开发一个完整的JavaScript组件

  作为一名开发者,大家应该都知道在浏览器中存在一些内置的控件:Alert,Confirm等,但是这些控件通常根据浏览器产商的不同而形态各异,视觉效果往往达不到UI设计师的要求。更重要的是,这类内置控件的风格很难与形形色色的各种风格迥异的互联网产品的设计风格统一。因此,优秀的前端开发者们各自开发自己的个性化控件来替代浏览器内置的这些控件。当然,这类组件在网络上已经有不计其数相当优秀的,写这篇文章的目的不是为了说明我开发的这个组件有多优秀,也不是为了炫耀什么,只是希望通过这种方式,与更多的开发者互相交流,互相学习,共同进步。好,废话不多说,言归正传。

  功能介绍

  • 取代浏览器自带的Alert、Confirm控件
  • 自定义界面样式
  • 使用方式与内置控件基本保持一致

  效果预览

  1、Alert控件

  2、Confirm控件

  3、完整代码,在线预览(见底部,提供压缩包下载)

  开发过程

  1. 组件结构设计

首先,我们来看下内置组件的基本使用方法:


1

2

3

4

5

6

alert("内置Alert控件");

if (confirm("关闭内置Confirm控件?")) {

    alert("True");

} else {

    alert("False");

}

  为了保证我们的组件使用方式和内置控件保持一致,所以我们必须考虑覆盖内置控件。考虑到组件开发的风格统一,易用,易维护,以及面向对象等特性,我计划将自定义的alert和confirm方法作为一个类(Winpop)的实例方法,最后用实例方法去覆盖系统内置控件的方法。为了达到目的,我的基本做法如下:


1

2

3

4

5

6

7

8

9

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组件的使用方式也做了一些细微的调整,由内置返回布尔值的方式,改为使用回调函数的方式,以确保可以正确的添加“确定”和“取消”的逻辑。因此,自定义组件的使用方式就变成了下面这种形式:


1

2

3

4

5

6

7

8

alert("自定义Alert组件");

confirm("关闭自定义Confirm组件?", function(flag){

    if (flag) {

        alert("True");

    } else {

        alert("False");

    }

});

  2. 组件代码设计

  在正式介绍Winpop组件的代码之前,我们先来看一下一个Javascript组件的基本结构:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

