【原創】開發一個完整的JavaScript組件

做爲一名開發者,你們應該都知道在瀏覽器中存在一些內置的控件:Alert,Confirm等,可是這些控件一般根據瀏覽器產商的不一樣而形態萬千,視覺效果每每達不到UI設計師的要求。更重要的是,這類內置控件的風格很難與形形色色的各類風格迥異的互聯網產品的設計風格統一。所以,優秀的前端開發者們各自開發本身的個性化控件來替代瀏覽器內置的這些控件。固然,這類組件在網絡上已經有不可勝數至關優秀的,寫這篇文章的目的不是爲了說明我開發的這個組件有多優秀,也不是爲了炫耀什麼,只是但願經過這種方式,與更多的開發者互相交流,互相學習,共同進步。好,廢話很少說,言歸正傳。html

功能介紹

  • 取代瀏覽器自帶的Alert、Confirm控件
  • 自定義界面樣式
  • 使用方式與內置控件基本保持一致

效果預覽

一、Alert控件前端

二、Confirm控件瀏覽器

三、完整代碼,在線預覽(見底部,提供壓縮包下載)緩存

開發過程

1. 組件結構設計

首先,咱們來看下內置組件的基本使用方法:網絡

alert("內置Alert控件");
if (confirm("關閉內置Confirm控件?")) {
	alert("True");
} else {
	alert("False");
}

爲了保證咱們的組件使用方式和內置控件保持一致,因此咱們必須考慮覆蓋內置控件。考慮到組件開發的風格統一,易用,易維護,以及面向對象等特性,我計劃將自定義的alert和confirm方法做爲一個類(Winpop)的實例方法,最後用實例方法去覆蓋系統內置控件的方法。爲了達到目的,個人基本作法以下:app

var obj = new Winpop(); // 建立一個Winpop的實例對象
// 覆蓋alert控件
window.alert = function(str) {
    obj.alert.call(obj, str);
};
// 覆蓋confirm控件
window.confirm = function(str, cb) {
    obj.confirm.call(obj, str, cb);
};

須要注意的是,因爲瀏覽器內置的控件能夠阻止瀏覽器的其餘行爲,而咱們自定義的組件並不能具有這種能力,爲了儘量的作到統一,正如預覽圖上看到的,咱們在彈出自定義組件的時候使用了一個全屏半透明遮罩層。也正是因爲上述緣由,confirm組件的使用方式也作了一些細微的調整,由內置返回布爾值的方式,改成使用回調函數的方式,以確保能夠正確的添加「肯定」和「取消」的邏輯。所以,自定義組件的使用方式就變成了下面這種形式:ide

alert("自定義Alert組件");
confirm("關閉自定義Confirm組件?", function(flag){
    if (flag) {
        alert("True");
    } else {
        alert("False");
    }
});
2. 組件代碼設計

在正式介紹Winpop組件的代碼以前,咱們先來看一下一個Javascript組件的基本結構:函數

(function(window, undefined) {
    function JsClassName(cfg) {
        var config = cfg || {};
        this.get = function(n) {
            return config[n];
        }
        this.set = function(n, v) {
            config[n] = v;
        }
        this.init();
    }
    JsClassName.prototype = {
		init: function(){},
		otherMethod: function(){}
	};
	window.JsClassName = window.JsClassName || JsClassName;
})(window);

使用一個自執行的匿名函數將咱們的組件代碼包裹起來,儘量的減小全局污染,最後再將咱們的類附到全局window對象上,這是一種比較推薦的作法。學習

構造函數中的get、set方法不是必須的,只是筆者的我的習慣而已,以爲這樣寫能夠將配置參數和其餘組件內部全局變量緩存和讀取的調用方式統一,彷佛也更具備面向對象的型。歡迎讀者們說說各自的想法,說說這樣寫到底好很差。this

接下來咱們一塊兒看下Winpop組件的完整代碼:

