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();
};
};