jQuery.fn.toggleClass

elem.toggleClass(value,stateVal)

  • 2个实参,第一个是字符串,第二个是 布尔值
    • 第二个参数为true ,添加class value,
    • 第二个参数为false,删除class value
  • 第一个参数是函数
    • 遍历elem中 的每个dom元素执行回调函数,回调函数的this值指向dom元素,第一个参数是dom元素的下标,第二个参数为元素当前的class,第三个参数为toggleClass的第二个参数,然后吧回调函数的返回值作为toggleClass的第一个参数重新执行toggleClass方法
  • 只有一个参数
    • 类型为string ,如果已经存在这个class,则删除,如不过不存在,则添加
    • 值为 false 清空class
  • 没有参数
    • 如果有class,把class缓存起来并清除class
    • 如果没有cass,吧缓存的class添加回去
jQuery.fn.extend({
    toggleClass: function( value, stateVal ) {
        var type = typeof value;
        // 获取第一个参数的数据类型

        if ( typeof stateVal === "boolean" && type === "string" ) {
            // 第一个参数是个字符串并且第二个参数是布尔值
            return stateVal ? this.addClass( value ) : this.removeClass( value );
            //当第二个参数为true时,把value作为className添加进元素
            //当第二个参数为false时,移除元素中的 value class
        }

        //只有一个参数并且是函数
        if ( jQuery.isFunction( value ) ) {
            //第一个参数是函数
            //对当前每个jq对象执行一次函数
            //this指向jq对象数组里的每个元素,i是其下标
            return this.each( function( i ) {
                jQuery( this ).toggleClass(
                    value.call( this, i, getClass( this ), stateVal ),
                    //把i 当前元素的className,stateVal作为参数传递value函数
                    //并把value的this指向当前元素
                    //把返回的值作为第一个参数重新调用函数
                    stateVal
                );
            } );
        }
        //没有参数或只有一个参数是string或boolean
        return this.each( function() {
            var className, i, self, classNames;

            if ( type === "string" ) {
                //第一个参数为字符串类型
                // Toggle individual class names
                i = 0;
                self = jQuery( this );
                classNames = value.match( rnothtmlwhite ) || [];
                ///rnothtmlwhite = ([^\x20\t\r\n\f]+/g)
                //把value根据分隔符拆分成数组,没有匹配到就返回空数组
                while ( ( className = classNames[ i++ ] ) ) {

                    // Check each className given, space separated list
                    if ( self.hasClass( className ) ) {
                        //如果元素有className,则删除
                        self.removeClass( className );
                    } else {
                        //如果没有,则添加
                        self.addClass( className );
                    }
                }

            // Toggle whole class name
            } else if ( value === undefined || type === "boolean" ) {
                //value为undefined或则boolean
                className = getClass( this );
                if ( className ) {
                    //如果元素本来有classname,那么先缓存起来
                    // Store className if set
                    dataPriv.set( this, "__className__", className );
                }

                // If the element has a class name or if we're passed `false`,
                // then remove the whole classname (if there was one, the above saved it).
                // Otherwise bring back whatever was previously saved (if anything),
                // falling back to the empty string if nothing was stored.
                if ( this.setAttribute ) {
                    //元素存在setAttribute方法
                    this.setAttribute( "class",
                        className || value === false ?
                        "" :
                        dataPriv.get( this, "__className__" ) || ""
                        //className有值 或value === false 删除所有classname
                        // 
                    );
                }
            }
        } );
    },
    each: function( callback ) {
        return jQuery.each( this, callback );
    },    
})


jQuery.extend({
        //遍历对象的每个元素,执行回调函数,回调函数的this指向对象成员
        each: function( obj, callback ) {
            var length, i = 0;
            //判断obj是否是类数组对象,比如元素的jq对象
            if ( isArrayLike( obj ) ) {
                length = obj.length;
                for ( ; i < length; i++ ) {
                    //遍历obj对象,执行回调函数
                    if ( callback.call( obj[ i ], i, obj[ i ] ) === false ) {
                        break;
                    }
                }
            } else {
                for ( i in obj ) {
                    //遍历obj对象,执行回调函数
                    if ( callback.call( obj[ i ], i, obj[ i ] ) === false ) {
                        break;
                    }
                }
            }

            return obj;
        }
    })
//判断是否为类数组对象
function isArrayLike( obj ) {
    var length = !!obj && "length" in obj && obj.length,
        //length=obj.length,如果不存在,就=false
        type = jQuery.type( obj );

    if ( type === "function" || jQuery.isWindow( obj ) ) {
        return false;
        //如果obj是function或window对象,返回false
    }

    return type === "array" || length === 0 ||
        typeof length === "number" && length > 0 && ( length - 1 ) in obj;
     //如果obj是个数组类型,返回true
     //length存在并且等于0,返回true
     //length存在并且是数值类型而且大于0,obj存在length - 1元素,返回true
     //如果上面所有条件都不满足,返回false,不是个类数组对象
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,639评论 18 139
  • 1.JQuery 基础 改变web开发人员创造搞交互性界面的方式。设计者无需花费时间纠缠JS复杂的高级特性。 1....
    LaBaby_阅读 1,330评论 0 2
  • 1.JQuery 基础 改变web开发人员创造搞交互性界面的方式。设计者无需花费时间纠缠JS复杂的高级特性。 1....
    LaBaby_阅读 1,167评论 0 1
  • //Clojure入门教程: Clojure – Functional Programming for the J...
    葡萄喃喃呓语阅读 3,636评论 0 7
  • 第一部分 准入训练 第1章 进入忍者世界 js开发人员通常使用js库来实现通用和可重用的功能。这些库需要简单易用,...
    如201608阅读 1,345评论 1 2