feat: add designBoard drag

This commit is contained in:
ShawnPhang 2024-04-02 14:47:54 +08:00
parent 79039c901f
commit 3da23d3642
13 changed files with 118 additions and 54 deletions

View File

@ -15,7 +15,7 @@
</head> </head>
<body> <body>
<div id="app"></div> <div class="pointer-case" id="app"></div>
<script type="module" src="/src/main.ts"></script> <script type="module" src="/src/main.ts"></script>
<script defer src="/snap.svg-min.js"></script> <script defer src="/snap.svg-min.js"></script>
<script async src="//busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js"></script> <script async src="//busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js"></script>

View File

@ -7,18 +7,26 @@
} }
body { body {
--el-color-primary: #2254f4; --el-color-primary: #2254f4;
cursor: -webkit-image-set(
url()
2x
)
6 2,
default;
-moz-osx-font-smoothing: grayscale; -moz-osx-font-smoothing: grayscale;
-webkit-font-smoothing: antialiased; -webkit-font-smoothing: antialiased;
color: #333333; color: #333333;
font-family: Hiragino Sans GB, Hiragino Sans GB W3, Arial, Microsoft Yahei, STHeiti, sans-serif; font-family: Hiragino Sans GB, Hiragino Sans GB W3, Arial, Microsoft Yahei, STHeiti, sans-serif;
overflow: hidden; overflow: hidden;
} }
.pointer-case {
cursor: -webkit-image-set(
url()
2x
)
6 2,
default;
}
.move-case {
cursor: grab;
}
.move-case:active {
cursor: grabbing;
}
* { * {
box-sizing: border-box; box-sizing: border-box;

View File

@ -1,10 +1,11 @@
import Selecto from 'selecto' import Selecto from 'selecto'
import Moveable, { getElementInfo } from 'moveable' import Moveable, { getElementInfo } from 'moveable'
// import store from '@/store' // import store from '@/store'
import { useWidgetStore } from '@/store' import { useWidgetStore, useControlStore } from '@/store'
export default function(moveable: Moveable) { export default function(moveable: Moveable) {
const widgetStore = useWidgetStore() const widgetStore = useWidgetStore()
const controlStore = useControlStore()
const selecto = new Selecto({ const selecto = new Selecto({
container: document.getElementById('page-design'), container: document.getElementById('page-design'),
selectableTargets: ['.layer'], selectableTargets: ['.layer'],
@ -45,5 +46,9 @@ export default function(moveable: Moveable) {
moveable.renderDirections = [] // ['nw', 'ne', 'sw', 'se'] // [] moveable.renderDirections = [] // ['nw', 'ne', 'sw', 'se'] // []
moveable.rotatable = false moveable.rotatable = false
moveable.target = [].slice.call(document.querySelectorAll('.widget-selected')) moveable.target = [].slice.call(document.querySelectorAll('.widget-selected'))
}).on("dragStart", e => {
if (controlStore.dSpaceDown) {
e.stop();
}
}) })
} }

View File

@ -105,7 +105,7 @@ const { pageDesignCanvasId } = defineProps<TProps>()
const { dPage } = storeToRefs(usePageStore()) const { dPage } = storeToRefs(usePageStore())
const { dZoom, dPaddingTop, dScreen } = storeToRefs(canvasStore) const { dZoom, dPaddingTop, dScreen } = storeToRefs(canvasStore)
const { dDraging, showRotatable, dAltDown } = storeToRefs(controlStore) const { dDraging, showRotatable, dAltDown, dSpaceDown } = storeToRefs(controlStore)
const { dWidgets, dActiveElement, dSelectWidgets, dHoverUuid } = storeToRefs(widgetStore) const { dWidgets, dActiveElement, dSelectWidgets, dHoverUuid } = storeToRefs(widgetStore)
@ -118,6 +118,32 @@ onMounted(() => {
if (!pageDesignEl) return if (!pageDesignEl) return
pageDesignEl.addEventListener('mousedown', handleSelection, false) pageDesignEl.addEventListener('mousedown', handleSelection, false)
pageDesignEl.addEventListener('mousemove', debounce(100, false, handleMouseMove), false) pageDesignEl.addEventListener('mousemove', debounce(100, false, handleMouseMove), false)
//
const scrollContainer: any = document.querySelector('#main')
const dragContainer: any = pageDesignEl
dragContainer.onmousedown = (e: any) => {
let mouseDownScrollPosition = {
scrollLeft: scrollContainer.scrollLeft,
scrollTop: scrollContainer.scrollTop,
}
let mouseDownPoint = {
x: e.clientX,
y: e.clientY,
}
dragContainer.onmousemove = (e: any) => {
if (!dSpaceDown.value) return
let dragMoveDiff = {
x: mouseDownPoint.x - e.clientX,
y: mouseDownPoint.y - e.clientY,
}
scrollContainer.scrollLeft = mouseDownScrollPosition.scrollLeft + dragMoveDiff.x
scrollContainer.scrollTop = mouseDownScrollPosition.scrollTop + dragMoveDiff.y
}
document.onmouseup = (e) => {
dragContainer.onmousemove = null
document.onmouseup = null
}
}
}) })
// components: {lineGuides}, // components: {lineGuides},

View File

@ -1,7 +1,7 @@
<!-- <!--
* @Author: ShawnPhang * @Author: ShawnPhang
* @Date: 2022-04-08 10:31:34 * @Date: 2022-04-08 10:31:34
* @Description: * @Description: 标尺
* @LastEditors: ShawnPhang <https://m.palxp.cn> * @LastEditors: ShawnPhang <https://m.palxp.cn>
* @LastEditTime: 2024-03-11 01:42:25 * @LastEditTime: 2024-03-11 01:42:25
--> -->

View File

@ -81,8 +81,6 @@ watch(
realValue = calcZoom() realValue = calcZoom()
} }
canvasStore.updateZoom(realValue) canvasStore.updateZoom(realValue)
// store.dispatch('updateZoom', realValue)
// updateZoom(realValue)
autoFixTop() autoFixTop()
} }
) )
@ -155,8 +153,6 @@ function screenChange() {
// //
if (activezoomIndex.value === zoomList.value.length - 1) { if (activezoomIndex.value === zoomList.value.length - 1) {
canvasStore.updateZoom(calcZoom()) canvasStore.updateZoom(calcZoom())
// store.dispatch('updateZoom', calcZoom())
// this.updateZoom(this.calcZoom())
autoFixTop() autoFixTop()
} }
} }
@ -225,10 +221,9 @@ function mousewheelZoom(down: boolean) {
const value = Number(dZoom.value.toFixed(0)) const value = Number(dZoom.value.toFixed(0))
if (down && value <= 1) return if (down && value <= 1) return
canvasStore.updateZoom(down ? value - 1 : value + 1) canvasStore.updateZoom(down ? value - 1 : value + 1)
// store.dispatch('updateZoom', down ? value - 1 : value + 1)
// updateZoom(down ? value - 1 : value + 1)
zoom.value.text = (value + '%') as any zoom.value.text = (value + '%') as any
autoFixTop() autoFixTop()
activezoomIndex.value = -99
} }
function nearZoom(add?: boolean) { function nearZoom(add?: boolean) {
@ -246,7 +241,6 @@ function nearZoom(add?: boolean) {
function calcZoom() { function calcZoom() {
let widthZoom = ((dScreen.value.width - 142) * 100) / dPage.value.width let widthZoom = ((dScreen.value.width - 142) * 100) / dPage.value.width
let heightZoom = ((dScreen.value.height - 122) * 100) / dPage.value.height let heightZoom = ((dScreen.value.height - 122) * 100) / dPage.value.height
bestZoom.value = Math.min(widthZoom, heightZoom) bestZoom.value = Math.min(widthZoom, heightZoom)
return bestZoom.value return bestZoom.value
} }

View File

@ -93,7 +93,6 @@ const targetRef = ref<HTMLImageElement | null>(null)
let rotateTemp: number | null = null let rotateTemp: number | null = null
let flipTemp: string | null = null let flipTemp: string | null = null
let locksTemp: boolean[] | null = null
// const { // const {
// dActiveElement, dWidgets, dMouseXY, dDropOverUuid, dCropUuid // dActiveElement, dWidgets, dMouseXY, dDropOverUuid, dCropUuid
@ -129,7 +128,7 @@ watch(
el?.removeEventListener('mousedown', touchstart, false) el?.removeEventListener('mousedown', touchstart, false)
} }
fixRotate() fixRotate()
lockOthers() lockOthers(val)
} }
) )
@ -314,21 +313,15 @@ function fixRotate() {
}, 100) }, 100)
} }
function lockOthers() { function lockOthers(isCrop) {
// //
if (locksTemp && locksTemp.length > 0) { widgetStore.lockWidgets()
for (let i = 0; i < locksTemp.length; i++) { if (!isCrop) return
dWidgets.value[i].lock = locksTemp[i] for (const widget of dWidgets.value) {
if (widget.uuid === props.params.uuid) {
widget.lock = false
break;
} }
locksTemp = []
} else {
locksTemp = []
for (const widget of dWidgets.value) {
locksTemp.push(widget?.lock || false)
}
dWidgets.value.forEach((widget: any) => {
widget.uuid != props.params.uuid && (widget.lock = true)
})
} }
} }
// cropDone(e) { // cropDone(e) {

