成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術(shù)文章
文章詳情頁

Vue實(shí)現(xiàn)Tab標(biāo)簽路由效果并用Animate.css做轉(zhuǎn)場(chǎng)動(dòng)畫效果的代碼第1/3頁

瀏覽:150日期:2023-01-07 10:42:28

類似于瀏覽器窗口一樣的路由切換邏輯,看著還是挺高大上的,本以為有很多高級(jí)的玩意兒,奈何復(fù)雜的東西總是由簡(jiǎn)單的東西拼接而成的,這個(gè)功能也不例外。

本篇文章主要描述兩個(gè)問題:

如何實(shí)現(xiàn)這種Tab標(biāo)簽頁的路由效果 如何為路由切換添加轉(zhuǎn)場(chǎng)動(dòng)畫。

該功能的開發(fā)主要使用到 AntDesignVue 組件庫的Tab組件和 Animate.css

效果如下:

Vue實(shí)現(xiàn)Tab標(biāo)簽路由效果并用Animate.css做轉(zhuǎn)場(chǎng)動(dòng)畫效果的代碼第1/3頁

Tab標(biāo)簽頁實(shí)現(xiàn)

首先是該組件的模板部分, ContextMenu 組件是我們自定義的右鍵菜單,后面會(huì)說到。 a-tabs 組件則是 ant 的組件,具體用法不詳述,可以查看官方文檔。還有一個(gè) PageToggleTransition 組件,是我們用來實(shí)現(xiàn)動(dòng)畫切換的組件,非常簡(jiǎn)單。

/** * TabLayout.vue 的模板部分,簡(jiǎn)單看一下有個(gè)印象 */<template> <PageLayout> <ContextMenu :list='menuItems' :visible.sync='menuVisible' @select='onMenuSelect' /> <!-- 標(biāo)簽部分 --> <a-tabs type='editable-card' :hide-add='true' :active-key='activePage' @change='changePage' @edit='editPage' @contextmenu='onContextmenu' > <a-tab-pane v-for='page in pageList' :key='page.fullPath'> <template #tab> <span :data-key='page.fullPath'> {{ page.name }} </span> </template> </a-tab-pane> </a-tabs> <!-- 路由出口 --> <PageToggleTransition name='fadeIn'> <keep-alive :exclude='dustbin'> <router-view /> </keep-alive> </PageToggleTransition> </PageLayout></template>

原理

維護(hù)一個(gè) pageList ,通過監(jiān)聽路由變化動(dòng)態(tài)的添加和刪除page。而所謂的page,就是頁面的路由對(duì)象($route),我們正是通過 $route.fullPath 作為頁面的唯一標(biāo)識(shí)的。而刪除頁面時(shí)不光要操作 pageList ,還要利用 keep-alive 組件的 exclude 屬性刪除緩存。至于 a-tabs 組件的這個(gè)插槽,主要是為了綁定一個(gè)數(shù)據(jù)key,以便觸發(fā)contextmenu事件時(shí),可以更容易的獲取到對(duì)應(yīng)頁面的key值(fullPath)

理論存在,實(shí)踐開始。

路由監(jiān)聽

watch: { $route: { handler (route) { this.activePage = route.fullPath this.putCache(route) const index = this.pageList.findIndex(item => item.fullPath === route.fullPath) if (index === -1) { this.pageList.push(route) } }, immediate: true }}

路由變化時(shí),主要做三件事:

設(shè)置當(dāng)前頁(activePage) 將當(dāng)前頁加入緩存,即移出垃圾桶(dustbin) 如果當(dāng)前頁不在pageList中,則添加進(jìn)來。

頁面跳轉(zhuǎn)

