mirror of
https://gitee.com/vant-contrib/vant.git
synced 2025-04-06 03:57:59 +08:00
169 lines
3.6 KiB
JavaScript
169 lines
3.6 KiB
JavaScript
import manager from './manager';
|
|
import context from './context';
|
|
import scrollUtils from '../../utils/scroll';
|
|
import { on, off } from '../../utils/event';
|
|
import Touch from '../touch';
|
|
|
|
export default {
|
|
mixins: [Touch],
|
|
|
|
props: {
|
|
// whether to show popup
|
|
value: Boolean,
|
|
// whether to show overlay
|
|
overlay: Boolean,
|
|
// overlay custom style
|
|
overlayStyle: Object,
|
|
// overlay custom class name
|
|
overlayClass: String,
|
|
// whether to close popup when click overlay
|
|
closeOnClickOverlay: Boolean,
|
|
// z-index
|
|
zIndex: [String, Number],
|
|
// return the mount node for popup
|
|
getContainer: Function,
|
|
// prevent body scroll
|
|
lockScroll: {
|
|
type: Boolean,
|
|
default: true
|
|
}
|
|
},
|
|
|
|
watch: {
|
|
value(val) {
|
|
this[val ? 'open' : 'close']();
|
|
},
|
|
|
|
getContainer() {
|
|
this.move();
|
|
},
|
|
|
|
overlay() {
|
|
this.renderOverlay();
|
|
}
|
|
},
|
|
|
|
created() {
|
|
this._popupId = 'popup-' + context.plusKey('id');
|
|
},
|
|
|
|
mounted() {
|
|
if (this.getContainer) {
|
|
this.move();
|
|
}
|
|
if (this.value) {
|
|
this.open();
|
|
}
|
|
},
|
|
|
|
activated() {
|
|
/* istanbul ignore next */
|
|
if (this.value) {
|
|
this.open();
|
|
}
|
|
},
|
|
|
|
beforeDestroy() {
|
|
this.close();
|
|
},
|
|
|
|
deactivated() {
|
|
/* istanbul ignore next */
|
|
this.close();
|
|
},
|
|
|
|
methods: {
|
|
open() {
|
|
/* istanbul ignore next */
|
|
if (this.$isServer || this.opened) {
|
|
return;
|
|
}
|
|
|
|
// 如果属性中传入了`zIndex`,则覆盖`context`中对应的`zIndex`
|
|
if (this.zIndex !== undefined) {
|
|
context.zIndex = this.zIndex;
|
|
}
|
|
|
|
this.opened = true;
|
|
this.renderOverlay();
|
|
|
|
if (this.lockScroll) {
|
|
on(document, 'touchstart', this.touchStart);
|
|
on(document, 'touchmove', this.onTouchMove);
|
|
if (!context.lockCount) {
|
|
document.body.classList.add('van-overflow-hidden');
|
|
}
|
|
context.lockCount++;
|
|
}
|
|
},
|
|
|
|
close() {
|
|
if (!this.opened) {
|
|
return;
|
|
}
|
|
|
|
if (this.lockScroll) {
|
|
context.lockCount--;
|
|
off(document, 'touchstart', this.touchStart);
|
|
off(document, 'touchmove', this.onTouchMove);
|
|
if (!context.lockCount) {
|
|
document.body.classList.remove('van-overflow-hidden');
|
|
}
|
|
}
|
|
|
|
this.opened = false;
|
|
manager.close(this._popupId);
|
|
this.$emit('input', false);
|
|
},
|
|
|
|
move() {
|
|
if (this.getContainer) {
|
|
this.getContainer().appendChild(this.$el);
|
|
} else if (this.$parent) {
|
|
this.$parent.$el.appendChild(this.$el);
|
|
}
|
|
},
|
|
|
|
onTouchMove(e) {
|
|
this.touchMove(e);
|
|
const direction = this.deltaY > 0 ? '10' : '01';
|
|
const el = scrollUtils.getScrollEventTarget(e.target, this.$el);
|
|
const { scrollHeight, offsetHeight, scrollTop } = el;
|
|
let status = '11';
|
|
|
|
/* istanbul ignore next */
|
|
if (scrollTop === 0) {
|
|
status = offsetHeight >= scrollHeight ? '00' : '01';
|
|
} else if (scrollTop + offsetHeight >= scrollHeight) {
|
|
status = '10';
|
|
}
|
|
|
|
/* istanbul ignore next */
|
|
if (
|
|
status !== '11' &&
|
|
this.direction === 'vertical' &&
|
|
!(parseInt(status, 2) & parseInt(direction, 2))
|
|
) {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
}
|
|
},
|
|
|
|
renderOverlay() {
|
|
if (this.overlay) {
|
|
manager.open(this, {
|
|
zIndex: context.plusKey('zIndex'),
|
|
className: this.overlayClass,
|
|
customStyle: this.overlayStyle
|
|
});
|
|
} else {
|
|
manager.close(this._popupId);
|
|
}
|
|
|
|
this.$nextTick(() => {
|
|
this.$el.style.zIndex = context.plusKey('zIndex');
|
|
});
|
|
}
|
|
}
|
|
};
|