From be3fbd2499703ac2abbe727c171c12e83e337767 Mon Sep 17 00:00:00 2001 From: marxy <334905044@qq.com> Date: Fri, 20 Oct 2017 16:30:44 +0800 Subject: [PATCH] Tabs add scroll --- examples/routers/tabs.vue | 39 ++++++++++++++++++++++++++++++++++++++- src/components/tabs/tabs.vue | 105 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++----- src/styles/components/tabs.less | 23 +++++++++++++++++++++++ 3 files changed, 161 insertions(+), 6 deletions(-) diff --git a/examples/routers/tabs.vue b/examples/routers/tabs.vue index 18a4d81..9be14d2 100644 --- a/examples/routers/tabs.vue +++ b/examples/routers/tabs.vue @@ -70,7 +70,7 @@ <!--</script>--> -<template> +<!-- <template> <Tabs type="card" closable @on-tab-remove="handleTabRemove"> <TabPane label="标签一" v-if="tab0">标签一的内容</TabPane> <TabPane label="标签二" v-if="tab1">标签二的内容</TabPane> @@ -92,4 +92,41 @@ } } } +</script> --> + +<template> + <div> + <Button type="ghost" @click="toFirst" size="small">to first</Button> + <Button type="ghost" @click="toLast" size="small">to last</Button> + <Tabs type="card" :animated="animated" v-model="activeTab"> + <TabPane v-for="tab in tabs" :key="tab" :label="'标签' + tab" :name="tab+''" closable>标签{{ tab }}</TabPane> + <div slot="extra"> + <Button type="ghost" @click="handleTabsAdd" size="small">增加</Button> + </div> + </Tabs> + </div> +</template> +<script> + export default { + data () { + return { + tabs: 2, + activeTab:"2", + animated:true + } + }, + methods: { + handleTabsAdd () { + this.tabs ++; + this.activeTab = this.tabs + ''; + }, + toFirst () { + this.activeTab = '1'; + }, + toLast () { + this.activeTab = this.tabs+''; + } + } + } </script> + diff --git a/src/components/tabs/tabs.vue b/src/components/tabs/tabs.vue index 4159fe2..bc9afd9 100644 --- a/src/components/tabs/tabs.vue +++ b/src/components/tabs/tabs.vue @@ -1,10 +1,13 @@ <template> <div :class="classes"> <div :class="[prefixCls + '-bar']"> + <div :class="[prefixCls + '-nav-right']" v-if="showSlot"><slot name="extra"></slot></div> <div :class="[prefixCls + '-nav-container']"> - <div :class="[prefixCls + '-nav-wrap']"> - <div :class="[prefixCls + '-nav-scroll']"> - <div :class="[prefixCls + '-nav']" ref="nav"> + <div ref="navWrap" :class="[prefixCls + '-nav-wrap', scrollable ? prefixCls + '-nav-scrollable' : '']" > + <span :class="[prefixCls + '-nav-prev', scrollable ? '' : prefixCls + '-nav-scroll-disabled']" @click="scrollPrev"><Icon type="chevron-left"></Icon></span> + <span :class="[prefixCls + '-nav-next', scrollable ? '' : prefixCls + '-nav-scroll-disabled']" @click="scrollNext"><Icon type="chevron-right"></Icon></span> + <div ref="navScroll" :class="[prefixCls + '-nav-scroll']"> + <div ref="nav" :class="[prefixCls + '-nav']" class="nav-text" :style="navStyle"> <div :class="barClasses" :style="barStyle"></div> <div :class="tabCls(item)" v-for="(item, index) in navList" @click="handleChange(index)"> <Icon v-if="item.icon !== ''" :type="item.icon"></Icon> @@ -13,7 +16,6 @@ <Icon v-if="showClose(item)" type="ios-close-empty" @click.native.stop="handleRemove(index)"></Icon> </div> </div> - <div :class="[prefixCls + '-nav-right']" v-if="showSlot"><slot name="extra"></slot></div> </div> </div> </div> @@ -26,6 +28,7 @@ import Render from '../base/render'; import { oneOf } from '../../utils/assist'; import Emitter from '../../mixins/emitter'; + import elementResizeDetectorMaker from 'element-resize-detector'; const prefixCls = 'ivu-tabs'; @@ -65,7 +68,11 @@ barWidth: 0, barOffset: 0, activeKey: this.value, - showSlot: false + showSlot: false, + navStyle:{ + transform: '' + }, + scrollable:false }; }, computed: { @@ -163,6 +170,7 @@ } else { this.barOffset = 0; } + this.updateNavScroll(); }); }, updateStatus () { @@ -222,6 +230,85 @@ } else { return false; } + }, + scrollPrev() { + const containerWidth = this.$refs.navScroll.offsetWidth; + const currentOffset = this.getCurrentScrollOffset(); + + if (!currentOffset) return; + + let newOffset = currentOffset > containerWidth + ? currentOffset - containerWidth + : 0; + + 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); + + this.setOffset(newOffset); + }, + getCurrentScrollOffset() { + const { navStyle } = this; + return navStyle.transform + ? Number(navStyle.transform.match(/translateX\(-(\d+(\.\d+)*)px\)/)[1]) + : 0; + }, + 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(); } }, watch: { @@ -232,10 +319,18 @@ this.updateBar(); this.updateStatus(); this.broadcast('Table', 'on-visible-change', true); + this.$nextTick(function(){ + this.scrollToActiveTab(); + }); } }, mounted () { this.showSlot = this.$slots.extra !== undefined; + this.observer = elementResizeDetectorMaker(); + this.observer.listenTo(this.$refs.navWrap, this.handleResize); + }, + beforeDestroy() { + this.observer.removeListener(this.$refs.navWrap, this.handleResize); } }; </script> diff --git a/src/styles/components/tabs.less b/src/styles/components/tabs.less index fef5af4..ef813e8 100644 --- a/src/styles/components/tabs.less +++ b/src/styles/components/tabs.less @@ -56,6 +56,29 @@ &-nav-right{ float: right; + margin-left: 5px; + } + + &-nav-prev{ + position:absolute; + line-height: 32px; + cursor: pointer; + left:0; + } + + &-nav-next{ + position:absolute; + line-height: 32px; + cursor: pointer; + right:0; + } + + &-nav-scrollable{ + padding: 0 12px; + } + + &-nav-scroll-disabled{ + display: none; } &-nav { -- libgit2 0.21.4