为了账号安全,请及时绑定邮箱和手机立即绑定

制作移动端弹窗(box/toast/loading/progress)

说明

根据个人需求,闲时动手写写案例。

这个demo中,主要写了移动端的几种弹窗类型,包括box(alert/confirm)、toast、loading等

使用

  • 全局引用popup.csspopup.js
<link rel="stylesheet" href="./popup.css">
<script class="lazyload" src="" data-original="./popup.js"></script>
  • 使用方式:
// 1.alert/confirm
// 标题可选
Popup.alert([title], content, callback);

// 例如
Popup.alert("哈哈哈", "您的操作有误", [
    {name : "哈哈", ac : function() {alert("OK");}}
]);

Popup.confirm("哈哈哈", "AA", [
    {name : "哈哈", ac : function() {alert("haha");}},
    {name : "嘻嘻", ac : function() {alert("xixi");}},
    {name : "呵呵", ac : function() {alert("hehe");}}
]);

// 2.toast
// 内容、超时可选
Popup.toast([content], [timeout]);

// 例如
Popup.toast("去吧,皮卡丘!", 1000);

// 3.loading
// 文字可选,没填文字,默认转菊花
// <todo> 菊花瓣数、大小、颜色可配置
// Popup.showLoading([content]);

// 例如
Popup.showLoading("loading...");

// 4.close
// 关闭弹窗
Popup.close();

// 5.progress
// 文字可选,没填文字,默认"正在更新程序"
// Popup.progress([content]);

// 例如
var ProgressBar = Popup.progress();
var percent = 0;

var inter = setInterval(function() {
    if (percent > 100) {
        clearInterval(inter);
    }
    ProgressBar.update(percent);
    percent++;
}, 50);

详细代码

简单的Dialog对象,内置初始化属性,前期由于需求量不大,没有做过多类型区分,简单实用type字段来标识dialog类型,包括了alert/confirm/toast/loading,因为它们都不需要实例调用;后期要做进度条,需要获取实例实时更新进度,因此在原dialog的基础上做了继承。

代码中还存在着很多可以优化的地方,包括代码精简压缩以及测试处理,仅拿出来共享,给移动端的朋友们一些参考。可以到github上去下载源码跑一下^-^。代码在这里啊!!!

  • popup.js
