You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
315 lines
7.8 KiB
Vue
315 lines
7.8 KiB
Vue
6 months ago
|
<template>
|
||
|
<view class="ui-popover" :class="popover ? 'show' : 'hide'">
|
||
|
<view
|
||
|
class="ui-popover-button"
|
||
|
:class="[ui]"
|
||
|
:id="'popover-button-' + elId"
|
||
|
:style="{ zIndex: index + zIndexConfig.popover }"
|
||
|
@tap="popoverClick"
|
||
|
@mouseleave="mouseleave"
|
||
|
@mouseover="mouseover"
|
||
|
>
|
||
|
<slot></slot>
|
||
|
</view>
|
||
|
<view class="ui-popover-box" :style="BoxStyle">
|
||
|
<view class="ui-popover-content-box" :id="'popover-content-' + elId" :style="contentStyle">
|
||
|
<view
|
||
|
class="ui-popover-content radius text-a"
|
||
|
:class="bg"
|
||
|
:style="{ zIndex: index + zIndexConfig.popover + 2 }"
|
||
|
>
|
||
|
<view class="p-3 text-sm" v-if="tips">{{ tips }}</view>
|
||
|
<block v-else><slot name="content" /></block>
|
||
|
</view>
|
||
|
<view class="ui-popover-arrow" :class="bg" :style="arrowStyle"></view>
|
||
|
</view>
|
||
|
</view>
|
||
|
<view
|
||
|
class="ui-popover-mask"
|
||
|
:class="mask ? 'bg-mask-50' : ''"
|
||
|
:style="{ zIndex: index + zIndexConfig.popover - 1 }"
|
||
|
@tap="popover = false"
|
||
|
v-if="(popover && tips == '' && time == 0) || mask"
|
||
|
></view>
|
||
|
</view>
|
||
|
</template>
|
||
|
|
||
|
<script>
|
||
|
import { guid } from '@/sheep/helper';
|
||
|
import zIndexConfig from '@/sheep/config/zIndex.js';
|
||
|
import sheep from '@/sheep';
|
||
|
|
||
|
export default {
|
||
|
name: 'suPopover',
|
||
|
data() {
|
||
|
return {
|
||
|
elId: guid(),
|
||
|
zIndexConfig,
|
||
|
popover: false,
|
||
|
BoxStyle: '',
|
||
|
contentStyle: '',
|
||
|
arrowStyle: '',
|
||
|
button: {},
|
||
|
content: {},
|
||
|
};
|
||
|
},
|
||
|
props: {
|
||
|
ui: {
|
||
|
type: String,
|
||
|
default: '',
|
||
|
},
|
||
|
tips: {
|
||
|
type: String,
|
||
|
default: '',
|
||
|
},
|
||
|
bg: {
|
||
|
type: String,
|
||
|
default: 'ui-BG',
|
||
|
},
|
||
|
mask: {
|
||
|
type: Boolean,
|
||
|
default: false,
|
||
|
},
|
||
|
show: {
|
||
|
type: [Boolean, String],
|
||
|
default: 'change',
|
||
|
},
|
||
|
hover: {
|
||
|
type: Boolean,
|
||
|
default: false,
|
||
|
},
|
||
|
index: {
|
||
|
type: Number,
|
||
|
default: 0,
|
||
|
},
|
||
|
time: {
|
||
|
type: Number,
|
||
|
default: 0,
|
||
|
},
|
||
|
bottom: {
|
||
|
type: Boolean,
|
||
|
default: false,
|
||
|
},
|
||
|
isChange: {
|
||
|
type: Boolean,
|
||
|
default: false,
|
||
|
},
|
||
|
},
|
||
|
watch: {
|
||
|
popover(val) {
|
||
|
this._computedQuery(
|
||
|
sheep.$platform.device.windowWidth,
|
||
|
sheep.$platform.device.windowHeight,
|
||
|
);
|
||
|
if (val) {
|
||
|
if (this.tips != '' || this.time > 0) {
|
||
|
setTimeout(
|
||
|
() => {
|
||
|
this.popover = false;
|
||
|
},
|
||
|
this.time == 0 ? 3000 : this.time,
|
||
|
);
|
||
|
}
|
||
|
this.sys_layer = this.sys_layer + 100;
|
||
|
} else {
|
||
|
this.sys_layer = this.sys_layer - 100;
|
||
|
}
|
||
|
this.$emit('update:show', val);
|
||
|
},
|
||
|
show(val) {
|
||
|
this.popover = val;
|
||
|
},
|
||
|
},
|
||
|
mounted() {
|
||
|
this.$nextTick(() => {
|
||
|
this._computedQuery(
|
||
|
sheep.$platform.device.windowWidth,
|
||
|
sheep.$platform.device.windowHeight,
|
||
|
);
|
||
|
// #ifdef H5
|
||
|
uni.onWindowResize((res) => {
|
||
|
this._computedQuery(res.size.windowWidth, res.size.windowHeight);
|
||
|
});
|
||
|
// #endif
|
||
|
});
|
||
|
},
|
||
|
methods: {
|
||
|
_onHide() {
|
||
|
this.popover = false;
|
||
|
},
|
||
|
_computedQuery(w, h) {
|
||
|
uni
|
||
|
.createSelectorQuery()
|
||
|
.in(this)
|
||
|
.select('#popover-button-' + this.elId)
|
||
|
.boundingClientRect((button) => {
|
||
|
if (button != null) {
|
||
|
this.button = button;
|
||
|
} else {
|
||
|
console.log('popover-button-' + this.elId + ' data error');
|
||
|
}
|
||
|
})
|
||
|
.select('#popover-content-' + this.elId)
|
||
|
.boundingClientRect((content) => {
|
||
|
if (content != null) {
|
||
|
this.content = content;
|
||
|
let button = this.button;
|
||
|
//contentStyle
|
||
|
let contentStyle = '';
|
||
|
let arrowStyle = '';
|
||
|
this.BoxStyle = `width:${w}px; left:-${button.left}px;z-index: ${
|
||
|
this.index + this.sys_layer + 102
|
||
|
}`;
|
||
|
// 判断气泡在上面还是下面
|
||
|
if (button.bottom < h / 2 || this.bottom) {
|
||
|
// '下';
|
||
|
contentStyle = contentStyle + `top:10px;`;
|
||
|
arrowStyle = arrowStyle + `top:${-5}px;`;
|
||
|
} else {
|
||
|
// '上';
|
||
|
contentStyle = contentStyle + `bottom:${button.height + 10}px;`;
|
||
|
arrowStyle = arrowStyle + `bottom:${-5}px;`;
|
||
|
}
|
||
|
|
||
|
// 判断气泡箭头在左中右
|
||
|
let btnCenter = button.right - button.width / 2;
|
||
|
let contentCenter = content.right - content.width / 2;
|
||
|
if (
|
||
|
(btnCenter < w / 3 && content.width > btnCenter) ||
|
||
|
(content.width > w / 2 && btnCenter < w / 2)
|
||
|
) {
|
||
|
// '左';
|
||
|
contentStyle = contentStyle + `left:10px;`;
|
||
|
arrowStyle = arrowStyle + `left:${btnCenter - 17}px;`;
|
||
|
} else if (
|
||
|
(btnCenter > (w / 6) * 4 && content.width > w - btnCenter) ||
|
||
|
(content.width > w / 2 && btnCenter > w / 2)
|
||
|
) {
|
||
|
// '右';
|
||
|
contentStyle = contentStyle + `right:10px;`;
|
||
|
arrowStyle = arrowStyle + `right:${w - btnCenter - 17}px;`;
|
||
|
} else {
|
||
|
// '中';
|
||
|
contentStyle =
|
||
|
contentStyle + `left:${button.left - content.width / 2 + button.width / 2}px;`;
|
||
|
arrowStyle = arrowStyle + `left:0px;right:0px;margin:auto;`;
|
||
|
}
|
||
|
|
||
|
this.arrowStyle = arrowStyle + `z-index:${this.index + this.sys_layer + 1};`;
|
||
|
this.contentStyle = contentStyle + `z-index:${this.index + this.sys_layer + 2};`;
|
||
|
} else {
|
||
|
console.log('popover-content-' + this.elId + ' data error');
|
||
|
}
|
||
|
})
|
||
|
.exec();
|
||
|
},
|
||
|
popoverClick() {
|
||
|
if (this.isChange) {
|
||
|
return false;
|
||
|
}
|
||
|
if (this.tips == '') {
|
||
|
this.popover = !this.popover;
|
||
|
} else {
|
||
|
this.popover = true;
|
||
|
}
|
||
|
},
|
||
|
mouseover() {
|
||
|
if (this.hover && (this.tips != '' || this.content.height != 0)) {
|
||
|
this.popover = true;
|
||
|
}
|
||
|
},
|
||
|
mouseleave() {
|
||
|
if (this.hover) {
|
||
|
this.popover = false;
|
||
|
}
|
||
|
},
|
||
|
},
|
||
|
};
|
||
|
</script>
|
||
|
|
||
|
<style lang="scss">
|
||
|
.ui-popover {
|
||
|
position: relative;
|
||
|
|
||
|
.ui-popover-button {
|
||
|
position: relative;
|
||
|
}
|
||
|
|
||
|
.ui-popover-box {
|
||
|
position: absolute;
|
||
|
|
||
|
.ui-popover-content-box {
|
||
|
position: absolute;
|
||
|
|
||
|
.ui-popover-content {
|
||
|
position: relative;
|
||
|
}
|
||
|
|
||
|
.ui-popover-arrow {
|
||
|
position: absolute;
|
||
|
height: 15px;
|
||
|
width: 15px;
|
||
|
border-radius: 2px;
|
||
|
transform: rotate(45deg);
|
||
|
}
|
||
|
|
||
|
&::after {
|
||
|
content: '';
|
||
|
width: 100%;
|
||
|
height: 110%;
|
||
|
position: absolute;
|
||
|
background-color: #000000;
|
||
|
top: 5%;
|
||
|
left: 0;
|
||
|
filter: blur(15px);
|
||
|
opacity: 0.15;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
.ui-popover-mask {
|
||
|
position: fixed;
|
||
|
width: 100%;
|
||
|
height: 100%;
|
||
|
top: 0;
|
||
|
left: 0;
|
||
|
}
|
||
|
|
||
|
&.show {
|
||
|
.ui-popover-button {
|
||
|
}
|
||
|
|
||
|
.ui-popover-content-box {
|
||
|
opacity: 1;
|
||
|
pointer-events: auto;
|
||
|
}
|
||
|
|
||
|
.ui-popover-arrow {
|
||
|
display: block;
|
||
|
}
|
||
|
|
||
|
.ui-popover-mask {
|
||
|
display: block;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
&.hide {
|
||
|
.ui-popover-button {
|
||
|
}
|
||
|
|
||
|
.ui-popover-content-box {
|
||
|
opacity: 0;
|
||
|
pointer-events: none;
|
||
|
}
|
||
|
|
||
|
.ui-popover-arrow {
|
||
|
display: none;
|
||
|
}
|
||
|
|
||
|
.ui-popover-mask {
|
||
|
display: none;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
</style>
|