perf[TagsView]: set the scrollPane as a business component (#1660)
This commit is contained in:
		
							
								
								
									
										85
									
								
								src/views/layout/components/TagsView/ScrollPane.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								src/views/layout/components/TagsView/ScrollPane.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | ||||
| <template> | ||||
|   <el-scrollbar ref="scrollContainer" :vertical="false" class="scroll-container" @wheel.native.prevent="handleScroll"> | ||||
|     <slot/> | ||||
|   </el-scrollbar> | ||||
| </template> | ||||
|  | ||||
| <script> | ||||
| const tagAndTagSpacing = 4 // tagAndTagSpacing | ||||
|  | ||||
| export default { | ||||
|   name: 'ScrollPane', | ||||
|   data() { | ||||
|     return { | ||||
|       left: 0 | ||||
|     } | ||||
|   }, | ||||
|   computed: { | ||||
|     scrollWrapper() { | ||||
|       return this.$refs.scrollContainer.$refs.wrap | ||||
|     } | ||||
|   }, | ||||
|   methods: { | ||||
|     handleScroll(e) { | ||||
|       const eventDelta = e.wheelDelta || -e.deltaY * 40 | ||||
|       const $scrollWrapper = this.scrollWrapper | ||||
|       $scrollWrapper.scrollLeft = $scrollWrapper.scrollLeft + eventDelta / 4 | ||||
|     }, | ||||
|     moveToTarget(currentTag) { | ||||
|       const $container = this.$refs.scrollContainer.$el | ||||
|       const $containerWidth = $container.offsetWidth | ||||
|       const $scrollWrapper = this.scrollWrapper | ||||
|       const tagList = this.$parent.$refs.tag | ||||
|  | ||||
|       let firstTag = null | ||||
|       let lastTag = null | ||||
|  | ||||
|       // find first tag and last tag | ||||
|       if (tagList.length > 0) { | ||||
|         firstTag = tagList[0] | ||||
|         lastTag = tagList[tagList.length - 1] | ||||
|       } | ||||
|  | ||||
|       if (firstTag === currentTag) { | ||||
|         $scrollWrapper.scrollLeft = 0 | ||||
|       } else if (lastTag === currentTag) { | ||||
|         $scrollWrapper.scrollLeft = $scrollWrapper.scrollWidth - $containerWidth | ||||
|       } else { | ||||
|         // find preTag and nextTag | ||||
|         const currentIndex = tagList.findIndex(item => item === currentTag) | ||||
|         const prevTag = tagList[currentIndex - 1] | ||||
|         const nextTag = tagList[currentIndex + 1] | ||||
|  | ||||
|         // the tag's offsetLeft after of nextTag | ||||
|         const afterNextTagOffsetLeft = nextTag.$el.offsetLeft + nextTag.$el.offsetWidth + tagAndTagSpacing | ||||
|  | ||||
|         // the tag's offsetLeft before of prevTag | ||||
|         const beforePrevTagOffsetLeft = prevTag.$el.offsetLeft - tagAndTagSpacing | ||||
|  | ||||
|         if (afterNextTagOffsetLeft > $scrollWrapper.scrollLeft + $containerWidth) { | ||||
|           $scrollWrapper.scrollLeft = afterNextTagOffsetLeft - $containerWidth | ||||
|         } else if (beforePrevTagOffsetLeft < $scrollWrapper.scrollLeft) { | ||||
|           $scrollWrapper.scrollLeft = beforePrevTagOffsetLeft | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style rel="stylesheet/scss" lang="scss" scoped> | ||||
| .scroll-container { | ||||
|   white-space: nowrap; | ||||
|   position: relative; | ||||
|   overflow: hidden; | ||||
|   width: 100%; | ||||
|   /deep/ { | ||||
|     .el-scrollbar__bar { | ||||
|       bottom: 0px; | ||||
|     } | ||||
|     .el-scrollbar__wrap { | ||||
|       height: 49px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
| @@ -26,7 +26,7 @@ | ||||
| </template> | ||||
| 
 | ||||
| <script> | ||||
| import ScrollPane from '@/components/ScrollPane' | ||||
| import ScrollPane from './ScrollPane' | ||||
| import { generateTitle } from '@/utils/i18n' | ||||
| import path from 'path' | ||||
| 
 | ||||
| @@ -90,7 +90,6 @@ export default { | ||||
|           } | ||||
|         } | ||||
|       }) | ||||
| 
 | ||||
|       return tags | ||||
|     }, | ||||
|     initTags() { | ||||
| @@ -115,12 +114,10 @@ export default { | ||||
|         for (const tag of tags) { | ||||
|           if (tag.to.path === this.$route.path) { | ||||
|             this.$refs.scrollPane.moveToTarget(tag) | ||||
| 
 | ||||
|             // when query is different then update | ||||
|             if (tag.to.fullPath !== this.$route.fullPath) { | ||||
|               this.$store.dispatch('updateVisitedView', this.$route) | ||||
|             } | ||||
| 
 | ||||
|             break | ||||
|           } | ||||
|         } | ||||
| @@ -178,8 +175,8 @@ export default { | ||||
|       } else { | ||||
|         this.left = left | ||||
|       } | ||||
|       this.top = e.clientY | ||||
| 
 | ||||
|       this.top = e.clientY | ||||
|       this.visible = true | ||||
|       this.selectedTag = tag | ||||
|     }, | ||||
| @@ -1,4 +1,4 @@ | ||||
| export { default as Navbar } from './Navbar' | ||||
| export { default as Sidebar } from './Sidebar/index.vue' | ||||
| export { default as TagsView } from './TagsView' | ||||
| export { default as TagsView } from './TagsView/index.vue' | ||||
| export { default as AppMain } from './AppMain' | ||||
|   | ||||
		Reference in New Issue
	
	Block a user