vue插件实现弹框+滚动锁屏

1、实现该插件的背景

由于公司活动页面涉及弹框交互的操作比较多,为了实现弹框show出来的时候,可以在让背景元素在当前位置锁住,提高用户的体验,因此撸了一段代码,完美实现了此需求,并且解决了滚动穿透的问题

2、解决方案

方案一:
当弹框show出时,设置根元素和body元素的高度为屏幕高度,并且设置overflow为hidden,关闭弹框的时候移除样式

方案二:
弹框show出时,设置body元素的position为fixed和width属性为100%,并且设置滚动条的位置

3、实现过程遇到的问题

方案一:
(1)、可以实现弹框show出来,但是存在滚动穿透的问题,在touch时,背景元素还是会滚动,并且滚动条会自动滚到顶部;
(2)、不能记录到当前的位置
为了解决滚动穿透问题,实现过程给背景元素添加禁止touchmove事件的默认行为,但是在ios的safari和chrome浏览器下兼容性有问题
实现绑定事件函数如下:

function addEvent(oNode, event, func, capture) {
    if (typeof (document.attachEvent) != "undefined") {
        oNode.attachEvent("on" + event, func);
    } else {
        oNode.addEventListener(event, func, capture);
    }
}

function removeEvent(oNode, event, func, capture) {
    if (typeof (document.detachEvent) != "undefined") {
        oNode.detachEvent("on" + event, func);
    } else {
        oNode.removeEventListener(event, func, capture);
    }
}

并且在Vue的原型对象上实现$lock函数,在lock函数中实现body根元素样式添加以及绑定touchmove事件,但是在实现过程中绑定的touchmove事件不起作用,无法实现预期效果,查其原因ios对touchmove和touchstart事件支持有问题

方案二:
在方案一的基础上做出改变,采用position和width属性实现与height,overflow同样的效果,并且采用新的属性scrollingElement获取滚动条位置,设置body元素的top属性为滚动条的位置的负数,保证弹框show出时,停留在当前位置

两种方案实现比较,第一种方案可以实现锁屏,但是不完美,存在滚动穿透问题,并且在用户体验上不佳,第二种方案可以完美解决方案一存在的问题,但是引用了新的属性,需要引入一个 polyfill document.scrollingElement.js(感谢yangg提供scrollingElemeng.js)

最终实现依赖如下代码:

var screen =(function(configCss) {
	     return {
	     afterOpen: function () {
                scrollTop = document.scrollingElement.scrollTop;
                document.body.style.position = configCss.position;
                document.body.style.width = configCss.width;
                document.body.style.top = -scrollTop + 'px';
            },
            beforeClose: function () {
                document.body.style = null;
                document.scrollingElement.scrollTop = scrollTop;
            }
       }
})(config)
vueLockScreen.install = function (Vue) {
        //实现锁屏
        Vue.prototype.$lock = function () {
            screen.afterOpen();
        };
        Vue.prototype.$unlock = function () {
            screen.beforeClose();
        };
    };

你可能感兴趣的:(javascript)