mirror of
				https://github.com/youzan/vant.git
				synced 2025-10-22 03:44:48 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			268 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			268 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <template>
 | |
|   <div :class="b([type])">
 | |
|     <div
 | |
|       ref="wrap"
 | |
|       :class="[
 | |
|         b('wrap', [position, { scrollable }]),
 | |
|         { 'van-hairline--top-bottom': type === 'line' }
 | |
|       ]"
 | |
|     >
 | |
|       <div :class="b('nav', [type])" ref="nav">
 | |
|         <div v-if="type === 'line'" :class="b('line')" :style="lineStyle" />
 | |
|         <div
 | |
|           v-for="(tab, index) in tabs"
 | |
|           ref="tabs"
 | |
|           class="van-tab"
 | |
|           :class="{
 | |
|             'van-tab--active': index === curActive,
 | |
|             'van-tab--disabled': tab.disabled
 | |
|           }"
 | |
|           @click="onClick(index)"
 | |
|         >
 | |
|           <van-node v-if="tab.$slots.title" :node="tab.$slots.title" />
 | |
|           <span class="van-ellipsis" v-else>{{ tab.title }}</span>
 | |
|         </div>
 | |
|       </div>
 | |
|     </div>
 | |
|     <div :class="b('content')" ref="content">
 | |
|       <slot />
 | |
|     </div>
 | |
|   </div>
 | |
| </template>
 | |
| 
 | |
| <script>
 | |
| import create from '../utils/create';
 | |
| import { raf } from '../utils/raf';
 | |
| import { on, off } from '../utils/event';
 | |
| import VanNode from '../utils/node';
 | |
| import scrollUtils from '../utils/scroll';
 | |
| import Touch from '../mixins/touch';
 | |
| 
 | |
