资讯专栏INFORMATION COLUMN

Vue中对iframe实现keep alive(无刷新)

k00baa / 717人阅读

摘要:前言最近一个需求,需要在项目中加入含有的页面,同时在路由切换的过程中,要求的内容不会被刷新。的原理要实现对保持页的状态。实现的思路既然保持页里的状态很难实现,在这个时候我想到了一个别的方法。

前言

最近一个需求,需要在Vue项目中加入含有iframe的页面,同时在路由切换的过程中,要求iframe的内容不会被刷新。一开始使用了Vue自带的keep- alive发现没有用,于是自己研究了一下解决方案。。。。。。

Vue的keep-alive原理

要实现对保持iframe页的状态。我们先搞清楚为什么Vue的keep-alive不能凑效。keep-alive原理是把组件里的节点信息保留在了VNode(在内存里),在需要渲染时候从Vnode渲染到真实DOM上。iframe页里的内容并不属于节点的信息,所以使用keep-alive依然会重新渲染iframe内的内容。另外,我也尝试有过想法:如果把整个iframe节点保存起来,然后需要切换时把它渲染到目标节点上,能否实现iframe页不被刷新呢?————也是不可行的,iframe每一次渲染就相当于打开一个新的网页窗口,即使把节点保存下来,在渲染时iframe页还是刷新的。

实现的思路

既然保持iframe页里的状态很难实现,在这个时候我想到了一个别的方法。能否在Vue的route-view节点上动点手脚?使得在切换非iframe页的时候使用Vue的路由,当切换iframe页时则使用v-show切换显示与隐藏,使得iframe节点一直不被删除,这样就能保持iframe的状态了。

我们简陋的实现一下以上的效果,上代码:

入口main.js:

import Vue from "vue/dist/vue.js"
import App from "./App.vue"
import VueRouter from "vue-router";

const Index = { template: "
Index
" } const routes = [ // 含有iframe的两个页面 { path: "/f1", name: "f1" }, // 含有iframe的两个页面 { path: "/f2", name: "f2" }, { path: "/index", component: Index } ] const router = new VueRouter({ routes }); Vue.use(VueRouter); new Vue({ render: h => h(App), router }).$mount("#app")

根组件:



上面代码简单来说,关键的地方首先是main.js初始化路由时,对iframe页不填写属性component,这样页面就是空白的。然后在router-view节点旁边渲染iframe页组件,使用$route.path判断当前路由的指向,控制iframe页的显示与隐藏

优化

上面代码简单的解决了问题,但还有一些地方可以优化:

iframe页在根节点App.vue一渲染时已经渲染了,对此iframe页可以做成懒加载,只有在进入过相应页面了触发渲染,并且渲染过之后就用v-show切换显示与隐藏

每当增加一个iframe页都要增加一段的组件引入注册和调用的代码。比较繁琐。我们目标应该做到每增加一个iframe页,只需要添加尽量少的代码。这里思路是:

在路由配置中定义一个属性,用于标识该页面是否含有iframe的页面

根据标识,iframe页组件自动动态注册和渲染,无需再手写额外的代码

router-view和iframe切换的逻辑封装成新组件,用它替代原有的router-view

我们先修改router的配置,增加一个属性名iframeComponent,用于标识是否包含iframe,该属性的值是组件文件引用

main.js:

import F1 from "./components/f1";
import F2 from "./components/f2";

const routes = [
  {
    path: "/f1",
    name: "f1",
    iframeComponent: F1 // 用于标识是否含有iframe页
  },
  {
    path: "/f2",
    name: "f2",
    iframeComponent: F2 // 用于标识是否含有iframe页
  },
  {
    path: "/index",
    component: { template: "
Index
" } } ] const router = new VueRouter({ routes // (缩写)相当于 routes: routes }); new Vue({ render: h => h(App), router }).$mount("#app")

接下来我们第二步和第三步结合在一起,封装新的组件iframe-router-view.vue:



该组件主要做的是根据main.ja里的routes生成一个只含有iframe页的数组对象。

watch上监听$route,判断当前页面在iframe页列表里的话就设置hasOpen属性为true,渲染该组件

用v-show="$route.path === item.path"切换iframe页的显示与隐藏。

逻辑并不复杂,这里就不多赘述。

结语

大家如果有更好的实现方法,或者我上面还有什么需要更正的错误,欢迎交流。
上面demo的代码放在了个人github上https://github.com/jmx164491960/vue-iframe-demo

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/106029.html

相关文章

  • Vue中对iframe实现keep alive刷新

    摘要:前言最近一个需求,需要在项目中加入含有的页面,同时在路由切换的过程中,要求的内容不会被刷新。的原理要实现对保持页的状态。实现的思路既然保持页里的状态很难实现,在这个时候我想到了一个别的方法。 前言 最近一个需求,需要在Vue项目中加入含有iframe的页面,同时在路由切换的过程中,要求iframe的内容不会被刷新。一开始使用了Vue自带的keep- alive发现没有用,于是自己研究了...

    heartFollower 评论0 收藏0
  • 面试官常问——vue

    摘要:如果要相应状态改变,通常最好使用计算属性或取而代之。那解决问题的思路便是在改变的情况下,保证页面的不刷新。后面值的变化,并不会导致浏览器向服务器发出请求,浏览器不发出请求,也就不会刷新页面。 1.vue生命周期2.vue 双向绑定原理3.vue router原理4.vue router动态路由 1.vue 生命周期钩子 showImg(https://segmentfault.com/...

    BlackMass 评论0 收藏0
  • 面试官常问——vue

    摘要:如果要相应状态改变,通常最好使用计算属性或取而代之。那解决问题的思路便是在改变的情况下,保证页面的不刷新。后面值的变化,并不会导致浏览器向服务器发出请求,浏览器不发出请求,也就不会刷新页面。 1.vue生命周期2.vue 双向绑定原理3.vue router原理4.vue router动态路由 1.vue 生命周期钩子 showImg(https://segmentfault.com/...

    xingqiba 评论0 收藏0
  • 面试官常问——vue

    摘要:如果要相应状态改变,通常最好使用计算属性或取而代之。那解决问题的思路便是在改变的情况下,保证页面的不刷新。后面值的变化,并不会导致浏览器向服务器发出请求,浏览器不发出请求,也就不会刷新页面。 1.vue生命周期2.vue 双向绑定原理3.vue router原理4.vue router动态路由 1.vue 生命周期钩子 showImg(https://segmentfault.com/...

    quietin 评论0 收藏0
  • Vue开发仿旅游站webapp项目总结 (下)

    摘要:用开发仿旅游站项目总结上该说的话,该表明的上篇已经表明了。之后的路由切换不再请求数据是因为组件内容是从内存取了不会再重新创建了,对应的钩子函数不会再执行了。此时,通过新增的生命周期钩子函数以及这个缓存值就实现了我们要的功能了。 用Vue开发仿旅游站webapp项目总结 (上)该说的话,该表明的上篇已经表明了。谢谢上篇评论区一些同学~ 很鼓励我,不过下下篇估计没了,这篇总结完,下下篇可...

    libxd 评论0 收藏0

发表评论

0条评论

k00baa

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<