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

前言

最近一个需求,需要在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: '<div>Index</div>' }

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')

根组件:


<template>

  <div id="app">

    <div class="nav">

      <router-link class="router" to="/f1">Go to F1</router-link>

      <router-link class="router" to="/f2">Go to F2</router-link>

      <router-link class="router" to="/index">Go to Index</router-link>

    </div>



    <keep-alive>

      <!-- Vue的路由 -->

      <router-view></router-view>

    </keep-alive>



    <!-- iframe页面 -->

    <f1 v-show="$route.path == '/f1'"></f1>

    <f2 v-show="$route.path == '/f2'"></f2>

  </div>

</template>

<script>

import F1 from './components/f1';

import F2 from './components/f2';

export default {

  name: 'app',

  components: {

    F1,

    F2

  },



}

</script>

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

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

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

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

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

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

    3. 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: '<div>Index</div>' }

  }

]

const router = new VueRouter({

  routes // (缩写)相当于 routes: routes

});

new Vue({

  render: h => h(App),

  router

}).$mount('#app')

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


<template>

    <div>

        <!-- Vue的router-view -->

        <keep-alive>

            <router-view></router-view>

        </keep-alive>

        <!-- iframe页 -->

        <component

            v-for="item in hasOpenComponentsArr"

            :key="item.name"

            :is="item.name"

            v-show="$route.path === item.path"

        ></component>

    </div>

</template>

<script>

import Vue from 'vue/dist/vue.js'

export default {

    created() {

        // 设置iframe页的数组对象

        const componentsArr = this.getComponentsArr();

        componentsArr.forEach((item) => {

            Vue.component(item.name, item.component);

        });

        this.componentsArr = componentsArr;

        // 判断当前路由是否iframe页

        this.isOpenIframePage();

    },

    data() {

        return {

            componentsArr: [] // 含有iframe的页面

        }

    },

    watch: {

        $route() {

            // 判断当前路由是否iframe页

            this.isOpenIframePage();

        }

    },

    computed: {

        // 实现懒加载,只渲染已经打开过(hasOpen:true)的iframe页

        hasOpenComponentsArr() {

            return this.componentsArr.filter(item => item.hasOpen);

        }

    },

    methods: {

        // 根据当前路由设置hasOpen

        isOpenIframePage() {

            const target = this.componentsArr.find(item => {

                return item.path === this.$route.path

            });

            if (target && !target.hasOpen) {

                target.hasOpen = true;

            }

        },

        // 遍历路由的所有页面,把含有iframeComponent标识的收集起来

        getComponentsArr() {

            const router = this.$router;

            const routes = router.options.routes;

            const iframeArr = routes.filter(item => item.iframeComponent);



            return iframeArr.map((item) => {

                const name = item.name || item.path.replace('/', '');

                return {

                    name: name,

                    path: item.path,

                    hasOpen: false, // 是否打开过,默认false

                    component: item.iframeComponent // 组件文件的引用

                };

            });

        }

    }

}

</script>

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

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

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

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

结语

大家如果有更好的实现方法,或者我上面还有什么需要更正的错误,欢迎交流。

上面demo的代码放在了个人github上https://github.com/jmx164491960/vue-iframe-demo

wc(1).jpg
更多前端技术分享请订阅微信公众号“前端技术干货分享圈”
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容