diff --git a/docs/.vuepress/config.js b/docs/.vuepress/config.js index 7a9d3f1..156ab7d 100644 --- a/docs/.vuepress/config.js +++ b/docs/.vuepress/config.js @@ -36,7 +36,7 @@ module.exports = { title: '进阶', collapsable: false, children: [ - '/advance/i18n', '/advance/async', '/advance/authority', '/advance/login', '/advance/guard', '/advance/interceptors', '/advance/skill' + '/advance/i18n', '/advance/async', '/advance/authority', '/advance/login', '/advance/guard', '/advance/interceptors' ] }, { diff --git a/docs/advance/async.md b/docs/advance/async.md index 0e2ff8a..750f6d7 100644 --- a/docs/advance/async.md +++ b/docs/advance/async.md @@ -24,7 +24,7 @@ module.exports = { } ``` ### 注册路由组件 -基础路由组件包含路由基本配置和对应的视图组件,我们统一在 `/router/router.map.js` 文件中注册它们。它和正常的路由配置基本无异,相当于把完整的路由拆分成单个的路由配置进行注册,为后面的路由动态配置打好基础。 +基础路由组件包含路由基本配置和对应的视图组件,我们统一在 `/router/async/router.map.js` 文件中注册它们。它和正常的路由配置基本无异,相当于把完整的路由拆分成单个的路由配置进行注册,为后面的路由动态配置打好基础。 一个单独的路由组件注册示例如下: ```jsx registerName: { //路由组件注册名称,唯一标识 @@ -108,7 +108,7 @@ export default routerMap ``` ::: ### 配置基本路由 -如果没有任何路由,你的应用是无法访问的,所以我们需要在本地配置一些基本的路由,比如登录页、404、403 等。你可以在 `/router/config.async.js` 文件中配置一些本地必要的路由。如下: +如果没有任何路由,你的应用是无法访问的,所以我们需要在本地配置一些基本的路由,比如登录页、404、403 等。你可以在 `/router/async/config.async.js` 文件中配置一些本地必要的路由。如下: ```js const routesConfig = [ 'login', //匹配 router.map.js 中注册的 registerName = login 的路由 @@ -163,25 +163,25 @@ export default options 那么我们就需要先从后端服务获取异步路由配置,后端返回的异步路由配置 `routesConfig` 是一个异步路由配置数组, 应当如下格式: ```jsx [{ - router: 'root', //匹配 /router/router.map.js 中注册名 registerName = root 的路由 + router: 'root', //匹配 router.map.js 中注册名 registerName = root 的路由 children: [ //root 路由的子路由配置 { - router: 'dashboard', //匹配 /router/router.map.js 中注册名 registerName = dashboard 的路由 + router: 'dashboard', //匹配 router.map.js 中注册名 registerName = dashboard 的路由 children: ['workplace', 'analysis'], //dashboard 路由的子路由配置,依次匹配 registerName 为 workplace 和 analysis 的路由 }, { - router: 'form', //匹配 /router/router.map.js 中注册名 registerName = form 的路由 + router: 'form', //匹配 router.map.js 中注册名 registerName = form 的路由 children: [ //form 路由的子路由配置 - 'basicForm', //匹配 /router/router.map.js 中注册名 registerName = basicForm 的路由 - 'stepForm', //匹配 /router/router.map.js 中注册名 registerName = stepForm 的路由 + 'basicForm', //匹配 router.map.js 中注册名 registerName = basicForm 的路由 + 'stepForm', //匹配 router.map.js 中注册名 registerName = stepForm 的路由 { - router: 'advanceForm', //匹配 /router/router.map.js 中注册名 registerName = advanceForm 的路由 + router: 'advanceForm', //匹配 router.map.js 中注册名 registerName = advanceForm 的路由 path: 'advance' //重写 advanceForm 路由的 path 属性 } ] }, { - router: 'basicForm', //匹配 /router/router.map.js 中注册名 registerName = basicForm 的路由 + router: 'basicForm', //匹配 router.map.js 中注册名 registerName = basicForm 的路由 name: '验权表单', //重写 basicForm 路由的 name 属性 icon: 'file-excel', //重写 basicForm 路由的 icon 属性 authority: 'form' //重写 basicForm 路由的 authority 属性 @@ -203,7 +203,7 @@ getRoutesConfig().then(result => { 至此,异步路由的加载就完成了,你可以访问异步加载的路由了。 :::tip 上面获取异步路由的代码,在 /pages/login/Login.vue 文件中可以找到。 -loadRoutes 方法会合并 /router/config.async.js 文件中配置的基本路由。 +loadRoutes 方法会合并 /router/async/config.async.js 文件中配置的基本路由。 ::: :::details 点击查看 loadRoutes 的详细代码 ```js diff --git a/docs/advance/guard.md b/docs/advance/guard.md index a43508c..274abfe 100644 --- a/docs/advance/guard.md +++ b/docs/advance/guard.md @@ -1,7 +1,109 @@ --- -title: 导航守卫 +title: 路由守卫 lang: zn-CN --- -# 导航守卫 +# 路由守卫 +Vue Antd Admin 使用 vue-router 实现路由导航功能,因此可以为路由配置一些守卫。 +我们统一把导航守卫配置在 router/guards.js 文件中。 -### 作者还没来得及编辑该页面,如果你感兴趣,可以点击下方链接,帮助作者完善此页 +## 前置守卫 +Vue Antd Admin 为每个前置导航守卫函数注入 to,from,next,options 四个参数: +* `to: Route`: 即将要进入的目标[路由对象](https://router.vuejs.org/zh/api/#%E8%B7%AF%E7%94%B1%E5%AF%B9%E8%B1%A1) +* `from: Route`: 当前导航正要离开的路由对象 +* `next: Function`: 一定要调用该方法来 resolve 这个钩子。执行效果依赖 next 方法的调用参数。详情查看 [Vue Router #导航守卫](https://router.vuejs.org/zh/guide/advanced/navigation-guards.html) +* `options: Object`: 应用配置,包含: {router, i18n, store, message},可根据需要扩展。 +如下,是登录拦截导航守卫的定义 +```js +const loginGuard = (to, from, next, options) => { + const {message} = options + if (!loginIgnore.includes(to) && !checkAuthorization()) { + message.warning('登录已失效,请重新登录') + next({path: '/login'}) + } else { + next() + } +} +``` + +## 后置守卫 +你也可以定义后置导航守卫,Vue Antd Admin 为每个后置导航函数注入 to,from,options 三个参数: +* `to: Route`: 即将要进入的目标[路由对象](https://router.vuejs.org/zh/api/#%E8%B7%AF%E7%94%B1%E5%AF%B9%E8%B1%A1) +* `from: Route`: 当前导航正要离开的路由对象 +* `options: Object`: 应用配置,包含: {router, i18n, store, message},可根据需要扩展。 +如下,是一个后置导航守卫的定义 +```js +const afterGuard = (to, from, options) => { + const {store, message} = options + // 做些什么 + message.info('do something') +} +``` + +## 导出守卫配置 +定义好导航守卫后,只需按照类别在 guard.js 中导出即可。分为两类,`前置守卫`和`后置守卫`。如下: +```js +export default { + beforeEach: [loginGuard, authorityGuard], + afterEach: [afterGuard] +} +``` + +:::details 点击查看完整的导航守卫配置 +```js +import {loginIgnore} from '@/router/index' +import {checkAuthorization} from '@/utils/request' + +/** + * 登录守卫 + * @param to + * @param form + * @param next + * @param options + */ +const loginGuard = (to, from, next, options) => { + const {message} = options + if (!loginIgnore.includes(to) && !checkAuthorization()) { + message.warning('登录已失效,请重新登录') + next({path: '/login'}) + } else { + next() + } +} + +/** + * 权限守卫 + * @param to + * @param form + * @param next + * @param options + */ +const authorityGuard = (to, from, next, options) => { + const {store, message} = options + const permissions = store.getters['account/permissions'] + const roles = store.getters['account/roles'] + if (!hasAuthority(to, permissions, roles)) { + message.warning(`对不起,您无权访问页面: ${to.fullPath},请联系管理员`) + next({path: '/403'}) + } else { + next() + } +} + +/** + * 后置守卫 + * @param to + * @param form + * @param options + */ +const afterGuard = (to, from, options) => { + const {store, message} = options + // 做些什么 + message.info('do something') +} + +export default { + beforeEach: [loginGuard, authorityGuard], + afterEach: [afterGuard] +} +``` +::: \ No newline at end of file diff --git a/docs/advance/interceptors.md b/docs/advance/interceptors.md index 843bc6a..f34fc94 100644 --- a/docs/advance/interceptors.md +++ b/docs/advance/interceptors.md @@ -3,5 +3,129 @@ title: 拦截器配置 lang: zn-CN --- # 拦截器配置 +Vue Antd Admin 基于 aixos 封装了 http 通信功能,我们可以为 http 请求响应配置一些拦截器。拦截器统一配置在 /utils/axios-interceptors.js 文件中。 +## 请求拦截器 +你可以为每个请求拦截器配置 `onFulfilled` 或 `onRejected` 两个钩子函数。 +### onFulfilled +我们会为 onFulfilled 钩子函数注入 config 和 options 两个参数: +* `config: AxiosRequestConfig`: axios 请求配置,详情参考 [axios 请求配置](http://www.axios-js.com/zh-cn/docs/#%E8%AF%B7%E6%B1%82%E9%85%8D%E7%BD%AE) +* `options: Object`: 应用配置,包含: {router, i18n, store, message},可根据需要扩展。 -### 作者还没来得及编辑该页面,如果你感兴趣,可以点击下方链接,帮助作者完善此页 +### onRejected +我们会为 onFulfilled 钩子函数注入 error 和 options 两个参数: +* `error: Error`: axios 请求错误对象 +* `options: Object`: 应用配置,包含: {router, i18n, store, message},可根据需要扩展。 + +如下,为一个完整的请求拦截器配置: +```js +const tokenCheck = { + // 发送请求之前做些什么 + onFulfilled(config, options) { + const {message} = options + const {url, xsrfCookieName} = config + if (url.indexOf('login') === -1 && xsrfCookieName && !Cookie.get(xsrfCookieName)) { + message.warning('认证 token 已过期,请重新登录') + } + return config + }, + // 请求出错时做点什么 + onRejected(error, options) { + const {message} = options + message.error(error.message) + return Promise.reject(error) + } +} +``` +## 响应拦截器 +响应拦截器也同样可以配置 `onFulfilled` 或 `onRejected` 两个钩子函数。 +### onFulfilled +我们会为 onFulfilled 钩子函数注入 response 和 options 两个参数: +* `response: AxiosResponse`: axios 响应对象,详情参考 [axios 响应对象](http://www.axios-js.com/zh-cn/docs/#%E5%93%8D%E5%BA%94%E7%BB%93%E6%9E%84) +* `options: Object`: 应用配置,包含: {router, i18n, store, message},可根据需要扩展。 + +### onRejected +我们会为 onFulfilled 钩子函数注入 error 和 options 两个参数: +* `error: Error`: axios 请求错误对象 +* `options: Object`: 应用配置,包含: {router, i18n, store, message},可根据需要扩展。 + +如下,为一个完整的响应拦截器配置: +```js +const resp401 = { + // 响应数据之前做点什么 + onFulfilled(response, options) { + const {message} = options + if (response.status === 401) { + message.error('无此接口权限') + } + return response + }, + // 响应出错时做点什么 + onRejected(error, options) { + const {message} = options + if (response.status === 401) { + message.error('无此接口权限') + } + return Promise.reject(error) + } +} +``` +## 导出拦截器 +定义好拦截器后,只需在 axios-interceptors.js 文件中导出即可。分为两类,`请求拦截器`和`响应拦截器`。如下: +```js +export default { + request: [tokenCheck], // 请求拦截 + response: [resp401] // 响应拦截 +} +``` + +:::details 点击查看完整的拦截器配置示例 +```js +import Cookie from 'js-cookie' +// 401拦截 +const resp401 = { + onFulfilled(response, options) { + const {message} = options + if (response.status === 401) { + message.error('无此接口权限') + } + return response + }, + onRejected(error, options) { + const {message} = options + message.error(error.message) + return Promise.reject(error) + } +} + +const resp403 = { + onFulfilled(response, options) { + const {message} = options + if (response.status === 403) { + message.error(`请求被拒绝`) + } + return response + } +} + +const reqCommon = { + onFulfilled(config, options) { + const {message} = options + const {url, xsrfCookieName} = config + if (url.indexOf('login') === -1 && xsrfCookieName && !Cookie.get(xsrfCookieName)) { + message.warning('认证 token 已过期,请重新登录') + } + return config + }, + onRejected(error, options) { + const {message} = options + message.error(error.message) + return Promise.reject(error) + } +} + +export default { + request: [reqCommon], // 请求拦截 + response: [resp401, resp403] // 响应拦截 +} +``` +::: \ No newline at end of file diff --git a/docs/advance/login.md b/docs/advance/login.md index e0ae3e9..104caa1 100644 --- a/docs/advance/login.md +++ b/docs/advance/login.md @@ -3,5 +3,74 @@ title: 登录认证 lang: zn-CN --- # 登录认证 +Vue Antd Admin 使用 js-cookie.js 管理用户的 token,结合 axios 配置,可以为每个请求头加上 token 信息。 -### 作者还没来得及编辑该页面,如果你感兴趣,可以点击下方链接,帮助作者完善此页 +## token名称 +后端系统通常会从请求 header 中获取用户的 token,因此我们需要配置好 token 名称,好让后端能正确的识别到用户 token。 +Vue Antd Admin 默认token 名称为 `Authorization`,你可以在 /utils/request.js 中修改它。 +```js{5} +import axios from 'axios' +import Cookie from 'js-cookie' + +// 跨域认证信息 header 名 +const xsrfHeaderName = 'Authorization' +... +``` +## token 设置 +调用登录接口后拿到用户的 token 和 token 过期时间(如无过期时间,可忽略),并使用 /utils/request.js #setAuthorization 方法保存token。 +```js{5} +import {setAuthorization} from '@/utils/request' + +login(name, password).then(res => { + const {token, expireAt} = res.data + setAuthorization({token, expireAt: new Date(expireAt)}) +}) +``` +## token 校验 +Vue Antd Admin 默认添加了登录导航守卫,如检查到本地cookie 中不包含 token 信息,则会拦截跳转至登录页。你可以在 /router/index.js 中配置 +不需要登录拦截的路由 +```js +// 不需要登录拦截的路由配置 +const loginIgnore = { + names: ['404', '403'], //根据路由名称匹配 + paths: ['/login'], //根据路由fullPath匹配 + /** + * 判断路由是否包含在该配置中 + * @param route vue-router 的 route 对象 + * @returns {boolean} + */ + includes(route) { + return this.names.includes(route.name) || this.paths.includes(route.path) + } +} +``` +或者在 /router/guards.js 中移出登录守卫 +```diff +... +export default { +- beforeEach: [loginGuard, authorityGuard, redirectGuard], ++ beforeEach: [authorityGuard, redirectGuard], + afterEach: [] +} +``` +## Api +### setAuthorization(auth, authType) +来源:/utils/request.js +该方法用于保存用户 token,接收两个参数: +* **auth** +认证信息,包含 token、expireAt 等认证数据。 +* **authType** +认证类型,默认为 `AUTH_TYPE.BEARER`(AUTH_TYPE.BEARER 默认会给token 加上 Bearer 识别前缀),可根据自己的认证类型自行扩展。 + +### checkAuthorization(authType) +该方法用于校验用户 token 是否过期,接收一个参数: +* **authType** +认证类型,默认为 `AUTH_TYPE.BEARER`。 + +### removeAuthorization(authType) +该方法用于移出用户本地存储的 token,接收一个参数: +* **authType** +认证类型,默认为 `AUTH_TYPE.BEARER`。 +:::tip +以上 Api 均可在 /utils/request.js 文件中找到。 +::: \ No newline at end of file diff --git a/src/components/checkbox/ImgCheckbox.vue b/src/components/checkbox/ImgCheckbox.vue index b59dd22..ca06bdd 100644 --- a/src/components/checkbox/ImgCheckbox.vue +++ b/src/components/checkbox/ImgCheckbox.vue @@ -1,5 +1,5 @@