import { h, ref, defineComponent, computed, provide, watch, toRef, nextTick, withDirectives, vShow, watchEffect, cloneVNode, TransitionGroup, onMounted } from 'vue'; import { VResizeObserver, VXScroll } from 'vueuc'; import { throttle } from 'lodash-es'; import { useCompitable, onFontsReady, useMergedState } from 'vooks'; import { useConfig, useTheme, useThemeClass } from "../../_mixins/index.mjs"; import { createKey, call, flatten, warnOnce, resolveWrappedSlot } from "../../_utils/index.mjs"; import { tabsLight } from "../styles/index.mjs"; import { tabsInjectionKey } from "./interface.mjs"; import Tab from "./Tab.mjs"; import style from "./styles/index.cssr.mjs"; import { depx, getPadding } from 'seemly'; export const tabsProps = Object.assign(Object.assign({}, useTheme.props), { value: [String, Number], defaultValue: [String, Number], trigger: { type: String, default: 'click' }, type: { type: String, default: 'bar' }, closable: Boolean, justifyContent: String, size: { type: String, default: 'medium' }, placement: { type: String, default: 'top' }, tabStyle: [String, Object], tabClass: String, addTabStyle: [String, Object], addTabClass: String, barWidth: Number, paneClass: String, paneStyle: [String, Object], paneWrapperClass: String, paneWrapperStyle: [String, Object], addable: [Boolean, Object], tabsPadding: { type: Number, default: 0 }, animated: Boolean, onBeforeLeave: Function, onAdd: Function, 'onUpdate:value': [Function, Array], onUpdateValue: [Function, Array], onClose: [Function, Array], // deprecated labelSize: String, activeName: [String, Number], onActiveNameChange: [Function, Array] }); export default defineComponent({ name: 'Tabs', props: tabsProps, setup(props, { slots }) { var _a, _b, _c, _d; if (process.env.NODE_ENV !== 'production') { watchEffect(() => { if (props.labelSize !== undefined) { warnOnce('tabs', '`label-size` is deprecated, please use `size` instead.'); } if (props.activeName !== undefined) { warnOnce('tabs', '`active-name` is deprecated, please use `value` instead.'); } if (props.onActiveNameChange !== undefined) { warnOnce('tabs', '`on-active-name-change` is deprecated, please use `on-update:value` instead.'); } }); } const { mergedClsPrefixRef, inlineThemeDisabled } = useConfig(props); const themeRef = useTheme('Tabs', '-tabs', style, tabsLight, props, mergedClsPrefixRef); const tabsElRef = ref(null); const barElRef = ref(null); const scrollWrapperElRef = ref(null); const addTabInstRef = ref(null); const xScrollInstRef = ref(null); const yScrollElRef = ref(null); const startReachedRef = ref(true); const endReachedRef = ref(true); const compitableSizeRef = useCompitable(props, ['labelSize', 'size']); const compitableValueRef = useCompitable(props, ['activeName', 'value']); const uncontrolledValueRef = ref((_b = (_a = compitableValueRef.value) !== null && _a !== void 0 ? _a : props.defaultValue) !== null && _b !== void 0 ? _b : slots.default ? (_d = (_c = flatten(slots.default())[0]) === null || _c === void 0 ? void 0 : _c.props) === null || _d === void 0 ? void 0 : _d.name : null); const mergedValueRef = useMergedState(compitableValueRef, uncontrolledValueRef); const tabChangeIdRef = { id: 0 }; const tabWrapperStyleRef = computed(() => { if (!props.justifyContent || props.type === 'card') return undefined; return { display: 'flex', justifyContent: props.justifyContent }; }); watch(mergedValueRef, () => { tabChangeIdRef.id = 0; updateCurrentBarStyle(); updateCurrentScrollPosition(true); }); function getCurrentEl() { var _a; const { value } = mergedValueRef; if (value === null) return null; const tabEl = (_a = tabsElRef.value) === null || _a === void 0 ? void 0 : _a.querySelector(`[data-name="${value}"]`); return tabEl; } function updateBarStyle(tabEl) { if (props.type === 'card') return; const { value: barEl } = barElRef; if (!barEl) return; const barIsHide = barEl.style.opacity === '0'; if (tabEl) { const disabledClassName = `${mergedClsPrefixRef.value}-tabs-bar--disabled`; const { barWidth, placement } = props; if (tabEl.dataset.disabled === 'true') { barEl.classList.add(disabledClassName); } else { barEl.classList.remove(disabledClassName); } if (['top', 'bottom'].includes(placement)) { clearBarStyle(['top', 'maxHeight', 'height']); if (typeof barWidth === 'number' && tabEl.offsetWidth >= barWidth) { const offsetDiffLeft = Math.floor((tabEl.offsetWidth - barWidth) / 2) + tabEl.offsetLeft; barEl.style.left = `${offsetDiffLeft}px`; barEl.style.maxWidth = `${barWidth}px`; } else { barEl.style.left = `${tabEl.offsetLeft}px`; barEl.style.maxWidth = `${tabEl.offsetWidth}px`; } barEl.style.width = '8192px'; if (barIsHide) { barEl.style.transition = 'none'; } void barEl.offsetWidth; if (barIsHide) { barEl.style.transition = ''; barEl.style.opacity = '1'; } } else { clearBarStyle(['left', 'maxWidth', 'width']); if (typeof barWidth === 'number' && tabEl.offsetHeight >= barWidth) { const offsetDiffTop = Math.floor((tabEl.offsetHeight - barWidth) / 2) + tabEl.offsetTop; barEl.style.top = `${offsetDiffTop}px`; barEl.style.maxHeight = `${barWidth}px`; } else { barEl.style.top = `${tabEl.offsetTop}px`; barEl.style.maxHeight = `${tabEl.offsetHeight}px`; } barEl.style.height = '8192px'; if (barIsHide) { barEl.style.transition = 'none'; } void barEl.offsetHeight; if (barIsHide) { barEl.style.transition = ''; barEl.style.opacity = '1'; } } } } function hideBarStyle() { if (props.type === 'card') return; const { value: barEl } = barElRef; if (!barEl) return; barEl.style.opacity = '0'; } function clearBarStyle(styleProps) { const { value: barEl } = barElRef; if (!barEl) return; for (const prop of styleProps) { barEl.style[prop] = ''; } } function updateCurrentBarStyle() { if (props.type === 'card') return; const tabEl = getCurrentEl(); if (tabEl) { updateBarStyle(tabEl); } else { hideBarStyle(); } } function updateCurrentScrollPosition(smooth) { var _a; const scrollWrapperEl = (_a = xScrollInstRef.value) === null || _a === void 0 ? void 0 : _a.$el; if (!scrollWrapperEl) return; const tabEl = getCurrentEl(); if (!tabEl) return; const { scrollLeft: scrollWrapperElScrollLeft, offsetWidth: scrollWrapperElOffsetWidth } = scrollWrapperEl; const { offsetLeft: tabElOffsetLeft, offsetWidth: tabElOffsetWidth } = tabEl; if (scrollWrapperElScrollLeft > tabElOffsetLeft) { scrollWrapperEl.scrollTo({ top: 0, left: tabElOffsetLeft, behavior: 'smooth' }); } else if (tabElOffsetLeft + tabElOffsetWidth > scrollWrapperElScrollLeft + scrollWrapperElOffsetWidth) { scrollWrapperEl.scrollTo({ top: 0, left: tabElOffsetLeft + tabElOffsetWidth - scrollWrapperElOffsetWidth, behavior: 'smooth' }); } } const tabsPaneWrapperRef = ref(null); let fromHeight = 0; let hangingTransition = null; function onAnimationBeforeLeave(el) { const tabsPaneWrapperEl = tabsPaneWrapperRef.value; if (tabsPaneWrapperEl) { fromHeight = el.getBoundingClientRect().height; const fromHeightPx = `${fromHeight}px`; const applyFromStyle = () => { tabsPaneWrapperEl.style.height = fromHeightPx; tabsPaneWrapperEl.style.maxHeight = fromHeightPx; }; if (!hangingTransition) { hangingTransition = applyFromStyle; } else { applyFromStyle(); hangingTransition(); hangingTransition = null; } } } function onAnimationEnter(el) { const tabsPaneWrapperEl = tabsPaneWrapperRef.value; if (tabsPaneWrapperEl) { const targetHeight = el.getBoundingClientRect().height; const applyTargetStyle = () => { void document.body.offsetHeight; tabsPaneWrapperEl.style.maxHeight = `${targetHeight}px`; tabsPaneWrapperEl.style.height = `${Math.max(fromHeight, targetHeight)}px`; }; if (!hangingTransition) { hangingTransition = applyTargetStyle; } else { hangingTransition(); hangingTransition = null; applyTargetStyle(); } } } function onAnimationAfterEnter() { const tabsPaneWrapperEl = tabsPaneWrapperRef.value; if (tabsPaneWrapperEl) { tabsPaneWrapperEl.style.maxHeight = ''; tabsPaneWrapperEl.style.height = ''; const { paneWrapperStyle } = props; if (typeof paneWrapperStyle === 'string') { tabsPaneWrapperEl.style.cssText = paneWrapperStyle; } else if (paneWrapperStyle) { const { maxHeight, height } = paneWrapperStyle; if (maxHeight !== undefined) { tabsPaneWrapperEl.style.maxHeight = maxHeight; } if (height !== undefined) { tabsPaneWrapperEl.style.height = height; } } } } const renderNameListRef = { value: [] }; const animationDirectionRef = ref('next'); function activateTab(panelName) { const currentValue = mergedValueRef.value; let dir = 'next'; for (const name of renderNameListRef.value) { if (name === currentValue) { break; } if (name === panelName) { dir = 'prev'; break; } } animationDirectionRef.value = dir; doUpdateValue(panelName); } function doUpdateValue(panelName) { const { onActiveNameChange, onUpdateValue, 'onUpdate:value': _onUpdateValue } = props; if (onActiveNameChange) { call(onActiveNameChange, panelName); } if (onUpdateValue) call(onUpdateValue, panelName); if (_onUpdateValue) call(_onUpdateValue, panelName); uncontrolledValueRef.value = panelName; } function handleClose(panelName) { const { onClose } = props; if (onClose) call(onClose, panelName); } let firstTimeUpdatePosition = true; function updateBarPositionInstantly() { const { value: barEl } = barElRef; if (!barEl) return; if (!firstTimeUpdatePosition) firstTimeUpdatePosition = false; const disableTransitionClassName = 'transition-disabled'; barEl.classList.add(disableTransitionClassName); updateCurrentBarStyle(); // here we don't need to force layout after update bar style // since deriveScrollShadow will force layout barEl.classList.remove(disableTransitionClassName); } const segmentCapsuleElRef = ref(null); function updateSegmentPosition({ disabledTransition }) { const tabsEl = tabsElRef.value; if (!tabsEl) return; disabledTransition && tabsEl.classList.add('transition-disabled'); const activeTabEl = getCurrentEl(); if (activeTabEl && segmentCapsuleElRef.value) { const rect = activeTabEl.getBoundingClientRect(); // move segment capsule to match the position of the active tab segmentCapsuleElRef.value.style.width = `${rect.width}px`; segmentCapsuleElRef.value.style.height = `${rect.height}px`; segmentCapsuleElRef.value.style.transform = `translateX(${rect.left - tabsEl.getBoundingClientRect().left - depx(getComputedStyle(tabsEl).paddingLeft)}px)`; } disabledTransition && tabsEl.classList.remove('transition-disabled'); } watch([mergedValueRef], () => { if (props.type === 'segment') { void nextTick(() => { updateSegmentPosition({ disabledTransition: false }); }); } }); onMounted(() => { if (props.type === 'segment') { updateSegmentPosition({ disabledTransition: true }); } }); let memorizedWidth = 0; function _handleNavResize(entry) { var _a, _b; if (entry.contentRect.width === 0 && entry.contentRect.height === 0) { return; } if (memorizedWidth === entry.contentRect.width) { return; } memorizedWidth = entry.contentRect.width; const { type } = props; if (type === 'line' || type === 'bar') { if (firstTimeUpdatePosition || ((_a = props.justifyContent) === null || _a === void 0 ? void 0 : _a.startsWith('space'))) { updateBarPositionInstantly(); } } if (type !== 'segment') { const { placement } = props; deriveScrollShadow((placement === 'top' || placement === 'bottom' ? (_b = xScrollInstRef.value) === null || _b === void 0 ? void 0 : _b.$el : yScrollElRef.value) || null); } } const handleNavResize = throttle(_handleNavResize, 64); watch([() => props.justifyContent, () => props.size], () => { void nextTick(() => { const { type } = props; if (type === 'line' || type === 'bar') { updateBarPositionInstantly(); } }); }); const addTabFixedRef = ref(false); function _handleTabsResize(entry) { var _a; const { target, contentRect: { width } } = entry; // eslint-disable-next-line @typescript-eslint/no-non-null-assertion const containerWidth = target.parentElement.offsetWidth; if (!addTabFixedRef.value) { if (containerWidth < width) { addTabFixedRef.value = true; } } else { const { value: addTabInst } = addTabInstRef; if (!addTabInst) return; if (containerWidth - width > addTabInst.$el.offsetWidth) { addTabFixedRef.value = false; } } deriveScrollShadow(((_a = xScrollInstRef.value) === null || _a === void 0 ? void 0 : _a.$el) || null); } const handleTabsResize = throttle(_handleTabsResize, 64); function handleAdd() { const { onAdd } = props; if (onAdd) onAdd(); void nextTick(() => { const currentEl = getCurrentEl(); const { value: xScrollInst } = xScrollInstRef; if (!currentEl || !xScrollInst) return; xScrollInst.scrollTo({ left: currentEl.offsetLeft, top: 0, behavior: 'smooth' }); }); } function deriveScrollShadow(el) { if (!el) return; const { placement } = props; if (placement === 'top' || placement === 'bottom') { const { scrollLeft, scrollWidth, offsetWidth } = el; startReachedRef.value = scrollLeft <= 0; endReachedRef.value = scrollLeft + offsetWidth >= scrollWidth; } else { const { scrollTop, scrollHeight, offsetHeight } = el; startReachedRef.value = scrollTop <= 0; endReachedRef.value = scrollTop + offsetHeight >= scrollHeight; } } const handleScroll = throttle(e => { deriveScrollShadow(e.target); }, 64); provide(tabsInjectionKey, { triggerRef: toRef(props, 'trigger'), tabStyleRef: toRef(props, 'tabStyle'), tabClassRef: toRef(props, 'tabClass'), addTabStyleRef: toRef(props, 'addTabStyle'), addTabClassRef: toRef(props, 'addTabClass'), paneClassRef: toRef(props, 'paneClass'), paneStyleRef: toRef(props, 'paneStyle'), mergedClsPrefixRef, typeRef: toRef(props, 'type'), closableRef: toRef(props, 'closable'), valueRef: mergedValueRef, tabChangeIdRef, onBeforeLeaveRef: toRef(props, 'onBeforeLeave'), activateTab, handleClose, handleAdd }); onFontsReady(() => { updateCurrentBarStyle(); updateCurrentScrollPosition(true); }); // avoid useless rerender watchEffect(() => { const { value: el } = scrollWrapperElRef; if (!el) return; const { value: clsPrefix } = mergedClsPrefixRef; const shadowStartClass = `${clsPrefix}-tabs-nav-scroll-wrapper--shadow-start`; const shadowEndClass = `${clsPrefix}-tabs-nav-scroll-wrapper--shadow-end`; if (startReachedRef.value) { el.classList.remove(shadowStartClass); } else { el.classList.add(shadowStartClass); } if (endReachedRef.value) { el.classList.remove(shadowEndClass); } else { el.classList.add(shadowEndClass); } }); const exposedMethods = { syncBarPosition: () => { updateCurrentBarStyle(); } }; const cssVarsRef = computed(() => { const { value: size } = compitableSizeRef; const { type } = props; const typeSuffix = { card: 'Card', bar: 'Bar', line: 'Line', segment: 'Segment' }[type]; const sizeType = `${size}${typeSuffix}`; const { self: { barColor, closeIconColor, closeIconColorHover, closeIconColorPressed, tabColor, tabBorderColor, paneTextColor, tabFontWeight, tabBorderRadius, tabFontWeightActive, colorSegment, fontWeightStrong, tabColorSegment, closeSize, closeIconSize, closeColorHover, closeColorPressed, closeBorderRadius, [createKey('panePadding', size)]: panePadding, [createKey('tabPadding', sizeType)]: tabPadding, [createKey('tabPaddingVertical', sizeType)]: tabPaddingVertical, [createKey('tabGap', sizeType)]: tabGap, [createKey('tabGap', `${sizeType}Vertical`)]: tabGapVertical, [createKey('tabTextColor', type)]: tabTextColor, [createKey('tabTextColorActive', type)]: tabTextColorActive, [createKey('tabTextColorHover', type)]: tabTextColorHover, [createKey('tabTextColorDisabled', type)]: tabTextColorDisabled, [createKey('tabFontSize', size)]: tabFontSize }, common: { cubicBezierEaseInOut } } = themeRef.value; return { '--n-bezier': cubicBezierEaseInOut, '--n-color-segment': colorSegment, '--n-bar-color': barColor, '--n-tab-font-size': tabFontSize, '--n-tab-text-color': tabTextColor, '--n-tab-text-color-active': tabTextColorActive, '--n-tab-text-color-disabled': tabTextColorDisabled, '--n-tab-text-color-hover': tabTextColorHover, '--n-pane-text-color': paneTextColor, '--n-tab-border-color': tabBorderColor, '--n-tab-border-radius': tabBorderRadius, '--n-close-size': closeSize, '--n-close-icon-size': closeIconSize, '--n-close-color-hover': closeColorHover, '--n-close-color-pressed': closeColorPressed, '--n-close-border-radius': closeBorderRadius, '--n-close-icon-color': closeIconColor, '--n-close-icon-color-hover': closeIconColorHover, '--n-close-icon-color-pressed': closeIconColorPressed, '--n-tab-color': tabColor, '--n-tab-font-weight': tabFontWeight, '--n-tab-font-weight-active': tabFontWeightActive, '--n-tab-padding': tabPadding, '--n-tab-padding-vertical': tabPaddingVertical, '--n-tab-gap': tabGap, '--n-tab-gap-vertical': tabGapVertical, '--n-pane-padding-left': getPadding(panePadding, 'left'), '--n-pane-padding-right': getPadding(panePadding, 'right'), '--n-pane-padding-top': getPadding(panePadding, 'top'), '--n-pane-padding-bottom': getPadding(panePadding, 'bottom'), '--n-font-weight-strong': fontWeightStrong, '--n-tab-color-segment': tabColorSegment }; }); const themeClassHandle = inlineThemeDisabled ? useThemeClass('tabs', computed(() => { return `${compitableSizeRef.value[0]}${props.type[0]}`; }), cssVarsRef, props) : undefined; return Object.assign({ mergedClsPrefix: mergedClsPrefixRef, mergedValue: mergedValueRef, renderedNames: new Set(), segmentCapsuleElRef, tabsPaneWrapperRef, tabsElRef, barElRef, addTabInstRef, xScrollInstRef, scrollWrapperElRef, addTabFixed: addTabFixedRef, tabWrapperStyle: tabWrapperStyleRef, handleNavResize, mergedSize: compitableSizeRef, handleScroll, handleTabsResize, cssVars: inlineThemeDisabled ? undefined : cssVarsRef, themeClass: themeClassHandle === null || themeClassHandle === void 0 ? void 0 : themeClassHandle.themeClass, animationDirection: animationDirectionRef, renderNameListRef, yScrollElRef, onAnimationBeforeLeave, onAnimationEnter, onAnimationAfterEnter, onRender: themeClassHandle === null || themeClassHandle === void 0 ? void 0 : themeClassHandle.onRender }, exposedMethods); }, render() { const { mergedClsPrefix, type, placement, addTabFixed, addable, mergedSize, renderNameListRef, onRender, paneWrapperClass, paneWrapperStyle, $slots: { default: defaultSlot, prefix: prefixSlot, suffix: suffixSlot } } = this; onRender === null || onRender === void 0 ? void 0 : onRender(); const tabPaneChildren = defaultSlot ? flatten(defaultSlot()).filter(v => { return v.type.__TAB_PANE__ === true; }) : []; const tabChildren = defaultSlot ? flatten(defaultSlot()).filter(v => { return v.type.__TAB__ === true; }) : []; const showPane = !tabChildren.length; const isCard = type === 'card'; const isSegment = type === 'segment'; const mergedJustifyContent = !isCard && !isSegment && this.justifyContent; renderNameListRef.value = []; const scrollContent = () => { const tabs = h("div", { style: this.tabWrapperStyle, class: [`${mergedClsPrefix}-tabs-wrapper`] }, mergedJustifyContent ? null : h("div", { class: `${mergedClsPrefix}-tabs-scroll-padding`, style: { width: `${this.tabsPadding}px` } }), showPane ? tabPaneChildren.map((tabPaneVNode, index) => { renderNameListRef.value.push(tabPaneVNode.props.name); return justifyTabDynamicProps(h(Tab, Object.assign({}, tabPaneVNode.props, { internalCreatedByPane: true, internalLeftPadded: index !== 0 && (!mergedJustifyContent || mergedJustifyContent === 'center' || mergedJustifyContent === 'start' || mergedJustifyContent === 'end') }), tabPaneVNode.children ? { default: tabPaneVNode.children.tab } : undefined)); }) : tabChildren.map((tabVNode, index) => { renderNameListRef.value.push(tabVNode.props.name); if (index !== 0 && !mergedJustifyContent) { return justifyTabDynamicProps(createLeftPaddedTabVNode(tabVNode)); } else { return justifyTabDynamicProps(tabVNode); } }), !addTabFixed && addable && isCard ? createAddTag(addable, (showPane ? tabPaneChildren.length : tabChildren.length) !== 0) : null, mergedJustifyContent ? null : h("div", { class: `${mergedClsPrefix}-tabs-scroll-padding`, style: { width: `${this.tabsPadding}px` } })); return h("div", { ref: "tabsElRef", class: `${mergedClsPrefix}-tabs-nav-scroll-content` }, isCard && addable ? h(VResizeObserver, { onResize: this.handleTabsResize }, { default: () => tabs }) : tabs, isCard ? h("div", { class: `${mergedClsPrefix}-tabs-pad` }) : null, isCard ? null : h("div", { ref: "barElRef", class: `${mergedClsPrefix}-tabs-bar` })); }; const resolvedPlacement = isSegment ? 'top' : placement; return h("div", { class: [`${mergedClsPrefix}-tabs`, this.themeClass, `${mergedClsPrefix}-tabs--${type}-type`, `${mergedClsPrefix}-tabs--${mergedSize}-size`, mergedJustifyContent && `${mergedClsPrefix}-tabs--flex`, `${mergedClsPrefix}-tabs--${resolvedPlacement}`], style: this.cssVars }, h("div", { class: [ // the class should be applied here since it's possible // to make tabs nested in tabs, style may influence each // other. adding a class will make it easy to write the // style. `${mergedClsPrefix}-tabs-nav--${type}-type`, `${mergedClsPrefix}-tabs-nav--${resolvedPlacement}`, `${mergedClsPrefix}-tabs-nav`] }, resolveWrappedSlot(prefixSlot, children => children && h("div", { class: `${mergedClsPrefix}-tabs-nav__prefix` }, children)), isSegment ? h("div", { class: `${mergedClsPrefix}-tabs-rail`, ref: "tabsElRef" }, h("div", { class: `${mergedClsPrefix}-tabs-capsule`, ref: "segmentCapsuleElRef" }, h("div", { class: `${mergedClsPrefix}-tabs-wrapper` }, h("div", { class: `${mergedClsPrefix}-tabs-tab` }))), showPane ? tabPaneChildren.map((tabPaneVNode, index) => { renderNameListRef.value.push(tabPaneVNode.props.name); return h(Tab, Object.assign({}, tabPaneVNode.props, { internalCreatedByPane: true, internalLeftPadded: index !== 0 }), tabPaneVNode.children ? { default: tabPaneVNode.children.tab } : undefined); }) : tabChildren.map((tabVNode, index) => { renderNameListRef.value.push(tabVNode.props.name); if (index === 0) { return tabVNode; } else { return createLeftPaddedTabVNode(tabVNode); } })) : h(VResizeObserver, { onResize: this.handleNavResize }, { default: () => h("div", { class: `${mergedClsPrefix}-tabs-nav-scroll-wrapper`, ref: "scrollWrapperElRef" }, ['top', 'bottom'].includes(resolvedPlacement) ? h(VXScroll, { ref: "xScrollInstRef", onScroll: this.handleScroll }, { default: scrollContent }) : h("div", { class: `${mergedClsPrefix}-tabs-nav-y-scroll`, onScroll: this.handleScroll, ref: "yScrollElRef" }, scrollContent())) }), addTabFixed && addable && isCard ? createAddTag(addable, true) : null, resolveWrappedSlot(suffixSlot, children => children && h("div", { class: `${mergedClsPrefix}-tabs-nav__suffix` }, children))), showPane && (this.animated && (resolvedPlacement === 'top' || resolvedPlacement === 'bottom') ? h("div", { ref: "tabsPaneWrapperRef", style: paneWrapperStyle, class: [`${mergedClsPrefix}-tabs-pane-wrapper`, paneWrapperClass] }, filterMapTabPanes(tabPaneChildren, this.mergedValue, this.renderedNames, this.onAnimationBeforeLeave, this.onAnimationEnter, this.onAnimationAfterEnter, this.animationDirection)) : filterMapTabPanes(tabPaneChildren, this.mergedValue, this.renderedNames))); } }); function filterMapTabPanes(tabPaneVNodes, value, renderedNames, onBeforeLeave, onEnter, onAfterEnter, animationDirection) { const children = []; tabPaneVNodes.forEach(vNode => { const { name, displayDirective, 'display-directive': _displayDirective } = vNode.props; const matchDisplayDirective = directive => displayDirective === directive || _displayDirective === directive; const show = value === name; if (vNode.key !== undefined) { vNode.key = name; } if (show || matchDisplayDirective('show') || matchDisplayDirective('show:lazy') && renderedNames.has(name)) { if (!renderedNames.has(name)) { renderedNames.add(name); } const useVShow = !matchDisplayDirective('if'); children.push(useVShow ? withDirectives(vNode, [[vShow, show]]) : vNode); } }); if (!animationDirection) { return children; } return h(TransitionGroup, { name: `${animationDirection}-transition`, onBeforeLeave: onBeforeLeave, onEnter: onEnter, onAfterEnter: onAfterEnter }, { default: () => children }); } function createAddTag(addable, internalLeftPadded) { return h(Tab, { ref: "addTabInstRef", key: "__addable", name: "__addable", internalCreatedByPane: true, internalAddable: true, internalLeftPadded: internalLeftPadded, disabled: typeof addable === 'object' && addable.disabled }); } function createLeftPaddedTabVNode(tabVNode) { const modifiedVNode = cloneVNode(tabVNode); if (modifiedVNode.props) { modifiedVNode.props.internalLeftPadded = true; } else { modifiedVNode.props = { internalLeftPadded: true }; } return modifiedVNode; } function justifyTabDynamicProps(tabVNode) { if (Array.isArray(tabVNode.dynamicProps)) { if (!tabVNode.dynamicProps.includes('internalLeftPadded')) { tabVNode.dynamicProps.push('internalLeftPadded'); } } else { tabVNode.dynamicProps = ['internalLeftPadded']; } return tabVNode; }