(function(window, jQuery, undefined) {

    var HTMLS = {
        ovl: '<div class="J_WinpopMask winpop-mask" id="J_WinpopMask"></div>' + '<div class="J_WinpopBox winpop-box" id="J_WinpopBox">' + '<div class="J_WinpopMain winpop-main"></div>' + '<div class="J_WinpopBtns winpop-btns"></div>' + '</div>',
        alert: '<input type="button" class="J_AltBtn pop-btn alert-button" value="肯定">',
        confirm: '<input type="button" class="J_CfmFalse pop-btn confirm-false" value="取消">' + '<input type="button" class="J_CfmTrue pop-btn confirm-true" value="肯定">'
    }

    function Winpop() {
        var config = {};
        this.get = function(n) {
            return config[n];
        }

        this.set = function(n, v) {
            config[n] = v;
        }
        this.init();
    }

    Winpop.prototype = {
        init: function() {
            this.createDom();
            this.bindEvent();
        },
        createDom: function() {
            var body = jQuery("body"),
                ovl = jQuery("#J_WinpopBox");

            if (ovl.length === 0) {
                body.append(HTMLS.ovl);
            }

            this.set("ovl", jQuery("#J_WinpopBox"));
            this.set("mask", jQuery("#J_WinpopMask"));
        },
        bindEvent: function() {
            var _this = this,
                ovl = _this.get("ovl"),
                mask = _this.get("mask");
            ovl.on("click", ".J_AltBtn", function(e) {
                _this.hide();
            });
            ovl.on("click", ".J_CfmTrue", function(e) {
                var cb = _this.get("confirmBack");
                _this.hide();
                cb && cb(true);
            });
            ovl.on("click", ".J_CfmFalse", function(e) {
                var cb = _this.get("confirmBack");
                _this.hide();
                cb && cb(false);
            });
            mask.on("click", function(e) {
                _this.hide();
            });
            jQuery(document).on("keyup", function(e) {
                var kc = e.keyCode,
                    cb = _this.get("confirmBack");;
                if (kc === 27) {
                    _this.hide();
                } else if (kc === 13) {
                    _this.hide();
                    if (_this.get("type") === "confirm") {
                        cb && cb(true);
                    }
                }
            });
        },
        alert: function(str, btnstr) {
            var str = typeof str === 'string' ? str : str.toString(),
                ovl = this.get("ovl");
            this.set("type", "alert");
            ovl.find(".J_WinpopMain").html(str);
            if (typeof btnstr == "undefined") {
                ovl.find(".J_WinpopBtns").html(HTMLS.alert);
            } else {
                ovl.find(".J_WinpopBtns").html(btnstr);
            }
            this.show();
        },
        confirm: function(str, callback) {
            var str = typeof str === 'string' ? str : str.toString(),
                ovl = this.get("ovl");
            this.set("type", "confirm");
            ovl.find(".J_WinpopMain").html(str);
            ovl.find(".J_WinpopBtns").html(HTMLS.confirm);
            this.set("confirmBack", (callback || function() {}));
            this.show();
        },
        show: function() {
            this.get("ovl").show();
            this.get("mask").show();
        },
        hide: function() {
            var ovl = this.get("ovl");
            ovl.find(".J_WinpopMain").html("");
            ovl.find(".J_WinpopBtns").html("");
            ovl.hide();
            this.get("mask").hide();
        },
        destory: function() {
            this.get("ovl").remove();
            this.get("mask").remove();
            delete window.alert;
            delete window.confirm;
        }
    };

    var obj = new Winpop();
    window.alert = function(str) {
        obj.alert.call(obj, str);
    };
    window.confirm = function(str, cb) {
        obj.confirm.call(obj, str, cb);
    };
})(window, jQuery);

代碼略多,關鍵作如下幾點說明:

  • 筆者偷了懶,使用了jQuery,使用以前請先保證已經引入了jQuery
  • 自定義組件結構最終是追加到body中的,因此在引入以上js以前,請先確保文檔已經加載完成
  • 組件添加了按ESC、點遮罩層隱藏組件功能
  • **注意:**雖然本例中未用到destory方法,但讀者朋友能夠注意一下該方法中的delete window.alertdelete window.confirm,這樣寫的目的是保證在自定義組件銷燬後,將Alert、Confirm控件恢復到瀏覽器內置效果
  • 組件最後若是加上window.Winpop = Winpop,就能夠將對象全局化供其餘類調用了

最後

做爲一個前端開發工程師,我的以爲Javascript組件開發是一件頗有意思的事情,其中樂趣只有本身親自動手嘗試了纔會體會獲得。前端組件開發每每須要Javascript、CSS和html相互配合,才能事半功倍,上面提到的Winpop也不例外,這裏給你們提供一個完整的demo壓縮包,有興趣的讀者朋友,歡迎傳播。

做者博客:百碼山莊

相關文章
相關標籤/搜索