接着之前的笔记,参考源码的布局,增加Breadcrumb组件及screenfull的全屏功能。
Breadcrumb 面包屑导航
需要引入Breadcrumb, BreadcrumbItem。看了下官方文档,有官方组件,先自己弄一下。Elemet-ui中的图标分隔符例子如下:
<el-breadcrumb separator-class="el-icon-arrow-right">
<el-breadcrumb-item :to="{ path: '/' }">首页</el-breadcrumb-item>
<el-breadcrumb-item>活动管理</el-breadcrumb-item>
<el-breadcrumb-item>活动列表</el-breadcrumb-item>
<el-breadcrumb-item>活动详情</el-breadcrumb-item>
</el-breadcrumb>
思路还觉还好,获取当前的路由,然后遍历到顶层,在组件内生成例子中的结构就可以。
在src>components下新建Breadcrumb组件,并在NavBar中使用下。为了样式统一,根据elemet-vue-admin中的样式配置一下。Breadcrumb>index.vue代码如下(下把样子调整出来):
<template>
<div>
<el-breadcrumb class="app-breadcrumb" separator-class="el-icon-arrow-right">
<el-breadcrumb-item :to="{ path: '/' }">首页</el-breadcrumb-item>
<el-breadcrumb-item>活动管理</el-breadcrumb-item>
<el-breadcrumb-item>活动列表</el-breadcrumb-item>
<el-breadcrumb-item>活动详情</el-breadcrumb-item>
</el-breadcrumb>
</div>
</template>
<script>
export default {};
</script>
<style lang="scss" scoped>
.app-breadcrumb.el-breadcrumb {
display: inline-block;
font-size: 14px;
line-height: 50px;
margin-left: 8px;
.no-redirect {
color: #97a8be;
cursor: text;
}
}
</style>
下面需要根据路由生成面包屑导航,同时如果点击面包屑导航的链接时,左侧的Sidebar要选中相应的菜单。
由于route.js中多级菜单都配置了meta属性及name,导致面包屑导航显示不对,修改后左侧sidebar菜单路由中有,页面显示不加载发现是sidebarItem中,判断只有一个子节点处的的代码写错了:
原代码
<item
:icon="onlyOneChild.meta.icon || (item.meta && item.meta.icon)"
:title="item.meta.title"
/>
修改后代码
<item
:icon="onlyOneChild.meta.icon || (item.meta && item.meta.icon)"
:title="onlyOneChild.meta.title"
/>
修改route.js,将每级菜单的meta规范下,并增加table的多级路由嵌套菜单,单独提取出来。在src>router下新建modules下新建table.js文件,内容如下
import Layout from "@/layouts";
const tableRouter = {
path: "/table",
component: Layout,
name: "table",
meta: {
title: "Table表格",
icon: "table",
affix: true,
roles: ["user"]
},
alwaysShow: true,
redirect: "/table/basic-table",
children: [
{
path: "/table/basic-table",
name: "basicTable",
meta: { title: "基础表格", icon: "table", roles: ["user"] },
component: () =>
import(/* webpackChunkName: "table" */ "@/views/tables/basictable.vue")
},
{
path: "/table/page-table",
name: "pageTable",
meta: { title: "分页表格", icon: "table", roles: ["user"] },
redirect: "/table/page-table0",
component: { render: h => h("router-view") },
children: [
{
path: "/table/page-table0",
name: "pageTable0",
meta: { title: "分页表格0", icon: "table", roles: ["user"] },
component: () =>
import(
/* webpackChunkName: "table" */ "@/views/tables/pagetable/pagetable0.vue"
)
},
{
path: "/table/page-table1",
name: "pageTable1",
meta: { title: "分页表格1", icon: "table", roles: ["user"] },
component: () =>
import(
/* webpackChunkName: "table" */ "@/views/tables/pagetable/pagetable1.vue"
)
}
]
}
]
};
export default tableRouter;
最后在router>index.js文件中引入即可。测试页面时发现有些链接在面包屑导航上跳转正常,有的报错。经过排查发现路由中有的缺少了"redirect"属性配置,修改配置例子如下:
{
path: "/form",
// component: { render: h => h("router-view") },
component: Layout,
name: "form",
redirect: "/form/basic-form",
meta: {
title: "Form表单",
icon: "form",
affix: true,
roles: ["admin", "user"]
},
children: [
{
path: "/form/basic-form",
name: "basicForm",
meta: {
title: "Form基础表单1",
icon: "form",
affix: true,
roles: ["admin", "user"]
},
component: () =>
import(/* webpackChunkName: "form" */ "@/views/forms/basicForm.vue")
}
....
]
}
但发现面包屑到导航中的首页返回还是不生效,无法获取首页的redirect属性,没有找到原因。于是修改sr>compones>Breadcrumb>index.vue中的getBreadlist() 方法,在match的时候增加redirect属性,如下所示:
if (!this.isDashboard(first)) {
matched = [
{
path: "/",
meta: { title: "首页" },
redirect: "/dashboard"
}
].concat(matched);
}
Screenfull 全屏功能
此功能依赖screenfull.js这个库,先安装下
yarn add screenfull
参考vue-element-admin中源码,先添加一个div并添加相应的样式。在src>components下新建screenfull组件。组件逻辑需要考虑
- 加载的时候是否全屏,根据是否全屏判断下按钮的SVG加载
- 初始化和销毁的时候监听组件的状态
src>components>Screenfull组件如下,需要注意的是screenfull的API有所变化,"Rename screenfull.enabled
to screenfull.isEnabled
"vue-element-admin的源码中为"screenfull.enabled
<template>
<div>
<svg-icon
:icon-class="isFullscreen ? 'exit-fullscreen' : 'fullscreen'"
@click="click"
/>
</div>
</template>
<script>
import screenfull from "screenfull";
export default {
name: "Screenfull",
data() {
return {
isFullscreen: false
};
},
mounted() {
this.init();
},
beforeDestroy() {
this.destroy();
},
methods: {
click() {
if (!screenfull.isEnabled) {
this.$message({
message: "you browser can not work",
type: "warning"
});
return false;
}
screenfull.toggle();
},
change() {
this.isFullscreen = screenfull.isFullscreen;
},
init() {
if (screenfull.isEnabled) {
screenfull.on("change", this.change);
}
},
destroy() {
if (screenfull.isEnabled) {
screenfull.off("change", this.change);
}
}
}
};
</script>
screenfull.js的API如下:
- .request()
Make an element fullscreen.Accepts a DOM element. Default is . If called with another element than the currently active, it will switch to that if it's a decendant.
If your page is inside an iframe you will need to add a allowfullscreen attribute (+ webkitallowfullscreen and mozallowfullscreen).
Keep in mind that the browser will only enter fullscreen when initiated by user events like click, touch, key.Returns a promise that resolves after the element enters fullscreen."
- .exit()
Brings you out of fullscreen.Returns a promise that resolves after the element exits fullscreen.
- .toggle()
Requests fullscreen if not active, otherwise exits.Returns a promise that resolves after the element enters/exits fullscreen.
- .on(event, function)
Events: 'change' | 'error'.Add a listener for when the browser switches in and out of fullscreen or when there is an error.
- .off(event, function)
Remove a previously registered event listener.
- .onchange(function)
Alias for .on('change', function)
- .onerror(function)
Alias for .on('error', function)
- .isFullscreen
Returns a boolean whether fullscreen is active.
- .element
Returns the element currently in fullscreen, otherwise null.
- .isEnabled
Returns a boolean whether you are allowed to enter fullscreen. If your page is inside an iframe you will need to add a allow fullscreen attribute (+ webkitallowfullscreen and mozallowfullscreen).
- .raw
Exposes the raw properties (prefixed if needed) used internally: requestFullscreen, exitFullscreen, fullscreenElement, fullscreenEnabled, fullscreenchange, fullscreenerror