admin-vben/packages/@core/ui-kit/tabs-ui/src/tabs-view.vue

101 lines
2.4 KiB
Vue
Raw Normal View History

2024-05-25 14:10:21 +00:00
<script setup lang="ts">
import type { TabItem } from '@vben-core/typings';
import { nextTick, onMounted } from 'vue';
import { useSortable } from '@vben-core/hooks';
2024-05-25 14:10:21 +00:00
import { useForwardPropsEmits } from '@vben-core/shadcn-ui';
import { TabsChrome } from './components';
import { TabsProps } from './types';
2024-05-25 14:10:21 +00:00
interface Props extends TabsProps {}
defineOptions({
name: 'TabsView',
});
const props = withDefaults(defineProps<Props>(), {
contentClass: 'vben-tabs-content',
dragable: true,
});
2024-05-25 14:10:21 +00:00
const emit = defineEmits<{
close: [string];
sortTabs: [number, number];
unpin: [TabItem];
}>();
2024-05-25 14:10:21 +00:00
const forward = useForwardPropsEmits(props, emit);
// 可能会找到拖拽的子元素这里需要确保拖拽的dom时tab元素
const findParentElement = (element: HTMLElement) => {
const parentCls = 'group';
return element.classList.contains(parentCls)
? element
: element.closest(`.${parentCls}`);
};
async function initTabsSortable() {
await nextTick();
const { contentClass } = props;
const el = document.querySelectorAll(`.${contentClass}`)?.[0] as HTMLElement;
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) {
return;
}
const srcParent = findParentElement(srcElement);
if (!srcParent) {
return;
}
if (!srcParent.classList.contains('dragable')) {
return;
}
if (
oldIndex !== undefined &&
newIndex !== undefined &&
!Number.isNaN(oldIndex) &&
!Number.isNaN(newIndex) &&
oldIndex !== newIndex
) {
emit('sortTabs', oldIndex, newIndex);
}
el.classList.remove('dragging');
el.style.cursor = 'default';
},
onMove(evt) {
const parent = findParentElement(evt.related);
return parent?.classList.contains('dragable');
},
onStart: () => {
el.style.cursor = 'grabbing';
el.classList.add('dragging');
},
});
await initializeSortable();
}
onMounted(initTabsSortable);
2024-05-25 14:10:21 +00:00
</script>
<template>
<TabsChrome v-bind="forward" />
2024-05-25 14:10:21 +00:00
</template>