位置: IT常识 - 正文

Vue实现TagView标签导航栏缓存,解决重新进去页面不刷新问题(vue slot标签)

编辑:rootadmin
Vue实现TagView标签导航栏缓存,解决重新进去页面不刷新问题 实现的效果图

推荐整理分享Vue实现TagView标签导航栏缓存,解决重新进去页面不刷新问题(vue slot标签),希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:vue tabs标签页,vue中的标签,vue slot标签,vue的title,vue tab标签页,vue的title,vue标签中怎么写表达式,vue中的标签,内容如对您有帮助,希望把文章链接给更多的朋友!

Vue实现TagView标签导航栏缓存,解决重新进去页面不刷新问题(vue slot标签)

一、先实现导航栏的效果1、先在@/layout/components/Sidebar目录下新建一个组件ScrollPane.vue,把下面代码复制进去<template> <div ref="scrollContainer" class="scroll-container" @wheel.prevent="handleScroll"> <div ref="scrollWrapper" class="scroll-wrapper" :style="{left: left + 'px'}"> <slot /> </div> </div></template><script>const padding = 15 // tag's paddingexport default { name: 'ScrollPane', data() { return { left: 0 } }, methods: { handleScroll(e) { const eventDelta = e.wheelDelta || -e.deltaY * 3// wheelDelta:-120;deltaY:-120 const $container = this.$refs.scrollContainer// 外面的container const $containerWidth = $container.offsetWidth// 外面的container的宽度 const $wrapper = this.$refs.scrollWrapper// 里面 const $wrapperWidth = $wrapper.offsetWidth// 里面的宽度 if (eventDelta > 0) { this.left = Math.min(0, this.left + eventDelta)// min() 方法可返回指定的数字中带有最低值的数字。 } else { if ($containerWidth - padding < $wrapperWidth) { if (this.left < -($wrapperWidth - $containerWidth + padding)) { this.left = 1 } else { this.left = Math.max(this.left + eventDelta, $containerWidth - $wrapperWidth - padding) } } else { this.left = 0 } } }, moveToTarget($target) { const $container = this.$refs.scrollContainer const $containerWidth = $container.offsetWidth const $targetLeft = $target.offsetLeft const $targetWidth = $target.offsetWidth if ($targetLeft < -this.left) { // tag in the left this.left = -$targetLeft + padding } else if ($targetLeft + padding > -this.left && $targetLeft + $targetWidth < -this.left + $containerWidth - padding) { // tag in the current view // eslint-disable-line } else { // tag in the right this.left = -($targetLeft - ($containerWidth - $targetWidth) + padding) } } }}</script><style rel="stylesheet/scss" lang="scss" scoped> .scroll-container { white-space: nowrap; position: relative; overflow: hidden; width: 100%; .scroll-wrapper { position: absolute; } }</style>2、在相同目录下再新建一个TagsView.vue组件,把下面代码复制进去<template> <div class="tags-view-container"> <scroll-pane ref="scrollPane" class="tags-view-wrapper"> <router-link v-for="tag in Array.from(visitedViews)" ref="tag" :key="tag.path" :class="isActive(tag)?'active':''" :to="tag" class="tags-view-item" @contextmenu.prevent.native="openMenu(tag,$event)" > {{ tag.title }} <span class="el-icon-close" @click.prevent.stop="closeSelectedTag(tag)" /> </router-link> </scroll-pane> <ul v-show="visible" class="contextmenu" :style="{left:left+'px',top:top+'px'}"> <li @click="closeSelectedTag(selectedTag)">关闭</li> <li @click="closeOthersTags">关闭其他</li> <li @click="closeAllTags">关闭所有</li> </ul> </div></template><script>import ScrollPane from './ScrollPane.vue'export default { name: 'TagsView', components: { ScrollPane }, data() { return { visible: false, top: 0, left: 0, selectedTag: {} } }, computed: { visitedViews() { return this.$store.state.tagsView.visitedViews } }, watch: { $route() { this.addViewTags() this.moveToCurrentTag() }, visible(value) { if (value) { document.body.addEventListener('click', this.closeMenu) } else { document.body.removeEventListener('click', this.closeMenu) } } }, mounted() { this.addViewTags() }, methods: { generateRoute() { if (this.$route.name) { return this.$route } return false }, isActive(route) { return route.path === this.$route.path }, addViewTags() { const route = this.generateRoute() if (!route) { return false } this.$store.dispatch('addVisitedViews', route) }, moveToCurrentTag() { const tags = this.$refs.tag this.$nextTick(() => { for (const tag of tags) { if (tag.to.path === this.$route.path) { this.$refs.scrollPane.moveToTarget(tag.$el) break } } }) }, closeSelectedTag(view) { this.$store.dispatch('delVisitedViews', view).then((views) => { if (this.isActive(view)) { const latestView = views.slice(-1)[0] if (latestView) { this.$router.push(latestView) } else { this.$router.push('/') } } }) }, closeOthersTags() { this.$router.push(this.selectedTag) this.$store.dispatch('delOthersViews', this.selectedTag).then(() => { this.moveToCurrentTag() }) }, closeAllTags() { this.$store.dispatch('delAllViews') this.$router.push('/') }, openMenu(tag, e) { this.visible = true this.selectedTag = tag const offsetLeft = this.$el.getBoundingClientRect().left // container margin left this.left = e.clientX - offsetLeft + 15 // 15: margin right this.top = e.clientY }, closeMenu() { this.visible = false } }}</script><style rel="stylesheet/scss" lang="scss" scoped> .tags-view-container { .tags-view-wrapper { background: #fff; height: 34px; border-bottom: 1px solid #d8dce5; box-shadow: 0 1px 3px 0 rgba(0, 0, 0, .12), 0 0 3px 0 rgba(0, 0, 0, .04); .tags-view-item { display: inline-block; position: relative; height: 26px; line-height: 26px; border: 1px solid #d8dce5; color: #495060; background: #fff; padding: 0 8px; font-size: 12px; margin-left: 5px; margin-top: 4px; &:first-of-type { margin-left: 15px; } &.active { background-color: #42b983; color: #fff; border-color: #42b983; &::before { content: ''; background: #fff; display: inline-block; width: 8px; height: 8px; border-radius: 50%; position: relative; margin-right: 2px; } } } } .contextmenu { margin: 0; background: #fff; z-index: 100; position: absolute; list-style-type: none; padding: 5px 0; border-radius: 4px; font-size: 12px; font-weight: 400; color: #333; box-shadow: 2px 2px 3px 0 rgba(0, 0, 0, .3); li { margin: 0; padding: 7px 16px; cursor: pointer; &:hover { background: #eee; } } } }</style><style rel="stylesheet/scss" lang="scss"> //reset element css of el-icon-close .tags-view-wrapper { .tags-view-item { .el-icon-close { width: 16px; height: 16px; vertical-align: 2px; border-radius: 50%; text-align: center; transition: all .3s cubic-bezier(.645, .045, .355, 1); transform-origin: 100% 50%; &:before { transform: scale(.6); display: inline-block; vertical-align: -3px; } &:hover { background-color: #b4bccc; color: #fff; } } } }</style>3、在store文件夹下新建一个tagsView.js,同样把复制代码进去const tagsView = { state: { visitedViews: [], cachedViews: [] }, mutations: { ADD_VISITED_VIEWS: (state, view) => { if (state.visitedViews.some(v => v.path === view.path)) return state.visitedViews.push(Object.assign({}, view, { title: view.meta.title || 'no-name' })) if (!view.meta.noCache) { state.cachedViews.push(view.name) } }, DEL_VISITED_VIEWS: (state, view) => { for (const [i, v] of state.visitedViews.entries()) { if (v.path === view.path) { state.visitedViews.splice(i, 1) break } } for (const i of state.cachedViews) { if (i === view.name) { const index = state.cachedViews.indexOf(i) state.cachedViews.splice(index, 1) break } } }, DEL_OTHERS_VIEWS: (state, view) => { for (const [i, v] of state.visitedViews.entries()) { if (v.path === view.path) { state.visitedViews = state.visitedViews.slice(i, i + 1) break } } for (const i of state.cachedViews) { if (i === view.name) { const index = state.cachedViews.indexOf(i) state.cachedViews = state.cachedViews.slice(index, i + 1) break } } }, DEL_ALL_VIEWS: (state) => { state.visitedViews = [] state.cachedViews = [] } }, actions: { addVisitedViews({ commit }, view) { commit('ADD_VISITED_VIEWS', view) }, delVisitedViews({ commit, state }, view) { return new Promise((resolve) => { commit('DEL_VISITED_VIEWS', view) resolve([...state.visitedViews]) }) }, delOthersViews({ commit, state }, view) { return new Promise((resolve) => { commit('DEL_OTHERS_VIEWS', view) resolve([...state.visitedViews]) }) }, delAllViews({ commit, state }) { return new Promise((resolve) => { commit('DEL_ALL_VIEWS') resolve([...state.visitedViews]) }) } }}export default tagsView4、在store/index.js中引入tagsView.jsimport Vue from 'vue'import Vuex from 'vuex'import getters from './getters'// 自定义模块import app from './modules/app'import settings from './modules/settings'import user from './modules/user'import permission from './modules/permission'import tagsView from './modules/tagsView'Vue.use(Vuex)const store = new Vuex.Store({ modules: { app, settings, user, permission, tagsView }, getters})export default store5、在@/layout/的主组件中引入TagsView组件

