vue3路由緩存(點(diǎn)擊菜單欄標(biāo)簽實(shí)現(xiàn)切換頁面)

環(huán)境搭建 參考 vite+vue3環(huán)境搭建

其次ui框架用了antdV(感覺比element更專業(yè)啊)
上面的都安裝好了后就可以開始了

0.配置路由文件

image.png

1.創(chuàng)建pinia緩存文件

import { defineStore } from 'pinia'

export const useMenuStore = defineStore({
  id: 'menu',
  state: () => {
    return {
      openMenu: [],
      menuIdx: 0,
      menuRoute: []
    }
  },
  getters: {},
  actions: {},
  // 開啟數(shù)據(jù)緩存
  persist: {
    enabled: true,
    strategies: [{
      key: 'my_project',
      storage: sessionStorage,
    }]
  }
})

2.路由入口

<script setup>
import { useMenuStore } from '@/stores/menu';
import { storeToRefs } from "pinia"

const menuStore = useMenuStore()
const { openMenu } = storeToRefs(menuStore) 
</script>
 <router-view v-slot="{ Component, route }">
            <keep-alive :include="openMenu">
              <component v-if="route.meta.keepAlive" :is="Component" :key="route.name" />
            </keep-alive>
            <component v-if="!route.meta.keepAlive" :is="Component" :key="route.name" />
          </router-view>

3.簡單的寫個菜單樣式


image.png

image.png

抽出成獨(dú)立的組件

<template>
  <div style="position: relative;border-radius: 5px;background-color: #fff;overflow: hidden;" :style="`${isScrollBtn ? 'padding: 0 20px 0 27px;' : 'padding: 0 37px 0 10px;'}`">
    <div class="leftBtn" @click="handleLeft" v-show="isScrollBtn"><DoubleLeftOutlined /></div>
    <div class="menu-tag" ref="tagBox" @scroll="menuScroll">
  
      <a-tag :closable="v != 'Index'"  v-for="(v, index) in openMenu" :bordered="false" :color="`${index == menuIdx ? '#108ee9' : 'blue'}`" @click="handleTag(v)" @close.prevent="handleClose(v, index)">{{v}}</a-tag>

    </div>
    <div class="rightBtn" @click="handleRight" v-show="isScrollBtn"><DoubleRightOutlined /></div>
  </div>
</template>

<script setup>
  import { DoubleRightOutlined, DoubleLeftOutlined } from '@ant-design/icons-vue';
  import { ref, nextTick, watch } from 'vue'
  import { useMenuStore } from '@/stores/menu';
  import { storeToRefs } from "pinia"
  import { useRouter, useRoute } from "vue-router"
  const route = new useRoute()
  const router = new useRouter()
  const menuStore = useMenuStore()
  const { menuIdx, openMenu, menuRoute } = storeToRefs(menuStore) 
  const tagBox = ref(null)

  const isScrollBtn = ref(false)
  let offsetWidth = 0
  let scrollWidth = 0
  let scrollLeft = 0
  let isScroll = false // 是否在滾動
  let scrollLenth = 300 //滾動距離

  watch(() => route.name, async (now) => {
    let Idx = openMenu.value.indexOf(now)
    if(Idx == -1) {
      openMenu.value.push(now)
      menuRoute.value.push(route.path)
      menuIdx.value = openMenu.value.length - 1
      await nextTick()
      isBeyond()
    } else {
      menuIdx.value = Idx
    }
  }, {immediate: true})
  /** 監(jiān)聽容器盒子滾動事件 */
  let scrollTimer;
  const menuScroll = () => {
    isScroll = true
    if(scrollTimer) clearTimeout(scrollTimer);
    scrollTimer = setTimeout(() => {
      getOpton()
      isScroll = false
    }, 300);
  }
  /** 獲取菜單容器寬度等信息 */
  const getOpton = () => {
    offsetWidth = tagBox.value.offsetWidth
    scrollWidth = tagBox.value.scrollWidth
    scrollLeft = tagBox.value.scrollLeft
  }
  /** 計算寬度是否超出容器寬度 */
  const isBeyond =  () => {
    getOpton()
    if(scrollWidth > offsetWidth) {
      isScrollBtn.value = true
      tagBox.value.scrollTo(scrollWidth - offsetWidth + 47, 0)
    } else {
      isScrollBtn.value = false
    }
  }
  const handleLeft = () => {
    if(isScroll) return
    tagBox.value.scrollTo(scrollLeft - scrollLenth, 0)
  }
  const handleRight = () => {
    if(isScroll) return
    tagBox.value.scrollTo(scrollLeft + scrollLenth, 0)
  }
  // 點(diǎn)擊菜單
  const handleTag = (name) => {
    let Idx = openMenu.value.indexOf(name)
    menuIdx.value = Idx
    router.push(menuRoute.value[Idx])
  }
  // 關(guān)閉菜單
  const handleClose = async (name,index) => {
    
    // 刪除的是高亮菜單
    if(index == menuIdx.value) {
      menuIdx.value -= 1
      router.push(menuRoute.value[index - 1])
    }
    openMenu.value.splice(index, 1)
    menuRoute.value.splice(index, 1)
    await nextTick()
    isBeyond()
  }
