我正在将React Intl用于x种语言(下面的示例),此刻我将以下内容导入到我设置应用程序的位置:
import { addLocaleData } from 'react-intl';
import locale_en from 'react-intl/locale-data/en';
import locale_de from 'react-intl/locale-data/de';
import messages_en from './translations/en.json';
import messages_de from './translations/de.json';
addLocaleData([...locale_en, ...locale_de]);
...
export const messages = {
en: messages_en,
de: messages_de
}
由于这些语言文件都被导入,无论使用哪种语言,我的主包js文件都变得越来越大,尤其是来自.json文件。
如何使用Webpack拆分这些语言文件(或使用CopyWebpackPlugin将它们复制到我的dist文件夹中),然后根据当前使用的语言动态导入它们?
该应用程序是同构的,因此该相同的代码正在服务器上运行。
答案 0 :(得分:0)
尽管我的项目不需要SSR,但最近我一直在进行类似的工作。我发现将动态导入语法与React的Suspense组件配对可以达到预期的结果。这是我发现起作用的粗略概述,至少就我而言,其中不包括SSR:
// wrap this around your JSX in App.js:
<React.Suspense fallback={<SomeLoadingComponent />}>
<AsyncIntlProvider>
{/* app child components go here */}
</AsyncIntlProvider>
</React.Suspense>
// the rest is in support of this
// can be placed in another file
// simply import AsyncIntlProvider in App.js
const messagesCache = {};
const AsyncIntlProvider = ({ children }) => {
// replace with your app's locale getting logic
// if based on a hook like useState, should kick off re-render and load new message bundle when locale changes (but I haven't tested this yet)
const locale = getLocale();
const messages = getMessages(locale);
return (
<IntlProvider locale={locale} messages={messages}>
{children}
</IntlProvider>
);
};
function getMessages(locale) {
if (messagesCache[locale]) {
return messagesCache[locale];
}
// Suspense is based on ErrorBoundary
// throwing a promise will cause <SomeLoadingComponent /> to render until the promise resolves
throw loadMessages(locale);
}
async function loadMessages(locale) {
// dynamic import syntax tells webpack to split this module into its own chunk
const messages = await import('./path/to/${locale}.json`);
messagesCache[locale] = messages;
return messages;
}
Webpack应该将每个区域设置JSON文件拆分成自己的块。如果不是,则可能在将动态导入语法传递到webpack之前将其转换为其他模块系统(require等)。例如:如果使用Typescript,则tsconfig需要"module": "esnext"
来保留import()
语法。如果使用Babel,它可能也会尝试进行模块移植。
单个语言环境的块输出看起来像这样;绝对比通过CopyWebpackPlugin实现的功能还多:
(window["webpackJsonp"] = window["webpackJsonp"] || []).push([[0],{
/***/ "./path/to/en-US.json":
/*!*************************************!*\
!*** ./path/to/en-US.json ***!
\*************************************/
/*! exports provided: message.id, default */
/***/ (function(module) {
eval("module.exports = JSON.parse(\"{\\\"message.id\\\":\\\"Localized message text\\\"}\");//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvbG9jYWxpemF0aW9uL2VuLVVTLmpzb24uanMiLCJzb3VyY2VzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./path/to/en-US.json\n");
/***/ })
}]);
希望这是一个很好的起点,可以与SSR一起使用,也可以进行修改以与SSR一起使用。请报告您在该主题上的发现。 ?