二、实现重新返回页面时不刷新,保留原本页面内容1、在@/layout/components/Sidebar下的AppMain.vue中添加如下代码

主要是通过keep-alive 和 router-view结合来实现

<template> <section class="app-main"> <transition name="fade-transform" mode="out-in"> <keep-alive :include="cachedViews"> //这是新增加的 <router-view :key="key" /> </keep-alive> // </transition> </section></template><script>export default { name: 'AppMain', computed: { key() { return this.$route.path }, cachedViews() {//这是新增加的 return this.$store.state.tagsView.cachedViews }// }}</script>2、在route/index.js中为页面路由添加keepAlive为true,例如在首页 { path: '/', component: Layout, redirect: '/dashboard', children: [{ path: 'dashboard', name: 'Dashboard', component: () => import('@/views/dashboard/index'), //在meta对象中添加key值keepAlive为true meta: { title: '首页', icon: 'dashboard' ,keepAlive:true} }] }3、还有一点很重要!!要确保组件的命名和路由的命名是一样的,不然页面不会保留原本内容,会进行自动刷新

4、退出登录时,要清除缓存可以在store/modules/user.js中添加如下代码import store from '..'// user logout logout({ commit, state }) { return new Promise((resolve, reject) => { logout().then(() => { removeToken() resetRouter() store.dispatch('delAllViews') //增加多这行 commit('RESET_STATE') resolve() }).catch(error => { reject(error) }) }) },最后,这是项目的目录结构,红色框是新建的文件,蓝色是修改增加的文件

 