window.Popup = (function(d) {
    var _instance = null,
        _progressInstance = null,
        doc = d.body,
        docWidth = doc.clientWidth,
        docHeight = doc.clientHeight;

    function Dialog() {
        this.title = '标题';
        this.content = '内容';
        this.btns = [
            {name : '确定', ac : function() {}},
            {name : '取消', ac : function() {}}
        ];
        this.type = 'alert';
        this.hasTitle = true;
        this.timeout = 500;
        this.dialogWrapper = null;
        this.mask = null;
        this.fadeTime = 250;
        this.fadeInTime = 0;
        this.hasDialog = false;
        // loading
        this.loadingText = '正在加载';
        this.hasLoadingText = false;
        this.spinNumber = 8;            // 菊花瓣数
        this.loadingRadius = 18;        // 菊花半径
        this.startAngle = 90;           // 正上方菊花角度
        this.spinMargin = 2;
    }

    Dialog.prototype = {
        /**
         * 创建mask
         * 
         * @return {[type]} [description]
         */
        _createMask : function() {
            var _this = this;
            if (!this.mask) {
                this.mask = document.createElement('div');
                this.mask.className = 'ui-mask fade';
                doc.appendChild(this.mask);
                // 在创建了mask之后250ms,给其设置透明度为1,达到动画效果
                setTimeout(function() {
                    _this.mask.className = 'ui-mask fade in';
                }, this.fadeInTime);

            }
        },

        /**
         * 移除mask
         * 
         * @return {[type]} [description]
         */
        _removeMask : function() {
            var _this = this;
            if (this.mask) {
                // 移除dom之前,给其设置className,达到动画过渡效果
                this.mask.className = 'ui-mask fade';
                setTimeout(function() {
                    _this.mask.remove();
                    _this.mask = null;
                }, this.fadeTime);
            }
        },

        /**
         * 创建弹窗的容器
         * 
         * @return {[type]} [description]
         */
        _createWrapper : function() {
            if (!this.dialogWrapper) {
                this.dialogWrapper = document.createElement('div');
            }
        },

        /**
         * 移除弹窗的容器
         * 
         * @return {[type]} [description]
         */
        _removeWrapper : function() {
            var _this = this;
            if (this.dialogWrapper) {
                if (this.type === 'toast') {
                    this.dialogWrapper.className = 'ui-dialog ui-toast fade';
                } else if(this.type === 'loading') {
                    this.dialogWrapper.className = 'ui-dialog ui-loading fade';
                } else if(this.type === 'progress') {
                    this.dialogWrapper.className = 'ui-dialog ui-progress fade';
                } else {
                    this.dialogWrapper.className = 'ui-dialog fade';
                }
                setTimeout(function() {
                    _this.dialogWrapper.remove();
                    _this.dialogWrapper = null;
                    _this.hasDialog = false;
                }, this.fadeTime);
            }
        },

        /**
         * 如果存在标题,则创建标题容器
         * 
         * @return {[type]} [description]
         */
        _createTitle : function() {
            var title = document.createElement('div');
            title.className = 'ui-title';
            title.textContent = this.title;
            return title;
        },

        /**
         * 创建内容区域
         * 
         * @return {[type]} [description]
         */
        _createContent : function() {
            var content = document.createElement('div');
            content.className = 'ui-content';
            content.textContent = this.content;
            return content;
        },

        /**
         * 创建按钮组
         * 
         * @return {[type]} [description]
         */
        _createBtns : function() {
            var buttons = document.createElement('div'), _this = this;
            buttons.className = 'ui-btns';
            this.btns && this.btns.forEach(function(btn, index) {
                var link = document.createElement('a');
                link.href = 'javascript:void(0)';
                link.textContent = btn.name;
                buttons.appendChild(link);
                link.addEventListener('click', function() {
                    _this.close();
                    btn.ac && btn.ac();
                });
            });
            return buttons;
        },

        /**
         * 创建loading文字
         * 
         * @return {[type]} [description]
         */
        _createLoadingText : function() {
            var text = document.createElement('div');
            text.className = 'ui-text';
            text.textContent = this.loadingText;
            return text;
        },

        /**
         * 创建loading菊花
         * 
         * @return {[type]} [description]
         */
        _createLoadingContent : function() {
            var content = document.createElement('div'),
                lineSpin = document.createElement('div'),
                deltaAngle = 360 / this.spinNumber;
            content.className = 'ui-content';
            if (!this.hasLoadingText) {
                content.style.height = '86px';
                content.style.borderRadius = '5px';
            }
            lineSpin.className = 'ui-line-spin';
            lineSpin.style.width = (this.loadingRadius * 2 + this.spinMargin * 2) + 'px';
            lineSpin.style.height = (this.loadingRadius * 2 + this.spinMargin * 2) + 'px';
            for (var i = 0; i < this.spinNumber; i++) {
                var div = document.createElement('div');
                var angle = this.startAngle + i * (-deltaAngle),
                    rotateAngle = i * deltaAngle,
                    p = Math.PI * 2 * (angle / 360);
                div.style.top = this.loadingRadius * (1 - Math.sin(p)) + 'px';
                div.style.left = this.loadingRadius * (1 + Math.cos(p)) + 'px';
                div.style.transform = 'rotate('+rotateAngle+'deg)';
                div.style.WebkitTransform = 'rotate('+rotateAngle+'deg)';
                div.style.MozTransform = 'rotate('+rotateAngle+'deg)';
                div.style.msTransform = 'rotate('+rotateAngle+'deg)';
                div.style.OTransform = 'rotate('+rotateAngle+'deg)';
                lineSpin.appendChild(div);
            }
            content.appendChild(lineSpin);
            return content;
        },

        /**
         * 渲染弹窗
         * 
         * @return {[type]} [description]
         */
        _renderDialog : function() {
            this._createWrapper();
            switch (this.type) {
                case 'confirm':
                    this.dialogWrapper.className = 'ui-dialog fade';
                    var title, content, btns, _this = this;
                    content = this._createContent();
                    btns = this._createBtns();
                    if (this.hasTitle) {
                        title = this._createTitle();
                        this.dialogWrapper.appendChild(title);
                    }

                    this.dialogWrapper.appendChild(content);
                    this.dialogWrapper.appendChild(btns);
                    doc.appendChild(this.dialogWrapper);
                    setTimeout(function() {
                        _this.dialogWrapper.className = 'ui-dialog fade in';
                    }, this.fadeInTime);
                    break;
                case 'toast': 
                    this.dialogWrapper.className = 'ui-dialog ui-toast fade';
                    var content = this._createContent(), _this = this;
                    this.dialogWrapper.appendChild(content);
                    doc.appendChild(this.dialogWrapper);
                    setTimeout(function() {
                        _this.dialogWrapper.className = 'ui-dialog ui-toast fade in';
                    }, this.fadeInTime);
                    setTimeout(function() {
                        _this.close();
                    }, this.timeout);
                    break;
                case 'loading':
                    this.dialogWrapper.className = 'ui-dialog ui-loading fade';
                    var text, content, _this = this;
                    content = this._createLoadingContent();
                    if (this.hasLoadingText) {
                        text = this._createLoadingText();
                        this.dialogWrapper.appendChild(text);
                    }
                    this.dialogWrapper.appendChild(content);
                    doc.appendChild(this.dialogWrapper);
                    setTimeout(function() {
                        _this.dialogWrapper.className = 'ui-dialog ui-loading fade in';
                    }, this.fadeInTime);
                    break;
                default:
                    break;
            }
        },

        /**
         * confirm/alert 弹窗,返回供外部使用
         * 
         * @return {[type]} [description]
         */
        confirm : function() {
            var args = arguments && arguments[0],
                length = args && args.length;
            if (length === 3) {
                // 有标题
                this.title = args[0];
                this.content = args[1];
                this.btns = args[2];
                this.hasTitle = true;
            } else if(length === 2) {
                this.content = args[0];
                this.btns = args[1];
                this.hasTitle = false;
            }
            this.type = 'confirm';
            this._create();
        },

        /**
         * toast弹窗,返回供外部使用
         * 
         * @return {[type]} [description]
         */
        toast : function() {
            var args = arguments && arguments[0],
                length = args && args.length;
            if (length === 2) {
                this.content = args[0];
                this.timeout = args[1];
            } else if(length === 1) {
                this.content = args[0];
            }
            this.type = 'toast';
            if (!this.hasDialog) {
                this._create();
            }
            this.hasDialog = true;
        },

        /**
         * loading弹窗,返回供外部使用
         * 
         * @return {[type]} [description]
         */
        showLoading : function() {
            var args = arguments && arguments[0],
                length = args && args.length;

            // 带有参数
            if (length === 1) {
                this.loadingText = args[0];
                this.hasLoadingText = true;
            }

            this.type = 'loading';
            this._create();
        },

        /**
         * 创建弹窗 
         * 
         * @return {[type]} [description]
         */
        _create : function() {
            this._createMask();
            this._renderDialog();
        },

        /**
         * 关闭弹窗
         * 
         * @return {[type]} [description]
         */
        close : function() {
            this._removeMask();
            this._removeWrapper();
            _instance = null;
            _progressInstance = null;
        }
    };

    function Progress() {
        Dialog.call(this);
        this.percent = 0;
        this.text = '正在更新程序';
        this.innerBar = null;
    }

    /**
     * 继承
     * @param  {[type]} subClass   [description]
     * @param  {[type]} superClass [description]
     * @return {[type]}            [description]
     */
    function inherit(subClass, superClass) {  
        function F() {}  
        F.prototype = superClass.prototype;  
        subClass.prototype = new F();  
        subClass.prototype.constructor = subClass;  
    }  

    /**
     * 继承条用
     */
    inherit(Progress, Dialog);

    /**
     * 创建进度条文字
     * @return {[type]} [description]
     */
    Progress.prototype._createProgressText = function() {
        var text = document.createElement('div');
        text.className = 'ui-text';
        text.textContent = this.text;
        return text;
    }

    /**
     * 创建进度条
     * @return {[type]} [description]
     */
    Progress.prototype._createProgressContent = function() {
        var content = document.createElement('div'), 
            outer = document.createElement('div'),
            inner = document.createElement('div');
        content.className = 'ui-content';
        outer.className = 'ui-outer';
        inner.className = 'ui-inner';
        outer.appendChild(inner);
        content.appendChild(outer)

        this.innerBar = inner;

        return content;
    }

    /**
     * 进度条初始化
     * @return {[type]} [description]
     */
    Progress.prototype.init = function() {
        var text, content, bar, _this = this;
        // 创建mask
        this._createMask();
        // 创建warpper
        this._createWrapper();

        this.dialogWrapper.className = 'ui-dialog ui-progress fade';

        // 1. 创建文字
        text = this._createProgressText();
        // 2. 创建内容
        content = this._createProgressContent();
        // 3. 将文字和content加入到warpper中
        this.dialogWrapper.appendChild(text);
        this.dialogWrapper.appendChild(content);
        // 4. 将warpper加入到body中
        doc.appendChild(this.dialogWrapper);

        setTimeout(function() {
            _this.dialogWrapper.className = 'ui-dialog ui-progress fade in';
        }, this.fadeInTime);
    }

    /**
     * 创建进度条
     * @return {[type]} [description]
     */
    Progress.prototype.create = function() {
        var args = arguments && arguments[0],
            length = args && args.length;

        if (length === 1) {
            this.text = args[0];
        }

        this.type = 'progress';
        this.init();
        return this;
    }

    /**
     * 更新进度条进度
     * @return {[type]} [description]
     */
    Progress.prototype.update = function(percent) {
        if (this.percent === 100) {
            this.close();
        }
        this.percent = percent;
        this.innerBar.style.width = percent + '%';
    }

    /**
     * 关闭进度条
     * @return {[type]} [description]
     */
    Progress.prototype.closeBar = function() {
        this.percent = 0;
        this.close();
    }

    /**
     * 获取dialog实例
     * @return {[type]} [description]
     */
    function _getInstance() {
        if (!_instance) {
            _instance = new Dialog();
        } 
        return _instance;
    }

    /**
     * 获取进度条实例
     * @return {[type]} [description]
     */
    function _getProgressInstance() {
        if (!_progressInstance) {
            _progressInstance = new Progress();
        }
        return _progressInstance;
    }

    /**
     * return 调用
     */
    return {
        alert : function() {
            return _getInstance().confirm(arguments);
        },
        confirm : function() {
            return _getInstance().confirm(arguments);
        },
        toast : function() {
            return _getInstance().toast(arguments);
        },
        showLoading : function() {
            return _getInstance().showLoading(arguments);
        },
        progress : function() {
            return _getProgressInstance().create(arguments);
        },
        close : function() {
            return _getInstance().close();
        }
    }

})(document);

展示

alert

alert

confirm

confirm

loading

loading

toast

toast

progress

progress

gif

gif

点击查看更多内容
5人点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
JAVA开发工程师
手记
粉丝
70
获赞与收藏
513

关注作者,订阅最新文章

阅读免费教程

感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消