fix: i18n problem of dynamic route; 🐛

修复:动态路由的国际化问题;
This commit is contained in:
iczer 2020-08-04 11:30:44 +08:00
parent d7eba7e2b3
commit b796a4c9e4
5 changed files with 34 additions and 17 deletions

View File

@ -34,6 +34,7 @@
import Menu from 'ant-design-vue/es/menu' import Menu from 'ant-design-vue/es/menu'
import Icon from 'ant-design-vue/es/icon' import Icon from 'ant-design-vue/es/icon'
import fastEqual from 'fast-deep-equal' import fastEqual from 'fast-deep-equal'
import {getI18nKey} from '@/utils/routerUtil'
const {Item, SubMenu} = Menu const {Item, SubMenu} = Menu
@ -115,7 +116,7 @@ export default {
h('router-link', {props: {to: menu.fullPath}}, h('router-link', {props: {to: menu.fullPath}},
[ [
this.renderIcon(h, menu.meta ? menu.meta.icon : 'none'), this.renderIcon(h, menu.meta ? menu.meta.icon : 'none'),
h('span', [this.$t(menu.fullPath.substring(1).replace(new RegExp('/', 'g'), '.') + '.name')]) h('span', [this.$t(getI18nKey(menu.fullPath))])
] ]
) )
] ]
@ -126,7 +127,7 @@ export default {
let subItem = [h('span', {slot: 'title'}, let subItem = [h('span', {slot: 'title'},
[ [
this.renderIcon(h, menu.meta ? menu.meta.icon : 'none'), this.renderIcon(h, menu.meta ? menu.meta.icon : 'none'),
h('span', [this.$t(menu.fullPath.substring(1).replace(new RegExp('/', 'g'), '.') + '.name')]) h('span', [this.$t(getI18nKey(menu.fullPath))])
] ]
)] )]
let itemArr = [] let itemArr = []

View File

@ -22,6 +22,8 @@
<script> <script>
import PageHeader from '@/components/page/header/PageHeader' import PageHeader from '@/components/page/header/PageHeader'
import {mapState, mapMutations} from 'vuex' import {mapState, mapMutations} from 'vuex'
import {getI18nKey} from '@/utils/routerUtil'
export default { export default {
name: 'PageLayout', name: 'PageLayout',
components: {PageHeader}, components: {PageHeader},
@ -64,7 +66,8 @@ export default {
return this.title || this.$t(pageTitle) || this.routeName return this.title || this.$t(pageTitle) || this.routeName
}, },
routeName() { routeName() {
return this.$t(this.$route.path.substring(1).replace(new RegExp('/', 'g'), '.') + '.name') const route = this.$route
return this.$t(getI18nKey(route.matched[route.matched.length - 1].path))
}, },
breadcrumb() { breadcrumb() {
let page = this.page let page = this.page
@ -89,9 +92,8 @@ export default {
let routes = this.$route.matched let routes = this.$route.matched
let breadcrumb = [] let breadcrumb = []
routes.forEach(route => { routes.forEach(route => {
let path = route.path const path = route.path.length === 0 ? '/home' : route.path
let key = path.length == 0 ? '/home' : path breadcrumb.push(this.$t(getI18nKey(path)))
breadcrumb.push(this.$t(key.substring(1).replace(new RegExp('/', 'g'), '.') + '.name'))
}) })
return breadcrumb return breadcrumb
}, },

View File

@ -12,7 +12,7 @@
@contextmenu="onContextmenu" @contextmenu="onContextmenu"
> >
<a-tab-pane :key="page.fullPath" v-for="page in pageList"> <a-tab-pane :key="page.fullPath" v-for="page in pageList">
<span slot="tab" :pagekey="page.fullPath">{{pageName(page.path)}}</span> <span slot="tab" :pagekey="page.fullPath">{{pageName(page)}}</span>
</a-tab-pane> </a-tab-pane>
</a-tabs> </a-tabs>
<div class="tabs-view-content" :style="`margin-top: ${multiPage ? -24 : 0}px`"> <div class="tabs-view-content" :style="`margin-top: ${multiPage ? -24 : 0}px`">
@ -31,6 +31,7 @@ import AdminLayout from '@/layouts/AdminLayout'
import Contextmenu from '@/components/menu/Contextmenu' import Contextmenu from '@/components/menu/Contextmenu'
import PageToggleTransition from '@/components/transition/PageToggleTransition' import PageToggleTransition from '@/components/transition/PageToggleTransition'
import {mapState, mapMutations} from 'vuex' import {mapState, mapMutations} from 'vuex'
import {getI18nKey} from '@/utils/routerUtil'
export default { export default {
name: 'TabsView', name: 'TabsView',
@ -176,8 +177,8 @@ export default {
this.setDustbins(this.dustbins.filter(item => item !== componentName)) this.setDustbins(this.dustbins.filter(item => item !== componentName))
} }
}, },
pageName(path) { pageName(page) {
return this.$t(path.substring(1).replace(new RegExp('/', 'g'), '.') + '.name') return this.$t(getI18nKey(page.matched[page.matched.length - 1].path))
}, },
...mapMutations('setting', ['setDustbins', 'correctPageMinHeight']) ...mapMutations('setting', ['setDustbins', 'correctPageMinHeight'])
} }

View File

@ -2,6 +2,7 @@ import Vue from 'vue'
import VueI18n from 'vue-i18n' import VueI18n from 'vue-i18n'
import routesI18n from '@/router/i18n' import routesI18n from '@/router/i18n'
import './Objects' import './Objects'
import {getI18nKey} from '@/utils/routerUtil'
/** /**
* 创建 i18n 配置 * 创建 i18n 配置
@ -23,16 +24,17 @@ function initI18n(locale, fallback) {
/** /**
* 根据 router options 配置生成 国际化语言 * 根据 router options 配置生成 国际化语言
* @param lang * @param lang
* @param options * @param routes
* @param valueKey * @param valueKey
* @returns {*} * @returns {*}
*/ */
function generateI18n(lang, options, valueKey) { function generateI18n(lang, routes, valueKey) {
options.forEach(menu => { routes.forEach(route => {
let keys = menu.fullPath.substring(1).split('/').concat('name') let keys = getI18nKey(route.fullPath).split('.')
lang.assignProps(keys, menu[valueKey]) let value = valueKey === 'path' ? route[valueKey].split('/').filter(item => !item.startsWith(':') && item != '').join('.') : route[valueKey]
if (menu.children) { lang.assignProps(keys, value)
generateI18n(lang, menu.children, valueKey) if (route.children) {
generateI18n(lang, route.children, valueKey)
} }
}) })
return lang return lang

View File

@ -185,4 +185,15 @@ function formatAuthority(routes) {
}) })
} }
export {parseRoutes, loadRoutes, loginGuard, authorityGuard, formatAuthority} /**
* 从路由 path 解析 i18n key
* @param path
* @returns {*}
*/
function getI18nKey(path) {
const keys = path.split('/').filter(item => !item.startsWith(':') && item != '')
keys.push('name')
return keys.join('.')
}
export {parseRoutes, loadRoutes, loginGuard, authorityGuard, formatAuthority, getI18nKey}