@intlayer/docs 7.0.0-canary.2 → 7.0.0-canary.3
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/cjs/common.cjs.map +1 -1
- package/dist/esm/common.mjs.map +1 -1
- package/dist/types/common.d.ts +5 -0
- package/dist/types/common.d.ts.map +1 -1
- package/docs/ar/intlayer_with_nextjs_16.md +1652 -0
- package/docs/ar/releases/v7.md +486 -0
- package/docs/de/intlayer_with_nextjs_16.md +1662 -0
- package/docs/de/releases/v7.md +503 -0
- package/docs/en/intlayer_with_nextjs_15.md +5 -2
- package/docs/en/intlayer_with_nextjs_16.md +4 -4
- package/docs/en-GB/intlayer_with_nextjs_16.md +1642 -0
- package/docs/en-GB/releases/v7.md +486 -0
- package/docs/es/intlayer_with_nextjs_16.md +1670 -0
- package/docs/es/releases/v7.md +503 -0
- package/docs/fr/intlayer_with_nextjs_16.md +1692 -0
- package/docs/fr/releases/v7.md +504 -0
- package/docs/hi/intlayer_with_nextjs_16.md +1618 -0
- package/docs/hi/releases/v7.md +486 -0
- package/docs/id/intlayer_with_nextjs_16.md +1604 -0
- package/docs/id/releases/v7.md +503 -0
- package/docs/it/intlayer_with_nextjs_16.md +1600 -0
- package/docs/it/releases/v7.md +505 -0
- package/docs/ja/intlayer_CMS.md +0 -9
- package/docs/ja/intlayer_with_nextjs_16.md +1788 -0
- package/docs/ja/releases/v7.md +504 -0
- package/docs/ko/intlayer_with_nextjs_16.md +1641 -0
- package/docs/ko/releases/v7.md +504 -0
- package/docs/pl/intlayer_with_nextjs_16.md +1645 -0
- package/docs/pl/releases/v7.md +486 -0
- package/docs/pt/intlayer_with_nextjs_16.md +1646 -0
- package/docs/pt/introduction.md +0 -15
- package/docs/pt/releases/v7.md +486 -0
- package/docs/ru/intlayer_with_nextjs_16.md +1610 -0
- package/docs/ru/releases/v7.md +486 -0
- package/docs/tr/intlayer_with_nextjs_16.md +1599 -0
- package/docs/tr/releases/v7.md +486 -0
- package/docs/vi/intlayer_with_nextjs_16.md +1597 -0
- package/docs/vi/releases/v7.md +486 -0
- package/docs/zh/intlayer_CMS.md +0 -23
- package/docs/zh/intlayer_with_nextjs_16.md +1628 -0
- package/docs/zh/releases/v7.md +487 -0
- package/package.json +14 -14
- package/src/common.ts +5 -0
|
@@ -0,0 +1,1641 @@
|
|
|
1
|
+
---
|
|
2
|
+
createdAt: 2025-10-25
|
|
3
|
+
updatedAt: 2025-10-25
|
|
4
|
+
title: Next.js 16 앱 번역 방법 – 2025 i18n 가이드
|
|
5
|
+
description: Next.js 16 웹사이트를 다국어로 만드는 방법을 알아보세요. 국제화(i18n) 및 번역을 위한 문서를 따라가세요.
|
|
6
|
+
keywords:
|
|
7
|
+
- 국제화
|
|
8
|
+
- 문서
|
|
9
|
+
- Intlayer
|
|
10
|
+
- Next.js 16
|
|
11
|
+
- 자바스크립트
|
|
12
|
+
- 리액트
|
|
13
|
+
slugs:
|
|
14
|
+
- doc
|
|
15
|
+
- environment
|
|
16
|
+
- nextjs
|
|
17
|
+
applicationTemplate: https://github.com/aymericzip/intlayer-next-16-template
|
|
18
|
+
youtubeVideo: https://www.youtube.com/watch?v=e_PPG7PTqGU
|
|
19
|
+
history:
|
|
20
|
+
- version: 7.0.0
|
|
21
|
+
date: 2025-06-29
|
|
22
|
+
changes: 초기 기록
|
|
23
|
+
---
|
|
24
|
+
|
|
25
|
+
# Intlayer를 사용하여 Next.js 16 웹사이트 번역하기 | 국제화(i18n)
|
|
26
|
+
|
|
27
|
+
<iframe title="Next.js를 위한 최고의 i18n 솔루션? Intlayer를 발견하세요" class="m-auto aspect-[16/9] w-full overflow-hidden rounded-lg border-0" allow="autoplay; gyroscope;" loading="lazy" width="1080" height="auto" src="https://www.youtube.com/embed/e_PPG7PTqGU?autoplay=0&origin=http://intlayer.org&controls=0&rel=1"/>
|
|
28
|
+
|
|
29
|
+
GitHub에서 [애플리케이션 템플릿](https://github.com/aymericzip/intlayer-next-16-template)을 확인하세요.
|
|
30
|
+
|
|
31
|
+
## Intlayer란 무엇인가요?
|
|
32
|
+
|
|
33
|
+
**Intlayer**는 최신 웹 애플리케이션에서 다국어 지원을 간소화하기 위해 설계된 혁신적이고 오픈 소스인 국제화(i18n) 라이브러리입니다. Intlayer는 강력한 **App Router**를 포함한 최신 **Next.js 16** 프레임워크와 원활하게 통합됩니다. 효율적인 렌더링을 위해 **서버 컴포넌트**와 함께 작동하도록 최적화되어 있으며, [**Turbopack**](https://nextjs.org/docs/architecture/turbopack)과 완벽하게 호환됩니다.
|
|
34
|
+
|
|
35
|
+
Intlayer를 사용하면 다음을 할 수 있습니다:
|
|
36
|
+
|
|
37
|
+
- 컴포넌트 수준에서 선언적 사전을 사용하여 **번역을 쉽게 관리**할 수 있습니다.
|
|
38
|
+
- 메타데이터, 라우트 및 콘텐츠를 **동적으로 현지화**할 수 있습니다.
|
|
39
|
+
- 클라이언트 및 서버 측 컴포넌트 모두에서 **번역에 접근**할 수 있습니다.
|
|
40
|
+
- 자동 생성된 타입으로 **TypeScript 지원을 보장**하여 자동 완성 및 오류 감지를 향상시킵니다.
|
|
41
|
+
- **동적 로케일 감지 및 전환**과 같은 고급 기능을 활용할 수 있습니다.
|
|
42
|
+
|
|
43
|
+
> Intlayer는 Next.js 12, 13, 14, 16과 호환됩니다. Next.js Page Router를 사용하는 경우, 이 [가이드](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/intlayer_with_nextjs_page_router.md)를 참조하세요. Next.js 12, 13, 14에서 App Router를 사용하는 경우, 이 [가이드](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/intlayer_with_nextjs_14.md)를 참조하세요.
|
|
44
|
+
|
|
45
|
+
---
|
|
46
|
+
|
|
47
|
+
## Next.js 애플리케이션에서 Intlayer 설정 단계별 가이드
|
|
48
|
+
|
|
49
|
+
### 1단계: 의존성 설치
|
|
50
|
+
|
|
51
|
+
npm을 사용하여 필요한 패키지를 설치하세요:
|
|
52
|
+
|
|
53
|
+
```bash packageManager="npm"
|
|
54
|
+
npm install intlayer next-intlayer
|
|
55
|
+
```
|
|
56
|
+
|
|
57
|
+
```bash packageManager="pnpm"
|
|
58
|
+
pnpm add intlayer next-intlayer
|
|
59
|
+
```
|
|
60
|
+
|
|
61
|
+
```bash packageManager="yarn"
|
|
62
|
+
yarn add intlayer next-intlayer
|
|
63
|
+
```
|
|
64
|
+
|
|
65
|
+
- **intlayer**
|
|
66
|
+
|
|
67
|
+
구성 관리, 번역, [콘텐츠 선언](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/dictionary/content_file.md), 트랜스파일링 및 [CLI 명령어](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/intlayer_cli.md)를 위한 국제화 도구를 제공하는 핵심 패키지입니다.
|
|
68
|
+
|
|
69
|
+
- **next-intlayer**
|
|
70
|
+
|
|
71
|
+
Intlayer를 Next.js와 통합하는 패키지입니다. Next.js 국제화를 위한 컨텍스트 프로바이더와 훅을 제공합니다. 또한, Intlayer를 [Webpack](https://webpack.js.org/) 또는 [Turbopack](https://nextjs.org/docs/app/api-reference/turbopack)과 통합하기 위한 Next.js 플러그인과 사용자의 선호 로케일 감지, 쿠키 관리, URL 리디렉션 처리를 위한 프록시도 포함되어 있습니다.
|
|
72
|
+
|
|
73
|
+
### 2단계: 프로젝트 구성
|
|
74
|
+
|
|
75
|
+
애플리케이션의 언어를 구성하기 위해 설정 파일을 만드세요:
|
|
76
|
+
|
|
77
|
+
```typescript fileName="intlayer.config.ts" codeFormat="typescript"
|
|
78
|
+
import { Locales, type IntlayerConfig } from "intlayer";
|
|
79
|
+
|
|
80
|
+
const config: IntlayerConfig = {
|
|
81
|
+
internationalization: {
|
|
82
|
+
locales: [
|
|
83
|
+
Locales.ENGLISH,
|
|
84
|
+
Locales.FRENCH,
|
|
85
|
+
Locales.SPANISH,
|
|
86
|
+
// 다른 로케일들
|
|
87
|
+
],
|
|
88
|
+
defaultLocale: Locales.ENGLISH,
|
|
89
|
+
},
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
export default config;
|
|
93
|
+
```
|
|
94
|
+
|
|
95
|
+
```javascript fileName="intlayer.config.mjs" codeFormat="esm"
|
|
96
|
+
import { Locales } from "intlayer";
|
|
97
|
+
|
|
98
|
+
/** @type {import('intlayer').IntlayerConfig} */
|
|
99
|
+
const config = {
|
|
100
|
+
internationalization: {
|
|
101
|
+
locales: [
|
|
102
|
+
Locales.ENGLISH,
|
|
103
|
+
Locales.FRENCH,
|
|
104
|
+
Locales.SPANISH,
|
|
105
|
+
// 다른 로케일들
|
|
106
|
+
],
|
|
107
|
+
defaultLocale: Locales.ENGLISH,
|
|
108
|
+
},
|
|
109
|
+
};
|
|
110
|
+
|
|
111
|
+
export default config;
|
|
112
|
+
```
|
|
113
|
+
|
|
114
|
+
```javascript fileName="intlayer.config.cjs" codeFormat="commonjs"
|
|
115
|
+
const { Locales } = require("intlayer");
|
|
116
|
+
|
|
117
|
+
/** @type {import('intlayer').IntlayerConfig} */
|
|
118
|
+
const config = {
|
|
119
|
+
internationalization: {
|
|
120
|
+
locales: [
|
|
121
|
+
Locales.ENGLISH,
|
|
122
|
+
Locales.FRENCH,
|
|
123
|
+
Locales.SPANISH,
|
|
124
|
+
// 다른 로케일들
|
|
125
|
+
],
|
|
126
|
+
defaultLocale: Locales.ENGLISH,
|
|
127
|
+
},
|
|
128
|
+
};
|
|
129
|
+
|
|
130
|
+
module.exports = config;
|
|
131
|
+
```
|
|
132
|
+
|
|
133
|
+
> 이 구성 파일을 통해 지역화된 URL, 프록시 리디렉션, 쿠키 이름, 콘텐츠 선언의 위치 및 확장자 설정, 콘솔에서 Intlayer 로그 비활성화 등 다양한 설정을 할 수 있습니다. 사용 가능한 모든 매개변수 목록은 [구성 문서](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/configuration.md)를 참조하세요.
|
|
134
|
+
|
|
135
|
+
### 3단계: Next.js 구성에 Intlayer 통합하기
|
|
136
|
+
|
|
137
|
+
Next.js 설정을 Intlayer와 함께 사용하도록 구성합니다:
|
|
138
|
+
|
|
139
|
+
```typescript fileName="next.config.ts" codeFormat="typescript"
|
|
140
|
+
import type { NextConfig } from "next";
|
|
141
|
+
import { withIntlayer } from "next-intlayer/server";
|
|
142
|
+
|
|
143
|
+
const nextConfig: NextConfig = {
|
|
144
|
+
/* 여기에 구성 옵션을 입력하세요 */
|
|
145
|
+
};
|
|
146
|
+
|
|
147
|
+
export default withIntlayer(nextConfig);
|
|
148
|
+
```
|
|
149
|
+
|
|
150
|
+
```typescript fileName="next.config.mjs" codeFormat="esm"
|
|
151
|
+
import { withIntlayer } from "next-intlayer/server";
|
|
152
|
+
|
|
153
|
+
/** @type {import('next').NextConfig} */
|
|
154
|
+
const nextConfig = {
|
|
155
|
+
/* 여기에 구성 옵션을 입력하세요 */
|
|
156
|
+
};
|
|
157
|
+
|
|
158
|
+
export default withIntlayer(nextConfig);
|
|
159
|
+
```
|
|
160
|
+
|
|
161
|
+
```typescript fileName="next.config.cjs" codeFormat="commonjs"
|
|
162
|
+
const { withIntlayer } = require("next-intlayer/server");
|
|
163
|
+
|
|
164
|
+
/** @type {import('next').NextConfig} */
|
|
165
|
+
const nextConfig = {
|
|
166
|
+
/* 여기에 구성 옵션을 입력하세요 */
|
|
167
|
+
};
|
|
168
|
+
|
|
169
|
+
module.exports = withIntlayer(nextConfig);
|
|
170
|
+
```
|
|
171
|
+
|
|
172
|
+
> `withIntlayer()` Next.js 플러그인은 Intlayer를 Next.js와 통합하는 데 사용됩니다. 이 플러그인은 콘텐츠 선언 파일의 빌드를 보장하고 개발 모드에서 이를 모니터링합니다. 또한 [Webpack](https://webpack.js.org/) 또는 [Turbopack](https://nextjs.org/docs/app/api-reference/turbopack) 환경 내에서 Intlayer 환경 변수를 정의합니다. 추가로, 성능 최적화를 위한 별칭(alias)을 제공하며 서버 컴포넌트와의 호환성을 보장합니다.
|
|
173
|
+
|
|
174
|
+
> `withIntlayer()` 함수는 프로미스 함수입니다. 빌드가 시작되기 전에 Intlayer 사전을 준비할 수 있게 해줍니다. 다른 플러그인과 함께 사용하려면 `await`할 수 있습니다. 예시:
|
|
175
|
+
>
|
|
176
|
+
> ```tsx
|
|
177
|
+
> const nextConfig = await withIntlayer(nextConfig);
|
|
178
|
+
> const nextConfigWithOtherPlugins = withOtherPlugins(nextConfig);
|
|
179
|
+
>
|
|
180
|
+
> export default nextConfigWithOtherPlugins;
|
|
181
|
+
> ```
|
|
182
|
+
>
|
|
183
|
+
> 동기적으로 사용하고 싶다면 `withIntlayerSync()` 함수를 사용할 수 있습니다. 예시:
|
|
184
|
+
>
|
|
185
|
+
> ```tsx
|
|
186
|
+
> const nextConfig = withIntlayerSync(nextConfig);
|
|
187
|
+
> const nextConfigWithOtherPlugins = withOtherPlugins(nextConfig);
|
|
188
|
+
>
|
|
189
|
+
> export default nextConfigWithOtherPlugins;
|
|
190
|
+
> ```
|
|
191
|
+
|
|
192
|
+
### 4단계: 동적 로케일 경로 정의
|
|
193
|
+
|
|
194
|
+
`RootLayout`의 모든 내용을 제거하고 다음 코드로 교체하세요:
|
|
195
|
+
|
|
196
|
+
```tsx {3} fileName="src/app/layout.tsx" codeFormat="typescript"
|
|
197
|
+
import type { PropsWithChildren, FC } from "react";
|
|
198
|
+
import "./globals.css";
|
|
199
|
+
|
|
200
|
+
const RootLayout: FC<PropsWithChildren> = ({ children }) => (
|
|
201
|
+
// 여전히 `next-themes`, `react-query`, `framer-motion` 등과 같은 다른 프로바이더로 children을 감쌀 수 있습니다.
|
|
202
|
+
<>{children}</>
|
|
203
|
+
);
|
|
204
|
+
|
|
205
|
+
export default RootLayout;
|
|
206
|
+
```
|
|
207
|
+
|
|
208
|
+
```jsx {3} fileName="src/app/layout.mjx" codeFormat="esm"
|
|
209
|
+
import "./globals.css";
|
|
210
|
+
|
|
211
|
+
const RootLayout = ({ children }) => (
|
|
212
|
+
// 여전히 `next-themes`, `react-query`, `framer-motion` 등과 같은 다른 프로바이더로 children을 감쌀 수 있습니다.
|
|
213
|
+
<>{children}</>
|
|
214
|
+
);
|
|
215
|
+
|
|
216
|
+
export default RootLayout;
|
|
217
|
+
```
|
|
218
|
+
|
|
219
|
+
```jsx {1,8} fileName="src/app/layout.csx" codeFormat="commonjs"
|
|
220
|
+
require("./globals.css");
|
|
221
|
+
|
|
222
|
+
const RootLayout = ({ children }) => (
|
|
223
|
+
// 여전히 `next-themes`, `react-query`, `framer-motion` 등과 같은 다른 프로바이더로 children을 감쌀 수 있습니다.
|
|
224
|
+
<>{children}</>
|
|
225
|
+
);
|
|
226
|
+
|
|
227
|
+
module.exports = {
|
|
228
|
+
default: RootLayout,
|
|
229
|
+
generateStaticParams,
|
|
230
|
+
};
|
|
231
|
+
```
|
|
232
|
+
|
|
233
|
+
> `RootLayout` 컴포넌트를 비워두면 `<html>` 태그에 [`lang`](https://developer.mozilla.org/fr/docs/Web/HTML/Global_attributes/lang) 및 [`dir`](https://developer.mozilla.org/fr/docs/Web/HTML/Global_attributes/dir) 속성을 설정할 수 있습니다.
|
|
234
|
+
|
|
235
|
+
동적 라우팅을 구현하려면 `[locale]` 디렉토리에 새 레이아웃을 추가하여 로케일 경로를 제공하세요:
|
|
236
|
+
|
|
237
|
+
```tsx fileName="src/app/[locale]/layout.tsx" codeFormat="typescript"
|
|
238
|
+
import type { NextLayoutIntlayer } from "next-intlayer";
|
|
239
|
+
import { Inter } from "next/font/google";
|
|
240
|
+
import { getHTMLTextDir } from "intlayer";
|
|
241
|
+
|
|
242
|
+
const inter = Inter({ subsets: ["latin"] });
|
|
243
|
+
|
|
244
|
+
const LocaleLayout: NextLayoutIntlayer = async ({ children, params }) => {
|
|
245
|
+
const { locale } = await params;
|
|
246
|
+
return (
|
|
247
|
+
<html lang={locale} dir={getHTMLTextDir(locale)}>
|
|
248
|
+
<body className={inter.className}>{children}</body>
|
|
249
|
+
</html>
|
|
250
|
+
);
|
|
251
|
+
|
|
252
|
+
export default LocaleLayout;
|
|
253
|
+
```
|
|
254
|
+
|
|
255
|
+
```jsx fileName="src/app/[locale]/layout.mjx" codeFormat="esm"
|
|
256
|
+
import { getHTMLTextDir } from "intlayer";
|
|
257
|
+
|
|
258
|
+
const inter = Inter({ subsets: ["latin"] });
|
|
259
|
+
|
|
260
|
+
const LocaleLayout = async ({ children, params: { locale } }) => {
|
|
261
|
+
const { locale } = await params;
|
|
262
|
+
return (
|
|
263
|
+
<html lang={locale} dir={getHTMLTextDir(locale)}>
|
|
264
|
+
<body className={inter.className}>{children}</body>
|
|
265
|
+
</html>
|
|
266
|
+
);
|
|
267
|
+
};
|
|
268
|
+
|
|
269
|
+
export default LocaleLayout;
|
|
270
|
+
```
|
|
271
|
+
|
|
272
|
+
```jsx fileName="src/app/[locale]/layout.csx" codeFormat="commonjs"
|
|
273
|
+
const { Inter } = require("next/font/google");
|
|
274
|
+
const { getHTMLTextDir } = require("intlayer");
|
|
275
|
+
|
|
276
|
+
const inter = Inter({ subsets: ["latin"] });
|
|
277
|
+
|
|
278
|
+
// LocaleLayout 컴포넌트는 동적으로 로케일을 받아 HTML lang 및 dir 속성을 설정합니다.
|
|
279
|
+
const LocaleLayout = async ({ children, params: { locale } }) => {
|
|
280
|
+
const { locale } = await params;
|
|
281
|
+
return (
|
|
282
|
+
<html lang={locale} dir={getHTMLTextDir(locale)}>
|
|
283
|
+
<body className={inter.className}>{children}</body>
|
|
284
|
+
</html>
|
|
285
|
+
);
|
|
286
|
+
};
|
|
287
|
+
|
|
288
|
+
module.exports = LocaleLayout;
|
|
289
|
+
|
|
290
|
+
const inter = Inter({ subsets: ["latin"] });
|
|
291
|
+
|
|
292
|
+
const LocaleLayout = async ({ children, params: { locale } }) => {
|
|
293
|
+
const { locale } = await params;
|
|
294
|
+
return (
|
|
295
|
+
<html lang={locale} dir={getHTMLTextDir(locale)}>
|
|
296
|
+
<body className={inter.className}>{children}</body>
|
|
297
|
+
</html>
|
|
298
|
+
);
|
|
299
|
+
};
|
|
300
|
+
|
|
301
|
+
module.exports = LocaleLayout;
|
|
302
|
+
```
|
|
303
|
+
|
|
304
|
+
> `[locale]` 경로 세그먼트는 로케일을 정의하는 데 사용됩니다. 예: `/en-US/about`는 `en-US`를 가리키고 `/fr/about`는 `fr`를 가리킵니다.
|
|
305
|
+
|
|
306
|
+
> 이 단계에서 `Error: Missing <html> and <body> tags in the root layout.` 오류가 발생할 수 있습니다. 이는 `/app/page.tsx` 파일이 더 이상 사용되지 않으며 삭제할 수 있기 때문에 예상되는 현상입니다. 대신, `[locale]` 경로 세그먼트가 `/app/[locale]/page.tsx` 페이지를 활성화합니다. 따라서 브라우저에서 `/en`, `/fr`, `/es`와 같은 경로를 통해 페이지에 접근할 수 있습니다. 기본 로케일을 루트 페이지로 설정하려면 7단계의 `proxy` 설정을 참조하세요.
|
|
307
|
+
|
|
308
|
+
그런 다음, 애플리케이션 Layout에서 `generateStaticParams` 함수를 구현하세요.
|
|
309
|
+
|
|
310
|
+
```tsx {1} fileName="src/app/[locale]/layout.tsx" codeFormat="typescript"
|
|
311
|
+
export { generateStaticParams } from "next-intlayer"; // 삽입할 라인
|
|
312
|
+
|
|
313
|
+
const LocaleLayout: NextLayoutIntlayer = async ({ children, params }) => {
|
|
314
|
+
/*... 나머지 코드 */
|
|
315
|
+
};
|
|
316
|
+
|
|
317
|
+
tsx {1} fileName="src/app/[locale]/layout.tsx" codeFormat="typescript"
|
|
318
|
+
export { generateStaticParams } from "next-intlayer"; // 삽입할 라인
|
|
319
|
+
|
|
320
|
+
const LocaleLayout: NextLayoutIntlayer = async ({ children, params }) => {
|
|
321
|
+
/*... 나머지 코드 */
|
|
322
|
+
};
|
|
323
|
+
|
|
324
|
+
export default LocaleLayout;
|
|
325
|
+
```
|
|
326
|
+
|
|
327
|
+
```jsx {1} fileName="src/app/[locale]/layout.mjx" codeFormat="esm"
|
|
328
|
+
export { generateStaticParams } from "next-intlayer"; // 삽입할 라인
|
|
329
|
+
|
|
330
|
+
const LocaleLayout = async ({ children, params: { locale } }) => {
|
|
331
|
+
/*... 나머지 코드 */
|
|
332
|
+
};
|
|
333
|
+
|
|
334
|
+
// ... 나머지 코드
|
|
335
|
+
```
|
|
336
|
+
|
|
337
|
+
```jsx {1,7} fileName="src/app/[locale]/layout.csx" codeFormat="commonjs"
|
|
338
|
+
const { generateStaticParams } = require("next-intlayer"); // 삽입할 라인
|
|
339
|
+
|
|
340
|
+
const LocaleLayout = async ({ children, params: { locale } }) => {
|
|
341
|
+
/*... 나머지 코드 */
|
|
342
|
+
};
|
|
343
|
+
|
|
344
|
+
module.exports = { default: LocaleLayout, generateStaticParams };
|
|
345
|
+
```
|
|
346
|
+
|
|
347
|
+
> `generateStaticParams`는 애플리케이션이 모든 로케일에 대해 필요한 페이지를 사전 빌드하도록 보장하여 런타임 계산을 줄이고 사용자 경험을 향상시킵니다. 자세한 내용은 [Next.js의 generateStaticParams 문서](https://nextjs.org/docs/app/building-your-application/rendering/static-and-dynamic-rendering#generate-static-params)를 참조하세요.
|
|
348
|
+
|
|
349
|
+
> Intlayer는 `export const dynamic = 'force-static';`와 함께 작동하여 모든 로케일에 대해 페이지가 사전 빌드되도록 보장합니다.
|
|
350
|
+
|
|
351
|
+
### 5단계: 콘텐츠 선언하기
|
|
352
|
+
|
|
353
|
+
번역을 저장하기 위해 콘텐츠 선언을 생성하고 관리하세요:
|
|
354
|
+
|
|
355
|
+
```tsx fileName="src/app/[locale]/page.content.ts" contentDeclarationFormat="typescript"
|
|
356
|
+
import { t, type Dictionary } from "intlayer";
|
|
357
|
+
|
|
358
|
+
const pageContent = {
|
|
359
|
+
key: "page",
|
|
360
|
+
content: {
|
|
361
|
+
getStarted: {
|
|
362
|
+
main: t({
|
|
363
|
+
en: "Get started by editing",
|
|
364
|
+
fr: "Commencez par éditer",
|
|
365
|
+
es: "Comience por editar",
|
|
366
|
+
}),
|
|
367
|
+
pageLink: "src/app/page.tsx",
|
|
368
|
+
},
|
|
369
|
+
},
|
|
370
|
+
} satisfies Dictionary;
|
|
371
|
+
|
|
372
|
+
export default pageContent;
|
|
373
|
+
```
|
|
374
|
+
|
|
375
|
+
```javascript fileName="src/app/[locale]/page.content.mjs" contentDeclarationFormat="esm"
|
|
376
|
+
import { t } from "intlayer";
|
|
377
|
+
|
|
378
|
+
/** @type {import('intlayer').Dictionary} */
|
|
379
|
+
const pageContent = {
|
|
380
|
+
key: "page",
|
|
381
|
+
content: {
|
|
382
|
+
getStarted: {
|
|
383
|
+
main: t({
|
|
384
|
+
en: "Get started by editing",
|
|
385
|
+
fr: "Commencez par éditer",
|
|
386
|
+
es: "Comience por editar",
|
|
387
|
+
}),
|
|
388
|
+
pageLink: "src/app/page.tsx",
|
|
389
|
+
},
|
|
390
|
+
},
|
|
391
|
+
};
|
|
392
|
+
|
|
393
|
+
export default pageContent;
|
|
394
|
+
```
|
|
395
|
+
|
|
396
|
+
```javascript fileName="src/app/[locale]/page.content.cjs" contentDeclarationFormat="commonjs"
|
|
397
|
+
const { t } = require("intlayer");
|
|
398
|
+
|
|
399
|
+
/** @type {import('intlayer').Dictionary} */
|
|
400
|
+
const pageContent = {
|
|
401
|
+
key: "page",
|
|
402
|
+
content: {
|
|
403
|
+
getStarted: {
|
|
404
|
+
main: t({
|
|
405
|
+
en: "Get started by editing",
|
|
406
|
+
fr: "Commencez par éditer",
|
|
407
|
+
es: "Comience por editar",
|
|
408
|
+
}),
|
|
409
|
+
pageLink: "src/app/page.tsx",
|
|
410
|
+
},
|
|
411
|
+
},
|
|
412
|
+
};
|
|
413
|
+
|
|
414
|
+
module.exports = pageContent;
|
|
415
|
+
```
|
|
416
|
+
|
|
417
|
+
```json fileName="src/app/[locale]/page.content.json" contentDeclarationFormat="json"
|
|
418
|
+
{
|
|
419
|
+
"$schema": "https://intlayer.org/schema.json",
|
|
420
|
+
"key": "page",
|
|
421
|
+
"content": {
|
|
422
|
+
"getStarted": {
|
|
423
|
+
"nodeType": "translation",
|
|
424
|
+
"translation": {
|
|
425
|
+
"en": "수정을 시작하세요",
|
|
426
|
+
"fr": "Commencez par éditer",
|
|
427
|
+
"es": "Comience por editar"
|
|
428
|
+
}
|
|
429
|
+
},
|
|
430
|
+
"pageLink": "src/app/page.tsx"
|
|
431
|
+
}
|
|
432
|
+
}
|
|
433
|
+
```
|
|
434
|
+
|
|
435
|
+
> 콘텐츠 선언은 애플리케이션 내 어디에서나 정의할 수 있으며, `contentDir` 디렉토리(기본값: `./src`)에 포함되기만 하면 됩니다. 그리고 콘텐츠 선언 파일 확장자(기본값: `.content.{json,ts,tsx,js,jsx,mjs,mjx,cjs,cjx}`)와 일치해야 합니다.
|
|
436
|
+
|
|
437
|
+
> 자세한 내용은 [콘텐츠 선언 문서](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/dictionary/content_file.md)를 참조하세요.
|
|
438
|
+
|
|
439
|
+
### 6단계: 코드에서 콘텐츠 활용하기
|
|
440
|
+
|
|
441
|
+
애플리케이션 전반에서 콘텐츠 사전을 접근하세요:
|
|
442
|
+
|
|
443
|
+
```tsx fileName="src/app/[locale]/page.tsx" codeFormat="typescript"
|
|
444
|
+
import type { FC } from "react";
|
|
445
|
+
import { ClientComponentExample } from "@components/ClientComponentExample";
|
|
446
|
+
import { ServerComponentExample } from "@components/ServerComponentExample";
|
|
447
|
+
import { type NextPageIntlayer, IntlayerClientProvider } from "next-intlayer";
|
|
448
|
+
import { IntlayerServerProvider, useIntlayer } from "next-intlayer/server";
|
|
449
|
+
|
|
450
|
+
const PageContent: FC = () => {
|
|
451
|
+
const content = useIntlayer("page");
|
|
452
|
+
|
|
453
|
+
return (
|
|
454
|
+
<>
|
|
455
|
+
<p>{content.getStarted.main}</p> {/* 시작하기 메인 내용 */}
|
|
456
|
+
<code>{content.getStarted.pageLink}</code> {/* 시작하기 페이지 링크 */}
|
|
457
|
+
</>
|
|
458
|
+
);
|
|
459
|
+
};
|
|
460
|
+
|
|
461
|
+
const Page: NextPageIntlayer = async ({ params }) => {
|
|
462
|
+
const { locale } = await params;
|
|
463
|
+
|
|
464
|
+
return (
|
|
465
|
+
<IntlayerServerProvider locale={locale}>
|
|
466
|
+
<PageContent />
|
|
467
|
+
<ServerComponentExample />
|
|
468
|
+
|
|
469
|
+
<IntlayerClientProvider locale={locale}>
|
|
470
|
+
<ClientComponentExample />
|
|
471
|
+
</IntlayerClientProvider>
|
|
472
|
+
</IntlayerServerProvider>
|
|
473
|
+
);
|
|
474
|
+
};
|
|
475
|
+
|
|
476
|
+
export default Page;
|
|
477
|
+
```
|
|
478
|
+
|
|
479
|
+
```jsx fileName="src/app/[locale]/page.mjx" codeFormat="esm"
|
|
480
|
+
import { ClientComponentExample } from "@components/ClientComponentExample";
|
|
481
|
+
import { ServerComponentExample } from "@components/ServerComponentExample";
|
|
482
|
+
import { IntlayerClientProvider } from "next-intlayer";
|
|
483
|
+
import { IntlayerServerProvider, useIntlayer } from "next-intlayer/server";
|
|
484
|
+
|
|
485
|
+
const PageContent = () => {
|
|
486
|
+
const content = useIntlayer("page");
|
|
487
|
+
|
|
488
|
+
return (
|
|
489
|
+
<>
|
|
490
|
+
<p>{content.getStarted.main}</p>
|
|
491
|
+
<code>{content.getStarted.pageLink}</code>
|
|
492
|
+
</>
|
|
493
|
+
);
|
|
494
|
+
};
|
|
495
|
+
|
|
496
|
+
const Page = async ({ params }) => {
|
|
497
|
+
const { locale } = await params;
|
|
498
|
+
|
|
499
|
+
return (
|
|
500
|
+
<IntlayerServerProvider locale={locale}>
|
|
501
|
+
<PageContent />
|
|
502
|
+
<ServerComponentExample />
|
|
503
|
+
|
|
504
|
+
<IntlayerClientProvider locale={locale}>
|
|
505
|
+
<ClientComponentExample />
|
|
506
|
+
</IntlayerClientProvider>
|
|
507
|
+
</IntlayerServerProvider>
|
|
508
|
+
);
|
|
509
|
+
};
|
|
510
|
+
|
|
511
|
+
export default Page;
|
|
512
|
+
```
|
|
513
|
+
|
|
514
|
+
```jsx fileName="src/app/[locale]/page.csx" codeFormat="commonjs"
|
|
515
|
+
import { ClientComponentExample } from "@components/ClientComponentExample";
|
|
516
|
+
import { ServerComponentExample } from "@components/ServerComponentExample";
|
|
517
|
+
import { IntlayerClientProvider } from "next-intlayer";
|
|
518
|
+
import { IntlayerServerProvider, useIntlayer } from "next-intlayer/server";
|
|
519
|
+
|
|
520
|
+
const PageContent = () => {
|
|
521
|
+
const content = useIntlayer("page");
|
|
522
|
+
|
|
523
|
+
return (
|
|
524
|
+
<>
|
|
525
|
+
<p>{content.getStarted.main}</p>
|
|
526
|
+
<code>{content.getStarted.pageLink}</code>
|
|
527
|
+
</>
|
|
528
|
+
);
|
|
529
|
+
};
|
|
530
|
+
|
|
531
|
+
const Page = async ({ params }) => {
|
|
532
|
+
const { locale } = await params;
|
|
533
|
+
|
|
534
|
+
return (
|
|
535
|
+
<IntlayerServerProvider locale={locale}>
|
|
536
|
+
<PageContent />
|
|
537
|
+
<ServerComponentExample />
|
|
538
|
+
|
|
539
|
+
<IntlayerClientProvider locale={locale}>
|
|
540
|
+
<ClientComponentExample />
|
|
541
|
+
</IntlayerClientProvider>
|
|
542
|
+
</IntlayerServerProvider>
|
|
543
|
+
);
|
|
544
|
+
};
|
|
545
|
+
```
|
|
546
|
+
|
|
547
|
+
- **`IntlayerClientProvider`**는 클라이언트 측 컴포넌트에 로케일을 제공하는 데 사용됩니다. 레이아웃을 포함한 모든 상위 컴포넌트에 배치할 수 있습니다. 그러나 Next.js가 페이지 간에 레이아웃 코드를 공유하기 때문에 레이아웃에 배치하는 것이 권장됩니다. 레이아웃에서 `IntlayerClientProvider`를 사용하면 각 페이지마다 다시 초기화하는 것을 방지하여 성능을 향상시키고 애플리케이션 전체에서 일관된 로컬라이제이션 컨텍스트를 유지할 수 있습니다.
|
|
548
|
+
- **`IntlayerServerProvider`**는 서버 자식 컴포넌트에 로케일을 제공하는 데 사용됩니다. 레이아웃에서는 설정할 수 없습니다.
|
|
549
|
+
|
|
550
|
+
> 레이아웃과 페이지는 공통 서버 컨텍스트를 공유할 수 없습니다. 서버 컨텍스트 시스템은 요청별 데이터 저장소([React의 cache](https://react.dev/reference/react/cache) 메커니즘)를 기반으로 하여 애플리케이션의 서로 다른 세그먼트마다 각 "컨텍스트"가 다시 생성되기 때문입니다. 공급자를 공유 레이아웃에 배치하면 이 격리가 깨져 서버 컨텍스트 값이 서버 컴포넌트에 올바르게 전달되지 않습니다.
|
|
551
|
+
|
|
552
|
+
> 레이아웃과 페이지는 공통 서버 컨텍스트를 공유할 수 없습니다. 서버 컨텍스트 시스템이 요청별 데이터 저장소(React의 [캐시](https://react.dev/reference/react/cache) 메커니즘)를 기반으로 하기 때문에 애플리케이션의 서로 다른 세그먼트마다 각 "컨텍스트"가 다시 생성됩니다. 프로바이더를 공유 레이아웃에 배치하면 이러한 격리가 깨져 서버 컨텍스트 값이 서버 컴포넌트에 올바르게 전달되지 않습니다.
|
|
553
|
+
|
|
554
|
+
```tsx {4,7} fileName="src/components/ClientComponentExample.tsx" codeFormat="typescript"
|
|
555
|
+
"use client";
|
|
556
|
+
|
|
557
|
+
import type { FC } from "react";
|
|
558
|
+
import { useIntlayer } from "next-intlayer";
|
|
559
|
+
|
|
560
|
+
export const ClientComponentExample: FC = () => {
|
|
561
|
+
const content = useIntlayer("client-component-example"); // 관련 콘텐츠 선언 생성
|
|
562
|
+
|
|
563
|
+
return (
|
|
564
|
+
<div>
|
|
565
|
+
<h2>{content.title}</h2>
|
|
566
|
+
<p>{content.content}</p>
|
|
567
|
+
</div>
|
|
568
|
+
);
|
|
569
|
+
};
|
|
570
|
+
```
|
|
571
|
+
|
|
572
|
+
```jsx {3,6} fileName="src/components/ClientComponentExample.mjx" codeFormat="esm"
|
|
573
|
+
"use client";
|
|
574
|
+
|
|
575
|
+
import { useIntlayer } from "next-intlayer";
|
|
576
|
+
|
|
577
|
+
const ClientComponentExample = () => {
|
|
578
|
+
const content = useIntlayer("client-component-example"); // 관련 콘텐츠 선언 생성
|
|
579
|
+
|
|
580
|
+
return (
|
|
581
|
+
<div>
|
|
582
|
+
<h2>{content.title}</h2>
|
|
583
|
+
<p>{content.content}</p>
|
|
584
|
+
</div>
|
|
585
|
+
);
|
|
586
|
+
};
|
|
587
|
+
```
|
|
588
|
+
|
|
589
|
+
```jsx {3,6} fileName="src/components/ClientComponentExample.csx" codeFormat="commonjs"
|
|
590
|
+
"use client";
|
|
591
|
+
|
|
592
|
+
const { useIntlayer } = require("next-intlayer");
|
|
593
|
+
|
|
594
|
+
const ClientComponentExample = () => {
|
|
595
|
+
const content = useIntlayer("client-component-example"); // 관련 콘텐츠 선언 생성
|
|
596
|
+
|
|
597
|
+
return (
|
|
598
|
+
<div>
|
|
599
|
+
<h2>{content.title}</h2>
|
|
600
|
+
<p>{content.content}</p>
|
|
601
|
+
</div>
|
|
602
|
+
);
|
|
603
|
+
};
|
|
604
|
+
```
|
|
605
|
+
|
|
606
|
+
```tsx {2} fileName="src/components/ServerComponentExample.tsx" codeFormat="typescript"
|
|
607
|
+
import type { FC } from "react";
|
|
608
|
+
import { useIntlayer } from "next-intlayer/server";
|
|
609
|
+
|
|
610
|
+
export const ServerComponentExample: FC = () => {
|
|
611
|
+
const content = useIntlayer("server-component-example"); // 관련 콘텐츠 선언 생성
|
|
612
|
+
|
|
613
|
+
return (
|
|
614
|
+
<div>
|
|
615
|
+
<h2>{content.title}</h2>
|
|
616
|
+
<p>{content.content}</p>
|
|
617
|
+
</div>
|
|
618
|
+
);
|
|
619
|
+
};
|
|
620
|
+
```
|
|
621
|
+
|
|
622
|
+
```jsx {1} fileName="src/components/ServerComponentExample.mjx" codeFormat="esm"
|
|
623
|
+
import { useIntlayer } from "next-intlayer/server";
|
|
624
|
+
|
|
625
|
+
const ServerComponentExample = () => {
|
|
626
|
+
const content = useIntlayer("server-component-example"); // 관련 콘텐츠 선언 생성
|
|
627
|
+
|
|
628
|
+
return (
|
|
629
|
+
<div>
|
|
630
|
+
<h2>{content.title}</h2>
|
|
631
|
+
<p>{content.content}</p>
|
|
632
|
+
</div>
|
|
633
|
+
);
|
|
634
|
+
};
|
|
635
|
+
```
|
|
636
|
+
|
|
637
|
+
```jsx {1} fileName="src/components/ServerComponentExample.csx" codeFormat="commonjs"
|
|
638
|
+
const { useIntlayer } = require("next-intlayer/server");
|
|
639
|
+
|
|
640
|
+
const ServerComponentExample = () => {
|
|
641
|
+
const content = useIntlayer("server-component-example"); // 관련 콘텐츠 선언 생성
|
|
642
|
+
|
|
643
|
+
return (
|
|
644
|
+
<div>
|
|
645
|
+
<h2>{content.title}</h2>
|
|
646
|
+
<p>{content.content}</p>
|
|
647
|
+
</div>
|
|
648
|
+
);
|
|
649
|
+
};
|
|
650
|
+
```
|
|
651
|
+
|
|
652
|
+
> 콘텐츠를 `alt`, `title`, `href`, `aria-label` 등과 같은 `string` 속성에서 사용하려면, 함수의 값을 호출해야 합니다. 예를 들어:
|
|
653
|
+
|
|
654
|
+
> ```jsx
|
|
655
|
+
> <img src={content.image.src.value} alt={content.image.value} />
|
|
656
|
+
> ```
|
|
657
|
+
|
|
658
|
+
> `useIntlayer` 훅에 대해 더 알아보려면 [문서](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/packages/next-intlayer/useIntlayer.md)를 참조하세요.
|
|
659
|
+
|
|
660
|
+
### (선택 사항) 7단계: 로케일 감지를 위한 프록시 구성
|
|
661
|
+
|
|
662
|
+
사용자의 선호 로케일을 감지하기 위해 프록시를 설정합니다:
|
|
663
|
+
|
|
664
|
+
```typescript fileName="src/proxy.ts" codeFormat="typescript"
|
|
665
|
+
export { intlayerProxy as proxy } from "next-intlayer/proxy";
|
|
666
|
+
|
|
667
|
+
export const config = {
|
|
668
|
+
matcher:
|
|
669
|
+
"/((?!api|static|assets|robots|sitemap|sw|service-worker|manifest|.*\\..*|_next).*)",
|
|
670
|
+
};
|
|
671
|
+
```
|
|
672
|
+
|
|
673
|
+
```javascript fileName="src/proxy.mjs" codeFormat="esm"
|
|
674
|
+
export { intlayerProxy as proxy } from "next-intlayer/proxy";
|
|
675
|
+
|
|
676
|
+
export const config = {
|
|
677
|
+
matcher:
|
|
678
|
+
"/((?!api|static|assets|robots|sitemap|sw|service-worker|manifest|.*\\..*|_next).*)",
|
|
679
|
+
};
|
|
680
|
+
```
|
|
681
|
+
|
|
682
|
+
```javascript fileName="src/proxy.cjs" codeFormat="commonjs"
|
|
683
|
+
const { intlayerProxy } = require("next-intlayer/proxy");
|
|
684
|
+
|
|
685
|
+
const config = {
|
|
686
|
+
matcher:
|
|
687
|
+
"/((?!api|static|assets|robots|sitemap|sw|service-worker|manifest|.*\\..*|_next).*)",
|
|
688
|
+
};
|
|
689
|
+
|
|
690
|
+
module.exports = { proxy: intlayerProxy, config };
|
|
691
|
+
```
|
|
692
|
+
|
|
693
|
+
> `intlayerProxy`는 사용자의 선호 로케일을 감지하여 [구성](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/configuration.md)에 명시된 적절한 URL로 리디렉션하는 데 사용됩니다. 또한 사용자의 선호 로케일을 쿠키에 저장할 수 있도록 합니다.
|
|
694
|
+
|
|
695
|
+
> 여러 프록시를 함께 연결해야 하는 경우(예: 인증 또는 커스텀 프록시와 함께 `intlayerProxy`를 사용하는 경우), Intlayer는 이제 `multipleProxies`라는 헬퍼를 제공합니다.
|
|
696
|
+
|
|
697
|
+
```ts
|
|
698
|
+
import { multipleProxies, intlayerProxy } from "next-intlayer/proxy";
|
|
699
|
+
import { customProxy } from "@utils/customProxy";
|
|
700
|
+
|
|
701
|
+
export const proxy = multipleProxies([intlayerProxy, customProxy]);
|
|
702
|
+
```
|
|
703
|
+
|
|
704
|
+
### (선택 사항) 8단계: 메타데이터의 국제화
|
|
705
|
+
|
|
706
|
+
페이지 제목과 같은 메타데이터를 국제화하려는 경우, Next.js에서 제공하는 `generateMetadata` 함수를 사용할 수 있습니다. 내부에서 `getIntlayer` 함수로부터 콘텐츠를 가져와 메타데이터를 번역할 수 있습니다.
|
|
707
|
+
|
|
708
|
+
```typescript fileName="src/app/[locale]/metadata.content.ts" contentDeclarationFormat="typescript"
|
|
709
|
+
import { type Dictionary, t } from "intlayer";
|
|
710
|
+
import { Metadata } from "next";
|
|
711
|
+
|
|
712
|
+
const metadataContent = {
|
|
713
|
+
key: "page-metadata",
|
|
714
|
+
content: {
|
|
715
|
+
title: t({
|
|
716
|
+
en: "Create Next App",
|
|
717
|
+
fr: "Créer une application Next.js",
|
|
718
|
+
es: "Crear una aplicación Next.js",
|
|
719
|
+
}),
|
|
720
|
+
description: t({
|
|
721
|
+
en: "Generated by create next app",
|
|
722
|
+
fr: "Généré par create next app",
|
|
723
|
+
es: "Generado por create next app",
|
|
724
|
+
}),
|
|
725
|
+
},
|
|
726
|
+
} satisfies Dictionary<Metadata>;
|
|
727
|
+
|
|
728
|
+
export default metadataContent;
|
|
729
|
+
```
|
|
730
|
+
|
|
731
|
+
```javascript fileName="src/app/[locale]/metadata.content.mjs" contentDeclarationFormat="esm"
|
|
732
|
+
import { t } from "intlayer";
|
|
733
|
+
|
|
734
|
+
/** @type {import('intlayer').Dictionary<import('next').Metadata>} */
|
|
735
|
+
const metadataContent = {
|
|
736
|
+
key: "page-metadata",
|
|
737
|
+
content: {
|
|
738
|
+
title: t({
|
|
739
|
+
en: "Create Next App",
|
|
740
|
+
fr: "Créer une application Next.js",
|
|
741
|
+
es: "Crear una aplicación Next.js",
|
|
742
|
+
}),
|
|
743
|
+
description: t({
|
|
744
|
+
en: "create next app에 의해 생성됨", // create next app에 의해 생성됨
|
|
745
|
+
fr: "Généré par create next app",
|
|
746
|
+
es: "Generado por create next app",
|
|
747
|
+
}),
|
|
748
|
+
},
|
|
749
|
+
};
|
|
750
|
+
|
|
751
|
+
export default metadataContent;
|
|
752
|
+
```
|
|
753
|
+
|
|
754
|
+
```javascript fileName="src/app/[locale]/metadata.content.cjs" contentDeclarationFormat="commonjs"
|
|
755
|
+
const { t } = require("intlayer");
|
|
756
|
+
|
|
757
|
+
/** @type {import('intlayer').Dictionary<import('next').Metadata>} */
|
|
758
|
+
const metadataContent = {
|
|
759
|
+
key: "page-metadata",
|
|
760
|
+
content: {
|
|
761
|
+
title: t({
|
|
762
|
+
en: "Create Next App",
|
|
763
|
+
fr: "Créer une application Next.js",
|
|
764
|
+
es: "Crear una aplicación Next.js",
|
|
765
|
+
}),
|
|
766
|
+
description: t({
|
|
767
|
+
en: "create next app에 의해 생성됨", // create next app에 의해 생성됨
|
|
768
|
+
fr: "Généré par create next app",
|
|
769
|
+
es: "Generado por create next app",
|
|
770
|
+
}),
|
|
771
|
+
},
|
|
772
|
+
};
|
|
773
|
+
|
|
774
|
+
module.exports = metadataContent;
|
|
775
|
+
fr: "Généré par create next app",
|
|
776
|
+
es: "Generado por create next app",
|
|
777
|
+
}),
|
|
778
|
+
},
|
|
779
|
+
};
|
|
780
|
+
|
|
781
|
+
export default metadataContent;
|
|
782
|
+
```
|
|
783
|
+
|
|
784
|
+
```javascript fileName="src/app/[locale]/metadata.content.cjs" contentDeclarationFormat="commonjs"
|
|
785
|
+
const { t } = require("intlayer");
|
|
786
|
+
|
|
787
|
+
/** @type {import('intlayer').Dictionary<import('next').Metadata>} */
|
|
788
|
+
const metadataContent = {
|
|
789
|
+
key: "page-metadata",
|
|
790
|
+
content: {
|
|
791
|
+
title: t({
|
|
792
|
+
en: "Create Next App",
|
|
793
|
+
fr: "Créer une application Next.js",
|
|
794
|
+
es: "Crear una aplicación Next.js",
|
|
795
|
+
}),
|
|
796
|
+
description: t({
|
|
797
|
+
en: "Generated by create next app",
|
|
798
|
+
fr: "Généré par create next app",
|
|
799
|
+
es: "Generado por create next app",
|
|
800
|
+
}),
|
|
801
|
+
},
|
|
802
|
+
};
|
|
803
|
+
|
|
804
|
+
module.exports = metadataContent;
|
|
805
|
+
```
|
|
806
|
+
|
|
807
|
+
```json fileName="src/app/[locale]/metadata.content.json" contentDeclarationFormat="json"
|
|
808
|
+
{
|
|
809
|
+
"key": "page-metadata",
|
|
810
|
+
"content": {
|
|
811
|
+
"title": {
|
|
812
|
+
"nodeType": "translation",
|
|
813
|
+
"translation": {
|
|
814
|
+
"ko": "프리액트 로고",
|
|
815
|
+
"en": "Preact logo",
|
|
816
|
+
"fr": "Logo Preact",
|
|
817
|
+
"es": "Logo Preact"
|
|
818
|
+
}
|
|
819
|
+
},
|
|
820
|
+
"description": {
|
|
821
|
+
"nodeType": "translation",
|
|
822
|
+
"translation": {
|
|
823
|
+
"ko": "create next app에 의해 생성됨",
|
|
824
|
+
"en": "Generated by create next app",
|
|
825
|
+
"fr": "Généré par create next app",
|
|
826
|
+
"es": "Generado por create next app"
|
|
827
|
+
}
|
|
828
|
+
}
|
|
829
|
+
}
|
|
830
|
+
}
|
|
831
|
+
```
|
|
832
|
+
|
|
833
|
+
````typescript fileName="src/app/[locale]/layout.tsx or src/app/[locale]/page.tsx" codeFormat="typescript"
|
|
834
|
+
import { getIntlayer, getMultilingualUrls } from "intlayer";
|
|
835
|
+
import type { Metadata } from "next";
|
|
836
|
+
import type { LocalPromiseParams } from "next-intlayer";
|
|
837
|
+
|
|
838
|
+
export const generateMetadata = async ({
|
|
839
|
+
params,
|
|
840
|
+
}: LocalPromiseParams): Promise<Metadata> => {
|
|
841
|
+
const { locale } = await params;
|
|
842
|
+
|
|
843
|
+
const metadata = getIntlayer("page-metadata", locale);
|
|
844
|
+
|
|
845
|
+
/**
|
|
846
|
+
* 각 로케일에 대한 모든 URL을 포함하는 객체를 생성합니다.
|
|
847
|
+
*
|
|
848
|
+
* 예시:
|
|
849
|
+
* ```ts
|
|
850
|
+
* getMultilingualUrls('/about');
|
|
851
|
+
*
|
|
852
|
+
* // 반환값
|
|
853
|
+
* // {
|
|
854
|
+
* // en: '/about',
|
|
855
|
+
* // fr: '/fr/about',
|
|
856
|
+
* // es: '/es/about',
|
|
857
|
+
* // }
|
|
858
|
+
* ```
|
|
859
|
+
*/
|
|
860
|
+
const multilingualUrls = getMultilingualUrls("/");
|
|
861
|
+
|
|
862
|
+
return {
|
|
863
|
+
...metadata,
|
|
864
|
+
alternates: {
|
|
865
|
+
canonical: multilingualUrls[locale as keyof typeof multilingualUrls],
|
|
866
|
+
languages: { ...multilingualUrls, "x-default": "/" },
|
|
867
|
+
},
|
|
868
|
+
openGraph: {
|
|
869
|
+
url: multilingualUrls[locale],
|
|
870
|
+
},
|
|
871
|
+
};
|
|
872
|
+
};
|
|
873
|
+
|
|
874
|
+
// ... 나머지 코드
|
|
875
|
+
````
|
|
876
|
+
|
|
877
|
+
````javascript fileName="src/app/[locale]/layout.mjs or src/app/[locale]/page.mjs" codeFormat="esm"
|
|
878
|
+
import { getIntlayer, getMultilingualUrls } from "intlayer";
|
|
879
|
+
|
|
880
|
+
export const generateMetadata = async ({ params }) => {
|
|
881
|
+
const { locale } = await params;
|
|
882
|
+
|
|
883
|
+
const metadata = getIntlayer("page-metadata", locale);
|
|
884
|
+
|
|
885
|
+
/**
|
|
886
|
+
* 각 로케일에 대한 모든 URL을 포함하는 객체를 생성합니다.
|
|
887
|
+
*
|
|
888
|
+
* 예시:
|
|
889
|
+
* ```ts
|
|
890
|
+
* getMultilingualUrls('/about');
|
|
891
|
+
*
|
|
892
|
+
* // 반환값
|
|
893
|
+
* // {
|
|
894
|
+
* // en: '/about',
|
|
895
|
+
* // fr: '/fr/about',
|
|
896
|
+
* // es: '/es/about'
|
|
897
|
+
* // }
|
|
898
|
+
* ```
|
|
899
|
+
*/
|
|
900
|
+
const multilingualUrls = getMultilingualUrls("/");
|
|
901
|
+
|
|
902
|
+
return {
|
|
903
|
+
...metadata,
|
|
904
|
+
alternates: {
|
|
905
|
+
canonical: multilingualUrls[locale],
|
|
906
|
+
languages: { ...multilingualUrls, "x-default": "/" },
|
|
907
|
+
},
|
|
908
|
+
openGraph: {
|
|
909
|
+
url: multilingualUrls[locale],
|
|
910
|
+
},
|
|
911
|
+
};
|
|
912
|
+
};
|
|
913
|
+
|
|
914
|
+
// ... 나머지 코드
|
|
915
|
+
````
|
|
916
|
+
|
|
917
|
+
````javascript fileName="src/app/[locale]/layout.cjs or src/app/[locale]/page.cjs" codeFormat="commonjs"
|
|
918
|
+
const { getIntlayer, getMultilingualUrls } = require("intlayer");
|
|
919
|
+
|
|
920
|
+
const generateMetadata = async ({ params }) => {
|
|
921
|
+
const { locale } = await params;
|
|
922
|
+
|
|
923
|
+
const metadata = getIntlayer("page-metadata", locale);
|
|
924
|
+
|
|
925
|
+
/**
|
|
926
|
+
* 각 로케일에 대한 모든 URL을 포함하는 객체를 생성합니다.
|
|
927
|
+
*
|
|
928
|
+
* 예시:
|
|
929
|
+
* ```ts
|
|
930
|
+
* getMultilingualUrls('/about');
|
|
931
|
+
*
|
|
932
|
+
* // 반환값
|
|
933
|
+
* // {
|
|
934
|
+
* // en: '/about',
|
|
935
|
+
* // fr: '/fr/about',
|
|
936
|
+
* // es: '/es/about'
|
|
937
|
+
* // }
|
|
938
|
+
* ```
|
|
939
|
+
*/
|
|
940
|
+
const multilingualUrls = getMultilingualUrls("/");
|
|
941
|
+
|
|
942
|
+
return {
|
|
943
|
+
...metadata,
|
|
944
|
+
alternates: {
|
|
945
|
+
canonical: multilingualUrls[locale],
|
|
946
|
+
languages: { ...multilingualUrls, "x-default": "/" },
|
|
947
|
+
},
|
|
948
|
+
openGraph: {
|
|
949
|
+
url: multilingualUrls[locale],
|
|
950
|
+
},
|
|
951
|
+
};
|
|
952
|
+
};
|
|
953
|
+
|
|
954
|
+
module.exports = { generateMetadata };
|
|
955
|
+
|
|
956
|
+
// ... 나머지 코드
|
|
957
|
+
````
|
|
958
|
+
|
|
959
|
+
> `next-intlayer`에서 가져온 `getIntlayer` 함수는 콘텐츠를 `IntlayerNode`로 래핑하여 시각적 편집기와의 통합을 가능하게 합니다. 반면, `intlayer`에서 가져온 `getIntlayer` 함수는 추가 속성 없이 콘텐츠를 직접 반환합니다.
|
|
960
|
+
|
|
961
|
+
또는 `getTranslation` 함수를 사용하여 메타데이터를 선언할 수 있습니다. 그러나 메타데이터의 번역을 자동화하고 콘텐츠를 외부화하기 위해 콘텐츠 선언 파일을 사용하는 것이 권장됩니다.
|
|
962
|
+
|
|
963
|
+
```typescript fileName="src/app/[locale]/layout.tsx or src/app/[locale]/page.tsx" codeFormat="typescript"
|
|
964
|
+
import {
|
|
965
|
+
type IConfigLocales,
|
|
966
|
+
getTranslation,
|
|
967
|
+
getMultilingualUrls,
|
|
968
|
+
} from "intlayer";
|
|
969
|
+
import type { Metadata } from "next";
|
|
970
|
+
import type { LocalPromiseParams } from "next-intlayer";
|
|
971
|
+
|
|
972
|
+
export const generateMetadata = async ({
|
|
973
|
+
params,
|
|
974
|
+
}: LocalPromiseParams): Promise<Metadata> => {
|
|
975
|
+
const { locale } = await params;
|
|
976
|
+
const t = <T>(content: IConfigLocales<T>) => getTranslation(content, locale);
|
|
977
|
+
|
|
978
|
+
return {
|
|
979
|
+
title: t<string>({
|
|
980
|
+
en: "My title",
|
|
981
|
+
fr: "Mon titre",
|
|
982
|
+
es: "Mi título",
|
|
983
|
+
}),
|
|
984
|
+
description: t({
|
|
985
|
+
en: "내 설명",
|
|
986
|
+
fr: "Ma description",
|
|
987
|
+
es: "Mi descripción",
|
|
988
|
+
}),
|
|
989
|
+
};
|
|
990
|
+
};
|
|
991
|
+
|
|
992
|
+
// ... 나머지 코드
|
|
993
|
+
```
|
|
994
|
+
|
|
995
|
+
```javascript fileName="src/app/[locale]/layout.mjs or src/app/[locale]/page.mjs" codeFormat="esm"
|
|
996
|
+
import { getTranslation, getMultilingualUrls } from "intlayer";
|
|
997
|
+
|
|
998
|
+
export const generateMetadata = async ({ params }) => {
|
|
999
|
+
const { locale } = await params;
|
|
1000
|
+
const t = (content) => getTranslation(content, locale);
|
|
1001
|
+
|
|
1002
|
+
return {
|
|
1003
|
+
title: t({
|
|
1004
|
+
en: "내 제목",
|
|
1005
|
+
fr: "Mon titre",
|
|
1006
|
+
es: "Mi título",
|
|
1007
|
+
}),
|
|
1008
|
+
description: t({
|
|
1009
|
+
en: "내 설명",
|
|
1010
|
+
fr: "Ma description",
|
|
1011
|
+
es: "Mi descripción",
|
|
1012
|
+
}),
|
|
1013
|
+
};
|
|
1014
|
+
};
|
|
1015
|
+
|
|
1016
|
+
// ... 나머지 코드
|
|
1017
|
+
```
|
|
1018
|
+
|
|
1019
|
+
```javascript fileName="src/app/[locale]/layout.cjs or src/app/[locale]/page.cjs" codeFormat="commonjs"
|
|
1020
|
+
const { getTranslation, getMultilingualUrls } = require("intlayer");
|
|
1021
|
+
|
|
1022
|
+
const generateMetadata = async ({ params }) => {
|
|
1023
|
+
const { locale } = await params;
|
|
1024
|
+
|
|
1025
|
+
const t = (content) => getTranslation(content, locale);
|
|
1026
|
+
|
|
1027
|
+
return {
|
|
1028
|
+
title: t({
|
|
1029
|
+
en: "My title",
|
|
1030
|
+
fr: "Mon titre",
|
|
1031
|
+
es: "Mi título",
|
|
1032
|
+
}),
|
|
1033
|
+
description: t({
|
|
1034
|
+
en: "My description",
|
|
1035
|
+
fr: "Ma description",
|
|
1036
|
+
es: "Mi descripción",
|
|
1037
|
+
}),
|
|
1038
|
+
};
|
|
1039
|
+
};
|
|
1040
|
+
|
|
1041
|
+
module.exports = { generateMetadata };
|
|
1042
|
+
|
|
1043
|
+
// ... 나머지 코드
|
|
1044
|
+
```
|
|
1045
|
+
|
|
1046
|
+
> 메타데이터 최적화에 대해 더 알아보기 [공식 Next.js 문서에서](https://nextjs.org/docs/app/building-your-application/optimizing/metadata).
|
|
1047
|
+
|
|
1048
|
+
### (선택 사항) 9단계: sitemap.xml 및 robots.txt의 다국어화
|
|
1049
|
+
|
|
1050
|
+
`sitemap.xml` 및 `robots.txt`를 다국어화하려면 Intlayer에서 제공하는 `getMultilingualUrls` 함수를 사용할 수 있습니다. 이 함수는 사이트맵에 다국어 URL을 생성할 수 있게 해줍니다.
|
|
1051
|
+
|
|
1052
|
+
```tsx fileName="src/app/sitemap.ts" codeFormat="typescript"
|
|
1053
|
+
import { getMultilingualUrls } from "intlayer";
|
|
1054
|
+
import type { MetadataRoute } from "next";
|
|
1055
|
+
|
|
1056
|
+
const sitemap = (): MetadataRoute.Sitemap => [
|
|
1057
|
+
{
|
|
1058
|
+
url: "https://example.com",
|
|
1059
|
+
alternates: {
|
|
1060
|
+
languages: { ...getMultilingualUrls("https://example.com") },
|
|
1061
|
+
},
|
|
1062
|
+
},
|
|
1063
|
+
{
|
|
1064
|
+
url: "https://example.com/login",
|
|
1065
|
+
alternates: {
|
|
1066
|
+
languages: { ...getMultilingualUrls("https://example.com/login") },
|
|
1067
|
+
},
|
|
1068
|
+
},
|
|
1069
|
+
{
|
|
1070
|
+
url: "https://example.com/register",
|
|
1071
|
+
alternates: {
|
|
1072
|
+
languages: { ...getMultilingualUrls("https://example.com/register") },
|
|
1073
|
+
},
|
|
1074
|
+
},
|
|
1075
|
+
];
|
|
1076
|
+
|
|
1077
|
+
export default sitemap;
|
|
1078
|
+
```
|
|
1079
|
+
|
|
1080
|
+
```jsx fileName="src/app/sitemap.mjx" codeFormat="esm"
|
|
1081
|
+
import { getMultilingualUrls } from "intlayer";
|
|
1082
|
+
|
|
1083
|
+
const sitemap = () => [
|
|
1084
|
+
{
|
|
1085
|
+
url: "https://example.com",
|
|
1086
|
+
alternates: {
|
|
1087
|
+
languages: { ...getMultilingualUrls("https://example.com") },
|
|
1088
|
+
},
|
|
1089
|
+
},
|
|
1090
|
+
{
|
|
1091
|
+
url: "https://example.com/login",
|
|
1092
|
+
alternates: {
|
|
1093
|
+
languages: { ...getMultilingualUrls("https://example.com/login") },
|
|
1094
|
+
},
|
|
1095
|
+
},
|
|
1096
|
+
{
|
|
1097
|
+
url: "https://example.com/register",
|
|
1098
|
+
alternates: {
|
|
1099
|
+
languages: { ...getMultilingualUrls("https://example.com/register") },
|
|
1100
|
+
},
|
|
1101
|
+
},
|
|
1102
|
+
];
|
|
1103
|
+
|
|
1104
|
+
export default sitemap;
|
|
1105
|
+
```
|
|
1106
|
+
|
|
1107
|
+
```jsx fileName="src/app/sitemap.csx" codeFormat="commonjs"
|
|
1108
|
+
const { getMultilingualUrls } = require("intlayer");
|
|
1109
|
+
|
|
1110
|
+
const sitemap = () => [
|
|
1111
|
+
{
|
|
1112
|
+
url: "https://example.com",
|
|
1113
|
+
alternates: {
|
|
1114
|
+
languages: { ...getMultilingualUrls("https://example.com") },
|
|
1115
|
+
},
|
|
1116
|
+
},
|
|
1117
|
+
{
|
|
1118
|
+
url: "https://example.com/login",
|
|
1119
|
+
alternates: {
|
|
1120
|
+
languages: { ...getMultilingualUrls("https://example.com/login") },
|
|
1121
|
+
},
|
|
1122
|
+
},
|
|
1123
|
+
{
|
|
1124
|
+
url: "https://example.com/register",
|
|
1125
|
+
alternates: {
|
|
1126
|
+
languages: { ...getMultilingualUrls("https://example.com/register") },
|
|
1127
|
+
},
|
|
1128
|
+
},
|
|
1129
|
+
];
|
|
1130
|
+
|
|
1131
|
+
module.exports = sitemap;
|
|
1132
|
+
```
|
|
1133
|
+
|
|
1134
|
+
```tsx fileName="src/app/robots.ts" codeFormat="typescript"
|
|
1135
|
+
import type { MetadataRoute } from "next";
|
|
1136
|
+
import { getMultilingualUrls } from "intlayer";
|
|
1137
|
+
|
|
1138
|
+
const getAllMultilingualUrls = (urls: string[]) =>
|
|
1139
|
+
urls.flatMap((url) => Object.values(getMultilingualUrls(url)) as string[]);
|
|
1140
|
+
|
|
1141
|
+
// robots 메타데이터를 반환하는 함수
|
|
1142
|
+
const robots = (): MetadataRoute.Robots => ({
|
|
1143
|
+
rules: {
|
|
1144
|
+
userAgent: "*", // 모든 사용자 에이전트 허용
|
|
1145
|
+
allow: ["/"], // 루트 경로 허용
|
|
1146
|
+
disallow: getAllMultilingualUrls(["/login", "/register"]), // 로그인 및 회원가입 페이지 다국어 URL 차단
|
|
1147
|
+
},
|
|
1148
|
+
host: "https://example.com", // 사이트 호스트
|
|
1149
|
+
sitemap: `https://example.com/sitemap.xml`, // 사이트맵 위치
|
|
1150
|
+
});
|
|
1151
|
+
|
|
1152
|
+
export default robots;
|
|
1153
|
+
```
|
|
1154
|
+
|
|
1155
|
+
```jsx fileName="src/app/robots.mjx" codeFormat="esm"
|
|
1156
|
+
import { getMultilingualUrls } from "intlayer";
|
|
1157
|
+
|
|
1158
|
+
const getAllMultilingualUrls = (urls) =>
|
|
1159
|
+
urls.flatMap((url) => Object.values(getMultilingualUrls(url)));
|
|
1160
|
+
|
|
1161
|
+
const robots = () => ({
|
|
1162
|
+
rules: {
|
|
1163
|
+
userAgent: "*",
|
|
1164
|
+
allow: ["/"],
|
|
1165
|
+
disallow: getAllMultilingualUrls(["/login", "/register"]),
|
|
1166
|
+
},
|
|
1167
|
+
host: "https://example.com",
|
|
1168
|
+
sitemap: `https://example.com/sitemap.xml`,
|
|
1169
|
+
});
|
|
1170
|
+
|
|
1171
|
+
export default robots;
|
|
1172
|
+
```
|
|
1173
|
+
|
|
1174
|
+
```jsx fileName="src/app/robots.csx" codeFormat="commonjs"
|
|
1175
|
+
const { getMultilingualUrls } = require("intlayer");
|
|
1176
|
+
|
|
1177
|
+
const getAllMultilingualUrls = (urls) =>
|
|
1178
|
+
urls.flatMap((url) => Object.values(getMultilingualUrls(url)));
|
|
1179
|
+
|
|
1180
|
+
const robots = () => ({
|
|
1181
|
+
rules: {
|
|
1182
|
+
userAgent: "*",
|
|
1183
|
+
allow: ["/"],
|
|
1184
|
+
disallow: getAllMultilingualUrls(["/login", "/register"]),
|
|
1185
|
+
},
|
|
1186
|
+
host: "https://example.com",
|
|
1187
|
+
sitemap: `https://example.com/sitemap.xml`,
|
|
1188
|
+
});
|
|
1189
|
+
|
|
1190
|
+
module.exports = robots;
|
|
1191
|
+
```
|
|
1192
|
+
|
|
1193
|
+
> 사이트맵 최적화에 대해 더 알아보려면 [공식 Next.js 문서](https://nextjs.org/docs/app/api-reference/file-conventions/metadata/sitemap)를 참고하세요. robots.txt 최적화에 대해서는 [공식 Next.js 문서](https://nextjs.org/docs/app/api-reference/file-conventions/metadata/robots)를 참고하세요.
|
|
1194
|
+
|
|
1195
|
+
### (선택 사항) 10단계: 콘텐츠의 언어 변경하기
|
|
1196
|
+
|
|
1197
|
+
Next.js에서 콘텐츠의 언어를 변경하려면, 권장되는 방법은 `Link` 컴포넌트를 사용하여 사용자를 적절한 현지화된 페이지로 리디렉션하는 것입니다. `Link` 컴포넌트는 페이지의 사전 로딩(prefetching)을 가능하게 하여 전체 페이지 새로고침을 방지하는 데 도움이 됩니다.
|
|
1198
|
+
|
|
1199
|
+
```tsx fileName="src/components/LocaleSwitcher.tsx" codeFormat="typescript"
|
|
1200
|
+
"use client";
|
|
1201
|
+
|
|
1202
|
+
import type { FC } from "react";
|
|
1203
|
+
import {
|
|
1204
|
+
Locales,
|
|
1205
|
+
getHTMLTextDir,
|
|
1206
|
+
getLocaleName,
|
|
1207
|
+
getLocalizedUrl,
|
|
1208
|
+
} from "intlayer";
|
|
1209
|
+
import { useLocale } from "next-intlayer";
|
|
1210
|
+
import Link from "next/link";
|
|
1211
|
+
|
|
1212
|
+
export const LocaleSwitcher: FC = () => {
|
|
1213
|
+
const { locale, pathWithoutLocale, availableLocales, setLocale } =
|
|
1214
|
+
useLocale();
|
|
1215
|
+
|
|
1216
|
+
return (
|
|
1217
|
+
<div>
|
|
1218
|
+
<button popoverTarget="localePopover">{getLocaleName(locale)}</button>
|
|
1219
|
+
<div id="localePopover" popover="auto">
|
|
1220
|
+
{availableLocales.map((localeItem) => (
|
|
1221
|
+
<Link
|
|
1222
|
+
href={getLocalizedUrl(pathWithoutLocale, localeItem)}
|
|
1223
|
+
key={localeItem}
|
|
1224
|
+
aria-current={locale === localeItem ? "page" : undefined}
|
|
1225
|
+
onClick={() => setLocale(localeItem)}
|
|
1226
|
+
replace // "뒤로 가기" 브라우저 버튼이 이전 페이지로 리디렉션되도록 보장합니다.
|
|
1227
|
+
>
|
|
1228
|
+
<span>
|
|
1229
|
+
{/* 로케일 - 예: FR */}
|
|
1230
|
+
{localeItem}
|
|
1231
|
+
</span>
|
|
1232
|
+
<span>
|
|
1233
|
+
{/* 해당 로케일의 언어 - 예: Français */}
|
|
1234
|
+
{getLocaleName(localeItem, locale)}
|
|
1235
|
+
</span>
|
|
1236
|
+
<span dir={getHTMLTextDir(localeItem)} lang={localeItem}>
|
|
1237
|
+
{/* 현재 로케일에서의 언어 - 예: 현재 로케일이 Locales.SPANISH일 때 Francés */}
|
|
1238
|
+
{getLocaleName(localeItem)}
|
|
1239
|
+
</span>
|
|
1240
|
+
<span dir="ltr" lang={Locales.ENGLISH}>
|
|
1241
|
+
{/* 영어로 된 언어명 - 예: French */}
|
|
1242
|
+
{getLocaleName(localeItem, Locales.ENGLISH)}
|
|
1243
|
+
</span>
|
|
1244
|
+
</Link>
|
|
1245
|
+
))}
|
|
1246
|
+
</div>
|
|
1247
|
+
</div>
|
|
1248
|
+
);
|
|
1249
|
+
};
|
|
1250
|
+
```
|
|
1251
|
+
|
|
1252
|
+
```jsx fileName="src/components/LocaleSwitcher.msx" codeFormat="esm"
|
|
1253
|
+
"use client";
|
|
1254
|
+
|
|
1255
|
+
import {
|
|
1256
|
+
Locales,
|
|
1257
|
+
getHTMLTextDir,
|
|
1258
|
+
getLocaleName,
|
|
1259
|
+
getLocalizedUrl,
|
|
1260
|
+
} from "intlayer";
|
|
1261
|
+
import { useLocale } from "next-intlayer";
|
|
1262
|
+
import Link from "next/link";
|
|
1263
|
+
|
|
1264
|
+
export const LocaleSwitcher = () => {
|
|
1265
|
+
const { locale, pathWithoutLocale, availableLocales, setLocale } =
|
|
1266
|
+
useLocale();
|
|
1267
|
+
|
|
1268
|
+
return (
|
|
1269
|
+
<div>
|
|
1270
|
+
<button popoverTarget="localePopover">{getLocaleName(locale)}</button>
|
|
1271
|
+
<div id="localePopover" popover="auto">
|
|
1272
|
+
{availableLocales.map((localeItem) => (
|
|
1273
|
+
<Link
|
|
1274
|
+
href={getLocalizedUrl(pathWithoutLocale, localeItem)}
|
|
1275
|
+
key={localeItem}
|
|
1276
|
+
aria-current={locale === localeItem ? "page" : undefined}
|
|
1277
|
+
onClick={() => setLocale(localeItem)}
|
|
1278
|
+
replace // "뒤로 가기" 브라우저 버튼이 이전 페이지로 리디렉션되도록 보장합니다.
|
|
1279
|
+
>
|
|
1280
|
+
<span>
|
|
1281
|
+
{/* 로케일 - 예: FR */}
|
|
1282
|
+
{localeItem}
|
|
1283
|
+
</span>
|
|
1284
|
+
<span>
|
|
1285
|
+
{/* 해당 로케일의 언어 - 예: Français */}
|
|
1286
|
+
{getLocaleName(localeItem, locale)}
|
|
1287
|
+
</span>
|
|
1288
|
+
<span dir={getHTMLTextDir(localeItem)} lang={localeItem}>
|
|
1289
|
+
{/* 현재 로케일의 언어 - 예: 현재 로케일이 Locales.SPANISH로 설정된 경우 Francés */}
|
|
1290
|
+
{getLocaleName(localeItem)}
|
|
1291
|
+
</span>
|
|
1292
|
+
<span dir="ltr" lang={Locales.ENGLISH}>
|
|
1293
|
+
{/* 영어로 된 언어 - 예: French */}
|
|
1294
|
+
{getLocaleName(localeItem, Locales.ENGLISH)}
|
|
1295
|
+
</span>
|
|
1296
|
+
</Link>
|
|
1297
|
+
))}
|
|
1298
|
+
</div>
|
|
1299
|
+
</div>
|
|
1300
|
+
);
|
|
1301
|
+
};
|
|
1302
|
+
```
|
|
1303
|
+
|
|
1304
|
+
```jsx fileName="src/components/LocaleSwitcher.csx" codeFormat="commonjs"
|
|
1305
|
+
"use client";
|
|
1306
|
+
|
|
1307
|
+
const {
|
|
1308
|
+
Locales,
|
|
1309
|
+
getHTMLTextDir,
|
|
1310
|
+
getLocaleName,
|
|
1311
|
+
getLocalizedUrl,
|
|
1312
|
+
} = require("intlayer");
|
|
1313
|
+
const { useLocale } = require("next-intlayer");
|
|
1314
|
+
const Link = require("next/link");
|
|
1315
|
+
|
|
1316
|
+
export const LocaleSwitcher = () => {
|
|
1317
|
+
const { locale, pathWithoutLocale, availableLocales, setLocale } =
|
|
1318
|
+
useLocale();
|
|
1319
|
+
|
|
1320
|
+
return (
|
|
1321
|
+
<div>
|
|
1322
|
+
<button popoverTarget="localePopover">{getLocaleName(locale)}</button>
|
|
1323
|
+
<div id="localePopover" popover="auto">
|
|
1324
|
+
{availableLocales.map((localeItem) => (
|
|
1325
|
+
<Link
|
|
1326
|
+
href={getLocalizedUrl(pathWithoutLocale, localeItem)}
|
|
1327
|
+
key={localeItem}
|
|
1328
|
+
aria-current={locale === localeItem ? "page" : undefined}
|
|
1329
|
+
onClick={() => setLocale(localeItem)}
|
|
1330
|
+
replace // "뒤로 가기" 브라우저 버튼이 이전 페이지로 리디렉션되도록 보장합니다.
|
|
1331
|
+
>
|
|
1332
|
+
<span>
|
|
1333
|
+
{/* 로케일 - 예: FR */}
|
|
1334
|
+
{localeItem}
|
|
1335
|
+
</span>
|
|
1336
|
+
<span>
|
|
1337
|
+
{/* 해당 로케일 내 언어 - 예: Français */}
|
|
1338
|
+
{getLocaleName(localeItem, locale)}
|
|
1339
|
+
</span>
|
|
1340
|
+
<span dir={getHTMLTextDir(localeItem)} lang={localeItem}>
|
|
1341
|
+
{/* 현재 로케일 내 언어 - 예: 현재 로케일이 Locales.SPANISH일 때 Francés */}
|
|
1342
|
+
{getLocaleName(localeItem)}
|
|
1343
|
+
</span>
|
|
1344
|
+
<span dir="ltr" lang={Locales.ENGLISH}>
|
|
1345
|
+
{/* 영어로 된 언어명 - 예: French */}
|
|
1346
|
+
{getLocaleName(localeItem, Locales.ENGLISH)}
|
|
1347
|
+
</span>
|
|
1348
|
+
</Link>
|
|
1349
|
+
))}
|
|
1350
|
+
</div>
|
|
1351
|
+
</div>
|
|
1352
|
+
);
|
|
1353
|
+
};
|
|
1354
|
+
```
|
|
1355
|
+
|
|
1356
|
+
> 대안으로 `useLocale` 훅에서 제공하는 `setLocale` 함수를 사용할 수 있습니다. 이 함수는 페이지 사전 로딩(prefetching)을 허용하지 않습니다. 자세한 내용은 [`useLocale` 훅 문서](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/packages/next-intlayer/useLocale.md)를 참고하세요.
|
|
1357
|
+
|
|
1358
|
+
> 또한 `onLocaleChange` 옵션에 함수를 설정하여 로케일이 변경될 때 사용자 정의 함수를 실행할 수 있습니다.
|
|
1359
|
+
|
|
1360
|
+
```tsx fileName="src/components/LocaleSwitcher.tsx"
|
|
1361
|
+
"use client";
|
|
1362
|
+
|
|
1363
|
+
import { useRouter } from "next/navigation";
|
|
1364
|
+
import { useLocale } from "next-intlayer";
|
|
1365
|
+
import { getLocalizedUrl } from "intlayer";
|
|
1366
|
+
|
|
1367
|
+
// ... 나머지 코드
|
|
1368
|
+
|
|
1369
|
+
const router = useRouter();
|
|
1370
|
+
const { setLocale } = useLocale({
|
|
1371
|
+
onLocaleChange: (locale) => {
|
|
1372
|
+
router.push(getLocalizedUrl(pathWithoutLocale, locale));
|
|
1373
|
+
},
|
|
1374
|
+
});
|
|
1375
|
+
|
|
1376
|
+
return (
|
|
1377
|
+
<button onClick={() => setLocale(Locales.FRENCH)}>프랑스어로 변경</button>
|
|
1378
|
+
);
|
|
1379
|
+
```
|
|
1380
|
+
|
|
1381
|
+
> 문서 참고:
|
|
1382
|
+
>
|
|
1383
|
+
> - [`useLocale` 훅](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/packages/next-intlayer/useLocale.md)
|
|
1384
|
+
> - [`getLocaleName` 훅](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/packages/intlayer/getLocaleName.md)
|
|
1385
|
+
> - [`getLocalizedUrl` 훅](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/packages/intlayer/getLocalizedUrl.md)
|
|
1386
|
+
> - [`getHTMLTextDir` 훅](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/packages/intlayer/getHTMLTextDir.md)
|
|
1387
|
+
> - [`hrefLang` 속성](https://developers.google.com/search/docs/specialty/international/localized-versions?hl=fr)
|
|
1388
|
+
> - [`lang` 속성](https://developer.mozilla.org/ko/docs/Web/HTML/Global_attributes/lang)
|
|
1389
|
+
> - [`dir` 속성](https://developer.mozilla.org/ko/docs/Web/HTML/Global_attributes/dir)
|
|
1390
|
+
> - [`aria-current` 속성](https://developer.mozilla.org/ko/docs/Web/Accessibility/ARIA/Attributes/aria-current)
|
|
1391
|
+
|
|
1392
|
+
### (선택 사항) 11단계: 지역화된 링크 컴포넌트 생성하기
|
|
1393
|
+
|
|
1394
|
+
애플리케이션의 내비게이션이 현재 로케일을 준수하도록 하려면, 커스텀 `Link` 컴포넌트를 만들 수 있습니다. 이 컴포넌트는 내부 URL에 현재 언어를 자동으로 접두사로 붙여줍니다. 예를 들어, 프랑스어 사용자가 "About" 페이지로 가는 링크를 클릭하면 `/about` 대신 `/fr/about`로 리디렉션됩니다.
|
|
1395
|
+
|
|
1396
|
+
이 동작은 여러 가지 이유로 유용합니다:
|
|
1397
|
+
|
|
1398
|
+
- **SEO 및 사용자 경험**: 로컬라이즈된 URL은 검색 엔진이 언어별 페이지를 올바르게 색인하도록 도와주며, 사용자에게 선호하는 언어로 된 콘텐츠를 제공합니다.
|
|
1399
|
+
- **일관성**: 애플리케이션 전반에 걸쳐 로컬라이즈된 링크를 사용함으로써 내비게이션이 현재 로케일 내에서 유지되어 예상치 못한 언어 전환을 방지합니다.
|
|
1400
|
+
- **유지보수성**: 로컬라이제이션 로직을 단일 컴포넌트에 중앙 집중화하면 URL 관리를 단순화하여 애플리케이션이 성장함에 따라 코드베이스를 더 쉽게 유지보수하고 확장할 수 있습니다.
|
|
1401
|
+
|
|
1402
|
+
아래는 TypeScript로 구현한 로컬라이즈된 `Link` 컴포넌트입니다:
|
|
1403
|
+
|
|
1404
|
+
```tsx fileName="src/components/Link.tsx" codeFormat="typescript"
|
|
1405
|
+
"use client";
|
|
1406
|
+
|
|
1407
|
+
import { getLocalizedUrl } from "intlayer";
|
|
1408
|
+
import NextLink, { type LinkProps as NextLinkProps } from "next/link";
|
|
1409
|
+
import { useLocale } from "next-intlayer";
|
|
1410
|
+
import type { PropsWithChildren, FC } from "react";
|
|
1411
|
+
|
|
1412
|
+
/**
|
|
1413
|
+
* 주어진 URL이 외부 링크인지 확인하는 유틸리티 함수입니다.
|
|
1414
|
+
* URL이 http:// 또는 https://로 시작하면 외부 링크로 간주합니다.
|
|
1415
|
+
*/
|
|
1416
|
+
export const checkIsExternalLink = (href?: string): boolean =>
|
|
1417
|
+
/^https?:\/\//.test(href ?? "");
|
|
1418
|
+
|
|
1419
|
+
/**
|
|
1420
|
+
* 현재 로케일에 따라 href 속성을 조정하는 커스텀 Link 컴포넌트입니다.
|
|
1421
|
+
* 내부 링크의 경우, `getLocalizedUrl`을 사용하여 URL 앞에 로케일을 붙입니다 (예: /fr/about).
|
|
1422
|
+
* 이를 통해 내비게이션이 동일한 로케일 컨텍스트 내에서 이루어지도록 보장합니다.
|
|
1423
|
+
*/
|
|
1424
|
+
export const Link: FC<PropsWithChildren<NextLinkProps>> = ({
|
|
1425
|
+
href,
|
|
1426
|
+
children,
|
|
1427
|
+
...props
|
|
1428
|
+
}) => {
|
|
1429
|
+
const { locale } = useLocale();
|
|
1430
|
+
const isExternalLink = checkIsExternalLink(href.toString());
|
|
1431
|
+
|
|
1432
|
+
// 링크가 내부 링크이고 유효한 href가 제공된 경우, 로케일이 적용된 URL을 가져옵니다.
|
|
1433
|
+
const hrefI18n: NextLinkProps["href"] =
|
|
1434
|
+
href && !isExternalLink ? getLocalizedUrl(href.toString(), locale) : href;
|
|
1435
|
+
|
|
1436
|
+
return (
|
|
1437
|
+
<NextLink href={hrefI18n} {...props}>
|
|
1438
|
+
{children}
|
|
1439
|
+
</NextLink>
|
|
1440
|
+
);
|
|
1441
|
+
};
|
|
1442
|
+
```
|
|
1443
|
+
|
|
1444
|
+
```jsx fileName="src/components/Link.mjx" codeFormat="esm"
|
|
1445
|
+
"use client";
|
|
1446
|
+
|
|
1447
|
+
import { getLocalizedUrl } from "intlayer";
|
|
1448
|
+
import NextLink from "next/link";
|
|
1449
|
+
import { useLocale } from "next-intlayer";
|
|
1450
|
+
|
|
1451
|
+
/**
|
|
1452
|
+
* 주어진 URL이 외부 링크인지 확인하는 유틸리티 함수입니다.
|
|
1453
|
+
* URL이 http:// 또는 https://로 시작하면 외부 링크로 간주합니다.
|
|
1454
|
+
*/
|
|
1455
|
+
export const checkIsExternalLink = (href) => /^https?:\/\//.test(href ?? "");
|
|
1456
|
+
|
|
1457
|
+
/**
|
|
1458
|
+
* 현재 로케일에 따라 href 속성을 조정하는 커스텀 Link 컴포넌트입니다.
|
|
1459
|
+
* 내부 링크의 경우 `getLocalizedUrl`을 사용하여 URL 앞에 로케일 접두사를 붙입니다 (예: /fr/about).
|
|
1460
|
+
* 이를 통해 네비게이션이 동일한 로케일 컨텍스트 내에서 이루어지도록 보장합니다.
|
|
1461
|
+
*/
|
|
1462
|
+
export const Link = ({ href, children, ...props }) => {
|
|
1463
|
+
const { locale } = useLocale();
|
|
1464
|
+
const isExternalLink = checkIsExternalLink(href.toString());
|
|
1465
|
+
|
|
1466
|
+
// 링크가 내부 링크이고 유효한 href가 제공된 경우, 현지화된 URL을 가져옵니다.
|
|
1467
|
+
const hrefI18n =
|
|
1468
|
+
href && !isExternalLink ? getLocalizedUrl(href.toString(), locale) : href;
|
|
1469
|
+
|
|
1470
|
+
return (
|
|
1471
|
+
<NextLink href={hrefI18n} {...props}>
|
|
1472
|
+
{children}
|
|
1473
|
+
</NextLink>
|
|
1474
|
+
);
|
|
1475
|
+
};
|
|
1476
|
+
```
|
|
1477
|
+
|
|
1478
|
+
```jsx fileName="src/components/Link.csx" codeFormat="commonjs"
|
|
1479
|
+
"use client";
|
|
1480
|
+
|
|
1481
|
+
const { getLocalizedUrl } = require("intlayer");
|
|
1482
|
+
const NextLink = require("next/link");
|
|
1483
|
+
const { useLocale } = require("next-intlayer");
|
|
1484
|
+
|
|
1485
|
+
/**
|
|
1486
|
+
* 주어진 URL이 외부 링크인지 확인하는 유틸리티 함수입니다.
|
|
1487
|
+
* URL이 http:// 또는 https://로 시작하면 외부 링크로 간주합니다.
|
|
1488
|
+
*/
|
|
1489
|
+
const checkIsExternalLink = (href) => /^https?:\/\//.test(href ?? "");
|
|
1490
|
+
|
|
1491
|
+
/**
|
|
1492
|
+
* 현재 로케일에 따라 href 속성을 조정하는 커스텀 Link 컴포넌트입니다.
|
|
1493
|
+
* 내부 링크의 경우, `getLocalizedUrl`을 사용하여 URL 앞에 로케일을 붙입니다(예: /fr/about).
|
|
1494
|
+
* 이를 통해 내비게이션이 동일한 로케일 컨텍스트 내에서 이루어지도록 보장합니다.
|
|
1495
|
+
*/
|
|
1496
|
+
const Link = ({ href, children, ...props }) => {
|
|
1497
|
+
const { locale } = useLocale();
|
|
1498
|
+
const isExternalLink = checkIsExternalLink(href.toString());
|
|
1499
|
+
|
|
1500
|
+
// 링크가 내부 링크이고 유효한 href가 제공된 경우, 로케일이 적용된 URL을 가져옵니다.
|
|
1501
|
+
const hrefI18n =
|
|
1502
|
+
href && !isExternalLink ? getLocalizedUrl(href.toString(), locale) : href;
|
|
1503
|
+
|
|
1504
|
+
return (
|
|
1505
|
+
<NextLink href={hrefI18n} {...props}>
|
|
1506
|
+
{children}
|
|
1507
|
+
</NextLink>
|
|
1508
|
+
);
|
|
1509
|
+
};
|
|
1510
|
+
```
|
|
1511
|
+
|
|
1512
|
+
#### 작동 방식
|
|
1513
|
+
|
|
1514
|
+
- **외부 링크 감지**:
|
|
1515
|
+
도움 함수 `checkIsExternalLink`는 URL이 외부 링크인지 여부를 판단합니다. 외부 링크는 현지화가 필요 없으므로 변경되지 않습니다.
|
|
1516
|
+
|
|
1517
|
+
- **현재 로케일 가져오기**:
|
|
1518
|
+
`useLocale` 훅은 현재 로케일(예: 프랑스어의 경우 `fr`)을 제공합니다.
|
|
1519
|
+
|
|
1520
|
+
- **URL 현지화**:
|
|
1521
|
+
내부 링크(즉, 외부 링크가 아닌 경우)에 대해 `getLocalizedUrl`을 사용하여 URL 앞에 현재 로케일을 자동으로 붙입니다. 예를 들어 사용자가 프랑스어 로케일에 있다면, `href`에 `/about`을 전달할 경우 `/fr/about`로 변환됩니다.
|
|
1522
|
+
|
|
1523
|
+
- **링크 반환**:
|
|
1524
|
+
이 컴포넌트는 현지화된 URL을 가진 `<a>` 요소를 반환하여, 내비게이션이 로케일과 일관되도록 보장합니다.
|
|
1525
|
+
|
|
1526
|
+
이 `Link` 컴포넌트를 애플리케이션 전반에 통합하면 일관되고 언어에 민감한 사용자 경험을 유지할 수 있으며, SEO 및 사용성 향상이라는 이점도 누릴 수 있습니다.
|
|
1527
|
+
|
|
1528
|
+
### (선택 사항) 12단계: 서버 액션에서 현재 로케일 가져오기
|
|
1529
|
+
|
|
1530
|
+
서버 액션 내에서 활성 로케일이 필요할 경우(예: 이메일 현지화 또는 로케일 인식 로직 실행), `next-intlayer/server`에서 `getLocale`을 호출하세요:
|
|
1531
|
+
|
|
1532
|
+
```tsx fileName="src/app/actions/getLocale.ts" codeFormat="typescript"
|
|
1533
|
+
"use server";
|
|
1534
|
+
|
|
1535
|
+
import { getLocale } from "next-intlayer/server";
|
|
1536
|
+
|
|
1537
|
+
export const myServerAction = async () => {
|
|
1538
|
+
const locale = await getLocale();
|
|
1539
|
+
|
|
1540
|
+
// 로케일로 무언가를 수행합니다
|
|
1541
|
+
};
|
|
1542
|
+
```
|
|
1543
|
+
|
|
1544
|
+
> `getLocale` 함수는 사용자의 로케일을 결정하기 위해 계단식 전략을 따릅니다:
|
|
1545
|
+
>
|
|
1546
|
+
> 1. 먼저, 프록시에서 설정했을 수 있는 로케일 값을 요청 헤더에서 확인합니다.
|
|
1547
|
+
> 2. 헤더에서 로케일을 찾지 못하면, 쿠키에 저장된 로케일을 확인합니다.
|
|
1548
|
+
> 3. 쿠키도 없으면, 사용자의 브라우저 설정에서 선호하는 언어를 감지하려 시도합니다.
|
|
1549
|
+
> 4. 마지막으로, 애플리케이션에 설정된 기본 로케일로 대체합니다.
|
|
1550
|
+
>
|
|
1551
|
+
> 이를 통해 사용 가능한 컨텍스트에 기반하여 가장 적절한 로케일이 선택됩니다.
|
|
1552
|
+
|
|
1553
|
+
### (선택 사항) 13단계: 번들 크기 최적화
|
|
1554
|
+
|
|
1555
|
+
`next-intlayer`를 사용할 때, 사전(dictionary)은 기본적으로 모든 페이지 번들에 포함됩니다. 번들 크기를 최적화하기 위해 Intlayer는 매크로를 사용하여 `useIntlayer` 호출을 지능적으로 대체하는 선택적 SWC 플러그인을 제공합니다. 이를 통해 사전은 실제로 사용하는 페이지의 번들에만 포함됩니다.
|
|
1556
|
+
|
|
1557
|
+
이 최적화를 활성화하려면 `@intlayer/swc` 패키지를 설치하세요. 설치가 완료되면 `next-intlayer`가 자동으로 플러그인을 감지하고 사용합니다:
|
|
1558
|
+
|
|
1559
|
+
```bash packageManager="npm"
|
|
1560
|
+
npm install @intlayer/swc --save-dev
|
|
1561
|
+
```
|
|
1562
|
+
|
|
1563
|
+
```bash packageManager="pnpm"
|
|
1564
|
+
pnpm add @intlayer/swc --save-dev
|
|
1565
|
+
```
|
|
1566
|
+
|
|
1567
|
+
```bash packageManager="yarn"
|
|
1568
|
+
yarn add @intlayer/swc --save-dev
|
|
1569
|
+
```
|
|
1570
|
+
|
|
1571
|
+
> 참고: 이 최적화는 Next.js 13 이상에서만 사용할 수 있습니다.
|
|
1572
|
+
|
|
1573
|
+
> 참고: 이 패키지는 SWC 플러그인이 Next.js에서 아직 실험 단계이기 때문에 기본적으로 설치되어 있지 않습니다. 향후 변경될 수 있습니다.
|
|
1574
|
+
|
|
1575
|
+
### Turbopack에서 사전 변경 사항 감시하기
|
|
1576
|
+
|
|
1577
|
+
`next dev` 명령어로 Turbopack을 개발 서버로 사용할 때, 사전 변경 사항이 기본적으로 자동 감지되지 않습니다.
|
|
1578
|
+
|
|
1579
|
+
이 제한은 Turbopack이 콘텐츠 파일의 변경 사항을 모니터링하기 위해 webpack 플러그인을 병렬로 실행할 수 없기 때문에 발생합니다. 이를 해결하려면 `intlayer watch` 명령어를 사용하여 개발 서버와 Intlayer 빌드 감시자를 동시에 실행해야 합니다.
|
|
1580
|
+
|
|
1581
|
+
```json5 fileName="package.json"
|
|
1582
|
+
{
|
|
1583
|
+
// ... 기존 package.json 설정
|
|
1584
|
+
"scripts": {
|
|
1585
|
+
// ... 기존 스크립트 설정
|
|
1586
|
+
"dev": "intlayer watch --with 'next dev'",
|
|
1587
|
+
},
|
|
1588
|
+
}
|
|
1589
|
+
```
|
|
1590
|
+
|
|
1591
|
+
> next-intlayer@<=6.x.x 버전을 사용 중이라면, Next.js 16 애플리케이션이 Turbopack과 올바르게 작동하도록 `--turbopack` 플래그를 유지해야 합니다. 이 제한을 피하려면 next-intlayer@>=7.x.x 버전 사용을 권장합니다.
|
|
1592
|
+
|
|
1593
|
+
### TypeScript 구성
|
|
1594
|
+
|
|
1595
|
+
Intlayer는 TypeScript의 이점을 활용하고 코드베이스를 더욱 견고하게 만들기 위해 모듈 증강(module augmentation)을 사용합니다.
|
|
1596
|
+
|
|
1597
|
+

|
|
1598
|
+
|
|
1599
|
+

|
|
1600
|
+
|
|
1601
|
+
TypeScript 구성에 자동 생성된 타입이 포함되어 있는지 확인하세요.
|
|
1602
|
+
|
|
1603
|
+
```json5 fileName="tsconfig.json"
|
|
1604
|
+
{
|
|
1605
|
+
// ... 기존 TypeScript 구성
|
|
1606
|
+
"include": [
|
|
1607
|
+
// ... 기존 TypeScript 구성
|
|
1608
|
+
".intlayer/**/*.ts", // 자동 생성된 타입 포함
|
|
1609
|
+
],
|
|
1610
|
+
}
|
|
1611
|
+
```
|
|
1612
|
+
|
|
1613
|
+
### Git 구성
|
|
1614
|
+
|
|
1615
|
+
Intlayer에서 생성된 파일을 무시하는 것이 권장됩니다. 이렇게 하면 Git 저장소에 커밋하는 것을 방지할 수 있습니다.
|
|
1616
|
+
|
|
1617
|
+
이를 위해 `.gitignore` 파일에 다음 지침을 추가할 수 있습니다:
|
|
1618
|
+
|
|
1619
|
+
```plaintext fileName=".gitignore"
|
|
1620
|
+
# Intlayer에서 생성된 파일 무시
|
|
1621
|
+
.intlayer
|
|
1622
|
+
```
|
|
1623
|
+
|
|
1624
|
+
### VS Code 확장
|
|
1625
|
+
|
|
1626
|
+
Intlayer와 함께 개발 경험을 향상시키기 위해 공식 **Intlayer VS Code 확장**을 설치할 수 있습니다.
|
|
1627
|
+
|
|
1628
|
+
[VS Code 마켓플레이스에서 설치](https://marketplace.visualstudio.com/items?itemName=intlayer.intlayer-vs-code-extension)
|
|
1629
|
+
|
|
1630
|
+
이 확장은 다음을 제공합니다:
|
|
1631
|
+
|
|
1632
|
+
- 번역 키에 대한 **자동 완성**.
|
|
1633
|
+
- 누락된 번역에 대한 **실시간 오류 감지**.
|
|
1634
|
+
- 번역된 콘텐츠의 **인라인 미리보기**.
|
|
1635
|
+
- 번역을 쉽게 생성하고 업데이트할 수 있는 **빠른 작업**.
|
|
1636
|
+
|
|
1637
|
+
확장 기능 사용 방법에 대한 자세한 내용은 [Intlayer VS Code 확장 문서](https://intlayer.org/doc/vs-code-extension)를 참조하세요.
|
|
1638
|
+
|
|
1639
|
+
### 더 나아가기
|
|
1640
|
+
|
|
1641
|
+
더 나아가려면 [비주얼 에디터](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/intlayer_visual_editor.md)를 구현하거나 [CMS](https://github.com/aymericzip/intlayer/blob/main/docs/docs/ko/intlayer_CMS.md)를 사용하여 콘텐츠를 외부화할 수 있습니다.
|