| export default create({
 | |
|   name: 'tabs',
 | |
| 
 | |
|   mixins: [Touch],
 | |
| 
 | |
|   components: {
 | |
|     VanNode
 | |
|   },
 | |
| 
 | |
|   model: {
 | |
|     prop: 'active'
 | |
|   },
 | |
| 
 | |
|   props: {
 | |
|     sticky: Boolean,
 | |
|     lineWidth: Number,
 | |
|     swipeable: Boolean,
 | |
|     active: {
 | |
|       type: [Number, String],
 | |
|       default: 0
 | |
|     },
 | |
|     type: {
 | |
|       type: String,
 | |
|       default: 'line'
 | |
|     },
 | |
|     duration: {
 | |
|       type: Number,
 | |
|       default: 0.2
 | |
|     },
 | |
|     swipeThreshold: {
 | |
|       type: Number,
 | |
|       default: 4
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   data() {
 | |
|     return {
 | |
|       tabs: [],
 | |
|       position: 'content-top',
 | |
|       curActive: 0,
 | |
|       lineStyle: {}
 | |
|     };
 | |
|   },
 | |
| 
 | |
|   computed: {
 | |
|     // whether the nav is scrollable
 | |
|     scrollable() {
 | |
|       return this.tabs.length > this.swipeThreshold;
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   watch: {
 | |
|     active(val) {
 | |
|       if (val !== this.curActive) {
 | |
|         this.correctActive(val);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     tabs(tabs) {
 | |
|       this.correctActive(this.curActive || this.active);
 | |
|       this.setLine();
 | |
|     },
 | |
| 
 | |
|     curActive() {
 | |
|       this.scrollIntoView();
 | |
|       this.setLine();
 | |
| 
 | |
|       // scroll to correct position
 | |
|       if (this.position === 'page-top' || this.position === 'content-bottom') {
 | |
|         scrollUtils.setScrollTop(this.scrollEl, scrollUtils.getElementTop(this.$el));
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     sticky(isSticky) {
 | |
|       this.scrollHandler(isSticky);
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   mounted() {
 | |
|     this.correctActive(this.active);
 | |
|     this.setLine();
 | |
| 
 | |
|     this.$nextTick(() => {
 | |
|       if (this.sticky) {
 | |
|         this.scrollHandler(true);
 | |
|       }
 | |
|       if (this.swipeable) {
 | |
|         this.swipeableHandler(true);
 | |
|       }
 | |
|       this.scrollIntoView();
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   beforeDestroy() {
 | |
|     /* istanbul ignore next */
 | |
|     if (this.sticky) {
 | |
|       this.scrollHandler(false);
 | |
|     }
 | |
|     /* istanbul ignore next */
 | |
|     if (this.swipeable) {
 | |
|       this.swipeableHandler(false);
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   methods: {
 | |
|     // whether to bind sticky listener
 | |
|     scrollHandler(init) {
 | |
|       this.scrollEl = this.scrollEl || scrollUtils.getScrollEventTarget(this.$el);
 | |
|       (init ? on : off)(this.scrollEl, 'scroll', this.onScroll, true);
 | |
|       if (init) {
 | |
|         this.onScroll();
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     // whether to bind content swipe listener
 | |
|     swipeableHandler(init) {
 | |
|       const { content } = this.$refs;
 | |
|       const action = init ? on : off;
 | |
|       action(content, 'touchstart', this.touchStart);
 | |
|       action(content, 'touchmove', this.touchMove);
 | |
|       action(content, 'touchend', this.onTouchEnd);
 | |
|       action(content, 'touchcancel', this.onTouchEnd);
 | |
|     },
 | |
| 
 | |
|     // watch swipe touch end
 | |
|     onTouchEnd() {
 | |
|       const { direction, deltaX, curActive } = this;
 | |
|       const minSwipeDistance = 50;
 | |
| 
 | |
|       /* istanbul ignore else */
 | |
|       if (direction === 'horizontal' && this.offsetX >= minSwipeDistance) {
 | |
|         /* istanbul ignore else */
 | |
|         if (deltaX > 0 && curActive !== 0) {
 | |
|           this.setCurActive(curActive - 1);
 | |
|         } else if (deltaX < 0 && curActive !== this.tabs.length - 1) {
 | |
|           this.setCurActive(curActive + 1);
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     // adjust tab position
 | |
|     onScroll() {
 | |
|       const scrollTop = scrollUtils.getScrollTop(this.scrollEl);
 | |
|       const elTopToPageTop = scrollUtils.getElementTop(this.$el);
 | |
|       const elBottomToPageTop = elTopToPageTop + this.$el.offsetHeight - this.$refs.wrap.offsetHeight;
 | |
|       if (scrollTop > elBottomToPageTop) {
 | |
|         this.position = 'content-bottom';
 | |
|       } else if (scrollTop > elTopToPageTop) {
 | |
|         this.position = 'page-top';
 | |
|       } else {
 | |
|         this.position = 'content-top';
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     // update nav bar style
 | |
|     setLine() {
 | |
|       this.$nextTick(() => {
 | |
|         if (!this.$refs.tabs) {
 | |
|           return;
 | |
|         }
 | |
| 
 | |
|         const tab = this.$refs.tabs[this.curActive];
 | |
|         const width = this.lineWidth || tab.offsetWidth;
 | |
|         const left = tab.offsetLeft + (tab.offsetWidth - width) / 2;
 | |
| 
 | |
|         this.lineStyle = {
 | |
|           width: `${width}px`,
 | |
|           transform: `translateX(${left}px)`,
 | |
|           transitionDuration: `${this.duration}s`
 | |
|         };
 | |
|       });
 | |
|     },
 | |
| 
 | |
|     // correct the value of active
 | |
|     correctActive(active) {
 | |
|       active = +active;
 | |
|       const exist = this.tabs.some(tab => tab.index === active);
 | |
|       const defaultActive = (this.tabs[0] || {}).index || 0;
 | |
|       this.setCurActive(exist ? active : defaultActive);
 | |
|     },
 | |
| 
 | |
|     setCurActive(active) {
 | |
|       this.curActive = active;
 | |
|       this.$emit('input', active);
 | |
|     },
 | |
| 
 | |
|     // emit event when clicked
 | |
|     onClick(index) {
 | |
|       const { title, disabled } = this.tabs[index];
 | |
|       if (disabled) {
 | |
|         this.$emit('disabled', index, title);
 | |
|       } else {
 | |
|         this.$emit('click', index, title);
 | |
|         this.setCurActive(index);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     // scroll active tab into view
 | |
|     scrollIntoView() {
 | |
|       if (!this.scrollable || !this.$refs.tabs) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       const tab = this.$refs.tabs[this.curActive];
 | |
|       const { nav } = this.$refs;
 | |
|       const { scrollLeft, offsetWidth: navWidth } = nav;
 | |
|       const { offsetLeft, offsetWidth: tabWidth } = tab;
 | |
| 
 | |
|       this.scrollTo(nav, scrollLeft, offsetLeft - (navWidth - tabWidth) / 2);
 | |
|     },
 | |
| 
 | |
|     // animate the scrollLeft of nav
 | |
|     scrollTo(el, from, to) {
 | |
|       let count = 0;
 | |
|       const frames = Math.round(this.duration * 1000 / 16);
 | |
|       const animate = () => {
 | |
|         el.scrollLeft += (to - from) / frames;
 | |
|         /* istanbul ignore next */
 | |
|         if (++count < frames) {
 | |
|           raf(animate);
 | |
|         }
 | |
|       };
 | |
|       animate();
 | |
|     }
 | |
|   }
 | |
| });
 | |
| </script>
 | 
