如何让i18n完全加载完毕,其他.ts文件再去调用?

在枚举中使用多语言,枚举对象无法识别多语言。
文件结构如下所示
image.png

i18n.ts中的写法为

// 注册i18n实例并引入语言文件
let i18n: VueI18n = new VueI18n({
    locale: "zh_CN",
    messages: null
});

export async function setLanguages(filePath: string[] = [
    'languages/zh_CN/zh_CN', 'languages/en_US/en_US', 'languages/zh_TW/zh_TW'
]) {
    for (let i = 0; i < filePath.length; i++) {
        let request = (await import(`@/assets/${filePath[i]}`)).default
        if (filePath[i].indexOf('zh_CN') > -1) {
            combinat_comm_zh_CN = { comm: {...base_zh_CN.comm, ...request.comm} }
            i18n.setLocaleMessage("zh_CN",{ ...base_zh_CN, ...request, ...combinat_comm_zh_CN });
        } else if (filePath[i].indexOf('en_US') > -1) {
            combinat_comm_en_US = { comm: { ...base_en_US.comm, ...request.comm } }
            i18n.setLocaleMessage("en_US",{ ...base_en_US, ...request, ...combinat_comm_en_US });
        } else if (filePath[i].indexOf('zh_TW') > -1) {
            combinat_comm_zh_TW = { comm: { ...base_zh_TW.comm, ...request.comm } }
            i18n.setLocaleMessage("zh_TW",{ ...base_zh_TW, ...request, ...combinat_comm_zh_TW });
        }
    }
}

Vue.prototype.i18n = i18n

export default i18n;

在main.ts中注册

import i18n from "@/maxbase-ui/assets/languages/i18n";
import { setLanguages } from '@/maxbase-ui/assets/languages/i18n'
async function main_init() {
    try {
        await setLanguages()
        return "success";
    } catch (error) {
        throw new Error(error);
    }
}

main_init().then((res) => {
    new Vue({
        router,
        store,
        i18n,
        render: (h) => h(App),
    }).$mount("#app");
});

在enums/index.ts中使用如下所示

import Vue from 'vue'
/**
 * 设备状态枚举
 */
enum equipStatus {
    INUSE = 1,
    STANDBY = 2,
    FAULT = 3,
    SCRAPPED = 4,
    MAINTENANCE = 5,
}
export const equipmentStatus = {
    [equipStatus.INUSE]: Vue.prototype.i18n.t('comm.GL_INUSE_NAME'),
    [equipStatus.STANDBY]: Vue.prototype.i18n.t('comm.GL_SPARE_NAME'),
    [equipStatus.FAULT]: Vue.prototype.i18n.t('comm.GL_FAULT_NAME'),
    [equipStatus.SCRAPPED]: Vue.prototype.i18n.t('comm.GL_SCRAPPED_NAME'),
    [equipStatus.MAINTENANCE]: Vue.prototype.i18n.t('comm.GL_MAINTENANCE_NAME')

}

在使用到此枚举对象的模块,解析结果为
image.png

1、断点走下来发现执行顺序为,先直接导出i18n对象,然后走enums/index.ts,然后再去执行i118n.ts中的setLanguages()方法,初步判断是因为在执行enums/index.ts中的多语言转换时,i18n的语言包还未挂载到i18n中,所以导致在使用的时候,无法识别多语言。
请问怎样可以让enums/index.ts中的多语言正确识别。

阅读 498
avatarAI BotBETA

你的分析是正确的。问题的根源在于enums/index.ts中的代码在i18n.tssetLanguages方法完成之前就已经执行了。setLanguages是异步的,这意味着它不会立即完成,而是在后台执行。而enums/index.ts中的代码是同步执行的,因此它会在setLanguages完成之前尝试访问Vue.prototype.i18n,此时语言包可能还没有加载完成。

要解决这个问题,你可以将枚举的创建推迟到setLanguages完成之后。这可以通过在main.ts中等待setLanguages完成,然后再创建和使用枚举来实现。

首先,你需要在i18n.ts中导出setLanguages函数,以便在main.ts中使用。然后,在main.ts中等待setLanguages完成后,再创建并使用枚举。

这是一个可能的解决方案:

  1. 修改i18n.ts,使其导出setLanguages函数:
// 其他代码保持不变...

export { setLanguages };  // 添加这一行

export default i18n;
  1. 修改main.ts,使其在setLanguages完成后创建和使用枚举:
import i18n from "@/maxbase-ui/assets/languages/i18n";
import { setLanguages, equipmentStatus } from '@/maxbase-ui/assets/languages/i18n'  // 导入enums/index.ts中的enums
import Vue from 'vue'

async function main_init() {
    try {
        await setLanguages()

        // 在这里创建和使用枚举
        Vue.prototype.$equipmentStatus = {
            [equipmentStatus.INUSE]: Vue.prototype.i18n.t('comm.GL_INUSE_NAME'),
            [equipmentStatus.STANDBY]: Vue.prototype.i18n.t('comm.GL_SPARE_NAME'),
            [equipmentStatus.FAULT]: Vue.prototype.i18n.t('comm.GL_FAULT_NAME'),
            [equipmentStatus.SCRAPPED]: Vue.prototype.i18n.t('comm.GL_SCRAPPED_NAME'),
            [equipmentStatus.MAINTENANCE]: Vue.prototype.i18n.t('comm.GL_MAINTENANCE_NAME')
        }

        return "success";
    } catch (error) {
        throw new Error(error);
    }
}

main_init().then((res) => {
    new Vue({
        router,
        store,
        i18n,
        render: (h) => h(App),
    }).$mount("#app");
});

这样,枚举的创建和使用就会在setLanguages完成后进行,从而确保Vue.prototype.i18n已经包含了所有必要的语言包。同时,由于枚举是存储在Vue.prototype上的,你可以在应用的任何组件中通过this.$equipmentStatus来访问它。

请注意,这种方法将枚举的创建和使用推迟到了应用的初始化阶段,这可能会影响应用的启动时间。如果这是一个问题,你可能需要寻找其他解决方案,例如使用懒加载来动态加载语言包,或者在枚举中使用默认的语言字符串,然后在语言包加载完成后更新这些字符串。

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题