xyyd-fatfox/src/layout/grid/BlockWrapper.tsx

188 lines
6.4 KiB
TypeScript
Raw Normal View History

2024-09-25 18:23:54 +08:00
import { defineComponent, ref, toRaw } from 'vue'
import type { Block } from '../layout.types'
import { dragging, resetDragging } from './useSortable'
import { v4 as uuid } from 'uuid'
import useLayoutStore from '../useLayoutStore'
import LinkBlock from './LinkBlock'
import DirBlock from './DirBlock'
2024-09-29 15:18:00 +08:00
import WidgetBlock from './WidgetBlock'
2024-10-10 16:04:00 +08:00
import useSettingsStore from '@/settings/useSettingsStore'
import { useMenuStore } from '../GlobalMenu'
import { block } from '@milkdown/kit/plugin/block'
import clsx from 'clsx'
2024-09-25 18:23:54 +08:00
export default defineComponent({
props: {
block: {
type: Object as () => Block,
required: true
},
idx: {
type: Number,
required: true
}
},
setup(props) {
2024-10-10 16:04:00 +08:00
const settings = useSettingsStore()
2024-09-25 18:23:54 +08:00
const layout = useLayoutStore()
const menu = useMenuStore()
2024-11-07 18:48:45 +08:00
let it: any = 0
2024-09-25 18:23:54 +08:00
const hover = ref(false)
return () => (
<div
class={clsx(
'w-full h-full p-[var(--block-padding)] relative rounded-lg ',
menu.isEditPage && 'animate-wiggle'
)}
2024-09-25 18:23:54 +08:00
key={props.block.id}
id={props.block.id}
style={{
gridColumn: `span ${props.block.w}`,
gridRow: `span ${props.block.h}`,
transition: 'border .3s, transform .2s',
2024-09-25 18:23:54 +08:00
border: hover.value ? '2px solid rgba(255,255,255,.5)' : '2px solid rgba(255,255,255,0)'
}}
data-transportable={props.block.link && !props.block.link.startsWith('id:') ? '1' : ''}
onDragover={(e) => {
e.preventDefault()
clearTimeout(it)
hover.value = true
it = setTimeout(() => {
hover.value = false
}, 300)
}}
onDrop={() => {
// 处理移入
if (!dragging.id) return
if (dragging.type === 'dock') {
const oldIdx = layout.state.dock.findIndex((el) => el?.id === dragging.id)
if (oldIdx < 0) return
const block = layout.state.dock[oldIdx]
if (!block) return
layout.currentPage.list.splice(props.idx, 0, block)
layout.state.dock[oldIdx] = null
return
}
if (dragging.type === 'page' && dragging.id !== props.block.id) {
// 合并为文件夹
const link = props.block.link
// 小组件无法合并
if (!link) return
2024-10-28 18:51:44 +08:00
//游戏不能合并
if (props.block.w !== 1) return
2024-09-25 18:23:54 +08:00
const oldIdx = layout.currentPage.list.findIndex((el) => el.id === dragging.id)
if (oldIdx < 0) return
const oldBlock = layout.currentPage.list[oldIdx]
2024-10-28 18:51:44 +08:00
//游戏不能合并
if (oldBlock.w !== 1) return
2024-09-25 18:23:54 +08:00
// 文件夹不能移入文件夹
if (!oldBlock || oldBlock.link.startsWith('id:')) return
if (link.startsWith('id:')) {
// 本身就是文件夹
const id = link.slice(3)
const dir = layout.state.dir[id]
if (!dir) return
dir.list.push(toRaw(oldBlock))
layout.currentPage.list.splice(oldIdx, 1)
resetDragging()
} else {
// 本身不是文件夹
const id = props.block.id
layout.state.dir[id] = {
label: props.block.label,
list: [toRaw(props.block), toRaw(oldBlock)]
}
layout.currentPage.list.splice(props.idx, 1)
layout.currentPage.list.splice(props.idx, 0, {
id: uuid(),
link: `id:${id}`,
name: '',
label: '新建文件夹',
icon: '',
text: '',
background: '',
color: '',
w: 1,
h: 1
})
layout.currentPage.list.splice(oldIdx, 1)
resetDragging()
}
}
if (dragging.type && dragging.type !== 'dock' && dragging.type !== 'page') {
const list = layout.state.dir[dragging.type]?.list
if (!list) return
const idx = list.findIndex((el) => el.id === dragging.id)
if (idx < 0) return
const block = list[idx]
layout.currentPage.list.splice(props.idx, 0, toRaw(block))
list.splice(idx, 1)
layout.checkDir(dragging.type)
resetDragging()
}
}}
>
{menu.isEditPage && (
<div
v-outside-click={() => {
menu.isEditPage = false
}}
onClick={(e) => {
e.stopPropagation()
const idx = layout.state.content[layout.state.current].pages[
layout.state.currentPage
].list.findIndex((val) => val.id === props.block.id)
if (idx < 0) return
layout.state.content[layout.state.current].pages[
layout.state.currentPage
].list.splice(idx, 1)
}}
class={
'rounded-full cursor-pointer backdrop-blur-md absolute w-[20px] h-[20px] top-[8px] right-[12px] z-10 '
}
style={{
backgroundImage: `url('/tab/bg/del_icon_img.png')`,
backgroundSize: 'cover'
}}
></div>
)}
2024-10-23 14:57:27 +08:00
<div
class="w-full h-full overflow-hidden relative cursor-pointer shadow-lg hover-move "
2024-10-23 14:57:27 +08:00
style={{
borderRadius: `calc(var(--block-radius) * var(--block-size))`,
transition: 'transform 0.2s'
2024-10-23 14:57:27 +08:00
}}
2024-11-11 11:53:26 +08:00
onContextmenu={(e) => {
e.stopPropagation()
e.preventDefault()
}}
2024-10-23 14:57:27 +08:00
>
{props.block.link ? (
props.block.link.startsWith('id:') ? (
// 文件夹
<DirBlock block={props.block} big={props.block.w !== 1 || props.block.h !== 1} />
2024-10-22 16:27:19 +08:00
) : (
2024-10-23 14:57:27 +08:00
// 链接
<LinkBlock block={props.block} />
)
) : (
// 小组件
<WidgetBlock block={props.block} />
2024-10-22 16:27:19 +08:00
)}
</div>
2024-10-23 14:57:27 +08:00
{settings.state.showBlockLabel && (
<div
class="absolute left-1/2 -translate-x-1/2 -bottom-3 text-sm text-white text-center w-[172px] overflow-hidden text-ellipsis whitespace-nowrap break-all font-bold"
2024-10-23 14:57:27 +08:00
style="text-shadow: 0 0 4px rgba(0,0,0,.6)"
>
{layout.getLabel(props.block)}
</div>
)}
2024-09-25 18:23:54 +08:00
</div>
)
}
})