mirror of
				https://github.com/youzan/vant.git
				synced 2025-10-20 18:54:24 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			168 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			168 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
| <template>
 | |
|   <div :class="b()">
 | |
|     <div
 | |
|       :class="b('track')"
 | |
|       :style="style"
 | |
|       @touchstart="onTouchStart"
 | |
|       @touchmove="onTouchMove"
 | |
|       @touchend="onTouchEnd"
 | |
|       @touchcancel="onTouchEnd"
 | |
|     >
 | |
|       <div :class="b('head')">
 | |
|         <slot :name="status">
 | |
|           <div
 | |
|             v-if="status === 'pulling' || status === 'loosing'"
 | |
|             v-text="text"
 | |
|             :class="b('text')"
 | |
|           />
 | |
|           <div
 | |
|             v-if="status === 'loading'"
 | |
|             :class="b('loading')"
 | |
|           >
 | |
|             <loading />
 | |
|             <span v-text="text" />
 | |
|           </div>
 | |
|         </slot>
 | |
|       </div>
 | |
|       <slot />
 | |
|     </div>
 | |
|   </div>
 | |
| </template>
 | |
| 
 | |
| <script>
 | |
| import create from '../utils/create';
 | |
| import scrollUtils from '../utils/scroll';
 | |
| import Touch from '../mixins/touch';
 | |
| 
 | |
| export default create({
 | |
|   name: 'pull-refresh',
 | |
| 
 | |
|   mixins: [Touch],
 | |
| 
 | |
|   props: {
 | |
|     disabled: Boolean,
 | |
|     pullingText: String,
 | |
|     loosingText: String,
 | |
|     loadingText: String,
 | |
|     value: {
 | |
|       type: Boolean,
 | |
|       required: true
 | |
|     },
 | |
|     animationDuration: {
 | |
|       type: Number,
 | |
|       default: 300
 | |
|     },
 | |
|     headHeight: {
 | |
|       type: Number,
 | |
|       default: 50
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   data() {
 | |
|     return {
 | |
|       status: 'normal',
 | |
|       height: 0,
 | |
|       duration: 0
 | |
|     };
 | |
|   },
 | |
| 
 | |
|   computed: {
 | |
|     style() {
 | |
|       return {
 | |
|         transition: `${this.duration}ms`,
 | |
|         transform: `translate3d(0,${this.height}px, 0)`
 | |
|       };
 | |
|     },
 | |
| 
 | |
|     untouchable() {
 | |
|       return this.status === 'loading' || this.disabled;
 | |
|     },
 | |
| 
 | |
|     text() {
 | |
|       return this[`${this.status}Text`] || this.$t(this.status);
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   watch: {
 | |
|     value(val) {
 | |
|       this.duration = this.animationDuration;
 | |
|       this.getStatus(val ? this.headHeight : 0, val);
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   mounted() {
 | |
|     this.scrollEl = scrollUtils.getScrollEventTarget(this.$el);
 | |
|   },
 | |
| 
 | |
|   methods: {
 | |
|     onTouchStart(event) {
 | |
|       if (!this.untouchable && this.getCeiling()) {
 | |
|         this.duration = 0;
 | |
|         this.touchStart(event);
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     onTouchMove(event) {
 | |
|       if (this.untouchable) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       this.touchMove(event);
 | |
| 
 | |
|       if (!this.ceiling && this.getCeiling()) {
 | |
|         this.duration = 0;
 | |
|         this.startY = event.touches[0].clientY;
 | |
|         this.deltaY = 0;
 | |
|       }
 | |
| 
 | |
|       if (this.ceiling && this.deltaY >= 0) {
 | |
|         if (this.direction === 'vertical') {
 | |
|           this.getStatus(this.ease(this.deltaY));
 | |
|           event.cancelable && event.preventDefault();
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     onTouchEnd() {
 | |
|       if (!this.untouchable && this.ceiling && this.deltaY) {
 | |
|         this.duration = this.animationDuration;
 | |
|         if (this.status === 'loosing') {
 | |
|           this.getStatus(this.headHeight, true);
 | |
|           this.$emit('input', true);
 | |
|           this.$emit('refresh');
 | |
|         } else {
 | |
|           this.getStatus(0);
 | |
|         }
 | |
|       }
 | |
|     },
 | |
| 
 | |
|     getCeiling() {
 | |
|       this.ceiling = scrollUtils.getScrollTop(this.scrollEl) === 0;
 | |
|       return this.ceiling;
 | |
|     },
 | |
| 
 | |
|     ease(height) {
 | |
|       const { headHeight } = this;
 | |
|       return height < headHeight
 | |
|         ? height
 | |
|         : height < headHeight * 2
 | |
|           ? Math.round(headHeight + (height - headHeight) / 2)
 | |
|           : Math.round(headHeight * 1.5 + (height - headHeight * 2) / 4);
 | |
|     },
 | |
| 
 | |
|     getStatus(height, isLoading) {
 | |
|       this.height = height;
 | |
| 
 | |
|       const status = isLoading
 | |
|         ? 'loading' : height === 0
 | |
|           ? 'normal' : height < this.headHeight
 | |
|             ? 'pulling' : 'loosing';
 | |
| 
 | |
|       if (status !== this.status) {
 | |
|         this.status = status;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| });
 | |
| </script>
 | 