本文链接地址:https://www.jiuchutong.com/zhishi/277333.html 转载请保留说明!

上一篇:本地连接没有了怎么办(本地连接没有了怎么恢复)

下一篇:“讲述人”功能(讲述人功能怎么关闭)

  • excel表格怎么去掉网格线(excel表格怎么去掉公式)

    excel表格怎么去掉网格线(excel表格怎么去掉公式)

  • 微信好友删了怎么恢复回来(微信好友删了怎么找回)

    微信好友删了怎么恢复回来(微信好友删了怎么找回)

  • 点开朋友圈是一条横线是怎么回事(点开朋友圈是一条线)

    点开朋友圈是一条横线是怎么回事(点开朋友圈是一条线)

  • 华为mate30pro能root吗(华为mate30pro能装两张电信卡吗)

    华为mate30pro能root吗(华为mate30pro能装两张电信卡吗)

  • 荣耀30s屏幕是什么材质呢(荣耀30s屏幕是什么型号)

    荣耀30s屏幕是什么材质呢(荣耀30s屏幕是什么型号)

  • 红米note8指纹解锁在哪里(红米note8指纹解锁怎么不见了)

    红米note8指纹解锁在哪里(红米note8指纹解锁怎么不见了)

  • 苹果x卡贴机可以买吗(苹果x卡贴机可以更新吗)

    苹果x卡贴机可以买吗(苹果x卡贴机可以更新吗)

  • 苹果无法打开其他设备上通话(无法打开因为apple)

    苹果无法打开其他设备上通话(无法打开因为apple)

  • 华为mate9充电功率(mate9充电60问题解决)

    华为mate9充电功率(mate9充电60问题解决)

  • 平板电脑是电脑系统吗(平板电脑是电脑端还是手机端)

    平板电脑是电脑系统吗(平板电脑是电脑端还是手机端)

  • boss直聘会查看访客记录吗(boss直聘查看面试结果对方知道吗)

    boss直聘会查看访客记录吗(boss直聘查看面试结果对方知道吗)

  • 为什么一拍抖音就反光(为什么一拍抖音就关机)

    为什么一拍抖音就反光(为什么一拍抖音就关机)

  • 为什么面容突然不能用了(为什么面容突然识别不了)

    为什么面容突然不能用了(为什么面容突然识别不了)

  • 多媒体是指(多媒体是指什么的载体)

    多媒体是指(多媒体是指什么的载体)

  • excel2010默认工作簿名(excel2010默认工作表个数最多)

    excel2010默认工作簿名(excel2010默认工作表个数最多)

  • 怎么开启耳机安全音量(怎么设置耳机按钮)

    怎么开启耳机安全音量(怎么设置耳机按钮)

  • word关闭时点了不保存(word关闭按钮不见了)

    word关闭时点了不保存(word关闭按钮不见了)

  • 华为手机怎么卸载下载软件(华为手机怎么卸载不需要的软件)

    华为手机怎么卸载下载软件(华为手机怎么卸载不需要的软件)

  • 苹果xr不用按键怎么锁屏(苹果xr按键没反应)

    苹果xr不用按键怎么锁屏(苹果xr按键没反应)

  • 手机相册私密在哪里(手机相册私密相册在哪找到)

    手机相册私密在哪里(手机相册私密相册在哪找到)

  • 朋友圈每天删除有上限吗(朋友圈一天删除多少就删除失败了)

    朋友圈每天删除有上限吗(朋友圈一天删除多少就删除失败了)

  • vivo手机sos怎么设置(vivo手机sos紧急呼叫怎么设置)

    vivo手机sos怎么设置(vivo手机sos紧急呼叫怎么设置)

  • 电脑端订单回收站在哪里(电脑端订单回收站怎么删除)

    电脑端订单回收站在哪里(电脑端订单回收站怎么删除)

  • 苹果x怎么开省电模式(iphone x怎么开启省电)

    苹果x怎么开省电模式(iphone x怎么开启省电)

  • 天猫差评对卖家有何影响(给天猫店差评买家的评价管理会显示吗)

    天猫差评对卖家有何影响(给天猫店差评买家的评价管理会显示吗)

  • cad删除线段的一部分(cad中删除线段用哪个键)

    cad删除线段的一部分(cad中删除线段用哪个键)

  • 超详细Centos 6.5 文本模式图文系统安装教程(重庆四日游最佳攻略超详细)

    超详细Centos 6.5 文本模式图文系统安装教程(重庆四日游最佳攻略超详细)

  • PHPCMS 如何判断该栏目是否含有子栏目?(php判断https)

    PHPCMS 如何判断该栏目是否含有子栏目?(php判断https)

  • 动态规划篇——背包问题(动态规划知乎)

    动态规划篇——背包问题(动态规划知乎)

  • 一般纳税人和小规模纳税人哪个合适
  • 增值税加计抵减政策10%和15%
  • 国际税收协定是怎样避税的
  • 夏天单位发防暑物品的通知
  • 地税办理流程
  • 科技公司技术服务费计入什么科目
  • 转登记纳税人留抵退税政策
  • 会计准则哪些科目变化最大
  • 同一控制企业合并的特征包括
  • 事业单位退休职业年金发放多少个月
  • 增值税纳税义务发生时间
  • 公务出差乘坐飞机
  • 预付了订金收到发票怎么做分录?
  • 物业公司转售电费怎么开票
  • 中国银行结算包括哪些内容?怎么做会计分录?
  • 退预付账款会计分录
  • 增值税进项没收到必须开销项怎么办?
  • 住宿费餐费怎么做分录
  • 建筑行业预收账款
  • 所得税的税率有哪几种
  • 新公司值得入职吗
  • 小规模纳税人申报表怎么填
  • 房地产税开征的利弊分析
  • 实收资本印花税按次还是按年
  • 赠与合同要公证吗有效吗
  • 若依框架是谁写的
  • 挂账多年的往来账怎么处理
  • 牙痛怎么办怎么治疗
  • 房屋装修各项费用比例
  • 巨大的冰柱造句
  • 委托加工物资验收入库的金额
  • 单位缴费基数申报提交了能改吗
  • 适用执行企业会计准则的一般企业
  • 程序员的表白情书
  • 小微企业人员认定标准
  • phpwechat
  • 生产车间生产人员福利费
  • 智能优化算法可以用到哪里
  • 美国大学数学系排名
  • 以前多计提的工资怎么办
  • 公司的银行账号是不是和个人账号不一样
  • 高速公路通行费增值税税率
  • 社保可以抵扣吗
  • 人工智能自动对焦又叫什么
  • 市政道路基础设施
  • 残疾人工资加计扣除包括社保吗
  • 记账凭证核算处理方法
  • 事业单位结余是什么意思
  • 销售商品的运费的税费计入进项税额
  • 固定资产相关业务
  • 认证固定资产发票如何入账
  • 购买公司分录怎么写
  • 会计凭证移交凭证需要哪些程序
  • 企业筹建期间发生的费用应计入什么账户
  • sqlserver游标的使用场景
  • sql里面时间函数格式
  • Windows Server 2008使用软件授权管理工具
  • linux内核模块编译步骤
  • ubuntu14.04升级
  • winxp系统如何调整屏幕亮度
  • win10彻底恢复出厂设置
  • win8安全和维护在哪
  • windows 10更新卡住
  • win7系统检测不到u盘
  • win7如何查看电脑主板型号
  • linux查看nginx配置文件
  • python三角形角度
  • c#7.0新语法
  • 批处理在windows中的典型应用
  • 协程有什么用
  • unity怎么新建项目
  • jquery添加节点
  • unity3d图形学
  • javascript要怎么学
  • 基于javascript的毕业设计选题
  • [转]决算中甲供材如何扣除?
  • 总分机构在同一县城企业所得税需要预交吗
  • 北京国税电子税务局网址
  • 企业税务登记是什么证件
  • 公办幼儿园和企办幼儿园区别
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设