View File

@ -334,10 +334,7 @@ function imgCrop(val: boolean) {
const { left, top } = el.getBoundingClientRect() const { left, top } = el.getBoundingClientRect()
toolBarStyle = { left: left + 'px', top: top + 'px' } toolBarStyle = { left: left + 'px', top: top + 'px' }
state.innerElement.cropEdit = val state.innerElement.cropEdit = val
// store.commit('setShowRotatable', !val)
controlStore.setShowRotatable(!val) controlStore.setShowRotatable(!val)
} }

View File

@ -7,13 +7,13 @@
*/ */
import { useControlStore, useWidgetStore } from '@/store' import { useControlStore, useWidgetStore } from '@/store'
import { TdWidgetData } from '@/store/design/widget' import { TdWidgetData } from '@/store/design/widget'
// import store from '@/store'
const appContainer: any = document.querySelector('#app')
const controlStore = useControlStore()
const widgetStore = useWidgetStore()
export default function keyCodeOptions(e: any, params: any) { export default function keyCodeOptions(e: any, params: any) {
const { f } = params const { f } = params
const controlStore = useControlStore()
const widgetStore = useWidgetStore()
switch (e.keyCode) { switch (e.keyCode) {
case 38: case 38:
udlr('top', -1 * f, e) udlr('top', -1 * f, e)
@ -49,6 +49,10 @@ export default function keyCodeOptions(e: any, params: any) {
} }
break break
} }
if (e.key === ' ') {
dealWithSpace(e)
}
} }
/** /**
* *
@ -81,15 +85,18 @@ function udlr(type: keyof TdWidgetData, value: any, event: any) {
key: type, key: type,
value: result, value: result,
}) })
// store.dispatch('updateWidgetData', { }
// uuid: store.getters.dActiveElement.uuid, }
// key: type,
// value: result, function dealWithSpace(event: any) {
// }) const widgetStore: any = useWidgetStore()
// 防止编辑文字时空格按不出来
// TODO: 键盘移位需要防抖入栈 if (!widgetStore.dActiveElement.editable) {
// timer = setTimeout(() => { event.preventDefault()
// this.pushHistory() appContainer.classList.add('move-case');
// }, 100) if (!controlStore.dSpaceDown) {
widgetStore.lockWidgets()
}
controlStore.setSpaceDown(true)
} }
} }

View File

@ -12,6 +12,7 @@
import keyCodeOptions from './methods/keyCodeOptions' import keyCodeOptions from './methods/keyCodeOptions'
import dealWithCtrl from './methods/dealWithCtrl' import dealWithCtrl from './methods/dealWithCtrl'
import { TControlStore } from '@/store/design/control' import { TControlStore } from '@/store/design/control'
import { useControlStore, useWidgetStore } from '@/store'
const ignoreNode = ['INPUT', 'TEXTAREA'] const ignoreNode = ['INPUT', 'TEXTAREA']
@ -31,6 +32,9 @@ const systemKeyCode = [
] ]
let hadDown = false let hadDown = false
const appContainer: any = document.querySelector('#app')
const controlStore = useControlStore()
const widgetStore = useWidgetStore()
const shortcuts = { const shortcuts = {
methods: { methods: {
@ -118,18 +122,20 @@ const shortcuts = {
}, },
handleKeyup(store: TControlStore, checkCtrl: number | undefined) { handleKeyup(store: TControlStore, checkCtrl: number | undefined) {
return (e: any) => { return (e: any) => {
console.log(e)
clearInterval(checkCtrl) clearInterval(checkCtrl)
hadDown = false hadDown = false
if (e.key === 'Alt' || e.key === 'Shift' || e.key === 'Control' || e.key === 'Meta') { if (e.key === 'Alt' || e.key === 'Shift' || e.key === 'Control' || e.key === 'Meta') {
store.updateAltDown(false) store.updateAltDown(false)
// store.dispatch('updateAltDown', false) }
if (e.key === ' ') {
appContainer.classList.remove('move-case');
controlStore.setSpaceDown(false)
widgetStore.lockWidgets()
} }
} }
}, },
dealCtrl(e: any, instance: any) { dealCtrl(e: any, instance: any) {
dealWithCtrl(e, instance) dealWithCtrl(e, instance)
console.log(e.key, e.keyCode)
}, },
}, },
} }

View File

@ -25,7 +25,9 @@ type TControlState = {
showRotatable: boolean showRotatable: boolean
/** 记录是否按下alt键 / 或ctrl */ /** 记录是否按下alt键 / 或ctrl */
dAltDown: boolean dAltDown: boolean
/** 正在编辑or裁剪的组件id */ // 是否按下空格键
dSpaceDown: boolean
/** orid *
dCropUuid: string dCropUuid: string
} }
@ -43,6 +45,7 @@ type TControlAction = {
stopDMove: () => void stopDMove: () => void
/** 设置正在裁剪or编辑的组件 */ /** 设置正在裁剪or编辑的组件 */
setCropUuid: (uuid: string) => void setCropUuid: (uuid: string) => void
setSpaceDown: (uuid: boolean) => void // 设置是否按下空格键
} }
/** 全局控制配置 */ /** 全局控制配置 */
@ -56,6 +59,7 @@ const ControlStore = defineStore<"controlStore", TControlState, {}, TControlAct
showRotatable: true, // 是否显示moveable的旋转按钮 showRotatable: true, // 是否显示moveable的旋转按钮
dAltDown: false, // 记录是否按下alt键 / 或ctrl dAltDown: false, // 记录是否按下alt键 / 或ctrl
dCropUuid: '-1', // 正在编辑or裁剪的组件id dCropUuid: '-1', // 正在编辑or裁剪的组件id
dSpaceDown: false, // 记录是否按下空格键
}), }),
getters: {}, getters: {},
actions: { actions: {
@ -108,6 +112,9 @@ const ControlStore = defineStore<"controlStore", TControlState, {}, TControlAct
// 设置正在裁剪or编辑的组件 // 设置正在裁剪or编辑的组件
this.dCropUuid = uuid this.dCropUuid = uuid
}, },
setSpaceDown(val: boolean) {
this.dSpaceDown = val
}
} }
}) })

