• 04-DOM操作练习:Tab栏切换(通过className设置样式)

    京东网页上,可以看到下面这种tab栏的切换:

    我们把模型抽象出来,实现一下。

    # 举例引入:鼠标悬停时,current元素的背景变色

    本段我们先举一个例子,因为这里用到了排他思想(先干掉 all,然后保留我一个)。对于理解tab切换,很有帮助。

    完整的代码实现:

    <!DOCTYPE html>
    <html>
    <head lang="en">
        <meta charset="UTF-8">
        <title></title>
        <style>
            button {
                margin: 10px;
                width: 100px;
                height: 40px;
                cursor: pointer;
            }
            .current {
                background-color: red;
            }
        </style>
    </head>
    <body>
    <button>按钮1</button>
    <button>按钮2</button>
    <button>按钮3</button>
    <button>按钮4</button>
    <button>按钮5</button>
    
    <script>
        //需求:鼠标放到哪个button上,改button变成黄色背景(添加类)
    
    
        var btnArr = document.getElementsByTagName("button");
        //绑定事件
        for(var i=0;i<btnArr.length;i++){   //要为每一个按钮绑定事件,所以用到了for循环
            btnArr[i].onmouseover = function () {
                //【重要】排他思想:先把所有按钮的className设置为空,然后把我(this)这个按钮的className设置为current
                //排他思想和for循环连用
                for(var j=0;j<btnArr.length;j++){
                    btnArr[j].className = "";
                }
                this.className = "current";  //【重要】核心代码
            }
        }
    
        //鼠标离开current时,还原背景色
        for(var i=0;i<btnArr.length;i++){   //要为每一个按钮绑定事件,所以用到了for循环
            btnArr[i].onmouseout = function () { //鼠标离开任何一个按钮时,就把按钮的背景色还原
                this.className = "";
            }
        }
    
    </script>
    
    </body>
    </html>
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52

    代码解释:

    鼠标悬停时,current栏变色,这里用到了排他思想:先把所有按钮的className设置为空,然后把我(this)这个按钮的className设置为current,就可以达到变色的效果。核心代码是:

                //排他思想:先把所有按钮的className设置为空,然后把我(this)这个按钮的className设置为current
                //排他思想和for循环连用
                for(var j=0;j<btnArr.length;j++){
                    btnArr[j].className = "";
                }
                this.className = "current";
    
    1
    2
    3
    4
    5
    6

    实现的效果如下:

    # tab切换:初步的代码

    代码如下:

    <!DOCTYPE html>
    <html>
    <head lang="en">
        <meta charset="UTF-8">
        <title></title>
        <style>
            * {
                padding: 0;
                margin: 0;
            }
            .box {
                width: 500px;
                height: 200px;
                border: 1px solid #ccc;
                margin: 50px auto;
                overflow: hidden;
            }
            ul {
                width: 600px;
                height: 40px;
                margin-left: -1px;
                list-style: none;
            }
            li {
                float: left;
                width: 101px;
                height: 40px;
                text-align: center;
                font: 600 18px/40px "simsun";
                background-color: pink;
                cursor: pointer;
            }
            span {
                display: none;
                width: 500px;
                height: 160px;
                background-color: yellow;
                text-align: center;
                font: 700 100px/160px "simsun";
            }
            .show {
                display: block;
            }
            .current {
                background-color: yellow;
            }
        </style>
    
        <script>
            window.onload = function () {
                //需求:鼠标放到上面的li上,li本身变色(添加类),对应的span也显示出来(添加类);
                //思路:1.点亮上面的盒子。   2.利用索引值显示下面的盒子。
    
                var liArr = document.getElementsByTagName("li");
                var spanArr = document.getElementsByTagName("span");
    
                for(var i=0;i<liArr.length;i++){
                    //绑定索引值(新增一个自定义属性:index属性)
                    liArr[i].index = i;
                    liArr[i].onmouseover = function () {
    
                        //1.点亮上面的盒子。   2.利用索引值显示下面的盒子。(排他思想)
                        for(var j=0;j<liArr.length;j++){
                            liArr[j].className = "";
                            spanArr[j].className = "";
                        }
                        this.className = "current";
                        spanArr[this.index].className = "show"; //【重要代码】
                    }
                }
            }
        </script>
    </head>
    <body>
    
    <div class="box">
        <ul>
            <li class="current">鞋子</li>
            <li>袜子</li>
            <li>帽子</li>
            <li>裤子</li>
            <li>裙子</li>
        </ul>
        <span class="show">鞋子</span>
        <span>袜子</span>
        <span>帽子</span>
        <span>裤子</span>
        <span>裙子</span>
    </div>
    
    </body>
    </html>
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92

    实现效果如下:

    上方代码的核心部分是:

                for(var i=0;i<liArr.length;i++){
                    //绑定索引值(新增一个自定义属性:index属性)
                    liArr[i].index = i;
                    liArr[i].onmouseover = function () {
    
                        //1.点亮上面的盒子。   2.利用索引值显示下面的盒子。(排他思想)
                        for(var j=0;j<liArr.length;j++){
                            liArr[j].className = "";
                            spanArr[j].className = "";
                        }
                        this.className = "current";
                        spanArr[this.index].className = "show"; //【重要代码】
                    }
                }
    
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15

    这段代码中,我们是通过给 liArr[i]一个index属性:liArr[i].index = i,然后让下方的span对应的index也随之对应显示:spanArr[this.index].className = "show"

    这样做比较难理解,其实还有一种容易理解的方法是:给liArr[i]增加index属性时,通过attribute的方式,因为这种方式增加的属性是可以显示在标签上的。也就有了下面这样的代码:

                for(var i=0;i<liArr.length;i++){
                    //绑定索引值(自定义属性),通过Attribute的方式【重要】
                    liArr[i].setAttribute("index",i);
                    liArr[i].onmouseover = function () {
                        //3.书写事件驱动程序(排他思想)
                        //1.点亮盒子。   2.利用索引值显示盒子。(排他思想)
                        for(var j=0;j<liArr.length;j++){
                            liArr[j].removeAttribute("class");
                            spanArr[j].removeAttribute("class");
                        }
                        this.setAttribute("class","current");
                        spanArr[this.getAttribute("index")].setAttribute("class","show");
                    }
                }
    
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15

    显示的效果是一样的,不同的地方在于,我们审查元素发现,li标签中确实新增了自定义的index属性:

    本段中,我们的目的已经达到了,不足的地方在于,本段中的代码是通过document获取的的标签,如果网页中有很多个这种tab选项卡,必然互相影响。

    为了多个tab栏彼此独立,我们就需要通过封装函数的方式把他们抽取出来,于是就有了下面的改进版代码。

    # tab切换:改进版代码(函数封装)

    # 方式一:给current标签设置index值【推荐的代码】

    完整版代码实现:

    <!DOCTYPE html>
    <html>
    <head lang="en">
        <meta charset="UTF-8">
        <title></title>
        <style>
            * {
                padding: 0;
                margin: 0;
            }
    
            .box {
                width: 500px;
                height: 200px;
                border: 1px solid #ccc;
                margin: 50px auto;
                overflow: hidden;
            }
    
            ul {
                width: 600px;
                height: 40px;
                margin-left: -1px;
                list-style: none;
            }
    
            li {
                float: left;
                width: 101px;
                height: 40px;
                text-align: center;
                font: 600 18px/40px "simsun";
                background-color: pink;
                cursor: pointer;
            }
    
            span {
                display: none;
                width: 500px;
                height: 160px;
                background-color: yellow;
                text-align: center;
                font: 700 100px/160px "simsun";
            }
    
            .show {
                display: block;
            }
    
            .current {
                background-color: yellow;
            }
        </style>
    
        <script>
            window.onload = function () {
                //需求:鼠标放到上面的li上,li本身变色(添加类),下方对应的span也显示出来(添加类);
                //思路:1.点亮上面的盒子。   2.利用索引值显示下面的对应的盒子。
    
                //1、获取所有的box
                var boxArr = document.getElementsByClassName("box");
    
                //让每一个box都调用函数
                for (var i = 0; i < boxArr.length; i++) {
                    fn(boxArr[i]);
                }
    
                function fn(element) {
                    var liArr = element.getElementsByTagName("li");   //注意,是element获取事件源,不是document获取事件源
                    var spanArr = element.getElementsByTagName("span");
                    //2.绑定事件(循环绑定)
                    for (var i = 0; i < liArr.length; i++) {
                        //绑定索引值(新增一个自定义属性:index属性)
                        liArr[i].index = i;
                        liArr[i].onmouseover = function () {
                            //3.书写事件驱动程序(排他思想)
                            //1.点亮上面的盒子。   2.利用索引值显示下面的盒子。(排他思想)
                            for (var j = 0; j < liArr.length; j++) {
                                liArr[j].className = "";
                                spanArr[j].className = "";
                            }
                            this.className = "current";
                            spanArr[this.index].className = "show"; //【重要】核心代码
                        }
                    }
                }
            }
        </script>
    </head>
    <body>
    
    <div class="box">
        <ul>
            <li class="current">鞋子</li>
            <li>袜子</li>
            <li>帽子</li>
            <li>裤子</li>
            <li>裙子</li>
        </ul>
        <span class="show">鞋子</span>
        <span>袜子</span>
        <span>帽子</span>
        <span>裤子</span>
        <span>裙子</span>
    </div>
    
    <div class="box">
        <ul>
            <li class="current">鞋子</li>
            <li>袜子</li>
            <li>帽子</li>
            <li>裤子</li>
            <li>裙子</li>
        </ul>
        <span class="show">鞋子</span>
        <span>袜子</span>
        <span>帽子</span>
        <span>裤子</span>
        <span>裙子</span>
    </div>
    
    <div class="box">
        <ul>
            <li class="current">鞋子</li>
            <li>袜子</li>
            <li>帽子</li>
            <li>裤子</li>
            <li>裙子</li>
        </ul>
        <span class="show">鞋子</span>
        <span>袜子</span>
        <span>帽子</span>
        <span>裤子</span>
        <span>裙子</span>
    </div>
    
    </body>
    </html>
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    68
    69
    70
    71
    72
    73
    74
    75
    76
    77
    78
    79
    80
    81
    82
    83
    84
    85
    86
    87
    88
    89
    90
    91
    92
    93
    94
    95
    96
    97
    98
    99
    100
    101
    102
    103
    104
    105
    106
    107
    108
    109
    110
    111
    112
    113
    114
    115
    116
    117
    118
    119
    120
    121
    122
    123
    124
    125
    126
    127
    128
    129
    130
    131
    132
    133
    134
    135
    136
    137
    138

    注意,通过函数fn的封装之后,我们是通过参数element获取元素,而不再是document了。这样可以达到“抽象性”的作用,各个tab栏之间相互独立。

    上方代码中,我们是通过给 liArr[i]一个index属性:liArr[i].index = i,然后让下方的span对应的index也随之对应显示:spanArr[this.index].className = "show"

    这样做比较难理解,根据上一段的规律,当然还有一种容易理解的方法是:给liArr[i]增加index属性时,通过attribute的方式,因为这种方式增加的属性是可以显示在标签上的。也就有了下面的方式二。

    # 方式二:通过attribute设置index的值

    基于上面方式一中的代码,我们修改一下js部分的代码,其他部分的代码保持不变。js部分的代码如下:

        <script>
            window.onload = function () {
                //需求:鼠标放到上面的li上,li本身变色(添加类),下方对应的span也显示出来(添加类);
                //思路:1.点亮上面的盒子。   2.利用索引值显示下面的对应的盒子。
    
                //1、获取所有的box
                var boxArr = document.getElementsByClassName("box");
    
                //让每一个box都调用函数
                for (var i = 0; i < boxArr.length; i++) {
                    fn(boxArr[i]);
                }
    
                function fn(element) {
                    var liArr = element.getElementsByTagName("li");   //注意,是element获取事件源,不是document获取事件源
                    var spanArr = element.getElementsByTagName("span");
                    //2.绑定事件(循环绑定)
                    for (var i = 0; i < liArr.length; i++) {
                        //绑定索引值(自定义属性)
                        liArr[i].setAttribute("index", i);
                        liArr[i].onmouseover = function () {
                            //3.书写事件驱动程序(排他思想)
                            //1.点亮盒子。   2.利用索引值显示盒子。(排他思想)
                            for (var j = 0; j < liArr.length; j++) {
                                liArr[j].removeAttribute("class");    //注意,这里是class,不是className
                                spanArr[j].removeAttribute("class");
                            }
                            this.setAttribute("class", "current");
                            spanArr[this.getAttribute("index")].setAttribute("class", "show");
                        }
                    }
                }
            }
        </script>
    
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34

    不过,方式一的写法应该比方式二更常见。

    总结:通过函数封装的形式,可以保证各个tab栏之间的切换互不打扰。最终实现效果如下:

    想学习更多技能?不妨关注我的微信公众号:千古壹号(id:qianguyihao)。