@fluenti/next 0.1.3 → 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/dev-runner.d.ts +28 -0
- package/dist/dev-runner.d.ts.map +1 -0
- package/dist/generate-server-module.d.ts.map +1 -1
- package/dist/index.cjs +12 -12
- package/dist/index.cjs.map +1 -1
- package/dist/index.js +123 -52
- package/dist/index.js.map +1 -1
- package/dist/types.d.ts +4 -0
- package/dist/types.d.ts.map +1 -1
- package/dist/with-fluenti.d.ts.map +1 -1
- package/package.json +3 -3
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
export interface DevRunnerOptions {
|
|
2
|
+
cwd: string;
|
|
3
|
+
onSuccess?: () => void;
|
|
4
|
+
onError?: (err: Error) => void;
|
|
5
|
+
/** If true, reject the promise on failure instead of swallowing the error */
|
|
6
|
+
throwOnError?: boolean;
|
|
7
|
+
/** Run only compile (skip extract). Useful for production builds where source is unchanged. */
|
|
8
|
+
compileOnly?: boolean;
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* Walk up from `cwd` to find `node_modules/.bin/fluenti`.
|
|
12
|
+
* Returns the absolute path or null if not found.
|
|
13
|
+
*/
|
|
14
|
+
export declare function resolveCliBin(cwd: string): string | null;
|
|
15
|
+
/**
|
|
16
|
+
* Run compile in-process via `@fluenti/cli` (for compileOnly mode),
|
|
17
|
+
* or fall back to shell-out for extract + compile (dev mode).
|
|
18
|
+
*/
|
|
19
|
+
export declare function runExtractCompile(options: DevRunnerOptions): Promise<void>;
|
|
20
|
+
/**
|
|
21
|
+
* Create a debounced runner that collapses rapid calls.
|
|
22
|
+
*
|
|
23
|
+
* - If called while idle, schedules a run after `delay` ms.
|
|
24
|
+
* - If called while a run is in progress, marks a pending rerun.
|
|
25
|
+
* - Never runs concurrently.
|
|
26
|
+
*/
|
|
27
|
+
export declare function createDebouncedRunner(options: DevRunnerOptions, delay?: number): () => void;
|
|
28
|
+
//# sourceMappingURL=dev-runner.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"dev-runner.d.ts","sourceRoot":"","sources":["../src/dev-runner.ts"],"names":[],"mappings":"AAKA,MAAM,WAAW,gBAAgB;IAC/B,GAAG,EAAE,MAAM,CAAA;IACX,SAAS,CAAC,EAAE,MAAM,IAAI,CAAA;IACtB,OAAO,CAAC,EAAE,CAAC,GAAG,EAAE,KAAK,KAAK,IAAI,CAAA;IAC9B,6EAA6E;IAC7E,YAAY,CAAC,EAAE,OAAO,CAAA;IACtB,+FAA+F;IAC/F,WAAW,CAAC,EAAE,OAAO,CAAA;CACtB;AAED;;;GAGG;AACH,wBAAgB,aAAa,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI,CAUxD;AAED;;;GAGG;AACH,wBAAsB,iBAAiB,CAAC,OAAO,EAAE,gBAAgB,GAAG,OAAO,CAAC,IAAI,CAAC,CAuDhF;AAED;;;;;;GAMG;AACH,wBAAgB,qBAAqB,CACnC,OAAO,EAAE,gBAAgB,EACzB,KAAK,SAAM,GACV,MAAM,IAAI,CAiCZ"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"generate-server-module.d.ts","sourceRoot":"","sources":["../src/generate-server-module.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,SAAS,CAAA;AAEnD;;;;;;;GAOG;AACH,wBAAgB,oBAAoB,CAClC,WAAW,EAAE,MAAM,EACnB,MAAM,EAAE,oBAAoB,GAC3B,MAAM,
|
|
1
|
+
{"version":3,"file":"generate-server-module.d.ts","sourceRoot":"","sources":["../src/generate-server-module.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,oBAAoB,EAAE,MAAM,SAAS,CAAA;AAEnD;;;;;;;GAOG;AACH,wBAAgB,oBAAoB,CAClC,WAAW,EAAE,MAAM,EACnB,MAAM,EAAE,oBAAoB,GAC3B,MAAM,CAiMR"}
|
package/dist/index.cjs
CHANGED
|
@@ -1,22 +1,22 @@
|
|
|
1
|
-
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`});let e=require(`node:fs`),t=require(`node:
|
|
2
|
-
`),u=
|
|
1
|
+
Object.defineProperty(exports,Symbol.toStringTag,{value:`Module`});let e=require(`node:fs`),t=require(`node:child_process`),n=require(`node:path`),r=require(`node:module`),i=require(`@fluenti/core`);var a=typeof __filename==`string`?__filename:{}.url,{createJiti:o}=(0,r.createRequire)(a)(`jiti`);function s(e,t){let r=c(e),i=t?.defaultLocale??r?.sourceLocale??`en`,a=t?.locales??r?.locales??[i],o=t?.compiledDir??r?.compileOutDir??`./src/locales/compiled`,s=t?.serverModuleOutDir??(0,n.join)(`node_modules`,`.fluenti`),l={locales:a,defaultLocale:i,compiledDir:o,serverModule:t?.serverModule??null,serverModuleOutDir:s};return t?.resolveLocale&&(l.resolveLocale=t.resolveLocale),t?.dateFormats&&(l.dateFormats=t.dateFormats),t?.numberFormats&&(l.numberFormats=t.numberFormats),t?.fallbackChain&&(l.fallbackChain=t.fallbackChain),l}function c(t){let r=o(a,{moduleCache:!1,interopDefault:!0});for(let i of[`fluenti.config.ts`,`fluenti.config.js`,`fluenti.config.mjs`]){let a=(0,n.resolve)(t,i);if((0,e.existsSync)(a))try{return u(r(a))}catch{return l(a,r)||null}}return null}function l(t,r){let i=(0,e.readFileSync)(t,`utf8`),a=i.match(/import\s*\{\s*defineConfig(?:\s+as\s+([A-Za-z_$][\w$]*))?\s*\}\s*from\s*['"]@fluenti\/cli['"]\s*;?/);if(!a)return null;let o=a[1]??`defineConfig`,s=i.replace(a[0],``),c=(0,n.join)((0,n.dirname)(t),`.${(0,n.basename)(t,(0,n.extname)(t))}.next-plugin-read-config${(0,n.extname)(t)||`.ts`}`);(0,e.writeFileSync)(c,`const ${o} = (config) => config\n${s}`,`utf8`);try{return u(r(c))}catch{return null}finally{(0,e.rmSync)(c,{force:!0})}}function u(e){return typeof e==`object`&&e&&`default`in e?e.default??{}:e}function d(t,r){if(r.serverModule)return(0,n.resolve)(t,r.serverModule);let a=(0,n.resolve)(t,r.serverModuleOutDir),o=(0,n.resolve)(a,`server.js`),s=(0,n.resolve)(a,`server.d.ts`);(0,e.existsSync)(a)||(0,e.mkdirSync)(a,{recursive:!0});for(let e of r.locales)(0,i.validateLocale)(e,`next-plugin`);let c=f((0,n.relative)(a,(0,n.resolve)(t,r.compiledDir))),l=r.locales.map(e=>` case '${e}': return import('${c}/${e}')`).join(`
|
|
2
|
+
`),u=r.fallbackChain?JSON.stringify(r.fallbackChain):`undefined`;(0,e.writeFileSync)((0,n.resolve)(a,`client-provider.js`),`"use client";
|
|
3
3
|
// Auto-generated by @fluenti/next — do not edit
|
|
4
4
|
import { createElement } from 'react'
|
|
5
5
|
import { I18nProvider } from '@fluenti/react'
|
|
6
|
-
${
|
|
6
|
+
${r.locales.map(e=>`import ${e.replace(/[^a-zA-Z0-9]/g,`_`)} from '${c}/${e}'`).join(`
|
|
7
7
|
`)}
|
|
8
8
|
|
|
9
|
-
const __allMessages = { ${
|
|
9
|
+
const __allMessages = { ${r.locales.map(e=>`'${e}': ${e.replace(/[^a-zA-Z0-9]/g,`_`)}`).join(`, `)} }
|
|
10
10
|
|
|
11
11
|
export function ClientI18nProvider({ locale, fallbackLocale, fallbackChain, children }) {
|
|
12
12
|
return createElement(I18nProvider, { locale, fallbackLocale, messages: __allMessages, fallbackChain }, children)
|
|
13
13
|
}
|
|
14
|
-
`,`utf-8`);let d=
|
|
14
|
+
`,`utf-8`);let d=r.resolveLocale?`import __resolveLocale from '${f((0,n.relative)(a,(0,n.resolve)(t,r.resolveLocale)))}'`:null,p=r.resolveLocale?`resolveLocale: __resolveLocale,`:`resolveLocale: async () => {
|
|
15
15
|
try {
|
|
16
16
|
const { cookies } = await import('next/headers')
|
|
17
|
-
return (await cookies()).get('locale')?.value ?? '${
|
|
17
|
+
return (await cookies()).get('locale')?.value ?? '${r.defaultLocale}'
|
|
18
18
|
} catch {
|
|
19
|
-
return '${
|
|
19
|
+
return '${r.defaultLocale}'
|
|
20
20
|
}
|
|
21
21
|
},`,m=`// Auto-generated by @fluenti/next — do not edit
|
|
22
22
|
import { createServerI18n } from '@fluenti/react/server'
|
|
@@ -26,10 +26,10 @@ const serverI18n = createServerI18n({
|
|
|
26
26
|
loadMessages: async (locale) => {
|
|
27
27
|
switch (locale) {
|
|
28
28
|
${l}
|
|
29
|
-
default: return import('${c}/${
|
|
29
|
+
default: return import('${c}/${r.defaultLocale}')
|
|
30
30
|
}
|
|
31
31
|
},
|
|
32
|
-
fallbackLocale: '${
|
|
32
|
+
fallbackLocale: '${r.defaultLocale}',
|
|
33
33
|
fallbackChain: ${u},
|
|
34
34
|
${p}
|
|
35
35
|
})
|
|
@@ -54,7 +54,7 @@ export const NumberFormat = serverI18n.NumberFormat
|
|
|
54
54
|
* Sets up both server-side (React.cache) and client-side (I18nProvider) i18n.
|
|
55
55
|
*/
|
|
56
56
|
export async function I18nProvider({ locale, children }) {
|
|
57
|
-
const activeLocale = locale ?? '${
|
|
57
|
+
const activeLocale = locale ?? '${r.defaultLocale}'
|
|
58
58
|
|
|
59
59
|
// 1. Initialize server-side i18n (React.cache scoped)
|
|
60
60
|
serverI18n.setLocale(activeLocale)
|
|
@@ -66,7 +66,7 @@ export async function I18nProvider({ locale, children }) {
|
|
|
66
66
|
|
|
67
67
|
return createElement(ClientI18nProvider, {
|
|
68
68
|
locale: activeLocale,
|
|
69
|
-
fallbackLocale: '${
|
|
69
|
+
fallbackLocale: '${r.defaultLocale}',
|
|
70
70
|
fallbackChain: ${u},
|
|
71
71
|
}, children)
|
|
72
72
|
}
|
|
@@ -124,5 +124,5 @@ export declare function I18nProvider(props: {
|
|
|
124
124
|
locale?: string
|
|
125
125
|
children: ReactNode
|
|
126
126
|
}): Promise<ReactElement>
|
|
127
|
-
`,`utf-8`),o}function f(e){return e.split(`\\`).join(`/`)}function p(e){if(e&&
|
|
127
|
+
`,`utf-8`),o}function f(e){return e.split(`\\`).join(`/`)}function p(t){let r=t;for(;;){let t=(0,n.resolve)(r,`node_modules/.bin/fluenti`);if((0,e.existsSync)(t))return t;let i=(0,n.dirname)(r);if(i===r)break;r=i}return null}async function m(e){if(e.compileOnly)try{let{runCompile:t}=(0,r.createRequire)((0,n.join)(e.cwd,`package.json`))(`@fluenti/cli`);await t(e.cwd),console.log(`[fluenti] Compiling... done`),e.onSuccess?.();return}catch(t){let n=t instanceof Error?t:Error(String(t));if(e.throwOnError)throw n;console.warn(`[fluenti] Compile failed:`,n.message),e.onError?.(n);return}let i=p(e.cwd);if(!i){let t=`[fluenti] CLI not found — skipping auto-compile. Install @fluenti/cli as a devDependency.`;return e.throwOnError?Promise.reject(Error(t)):(console.warn(t),Promise.resolve())}let a=`${i} extract && ${i} compile`;return new Promise((n,r)=>{(0,t.exec)(a,{cwd:e.cwd},(t,i,a)=>{if(t){let n=Error(a||t.message);if(e.throwOnError){r(n);return}console.warn(`[fluenti] Extract/compile failed:`,n.message),e.onError?.(n)}else console.log(`[fluenti] Extracting and compiling... done`),e.onSuccess?.();n()})})}function h(e,t=300){let n=null,r=!1,i=!1;async function a(){r=!0;try{await m(e)}finally{r=!1,i&&(i=!1,o())}}function o(){n!==null&&clearTimeout(n),n=setTimeout(()=>{n=null,r?i=!0:a()},t)}return o}function g(e){if(e&&_(e))return v({},e);let t=e??{};return function(e){return v(t,e??{})}}function _(e){return[`reactStrictMode`,`experimental`,`images`,`env`,`webpack`,`rewrites`,`redirects`,`headers`,`pageExtensions`,`output`,`basePath`,`i18n`,`trailingSlash`,`compiler`,`transpilePackages`].some(t=>t in e)}function v(r,i){let a=process.cwd(),o=s(a,r);(0,e.existsSync)((0,n.resolve)(a,o.compiledDir))||console.warn(`\n[fluenti] Compiled catalogs not found at ${o.compiledDir}.\nRun: npx fluenti extract && npx fluenti compile\n`);let c=d(a,o),l=(0,n.resolve)(typeof __dirname<`u`?__dirname:(0,n.dirname)(new URL({}.url).pathname),`loader.js`),u=i.webpack,f=!1;return{...i,webpack(e,i){e.module.rules.push({test:/\.[jt]sx?$/,enforce:`pre`,exclude:[/node_modules/,/\.next/],use:[{loader:l,options:{serverModulePath:c}}]}),e.resolve=e.resolve??{},e.resolve.alias=e.resolve.alias??{},e.resolve.alias[`@fluenti/next$`]=c;let s=r.buildAutoCompile??!0;if(!i.dev&&s&&!f){f=!0;try{(0,t.execSync)(`node --input-type=module -e "const { runCompile } = await import('@fluenti/cli'); await runCompile('${a.replace(/\\/g,`\\\\`).replace(/'/g,`\\'`)}')"`,{cwd:a,stdio:`inherit`})}catch{}}let d=r.devAutoCompile??!0;if(i.dev&&d){let t=r.devAutoCompileDelay??1e3,i=h({cwd:a},t),s=(0,n.resolve)(a,o.compiledDir);e.plugins=e.plugins??[],e.plugins.push({apply(e){let t=!0;e.hooks.watchRun.tapAsync(`fluenti-dev`,(e,n)=>{t&&(t=!1,i());let r=e.modifiedFiles;r&&[...r].some(e=>/\.[jt]sx?$/.test(e)&&!e.includes(`node_modules`)&&!e.includes(`.next`)&&!e.startsWith(s))&&i(),n()})}})}return u?u(e,i):e}}}var y='[fluenti] `withFluenti()` must be configured in next.config.ts before importing from "@fluenti/next".';function b(){throw Error(y)}var x=b,S=b,C=b,w=b,T=b,E=b,D=b,O=b,k=b;exports.DateTime=D,exports.I18nProvider=k,exports.NumberFormat=O,exports.Plural=T,exports.Select=E,exports.Trans=w,exports.getI18n=S,exports.setLocale=x,exports.t=C,exports.withFluenti=g;
|
|
128
128
|
//# sourceMappingURL=index.cjs.map
|
package/dist/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.cjs","names":[],"sources":["../src/read-config.ts","../src/generate-server-module.ts","../src/with-fluenti.ts","../src/index.ts"],"sourcesContent":["import { existsSync, readFileSync, rmSync, writeFileSync } from 'node:fs'\nimport { createRequire } from 'node:module'\nimport { basename, dirname, extname, join, resolve } from 'node:path'\nimport type { FluentiConfig } from '@fluenti/core'\nimport type { WithFluentConfig, ResolvedFluentConfig } from './types'\n\nconst runtimeModulePath = typeof __filename === 'string'\n ? __filename\n : import.meta.url\nconst require = createRequire(runtimeModulePath)\nconst { createJiti } = require('jiti') as {\n createJiti: (\n url: string,\n options?: { moduleCache?: boolean; interopDefault?: boolean },\n ) => (path: string) => unknown\n}\n\n/**\n * Read fluenti.config.ts and merge with withFluenti() overrides.\n */\nexport function resolveConfig(\n projectRoot: string,\n overrides?: WithFluentConfig,\n): ResolvedFluentConfig {\n const fileConfig = readFluentConfigSync(projectRoot)\n\n const defaultLocale = overrides?.defaultLocale\n ?? fileConfig?.sourceLocale\n ?? 'en'\n\n const locales = overrides?.locales\n ?? fileConfig?.locales\n ?? [defaultLocale]\n\n const compiledDir = overrides?.compiledDir\n ?? fileConfig?.compileOutDir\n ?? './src/locales/compiled'\n\n const serverModuleOutDir = overrides?.serverModuleOutDir\n ?? join('node_modules', '.fluenti')\n\n const resolved: ResolvedFluentConfig = {\n locales,\n defaultLocale,\n compiledDir,\n serverModule: overrides?.serverModule ?? null,\n serverModuleOutDir,\n }\n if (overrides?.resolveLocale) resolved.resolveLocale = overrides.resolveLocale\n if (overrides?.dateFormats) resolved.dateFormats = overrides.dateFormats\n if (overrides?.numberFormats) resolved.numberFormats = overrides.numberFormats\n if (overrides?.fallbackChain) resolved.fallbackChain = overrides.fallbackChain\n return resolved\n}\n\n/**\n * Attempt to read fluenti.config.ts synchronously.\n * Returns null if file doesn't exist or can't be parsed.\n */\nfunction readFluentConfigSync(projectRoot: string): FluentiConfig | null {\n const jiti = createJiti(runtimeModulePath, {\n moduleCache: false,\n interopDefault: true,\n })\n const candidates = [\n 'fluenti.config.ts',\n 'fluenti.config.js',\n 'fluenti.config.mjs',\n ]\n\n for (const name of candidates) {\n const configPath = resolve(projectRoot, name)\n if (existsSync(configPath)) {\n try {\n return normalizeLoadedConfig(jiti(configPath) as FluentiConfig | { default?: FluentiConfig })\n } catch {\n const rewritten = tryLoadConfigViaDefineConfigShim(configPath, jiti)\n if (rewritten) {\n return rewritten\n }\n return null\n }\n }\n }\n\n return null\n}\n\nfunction tryLoadConfigViaDefineConfigShim(\n configPath: string,\n jiti: (path: string) => unknown,\n): FluentiConfig | null {\n const source = readFileSync(configPath, 'utf8')\n const importMatch = source.match(\n /import\\s*\\{\\s*defineConfig(?:\\s+as\\s+([A-Za-z_$][\\w$]*))?\\s*\\}\\s*from\\s*['\"]@fluenti\\/cli['\"]\\s*;?/,\n )\n if (!importMatch) {\n return null\n }\n\n const helperName = importMatch[1] ?? 'defineConfig'\n const rewrittenSource = source.replace(importMatch[0], '')\n const tempPath = join(\n dirname(configPath),\n `.${basename(configPath, extname(configPath))}.next-plugin-read-config${extname(configPath) || '.ts'}`,\n )\n\n writeFileSync(tempPath, `const ${helperName} = (config) => config\\n${rewrittenSource}`, 'utf8')\n\n try {\n return normalizeLoadedConfig(jiti(tempPath) as FluentiConfig | { default?: FluentiConfig })\n } catch {\n return null\n } finally {\n rmSync(tempPath, { force: true })\n }\n}\n\nfunction normalizeLoadedConfig(\n mod: FluentiConfig | { default?: FluentiConfig },\n): FluentiConfig {\n return typeof mod === 'object' && mod !== null && 'default' in mod\n ? (mod.default ?? {}) as FluentiConfig\n : mod as FluentiConfig\n}\n","import { writeFileSync, mkdirSync, existsSync } from 'node:fs'\nimport { resolve, relative } from 'node:path'\nimport { validateLocale } from '@fluenti/core'\nimport type { ResolvedFluentConfig } from './types'\n\n/**\n * Generate the server module that provides:\n * - setLocale / getI18n\n * - Trans / Plural / Select / DateTime / NumberFormat (server components)\n * - I18nProvider (async server component for layouts)\n *\n * @returns Absolute path to the generated server module.\n */\nexport function generateServerModule(\n projectRoot: string,\n config: ResolvedFluentConfig,\n): string {\n if (config.serverModule) {\n return resolve(projectRoot, config.serverModule)\n }\n\n const outDir = resolve(projectRoot, config.serverModuleOutDir)\n const outPath = resolve(outDir, 'server.js')\n const dtsPath = resolve(outDir, 'server.d.ts')\n\n if (!existsSync(outDir)) {\n mkdirSync(outDir, { recursive: true })\n }\n\n for (const locale of config.locales) {\n validateLocale(locale, 'next-plugin')\n }\n\n const compiledDirAbs = resolve(projectRoot, config.compiledDir)\n const compiledRelative = toForwardSlash(relative(outDir, compiledDirAbs))\n\n const localeImports = config.locales\n .map((locale) => ` case '${locale}': return import('${compiledRelative}/${locale}')`)\n .join('\\n')\n\n const fallbackChainStr = config.fallbackChain\n ? JSON.stringify(config.fallbackChain)\n : 'undefined'\n\n // Generate a 'use client' provider that imports messages statically.\n // Messages contain functions (interpolation) which can't cross the RSC boundary.\n const clientProviderPath = resolve(outDir, 'client-provider.js')\n\n const clientStaticImports = config.locales\n .map((locale) => {\n const safe = locale.replace(/[^a-zA-Z0-9]/g, '_')\n return `import ${safe} from '${compiledRelative}/${locale}'`\n })\n .join('\\n')\n\n const clientAllMessagesEntries = config.locales\n .map((locale) => {\n const safe = locale.replace(/[^a-zA-Z0-9]/g, '_')\n return `'${locale}': ${safe}`\n })\n .join(', ')\n\n const clientProviderSource = `\"use client\";\n// Auto-generated by @fluenti/next — do not edit\nimport { createElement } from 'react'\nimport { I18nProvider } from '@fluenti/react'\n${clientStaticImports}\n\nconst __allMessages = { ${clientAllMessagesEntries} }\n\nexport function ClientI18nProvider({ locale, fallbackLocale, fallbackChain, children }) {\n return createElement(I18nProvider, { locale, fallbackLocale, messages: __allMessages, fallbackChain }, children)\n}\n`\n writeFileSync(clientProviderPath, clientProviderSource, 'utf-8')\n\n const resolveLocaleImport = config.resolveLocale\n ? `import __resolveLocale from '${config.resolveLocale}'`\n : null\n\n const resolveLocaleFn = config.resolveLocale\n ? `resolveLocale: __resolveLocale,`\n : `resolveLocale: async () => {\n try {\n const { cookies } = await import('next/headers')\n return (await cookies()).get('locale')?.value ?? '${config.defaultLocale}'\n } catch {\n return '${config.defaultLocale}'\n }\n },`\n\n const moduleSource = `// Auto-generated by @fluenti/next — do not edit\nimport { createServerI18n } from '@fluenti/react/server'\nimport { createElement } from 'react'\n${resolveLocaleImport ? `${resolveLocaleImport}\\n` : ''}\nconst serverI18n = createServerI18n({\n loadMessages: async (locale) => {\n switch (locale) {\n${localeImports}\n default: return import('${compiledRelative}/${config.defaultLocale}')\n }\n },\n fallbackLocale: '${config.defaultLocale}',\n fallbackChain: ${fallbackChainStr},\n ${resolveLocaleFn}\n})\n\nexport const setLocale = serverI18n.setLocale\nexport const getI18n = serverI18n.getI18n\nexport const t = (..._args) => {\n throw new Error(\n \"[fluenti] \\`t\\` imported from '@fluenti/next' is a compile-time API. \" +\n 'Use it only with the Fluenti loader inside an async server scope.',\n )\n}\nexport const Trans = serverI18n.Trans\nexport const Plural = serverI18n.Plural\nexport const Select = serverI18n.Select\nexport const DateTime = serverI18n.DateTime\nexport const NumberFormat = serverI18n.NumberFormat\n\n/**\n * Async server component for root layouts.\n *\n * Sets up both server-side (React.cache) and client-side (I18nProvider) i18n.\n */\nexport async function I18nProvider({ locale, children }) {\n const activeLocale = locale ?? '${config.defaultLocale}'\n\n // 1. Initialize server-side i18n (React.cache scoped)\n serverI18n.setLocale(activeLocale)\n await serverI18n.getI18n()\n\n // 2. Import the local 'use client' provider that has messages statically bundled.\n // Messages contain functions (interpolation) which can't be serialized across the RSC boundary.\n const { ClientI18nProvider } = await import('./client-provider.js')\n\n return createElement(ClientI18nProvider, {\n locale: activeLocale,\n fallbackLocale: '${config.defaultLocale}',\n fallbackChain: ${fallbackChainStr},\n }, children)\n}\n`\n\n const dtsSource = `// Auto-generated by @fluenti/next — do not edit\nimport type { ReactNode, ReactElement } from 'react'\nimport type { CompileTimeT, FluentInstanceExtended } from '@fluenti/core'\n\nexport declare function setLocale(locale: string): void\nexport declare function getI18n(): Promise<FluentInstanceExtended & { locale: string }>\nexport declare const t: CompileTimeT\n\nexport declare function Trans(props: {\n children: ReactNode\n id?: string\n context?: string\n comment?: string\n render?: (translation: ReactNode) => ReactNode\n}): Promise<ReactElement>\n\nexport declare function Plural(props: {\n value: number\n id?: string\n context?: string\n comment?: string\n zero?: ReactNode\n one?: ReactNode\n two?: ReactNode\n few?: ReactNode\n many?: ReactNode\n other: ReactNode\n offset?: number\n}): Promise<ReactElement>\n\nexport declare function Select(props: {\n value: string\n id?: string\n context?: string\n comment?: string\n other: ReactNode\n options?: Record<string, ReactNode>\n [key: string]: ReactNode | Record<string, ReactNode> | string | undefined\n}): Promise<ReactElement>\n\nexport declare function DateTime(props: {\n value: Date | number\n style?: string\n}): Promise<ReactElement>\n\nexport declare function NumberFormat(props: {\n value: number\n style?: string\n}): Promise<ReactElement>\n\nexport declare function I18nProvider(props: {\n locale?: string\n children: ReactNode\n}): Promise<ReactElement>\n`\n\n writeFileSync(outPath, moduleSource, 'utf-8')\n writeFileSync(dtsPath, dtsSource, 'utf-8')\n\n return outPath\n}\n\nfunction toForwardSlash(p: string): string {\n return p.split('\\\\').join('/')\n}\n","import { existsSync } from 'node:fs'\nimport { resolve, dirname } from 'node:path'\nimport type { WithFluentConfig } from './types'\nimport { resolveConfig } from './read-config'\nimport { generateServerModule } from './generate-server-module'\nimport { createDebouncedRunner } from '@fluenti/core/internal'\n\ntype NextConfig = Record<string, unknown>\n\n/**\n * Wrap your Next.js config with Fluenti support.\n *\n * Adds a webpack loader that transforms `t\\`\\`` and `t()` calls,\n * and generates a server module for RSC i18n.\n *\n * @example\n * ```ts\n * // next.config.ts — function style (recommended)\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti()({ reactStrictMode: true })\n * ```\n *\n * @example\n * ```ts\n * // next.config.ts — direct style\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti({ reactStrictMode: true })\n * ```\n */\nexport function withFluenti(fluentConfig?: WithFluentConfig): (nextConfig?: NextConfig) => NextConfig\nexport function withFluenti(nextConfig: NextConfig): NextConfig\nexport function withFluenti(\n configOrNext?: WithFluentConfig | NextConfig,\n): NextConfig | ((nextConfig?: NextConfig) => NextConfig) {\n if (configOrNext && isNextConfig(configOrNext as NextConfig)) {\n return applyFluenti({}, configOrNext as NextConfig)\n }\n\n const fluentConfig = (configOrNext ?? {}) as WithFluentConfig\n return function wrappedConfig(nextConfig?: NextConfig): NextConfig {\n return applyFluenti(fluentConfig, nextConfig ?? {})\n }\n}\n\nfunction isNextConfig(obj: NextConfig): boolean {\n const nextKeys = [\n 'reactStrictMode', 'experimental', 'images', 'env', 'webpack',\n 'rewrites', 'redirects', 'headers', 'pageExtensions', 'output',\n 'basePath', 'i18n', 'trailingSlash', 'compiler', 'transpilePackages',\n ]\n return nextKeys.some((key) => key in obj)\n}\n\nfunction applyFluenti(\n fluentConfig: WithFluentConfig,\n nextConfig: NextConfig,\n): NextConfig {\n const projectRoot = process.cwd()\n const resolved = resolveConfig(projectRoot, fluentConfig)\n\n // Warn if compiled catalogs directory doesn't exist yet\n const compiledDir = resolve(projectRoot, resolved.compiledDir)\n if (!existsSync(compiledDir)) {\n console.warn(\n `\\n[fluenti] Compiled catalogs not found at ${resolved.compiledDir}.\\n` +\n `Run: npx fluenti extract && npx fluenti compile\\n`,\n )\n }\n\n // Generate server module for RSC\n const serverModulePath = generateServerModule(projectRoot, resolved)\n\n // Resolve the loader path — use import.meta.url for ESM compatibility\n const thisDir = typeof __dirname !== 'undefined'\n ? __dirname\n : dirname(new URL(import.meta.url).pathname)\n const loaderPath = resolve(thisDir, 'loader.js')\n\n const existingWebpack = nextConfig['webpack'] as\n | ((config: WebpackConfig, options: WebpackOptions) => WebpackConfig)\n | undefined\n\n return {\n ...nextConfig,\n webpack(config: WebpackConfig, options: WebpackOptions) {\n // Add fluenti loader (enforce: pre — runs before other loaders)\n config.module.rules.push({\n test: /\\.[jt]sx?$/,\n enforce: 'pre' as const,\n exclude: [/node_modules/, /\\.next/],\n use: [\n {\n loader: loaderPath,\n options: {\n serverModulePath,\n },\n },\n ],\n })\n\n // Add resolve alias so loader can import from generated server module\n config.resolve = config.resolve ?? {} as WebpackConfig['resolve']\n config.resolve.alias = config.resolve.alias ?? {}\n config.resolve.alias['@fluenti/next$'] = serverModulePath\n\n // Auto extract+compile in dev mode\n const devAutoCompile = fluentConfig.devAutoCompile ?? true\n if (options.dev && devAutoCompile) {\n const debouncedRun = createDebouncedRunner({ cwd: projectRoot })\n\n config.plugins = config.plugins ?? []\n config.plugins.push({\n apply(compiler: WebpackCompiler) {\n let isFirstBuild = true\n compiler.hooks.watchRun.tapAsync('fluenti-dev', (_compiler: WebpackCompiler, callback: () => void) => {\n if (isFirstBuild) {\n isFirstBuild = false\n debouncedRun()\n }\n const modifiedFiles = _compiler.modifiedFiles\n if (modifiedFiles) {\n const hasSourceChange = [...modifiedFiles].some((f: string) =>\n /\\.[jt]sx?$/.test(f) && !f.includes('node_modules') && !f.includes('.next'),\n )\n if (hasSourceChange) {\n debouncedRun()\n }\n }\n callback()\n })\n },\n })\n }\n\n // Call user's webpack config if provided\n if (existingWebpack) {\n return existingWebpack(config, options)\n }\n\n return config\n },\n }\n}\n\n// Minimal webpack types for the config function\ninterface WebpackConfig {\n module: {\n rules: Array<{\n test: RegExp\n enforce?: 'pre' | 'post'\n exclude?: Array<RegExp>\n use: Array<{ loader: string; options: Record<string, unknown> }>\n }>\n }\n resolve: {\n alias?: Record<string, string>\n }\n plugins?: Array<{ apply(compiler: WebpackCompiler): void }>\n}\n\ninterface WebpackOptions {\n isServer: boolean\n dev: boolean\n}\n\ninterface WebpackCompiler {\n hooks: {\n watchRun: {\n tapAsync(name: string, cb: (compiler: WebpackCompiler, callback: () => void) => void): void\n }\n }\n modifiedFiles?: Set<string>\n}\n","/**\n * @fluenti/next — Next.js plugin for Fluenti\n *\n * Provides:\n * - `withFluenti()` — wraps next.config.ts with t`` transform support\n * - I18nProvider — async server component (exported from generated module)\n * - Webpack loader for strict, binding-aware tagged-template optimization\n *\n * @example\n * ```ts\n * // next.config.ts\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti()({ reactStrictMode: true })\n * ```\n *\n * @example\n * ```tsx\n * // app/layout.tsx — resolved by webpack alias to the generated module\n * import { I18nProvider } from '@fluenti/next'\n * ```\n */\nexport { withFluenti } from './with-fluenti'\nexport type { WithFluentConfig, I18nProviderProps } from './types'\n\n// ── Runtime stubs ────────────────────────────────────────────────────\n// TypeScript resolves types from this file (via package.json exports).\n// At runtime, webpack `resolve.alias` redirects `@fluenti/next$` to the\n// generated server module, so these stubs are never actually called in\n// a correctly configured project. They exist only to provide helpful\n// errors if `withFluenti()` is not configured.\n\nimport type { ReactNode, ReactElement } from 'react'\nimport type { CompileTimeT, FluentInstanceExtended } from '@fluenti/core'\n\nconst NOT_CONFIGURED =\n '[fluenti] `withFluenti()` must be configured in next.config.ts before importing from \"@fluenti/next\".'\n\nfunction throwNotConfigured(): never {\n throw new Error(NOT_CONFIGURED)\n}\n\n/** @see Generated module for the real implementation. */\nexport const setLocale: (locale: string) => void = throwNotConfigured\n/** @see Generated module for the real implementation. */\nexport const getI18n: () => Promise<FluentInstanceExtended & { locale: string }> = throwNotConfigured as () => Promise<FluentInstanceExtended & { locale: string }>\n/** @see Generated module for the real implementation. */\nexport const t: CompileTimeT = throwNotConfigured as unknown as CompileTimeT\n/** @see Generated module for the real implementation. */\nexport const Trans: (props: { children: ReactNode; id?: string; context?: string; comment?: string; render?: (translation: ReactNode) => ReactNode }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Trans\n/** @see Generated module for the real implementation. */\nexport const Plural: (props: { value: number; id?: string; context?: string; comment?: string; zero?: ReactNode; one?: ReactNode; two?: ReactNode; few?: ReactNode; many?: ReactNode; other: ReactNode; offset?: number }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Plural\n/** @see Generated module for the real implementation. */\nexport const Select: (props: { value: string; id?: string; context?: string; comment?: string; other: ReactNode; options?: Record<string, ReactNode>; [key: string]: ReactNode | Record<string, ReactNode> | string | undefined }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Select\n/** @see Generated module for the real implementation. */\nexport const DateTime: (props: { value: Date | number; style?: string }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof DateTime\n/** @see Generated module for the real implementation. */\nexport const NumberFormat: (props: { value: number; style?: string }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof NumberFormat\n/** @see Generated module for the real implementation. */\nexport const I18nProvider: (props: { locale?: string; children: ReactNode }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof I18nProvider\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.cjs","names":[],"sources":["../src/read-config.ts","../src/generate-server-module.ts","../src/dev-runner.ts","../src/with-fluenti.ts","../src/index.ts"],"sourcesContent":["import { existsSync, readFileSync, rmSync, writeFileSync } from 'node:fs'\nimport { createRequire } from 'node:module'\nimport { basename, dirname, extname, join, resolve } from 'node:path'\nimport type { FluentiConfig } from '@fluenti/core'\nimport type { WithFluentConfig, ResolvedFluentConfig } from './types'\n\nconst runtimeModulePath = typeof __filename === 'string'\n ? __filename\n : import.meta.url\nconst require = createRequire(runtimeModulePath)\nconst { createJiti } = require('jiti') as {\n createJiti: (\n url: string,\n options?: { moduleCache?: boolean; interopDefault?: boolean },\n ) => (path: string) => unknown\n}\n\n/**\n * Read fluenti.config.ts and merge with withFluenti() overrides.\n */\nexport function resolveConfig(\n projectRoot: string,\n overrides?: WithFluentConfig,\n): ResolvedFluentConfig {\n const fileConfig = readFluentConfigSync(projectRoot)\n\n const defaultLocale = overrides?.defaultLocale\n ?? fileConfig?.sourceLocale\n ?? 'en'\n\n const locales = overrides?.locales\n ?? fileConfig?.locales\n ?? [defaultLocale]\n\n const compiledDir = overrides?.compiledDir\n ?? fileConfig?.compileOutDir\n ?? './src/locales/compiled'\n\n const serverModuleOutDir = overrides?.serverModuleOutDir\n ?? join('node_modules', '.fluenti')\n\n const resolved: ResolvedFluentConfig = {\n locales,\n defaultLocale,\n compiledDir,\n serverModule: overrides?.serverModule ?? null,\n serverModuleOutDir,\n }\n if (overrides?.resolveLocale) resolved.resolveLocale = overrides.resolveLocale\n if (overrides?.dateFormats) resolved.dateFormats = overrides.dateFormats\n if (overrides?.numberFormats) resolved.numberFormats = overrides.numberFormats\n if (overrides?.fallbackChain) resolved.fallbackChain = overrides.fallbackChain\n return resolved\n}\n\n/**\n * Attempt to read fluenti.config.ts synchronously.\n * Returns null if file doesn't exist or can't be parsed.\n */\nfunction readFluentConfigSync(projectRoot: string): FluentiConfig | null {\n const jiti = createJiti(runtimeModulePath, {\n moduleCache: false,\n interopDefault: true,\n })\n const candidates = [\n 'fluenti.config.ts',\n 'fluenti.config.js',\n 'fluenti.config.mjs',\n ]\n\n for (const name of candidates) {\n const configPath = resolve(projectRoot, name)\n if (existsSync(configPath)) {\n try {\n return normalizeLoadedConfig(jiti(configPath) as FluentiConfig | { default?: FluentiConfig })\n } catch {\n const rewritten = tryLoadConfigViaDefineConfigShim(configPath, jiti)\n if (rewritten) {\n return rewritten\n }\n return null\n }\n }\n }\n\n return null\n}\n\nfunction tryLoadConfigViaDefineConfigShim(\n configPath: string,\n jiti: (path: string) => unknown,\n): FluentiConfig | null {\n const source = readFileSync(configPath, 'utf8')\n const importMatch = source.match(\n /import\\s*\\{\\s*defineConfig(?:\\s+as\\s+([A-Za-z_$][\\w$]*))?\\s*\\}\\s*from\\s*['\"]@fluenti\\/cli['\"]\\s*;?/,\n )\n if (!importMatch) {\n return null\n }\n\n const helperName = importMatch[1] ?? 'defineConfig'\n const rewrittenSource = source.replace(importMatch[0], '')\n const tempPath = join(\n dirname(configPath),\n `.${basename(configPath, extname(configPath))}.next-plugin-read-config${extname(configPath) || '.ts'}`,\n )\n\n writeFileSync(tempPath, `const ${helperName} = (config) => config\\n${rewrittenSource}`, 'utf8')\n\n try {\n return normalizeLoadedConfig(jiti(tempPath) as FluentiConfig | { default?: FluentiConfig })\n } catch {\n return null\n } finally {\n rmSync(tempPath, { force: true })\n }\n}\n\nfunction normalizeLoadedConfig(\n mod: FluentiConfig | { default?: FluentiConfig },\n): FluentiConfig {\n return typeof mod === 'object' && mod !== null && 'default' in mod\n ? (mod.default ?? {}) as FluentiConfig\n : mod as FluentiConfig\n}\n","import { writeFileSync, mkdirSync, existsSync } from 'node:fs'\nimport { resolve, relative } from 'node:path'\nimport { validateLocale } from '@fluenti/core'\nimport type { ResolvedFluentConfig } from './types'\n\n/**\n * Generate the server module that provides:\n * - setLocale / getI18n\n * - Trans / Plural / Select / DateTime / NumberFormat (server components)\n * - I18nProvider (async server component for layouts)\n *\n * @returns Absolute path to the generated server module.\n */\nexport function generateServerModule(\n projectRoot: string,\n config: ResolvedFluentConfig,\n): string {\n if (config.serverModule) {\n return resolve(projectRoot, config.serverModule)\n }\n\n const outDir = resolve(projectRoot, config.serverModuleOutDir)\n const outPath = resolve(outDir, 'server.js')\n const dtsPath = resolve(outDir, 'server.d.ts')\n\n if (!existsSync(outDir)) {\n mkdirSync(outDir, { recursive: true })\n }\n\n for (const locale of config.locales) {\n validateLocale(locale, 'next-plugin')\n }\n\n const compiledDirAbs = resolve(projectRoot, config.compiledDir)\n const compiledRelative = toForwardSlash(relative(outDir, compiledDirAbs))\n\n const localeImports = config.locales\n .map((locale) => ` case '${locale}': return import('${compiledRelative}/${locale}')`)\n .join('\\n')\n\n const fallbackChainStr = config.fallbackChain\n ? JSON.stringify(config.fallbackChain)\n : 'undefined'\n\n // Generate a 'use client' provider that imports messages statically.\n // Messages contain functions (interpolation) which can't cross the RSC boundary.\n const clientProviderPath = resolve(outDir, 'client-provider.js')\n\n const clientStaticImports = config.locales\n .map((locale) => {\n const safe = locale.replace(/[^a-zA-Z0-9]/g, '_')\n return `import ${safe} from '${compiledRelative}/${locale}'`\n })\n .join('\\n')\n\n const clientAllMessagesEntries = config.locales\n .map((locale) => {\n const safe = locale.replace(/[^a-zA-Z0-9]/g, '_')\n return `'${locale}': ${safe}`\n })\n .join(', ')\n\n const clientProviderSource = `\"use client\";\n// Auto-generated by @fluenti/next — do not edit\nimport { createElement } from 'react'\nimport { I18nProvider } from '@fluenti/react'\n${clientStaticImports}\n\nconst __allMessages = { ${clientAllMessagesEntries} }\n\nexport function ClientI18nProvider({ locale, fallbackLocale, fallbackChain, children }) {\n return createElement(I18nProvider, { locale, fallbackLocale, messages: __allMessages, fallbackChain }, children)\n}\n`\n writeFileSync(clientProviderPath, clientProviderSource, 'utf-8')\n\n const resolveLocaleImport = config.resolveLocale\n ? (() => {\n const absPath = resolve(projectRoot, config.resolveLocale)\n const relPath = toForwardSlash(relative(outDir, absPath))\n return `import __resolveLocale from '${relPath}'`\n })()\n : null\n\n const resolveLocaleFn = config.resolveLocale\n ? `resolveLocale: __resolveLocale,`\n : `resolveLocale: async () => {\n try {\n const { cookies } = await import('next/headers')\n return (await cookies()).get('locale')?.value ?? '${config.defaultLocale}'\n } catch {\n return '${config.defaultLocale}'\n }\n },`\n\n const moduleSource = `// Auto-generated by @fluenti/next — do not edit\nimport { createServerI18n } from '@fluenti/react/server'\nimport { createElement } from 'react'\n${resolveLocaleImport ? `${resolveLocaleImport}\\n` : ''}\nconst serverI18n = createServerI18n({\n loadMessages: async (locale) => {\n switch (locale) {\n${localeImports}\n default: return import('${compiledRelative}/${config.defaultLocale}')\n }\n },\n fallbackLocale: '${config.defaultLocale}',\n fallbackChain: ${fallbackChainStr},\n ${resolveLocaleFn}\n})\n\nexport const setLocale = serverI18n.setLocale\nexport const getI18n = serverI18n.getI18n\nexport const t = (..._args) => {\n throw new Error(\n \"[fluenti] \\`t\\` imported from '@fluenti/next' is a compile-time API. \" +\n 'Use it only with the Fluenti loader inside an async server scope.',\n )\n}\nexport const Trans = serverI18n.Trans\nexport const Plural = serverI18n.Plural\nexport const Select = serverI18n.Select\nexport const DateTime = serverI18n.DateTime\nexport const NumberFormat = serverI18n.NumberFormat\n\n/**\n * Async server component for root layouts.\n *\n * Sets up both server-side (React.cache) and client-side (I18nProvider) i18n.\n */\nexport async function I18nProvider({ locale, children }) {\n const activeLocale = locale ?? '${config.defaultLocale}'\n\n // 1. Initialize server-side i18n (React.cache scoped)\n serverI18n.setLocale(activeLocale)\n await serverI18n.getI18n()\n\n // 2. Import the local 'use client' provider that has messages statically bundled.\n // Messages contain functions (interpolation) which can't be serialized across the RSC boundary.\n const { ClientI18nProvider } = await import('./client-provider.js')\n\n return createElement(ClientI18nProvider, {\n locale: activeLocale,\n fallbackLocale: '${config.defaultLocale}',\n fallbackChain: ${fallbackChainStr},\n }, children)\n}\n`\n\n const dtsSource = `// Auto-generated by @fluenti/next — do not edit\nimport type { ReactNode, ReactElement } from 'react'\nimport type { CompileTimeT, FluentInstanceExtended } from '@fluenti/core'\n\nexport declare function setLocale(locale: string): void\nexport declare function getI18n(): Promise<FluentInstanceExtended & { locale: string }>\nexport declare const t: CompileTimeT\n\nexport declare function Trans(props: {\n children: ReactNode\n id?: string\n context?: string\n comment?: string\n render?: (translation: ReactNode) => ReactNode\n}): Promise<ReactElement>\n\nexport declare function Plural(props: {\n value: number\n id?: string\n context?: string\n comment?: string\n zero?: ReactNode\n one?: ReactNode\n two?: ReactNode\n few?: ReactNode\n many?: ReactNode\n other: ReactNode\n offset?: number\n}): Promise<ReactElement>\n\nexport declare function Select(props: {\n value: string\n id?: string\n context?: string\n comment?: string\n other: ReactNode\n options?: Record<string, ReactNode>\n [key: string]: ReactNode | Record<string, ReactNode> | string | undefined\n}): Promise<ReactElement>\n\nexport declare function DateTime(props: {\n value: Date | number\n style?: string\n}): Promise<ReactElement>\n\nexport declare function NumberFormat(props: {\n value: number\n style?: string\n}): Promise<ReactElement>\n\nexport declare function I18nProvider(props: {\n locale?: string\n children: ReactNode\n}): Promise<ReactElement>\n`\n\n writeFileSync(outPath, moduleSource, 'utf-8')\n writeFileSync(dtsPath, dtsSource, 'utf-8')\n\n return outPath\n}\n\nfunction toForwardSlash(p: string): string {\n return p.split('\\\\').join('/')\n}\n","import { exec } from 'node:child_process'\nimport { existsSync } from 'node:fs'\nimport { resolve, dirname, join } from 'node:path'\nimport { createRequire } from 'node:module'\n\nexport interface DevRunnerOptions {\n cwd: string\n onSuccess?: () => void\n onError?: (err: Error) => void\n /** If true, reject the promise on failure instead of swallowing the error */\n throwOnError?: boolean\n /** Run only compile (skip extract). Useful for production builds where source is unchanged. */\n compileOnly?: boolean\n}\n\n/**\n * Walk up from `cwd` to find `node_modules/.bin/fluenti`.\n * Returns the absolute path or null if not found.\n */\nexport function resolveCliBin(cwd: string): string | null {\n let dir = cwd\n for (;;) {\n const bin = resolve(dir, 'node_modules/.bin/fluenti')\n if (existsSync(bin)) return bin\n const parent = dirname(dir)\n if (parent === dir) break\n dir = parent\n }\n return null\n}\n\n/**\n * Run compile in-process via `@fluenti/cli` (for compileOnly mode),\n * or fall back to shell-out for extract + compile (dev mode).\n */\nexport async function runExtractCompile(options: DevRunnerOptions): Promise<void> {\n if (options.compileOnly) {\n try {\n // Resolve @fluenti/cli from the project's cwd (not from this package's location)\n // using createRequire so pnpm's strict node_modules layout works correctly.\n // Use require() (not import()) to load @fluenti/cli — avoids CJS/ESM interop\n // issues when dynamic import() loads minified CJS with chunk requires.\n const projectRequire = createRequire(join(options.cwd, 'package.json'))\n const { runCompile } = projectRequire('@fluenti/cli')\n await runCompile(options.cwd)\n console.log('[fluenti] Compiling... done')\n options.onSuccess?.()\n return\n } catch (e) {\n const error = e instanceof Error ? e : new Error(String(e))\n if (options.throwOnError) throw error\n console.warn('[fluenti] Compile failed:', error.message)\n options.onError?.(error)\n return\n }\n }\n\n // Dev mode: shell out for extract + compile\n const bin = resolveCliBin(options.cwd)\n if (!bin) {\n const msg = '[fluenti] CLI not found — skipping auto-compile. Install @fluenti/cli as a devDependency.'\n if (options.throwOnError) {\n return Promise.reject(new Error(msg))\n }\n console.warn(msg)\n return Promise.resolve()\n }\n\n const command = `${bin} extract && ${bin} compile`\n return new Promise<void>((resolve, reject) => {\n exec(\n command,\n { cwd: options.cwd },\n (err, _stdout, stderr) => {\n if (err) {\n const error = new Error(stderr || err.message)\n if (options.throwOnError) {\n reject(error)\n return\n }\n console.warn('[fluenti] Extract/compile failed:', error.message)\n options.onError?.(error)\n } else {\n console.log('[fluenti] Extracting and compiling... done')\n options.onSuccess?.()\n }\n resolve()\n },\n )\n })\n}\n\n/**\n * Create a debounced runner that collapses rapid calls.\n *\n * - If called while idle, schedules a run after `delay` ms.\n * - If called while a run is in progress, marks a pending rerun.\n * - Never runs concurrently.\n */\nexport function createDebouncedRunner(\n options: DevRunnerOptions,\n delay = 300,\n): () => void {\n let timer: ReturnType<typeof setTimeout> | null = null\n let running = false\n let pendingRerun = false\n\n async function execute(): Promise<void> {\n running = true\n try {\n await runExtractCompile(options)\n } finally {\n running = false\n if (pendingRerun) {\n pendingRerun = false\n schedule()\n }\n }\n }\n\n function schedule(): void {\n if (timer !== null) {\n clearTimeout(timer)\n }\n timer = setTimeout(() => {\n timer = null\n if (running) {\n pendingRerun = true\n } else {\n execute()\n }\n }, delay)\n }\n\n return schedule\n}\n","import { existsSync } from 'node:fs'\nimport { execSync } from 'node:child_process'\nimport { resolve, dirname } from 'node:path'\nimport type { WithFluentConfig } from './types'\nimport { resolveConfig } from './read-config'\nimport { generateServerModule } from './generate-server-module'\nimport { createDebouncedRunner } from './dev-runner'\n\ntype NextConfig = Record<string, unknown>\n\n/**\n * Wrap your Next.js config with Fluenti support.\n *\n * Adds a webpack loader that transforms `t\\`\\`` and `t()` calls,\n * and generates a server module for RSC i18n.\n *\n * @example\n * ```ts\n * // next.config.ts — function style (recommended)\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti()({ reactStrictMode: true })\n * ```\n *\n * @example\n * ```ts\n * // next.config.ts — direct style\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti({ reactStrictMode: true })\n * ```\n */\nexport function withFluenti(fluentConfig?: WithFluentConfig): (nextConfig?: NextConfig) => NextConfig\nexport function withFluenti(nextConfig: NextConfig): NextConfig\nexport function withFluenti(\n configOrNext?: WithFluentConfig | NextConfig,\n): NextConfig | ((nextConfig?: NextConfig) => NextConfig) {\n if (configOrNext && isNextConfig(configOrNext as NextConfig)) {\n return applyFluenti({}, configOrNext as NextConfig)\n }\n\n const fluentConfig = (configOrNext ?? {}) as WithFluentConfig\n return function wrappedConfig(nextConfig?: NextConfig): NextConfig {\n return applyFluenti(fluentConfig, nextConfig ?? {})\n }\n}\n\nfunction isNextConfig(obj: NextConfig): boolean {\n const nextKeys = [\n 'reactStrictMode', 'experimental', 'images', 'env', 'webpack',\n 'rewrites', 'redirects', 'headers', 'pageExtensions', 'output',\n 'basePath', 'i18n', 'trailingSlash', 'compiler', 'transpilePackages',\n ]\n return nextKeys.some((key) => key in obj)\n}\n\nfunction applyFluenti(\n fluentConfig: WithFluentConfig,\n nextConfig: NextConfig,\n): NextConfig {\n const projectRoot = process.cwd()\n const resolved = resolveConfig(projectRoot, fluentConfig)\n\n // Warn if compiled catalogs directory doesn't exist yet\n const compiledDir = resolve(projectRoot, resolved.compiledDir)\n if (!existsSync(compiledDir)) {\n console.warn(\n `\\n[fluenti] Compiled catalogs not found at ${resolved.compiledDir}.\\n` +\n `Run: npx fluenti extract && npx fluenti compile\\n`,\n )\n }\n\n // Generate server module for RSC\n const serverModulePath = generateServerModule(projectRoot, resolved)\n\n // Resolve the loader path — use import.meta.url for ESM compatibility\n const thisDir = typeof __dirname !== 'undefined'\n ? __dirname\n : dirname(new URL(import.meta.url).pathname)\n const loaderPath = resolve(thisDir, 'loader.js')\n\n const existingWebpack = nextConfig['webpack'] as\n | ((config: WebpackConfig, options: WebpackOptions) => WebpackConfig)\n | undefined\n\n let buildCompileRan = false\n\n return {\n ...nextConfig,\n webpack(config: WebpackConfig, options: WebpackOptions) {\n // Add fluenti loader (enforce: pre — runs before other loaders)\n config.module.rules.push({\n test: /\\.[jt]sx?$/,\n enforce: 'pre' as const,\n exclude: [/node_modules/, /\\.next/],\n use: [\n {\n loader: loaderPath,\n options: {\n serverModulePath,\n },\n },\n ],\n })\n\n // Add resolve alias so loader can import from generated server module\n config.resolve = config.resolve ?? {} as WebpackConfig['resolve']\n config.resolve.alias = config.resolve.alias ?? {}\n config.resolve.alias['@fluenti/next$'] = serverModulePath\n\n // Auto compile before production build (run once across server+client passes)\n const buildAutoCompile = fluentConfig.buildAutoCompile ?? true\n if (!options.dev && buildAutoCompile && !buildCompileRan) {\n buildCompileRan = true\n try {\n // Use node -e with dynamic import — avoids CLI binary resolution issues.\n // webpack() is sync, so we use execSync to block until compile finishes.\n const escapedRoot = projectRoot.replace(/\\\\/g, '\\\\\\\\').replace(/'/g, \"\\\\'\")\n execSync(\n `node --input-type=module -e \"const { runCompile } = await import('@fluenti/cli'); await runCompile('${escapedRoot}')\"`,\n { cwd: projectRoot, stdio: 'inherit' },\n )\n } catch {\n // @fluenti/cli not available or compile failed — skip silently\n }\n }\n\n // Auto extract+compile in dev mode\n const devAutoCompile = fluentConfig.devAutoCompile ?? true\n if (options.dev && devAutoCompile) {\n const devDelay = fluentConfig.devAutoCompileDelay ?? 1000\n const debouncedRun = createDebouncedRunner({ cwd: projectRoot }, devDelay)\n const compiledDirResolved = resolve(projectRoot, resolved.compiledDir)\n\n config.plugins = config.plugins ?? []\n config.plugins.push({\n apply(compiler: WebpackCompiler) {\n let isFirstBuild = true\n compiler.hooks.watchRun.tapAsync('fluenti-dev', (_compiler: WebpackCompiler, callback: () => void) => {\n if (isFirstBuild) {\n isFirstBuild = false\n debouncedRun()\n }\n const modifiedFiles = _compiler.modifiedFiles\n if (modifiedFiles) {\n const hasSourceChange = [...modifiedFiles].some((f: string) =>\n /\\.[jt]sx?$/.test(f)\n && !f.includes('node_modules')\n && !f.includes('.next')\n && !f.startsWith(compiledDirResolved),\n )\n if (hasSourceChange) {\n debouncedRun()\n }\n }\n callback()\n })\n },\n })\n }\n\n // Call user's webpack config if provided\n if (existingWebpack) {\n return existingWebpack(config, options)\n }\n\n return config\n },\n }\n}\n\n// Minimal webpack types for the config function\ninterface WebpackConfig {\n module: {\n rules: Array<{\n test: RegExp\n enforce?: 'pre' | 'post'\n exclude?: Array<RegExp>\n use: Array<{ loader: string; options: Record<string, unknown> }>\n }>\n }\n resolve: {\n alias?: Record<string, string>\n }\n plugins?: Array<{ apply(compiler: WebpackCompiler): void }>\n}\n\ninterface WebpackOptions {\n isServer: boolean\n dev: boolean\n}\n\ninterface WebpackCompiler {\n hooks: {\n watchRun: {\n tapAsync(name: string, cb: (compiler: WebpackCompiler, callback: () => void) => void): void\n }\n }\n modifiedFiles?: Set<string>\n}\n","/**\n * @fluenti/next — Next.js plugin for Fluenti\n *\n * Provides:\n * - `withFluenti()` — wraps next.config.ts with t`` transform support\n * - I18nProvider — async server component (exported from generated module)\n * - Webpack loader for strict, binding-aware tagged-template optimization\n *\n * @example\n * ```ts\n * // next.config.ts\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti()({ reactStrictMode: true })\n * ```\n *\n * @example\n * ```tsx\n * // app/layout.tsx — resolved by webpack alias to the generated module\n * import { I18nProvider } from '@fluenti/next'\n * ```\n */\nexport { withFluenti } from './with-fluenti'\nexport type { WithFluentConfig, I18nProviderProps } from './types'\n\n// ── Runtime stubs ────────────────────────────────────────────────────\n// TypeScript resolves types from this file (via package.json exports).\n// At runtime, webpack `resolve.alias` redirects `@fluenti/next$` to the\n// generated server module, so these stubs are never actually called in\n// a correctly configured project. They exist only to provide helpful\n// errors if `withFluenti()` is not configured.\n\nimport type { ReactNode, ReactElement } from 'react'\nimport type { CompileTimeT, FluentInstanceExtended } from '@fluenti/core'\n\nconst NOT_CONFIGURED =\n '[fluenti] `withFluenti()` must be configured in next.config.ts before importing from \"@fluenti/next\".'\n\nfunction throwNotConfigured(): never {\n throw new Error(NOT_CONFIGURED)\n}\n\n/** @see Generated module for the real implementation. */\nexport const setLocale: (locale: string) => void = throwNotConfigured\n/** @see Generated module for the real implementation. */\nexport const getI18n: () => Promise<FluentInstanceExtended & { locale: string }> = throwNotConfigured as () => Promise<FluentInstanceExtended & { locale: string }>\n/** @see Generated module for the real implementation. */\nexport const t: CompileTimeT = throwNotConfigured as unknown as CompileTimeT\n/** @see Generated module for the real implementation. */\nexport const Trans: (props: { children: ReactNode; id?: string; context?: string; comment?: string; render?: (translation: ReactNode) => ReactNode }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Trans\n/** @see Generated module for the real implementation. */\nexport const Plural: (props: { value: number; id?: string; context?: string; comment?: string; zero?: ReactNode; one?: ReactNode; two?: ReactNode; few?: ReactNode; many?: ReactNode; other: ReactNode; offset?: number }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Plural\n/** @see Generated module for the real implementation. */\nexport const Select: (props: { value: string; id?: string; context?: string; comment?: string; other: ReactNode; options?: Record<string, ReactNode>; [key: string]: ReactNode | Record<string, ReactNode> | string | undefined }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Select\n/** @see Generated module for the real implementation. */\nexport const DateTime: (props: { value: Date | number; style?: string }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof DateTime\n/** @see Generated module for the real implementation. */\nexport const NumberFormat: (props: { value: number; style?: string }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof NumberFormat\n/** @see Generated module for the real implementation. */\nexport const I18nProvider: (props: { locale?: string; children: ReactNode }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof I18nProvider\n"],"mappings":"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"}
|
package/dist/index.js
CHANGED
|
@@ -1,12 +1,12 @@
|
|
|
1
1
|
import { existsSync as e, mkdirSync as t, readFileSync as n, rmSync as r, writeFileSync as i } from "node:fs";
|
|
2
|
-
import {
|
|
3
|
-
import {
|
|
4
|
-
import {
|
|
5
|
-
import {
|
|
2
|
+
import { exec as a, execSync as o } from "node:child_process";
|
|
3
|
+
import { basename as s, dirname as c, extname as l, join as u, relative as d, resolve as f } from "node:path";
|
|
4
|
+
import { createRequire as p } from "node:module";
|
|
5
|
+
import { validateLocale as m } from "@fluenti/core";
|
|
6
6
|
//#region src/read-config.ts
|
|
7
|
-
var
|
|
8
|
-
function
|
|
9
|
-
let n =
|
|
7
|
+
var h = typeof __filename == "string" ? __filename : import.meta.url, { createJiti: g } = p(h)("jiti");
|
|
8
|
+
function _(e, t) {
|
|
9
|
+
let n = v(e), r = t?.defaultLocale ?? n?.sourceLocale ?? "en", i = t?.locales ?? n?.locales ?? [r], a = t?.compiledDir ?? n?.compileOutDir ?? "./src/locales/compiled", o = t?.serverModuleOutDir ?? u("node_modules", ".fluenti"), s = {
|
|
10
10
|
locales: i,
|
|
11
11
|
defaultLocale: r,
|
|
12
12
|
compiledDir: a,
|
|
@@ -15,8 +15,8 @@ function g(e, t) {
|
|
|
15
15
|
};
|
|
16
16
|
return t?.resolveLocale && (s.resolveLocale = t.resolveLocale), t?.dateFormats && (s.dateFormats = t.dateFormats), t?.numberFormats && (s.numberFormats = t.numberFormats), t?.fallbackChain && (s.fallbackChain = t.fallbackChain), s;
|
|
17
17
|
}
|
|
18
|
-
function
|
|
19
|
-
let n = h
|
|
18
|
+
function v(t) {
|
|
19
|
+
let n = g(h, {
|
|
20
20
|
moduleCache: !1,
|
|
21
21
|
interopDefault: !0
|
|
22
22
|
});
|
|
@@ -25,40 +25,40 @@ function _(t) {
|
|
|
25
25
|
"fluenti.config.js",
|
|
26
26
|
"fluenti.config.mjs"
|
|
27
27
|
]) {
|
|
28
|
-
let i =
|
|
28
|
+
let i = f(t, r);
|
|
29
29
|
if (e(i)) try {
|
|
30
|
-
return
|
|
30
|
+
return b(n(i));
|
|
31
31
|
} catch {
|
|
32
|
-
return
|
|
32
|
+
return y(i, n) || null;
|
|
33
33
|
}
|
|
34
34
|
}
|
|
35
35
|
return null;
|
|
36
36
|
}
|
|
37
|
-
function
|
|
38
|
-
let
|
|
39
|
-
if (!
|
|
40
|
-
let d =
|
|
37
|
+
function y(e, t) {
|
|
38
|
+
let a = n(e, "utf8"), o = a.match(/import\s*\{\s*defineConfig(?:\s+as\s+([A-Za-z_$][\w$]*))?\s*\}\s*from\s*['"]@fluenti\/cli['"]\s*;?/);
|
|
39
|
+
if (!o) return null;
|
|
40
|
+
let d = o[1] ?? "defineConfig", f = a.replace(o[0], ""), p = u(c(e), `.${s(e, l(e))}.next-plugin-read-config${l(e) || ".ts"}`);
|
|
41
41
|
i(p, `const ${d} = (config) => config\n${f}`, "utf8");
|
|
42
42
|
try {
|
|
43
|
-
return
|
|
43
|
+
return b(t(p));
|
|
44
44
|
} catch {
|
|
45
45
|
return null;
|
|
46
46
|
} finally {
|
|
47
47
|
r(p, { force: !0 });
|
|
48
48
|
}
|
|
49
49
|
}
|
|
50
|
-
function
|
|
50
|
+
function b(e) {
|
|
51
51
|
return typeof e == "object" && e && "default" in e ? e.default ?? {} : e;
|
|
52
52
|
}
|
|
53
53
|
//#endregion
|
|
54
54
|
//#region src/generate-server-module.ts
|
|
55
|
-
function
|
|
56
|
-
if (r.serverModule) return
|
|
57
|
-
let a =
|
|
55
|
+
function x(n, r) {
|
|
56
|
+
if (r.serverModule) return f(n, r.serverModule);
|
|
57
|
+
let a = f(n, r.serverModuleOutDir), o = f(a, "server.js"), s = f(a, "server.d.ts");
|
|
58
58
|
e(a) || t(a, { recursive: !0 });
|
|
59
|
-
for (let e of r.locales)
|
|
60
|
-
let c =
|
|
61
|
-
i(
|
|
59
|
+
for (let e of r.locales) m(e, "next-plugin");
|
|
60
|
+
let c = S(d(a, f(n, r.compiledDir))), l = r.locales.map((e) => ` case '${e}': return import('${c}/${e}')`).join("\n"), u = r.fallbackChain ? JSON.stringify(r.fallbackChain) : "undefined";
|
|
61
|
+
i(f(a, "client-provider.js"), `"use client";
|
|
62
62
|
// Auto-generated by @fluenti/next — do not edit
|
|
63
63
|
import { createElement } from 'react'
|
|
64
64
|
import { I18nProvider } from '@fluenti/react'
|
|
@@ -70,7 +70,7 @@ export function ClientI18nProvider({ locale, fallbackLocale, fallbackChain, chil
|
|
|
70
70
|
return createElement(I18nProvider, { locale, fallbackLocale, messages: __allMessages, fallbackChain }, children)
|
|
71
71
|
}
|
|
72
72
|
`, "utf-8");
|
|
73
|
-
let
|
|
73
|
+
let p = r.resolveLocale ? `import __resolveLocale from '${S(d(a, f(n, r.resolveLocale)))}'` : null, h = r.resolveLocale ? "resolveLocale: __resolveLocale," : `resolveLocale: async () => {
|
|
74
74
|
try {
|
|
75
75
|
const { cookies } = await import('next/headers')
|
|
76
76
|
return (await cookies()).get('locale')?.value ?? '${r.defaultLocale}'
|
|
@@ -81,16 +81,16 @@ export function ClientI18nProvider({ locale, fallbackLocale, fallbackChain, chil
|
|
|
81
81
|
return i(o, `// Auto-generated by @fluenti/next — do not edit
|
|
82
82
|
import { createServerI18n } from '@fluenti/react/server'
|
|
83
83
|
import { createElement } from 'react'
|
|
84
|
-
${
|
|
84
|
+
${p ? `${p}\n` : ""}
|
|
85
85
|
const serverI18n = createServerI18n({
|
|
86
86
|
loadMessages: async (locale) => {
|
|
87
87
|
switch (locale) {
|
|
88
|
-
${
|
|
88
|
+
${l}
|
|
89
89
|
default: return import('${c}/${r.defaultLocale}')
|
|
90
90
|
}
|
|
91
91
|
},
|
|
92
92
|
fallbackLocale: '${r.defaultLocale}',
|
|
93
|
-
fallbackChain: ${
|
|
93
|
+
fallbackChain: ${u},
|
|
94
94
|
${h}
|
|
95
95
|
})
|
|
96
96
|
|
|
@@ -127,24 +127,85 @@ export async function I18nProvider({ locale, children }) {
|
|
|
127
127
|
return createElement(ClientI18nProvider, {
|
|
128
128
|
locale: activeLocale,
|
|
129
129
|
fallbackLocale: '${r.defaultLocale}',
|
|
130
|
-
fallbackChain: ${
|
|
130
|
+
fallbackChain: ${u},
|
|
131
131
|
}, children)
|
|
132
132
|
}
|
|
133
133
|
`, "utf-8"), i(s, "// Auto-generated by @fluenti/next — do not edit\nimport type { ReactNode, ReactElement } from 'react'\nimport type { CompileTimeT, FluentInstanceExtended } from '@fluenti/core'\n\nexport declare function setLocale(locale: string): void\nexport declare function getI18n(): Promise<FluentInstanceExtended & { locale: string }>\nexport declare const t: CompileTimeT\n\nexport declare function Trans(props: {\n children: ReactNode\n id?: string\n context?: string\n comment?: string\n render?: (translation: ReactNode) => ReactNode\n}): Promise<ReactElement>\n\nexport declare function Plural(props: {\n value: number\n id?: string\n context?: string\n comment?: string\n zero?: ReactNode\n one?: ReactNode\n two?: ReactNode\n few?: ReactNode\n many?: ReactNode\n other: ReactNode\n offset?: number\n}): Promise<ReactElement>\n\nexport declare function Select(props: {\n value: string\n id?: string\n context?: string\n comment?: string\n other: ReactNode\n options?: Record<string, ReactNode>\n [key: string]: ReactNode | Record<string, ReactNode> | string | undefined\n}): Promise<ReactElement>\n\nexport declare function DateTime(props: {\n value: Date | number\n style?: string\n}): Promise<ReactElement>\n\nexport declare function NumberFormat(props: {\n value: number\n style?: string\n}): Promise<ReactElement>\n\nexport declare function I18nProvider(props: {\n locale?: string\n children: ReactNode\n}): Promise<ReactElement>\n", "utf-8"), o;
|
|
134
134
|
}
|
|
135
|
-
function
|
|
135
|
+
function S(e) {
|
|
136
136
|
return e.split("\\").join("/");
|
|
137
137
|
}
|
|
138
138
|
//#endregion
|
|
139
|
+
//#region src/dev-runner.ts
|
|
140
|
+
function C(t) {
|
|
141
|
+
let n = t;
|
|
142
|
+
for (;;) {
|
|
143
|
+
let t = f(n, "node_modules/.bin/fluenti");
|
|
144
|
+
if (e(t)) return t;
|
|
145
|
+
let r = c(n);
|
|
146
|
+
if (r === n) break;
|
|
147
|
+
n = r;
|
|
148
|
+
}
|
|
149
|
+
return null;
|
|
150
|
+
}
|
|
151
|
+
async function w(e) {
|
|
152
|
+
if (e.compileOnly) try {
|
|
153
|
+
let { runCompile: t } = p(u(e.cwd, "package.json"))("@fluenti/cli");
|
|
154
|
+
await t(e.cwd), console.log("[fluenti] Compiling... done"), e.onSuccess?.();
|
|
155
|
+
return;
|
|
156
|
+
} catch (t) {
|
|
157
|
+
let n = t instanceof Error ? t : Error(String(t));
|
|
158
|
+
if (e.throwOnError) throw n;
|
|
159
|
+
console.warn("[fluenti] Compile failed:", n.message), e.onError?.(n);
|
|
160
|
+
return;
|
|
161
|
+
}
|
|
162
|
+
let t = C(e.cwd);
|
|
163
|
+
if (!t) {
|
|
164
|
+
let t = "[fluenti] CLI not found — skipping auto-compile. Install @fluenti/cli as a devDependency.";
|
|
165
|
+
return e.throwOnError ? Promise.reject(Error(t)) : (console.warn(t), Promise.resolve());
|
|
166
|
+
}
|
|
167
|
+
let n = `${t} extract && ${t} compile`;
|
|
168
|
+
return new Promise((t, r) => {
|
|
169
|
+
a(n, { cwd: e.cwd }, (n, i, a) => {
|
|
170
|
+
if (n) {
|
|
171
|
+
let t = Error(a || n.message);
|
|
172
|
+
if (e.throwOnError) {
|
|
173
|
+
r(t);
|
|
174
|
+
return;
|
|
175
|
+
}
|
|
176
|
+
console.warn("[fluenti] Extract/compile failed:", t.message), e.onError?.(t);
|
|
177
|
+
} else console.log("[fluenti] Extracting and compiling... done"), e.onSuccess?.();
|
|
178
|
+
t();
|
|
179
|
+
});
|
|
180
|
+
});
|
|
181
|
+
}
|
|
182
|
+
function T(e, t = 300) {
|
|
183
|
+
let n = null, r = !1, i = !1;
|
|
184
|
+
async function a() {
|
|
185
|
+
r = !0;
|
|
186
|
+
try {
|
|
187
|
+
await w(e);
|
|
188
|
+
} finally {
|
|
189
|
+
r = !1, i && (i = !1, o());
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
function o() {
|
|
193
|
+
n !== null && clearTimeout(n), n = setTimeout(() => {
|
|
194
|
+
n = null, r ? i = !0 : a();
|
|
195
|
+
}, t);
|
|
196
|
+
}
|
|
197
|
+
return o;
|
|
198
|
+
}
|
|
199
|
+
//#endregion
|
|
139
200
|
//#region src/with-fluenti.ts
|
|
140
|
-
function
|
|
141
|
-
if (e &&
|
|
201
|
+
function E(e) {
|
|
202
|
+
if (e && D(e)) return O({}, e);
|
|
142
203
|
let t = e ?? {};
|
|
143
204
|
return function(e) {
|
|
144
|
-
return
|
|
205
|
+
return O(t, e ?? {});
|
|
145
206
|
};
|
|
146
207
|
}
|
|
147
|
-
function
|
|
208
|
+
function D(e) {
|
|
148
209
|
return [
|
|
149
210
|
"reactStrictMode",
|
|
150
211
|
"experimental",
|
|
@@ -163,10 +224,10 @@ function C(e) {
|
|
|
163
224
|
"transpilePackages"
|
|
164
225
|
].some((t) => t in e);
|
|
165
226
|
}
|
|
166
|
-
function
|
|
167
|
-
let r = process.cwd(), i =
|
|
168
|
-
e(
|
|
169
|
-
let a =
|
|
227
|
+
function O(t, n) {
|
|
228
|
+
let r = process.cwd(), i = _(r, t);
|
|
229
|
+
e(f(r, i.compiledDir)) || console.warn(`\n[fluenti] Compiled catalogs not found at ${i.compiledDir}.\nRun: npx fluenti extract && npx fluenti compile\n`);
|
|
230
|
+
let a = x(r, i), s = f(typeof __dirname < "u" ? __dirname : c(new URL(import.meta.url).pathname), "loader.js"), l = n.webpack, u = !1;
|
|
170
231
|
return {
|
|
171
232
|
...n,
|
|
172
233
|
webpack(e, n) {
|
|
@@ -179,30 +240,40 @@ function w(t, n) {
|
|
|
179
240
|
options: { serverModulePath: a }
|
|
180
241
|
}]
|
|
181
242
|
}), e.resolve = e.resolve ?? {}, e.resolve.alias = e.resolve.alias ?? {}, e.resolve.alias["@fluenti/next$"] = a;
|
|
182
|
-
let
|
|
183
|
-
if (n.dev &&
|
|
184
|
-
|
|
243
|
+
let c = t.buildAutoCompile ?? !0;
|
|
244
|
+
if (!n.dev && c && !u) {
|
|
245
|
+
u = !0;
|
|
246
|
+
try {
|
|
247
|
+
o(`node --input-type=module -e "const { runCompile } = await import('@fluenti/cli'); await runCompile('${r.replace(/\\/g, "\\\\").replace(/'/g, "\\'")}')"`, {
|
|
248
|
+
cwd: r,
|
|
249
|
+
stdio: "inherit"
|
|
250
|
+
});
|
|
251
|
+
} catch {}
|
|
252
|
+
}
|
|
253
|
+
let d = t.devAutoCompile ?? !0;
|
|
254
|
+
if (n.dev && d) {
|
|
255
|
+
let n = t.devAutoCompileDelay ?? 1e3, a = T({ cwd: r }, n), o = f(r, i.compiledDir);
|
|
185
256
|
e.plugins = e.plugins ?? [], e.plugins.push({ apply(e) {
|
|
186
|
-
let
|
|
187
|
-
e.hooks.watchRun.tapAsync("fluenti-dev", (e,
|
|
188
|
-
|
|
189
|
-
let
|
|
190
|
-
|
|
257
|
+
let t = !0;
|
|
258
|
+
e.hooks.watchRun.tapAsync("fluenti-dev", (e, n) => {
|
|
259
|
+
t && (t = !1, a());
|
|
260
|
+
let r = e.modifiedFiles;
|
|
261
|
+
r && [...r].some((e) => /\.[jt]sx?$/.test(e) && !e.includes("node_modules") && !e.includes(".next") && !e.startsWith(o)) && a(), n();
|
|
191
262
|
});
|
|
192
263
|
} });
|
|
193
264
|
}
|
|
194
|
-
return
|
|
265
|
+
return l ? l(e, n) : e;
|
|
195
266
|
}
|
|
196
267
|
};
|
|
197
268
|
}
|
|
198
269
|
//#endregion
|
|
199
270
|
//#region src/index.ts
|
|
200
|
-
var
|
|
201
|
-
function
|
|
202
|
-
throw Error(
|
|
271
|
+
var k = "[fluenti] `withFluenti()` must be configured in next.config.ts before importing from \"@fluenti/next\".";
|
|
272
|
+
function A() {
|
|
273
|
+
throw Error(k);
|
|
203
274
|
}
|
|
204
|
-
var
|
|
275
|
+
var j = A, M = A, N = A, P = A, F = A, I = A, L = A, R = A, z = A;
|
|
205
276
|
//#endregion
|
|
206
|
-
export {
|
|
277
|
+
export { L as DateTime, z as I18nProvider, R as NumberFormat, F as Plural, I as Select, P as Trans, M as getI18n, j as setLocale, N as t, E as withFluenti };
|
|
207
278
|
|
|
208
279
|
//# sourceMappingURL=index.js.map
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","names":[],"sources":["../src/read-config.ts","../src/generate-server-module.ts","../src/with-fluenti.ts","../src/index.ts"],"sourcesContent":["import { existsSync, readFileSync, rmSync, writeFileSync } from 'node:fs'\nimport { createRequire } from 'node:module'\nimport { basename, dirname, extname, join, resolve } from 'node:path'\nimport type { FluentiConfig } from '@fluenti/core'\nimport type { WithFluentConfig, ResolvedFluentConfig } from './types'\n\nconst runtimeModulePath = typeof __filename === 'string'\n ? __filename\n : import.meta.url\nconst require = createRequire(runtimeModulePath)\nconst { createJiti } = require('jiti') as {\n createJiti: (\n url: string,\n options?: { moduleCache?: boolean; interopDefault?: boolean },\n ) => (path: string) => unknown\n}\n\n/**\n * Read fluenti.config.ts and merge with withFluenti() overrides.\n */\nexport function resolveConfig(\n projectRoot: string,\n overrides?: WithFluentConfig,\n): ResolvedFluentConfig {\n const fileConfig = readFluentConfigSync(projectRoot)\n\n const defaultLocale = overrides?.defaultLocale\n ?? fileConfig?.sourceLocale\n ?? 'en'\n\n const locales = overrides?.locales\n ?? fileConfig?.locales\n ?? [defaultLocale]\n\n const compiledDir = overrides?.compiledDir\n ?? fileConfig?.compileOutDir\n ?? './src/locales/compiled'\n\n const serverModuleOutDir = overrides?.serverModuleOutDir\n ?? join('node_modules', '.fluenti')\n\n const resolved: ResolvedFluentConfig = {\n locales,\n defaultLocale,\n compiledDir,\n serverModule: overrides?.serverModule ?? null,\n serverModuleOutDir,\n }\n if (overrides?.resolveLocale) resolved.resolveLocale = overrides.resolveLocale\n if (overrides?.dateFormats) resolved.dateFormats = overrides.dateFormats\n if (overrides?.numberFormats) resolved.numberFormats = overrides.numberFormats\n if (overrides?.fallbackChain) resolved.fallbackChain = overrides.fallbackChain\n return resolved\n}\n\n/**\n * Attempt to read fluenti.config.ts synchronously.\n * Returns null if file doesn't exist or can't be parsed.\n */\nfunction readFluentConfigSync(projectRoot: string): FluentiConfig | null {\n const jiti = createJiti(runtimeModulePath, {\n moduleCache: false,\n interopDefault: true,\n })\n const candidates = [\n 'fluenti.config.ts',\n 'fluenti.config.js',\n 'fluenti.config.mjs',\n ]\n\n for (const name of candidates) {\n const configPath = resolve(projectRoot, name)\n if (existsSync(configPath)) {\n try {\n return normalizeLoadedConfig(jiti(configPath) as FluentiConfig | { default?: FluentiConfig })\n } catch {\n const rewritten = tryLoadConfigViaDefineConfigShim(configPath, jiti)\n if (rewritten) {\n return rewritten\n }\n return null\n }\n }\n }\n\n return null\n}\n\nfunction tryLoadConfigViaDefineConfigShim(\n configPath: string,\n jiti: (path: string) => unknown,\n): FluentiConfig | null {\n const source = readFileSync(configPath, 'utf8')\n const importMatch = source.match(\n /import\\s*\\{\\s*defineConfig(?:\\s+as\\s+([A-Za-z_$][\\w$]*))?\\s*\\}\\s*from\\s*['\"]@fluenti\\/cli['\"]\\s*;?/,\n )\n if (!importMatch) {\n return null\n }\n\n const helperName = importMatch[1] ?? 'defineConfig'\n const rewrittenSource = source.replace(importMatch[0], '')\n const tempPath = join(\n dirname(configPath),\n `.${basename(configPath, extname(configPath))}.next-plugin-read-config${extname(configPath) || '.ts'}`,\n )\n\n writeFileSync(tempPath, `const ${helperName} = (config) => config\\n${rewrittenSource}`, 'utf8')\n\n try {\n return normalizeLoadedConfig(jiti(tempPath) as FluentiConfig | { default?: FluentiConfig })\n } catch {\n return null\n } finally {\n rmSync(tempPath, { force: true })\n }\n}\n\nfunction normalizeLoadedConfig(\n mod: FluentiConfig | { default?: FluentiConfig },\n): FluentiConfig {\n return typeof mod === 'object' && mod !== null && 'default' in mod\n ? (mod.default ?? {}) as FluentiConfig\n : mod as FluentiConfig\n}\n","import { writeFileSync, mkdirSync, existsSync } from 'node:fs'\nimport { resolve, relative } from 'node:path'\nimport { validateLocale } from '@fluenti/core'\nimport type { ResolvedFluentConfig } from './types'\n\n/**\n * Generate the server module that provides:\n * - setLocale / getI18n\n * - Trans / Plural / Select / DateTime / NumberFormat (server components)\n * - I18nProvider (async server component for layouts)\n *\n * @returns Absolute path to the generated server module.\n */\nexport function generateServerModule(\n projectRoot: string,\n config: ResolvedFluentConfig,\n): string {\n if (config.serverModule) {\n return resolve(projectRoot, config.serverModule)\n }\n\n const outDir = resolve(projectRoot, config.serverModuleOutDir)\n const outPath = resolve(outDir, 'server.js')\n const dtsPath = resolve(outDir, 'server.d.ts')\n\n if (!existsSync(outDir)) {\n mkdirSync(outDir, { recursive: true })\n }\n\n for (const locale of config.locales) {\n validateLocale(locale, 'next-plugin')\n }\n\n const compiledDirAbs = resolve(projectRoot, config.compiledDir)\n const compiledRelative = toForwardSlash(relative(outDir, compiledDirAbs))\n\n const localeImports = config.locales\n .map((locale) => ` case '${locale}': return import('${compiledRelative}/${locale}')`)\n .join('\\n')\n\n const fallbackChainStr = config.fallbackChain\n ? JSON.stringify(config.fallbackChain)\n : 'undefined'\n\n // Generate a 'use client' provider that imports messages statically.\n // Messages contain functions (interpolation) which can't cross the RSC boundary.\n const clientProviderPath = resolve(outDir, 'client-provider.js')\n\n const clientStaticImports = config.locales\n .map((locale) => {\n const safe = locale.replace(/[^a-zA-Z0-9]/g, '_')\n return `import ${safe} from '${compiledRelative}/${locale}'`\n })\n .join('\\n')\n\n const clientAllMessagesEntries = config.locales\n .map((locale) => {\n const safe = locale.replace(/[^a-zA-Z0-9]/g, '_')\n return `'${locale}': ${safe}`\n })\n .join(', ')\n\n const clientProviderSource = `\"use client\";\n// Auto-generated by @fluenti/next — do not edit\nimport { createElement } from 'react'\nimport { I18nProvider } from '@fluenti/react'\n${clientStaticImports}\n\nconst __allMessages = { ${clientAllMessagesEntries} }\n\nexport function ClientI18nProvider({ locale, fallbackLocale, fallbackChain, children }) {\n return createElement(I18nProvider, { locale, fallbackLocale, messages: __allMessages, fallbackChain }, children)\n}\n`\n writeFileSync(clientProviderPath, clientProviderSource, 'utf-8')\n\n const resolveLocaleImport = config.resolveLocale\n ? `import __resolveLocale from '${config.resolveLocale}'`\n : null\n\n const resolveLocaleFn = config.resolveLocale\n ? `resolveLocale: __resolveLocale,`\n : `resolveLocale: async () => {\n try {\n const { cookies } = await import('next/headers')\n return (await cookies()).get('locale')?.value ?? '${config.defaultLocale}'\n } catch {\n return '${config.defaultLocale}'\n }\n },`\n\n const moduleSource = `// Auto-generated by @fluenti/next — do not edit\nimport { createServerI18n } from '@fluenti/react/server'\nimport { createElement } from 'react'\n${resolveLocaleImport ? `${resolveLocaleImport}\\n` : ''}\nconst serverI18n = createServerI18n({\n loadMessages: async (locale) => {\n switch (locale) {\n${localeImports}\n default: return import('${compiledRelative}/${config.defaultLocale}')\n }\n },\n fallbackLocale: '${config.defaultLocale}',\n fallbackChain: ${fallbackChainStr},\n ${resolveLocaleFn}\n})\n\nexport const setLocale = serverI18n.setLocale\nexport const getI18n = serverI18n.getI18n\nexport const t = (..._args) => {\n throw new Error(\n \"[fluenti] \\`t\\` imported from '@fluenti/next' is a compile-time API. \" +\n 'Use it only with the Fluenti loader inside an async server scope.',\n )\n}\nexport const Trans = serverI18n.Trans\nexport const Plural = serverI18n.Plural\nexport const Select = serverI18n.Select\nexport const DateTime = serverI18n.DateTime\nexport const NumberFormat = serverI18n.NumberFormat\n\n/**\n * Async server component for root layouts.\n *\n * Sets up both server-side (React.cache) and client-side (I18nProvider) i18n.\n */\nexport async function I18nProvider({ locale, children }) {\n const activeLocale = locale ?? '${config.defaultLocale}'\n\n // 1. Initialize server-side i18n (React.cache scoped)\n serverI18n.setLocale(activeLocale)\n await serverI18n.getI18n()\n\n // 2. Import the local 'use client' provider that has messages statically bundled.\n // Messages contain functions (interpolation) which can't be serialized across the RSC boundary.\n const { ClientI18nProvider } = await import('./client-provider.js')\n\n return createElement(ClientI18nProvider, {\n locale: activeLocale,\n fallbackLocale: '${config.defaultLocale}',\n fallbackChain: ${fallbackChainStr},\n }, children)\n}\n`\n\n const dtsSource = `// Auto-generated by @fluenti/next — do not edit\nimport type { ReactNode, ReactElement } from 'react'\nimport type { CompileTimeT, FluentInstanceExtended } from '@fluenti/core'\n\nexport declare function setLocale(locale: string): void\nexport declare function getI18n(): Promise<FluentInstanceExtended & { locale: string }>\nexport declare const t: CompileTimeT\n\nexport declare function Trans(props: {\n children: ReactNode\n id?: string\n context?: string\n comment?: string\n render?: (translation: ReactNode) => ReactNode\n}): Promise<ReactElement>\n\nexport declare function Plural(props: {\n value: number\n id?: string\n context?: string\n comment?: string\n zero?: ReactNode\n one?: ReactNode\n two?: ReactNode\n few?: ReactNode\n many?: ReactNode\n other: ReactNode\n offset?: number\n}): Promise<ReactElement>\n\nexport declare function Select(props: {\n value: string\n id?: string\n context?: string\n comment?: string\n other: ReactNode\n options?: Record<string, ReactNode>\n [key: string]: ReactNode | Record<string, ReactNode> | string | undefined\n}): Promise<ReactElement>\n\nexport declare function DateTime(props: {\n value: Date | number\n style?: string\n}): Promise<ReactElement>\n\nexport declare function NumberFormat(props: {\n value: number\n style?: string\n}): Promise<ReactElement>\n\nexport declare function I18nProvider(props: {\n locale?: string\n children: ReactNode\n}): Promise<ReactElement>\n`\n\n writeFileSync(outPath, moduleSource, 'utf-8')\n writeFileSync(dtsPath, dtsSource, 'utf-8')\n\n return outPath\n}\n\nfunction toForwardSlash(p: string): string {\n return p.split('\\\\').join('/')\n}\n","import { existsSync } from 'node:fs'\nimport { resolve, dirname } from 'node:path'\nimport type { WithFluentConfig } from './types'\nimport { resolveConfig } from './read-config'\nimport { generateServerModule } from './generate-server-module'\nimport { createDebouncedRunner } from '@fluenti/core/internal'\n\ntype NextConfig = Record<string, unknown>\n\n/**\n * Wrap your Next.js config with Fluenti support.\n *\n * Adds a webpack loader that transforms `t\\`\\`` and `t()` calls,\n * and generates a server module for RSC i18n.\n *\n * @example\n * ```ts\n * // next.config.ts — function style (recommended)\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti()({ reactStrictMode: true })\n * ```\n *\n * @example\n * ```ts\n * // next.config.ts — direct style\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti({ reactStrictMode: true })\n * ```\n */\nexport function withFluenti(fluentConfig?: WithFluentConfig): (nextConfig?: NextConfig) => NextConfig\nexport function withFluenti(nextConfig: NextConfig): NextConfig\nexport function withFluenti(\n configOrNext?: WithFluentConfig | NextConfig,\n): NextConfig | ((nextConfig?: NextConfig) => NextConfig) {\n if (configOrNext && isNextConfig(configOrNext as NextConfig)) {\n return applyFluenti({}, configOrNext as NextConfig)\n }\n\n const fluentConfig = (configOrNext ?? {}) as WithFluentConfig\n return function wrappedConfig(nextConfig?: NextConfig): NextConfig {\n return applyFluenti(fluentConfig, nextConfig ?? {})\n }\n}\n\nfunction isNextConfig(obj: NextConfig): boolean {\n const nextKeys = [\n 'reactStrictMode', 'experimental', 'images', 'env', 'webpack',\n 'rewrites', 'redirects', 'headers', 'pageExtensions', 'output',\n 'basePath', 'i18n', 'trailingSlash', 'compiler', 'transpilePackages',\n ]\n return nextKeys.some((key) => key in obj)\n}\n\nfunction applyFluenti(\n fluentConfig: WithFluentConfig,\n nextConfig: NextConfig,\n): NextConfig {\n const projectRoot = process.cwd()\n const resolved = resolveConfig(projectRoot, fluentConfig)\n\n // Warn if compiled catalogs directory doesn't exist yet\n const compiledDir = resolve(projectRoot, resolved.compiledDir)\n if (!existsSync(compiledDir)) {\n console.warn(\n `\\n[fluenti] Compiled catalogs not found at ${resolved.compiledDir}.\\n` +\n `Run: npx fluenti extract && npx fluenti compile\\n`,\n )\n }\n\n // Generate server module for RSC\n const serverModulePath = generateServerModule(projectRoot, resolved)\n\n // Resolve the loader path — use import.meta.url for ESM compatibility\n const thisDir = typeof __dirname !== 'undefined'\n ? __dirname\n : dirname(new URL(import.meta.url).pathname)\n const loaderPath = resolve(thisDir, 'loader.js')\n\n const existingWebpack = nextConfig['webpack'] as\n | ((config: WebpackConfig, options: WebpackOptions) => WebpackConfig)\n | undefined\n\n return {\n ...nextConfig,\n webpack(config: WebpackConfig, options: WebpackOptions) {\n // Add fluenti loader (enforce: pre — runs before other loaders)\n config.module.rules.push({\n test: /\\.[jt]sx?$/,\n enforce: 'pre' as const,\n exclude: [/node_modules/, /\\.next/],\n use: [\n {\n loader: loaderPath,\n options: {\n serverModulePath,\n },\n },\n ],\n })\n\n // Add resolve alias so loader can import from generated server module\n config.resolve = config.resolve ?? {} as WebpackConfig['resolve']\n config.resolve.alias = config.resolve.alias ?? {}\n config.resolve.alias['@fluenti/next$'] = serverModulePath\n\n // Auto extract+compile in dev mode\n const devAutoCompile = fluentConfig.devAutoCompile ?? true\n if (options.dev && devAutoCompile) {\n const debouncedRun = createDebouncedRunner({ cwd: projectRoot })\n\n config.plugins = config.plugins ?? []\n config.plugins.push({\n apply(compiler: WebpackCompiler) {\n let isFirstBuild = true\n compiler.hooks.watchRun.tapAsync('fluenti-dev', (_compiler: WebpackCompiler, callback: () => void) => {\n if (isFirstBuild) {\n isFirstBuild = false\n debouncedRun()\n }\n const modifiedFiles = _compiler.modifiedFiles\n if (modifiedFiles) {\n const hasSourceChange = [...modifiedFiles].some((f: string) =>\n /\\.[jt]sx?$/.test(f) && !f.includes('node_modules') && !f.includes('.next'),\n )\n if (hasSourceChange) {\n debouncedRun()\n }\n }\n callback()\n })\n },\n })\n }\n\n // Call user's webpack config if provided\n if (existingWebpack) {\n return existingWebpack(config, options)\n }\n\n return config\n },\n }\n}\n\n// Minimal webpack types for the config function\ninterface WebpackConfig {\n module: {\n rules: Array<{\n test: RegExp\n enforce?: 'pre' | 'post'\n exclude?: Array<RegExp>\n use: Array<{ loader: string; options: Record<string, unknown> }>\n }>\n }\n resolve: {\n alias?: Record<string, string>\n }\n plugins?: Array<{ apply(compiler: WebpackCompiler): void }>\n}\n\ninterface WebpackOptions {\n isServer: boolean\n dev: boolean\n}\n\ninterface WebpackCompiler {\n hooks: {\n watchRun: {\n tapAsync(name: string, cb: (compiler: WebpackCompiler, callback: () => void) => void): void\n }\n }\n modifiedFiles?: Set<string>\n}\n","/**\n * @fluenti/next — Next.js plugin for Fluenti\n *\n * Provides:\n * - `withFluenti()` — wraps next.config.ts with t`` transform support\n * - I18nProvider — async server component (exported from generated module)\n * - Webpack loader for strict, binding-aware tagged-template optimization\n *\n * @example\n * ```ts\n * // next.config.ts\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti()({ reactStrictMode: true })\n * ```\n *\n * @example\n * ```tsx\n * // app/layout.tsx — resolved by webpack alias to the generated module\n * import { I18nProvider } from '@fluenti/next'\n * ```\n */\nexport { withFluenti } from './with-fluenti'\nexport type { WithFluentConfig, I18nProviderProps } from './types'\n\n// ── Runtime stubs ────────────────────────────────────────────────────\n// TypeScript resolves types from this file (via package.json exports).\n// At runtime, webpack `resolve.alias` redirects `@fluenti/next$` to the\n// generated server module, so these stubs are never actually called in\n// a correctly configured project. They exist only to provide helpful\n// errors if `withFluenti()` is not configured.\n\nimport type { ReactNode, ReactElement } from 'react'\nimport type { CompileTimeT, FluentInstanceExtended } from '@fluenti/core'\n\nconst NOT_CONFIGURED =\n '[fluenti] `withFluenti()` must be configured in next.config.ts before importing from \"@fluenti/next\".'\n\nfunction throwNotConfigured(): never {\n throw new Error(NOT_CONFIGURED)\n}\n\n/** @see Generated module for the real implementation. */\nexport const setLocale: (locale: string) => void = throwNotConfigured\n/** @see Generated module for the real implementation. */\nexport const getI18n: () => Promise<FluentInstanceExtended & { locale: string }> = throwNotConfigured as () => Promise<FluentInstanceExtended & { locale: string }>\n/** @see Generated module for the real implementation. */\nexport const t: CompileTimeT = throwNotConfigured as unknown as CompileTimeT\n/** @see Generated module for the real implementation. */\nexport const Trans: (props: { children: ReactNode; id?: string; context?: string; comment?: string; render?: (translation: ReactNode) => ReactNode }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Trans\n/** @see Generated module for the real implementation. */\nexport const Plural: (props: { value: number; id?: string; context?: string; comment?: string; zero?: ReactNode; one?: ReactNode; two?: ReactNode; few?: ReactNode; many?: ReactNode; other: ReactNode; offset?: number }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Plural\n/** @see Generated module for the real implementation. */\nexport const Select: (props: { value: string; id?: string; context?: string; comment?: string; other: ReactNode; options?: Record<string, ReactNode>; [key: string]: ReactNode | Record<string, ReactNode> | string | undefined }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Select\n/** @see Generated module for the real implementation. */\nexport const DateTime: (props: { value: Date | number; style?: string }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof DateTime\n/** @see Generated module for the real implementation. */\nexport const NumberFormat: (props: { value: number; style?: string }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof NumberFormat\n/** @see Generated module for the real implementation. */\nexport const I18nProvider: (props: { locale?: string; children: ReactNode }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof I18nProvider\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.js","names":[],"sources":["../src/read-config.ts","../src/generate-server-module.ts","../src/dev-runner.ts","../src/with-fluenti.ts","../src/index.ts"],"sourcesContent":["import { existsSync, readFileSync, rmSync, writeFileSync } from 'node:fs'\nimport { createRequire } from 'node:module'\nimport { basename, dirname, extname, join, resolve } from 'node:path'\nimport type { FluentiConfig } from '@fluenti/core'\nimport type { WithFluentConfig, ResolvedFluentConfig } from './types'\n\nconst runtimeModulePath = typeof __filename === 'string'\n ? __filename\n : import.meta.url\nconst require = createRequire(runtimeModulePath)\nconst { createJiti } = require('jiti') as {\n createJiti: (\n url: string,\n options?: { moduleCache?: boolean; interopDefault?: boolean },\n ) => (path: string) => unknown\n}\n\n/**\n * Read fluenti.config.ts and merge with withFluenti() overrides.\n */\nexport function resolveConfig(\n projectRoot: string,\n overrides?: WithFluentConfig,\n): ResolvedFluentConfig {\n const fileConfig = readFluentConfigSync(projectRoot)\n\n const defaultLocale = overrides?.defaultLocale\n ?? fileConfig?.sourceLocale\n ?? 'en'\n\n const locales = overrides?.locales\n ?? fileConfig?.locales\n ?? [defaultLocale]\n\n const compiledDir = overrides?.compiledDir\n ?? fileConfig?.compileOutDir\n ?? './src/locales/compiled'\n\n const serverModuleOutDir = overrides?.serverModuleOutDir\n ?? join('node_modules', '.fluenti')\n\n const resolved: ResolvedFluentConfig = {\n locales,\n defaultLocale,\n compiledDir,\n serverModule: overrides?.serverModule ?? null,\n serverModuleOutDir,\n }\n if (overrides?.resolveLocale) resolved.resolveLocale = overrides.resolveLocale\n if (overrides?.dateFormats) resolved.dateFormats = overrides.dateFormats\n if (overrides?.numberFormats) resolved.numberFormats = overrides.numberFormats\n if (overrides?.fallbackChain) resolved.fallbackChain = overrides.fallbackChain\n return resolved\n}\n\n/**\n * Attempt to read fluenti.config.ts synchronously.\n * Returns null if file doesn't exist or can't be parsed.\n */\nfunction readFluentConfigSync(projectRoot: string): FluentiConfig | null {\n const jiti = createJiti(runtimeModulePath, {\n moduleCache: false,\n interopDefault: true,\n })\n const candidates = [\n 'fluenti.config.ts',\n 'fluenti.config.js',\n 'fluenti.config.mjs',\n ]\n\n for (const name of candidates) {\n const configPath = resolve(projectRoot, name)\n if (existsSync(configPath)) {\n try {\n return normalizeLoadedConfig(jiti(configPath) as FluentiConfig | { default?: FluentiConfig })\n } catch {\n const rewritten = tryLoadConfigViaDefineConfigShim(configPath, jiti)\n if (rewritten) {\n return rewritten\n }\n return null\n }\n }\n }\n\n return null\n}\n\nfunction tryLoadConfigViaDefineConfigShim(\n configPath: string,\n jiti: (path: string) => unknown,\n): FluentiConfig | null {\n const source = readFileSync(configPath, 'utf8')\n const importMatch = source.match(\n /import\\s*\\{\\s*defineConfig(?:\\s+as\\s+([A-Za-z_$][\\w$]*))?\\s*\\}\\s*from\\s*['\"]@fluenti\\/cli['\"]\\s*;?/,\n )\n if (!importMatch) {\n return null\n }\n\n const helperName = importMatch[1] ?? 'defineConfig'\n const rewrittenSource = source.replace(importMatch[0], '')\n const tempPath = join(\n dirname(configPath),\n `.${basename(configPath, extname(configPath))}.next-plugin-read-config${extname(configPath) || '.ts'}`,\n )\n\n writeFileSync(tempPath, `const ${helperName} = (config) => config\\n${rewrittenSource}`, 'utf8')\n\n try {\n return normalizeLoadedConfig(jiti(tempPath) as FluentiConfig | { default?: FluentiConfig })\n } catch {\n return null\n } finally {\n rmSync(tempPath, { force: true })\n }\n}\n\nfunction normalizeLoadedConfig(\n mod: FluentiConfig | { default?: FluentiConfig },\n): FluentiConfig {\n return typeof mod === 'object' && mod !== null && 'default' in mod\n ? (mod.default ?? {}) as FluentiConfig\n : mod as FluentiConfig\n}\n","import { writeFileSync, mkdirSync, existsSync } from 'node:fs'\nimport { resolve, relative } from 'node:path'\nimport { validateLocale } from '@fluenti/core'\nimport type { ResolvedFluentConfig } from './types'\n\n/**\n * Generate the server module that provides:\n * - setLocale / getI18n\n * - Trans / Plural / Select / DateTime / NumberFormat (server components)\n * - I18nProvider (async server component for layouts)\n *\n * @returns Absolute path to the generated server module.\n */\nexport function generateServerModule(\n projectRoot: string,\n config: ResolvedFluentConfig,\n): string {\n if (config.serverModule) {\n return resolve(projectRoot, config.serverModule)\n }\n\n const outDir = resolve(projectRoot, config.serverModuleOutDir)\n const outPath = resolve(outDir, 'server.js')\n const dtsPath = resolve(outDir, 'server.d.ts')\n\n if (!existsSync(outDir)) {\n mkdirSync(outDir, { recursive: true })\n }\n\n for (const locale of config.locales) {\n validateLocale(locale, 'next-plugin')\n }\n\n const compiledDirAbs = resolve(projectRoot, config.compiledDir)\n const compiledRelative = toForwardSlash(relative(outDir, compiledDirAbs))\n\n const localeImports = config.locales\n .map((locale) => ` case '${locale}': return import('${compiledRelative}/${locale}')`)\n .join('\\n')\n\n const fallbackChainStr = config.fallbackChain\n ? JSON.stringify(config.fallbackChain)\n : 'undefined'\n\n // Generate a 'use client' provider that imports messages statically.\n // Messages contain functions (interpolation) which can't cross the RSC boundary.\n const clientProviderPath = resolve(outDir, 'client-provider.js')\n\n const clientStaticImports = config.locales\n .map((locale) => {\n const safe = locale.replace(/[^a-zA-Z0-9]/g, '_')\n return `import ${safe} from '${compiledRelative}/${locale}'`\n })\n .join('\\n')\n\n const clientAllMessagesEntries = config.locales\n .map((locale) => {\n const safe = locale.replace(/[^a-zA-Z0-9]/g, '_')\n return `'${locale}': ${safe}`\n })\n .join(', ')\n\n const clientProviderSource = `\"use client\";\n// Auto-generated by @fluenti/next — do not edit\nimport { createElement } from 'react'\nimport { I18nProvider } from '@fluenti/react'\n${clientStaticImports}\n\nconst __allMessages = { ${clientAllMessagesEntries} }\n\nexport function ClientI18nProvider({ locale, fallbackLocale, fallbackChain, children }) {\n return createElement(I18nProvider, { locale, fallbackLocale, messages: __allMessages, fallbackChain }, children)\n}\n`\n writeFileSync(clientProviderPath, clientProviderSource, 'utf-8')\n\n const resolveLocaleImport = config.resolveLocale\n ? (() => {\n const absPath = resolve(projectRoot, config.resolveLocale)\n const relPath = toForwardSlash(relative(outDir, absPath))\n return `import __resolveLocale from '${relPath}'`\n })()\n : null\n\n const resolveLocaleFn = config.resolveLocale\n ? `resolveLocale: __resolveLocale,`\n : `resolveLocale: async () => {\n try {\n const { cookies } = await import('next/headers')\n return (await cookies()).get('locale')?.value ?? '${config.defaultLocale}'\n } catch {\n return '${config.defaultLocale}'\n }\n },`\n\n const moduleSource = `// Auto-generated by @fluenti/next — do not edit\nimport { createServerI18n } from '@fluenti/react/server'\nimport { createElement } from 'react'\n${resolveLocaleImport ? `${resolveLocaleImport}\\n` : ''}\nconst serverI18n = createServerI18n({\n loadMessages: async (locale) => {\n switch (locale) {\n${localeImports}\n default: return import('${compiledRelative}/${config.defaultLocale}')\n }\n },\n fallbackLocale: '${config.defaultLocale}',\n fallbackChain: ${fallbackChainStr},\n ${resolveLocaleFn}\n})\n\nexport const setLocale = serverI18n.setLocale\nexport const getI18n = serverI18n.getI18n\nexport const t = (..._args) => {\n throw new Error(\n \"[fluenti] \\`t\\` imported from '@fluenti/next' is a compile-time API. \" +\n 'Use it only with the Fluenti loader inside an async server scope.',\n )\n}\nexport const Trans = serverI18n.Trans\nexport const Plural = serverI18n.Plural\nexport const Select = serverI18n.Select\nexport const DateTime = serverI18n.DateTime\nexport const NumberFormat = serverI18n.NumberFormat\n\n/**\n * Async server component for root layouts.\n *\n * Sets up both server-side (React.cache) and client-side (I18nProvider) i18n.\n */\nexport async function I18nProvider({ locale, children }) {\n const activeLocale = locale ?? '${config.defaultLocale}'\n\n // 1. Initialize server-side i18n (React.cache scoped)\n serverI18n.setLocale(activeLocale)\n await serverI18n.getI18n()\n\n // 2. Import the local 'use client' provider that has messages statically bundled.\n // Messages contain functions (interpolation) which can't be serialized across the RSC boundary.\n const { ClientI18nProvider } = await import('./client-provider.js')\n\n return createElement(ClientI18nProvider, {\n locale: activeLocale,\n fallbackLocale: '${config.defaultLocale}',\n fallbackChain: ${fallbackChainStr},\n }, children)\n}\n`\n\n const dtsSource = `// Auto-generated by @fluenti/next — do not edit\nimport type { ReactNode, ReactElement } from 'react'\nimport type { CompileTimeT, FluentInstanceExtended } from '@fluenti/core'\n\nexport declare function setLocale(locale: string): void\nexport declare function getI18n(): Promise<FluentInstanceExtended & { locale: string }>\nexport declare const t: CompileTimeT\n\nexport declare function Trans(props: {\n children: ReactNode\n id?: string\n context?: string\n comment?: string\n render?: (translation: ReactNode) => ReactNode\n}): Promise<ReactElement>\n\nexport declare function Plural(props: {\n value: number\n id?: string\n context?: string\n comment?: string\n zero?: ReactNode\n one?: ReactNode\n two?: ReactNode\n few?: ReactNode\n many?: ReactNode\n other: ReactNode\n offset?: number\n}): Promise<ReactElement>\n\nexport declare function Select(props: {\n value: string\n id?: string\n context?: string\n comment?: string\n other: ReactNode\n options?: Record<string, ReactNode>\n [key: string]: ReactNode | Record<string, ReactNode> | string | undefined\n}): Promise<ReactElement>\n\nexport declare function DateTime(props: {\n value: Date | number\n style?: string\n}): Promise<ReactElement>\n\nexport declare function NumberFormat(props: {\n value: number\n style?: string\n}): Promise<ReactElement>\n\nexport declare function I18nProvider(props: {\n locale?: string\n children: ReactNode\n}): Promise<ReactElement>\n`\n\n writeFileSync(outPath, moduleSource, 'utf-8')\n writeFileSync(dtsPath, dtsSource, 'utf-8')\n\n return outPath\n}\n\nfunction toForwardSlash(p: string): string {\n return p.split('\\\\').join('/')\n}\n","import { exec } from 'node:child_process'\nimport { existsSync } from 'node:fs'\nimport { resolve, dirname, join } from 'node:path'\nimport { createRequire } from 'node:module'\n\nexport interface DevRunnerOptions {\n cwd: string\n onSuccess?: () => void\n onError?: (err: Error) => void\n /** If true, reject the promise on failure instead of swallowing the error */\n throwOnError?: boolean\n /** Run only compile (skip extract). Useful for production builds where source is unchanged. */\n compileOnly?: boolean\n}\n\n/**\n * Walk up from `cwd` to find `node_modules/.bin/fluenti`.\n * Returns the absolute path or null if not found.\n */\nexport function resolveCliBin(cwd: string): string | null {\n let dir = cwd\n for (;;) {\n const bin = resolve(dir, 'node_modules/.bin/fluenti')\n if (existsSync(bin)) return bin\n const parent = dirname(dir)\n if (parent === dir) break\n dir = parent\n }\n return null\n}\n\n/**\n * Run compile in-process via `@fluenti/cli` (for compileOnly mode),\n * or fall back to shell-out for extract + compile (dev mode).\n */\nexport async function runExtractCompile(options: DevRunnerOptions): Promise<void> {\n if (options.compileOnly) {\n try {\n // Resolve @fluenti/cli from the project's cwd (not from this package's location)\n // using createRequire so pnpm's strict node_modules layout works correctly.\n // Use require() (not import()) to load @fluenti/cli — avoids CJS/ESM interop\n // issues when dynamic import() loads minified CJS with chunk requires.\n const projectRequire = createRequire(join(options.cwd, 'package.json'))\n const { runCompile } = projectRequire('@fluenti/cli')\n await runCompile(options.cwd)\n console.log('[fluenti] Compiling... done')\n options.onSuccess?.()\n return\n } catch (e) {\n const error = e instanceof Error ? e : new Error(String(e))\n if (options.throwOnError) throw error\n console.warn('[fluenti] Compile failed:', error.message)\n options.onError?.(error)\n return\n }\n }\n\n // Dev mode: shell out for extract + compile\n const bin = resolveCliBin(options.cwd)\n if (!bin) {\n const msg = '[fluenti] CLI not found — skipping auto-compile. Install @fluenti/cli as a devDependency.'\n if (options.throwOnError) {\n return Promise.reject(new Error(msg))\n }\n console.warn(msg)\n return Promise.resolve()\n }\n\n const command = `${bin} extract && ${bin} compile`\n return new Promise<void>((resolve, reject) => {\n exec(\n command,\n { cwd: options.cwd },\n (err, _stdout, stderr) => {\n if (err) {\n const error = new Error(stderr || err.message)\n if (options.throwOnError) {\n reject(error)\n return\n }\n console.warn('[fluenti] Extract/compile failed:', error.message)\n options.onError?.(error)\n } else {\n console.log('[fluenti] Extracting and compiling... done')\n options.onSuccess?.()\n }\n resolve()\n },\n )\n })\n}\n\n/**\n * Create a debounced runner that collapses rapid calls.\n *\n * - If called while idle, schedules a run after `delay` ms.\n * - If called while a run is in progress, marks a pending rerun.\n * - Never runs concurrently.\n */\nexport function createDebouncedRunner(\n options: DevRunnerOptions,\n delay = 300,\n): () => void {\n let timer: ReturnType<typeof setTimeout> | null = null\n let running = false\n let pendingRerun = false\n\n async function execute(): Promise<void> {\n running = true\n try {\n await runExtractCompile(options)\n } finally {\n running = false\n if (pendingRerun) {\n pendingRerun = false\n schedule()\n }\n }\n }\n\n function schedule(): void {\n if (timer !== null) {\n clearTimeout(timer)\n }\n timer = setTimeout(() => {\n timer = null\n if (running) {\n pendingRerun = true\n } else {\n execute()\n }\n }, delay)\n }\n\n return schedule\n}\n","import { existsSync } from 'node:fs'\nimport { execSync } from 'node:child_process'\nimport { resolve, dirname } from 'node:path'\nimport type { WithFluentConfig } from './types'\nimport { resolveConfig } from './read-config'\nimport { generateServerModule } from './generate-server-module'\nimport { createDebouncedRunner } from './dev-runner'\n\ntype NextConfig = Record<string, unknown>\n\n/**\n * Wrap your Next.js config with Fluenti support.\n *\n * Adds a webpack loader that transforms `t\\`\\`` and `t()` calls,\n * and generates a server module for RSC i18n.\n *\n * @example\n * ```ts\n * // next.config.ts — function style (recommended)\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti()({ reactStrictMode: true })\n * ```\n *\n * @example\n * ```ts\n * // next.config.ts — direct style\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti({ reactStrictMode: true })\n * ```\n */\nexport function withFluenti(fluentConfig?: WithFluentConfig): (nextConfig?: NextConfig) => NextConfig\nexport function withFluenti(nextConfig: NextConfig): NextConfig\nexport function withFluenti(\n configOrNext?: WithFluentConfig | NextConfig,\n): NextConfig | ((nextConfig?: NextConfig) => NextConfig) {\n if (configOrNext && isNextConfig(configOrNext as NextConfig)) {\n return applyFluenti({}, configOrNext as NextConfig)\n }\n\n const fluentConfig = (configOrNext ?? {}) as WithFluentConfig\n return function wrappedConfig(nextConfig?: NextConfig): NextConfig {\n return applyFluenti(fluentConfig, nextConfig ?? {})\n }\n}\n\nfunction isNextConfig(obj: NextConfig): boolean {\n const nextKeys = [\n 'reactStrictMode', 'experimental', 'images', 'env', 'webpack',\n 'rewrites', 'redirects', 'headers', 'pageExtensions', 'output',\n 'basePath', 'i18n', 'trailingSlash', 'compiler', 'transpilePackages',\n ]\n return nextKeys.some((key) => key in obj)\n}\n\nfunction applyFluenti(\n fluentConfig: WithFluentConfig,\n nextConfig: NextConfig,\n): NextConfig {\n const projectRoot = process.cwd()\n const resolved = resolveConfig(projectRoot, fluentConfig)\n\n // Warn if compiled catalogs directory doesn't exist yet\n const compiledDir = resolve(projectRoot, resolved.compiledDir)\n if (!existsSync(compiledDir)) {\n console.warn(\n `\\n[fluenti] Compiled catalogs not found at ${resolved.compiledDir}.\\n` +\n `Run: npx fluenti extract && npx fluenti compile\\n`,\n )\n }\n\n // Generate server module for RSC\n const serverModulePath = generateServerModule(projectRoot, resolved)\n\n // Resolve the loader path — use import.meta.url for ESM compatibility\n const thisDir = typeof __dirname !== 'undefined'\n ? __dirname\n : dirname(new URL(import.meta.url).pathname)\n const loaderPath = resolve(thisDir, 'loader.js')\n\n const existingWebpack = nextConfig['webpack'] as\n | ((config: WebpackConfig, options: WebpackOptions) => WebpackConfig)\n | undefined\n\n let buildCompileRan = false\n\n return {\n ...nextConfig,\n webpack(config: WebpackConfig, options: WebpackOptions) {\n // Add fluenti loader (enforce: pre — runs before other loaders)\n config.module.rules.push({\n test: /\\.[jt]sx?$/,\n enforce: 'pre' as const,\n exclude: [/node_modules/, /\\.next/],\n use: [\n {\n loader: loaderPath,\n options: {\n serverModulePath,\n },\n },\n ],\n })\n\n // Add resolve alias so loader can import from generated server module\n config.resolve = config.resolve ?? {} as WebpackConfig['resolve']\n config.resolve.alias = config.resolve.alias ?? {}\n config.resolve.alias['@fluenti/next$'] = serverModulePath\n\n // Auto compile before production build (run once across server+client passes)\n const buildAutoCompile = fluentConfig.buildAutoCompile ?? true\n if (!options.dev && buildAutoCompile && !buildCompileRan) {\n buildCompileRan = true\n try {\n // Use node -e with dynamic import — avoids CLI binary resolution issues.\n // webpack() is sync, so we use execSync to block until compile finishes.\n const escapedRoot = projectRoot.replace(/\\\\/g, '\\\\\\\\').replace(/'/g, \"\\\\'\")\n execSync(\n `node --input-type=module -e \"const { runCompile } = await import('@fluenti/cli'); await runCompile('${escapedRoot}')\"`,\n { cwd: projectRoot, stdio: 'inherit' },\n )\n } catch {\n // @fluenti/cli not available or compile failed — skip silently\n }\n }\n\n // Auto extract+compile in dev mode\n const devAutoCompile = fluentConfig.devAutoCompile ?? true\n if (options.dev && devAutoCompile) {\n const devDelay = fluentConfig.devAutoCompileDelay ?? 1000\n const debouncedRun = createDebouncedRunner({ cwd: projectRoot }, devDelay)\n const compiledDirResolved = resolve(projectRoot, resolved.compiledDir)\n\n config.plugins = config.plugins ?? []\n config.plugins.push({\n apply(compiler: WebpackCompiler) {\n let isFirstBuild = true\n compiler.hooks.watchRun.tapAsync('fluenti-dev', (_compiler: WebpackCompiler, callback: () => void) => {\n if (isFirstBuild) {\n isFirstBuild = false\n debouncedRun()\n }\n const modifiedFiles = _compiler.modifiedFiles\n if (modifiedFiles) {\n const hasSourceChange = [...modifiedFiles].some((f: string) =>\n /\\.[jt]sx?$/.test(f)\n && !f.includes('node_modules')\n && !f.includes('.next')\n && !f.startsWith(compiledDirResolved),\n )\n if (hasSourceChange) {\n debouncedRun()\n }\n }\n callback()\n })\n },\n })\n }\n\n // Call user's webpack config if provided\n if (existingWebpack) {\n return existingWebpack(config, options)\n }\n\n return config\n },\n }\n}\n\n// Minimal webpack types for the config function\ninterface WebpackConfig {\n module: {\n rules: Array<{\n test: RegExp\n enforce?: 'pre' | 'post'\n exclude?: Array<RegExp>\n use: Array<{ loader: string; options: Record<string, unknown> }>\n }>\n }\n resolve: {\n alias?: Record<string, string>\n }\n plugins?: Array<{ apply(compiler: WebpackCompiler): void }>\n}\n\ninterface WebpackOptions {\n isServer: boolean\n dev: boolean\n}\n\ninterface WebpackCompiler {\n hooks: {\n watchRun: {\n tapAsync(name: string, cb: (compiler: WebpackCompiler, callback: () => void) => void): void\n }\n }\n modifiedFiles?: Set<string>\n}\n","/**\n * @fluenti/next — Next.js plugin for Fluenti\n *\n * Provides:\n * - `withFluenti()` — wraps next.config.ts with t`` transform support\n * - I18nProvider — async server component (exported from generated module)\n * - Webpack loader for strict, binding-aware tagged-template optimization\n *\n * @example\n * ```ts\n * // next.config.ts\n * import { withFluenti } from '@fluenti/next'\n * export default withFluenti()({ reactStrictMode: true })\n * ```\n *\n * @example\n * ```tsx\n * // app/layout.tsx — resolved by webpack alias to the generated module\n * import { I18nProvider } from '@fluenti/next'\n * ```\n */\nexport { withFluenti } from './with-fluenti'\nexport type { WithFluentConfig, I18nProviderProps } from './types'\n\n// ── Runtime stubs ────────────────────────────────────────────────────\n// TypeScript resolves types from this file (via package.json exports).\n// At runtime, webpack `resolve.alias` redirects `@fluenti/next$` to the\n// generated server module, so these stubs are never actually called in\n// a correctly configured project. They exist only to provide helpful\n// errors if `withFluenti()` is not configured.\n\nimport type { ReactNode, ReactElement } from 'react'\nimport type { CompileTimeT, FluentInstanceExtended } from '@fluenti/core'\n\nconst NOT_CONFIGURED =\n '[fluenti] `withFluenti()` must be configured in next.config.ts before importing from \"@fluenti/next\".'\n\nfunction throwNotConfigured(): never {\n throw new Error(NOT_CONFIGURED)\n}\n\n/** @see Generated module for the real implementation. */\nexport const setLocale: (locale: string) => void = throwNotConfigured\n/** @see Generated module for the real implementation. */\nexport const getI18n: () => Promise<FluentInstanceExtended & { locale: string }> = throwNotConfigured as () => Promise<FluentInstanceExtended & { locale: string }>\n/** @see Generated module for the real implementation. */\nexport const t: CompileTimeT = throwNotConfigured as unknown as CompileTimeT\n/** @see Generated module for the real implementation. */\nexport const Trans: (props: { children: ReactNode; id?: string; context?: string; comment?: string; render?: (translation: ReactNode) => ReactNode }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Trans\n/** @see Generated module for the real implementation. */\nexport const Plural: (props: { value: number; id?: string; context?: string; comment?: string; zero?: ReactNode; one?: ReactNode; two?: ReactNode; few?: ReactNode; many?: ReactNode; other: ReactNode; offset?: number }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Plural\n/** @see Generated module for the real implementation. */\nexport const Select: (props: { value: string; id?: string; context?: string; comment?: string; other: ReactNode; options?: Record<string, ReactNode>; [key: string]: ReactNode | Record<string, ReactNode> | string | undefined }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof Select\n/** @see Generated module for the real implementation. */\nexport const DateTime: (props: { value: Date | number; style?: string }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof DateTime\n/** @see Generated module for the real implementation. */\nexport const NumberFormat: (props: { value: number; style?: string }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof NumberFormat\n/** @see Generated module for the real implementation. */\nexport const I18nProvider: (props: { locale?: string; children: ReactNode }) => Promise<ReactElement> = throwNotConfigured as unknown as typeof I18nProvider\n"],"mappings":"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"}
|
package/dist/types.d.ts
CHANGED
|
@@ -35,6 +35,10 @@ export interface WithFluentConfig {
|
|
|
35
35
|
fallbackChain?: Record<string, Locale[]>;
|
|
36
36
|
/** Auto extract+compile in dev mode (default: true) */
|
|
37
37
|
devAutoCompile?: boolean;
|
|
38
|
+
/** Debounce delay in ms for dev auto-compile (default: 1000). Increase if you see excessive recompilation. */
|
|
39
|
+
devAutoCompileDelay?: number;
|
|
40
|
+
/** Auto extract+compile before production build (default: true) */
|
|
41
|
+
buildAutoCompile?: boolean;
|
|
38
42
|
}
|
|
39
43
|
/**
|
|
40
44
|
* Resolved config after merging fluenti.config.ts + withFluenti() overrides.
|
package/dist/types.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,iBAAiB,EAAE,mBAAmB,EAAE,MAAM,EAAE,MAAM,eAAe,CAAA;AAEnF;;;;;GAKG;AACH,MAAM,WAAW,gBAAgB;IAC/B,2CAA2C;IAC3C,OAAO,CAAC,EAAE,MAAM,EAAE,CAAA;IAClB,wEAAwE;IACxE,aAAa,CAAC,EAAE,MAAM,CAAA;IACtB,iDAAiD;IACjD,WAAW,CAAC,EAAE,MAAM,CAAA;IAEpB,sDAAsD;IACtD,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB,0EAA0E;IAC1E,kBAAkB,CAAC,EAAE,MAAM,CAAA;IAE3B;;;;;;;;;OASG;IACH,aAAa,CAAC,EAAE,MAAM,CAAA;IAEtB,gCAAgC;IAChC,WAAW,CAAC,EAAE,iBAAiB,CAAA;IAC/B,kCAAkC;IAClC,aAAa,CAAC,EAAE,mBAAmB,CAAA;IACnC,gCAAgC;IAChC,aAAa,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,EAAE,CAAC,CAAA;IACxC,uDAAuD;IACvD,cAAc,CAAC,EAAE,OAAO,CAAA;
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../src/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,iBAAiB,EAAE,mBAAmB,EAAE,MAAM,EAAE,MAAM,eAAe,CAAA;AAEnF;;;;;GAKG;AACH,MAAM,WAAW,gBAAgB;IAC/B,2CAA2C;IAC3C,OAAO,CAAC,EAAE,MAAM,EAAE,CAAA;IAClB,wEAAwE;IACxE,aAAa,CAAC,EAAE,MAAM,CAAA;IACtB,iDAAiD;IACjD,WAAW,CAAC,EAAE,MAAM,CAAA;IAEpB,sDAAsD;IACtD,YAAY,CAAC,EAAE,MAAM,CAAA;IACrB,0EAA0E;IAC1E,kBAAkB,CAAC,EAAE,MAAM,CAAA;IAE3B;;;;;;;;;OASG;IACH,aAAa,CAAC,EAAE,MAAM,CAAA;IAEtB,gCAAgC;IAChC,WAAW,CAAC,EAAE,iBAAiB,CAAA;IAC/B,kCAAkC;IAClC,aAAa,CAAC,EAAE,mBAAmB,CAAA;IACnC,gCAAgC;IAChC,aAAa,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,EAAE,CAAC,CAAA;IACxC,uDAAuD;IACvD,cAAc,CAAC,EAAE,OAAO,CAAA;IACxB,8GAA8G;IAC9G,mBAAmB,CAAC,EAAE,MAAM,CAAA;IAC5B,mEAAmE;IACnE,gBAAgB,CAAC,EAAE,OAAO,CAAA;CAC3B;AAED;;GAEG;AACH,MAAM,WAAW,oBAAoB;IACnC,OAAO,EAAE,MAAM,EAAE,CAAA;IACjB,aAAa,EAAE,MAAM,CAAA;IACrB,WAAW,EAAE,MAAM,CAAA;IACnB,YAAY,EAAE,MAAM,GAAG,IAAI,CAAA;IAC3B,kBAAkB,EAAE,MAAM,CAAA;IAC1B,aAAa,CAAC,EAAE,MAAM,CAAA;IACtB,WAAW,CAAC,EAAE,iBAAiB,CAAA;IAC/B,aAAa,CAAC,EAAE,mBAAmB,CAAA;IACnC,aAAa,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,EAAE,CAAC,CAAA;CACzC;AAED;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAChC,iEAAiE;IACjE,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,QAAQ,EAAE,KAAK,CAAC,SAAS,CAAA;CAC1B"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"with-fluenti.d.ts","sourceRoot":"","sources":["../src/with-fluenti.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"with-fluenti.d.ts","sourceRoot":"","sources":["../src/with-fluenti.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,SAAS,CAAA;AAK/C,KAAK,UAAU,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAA;AAEzC;;;;;;;;;;;;;;;;;;;GAmBG;AACH,wBAAgB,WAAW,CAAC,YAAY,CAAC,EAAE,gBAAgB,GAAG,CAAC,UAAU,CAAC,EAAE,UAAU,KAAK,UAAU,CAAA;AACrG,wBAAgB,WAAW,CAAC,UAAU,EAAE,UAAU,GAAG,UAAU,CAAA"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@fluenti/next",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.2.0",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"description": "Next.js plugin for Fluenti — withFluenti, I18nProvider, t`` transforms for App Router and Pages Router",
|
|
6
6
|
"homepage": "https://fluenti.dev",
|
|
@@ -71,8 +71,8 @@
|
|
|
71
71
|
},
|
|
72
72
|
"dependencies": {
|
|
73
73
|
"jiti": "^2",
|
|
74
|
-
"@fluenti/core": "0.
|
|
75
|
-
"@fluenti/react": "0.
|
|
74
|
+
"@fluenti/core": "0.2.0",
|
|
75
|
+
"@fluenti/react": "0.2.0"
|
|
76
76
|
},
|
|
77
77
|
"devDependencies": {
|
|
78
78
|
"@types/node": "^22",
|