美文网首页饥人谷技术博客
从封装函数到实现简易版自用jQuery (二)

从封装函数到实现简易版自用jQuery (二)

作者: SarahZ | 来源:发表于2018-03-02 13:30 被阅读38次

    回顾

    上一篇文章 从封装函数到实现简易版自用jQuery (一) 已经介绍了如何实现基本功能和封装成自己的库,这篇文章着重讲对自己 API 功能的拓展,使其更强大。

    以下是基于第一篇文章,在本次练习中要用到的代码,以 addClass( ) 为例进行拓展。

    <ul>
      <li id="item1">item1</li>
      <li id="item2">item2</li>
      <li id="item3">item3</li>
      <li id="item4">item4</li>
      <li id="item5">item5</li>
    </ul>
    
    .red{
      color:red;
    }
    
    window.simpleTools = function(node){
      return{
        addClass:function(classes){
          classes.forEach((value) =>
                        node.classList.add(value));
        }
      };
    };
    
    

    思考1: 如果传参是个选择器该怎么办?

    解决方案

    加个类型判断来解决吧!为了让我们的代码更加语义化,传来的参数由 node 改名为 nodeOrSelector,在函数内再定义一个 node 变量,
    如果传参是选择器,通过 document.querySelector() 来找到相应的节点,赋值给 node;
    如果传参不是选择器,直接赋值给 node 存起来。

     var node;
     if(typeof nodeOrSelector === 'string'){
        node = document.querySelector(nodeOrSelector);
     }else{
        node = nodeOrSelector;
     }
    

    测试运行

    var nodeTest = simpleTools('#item3');
    nodeTest.addClass(['red']);
    console.log(document.querySelectorAll('#item3'));
    
    image

    思考2:如果传参是多个选择器该怎么办?

    解决方案

    1. document.querySelector( ) 改成 document.querySelectorAll( ) ,
      此时变量 node 变成 nodes 对象。
    2. 遍历 nodes ,依次加上 red 效果
    window.simpleTools = function(nodeOrSelector){
      var nodes = {};
      if(typeof nodeOrSelector === 'string'){
        nodes = document.querySelectorAll(nodeOrSelector);
      }else{
        nodes = nodeOrSelector;
      }
      return{
        addClass:function(classes){
          classes.forEach((value) =>{
          for(var i = 0;i < nodes.length;i++){
            nodes[i].classList.add(value);
          }
          });
        }
      };
    };
    

    测试运行

    var nodeTest = simpleTools('ul>li');
    nodeTest.addClass(['red']);
    console.log(document.querySelectorAll('ul>li'))
    
    image
    var nodeTest2 = simpleTools('#item3');
    nodeTest2.addClass(['red']);
    console.log(document.querySelectorAll('#item3'))
    
    image

    思考3: 想要改变原型链怎么办?

    现在的 nodes 是一个连接着
    NodeList.prototype 的对象,我的原型链想直接是 Object.prototype, 怎么办呢?

    解决方案

    借助一个临时变量,通过循环遍历得到一个纯净的对象。

    window.simpleTools = function(nodeOrSelector){
      var nodes = {};
      if(typeof nodeOrSelector === 'string'){
        var temp = document.querySelectorAll(nodeOrSelector);
        for(var i =0 ;i<temp.length;i++){
          nodes[i] = temp[i];
        }
        nodes.length = temp.length;
      }else if(nodeOrSelector instanceof Node){
        nodes = {
          0 :nodeOrSelector,
          length :1
        };
      }
      return nodes; // 只看nodes的变化,暂时先忽略addClass( )方法
    };
    

    如果是多个选择器,遍历并存储,不要忘记了nodes.length哦。如果是一个节点,也需要把 nodeOrSelector 构造出和上面分支一样的形式存到 node 对象中。现在无论是多个选择器还是一个节点,都转化成了只链接 Object.prototype 的对象。

    测试运行

    var nodeTest = simpleTools('#item3');
    console.log(nodeTest);
    
    image
    var nodeTest2 = simpleTools('ul>li');
    console.log(nodeTest2);
    
    image

    思考4: 设置文字怎么做?

    如果是获取文字,那么把 nodes 中每一项的 textContent 存起来。

    getText : function(){
        var texts = [];
        for(var i = 0; i < nodes.length;i++){
            texts.push(nodes[i].textContent);
        }
        return texts;
    }
    

    如果是设置文字,通过遍历,将要设置的文字依次赋值给 textContent 。

    setText : function(text){
        for(var i = 0; i < nodes.length;i++){
            nodes[i].textContent = text;
        }
        return text;
    }
    

    测试运行

    image
    var nodeTest = simpleTools('#item3');
    nodeTest.setText('hello');
    
    image
    代码优化

    无论是设置还是获取,上面的代码看起来是那么类似,说明这就存在着优化的可能。我们试图将这两个函数合并成为一个,如果你有参数传递,那么就说明你是需要设置文本,如果没有参数传入,那么就说明你是想获取文本。

    text: function (text) {
        if (text == undefined) {
            var texts = [];
            for (var i = 0; i < nodes.length; i++) {
                    texts.push(nodes[i].textContent);
                }
            return texts;
        }
        else {
            for (var i = 0; i < nodes.length; i++) {
                nodes[i].textContent = text;
            }
        }
    }
    

    再给个 alias 吧

    window.$ = function simpleTools(){...}
    

    使用全局变量 $ 就相当于在用 simpleTools。

    tips:
    如果某变量是由 jQuery 构造出来的,在变量前加上一个 $, 防止变量弄混。

    eg:var $node = $(#item3)

    关于 return 的两种形式

    第一种

    window.$ = function simpleTools(nodeOrSelector) {
        var nodes = {};
        if (typeof nodeOrSelector === 'string') {
            var temp = document.querySelectorAll(nodeOrSelector);
            for (var i = 0; i < temp.length; i++) {
                nodes[i] = temp[i];
            }
            nodes.length = temp.length;
        } else if (nodeOrSelector instanceof Node) {
            nodes = {
                0 : nodeOrSelector,
                length: 1
            };
        }
    
        return {
            addClass: function(classes) {
                classes.forEach((value) =>{
                    for (var i = 0; i < nodes.length; i++) {
                        nodes[i].classList.add(value);
                    }
                });
            },
            text: function(text) {
                if (text == undefined) {
                    var texts = [];
                    for (var i = 0; i < nodes.length; i++) {
                        texts.push(nodes[i].textContent);
                    }
                    return texts;
                } else {
                    for (var i = 0; i < nodes.length; i++) {
                        nodes[i].textContent = text;
                    }
                }
            }
    
        };
    };
    

    第二种

    window.$ = function simpleTools(nodeOrSelector) {
        var nodes = {};
        if (typeof nodeOrSelector === 'string') {
            var temp = document.querySelectorAll(nodeOrSelector);
            for (var i = 0; i < temp.length; i++) {
                nodes[i] = temp[i];
            }
            nodes.length = temp.length;
        } else if (nodeOrSelector instanceof Node) {
            nodes = {
                0 : nodeOrSelector,
                length: 1
            };
        }
    
        nodes.addClass = function(classes) {
            classes.forEach((value) =>{
                for (var i = 0; i < nodes.length; i++) {
                    nodes[i].classList.add(value);
                }
            });
        };
    
        nodes.text = function(text) {
            if (text == undefined) {
                var texts = [];
                for (var i = 0; i < nodes.length; i++) {
                    texts.push(nodes[i].textContent);
                }
                return texts;
            } else {
                for (var i = 0; i < nodes.length; i++) {
                    nodes[i].textContent = text;
                }
            }
        };
        return nodes;
    };
    
    

    你喜欢哪种就挑哪种啦

    小结

    1. 拓展了自己的addClass( )方法,不仅可以传节点,还可以接收选择器。
    2. 进一步加深了对原型链的了解。

    快动手试试写个自己的 API 吧!

    相关文章

      网友评论

        本文标题:从封装函数到实现简易版自用jQuery (二)

        本文链接:https://www.haomeiwen.com/subject/hhzoxftx.html