(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方法不是必须的,只是笔者的个人习惯而已,觉得这样写可以将配置参数和其他组件内部全局变量缓存和读取的调用方式统一,似乎也更具有面向对象的型。欢迎读者们说说各自的想法,说说这样写到底好不好。

  接下来我们一起看下Winpop组件的完整代码:


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

(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.alert 和 delete window.confirm ,这样写的目的是保证在自定义组件销毁后,将Alert、Confirm控件恢复到浏览器内置效果
  • 组件最后如果加上 window.Winpop = Winpop ,就可以将对象全局化供其他类调用了

  最后

  作为一个前端开发工程师,个人觉得Javascript组件开发是一件很有意思的事情,其中乐趣只有自己亲自动手尝试了才会体会得到。前端组件开发往往需要Javascript、CSS和html相互配合,才能事半功倍,上面提到的Winpop也不例外,这里给大家提供一个完整的demo压缩包,有兴趣的读者朋友,欢迎传播。

时间: 2024-12-17 16:21:57

开发一个完整的JavaScript组件的相关文章

【原创】开发一个完整的JavaScript组件

作为一名开发者,大家应该都知道在浏览器中存在一些内置的控件:Alert,Confirm等,但是这些控件通常根据浏览器产商的不同而形态各异,视觉效果往往达不到UI设计师的要求.更重要的是,这类内置控件的风格很难与形形色色的各种风格迥异的互联网产品的设计风格统一.因此,优秀的前端开发者们各自开发自己的个性化控件来替代浏览器内置的这些控件.当然,这类组件在网络上已经有不计其数相当优秀的,写这篇文章的目的不是为了说明我开发的这个组件有多优秀,也不是为了炫耀什么,只是希望通过这种方式,与更多的开发者互相交

[转]开发一个完整的JavaScript组件

原创:http://www.admin10000.com/document/5961.html 作为一名开发者,大家应该都知道在浏览器中存在一些内置的控件:Alert,Confirm等,但是这些控件通常根据浏览器产商的不同而形态各异,视觉效果往往达不到UI设计师的要求.更重要的是,这类内置控件的风格很难与形形色色的各种风格迥异的互联网产品的设计风格统一.因此,优秀的前端开发者们各自开发自己的个性化控件来替代浏览器内置的这些控件.当然,这类组件在网络上已经有不计其数相当优秀的,写这篇文章的目的不是

php怎么做网站?如何用PHP开发一个完整的网站?

1.PHPer应具备的知识 (1)PHP知识: 熟练掌握基础函数,PHP语句(条件.循环),数组(排序.读取),函数(内部 构造),运算(数学 逻辑),面向对象(继承 接口 封装 多态静态属性)等. 了解Cookie或者Session一种机制 了解一种模板操作机制和使用 手头收藏一些好的常用类或方法,能提高我们的开发的速度.如:分页.上传.字符过滤.数据库操作.生成HTML.文件操作等. (2)HTML知识 (3)JavaScript事件处理 (4)数据库知识:SQL语句"增删改查"

【如何快速的开发一个完整的iOS直播app】(原理篇)

一.个人见解(直播难与易) 直播难:个人认为要想把直播从零开始做出来,绝对是牛逼中的牛逼,大牛中的大牛,因为直播中运用到的技术难点非常之多,视频/音频处理,图形处理,视频/音频压缩,CDN分发,即时通讯等技术,每一个技术都够你学几年的. 直播易:已经有各个领域的大牛,封装好了许多牛逼的框架,我们只需要用别人写好的框架,就能快速的搭建一个直播app,也就是传说中的站在大牛肩膀上编程. 二.了解直播 热门直播产品 映客,斗鱼,熊猫,虎牙,花椒等等 直播效果图 直播效果.jpeg 1.一个完整直播ap

【如何快速的开发一个完整的iOS直播app】(播放篇)

前言 在看这篇之前,如果您还不了解直播原理,请查看上篇文章如何快速的开发一个完整的iOS直播app(原理篇) 开发一款直播app,集成ijkplayer成功后,就算完成直播功能一半的工程了,只要有拉流url,就能播放直播啦 本篇主要讲解的是直播app中,需要用到的一个很重要的开源框架ijkplayer,然后集成这个框架可能对大多数初学者还是比较有难度的,所以本篇主要教你解决集成[ijkplayer]遇见的各种坑. 很多文章,可能讲解的是如何做,我比较注重讲解为什么这样做,大家有什么不明白,还可以

如何快速的开发一个完整的iOS直播app(原理篇)

前言 大半年没写博客了,但我一直关注着互联网的动向,最近会研究很多东西,并分享,今年移动直播行业的兴起,诞生了一大批网红,甚至明星也开始直播了,因此不得不跟上时代的步伐,由于第一次接触的原因,因此花了很多时间了解直播,整理了直播的原理,当前只是原理篇,后续会持续发布实战篇,教你从零开始搭建一个完整的iOS直播app,希望能帮助到更多的人更快的了解直播. 一.个人见解(直播难与易) 直播难:个人认为要想把直播从零开始做出来,绝对是牛逼中的牛逼,大牛中的大牛,因为直播中运用到的技术难点非常之多,视频

【如何快速的开发一个完整的 iOS 直播 app】(美颜篇)

来源:袁峥Seemygo 链接:http://www.jianshu.com/p/4646894245ba 前言 在看这篇之前,如果您还不了解直播原理,请查看这篇文章如何快速的开发一个完整的iOS直播app(原理篇) 开发一款直播app,美颜功能是很重要的,如果没有美颜功能,可能分分钟钟掉粉千万,本篇主要讲解直播中美颜功能的实现原理,并且实现美颜功能. 利用GPUImage处理直播过程中美颜的流程 采集视频 => 获取每一帧图片 => 滤镜处理 => GPUImageView展示 美颜原

使用vue来开发一个下拉菜单组件(1)

一.新建demo工程 vue init webpack-simple demo 添加src/mixins/emitter.js文件(见前几篇博文) 安装font-awesome字体库: cnpm install font-awesome --save 配置webpack.config.js,引入字体文件: { test: /\.(otf|eot|ttf|woff|woff2)$/, loader: 'file-loader' } 在src/main.js中引入font-awesome: impo

【如何快速的开发一个完整的iOS直播app】(采集篇)

效果 为了采集效果图,我也是豁出去了,请忽略人物,关注技术. 忽略本人.png 基本知识介绍 AVFoundation: 音视频数据采集需要用AVFoundation框架. AVCaptureDevice:硬件设备,包括麦克风.摄像头,通过该对象可以设置物理设备的一些属性(例如相机聚焦.白平衡等) AVCaptureDeviceInput:硬件输入对象,可以根据AVCaptureDevice创建对应的AVCaptureDeviceInput对象,用于管理硬件输入数据. AVCaptureOutp