</script>
<style lang='less' scoped>  
  :deep(.menu-tag) {
    cursor: pointer;
    user-select: none;
  }
  .leftBtn,.rightBtn {
    display: flex;
    align-items: center;
    justify-content: center;
    font-size: 18px;
    color: #fff2f0;
    position: absolute;
    top: 0;
    
    bottom: 0;
    background-color: rgb(60 60 60 / 88%);
    width: 20px;
    z-index: 1;
    box-shadow: #fff inset 0 0px 17px;
    cursor: pointer;
    &:hover {
      color: #d8d5d5;
      background-color: #999696;
    }
  }
  .leftBtn {
    left: 0;
  }
  .rightBtn {
    right: 0;
  }
  .menu-tag {
    // width: 1000px;
    height: 40px;
    
    box-sizing: border-box;
    // margin: 10px 16px;
    
    display: flex;
    align-items: center;
    overflow: hidden;
    justify-content: flex-start;
    &>* {
      flex-shrink: 0;
    }
    
  }
</style>

最后路由緩存需要頁面申明name
使用steup語法糖 可以安裝插件 vite-plugin-vue-setup-extend
vite.config.js內(nèi)配置


image.png

就可以直接

<script steup name="aaaa">
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末茄唐,一起剝皮案震驚了整個濱河市刻获,隨后出現(xiàn)的幾起案子蚀狰,更是在濱河造成了極大的恐慌豁遭,老刑警劉巖椒惨,帶你破解...
    沈念sama閱讀 216,692評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡搪桂,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,482評論 3 392
  • 文/潘曉璐 我一進(jìn)店門盯滚,熙熙樓的掌柜王于貴愁眉苦臉地迎上來踢械,“玉大人,你說我怎么就攤上這事魄藕∧诹校” “怎么了?”我有些...
    開封第一講書人閱讀 162,995評論 0 353
  • 文/不壞的土叔 我叫張陵背率,是天一觀的道長话瞧。 經(jīng)常有香客問我嫩与,道長,這世上最難降的妖魔是什么移稳? 我笑而不...
    開封第一講書人閱讀 58,223評論 1 292
  • 正文 為了忘掉前任蕴纳,我火速辦了婚禮,結(jié)果婚禮上个粱,老公的妹妹穿的比我還像新娘古毛。我一直安慰自己,他們只是感情好都许,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,245評論 6 388
  • 文/花漫 我一把揭開白布稻薇。 她就那樣靜靜地躺著,像睡著了一般胶征。 火紅的嫁衣襯著肌膚如雪塞椎。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,208評論 1 299
  • 那天睛低,我揣著相機(jī)與錄音案狠,去河邊找鬼。 笑死钱雷,一個胖子當(dāng)著我的面吹牛骂铁,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播罩抗,決...
    沈念sama閱讀 40,091評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼拉庵,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了套蒂?” 一聲冷哼從身側(cè)響起钞支,我...
    開封第一講書人閱讀 38,929評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎操刀,沒想到半個月后烁挟,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,346評論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡骨坑,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,570評論 2 333
  • 正文 我和宋清朗相戀三年撼嗓,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片卡啰。...
    茶點(diǎn)故事閱讀 39,739評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡静稻,死狀恐怖警没,靈堂內(nèi)的尸體忽然破棺而出匈辱,到底是詐尸還是另有隱情,我是刑警寧澤杀迹,帶...
    沈念sama閱讀 35,437評論 5 344
  • 正文 年R本政府宣布亡脸,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏浅碾。R本人自食惡果不足惜大州,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,037評論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望垂谢。 院中可真熱鬧厦画,春花似錦、人聲如沸滥朱。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,677評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽徙邻。三九已至排嫌,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間缰犁,已是汗流浹背淳地。 一陣腳步聲響...
    開封第一講書人閱讀 32,833評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留帅容,地道東北人颇象。 一個月前我還...
    沈念sama閱讀 47,760評論 2 369
  • 正文 我出身青樓,卻偏偏與公主長得像丰嘉,于是被迫代替她去往敵國和親夯到。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,647評論 2 354

推薦閱讀更多精彩內(nèi)容