feat: 暴露语言变化事件 & layout的title支持配置国际化 (#255)

* feat: layout的title支持国际化

* feat: locale插件暴漏onLocaleChange事件

* chore: change docs

* fix: 去掉无用代码

---------

Co-authored-by: winixt <haizekuo@gmail.com>
This commit is contained in:
听海 2024-10-23 18:58:36 +08:00 committed by GitHub
parent db358576e8
commit a303e171c9
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
9 changed files with 87 additions and 50 deletions

View File

@ -11,7 +11,7 @@
1. 添加 Webpack 构建依赖包: `npm i @fesjs/builder-webpack -D` 1. 添加 Webpack 构建依赖包: `npm i @fesjs/builder-webpack -D`
2. 如果设置了 `publicPath: './'`,请更改为 `publicPath: ''` 2. 如果设置了 `publicPath: './'`,请更改为 `publicPath: ''`
3. 如果有,`public/index.html` 文件挪到项目根目录,移除 [html-webpack-plugin](https://github.com/jantimon/html-webpack-plugin) 相关配置,具体模版变量使用请查看[HTML 模版](../guide/template.html)。 3. 将 html 模版文件从 `public/index.html` 文件挪到项目根目录,移除 [html-webpack-plugin](https://github.com/jantimon/html-webpack-plugin) 相关配置,具体模版变量使用请查看[HTML 模版](../guide/template.html)。
## 换成 Vite ## 换成 Vite

View File

@ -233,7 +233,7 @@ export function layout(layoutConfig, { initialState }) {
- **类型**`String` - **类型**`String`
- **默认值**:默认为 [编译时配置 title](../../../reference/config/#title) - **默认值**:默认为 [编译时配置 title](../../../reference/config/#title)
- **详情**:产品名。 - **详情**:产品名,当配置为"$"开头时,开启国际化,使用`$`后面的内容去匹配语言设置
### logo ### logo

View File

@ -69,11 +69,9 @@ export default {
想了解更多语言信息配置、匹配规则,请参考 [Vue I18n](https://vue-i18n.intlify.dev/guide/essentials/syntax.html) 文档。 想了解更多语言信息配置、匹配规则,请参考 [Vue I18n](https://vue-i18n.intlify.dev/guide/essentials/syntax.html) 文档。
### 多层配置 ### 多层配置
如果国际化内容较多,希望模块化配置,则可以这样:
``` 如果国际化内容较多,希望模块化配置,则可以这样:
src src
├── locales ├── locales
│ ├── zh-CN.js │ ├── zh-CN.js
@ -84,6 +82,7 @@ src
└── pages └── pages
│ └── index.vue │ └── index.vue
└── app.js └── app.js
``` ```
插件会把相同语言的配置合并在一起! 插件会把相同语言的配置合并在一起!
@ -145,7 +144,19 @@ export default {
### 运行时配置 ### 运行时配置
暂无。 ## onLocaleChange
当语言环境发生变化时,会触发此函数。
```js
import { defineRuntimeConfig } from '@fesjs/fes';
export default defineRuntimeConfig({
onLocaleChange: ({ t, locale }) => {
}
});
```
## API ## API

View File

@ -11,8 +11,8 @@
> >
<div class="layout-logo"> <div class="layout-logo">
<img v-if="logo" :src="logo" class="logo-img"> <img v-if="logo" :src="logo" class="logo-img">
<div v-if="title" class="logo-name"> <div v-if="transformedTitle" class="logo-name">
{{ title }} {{ transformedTitle }}
</div> </div>
</div> </div>
<LayoutMenu <LayoutMenu
@ -58,8 +58,8 @@
<div> <div>
<div class="layout-logo"> <div class="layout-logo">
<img v-if="logo" :src="logo" class="logo-img"> <img v-if="logo" :src="logo" class="logo-img">
<div v-if="title" class="logo-name"> <div v-if="transformedTitle" class="logo-name">
{{ title }} {{ transformedTitle }}
</div> </div>
</div> </div>
<LayoutMenu <LayoutMenu
@ -98,8 +98,8 @@
<FHeader ref="headerRef" class="layout-header" :inverted="theme === 'dark'" :fixed="currentFixedHeaderRef"> <FHeader ref="headerRef" class="layout-header" :inverted="theme === 'dark'" :fixed="currentFixedHeaderRef">
<div class="layout-logo"> <div class="layout-logo">
<img v-if="logo" :src="logo" class="logo-img"> <img v-if="logo" :src="logo" class="logo-img">
<div v-if="title" class="logo-name"> <div v-if="transformedTitle" class="logo-name">
{{ title }} {{ transformedTitle }}
</div> </div>
</div> </div>
<LayoutMenu <LayoutMenu
@ -131,8 +131,8 @@
<FHeader ref="headerRef" class="layout-header" :inverted="theme === 'dark'" :fixed="currentFixedHeaderRef"> <FHeader ref="headerRef" class="layout-header" :inverted="theme === 'dark'" :fixed="currentFixedHeaderRef">
<div class="layout-logo"> <div class="layout-logo">
<img v-if="logo" :src="logo" class="logo-img"> <img v-if="logo" :src="logo" class="logo-img">
<div v-if="title" class="logo-name"> <div v-if="transformedTitle" class="logo-name">
{{ title }} {{ transformedTitle }}
</div> </div>
</div> </div>
<LayoutMenu <LayoutMenu
@ -184,8 +184,8 @@
<FHeader ref="headerRef" class="layout-header" :fixed="currentFixedHeaderRef" :inverted="theme === 'dark'"> <FHeader ref="headerRef" class="layout-header" :fixed="currentFixedHeaderRef" :inverted="theme === 'dark'">
<div class="layout-logo"> <div class="layout-logo">
<img v-if="logo" :src="logo" class="logo-img"> <img v-if="logo" :src="logo" class="logo-img">
<div v-if="title" class="logo-name"> <div v-if="transformedTitle" class="logo-name">
{{ title }} {{ transformedTitle }}
</div> </div>
</div> </div>
<div class="layout-header-custom"> <div class="layout-header-custom">
@ -231,6 +231,7 @@ import { FAside, FFooter, FHeader, FLayout, FMain } from '@fesjs/fes-design';
import { computed, nextTick, ref, watch } from 'vue'; import { computed, nextTick, ref, watch } from 'vue';
import defaultLogo from '../assets/logo.png'; import defaultLogo from '../assets/logo.png';
import { flatNodes } from '../helpers/utils'; import { flatNodes } from '../helpers/utils';
import { transTitle } from '../helpers/pluginLocale';
import LayoutMenu from './Menu.vue'; import LayoutMenu from './Menu.vue';
import MultiTabProvider from './MultiTabProvider.vue'; import MultiTabProvider from './MultiTabProvider.vue';
@ -302,6 +303,10 @@ export default {
const route = useRoute(); const route = useRoute();
const router = useRouter(); const router = useRouter();
const transformedTitle = computed(() => {
return transTitle(props.title);
});
const currentNavigation = computed(() => { const currentNavigation = computed(() => {
if (route.meta.layout && route.meta.layout.navigation !== undefined) { if (route.meta.layout && route.meta.layout.navigation !== undefined) {
return route.meta.layout.navigation; return route.meta.layout.navigation;
@ -397,6 +402,7 @@ export default {
currentNavigation, currentNavigation,
rootMenus, rootMenus,
activeSubMenus, activeSubMenus,
transformedTitle,
}; };
}, },
}; };

View File

@ -24,6 +24,8 @@ export default (api) => {
api.addRuntimePluginKey(() => 'locale'); api.addRuntimePluginKey(() => 'locale');
api.addRuntimePluginKey(() => 'onLocaleChange');
const absoluteFilePath = join(namespace, 'core.js'); const absoluteFilePath = join(namespace, 'core.js');
const absRuntimeFilePath = join(namespace, 'runtime.js'); const absRuntimeFilePath = join(namespace, 'runtime.js');

View File

@ -8,7 +8,7 @@
import { isRef, unref } from 'vue'; import { isRef, unref } from 'vue';
import { createI18n, useI18n } from '{{{ VUE_I18N_PATH }}}'; import { createI18n, useI18n } from '{{{ VUE_I18N_PATH }}}';
import locales from './locales' import locales from './locales'
import { plugin, ApplyPluginsType } from '@@/core/coreExports';
const defaultOptions = {{{REPLACE_DEFAULT_OPTIONS}}}; const defaultOptions = {{{REPLACE_DEFAULT_OPTIONS}}};
@ -51,6 +51,8 @@ const i18n = createI18n({
messages, messages,
}); });
const t = i18n.global.t;
window.localStorage.setItem("fes_locale", unref(i18n.global.locale)); window.localStorage.setItem("fes_locale", unref(i18n.global.locale));
const setLocale = ({ locale }) => { const setLocale = ({ locale }) => {
if (isRef(i18n.global.locale)) { if (isRef(i18n.global.locale)) {
@ -59,6 +61,11 @@ const setLocale = ({ locale }) => {
i18n.global.locale = locale; i18n.global.locale = locale;
} }
window.localStorage.setItem("fes_locale", locale); window.localStorage.setItem("fes_locale", locale);
plugin.applyPlugins({
key: 'onLocaleChange',
type: ApplyPluginsType.event,
args: { i18n, t, locale: unref(i18n.global.locale) },
});
}; };
const getLocale = () => { const getLocale = () => {
@ -82,17 +89,20 @@ const getAllLocales = () => {
const install = (app) => { const install = (app) => {
app.use(i18n); app.use(i18n);
plugin.applyPlugins({
key: 'onLocaleChange',
type: ApplyPluginsType.event,
args: { i18n, t, locale: unref(i18n.global.locale) },
});
}; };
const t = i18n.global.t;
const locale = { const locale = {
setLocale, setLocale,
getLocale, getLocale,
addLocale, addLocale,
getAllLocales, getAllLocales,
messages, messages,
t t,
}; };
export { useI18n, locale, install }; export { useI18n, locale, install };

View File

@ -21,4 +21,7 @@ declare module '@fesjs/fes' {
} }
| false; | false;
} }
interface PluginRuntimeConfig {
onLocaleChange: (params: { t: VueI18n['t']; locale: string }) => void;
}
} }

View File

@ -33,7 +33,7 @@ export default defineBuildConfig({
}, },
}, },
layout: { layout: {
title: 'Fes.js', title: '$home',
footer: 'Created by MumbleFE', footer: 'Created by MumbleFE',
multiTabs: true, multiTabs: true,
navigation: 'mixin', navigation: 'mixin',

View File

@ -3,6 +3,11 @@ import { ref, watch } from 'vue';
import PageLoading from '@/components/pageLoading.vue'; import PageLoading from '@/components/pageLoading.vue';
import UserCenter from '@/components/userCenter.vue'; import UserCenter from '@/components/userCenter.vue';
export function onLocaleChange({ locale, t }) {
//
console.log(locale, t);
}
export const beforeRender = { export const beforeRender = {
loading: <PageLoading />, loading: <PageLoading />,
action() { action() {