UNPKG

element-plus

Version:

A Component Library for Vue 3

268 lines (265 loc) 10 kB
import { defineComponent, inject, ref, computed, watch, onMounted, onUpdated, createVNode } from 'vue'; import { NOOP, capitalize } from '@vue/shared'; import { useDocumentVisibility, useWindowFocus, useResizeObserver } from '@vueuse/core'; import '../../../utils/index.mjs'; import '../../../constants/index.mjs'; import { ElIcon } from '../../icon/index.mjs'; import { ArrowLeft, ArrowRight, Close } from '@element-plus/icons-vue'; import '../../../tokens/index.mjs'; import '../../../hooks/index.mjs'; import TabBar from './tab-bar2.mjs'; import { buildProps, definePropType } from '../../../utils/vue/props.mjs'; import { mutable } from '../../../utils/typescript.mjs'; import { tabsRootContextKey } from '../../../tokens/tabs.mjs'; import { throwError } from '../../../utils/error.mjs'; import { useNamespace } from '../../../hooks/use-namespace/index.mjs'; import { EVENT_CODE } from '../../../constants/aria.mjs'; const tabNavProps = buildProps({ panes: { type: definePropType(Array), default: () => mutable([]) }, currentName: { type: [String, Number], default: "" }, editable: Boolean, onTabClick: { type: definePropType(Function), default: NOOP }, onTabRemove: { type: definePropType(Function), default: NOOP }, type: { type: String, values: ["card", "border-card", ""], default: "" }, stretch: Boolean }); const COMPONENT_NAME = "ElTabNav"; const TabNav = defineComponent({ name: COMPONENT_NAME, props: tabNavProps, setup(props, { expose }) { const rootTabs = inject(tabsRootContextKey); if (!rootTabs) throwError(COMPONENT_NAME, `<el-tabs><tab-nav /></el-tabs>`); const ns = useNamespace("tabs"); const visibility = useDocumentVisibility(); const focused = useWindowFocus(); const navScroll$ = ref(); const nav$ = ref(); const el$ = ref(); const scrollable = ref(false); const navOffset = ref(0); const isFocus = ref(false); const focusable = ref(true); const sizeName = computed(() => ["top", "bottom"].includes(rootTabs.props.tabPosition) ? "width" : "height"); const navStyle = computed(() => { const dir = sizeName.value === "width" ? "X" : "Y"; return { transform: `translate${dir}(-${navOffset.value}px)` }; }); const scrollPrev = () => { if (!navScroll$.value) return; const containerSize = navScroll$.value[`offset${capitalize(sizeName.value)}`]; const currentOffset = navOffset.value; if (!currentOffset) return; const newOffset = currentOffset > containerSize ? currentOffset - containerSize : 0; navOffset.value = newOffset; }; const scrollNext = () => { if (!navScroll$.value || !nav$.value) return; const navSize = nav$.value[`offset${capitalize(sizeName.value)}`]; const containerSize = navScroll$.value[`offset${capitalize(sizeName.value)}`]; const currentOffset = navOffset.value; if (navSize - currentOffset <= containerSize) return; const newOffset = navSize - currentOffset > containerSize * 2 ? currentOffset + containerSize : navSize - containerSize; navOffset.value = newOffset; }; const scrollToActiveTab = () => { const nav = nav$.value; if (!scrollable.value || !el$.value || !navScroll$.value || !nav) return; const activeTab = el$.value.querySelector(".is-active"); if (!activeTab) return; const navScroll = navScroll$.value; const isHorizontal = ["top", "bottom"].includes(rootTabs.props.tabPosition); const activeTabBounding = activeTab.getBoundingClientRect(); const navScrollBounding = navScroll.getBoundingClientRect(); const maxOffset = isHorizontal ? nav.offsetWidth - navScrollBounding.width : nav.offsetHeight - navScrollBounding.height; const currentOffset = navOffset.value; let newOffset = currentOffset; if (isHorizontal) { if (activeTabBounding.left < navScrollBounding.left) { newOffset = currentOffset - (navScrollBounding.left - activeTabBounding.left); } if (activeTabBounding.right > navScrollBounding.right) { newOffset = currentOffset + activeTabBounding.right - navScrollBounding.right; } } else { if (activeTabBounding.top < navScrollBounding.top) { newOffset = currentOffset - (navScrollBounding.top - activeTabBounding.top); } if (activeTabBounding.bottom > navScrollBounding.bottom) { newOffset = currentOffset + (activeTabBounding.bottom - navScrollBounding.bottom); } } newOffset = Math.max(newOffset, 0); navOffset.value = Math.min(newOffset, maxOffset); }; const update = () => { if (!nav$.value || !navScroll$.value) return; const navSize = nav$.value[`offset${capitalize(sizeName.value)}`]; const containerSize = navScroll$.value[`offset${capitalize(sizeName.value)}`]; const currentOffset = navOffset.value; if (containerSize < navSize) { const currentOffset2 = navOffset.value; scrollable.value = scrollable.value || {}; scrollable.value.prev = currentOffset2; scrollable.value.next = currentOffset2 + containerSize < navSize; if (navSize - currentOffset2 < containerSize) { navOffset.value = navSize - containerSize; } } else { scrollable.value = false; if (currentOffset > 0) { navOffset.value = 0; } } }; const changeTab = (e) => { const code = e.code; const { up, down, left, right } = EVENT_CODE; if (![up, down, left, right].includes(code)) return; const tabList = Array.from(e.currentTarget.querySelectorAll("[role=tab]")); const currentIndex = tabList.indexOf(e.target); let nextIndex; if (code === left || code === up) { if (currentIndex === 0) { nextIndex = tabList.length - 1; } else { nextIndex = currentIndex - 1; } } else { if (currentIndex < tabList.length - 1) { nextIndex = currentIndex + 1; } else { nextIndex = 0; } } tabList[nextIndex].focus(); tabList[nextIndex].click(); setFocus(); }; const setFocus = () => { if (focusable.value) isFocus.value = true; }; const removeFocus = () => isFocus.value = false; watch(visibility, (visibility2) => { if (visibility2 === "hidden") { focusable.value = false; } else if (visibility2 === "visible") { setTimeout(() => focusable.value = true, 50); } }); watch(focused, (focused2) => { if (focused2) { setTimeout(() => focusable.value = true, 50); } else { focusable.value = false; } }); useResizeObserver(el$, update); onMounted(() => setTimeout(() => scrollToActiveTab(), 0)); onUpdated(() => update()); expose({ scrollToActiveTab, removeFocus }); return () => { const scrollBtn = scrollable.value ? [createVNode("span", { "class": [ns.e("nav-prev"), ns.is("disabled", !scrollable.value.prev)], "onClick": scrollPrev }, [createVNode(ElIcon, null, { default: () => [createVNode(ArrowLeft, null, null)] })]), createVNode("span", { "class": [ns.e("nav-next"), ns.is("disabled", !scrollable.value.next)], "onClick": scrollNext }, [createVNode(ElIcon, null, { default: () => [createVNode(ArrowRight, null, null)] })])] : null; const tabs = props.panes.map((pane, index) => { var _a, _b; const tabName = pane.props.name || pane.index || `${index}`; const closable = pane.isClosable || props.editable; pane.index = `${index}`; const btnClose = closable ? createVNode(ElIcon, { "class": "is-icon-close", "onClick": (ev) => props.onTabRemove(pane, ev) }, { default: () => [createVNode(Close, null, null)] }) : null; const tabLabelContent = ((_b = (_a = pane.instance.slots).label) == null ? void 0 : _b.call(_a)) || pane.props.label; const tabindex = pane.active ? 0 : -1; return createVNode("div", { "ref": `tab-${tabName}`, "class": [ns.e("item"), ns.is(rootTabs.props.tabPosition), ns.is("active", pane.active), ns.is("disabled", pane.props.disabled), ns.is("closable", closable), ns.is("focus", isFocus.value)], "id": `tab-${tabName}`, "key": `tab-${tabName}`, "aria-controls": `pane-${tabName}`, "role": "tab", "aria-selected": pane.active, "tabindex": tabindex, "onFocus": () => setFocus(), "onBlur": () => removeFocus(), "onClick": (ev) => { removeFocus(); props.onTabClick(pane, tabName, ev); }, "onKeydown": (ev) => { if (closable && (ev.code === EVENT_CODE.delete || ev.code === EVENT_CODE.backspace)) { props.onTabRemove(pane, ev); } } }, [...[tabLabelContent, btnClose]]); }); return createVNode("div", { "ref": el$, "class": [ns.e("nav-wrap"), ns.is("scrollable", !!scrollable.value), ns.is(rootTabs.props.tabPosition)] }, [scrollBtn, createVNode("div", { "class": ns.e("nav-scroll"), "ref": navScroll$ }, [createVNode("div", { "class": [ns.e("nav"), ns.is(rootTabs.props.tabPosition), ns.is("stretch", props.stretch && ["top", "bottom"].includes(rootTabs.props.tabPosition))], "ref": nav$, "style": navStyle.value, "role": "tablist", "onKeydown": changeTab }, [...[!props.type ? createVNode(TabBar, { "tabs": [...props.panes] }, null) : null, tabs]])])]); }; } }); export { TabNav as default, tabNavProps }; //# sourceMappingURL=tab-nav.mjs.map