121 lines
2.9 KiB
TypeScript
121 lines
2.9 KiB
TypeScript
import type { EmitType } from '@vben-core/typings';
|
||
|
||
import type { TabsProps } from './types';
|
||
|
||
import { nextTick, onMounted, onUnmounted, ref, watch } from 'vue';
|
||
|
||
import {
|
||
type Sortable,
|
||
useIsMobile,
|
||
useSortable,
|
||
} from '@vben-core/composables';
|
||
|
||
// 可能会找到拖拽的子元素,这里需要确保拖拽的dom时tab元素
|
||
function findParentElement(element: HTMLElement) {
|
||
const parentCls = 'group';
|
||
return element.classList.contains(parentCls)
|
||
? element
|
||
: element.closest(`.${parentCls}`);
|
||
}
|
||
|
||
export function useTabsDrag(props: TabsProps, emit: EmitType) {
|
||
const sortableInstance = ref<null | Sortable>(null);
|
||
|
||
async function initTabsSortable() {
|
||
await nextTick();
|
||
|
||
const el = document.querySelectorAll(
|
||
`.${props.contentClass}`,
|
||
)?.[0] as HTMLElement;
|
||
|
||
if (!el) {
|
||
console.warn('Element not found for sortable initialization');
|
||
return;
|
||
}
|
||
|
||
const resetElState = async () => {
|
||
el.style.cursor = 'default';
|
||
// el.classList.remove('dragging');
|
||
el.querySelector('.draggable')?.classList.remove('dragging');
|
||
};
|
||
|
||
const { initializeSortable } = useSortable(el, {
|
||
filter: (_evt, target: HTMLElement) => {
|
||
const parent = findParentElement(target);
|
||
const dragable = parent?.classList.contains('dragable');
|
||
return !dragable || !props.dragable;
|
||
},
|
||
onEnd(evt) {
|
||
const { newIndex, oldIndex } = evt;
|
||
// const fromElement = evt.item;
|
||
const { srcElement } = (evt as any).originalEvent;
|
||
|
||
if (!srcElement) {
|
||
resetElState();
|
||
return;
|
||
}
|
||
|
||
const srcParent = findParentElement(srcElement);
|
||
|
||
if (!srcParent) {
|
||
resetElState();
|
||
return;
|
||
}
|
||
|
||
if (!srcParent.classList.contains('dragable')) {
|
||
resetElState();
|
||
|
||
return;
|
||
}
|
||
|
||
if (
|
||
oldIndex !== undefined &&
|
||
newIndex !== undefined &&
|
||
!Number.isNaN(oldIndex) &&
|
||
!Number.isNaN(newIndex) &&
|
||
oldIndex !== newIndex
|
||
) {
|
||
emit('sortTabs', oldIndex, newIndex);
|
||
}
|
||
resetElState();
|
||
},
|
||
onMove(evt) {
|
||
const parent = findParentElement(evt.related);
|
||
return parent?.classList.contains('dragable') && props.dragable;
|
||
},
|
||
onStart: () => {
|
||
el.style.cursor = 'grabbing';
|
||
el.querySelector('.draggable')?.classList.add('dragging');
|
||
// el.classList.add('dragging');
|
||
},
|
||
});
|
||
|
||
sortableInstance.value = await initializeSortable();
|
||
}
|
||
|
||
async function init() {
|
||
const { isMobile } = useIsMobile();
|
||
|
||
// 移动端下tab不需要拖拽
|
||
if (isMobile.value) {
|
||
return;
|
||
}
|
||
await nextTick();
|
||
initTabsSortable();
|
||
}
|
||
|
||
onMounted(init);
|
||
|
||
watch(
|
||
() => props.styleType,
|
||
() => {
|
||
sortableInstance.value?.destroy();
|
||
init();
|
||
},
|
||
);
|
||
|
||
onUnmounted(() => {
|
||
sortableInstance.value?.destroy();
|
||
});
|
||
}
|