最近一个需求,须要在Vue项目中加入含有iframe的页面,同时在路由切换的过程当中,要求iframe的内容不会被刷新。一开始使用了Vue自带的keep- alive发现没有用,因而本身研究了一下解决方案。。。。。。vue
要实现对保持iframe页的状态。咱们先搞清楚为何Vue的keep-alive不能凑效。keep-alive原理是把组件里的节点信息保留在了VNode(在内存里),在须要渲染时候从Vnode渲染到真实DOM上。iframe页里的内容并不属于节点的信息,因此使用keep-alive依然会从新渲染iframe内的内容。另外,我也尝试有过想法:若是把整个iframe节点保存起来,而后须要切换时把它渲染到目标节点上,可否实现iframe页不被刷新呢?————也是不可行的,iframe每一次渲染就至关于打开一个新的网页窗口,即便把节点保存下来,在渲染时iframe页仍是刷新的。node
既然保持iframe页里的状态很难实现,在这个时候我想到了一个别的方法。可否在Vue的route-view节点上动点手脚?使得在切换非iframe页的时候使用Vue的路由,当切换iframe页时则使用v-show切换显示与隐藏,使得iframe节点一直不被删除,这样就能保持iframe的状态了。git
咱们简陋的实现一下以上的效果,上代码:github
入口main.js:vue-router
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页的显示与隐藏。bash
上面代码简单的解决了问题,但还有一些地方能够优化:app
咱们先修改router的配置,增长一个属性名iframeComponent,用于标识是否包含iframe,该属性的值是组件文件引用优化
main.js:ui
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>
复制代码
逻辑并不复杂,这里就很少赘述。
你们若是有更好的实现方法,或者我上面还有什么须要更正的错误,欢迎交流。 上面demo的代码放在了我的github上github.com/jmx16449196…