methods: { changePage (key) { this.activePage = key this.$router.push(key) }, editPage (key, action) { if (action === ’remove’) { this.remove(key) } }, remove (key) { if (this.pageList.length <= 1) { return message.info(’最后一頁了哦~’) } let curIndex = this.pageList.findIndex(item => item.fullPath === key) const { matched } = this.pageList[curIndex] const componentName = last(matched).components.default.name this.dustbin.push(componentName) this.pageList.splice(curIndex, 1) // 如果刪除的是當(dāng)前頁才需要跳轉(zhuǎn) if (key === this.activePage) { // 判斷向左跳還是向右跳 curIndex = curIndex >= this.pageList.length ? this.pageList.length - 1 : curIndex const page = this.pageList[curIndex] this.$router.push(page.fullPath).finally(() => { this.dustbin.splice(0) // 重置,否則會(huì)影響到某些組件的緩存 }) } } ... ...}

這里主要主要說一下remove方法:

如果是最后一頁,則忽略 在pageList中找到當(dāng)前頁對(duì)應(yīng)的組件名用于刪除緩存(這里不清楚的可以看一下 keep-alive組件 ,和 $route.matched ) 如果刪除的是當(dāng)前頁,需要進(jìn)行頁面跳轉(zhuǎn),向左挑還是向右跳呢?

需要強(qiáng)調(diào)的時(shí) keep-alive 的 exclude 屬性,當(dāng)組件名被匹配到的時(shí)候就會(huì)立即清除緩存,所以, dustbin 添加完之后記得要重置,否則下次就不會(huì)緩存了。

自定義contextmenu事件

解釋下,contextmenu事件就是右鍵菜單事件,我們可以通過監(jiān)聽事件,使得右鍵菜單事件觸發(fā)的時(shí)候顯示我們的自定義菜單。

methods: { // 自定義右鍵菜單的關(guān)閉功能 onContextmenu (e) { const key = getTabKey(e.target) // 這里的判斷,用到了前面在span標(biāo)簽上加的data-key自定義屬性 if (!key) return // 主要是為了控制菜單的顯示或隱藏 e.preventDefault() // 組織默認(rèn)行為,顯示我們的自定義郵件菜單 this.menuVisible = true } ... ...}/** * 由于ant-design-vue組件庫的TabPane組件暫不支持自定義監(jiān)聽器,無法直接獲取到右鍵target所在標(biāo)簽頁的key 。故增加此方法用于 * 查詢右鍵target所在標(biāo)簽頁的標(biāo)識(shí) key ,以用于自定義右鍵菜單的事件處理。 * 注:TabPane組件支持自定義監(jiān)聽器后可去除該方法并重構(gòu) ‘自定義右鍵菜單的事件處理’ * @param target 查詢開始目標(biāo) * @param depth 查詢層級(jí)深度 (查找層級(jí)最多不超過3層,超過3層深度直接返回 null) * @returns {String} */function getTabKey (target, depth = 0) { if (depth > 2 || !target) { return null } return target.dataset.key || getTabKey(target.firstElementChild, ++depth)}

另外要說的是,dom元素上以 data- 開頭的屬性會(huì)被收錄進(jìn)元素的 dataset 屬性中, data-key 訪問時(shí)就是 dom.dataset.key

下面就是我們的 ContextMenu 組件了:

效果圖:

Vue實(shí)現(xiàn)Tab標(biāo)簽路由效果并用Animate.css做轉(zhuǎn)場(chǎng)動(dòng)畫效果的代碼第1/3頁

代碼如下:

<template> <a-menu v-show='visible' : :selectedKeys='selectedKeys' @click='handleClick' > <a-menu-item v-for='item in list' :key='item.key'> <a-icon v-if='item.icon' :type='item.icon'/> <span>{{ item.text }}</span> </a-menu-item> </a-menu></template><script>export default { name: ’ContextMenu’, props: { visible: { type: Boolean, required: false, default: false }, list: { type: Array, required: true, default: () => [] } }, data () { return { left: 0, top: 0, target: null, selectedKeys: [] } }, computed: { style () { return { left: this.left + ’px’, top: this.top + ’px’ } } }, created () { const clickHandler = () => this.closeMenu() const contextMenuHandler = e => this.setPosition(e) window.addEventListener(’click’, clickHandler) window.addEventListener(’contextmenu’, contextMenuHandler) this.$emit(’hook:beforeDestroy’, () => { window.removeEventListener(’click’, clickHandler) window.removeEventListener(’contextmenu’, contextMenuHandler) }) }, methods: { closeMenu () { this.$emit(’update:visible’, false) }, setPosition (e) { this.left = e.clientX this.top = e.clientY this.target = e.target }, handleClick ({ key }) { this.$emit(’select’, key, this.target) this.closeMenu() } }}</script><style lang='stylus' scoped> .contextmenu position fixed z-index 1000 border-radius 4px border 1px lightgrey solid box-shadow 4px 4px 10px lightgrey !important .ant-menu-item margin 0 !important</style>

這里需要強(qiáng)調(diào)的是鉤子函數(shù) created 的內(nèi)容:

1.首先全局事件需要成對(duì)出現(xiàn),有添加就要有移除,否則可能造成內(nèi)存泄漏,并導(dǎo)致一些其他的bug。就比如在模塊熱替換的項(xiàng)目中,會(huì)造成反復(fù)綁定的問題。

2.為什么這里要給window綁定contextmenu事件和click事件,之前不是綁過了嗎?這里的click事件主要是為了關(guān)閉菜單,右鍵菜單的特點(diǎn)是,不論點(diǎn)了什么點(diǎn)了哪里,只要點(diǎn)一下就會(huì)關(guān)閉。這里的contextmenu事件主要是為了獲取到事件對(duì)象 event ,以此來設(shè)置菜單的位置。而之前綁定在 a-tabs 組件上的contextmenu事件主要是為了阻止默認(rèn)事件,我們只攔截了該組件,而不需要攔截全局范圍。

自定義右鍵菜單主要是為了 從 event.target 中獲取到我們需要的key并以事件的形式傳遞出來 ,便于分發(fā)后面的邏輯,即:

onMenuSelect (key, target) { const tabKey = getTabKey(target) switch (key) { case ’1’: this.closeLeft(tabKey); break case ’2’: this.closeRight(tabKey); break case ’3’: this.closeOthers(tabKey); break default: break }}

這三種情況的邏輯是基本一致的,主要做了三件事:

清除緩存 刪除頁面,并設(shè)置當(dāng)前頁面 頁面跳轉(zhuǎn)

以closeOthers為例:

closeOthers (tabKey) { const index = this.pageList.findIndex(item => item.fullPath === tabKey) // 找到觸發(fā)事件時(shí)鼠標(biāo)停留在那個(gè)tab上 for (const route of this.pageList) { if (route.fullPath !== tabKey) { this.clearCache(route) // 清緩存 } } const page = this.pageList[index] this.pageList = 123下一頁閱讀全文

標(biāo)簽: Vue
相關(guān)文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
国产亚洲精品久| 久久成人久久鬼色| 国产永久精品大片wwwapp| 免费在线国产精品| 亚洲柠檬福利资源导航| 91猫先生在线| 精品免费国产一区二区三区四区| 国产精品123区| 7777精品伊人久久久大香线蕉的 | 久久精品久久精品| 色综合久久88色综合天天6 | 中文字幕av一区二区三区高 | 欧美主播一区二区三区| 日韩精品亚洲一区二区三区免费| 国产精品日韩一区二区三区| 亚洲图片另类小说| 激情综合自拍| 亚洲人成亚洲人成在线观看图片| 在线播放亚洲| 国产精品国产三级国产专播品爱网| 欧美福利视频| 中文字幕在线一区免费| 在线精品一区二区| 亚洲色图另类专区| 在线一区视频| 性久久久久久久久久久久 | 久久亚洲二区三区| 91麻豆123| 国产欧美精品在线观看| 欧美午夜电影在线观看| 最新成人av在线| 国产亚洲一区在线| 日本三级韩国三级欧美三级| 欧美午夜宅男影院| 国产精品一品二品| 日韩欧美黄色影院| 不卡一区在线观看| 国产欧美日韩久久| 亚洲一二三区精品| 亚洲猫色日本管| 久久蜜桃资源一区二区老牛| 免费高清不卡av| 日韩欧美国产高清| 欧美激情1区2区| 亚洲特级片在线| 国产日韩视频| 日韩电影在线免费看| 欧美日韩免费精品| 亚洲第一av色| 亚洲成人激情综合网| 久久精品一区二区| 欧美韩日精品| 日韩精品电影一区亚洲| 国产欧美一区在线| 国产一区二区无遮挡| 久久久一区二区| 亚洲一区二区精品在线| 精品动漫一区| 在线成人国产| 国产欧美亚洲一区| 日韩亚洲国产精品| 亚洲黄色一区二区三区| 色狠狠色噜噜噜综合网| 91久久精品一区二区别| 韩国精品主播一区二区在线观看| 丰满少妇在线播放bd日韩电影| 欧美自拍偷拍一区| 欧美精选午夜久久久乱码6080| 国产成人精品亚洲日本在线桃色| 久久老女人爱爱| 在线欧美日韩| 麻豆91精品91久久久的内涵| 日韩欧美亚洲国产精品字幕久久久 | 成人一道本在线| 中文字幕日韩av资源站| 亚洲综合精品| 高清在线成人网| 亚洲欧美另类在线| 欧美午夜一区二区三区免费大片| 91蜜桃在线免费视频| 亚洲gay无套男同| 日韩欧美高清dvd碟片| 亚洲国产二区| 精品一区二区三区久久| 国产人成一区二区三区影院| 亚洲欧美日韩精品久久久| 国产精品456| 亚洲精品中文在线影院| 欧美日韩中文字幕一区二区| 97se亚洲国产综合自在线| 亚洲自拍偷拍网站| 欧美一区二区三区视频免费播放| 国产中文一区| 久久99国内精品| 国产欧美日韩一区二区三区在线观看| 久久精品盗摄| av男人天堂一区| 日韩电影在线观看电影| 精品福利二区三区| 男女av一区三区二区色多| 国产69精品久久99不卡| 亚洲成人一区二区| 久久亚洲一区二区三区明星换脸 | 国产精品日产欧美久久久久| 欧美三级电影一区| 日韩视频中文| 9l国产精品久久久久麻豆| 日本伊人精品一区二区三区观看方式 | 大尺度一区二区| 亚洲成人av电影| 国产日产欧美一区| 欧美日韩一本到| 亚洲美女黄网| 成人国产精品视频| 奇米在线7777在线精品| 亚洲日本在线视频观看| 欧美成人video| 91国偷自产一区二区三区观看| 欧美日韩一区二区三区在线观看免| 国产精品中文字幕欧美| 亚洲成人7777| 中文字幕色av一区二区三区| 日韩精品自拍偷拍| 欧美日韩综合一区| 亚洲一区免费看| 欧美精品日本| 波波电影院一区二区三区| 捆绑变态av一区二区三区| 自拍偷拍亚洲欧美日韩| 久久奇米777| 91精品国产一区二区三区| 色一情一伦一子一伦一区| 亚洲精品视频啊美女在线直播| 色综合久久综合中文综合网| 丁香啪啪综合成人亚洲小说| 国内精品免费**视频| 天天免费综合色| 一区二区三区四区av| 中文字幕免费一区| 日韩一区二区三区在线视频| 欧美日韩在线播放三区四区| 色综合av在线| 国产精品日韩欧美一区二区| 国色天香一区二区| 色综合久久综合网欧美综合网| 成人黄色av电影| 国产一区视频网站| 国内精品不卡在线| 蜜臀av一区二区在线免费观看| 伊人一区二区三区| 中文字幕在线不卡| 国产精品久久毛片av大全日韩| 国产目拍亚洲精品99久久精品| 久久毛片高清国产| 久久精品亚洲国产奇米99| 日韩欧美激情一区| 日韩一区二区三区电影在线观看 | 中文字幕一区二区三| 国产欧美一区二区精品忘忧草| 日韩欧美一级二级三级| 91精品欧美一区二区三区综合在| 欧美日本在线播放| 欧美人体做爰大胆视频| 欧美色图片你懂的| 欧美日韩中文精品| 欧美亚洲图片小说| 在线免费不卡视频| 欧美唯美清纯偷拍| 欧美日韩不卡在线| 91精品国产91热久久久做人人| 欧美一区二区三区婷婷月色| 日韩一区二区在线观看视频 | 国产色综合久久| 中文字幕精品一区二区精品绿巨人 | 亚洲免费在线精品一区| 国产精品夜夜夜一区二区三区尤| 亚洲第一在线综合在线| 亚洲全部视频| 国产精品推荐精品| 一本大道久久a久久综合| 在线观看av一区| 欧美精品少妇一区二区三区| 日韩一区二区三区四区五区六区 | 奇米影视一区二区三区| 麻豆精品新av中文字幕| 精品一区二区av| 国产sm精品调教视频网站| 成人小视频在线| 99精品视频在线免费观看| 色综合天天做天天爱| 欧美精品一级| 日韩一级网站| 久久久久高清| 欧美挠脚心视频网站| 欧美草草影院在线视频| 国产精品丝袜一区| 亚洲精品美国一| 日韩精品一级二级 | 亚洲免费婷婷| 欧美亚男人的天堂|