iview/src/components/tabs/tabs.vue

529 lines
21 KiB
Vue
Raw Normal View History

<template>
2016-12-06 22:51:06 +08:00
<div :class="classes">
<div :class="[prefixCls + '-bar']">
2017-10-20 16:30:44 +08:00
<div :class="[prefixCls + '-nav-right']" v-if="showSlot"><slot name="extra"></slot></div>
2018-05-04 15:32:31 +02:00
<div
:class="[prefixCls + '-nav-container']"
tabindex="0"
ref="navContainer"
@keydown="handleTabKeyNavigation"
2018-05-18 11:01:30 +08:00
@keydown.space.prevent="handleTabKeyboardSelect(false)"
2018-05-04 15:32:31 +02:00
>
2017-11-09 09:48:56 +08:00
<div ref="navWrap" :class="[prefixCls + '-nav-wrap', scrollable ? prefixCls + '-nav-scrollable' : '']">
2018-07-02 16:22:45 +08:00
<span :class="[prefixCls + '-nav-prev', scrollable ? '' : prefixCls + '-nav-scroll-disabled']" @click="scrollPrev"><Icon type="ios-arrow-back"></Icon></span>
<span :class="[prefixCls + '-nav-next', scrollable ? '' : prefixCls + '-nav-scroll-disabled']" @click="scrollNext"><Icon type="ios-arrow-forward"></Icon></span>
2017-10-20 16:30:44 +08:00
<div ref="navScroll" :class="[prefixCls + '-nav-scroll']">
2019-04-10 16:48:04 +08:00
<div ref="nav" :class="[prefixCls + '-nav']" :style="navStyle">
2016-12-06 22:51:06 +08:00
<div :class="barClasses" :style="barStyle"></div>
2017-03-03 13:38:46 +08:00
<div :class="tabCls(item)" v-for="(item, index) in navList" @click="handleChange(index)">
2016-12-06 22:51:06 +08:00
<Icon v-if="item.icon !== ''" :type="item.icon"></Icon>
2017-06-02 15:05:01 +08:00
<Render v-if="item.labelType === 'function'" :render="item.label"></Render>
<template v-else>{{ item.label }}</template>
2019-04-10 14:07:07 +08:00
<Icon :class="[prefixCls + '-close']" v-if="showClose(item)" :type="arrowType" :custom="customArrowType" :size="arrowSize" @click.native.stop="handleRemove(index)"></Icon>
2016-12-06 22:51:06 +08:00
</div>
</div>
</div>
</div>
</div>
</div>
2018-05-04 15:32:31 +02:00
<div :class="contentClasses" :style="contentStyle" ref="panes"><slot></slot></div>
2016-12-06 22:51:06 +08:00
</div>
</template>
<script>
2016-12-06 22:51:06 +08:00
import Icon from '../icon/icon.vue';
2017-06-09 16:54:20 +08:00
import Render from '../base/render';
import { oneOf, MutationObserver, findComponentsDownward } from '../../utils/assist';
2017-04-10 09:58:51 +08:00
import Emitter from '../../mixins/emitter';
2017-10-20 16:30:44 +08:00
import elementResizeDetectorMaker from 'element-resize-detector';
2016-12-06 22:51:06 +08:00
const prefixCls = 'ivu-tabs';
2018-05-04 15:32:31 +02:00
const transitionTime = 300; // from CSS
const getNextTab = (list, activeKey, direction, countDisabledAlso) => {
const currentIndex = list.findIndex(tab => tab.name === activeKey);
const nextIndex = (currentIndex + direction + list.length) % list.length;
const nextTab = list[nextIndex];
2018-05-08 12:35:10 +02:00
if (nextTab.disabled) return getNextTab(list, nextTab.name, direction, countDisabledAlso);
2018-05-04 15:32:31 +02:00
else return nextTab;
};
const focusFirst = (element, root) => {
try {element.focus();}
2018-05-04 15:41:32 +02:00
catch(err) {} // eslint-disable-line no-empty
2018-05-04 15:32:31 +02:00
if (document.activeElement == element && element !== root) return true;
const candidates = element.children;
for (let candidate of candidates) {
if (focusFirst(candidate, root)) return true;
}
return false;
};
2016-12-06 22:51:06 +08:00
export default {
2017-03-03 13:38:46 +08:00
name: 'Tabs',
2017-04-10 09:58:51 +08:00
mixins: [ Emitter ],
2017-06-02 15:05:01 +08:00
components: { Icon, Render },
provide () {
return { TabsInstance: this };
},
2016-12-06 22:51:06 +08:00
props: {
2017-03-03 13:38:46 +08:00
value: {
2016-12-06 22:51:06 +08:00
type: [String, Number]
},
type: {
validator (value) {
return oneOf(value, ['line', 'card']);
},
default: 'line'
},
size: {
validator (value) {
return oneOf(value, ['small', 'default']);
},
default: 'default'
},
animated: {
type: Boolean,
default: true
},
captureFocus: {
type: Boolean,
default: false
},
2016-12-06 22:51:06 +08:00
closable: {
type: Boolean,
default: false
},
beforeRemove: Function,
2019-03-08 14:56:06 +08:00
// Tabs 嵌套时,用 name 区分层级
name: {
type: String
},
custContentClass: {
type: String,
default: ''
},
custContentStyle: {
type: Object,
}
2016-12-06 22:51:06 +08:00
},
data () {
2016-12-06 22:51:06 +08:00
return {
prefixCls: prefixCls,
navList: [],
barWidth: 0,
2017-03-03 13:38:46 +08:00
barOffset: 0,
activeKey: this.value,
2018-05-04 15:32:31 +02:00
focusedKey: this.value,
2017-10-20 16:30:44 +08:00
showSlot: false,
2017-10-20 17:44:58 +08:00
navStyle: {
2017-10-20 16:30:44 +08:00
transform: ''
},
2018-05-04 15:32:31 +02:00
scrollable: false,
transitioning: false,
2016-12-25 22:49:42 +08:00
};
2016-12-06 22:51:06 +08:00
},
computed: {
classes () {
return [
`${prefixCls}`,
{
[`${prefixCls}-card`]: this.type === 'card',
[`${prefixCls}-mini`]: this.size === 'small' && this.type === 'line',
[`${prefixCls}-no-animation`]: !this.animated
}
2016-12-25 22:49:42 +08:00
];
2016-12-06 22:51:06 +08:00
},
contentClasses () {
return [
`${prefixCls}-content`,
2016-12-06 23:11:44 +08:00
{
[`${prefixCls}-content-animated`]: this.animated
},
this.custContentClass
2016-12-25 22:49:42 +08:00
];
2016-12-06 22:51:06 +08:00
},
barClasses () {
return [
`${prefixCls}-ink-bar`,
2016-12-06 23:11:44 +08:00
{
[`${prefixCls}-ink-bar-animated`]: this.animated
}
2016-12-25 22:49:42 +08:00
];
2016-12-06 22:51:06 +08:00
},
contentStyle () {
const x = this.getTabIndex(this.activeKey);
2016-12-06 22:51:06 +08:00
const p = x === 0 ? '0%' : `-${x}00%`;
let style = {};
if (x > -1) {
style = {
transform: `translateX(${p}) translateZ(0px)`
2016-12-25 22:49:42 +08:00
};
2016-12-06 22:51:06 +08:00
}
const { custContentStyle } = this;
2019-09-30 23:04:53 +08:00
if (custContentStyle) {
for (const key in custContentStyle){
style[key] = custContentStyle[key];
}
}
2016-12-06 22:51:06 +08:00
return style;
},
barStyle () {
let style = {
2018-05-18 15:24:49 +02:00
visibility: 'hidden',
2016-12-06 23:11:44 +08:00
width: `${this.barWidth}px`
2016-12-06 22:51:06 +08:00
};
2018-05-21 10:51:23 +08:00
if (this.type === 'line') style.visibility = 'visible';
2016-12-06 23:11:44 +08:00
if (this.animated) {
style.transform = `translate3d(${this.barOffset}px, 0px, 0px)`;
} else {
style.left = `${this.barOffset}px`;
}
2016-12-06 22:51:06 +08:00
return style;
2019-04-10 14:07:07 +08:00
},
// 3.4.0, global setting customArrow 有值时arrow 赋值空
arrowType () {
let type = 'ios-close';
if (this.$IVIEW) {
if (this.$IVIEW.tabs.customCloseIcon) {
type = '';
} else if (this.$IVIEW.tabs.closeIcon) {
type = this.$IVIEW.tabs.closeIcon;
}
}
return type;
},
// 3.4.0, global setting
customArrowType () {
let type = '';
if (this.$IVIEW) {
if (this.$IVIEW.tabs.customCloseIcon) {
type = this.$IVIEW.tabs.customCloseIcon;
}
}
return type;
},
// 3.4.0, global setting
arrowSize () {
let size = '';
if (this.$IVIEW) {
if (this.$IVIEW.tabs.closeIconSize) {
size = this.$IVIEW.tabs.closeIconSize;
}
}
return size;
2016-12-06 22:51:06 +08:00
}
},
methods: {
getTabs () {
// return this.$children.filter(item => item.$options.name === 'TabPane');
2019-03-08 14:56:06 +08:00
const AllTabPanes = findComponentsDownward(this, 'TabPane');
const TabPanes = [];
AllTabPanes.forEach(item => {
if (item.tab && this.name) {
if (item.tab === this.name) {
TabPanes.push(item);
}
} else {
TabPanes.push(item);
}
});
2019-03-15 15:44:27 +08:00
// 在 TabPane 使用 v-if 时,并不会按照预先的顺序渲染,这时可设置 index并从小到大排序
TabPanes.sort((a, b) => {
if (a.index && b.index) {
return a.index > b.index ? 1 : -1;
}
});
2019-03-08 14:56:06 +08:00
return TabPanes;
2016-12-06 22:51:06 +08:00
},
updateNav () {
this.navList = [];
this.getTabs().forEach((pane, index) => {
this.navList.push({
2017-06-02 15:05:01 +08:00
labelType: typeof pane.label,
2016-12-06 22:51:06 +08:00
label: pane.label,
icon: pane.icon || '',
2017-03-03 13:38:46 +08:00
name: pane.currentName || index,
2017-01-16 14:36:53 +08:00
disabled: pane.disabled,
closable: pane.closable
2016-12-06 22:51:06 +08:00
});
2017-03-03 13:38:46 +08:00
if (!pane.currentName) pane.currentName = index;
2016-12-06 22:51:06 +08:00
if (index === 0) {
2017-03-03 13:38:46 +08:00
if (!this.activeKey) this.activeKey = pane.currentName || index;
2016-12-06 22:51:06 +08:00
}
});
2016-12-06 23:11:44 +08:00
this.updateStatus();
2016-12-06 22:51:06 +08:00
this.updateBar();
},
updateBar () {
this.$nextTick(() => {
const index = this.getTabIndex(this.activeKey);
2017-10-26 11:26:05 +08:00
if (!this.$refs.nav) return; // 页面销毁时,这里会报错,为了解决 #2100
2017-03-03 13:38:46 +08:00
const prevTabs = this.$refs.nav.querySelectorAll(`.${prefixCls}-tab`);
2016-12-06 22:51:06 +08:00
const tab = prevTabs[index];
2017-09-13 15:06:02 +08:00
this.barWidth = tab ? parseFloat(tab.offsetWidth) : 0;
2016-12-06 22:51:06 +08:00
if (index > 0) {
let offset = 0;
const gutter = this.size === 'small' ? 0 : 16;
for (let i = 0; i < index; i++) {
offset += parseFloat(prevTabs[i].offsetWidth) + gutter;
2016-12-06 22:51:06 +08:00
}
this.barOffset = offset;
} else {
this.barOffset = 0;
}
2017-10-20 16:30:44 +08:00
this.updateNavScroll();
2016-12-06 22:51:06 +08:00
});
},
2016-12-06 23:11:44 +08:00
updateStatus () {
const tabs = this.getTabs();
2017-03-03 13:38:46 +08:00
tabs.forEach(tab => tab.show = (tab.currentName === this.activeKey) || this.animated);
2016-12-06 23:11:44 +08:00
},
2016-12-06 22:51:06 +08:00
tabCls (item) {
return [
`${prefixCls}-tab`,
{
[`${prefixCls}-tab-disabled`]: item.disabled,
2018-05-04 15:32:31 +02:00
[`${prefixCls}-tab-active`]: item.name === this.activeKey,
[`${prefixCls}-tab-focused`]: item.name === this.focusedKey,
2016-12-06 22:51:06 +08:00
}
2016-12-25 22:49:42 +08:00
];
2016-12-06 22:51:06 +08:00
},
handleChange (index) {
2018-05-04 15:32:31 +02:00
if (this.transitioning) return;
this.transitioning = true;
setTimeout(() => this.transitioning = false, transitionTime);
2016-12-06 22:51:06 +08:00
const nav = this.navList[index];
if (nav.disabled) return;
2017-03-03 13:38:46 +08:00
this.activeKey = nav.name;
this.$emit('input', nav.name);
this.$emit('on-click', nav.name);
2016-12-06 22:51:06 +08:00
},
2018-05-04 15:32:31 +02:00
handleTabKeyNavigation(e){
if (e.keyCode !== 37 && e.keyCode !== 39) return;
const direction = e.keyCode === 39 ? 1 : -1;
const nextTab = getNextTab(this.navList, this.focusedKey, direction);
this.focusedKey = nextTab.name;
},
handleTabKeyboardSelect(init = false){
if (init) return;
const focused = this.focusedKey || 0;
const index = this.getTabIndex(focused);
this.handleChange(index);
2018-05-04 15:32:31 +02:00
},
2016-12-06 22:51:06 +08:00
handleRemove (index) {
if (!this.beforeRemove) {
return this.handleRemoveTab(index);
}
const before = this.beforeRemove(index);
if (before && before.then) {
before.then(() => {
this.handleRemoveTab(index);
});
} else {
this.handleRemoveTab(index);
}
},
handleRemoveTab (index) {
2016-12-06 22:51:06 +08:00
const tabs = this.getTabs();
const tab = tabs[index];
2017-03-23 11:03:43 +08:00
tab.$destroy();
2016-12-06 22:51:06 +08:00
2017-03-03 13:38:46 +08:00
if (tab.currentName === this.activeKey) {
2016-12-06 22:51:06 +08:00
const newTabs = this.getTabs();
let activeKey = -1;
if (newTabs.length) {
const leftNoDisabledTabs = tabs.filter((item, itemIndex) => !item.disabled && itemIndex < index);
const rightNoDisabledTabs = tabs.filter((item, itemIndex) => !item.disabled && itemIndex > index);
if (rightNoDisabledTabs.length) {
2017-03-03 13:38:46 +08:00
activeKey = rightNoDisabledTabs[0].currentName;
2016-12-06 22:51:06 +08:00
} else if (leftNoDisabledTabs.length) {
2017-03-03 13:38:46 +08:00
activeKey = leftNoDisabledTabs[leftNoDisabledTabs.length - 1].currentName;
2016-12-06 22:51:06 +08:00
} else {
2017-03-03 13:38:46 +08:00
activeKey = newTabs[0].currentName;
2016-12-06 22:51:06 +08:00
}
}
this.activeKey = activeKey;
2017-03-23 11:03:43 +08:00
this.$emit('input', activeKey);
2016-12-06 22:51:06 +08:00
}
2017-03-03 13:38:46 +08:00
this.$emit('on-tab-remove', tab.currentName);
2016-12-06 22:51:06 +08:00
this.updateNav();
2017-01-16 14:36:53 +08:00
},
showClose (item) {
if (this.type === 'card') {
if (item.closable !== null) {
return item.closable;
} else {
return this.closable;
}
} else {
return false;
}
2017-10-20 16:30:44 +08:00
},
scrollPrev() {
const containerWidth = this.$refs.navScroll.offsetWidth;
const currentOffset = this.getCurrentScrollOffset();
if (!currentOffset) return;
let newOffset = currentOffset > containerWidth
? currentOffset - containerWidth
: 0;
2017-10-20 16:30:44 +08:00
this.setOffset(newOffset);
},
scrollNext() {
const navWidth = this.$refs.nav.offsetWidth;
const containerWidth = this.$refs.navScroll.offsetWidth;
const currentOffset = this.getCurrentScrollOffset();
if (navWidth - currentOffset <= containerWidth) return;
let newOffset = navWidth - currentOffset > containerWidth * 2
? currentOffset + containerWidth
: (navWidth - containerWidth);
2017-10-20 16:30:44 +08:00
this.setOffset(newOffset);
},
getCurrentScrollOffset() {
const { navStyle } = this;
return navStyle.transform
? Number(navStyle.transform.match(/translateX\(-(\d+(\.\d+)*)px\)/)[1])
: 0;
2017-10-20 16:30:44 +08:00
},
getTabIndex(name){
return this.navList.findIndex(nav => nav.name === name);
},
2017-10-20 16:30:44 +08:00
setOffset(value) {
this.navStyle.transform = `translateX(-${value}px)`;
},
scrollToActiveTab() {
if (!this.scrollable) return;
const nav = this.$refs.nav;
const activeTab = this.$el.querySelector(`.${prefixCls}-tab-active`);
if(!activeTab) return;
const navScroll = this.$refs.navScroll;
const activeTabBounding = activeTab.getBoundingClientRect();
const navScrollBounding = navScroll.getBoundingClientRect();
const navBounding = nav.getBoundingClientRect();
const currentOffset = this.getCurrentScrollOffset();
let newOffset = currentOffset;
if (navBounding.right < navScrollBounding.right) {
newOffset = nav.offsetWidth - navScrollBounding.width;
}
if (activeTabBounding.left < navScrollBounding.left) {
newOffset = currentOffset - (navScrollBounding.left - activeTabBounding.left);
}else if (activeTabBounding.right > navScrollBounding.right) {
newOffset = currentOffset + activeTabBounding.right - navScrollBounding.right;
}
if(currentOffset !== newOffset){
this.setOffset(Math.max(newOffset, 0));
}
},
updateNavScroll(){
const navWidth = this.$refs.nav.offsetWidth;
const containerWidth = this.$refs.navScroll.offsetWidth;
const currentOffset = this.getCurrentScrollOffset();
if (containerWidth < navWidth) {
this.scrollable = true;
if (navWidth - currentOffset < containerWidth) {
this.setOffset(navWidth - containerWidth);
}
} else {
this.scrollable = false;
if (currentOffset > 0) {
this.setOffset(0);
}
}
},
handleResize(){
this.updateNavScroll();
2017-10-29 15:39:45 -07:00
},
isInsideHiddenElement () {
let parentNode = this.$el.parentNode;
while(parentNode && parentNode !== document.body) {
if (parentNode.style && parentNode.style.display === 'none') {
2017-10-29 15:39:45 -07:00
return parentNode;
}
parentNode = parentNode.parentNode;
}
return false;
},
updateVisibility(index){
[...this.$refs.panes.querySelectorAll(`.${prefixCls}-tabpane`)].forEach((el, i) => {
if (index === i) {
2018-08-20 17:11:59 +08:00
[...el.children].filter(child=> child.classList.contains(`${prefixCls}-tabpane`)).forEach(child => child.style.visibility = 'visible');
if (this.captureFocus) setTimeout(() => focusFirst(el, el), transitionTime);
} else {
setTimeout(() => {
2018-08-20 17:11:59 +08:00
[...el.children].filter(child=> child.classList.contains(`${prefixCls}-tabpane`)).forEach(child => child.style.visibility = 'hidden');
}, transitionTime);
}
});
2016-12-06 22:51:06 +08:00
}
},
watch: {
2017-03-03 13:38:46 +08:00
value (val) {
this.activeKey = val;
2018-05-04 15:32:31 +02:00
this.focusedKey = val;
2017-03-03 13:38:46 +08:00
},
2018-05-04 15:32:31 +02:00
activeKey (val) {
this.focusedKey = val;
2016-12-06 22:51:06 +08:00
this.updateBar();
2017-01-11 15:41:23 +08:00
this.updateStatus();
2017-04-10 09:58:51 +08:00
this.broadcast('Table', 'on-visible-change', true);
2017-10-20 17:44:58 +08:00
this.$nextTick(() => {
2017-10-20 16:30:44 +08:00
this.scrollToActiveTab();
});
// update visibility
const nextIndex = Math.max(this.getTabIndex(this.focusedKey), 0);
this.updateVisibility(nextIndex);
2016-12-06 22:51:06 +08:00
}
},
2017-03-27 15:24:48 +08:00
mounted () {
this.showSlot = this.$slots.extra !== undefined;
2017-10-20 16:30:44 +08:00
this.observer = elementResizeDetectorMaker();
this.observer.listenTo(this.$refs.navWrap, this.handleResize);
2017-10-29 15:39:45 -07:00
const hiddenParentNode = this.isInsideHiddenElement();
if (hiddenParentNode) {
this.mutationObserver = new MutationObserver(() => {
2017-10-29 15:39:45 -07:00
if (hiddenParentNode.style.display !== 'none') {
this.updateBar();
this.mutationObserver.disconnect();
2017-10-29 15:39:45 -07:00
}
});
this.mutationObserver.observe(hiddenParentNode, { attributes: true, childList: true, characterData: true, attributeFilter: ['style'] });
2017-10-29 15:39:45 -07:00
}
2018-05-08 12:35:10 +02:00
this.handleTabKeyboardSelect(true);
this.updateVisibility(this.getTabIndex(this.activeKey));
2017-10-20 16:30:44 +08:00
},
beforeDestroy() {
this.observer.removeListener(this.$refs.navWrap, this.handleResize);
2017-11-14 09:53:58 +08:00
if (this.mutationObserver) this.mutationObserver.disconnect();
2016-12-06 22:51:06 +08:00
}
2016-12-25 22:49:42 +08:00
};
</script>