成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

基于zepto的組件系統(tǒng)搭建

EasonTyler / 1091人閱讀

摘要:前言隨著前端開發(fā)復(fù)雜度的日益提升,組件化開發(fā)應(yīng)運而生,對于一個相對簡單的活動頁面開發(fā)如何進(jìn)行組件化是本文的主要內(nèi)容。

前言

隨著前端開發(fā)復(fù)雜度的日益提升,組件化開發(fā)應(yīng)運而生,對于一個相對簡單的活動頁面開發(fā)如何進(jìn)行組件化是本文的主要內(nèi)容。

概述

下面我們看一下在zepto的基礎(chǔ)上如何構(gòu)建組件系統(tǒng),首先,我們要解決第一個問題,如何引用一個組件,我們可以通過設(shè)置一個屬性data-component來引用自定義的組件:

那么如何向組件中傳入數(shù)據(jù)呢,我們同樣也可以通過設(shè)置屬性來向組件傳遞數(shù)據(jù),比如傳入一個id值:

那么組件之間如何進(jìn)行通信呢,我們可以采用觀察者模式來實現(xiàn)。

寫一個組件

我們先來看看我們?nèi)绾蝸韺懸粋€組件

//a.js
defineComponent("a", function (component) {
    var el = "

input-editor

"; var id = component.getProp("id");//獲取參數(shù)id $(this).append(el);//視圖渲染 component.setStyle(".a{color:green}");//定義樣式 $(this).find("p").on("click", function () { component.emit("test", id, "2");//觸發(fā)test }); });

我們先看看這個組件是怎么定義的,首先調(diào)用defineComponent(先不管這個函數(shù)在哪定義的)定義一個組件a,后面那個函數(shù)是組件a的組要邏輯,這個函數(shù)傳入了一個component(先不管這個是哪來的,先看它能干啥),在前面我們說過如何向組件傳遞數(shù)據(jù),在組件里我們通過component.getProp("id")來獲取,樣式我們通過component.setStyle(".a{color:green}")來定義,組件之前的通信我們通過component.emit()來觸發(fā)(在別的組件里通過component.on()來注冊),看上去我們基本解決了前面關(guān)于組件的一些問題,那么這個是怎么實現(xiàn)的呢?

組件實現(xiàn)原理

我們先來看看上面那個組件我們應(yīng)該如何來實現(xiàn),從上面定義一個組件來看有兩個地方是比較關(guān)鍵的,一個是defineComponent是怎么實現(xiàn)的,一個就是component是什么。

我們先來看看defineComponent是怎么實現(xiàn)的,很顯然defineComponent必須定義為全局的(要不然a.js就無法使用了,而且必須在加載a.js之前定義defineComponent),我們來看看defineComponent的代碼

//component.js
  var component = new Component();
  window.defineComponent = function (name, fn) {
        component.components[name] = {
            init: function () {
                //設(shè)置currentComponent為當(dāng)前組件
                currentComponent = this;
                fn.call(this, component);
                component.init(this);
            }
        };
    }

這里我們可以看到定義了一個類Componentcomponent是它的一個實例,defineComponent就是在component.components注冊一個組件,這里的關(guān)鍵是Component類,我們來看看Component是怎么定義的

//component.js
  /**
     * Component類
     * @constructor
     */
    function Component() {
        this.components = {};//所有的組件
        this.events = {};//注冊的事件
        this.loadStyle = {};
        this.init("body");//初始化
    }

    var currentComponent = null;//當(dāng)前的組件
    /**
     * 類的初始化函數(shù)
     * @param container 初始化的范圍,默認(rèn)情況下是body
     */
    Component.prototype.init = function (container) {
        var self = this;
        container = container || "body";
        $(container).find("[data-component]").each(function () {
            self.initComponent(this);
        });

    };
    /**
     *  初始化單個組件
     * @param context 當(dāng)前組件
     */
    Component.prototype.initComponent = function (context) {

        var self = this;
        var componentName = $(context).attr("data-component");
        if (this.components[componentName]) {
            this.components[componentName].init.call(context);
        } else {
            _loadScript("http://" + document.domain + ":5000/dist/components/" + componentName + ".js", function () {
                self.components[componentName].init.call(context);
                //設(shè)置樣式,同一個組件只設(shè)置一次
                if (!self.loadStyle[componentName] && self.components[componentName].style) {
                    $("head").append("");
                    self.loadStyle[componentName] = true;
                }
            });
        }

    };
    /**
     * 設(shè)置樣式
     * @param style 樣式
     */
    Component.prototype.setStyle = function (style) {
        //獲取當(dāng)前組件的名稱,currentComponent就是當(dāng)前組件
        var currentComponentName = $(currentComponent).attr("data-component");
        var component = this.components[currentComponentName];
        if (component && !component.style) {
            component.style = style;
        }
    };
    /**
     * 獲取組件參數(shù)
     * @param prop 參數(shù)名
     * @returns {*|jQuery}
     */
    Component.prototype.getProp = function (prop) {
        var currentComponentNme = $(currentComponent).attr("data-component");
        if ($(currentComponent).attr("data-" + prop)) {
            return $(currentComponent).attr("data-" + prop)
        } else {
            //屬性不存在時報錯
            throw Error("the attribute data-" + prop + " of " + currentComponentNme + " is undefined or empty")
        }

    };
    /**
     * 注冊事件
     * @param name 事件名
     * @param fn 事件函數(shù)
     */
    Component.prototype.on = function (name, fn) {
        this.events[name] = this.events[name] ? this.events[name] : [];
        this.events[name].push(fn);
    };
    /**
     * 觸發(fā)事件
     */
    Component.prototype.emit = function () {
        var args = [].slice.apply(arguments);
        var eventName = args[0];
        var params = args.slice(1);
        if(this.events[eventName]){
            this.events[eventName].map(function (fn) {
                fn.apply(null, params);
            });
        }else{
            //事件不存在時報錯
            throw Error("the event " + eventName + " is undefined")
        }

    };
    /**
     * 動態(tài)加載組價
     * @param url 組件路徑
     * @param callback 回調(diào)函數(shù)
     * @private
     */
    function _loadScript(url, callback) {
        var script = document.createElement("script");
        script.type = "text/javascript";
        if (typeof(callback) != "undefined") {
            if (script.readyState) {
                script.onreadystatechange = function () {
                    if (script.readyState == "loaded" || script.readyState == "complete") {
                        script.onreadystatechange = null;
                        callback();
                        $(script).remove();
                    }
                };
            } else {
                script.onload = function () {
                    callback();
                    $(script).remove();
                };
            }
        }
        script.src = url;
        $("body").append(script);
    }

我們先了解一下大概的流程


大致的流程就是上面這張流程圖了,我們所有的組件都是注冊在component.components里,事件都是在component.events里面。

我們回頭看一下組件components里頭的init方法

//component.js
  var component = new Component();
  window.defineComponent = function (name, fn) {
        component.components[name] = {
            init: function () {
                //設(shè)置currentComponent為當(dāng)前組件
                currentComponent = this;
                fn.call(this, component);
                component.init(this);
            }
        };
    }

首先,將this賦給currentComponent,這個在哪里會用到呢?在個getProp和setStyle這兩個方法里都用到了

//component.js
        /**
     * 設(shè)置樣式
     * @param style 樣式
     */
    Component.prototype.setStyle = function (style) {
        console.log(currentComponent);
        //獲取當(dāng)前組件的名稱,currentComponent就是當(dāng)前組件
        var currentComponentName = $(currentComponent).attr("data-component");
        var component = this.components[currentComponentName];
        if (component && !component.style) {
            component.style = style;
        }
    };
    /**
     * 獲取組件參數(shù)
     * @param prop 參數(shù)名
     * @returns {*|jQuery}
     */
    Component.prototype.getProp = function (prop) {
        return $(currentComponent).attr("data-" + prop)
    };

到這里大家可能會對this比較疑惑,這個this到底是什么,我們可以先看在那個地方調(diào)用了組件的init方法

//component.js
        /**
     *  初始化單個組件
     * @param componentName 組件名
     * @param context 當(dāng)前組件
     */
    Component.prototype.initComponent = function (componentName, context) {

        var self = this;
        if (this.components[componentName]) {
            this.components[componentName].init.call(context);
        } else {
            _loadScript("http://" + document.domain + ":5000/components/" + componentName + ".js", function () {
                self.components[componentName].init.call(context);
                //設(shè)置樣式,同一個組件只設(shè)置一次
                if (!self.loadStyle[componentName] && self.components[componentName].style) {
                    $("head").append("");
                    self.loadStyle[componentName] = true;
                }
            });
        }

    };

就是在單個組件初始化的調(diào)用了init方法,這里有call改變了init的this,使得this=context,那么這個context又是啥呢

//component.js
       /**
     * 類的初始化函數(shù)
     * @param container 初始化的范圍,默認(rèn)情況下是body
     */
    Component.prototype.init = function (container) {
        var self = this;
        container = container || "body";
        $(container).find("[data-component]").each(function () {
            var componentName = $(this).attr("data-component");
            console.log(this);
            self.initComponent(componentName, this);
        });

    };

context其實就是遍歷的每一個組件,到這里我們回過頭來看看我們是怎么定義一個組件

//b.js
defineComponent("b", function (component) {
    var el = "

text-editor

"; $(this).append(el); component.on("test", function (a, b) { console.log(a + b); }); var style = ".text-editor{color:red}"; component.setStyle(style) });

我們知道this就是組件本身也就是下面這個

這個組件通過component.on注冊了一個test事件,在前面我們知道test事件是在a組件觸發(fā)的,到這里我們就把整個組件系統(tǒng)框架開發(fā)完成了,下面就是一個個去增加組件就好了,整個的代碼如下:

//component.js
(function () {
    /**
     * Component類
     * @constructor
     */
    function Component() {
        this.components = {};//所有的組件
        this.events = {};//注冊的事件
        this.loadStyle = {};
        this.init("body");//初始化
    }

    var currentComponent = null;//當(dāng)前的組件
    /**
     * 類的初始化函數(shù)
     * @param container 初始化的范圍,默認(rèn)情況下是body
     */
    Component.prototype.init = function (container) {
        var self = this;
        container = container || "body";
        $(container).find("[data-component]").each(function () {
            self.initComponent(this);
        });

    };
    /**
     *  初始化單個組件
     * @param context 當(dāng)前組件
     */
    Component.prototype.initComponent = function (context) {

        var self = this;
        var componentName = $(context).attr("data-component");
        if (this.components[componentName]) {
            this.components[componentName].init.call(context);
        } else {
            _loadScript("http://" + document.domain + ":5000/dist/components/" + componentName + ".js", function () {
                self.components[componentName].init.call(context);
                //設(shè)置樣式,同一個組件只設(shè)置一次
                if (!self.loadStyle[componentName] && self.components[componentName].style) {
                    $("head").append("");
                    self.loadStyle[componentName] = true;
                }
            });
        }

    };
    /**
     * 設(shè)置樣式
     * @param style 樣式
     */
    Component.prototype.setStyle = function (style) {
        //獲取當(dāng)前組件的名稱,currentComponent就是當(dāng)前組件
        var currentComponentName = $(currentComponent).attr("data-component");
        var component = this.components[currentComponentName];
        if (component && !component.style) {
            component.style = style;
        }
    };
    /**
     * 獲取組件參數(shù)
     * @param prop 參數(shù)名
     * @returns {*|jQuery}
     */
    Component.prototype.getProp = function (prop) {
        var currentComponentNme = $(currentComponent).attr("data-component");
        if ($(currentComponent).attr("data-" + prop)) {
            return $(currentComponent).attr("data-" + prop)
        } else {
            //屬性不存在時報錯
            throw Error("the attribute data-" + prop + " of " + currentComponentNme + " is undefined or empty")
        }

    };
    /**
     * 注冊事件
     * @param name 事件名
     * @param fn 事件函數(shù)
     */
    Component.prototype.on = function (name, fn) {
        this.events[name] = this.events[name] ? this.events[name] : [];
        this.events[name].push(fn);
    };
    /**
     * 觸發(fā)事件
     */
    Component.prototype.emit = function () {
        var args = [].slice.apply(arguments);
        var eventName = args[0];
        var params = args.slice(1);
        if(this.events[eventName]){
            this.events[eventName].map(function (fn) {
                fn.apply(null, params);
            });
        }else{
            //事件不存在時報錯
            throw Error("the event " + eventName + " is undefined")
        }

    };
    /**
     * 動態(tài)加載組價
     * @param url 組件路徑
     * @param callback 回調(diào)函數(shù)
     * @private
     */
    function _loadScript(url, callback) {
        var script = document.createElement("script");
        script.type = "text/javascript";
        if (typeof(callback) != "undefined") {
            if (script.readyState) {
                script.onreadystatechange = function () {
                    if (script.readyState == "loaded" || script.readyState == "complete") {
                        script.onreadystatechange = null;
                        callback();
                        $(script).remove();
                    }
                };
            } else {
                script.onload = function () {
                    callback();
                    $(script).remove();
                };
            }
        }
        script.src = url;
        $("body").append(script);
    }

    var component = new Component();

    window.defineComponent = function (name, fn) {
        component.components[name] = {
            init: function () {
                //設(shè)置currentComponent為當(dāng)前組件
                currentComponent = this;
                fn.call(this, component);
                component.init(this);
            }
        };
    }

})();
工程化

上面搭建的組件系統(tǒng)有個不好的地方,就是我們定義的htmlstyle都是字符串,對于一些大的組件來說,htmlstyle都是非常長的,這樣的話調(diào)試就會很困難,因此,我們需要對組件系統(tǒng)進(jìn)行工程化,最終目標(biāo)是html,jscss可以分開開發(fā),現(xiàn)有的工程化工具比較多,你可以用gulp或者node自己寫一個工具,這里介紹一下如何使用node來實現(xiàn)組件系統(tǒng)的工程化。

我們先來看看目錄結(jié)構(gòu)


我們首先要獲取到編譯前組件的路徑

//get-path.js
var glob = require("glob");
exports.getEntries = function (globPath) {
    var entries = {};
    /**
     * 讀取src目錄,并進(jìn)行路徑裁剪
     */
    glob.sync(globPath).forEach(function (entry) {
        var tmp = entry.split("/");
        tmp.shift();
        tmp.pop();
        var pathname = tmp.join("/"); // 獲取前兩個元素

        entries[pathname] = entry;

    });

    return entries;
};

然后根據(jù)路徑分別讀取index.js,index.html,index.css

//read-file.js
var readline = require("readline");
var fs = require("fs");

exports.readFile = function (file, fn) {
    console.log(file);
    var fRead = fs.createReadStream(file);
    var objReadline = readline.createInterface({
        input: fRead
    });
    function trim(str) {
        return str.replace(/(^s*)|(s*$)|(//(.*))|(/*(.*)*/)/g, "");
    }
    var fileStr = "";
    objReadline.on("line", function (line) {
        fileStr += trim(line);
    });
    objReadline.on("close", function () {
        fn(fileStr)
    });
};


//get-component.js
var fs = require("fs");
var os = require("os");

var getPaths = require("./get-path.js");
var routesPath = getPaths.getEntries("./src/components/**/index.js");

var readFile = require("./read-file");

for (var i in routesPath) {
    (function (i) {
        var outFile = i.replace("src", "dist");
        readFile.readFile(i + "/index.js", function (fileStr) {
            var js = fileStr;
            readFile.readFile(i + "/index.html", function (fileStr) {
                js = js.replace("", fileStr);
                readFile.readFile(i + "/index.css", function (fileStr) {
                    js = js.replace("

              <