View File

@ -235,3 +235,22 @@ export function setWidgetStyle(state: TWidgetStore, { uuid, key, value, pushHist
export function setDWidgets(state: TWidgetStore, e: TdWidgetData[]) { export function setDWidgets(state: TWidgetStore, e: TdWidgetData[]) {
state.dWidgets = e state.dWidgets = e
} }
// 锁定所有图层 / 再次调用时还原图层
let lastLocks: boolean[] | null = null
export function lockWidgets(state: TWidgetStore) {
if (lastLocks && lastLocks.length > 0) {
for (let i = 0; i < lastLocks.length; i++) {
state.dWidgets[i].lock = lastLocks[i]
}
lastLocks = []
} else {
lastLocks = []
for (const widget of state.dWidgets) {
lastLocks.push(widget?.lock || false)
}
state.dWidgets.forEach((widget: any) => {
widget.lock = true
})
}
}

View File

@ -10,7 +10,7 @@ import { Store, defineStore } from "pinia";
import { TInidDMovePayload, TMovePayload, dMove, initDMove, setDropOver, setMouseEvent, setdActiveElement, updateGroupSize, updateHoverUuid } from "./actions"; import { TInidDMovePayload, TMovePayload, dMove, initDMove, setDropOver, setMouseEvent, setdActiveElement, updateGroupSize, updateHoverUuid } from "./actions";
import { TPageState } from "../page"; import { TPageState } from "../page";
import { TInitResize, TResize, TdResizePayload, dResize, initDResize, resize } from "./actions/resize"; import { TInitResize, TResize, TdResizePayload, dResize, initDResize, resize } from "./actions/resize";
import { TUpdateWidgetMultiplePayload, TUpdateWidgetPayload, TsetWidgetStyleData, addWidget, deleteWidget, setDWidgets, setWidgetStyle, updateWidgetData, updateWidgetMultiple } from "./actions/widget"; import { TUpdateWidgetMultiplePayload, TUpdateWidgetPayload, TsetWidgetStyleData, addWidget, deleteWidget, setDWidgets, setWidgetStyle, updateWidgetData, updateWidgetMultiple, lockWidgets } from "./actions/widget";
import { addGroup } from "./actions/group"; import { addGroup } from "./actions/group";
import { setTemplate } from "./actions/template"; import { setTemplate } from "./actions/template";
import { copyWidget, pasteWidget } from "./actions/clone"; import { copyWidget, pasteWidget } from "./actions/clone";
@ -116,6 +116,7 @@ type TAction = {
resize: (data: TResize) => void resize: (data: TResize) => void
setWidgetStyle: (data: TsetWidgetStyleData) => void setWidgetStyle: (data: TsetWidgetStyleData) => void
setDWidgets: (data: TdWidgetData[]) => void setDWidgets: (data: TdWidgetData[]) => void
lockWidgets: () => void
setMouseEvent: (e: MouseEvent | null) => void setMouseEvent: (e: MouseEvent | null) => void
setdActiveElement: (data: TdWidgetData) => void setdActiveElement: (data: TdWidgetData) => void
} }
@ -176,6 +177,7 @@ const WidgetStore = defineStore<"widgetStore", TWidgetState, TGetter, TAction>("
resize(data) { resize(this, data) }, resize(data) { resize(this, data) },
setWidgetStyle(data) { setWidgetStyle(this, data) }, setWidgetStyle(data) { setWidgetStyle(this, data) },
setDWidgets(data) { setDWidgets(this, data) }, setDWidgets(data) { setDWidgets(this, data) },
lockWidgets() { lockWidgets(this) },
setMouseEvent(event) { setMouseEvent(this, event) }, setMouseEvent(event) { setMouseEvent(this, event) },
setdActiveElement(data) { setdActiveElement(this, data) }, setdActiveElement(data) { setdActiveElement(this, data) },
} }