我在 Nuxtjs 项目中使用 vue-i18n,我想用 vite 动态导入我的语言环境文件。
当我使用 webpack 时,这些代码运行良好
插件/i18n.js
import Vue from 'vue';
import VueI18n from 'vue-i18n';
import config from '@/config';
Vue.use(VueI18n);
let messages = Object;
config.locale.available.forEach(locale => {
messages[locale] = require(`~/locales/${locale}.json`);
});
export default ({ app, store }) => {
app.i18n = new VueI18n({
locale: store.state.locale.locale,
messages: messages
});
}
我发现vitejs中没有
require()
,还有vitejs的glob-import功能
let messages = Object,
languages = import.meta.glob('../locales/*.json'); // => languages = {} (languages only get {} value)
config.locale.available.forEach(locale => {
messages[locale] = languages[`../locales/${locale}.json`];
});
但是
languages
只得到了 {}
值。
import()
let messages = Object,
translate = lang => () => import(`@/locales/${lang}.json`).then(i => i.default || i);
config.locale.available.forEach(locale => {
messages[locale] = translate(locale);
});
终端和控制台都没有错误,但没有正确加载语言环境文件。
只要我一一
import()
,问题就会消失:
import en from '@/locales/en.json';
import fr from '@/locales/fr.json';
import ja from '@/locales/ja.json';
let messages = Object;
messages['en'] = en;
messages['fr'] = fr;
messages['ja'] = ja;
但是,如何动态导入呢?
我用谷歌搜索了它,但帮助不大。非常感谢任何人的帮助!
所以,我想我根据 @LiuQixuan 在 GitHub 上的回答找到了一些答案。
//Importing your data
const data = import.meta.glob('../locales/*.json')
//Ref helps with promises I think. I'm sure there are more elegant ways.
const imp = ref([{}])
// From https://github.com/vitejs/vite/issues/77
// by LiuQixuan commented on Jun 20
for (const path in data) {
data[path]().then((mod) => {
imp.value.push(mod)
})
}
从那里开始,我迭代了
imp.values
,然后我可以循环调用每个文件,并使用以下方法获取数据:
JSON.parse(JSON.stringify(**Your data**))
我的示例 Vue HTML 是这样的:
<div v-for="(module,i) in imp" :key="i">
<div v-for="(data,j) in module" :key="j">
//at this point you can read it fully with {{ data }}
<div v-for="(jsonText, k) in JSON.parse(JSON.stringify(data))" :key=k">
{{ jsonText.text }}
<div v-for="insideJson in jsonText" :key="insideJson">
{{ insideJson.yourtext }}
</div>
</div>
</div>
</div>
这样,我就可以访问文件中的每个对象了。我认为您还有其他需求,但这证明您可以访问每个文件而无需独立导入。
我知道这有点粗糙。我使用了 Stringify 解析,因为数据总是像以前一样返回,所以我无法直接访问。我确信有一个更优雅的解决方案,但这就是我想到的。
我最初是尝试动态导入图像,在弄清楚之后,我将该方法应用于您的问题。
对于任何想从文件夹动态导入多图像的人来说,请使用:
new URL(*, import.meta.url)
与之前一样,但添加了:
for (const path in modules) {
modules[path]().then(() => {
//*************
const imgURL = new URL(path, import.meta.url)
//*************
gallery.value.push(imgURL)
})
}
//Then reference that gallery.value in your :src
基于@Lucas Dawson的回答
对于那些必须在 vite 和 webpack 上运行代码的人
插件/i18n.js
import Vue from 'vue';
import VueI18n from 'vue-i18n';
Vue.use(VueI18n);
/***
make sure your public config has `VITE_` prefix or it can't be seen in the client side
https://vitejs.dev/guide/env-and-mode.html#env-files
***/
let messages = Object,
locales = process.env.VITE_AVAILABLE_LOCALES.split(',');
// check is using vite or webpack
if (typeof __webpack_require__ !== 'function') {
// for vite
/***
glob and globEager are both work for this
https://vitejs.dev/guide/features.html#glob-import
the differences is
`glob` will return a dynamic import function (lazy load)
`globEager` will return the data of the file from the path directly
if you use `glob`, don't forget `JSON.parse(JSON.stringify(**Your data**))`
if you have trouble with this, use `globEager` may save your day
***/
let modules = import.meta.globEager('/lang/*.json');
locales.forEach(locale => {
messages[locale] = modules[`/lang/${locale}.json`];
});
}
else {
// for webpack (storybook)
locales.forEach(locale => {
messages[locale] = require(`~/lang/${locale}.json`);
});
}
export default ({ app, store }) => {
app.i18n = new VueI18n({
locale: store.getters['locale/locale'],
messages
});
}
再次感谢@Lucas Dawson的回答!
接受的答案适用于某些情况,但就我而言,我需要从一开始就访问本地化文本,而不会因
import.meta.glob
生成的 Promises 造成延迟。
这对我有帮助,因为当文件已经可用时它是即时的:
const requireLocale = async fileName => {
try {
const files = import.meta.globEager("./i18n/*.json")
const texts = files[`./i18n/${fileName}.json`]
return texts?.default || {}
}
catch (e) {
console.warn(`The file "./i18n/${fileName}.json" could not be loaded.`)
return {}
}
}