<!-- msw.svelte -->
<div class="msw-container">
<div on:click|stopPropagation={showModal}
bind:this={btnDOM}
class="msw-show">MSW</div>
</div>
<script>
import { onMount } from "svelte";
// 区分当前是PC端,还是移动端,来设置 mouse 事件 或 touch 事件
function getModels() {
let userAgentInfo = navigator.userAgent;
let mobileAgents = ["Android", "iPhone", "SymbianOS", "Windows Phone", "iPad", "iPod"];
return mobileAgents.reduce((prev, ua)=>{
return userAgentInfo.includes(ua) || prev
}, false)
};
const MSW_BTN_POSITION = '__MSW_BTN_POSITION__'
let show = false;
let btnDOM = null;
let isDrop = false;
let isMoving = false;
let offset = {
x: 0,
y: 0,
};
let offsetDown = {};
let dropTimer = null;
let isMobile = getModels();
let btnW = 0;
let btnH = 0;
let clientW = 0;
let clientH = 0;
let eventType = isMobile ? 'touchstart' : 'mousedown';
// DOM 挂载后执行
onMount(async () => {
// 初始化,获取按钮、视口宽高、计算边界值
initClientData();
return () => {
// component 卸载后,解除事件绑定
btnDOM.removeEventListener(eventType, btnMousedown)
}
});
function initClientData() {
// 按钮位置保存在本地,可以记住位置,避免每次去拖拽
let local = localStorage.getItem(MSW_BTN_POSITION)
if (local) {
offset = JSON.parse(local)
btnMove()
}
let w = window.innerWidth || document.documentElement.clientWidth || document.body.clientWidth
clientW = isMobile ? w : document.body.clientWidth
clientH = window.innerHeight || document.documentElement.clientHeight || document.body.clientHeight
btnW = btnDOM.offsetWidth
btnH = btnDOM.offsetHeight
// 给按钮绑定 mousedown 或 touchstart 事件
btnDOM.addEventListener(eventType, btnMousedown)
}
function eventHandle (type) {
if (isMobile) {
document[`${type}EventListener`]('touchmove', mousemove);
document[`${type}EventListener`]('touchend', mouseup);
} else {
document[`${type}EventListener`]('mousemove', mousemove);
document[`${type}EventListener`]('mouseup', mouseup);
}
}
function showModal () {
if (!isMoving) {
show = true;
}
}
function btnMousedown(e) {
e = e || window.event
isDrop = true
offsetDown = {
...getOffset(e)
};
eventHandle('add')
}
function mousemove(e) {
e = e || window.event
if (isDrop) {
let data = getOffset(e);
// 判断是否移动了
isMoving = !(offsetDown.x === data.x && offsetDown.y === data.y)
let x = data.x - btnW / 2;
let y = data.y - btnH / 2;
if (x > 5 && x < (clientW-btnW - 5)) {
offset.x = x;
}
if (y > 5 && y < (clientH-btnH - 5)) {
offset.y = y;
}
if (isMoving) {
btnMove()
}
clearTimeout(dropTimer);
dropTimer = setTimeout(()=>{
isMoving = false;
clearTimeout(dropTimer);
dropTimer = null;
}, 300);
}
}
function mouseup() {
if (isDrop) {
window.localStorage.setItem(MSW_BTN_POSITION, JSON.stringify(offset))
eventHandle('remove')
}
isDrop = false
// console.log('mouseup')
}
function btnMove (){
btnDOM.style.cssText = `
left: ${offset.x}px;
top: ${offset.y}px;
right: auto;
bottom: auto;
`
}
function getOffset(e) {
return isMobile ? {
x: e.targetTouches[0].clientX,
y: e.targetTouches[0].clientY,
} : {
x: e.clientX,
y: e.clientY,
}
}
</script>
<style lang="scss" type="text/scss">
@import "index";
</style>