Intlayer의 향후 출시 예정에 대한 알림을 받으세요
    생성:2025-04-18마지막 업데이트:2025-06-29

    Intlayer와 Vite, Preact로 국제화(i18n) 시작하기

    이 패키지는 개발 중입니다. 자세한 내용은 이슈를 참조하세요. 이 이슈에 좋아요를 눌러 Preact용 Intlayer에 대한 관심을 표현해 주세요.

    GitHub에서 애플리케이션 템플릿을 확인하세요.

    Intlayer란 무엇인가요?

    Intlayer는 최신 웹 애플리케이션에서 다국어 지원을 간소화하기 위해 설계된 혁신적인 오픈 소스 국제화(i18n) 라이브러리입니다.

    Intlayer를 사용하면 다음을 할 수 있습니다:

    • 컴포넌트 수준에서 선언적 사전을 사용하여 번역을 쉽게 관리할 수 있습니다.
    • 메타데이터, 라우트 및 콘텐츠를 동적으로 현지화할 수 있습니다.
    • 자동 생성된 타입으로 TypeScript 지원을 보장하여 자동 완성 및 오류 감지를 향상시킵니다.
    • 동적 로케일 감지 및 전환과 같은 고급 기능을 활용할 수 있습니다.

    Vite 및 Preact 애플리케이션에서 Intlayer 설정 단계별 가이드

    1단계: 의존성 설치

    npm을 사용하여 필요한 패키지를 설치하세요:

    bash
    npm install intlayer preact-intlayernpm install vite-intlayer --save-dev
    • intlayer

    • intlayer

      구성 관리, 번역, 콘텐츠 선언, 트랜스파일링 및 CLI 명령어를 위한 국제화 도구를 제공하는 핵심 패키지입니다.

    • preact-intlayer Intlayer를 Preact 애플리케이션과 통합하는 패키지입니다. Preact 국제화를 위한 컨텍스트 제공자와 훅을 제공합니다.

    • vite-intlayer Intlayer를 Vite 번들러와 통합하기 위한 Vite 플러그인과, 사용자의 선호 로케일 감지, 쿠키 관리, URL 리디렉션 처리를 위한 미들웨어를 포함합니다.

    2단계: 프로젝트 구성

    애플리케이션의 언어를 구성하기 위해 설정 파일을 만드세요:

    intlayer.config.ts
    import { Locales, type IntlayerConfig } from "intlayer";const config: IntlayerConfig = {  internationalization: {    locales: [      Locales.ENGLISH,      Locales.FRENCH,      Locales.SPANISH,      // 다른 로케일들    ],    defaultLocale: Locales.ENGLISH,  },};export default config;
    intlayer.config.ts
    import { Locales, type IntlayerConfig } from "intlayer";const config: IntlayerConfig = {  internationalization: {    locales: [      Locales.ENGLISH,      Locales.FRENCH,      Locales.SPANISH,      // 다른 로케일들    ],    defaultLocale: Locales.ENGLISH,  },};export default config;

    이 설정 파일을 통해 로컬라이즈된 URL, 미들웨어 리디렉션, 쿠키 이름, 콘텐츠 선언 위치 및 확장자, 콘솔에서 Intlayer 로그 비활성화 등 다양한 설정을 할 수 있습니다. 사용 가능한 모든 매개변수 목록은 설정 문서를 참고하세요.

    3단계: Vite 구성에 Intlayer 통합하기

    intlayer 플러그인을 구성에 추가하세요.

    vite.config.ts
    import { defineConfig } from "vite";import preact from "@preact/preset-vite";import { intlayerPlugin } from "vite-intlayer";// https://vitejs.dev/config/export default defineConfig({  plugins: [preact(), intlayerPlugin()],});

    이 구성 파일을 통해 지역화된 URL, 미들웨어 리디렉션, 쿠키 이름, 콘텐츠 선언의 위치 및 확장자 설정, 콘솔에서 Intlayer 로그 비활성화 등 다양한 설정을 할 수 있습니다. 사용 가능한 모든 매개변수 목록은 구성 문서를 참조하세요.

    3단계: Vite 구성에 Intlayer 통합하기

    intlayer 플러그인을 구성에 추가하세요.

    vite.config.ts
    import { defineConfig } from "vite";import preact from "@preact/preset-vite";import { intlayerPlugin } from "vite-intlayer";// https://vitejs.dev/config/export default defineConfig({  plugins: [preact(), intlayerPlugin()],});

    intlayerPlugin() Vite 플러그인은 Intlayer를 Vite와 통합하기 위해 사용됩니다. 이 플러그인은 콘텐츠 선언 파일의 빌드를 보장하고 개발 모드에서 이를 모니터링합니다. 또한 Vite 애플리케이션 내에서 Intlayer 환경 변수를 정의하며, 성능 최적화를 위해 별칭(alias)도 제공합니다.

    4단계: 콘텐츠 선언하기

    번역을 저장하기 위해 콘텐츠 선언을 생성하고 관리하세요:

    src/app.content.tsx
    import { t, type Dictionary } from "intlayer";import type { ComponentChildren } from "preact";const appContent = {  key: "app",  content: {    viteLogo: t({      en: "Vite logo",      fr: "Logo Vite",      es: "Logo Vite",    }),    preactLogo: t({      en: "Preact logo",      fr: "Logo Preact",      es: "Logo Preact",    }),    title: "Vite + Preact",    count: t({      en: "count is ",      fr: "le compte est ",      es: "el recuento es ",    }),    edit: t<ComponentChildren>({      en: (        <>          <code>src/app.tsx</code>를 편집하고 저장하여 HMR을 테스트하세요        </>      ),      fr: (        <>          <code>src/app.tsx</code>를 편집하고 저장하여 HMR을 테스트하세요        </>      ),      es: (        <>          <code>src/app.tsx</code>를 편집하고 저장하여 HMR을 테스트하세요        </>      ),    }),    readTheDocs: t({      en: "Click on the Vite and Preact logos to learn more",      fr: "Cliquez sur les logos Vite et Preact pour en savoir plus",      es: "Haga clic en los logotipos de Vite y Preact para obtener más información",    }),  },} satisfies Dictionary;export default appContent;

    콘텐츠 선언은 애플리케이션 내 어디에서든 contentDir 디렉토리(기본값은 ./src)에 포함되기만 하면 정의할 수 있습니다. 그리고 콘텐츠 선언 파일 확장자(기본값은 .content.{json,ts,tsx,js,jsx,mjs,mjx,cjs,cjx})와 일치해야 합니다.

    자세한 내용은 콘텐츠 선언 문서를 참조하세요.

    콘텐츠 파일에 TSX 코드가 포함된 경우, import { h } from "preact";를 임포트하거나 JSX 프래그마가 Preact에 맞게 올바르게 설정되어 있는지 확인해야 할 수 있습니다.

    5단계: 코드에서 Intlayer 사용하기

    애플리케이션 전반에서 콘텐츠 사전을 접근하세요:

    src/app.tsx
    import { useState } from "preact/hooks";import type { FunctionalComponent } from "preact";import preactLogo from "./assets/preact.svg"; // preact.svg 파일이 있다고 가정합니다import viteLogo from "/vite.svg";import "./app.css"; // CSS 파일 이름이 app.css라고 가정합니다import { IntlayerProvider, useIntlayer } from "preact-intlayer";const AppContent: FunctionalComponent = () => {  const [count, setCount] = useState(0);  const content = useIntlayer("app");  return (    <>      <div>        <a href="https://vitejs.dev" target="_blank">          <img src={viteLogo} class="logo" alt={content.viteLogo.value} />        </a>        <a href="https://preactjs.com" target="_blank">          <img            src={preactLogo}            class="logo preact"            alt={content.preactLogo.value}          />        </a>      </div>      <h1>{content.title}</h1>      <div class="card">        <button onClick={() => setCount((count) => count + 1)}>          {content.count}          {count}        </button>        <p>{content.edit}</p>      </div>      <p class="read-the-docs">{content.readTheDocs}</p>    </>  );};const App: FunctionalComponent = () => (  <IntlayerProvider>    <AppContent />  </IntlayerProvider>);export default App;

    만약 alt, title, href, aria-label 등과 같은 string 속성에서 콘텐츠를 사용하려면, 함수의 값을 호출해야 합니다. 예를 들어:

    jsx
    <img src={content.image.src.value} alt={content.image.value} />

    참고: Preact에서는 className 대신 일반적으로 class를 사용합니다.

    useIntlayer 훅에 대해 더 알아보려면 문서를 참고하세요 (preact-intlayer도 API가 유사합니다).

    (선택 사항) 6단계: 콘텐츠의 언어 변경하기

    콘텐츠의 언어를 변경하려면 useLocale 훅에서 제공하는 setLocale 함수를 사용할 수 있습니다. 이 함수는 애플리케이션의 로케일을 설정하고 그에 따라 콘텐츠를 업데이트할 수 있게 해줍니다.

    src/components/LocaleSwitcher.tsx
    import type { FunctionalComponent } from "preact";import { Locales } from "intlayer";import { useLocale } from "preact-intlayer";const LocaleSwitcher: FunctionalComponent = () => {  const { setLocale } = useLocale();  return (    <button onClick={() => setLocale(Locales.ENGLISH)}>영어로 언어 변경</button>  );};export default LocaleSwitcher;

    useLocale 훅에 대해 더 알아보려면 문서를 참조하세요 (API는 preact-intlayer와 유사합니다).

    (선택 사항) 7단계: 애플리케이션에 지역화된 라우팅 추가하기

    이 단계의 목적은 각 언어별로 고유한 경로를 만드는 것입니다. 이는 SEO 및 SEO 친화적인 URL에 유용합니다. 예시:

    plaintext
    - https://example.com/about- https://example.com/es/about- https://example.com/fr/about

    기본적으로 경로는 기본 로케일에 대해 접두사가 붙지 않습니다. 기본 로케일에 접두사를 붙이고 싶다면, 구성에서 middleware.prefixDefault 옵션을 true로 설정할 수 있습니다. 자세한 내용은 구성 문서를 참조하세요.

    애플리케이션에 지역화된 라우팅을 추가하려면, 애플리케이션의 라우트를 감싸고 로케일 기반 라우팅을 처리하는 LocaleRouter 컴포넌트를 생성할 수 있습니다. 다음은 preact-iso를 사용한 예제입니다:

    먼저, preact-iso를 설치하세요:

    bash
    npm install preact-iso
    src/components/LocaleRouter.tsx
    import { type Locales, configuration, getPathWithoutLocale } from "intlayer";import { ComponentChildren, FunctionalComponent } from "preact";import { IntlayerProvider } from "preact-intlayer";import { LocationProvider, useLocation } from "preact-iso";import { useEffect } from "preact/hooks";const { internationalization, middleware } = configuration;const { locales, defaultLocale } = internationalization;const Navigate: FunctionalComponent<{ to: string; replace?: boolean }> = ({  to,  replace,}) => {  const { route } = useLocation();  useEffect(() => {    route(to, replace);  }, [to, replace, route]);  return null;};/**/** * 지역화 처리를 담당하며 자식 컴포넌트를 적절한 로케일 컨텍스트로 감싸는 컴포넌트입니다. * URL 기반의 로케일 감지 및 검증을 관리합니다. */const AppLocalized: FunctionalComponent<{  children: ComponentChildren;  locale?: Locales;}> = ({ children, locale }) => {  const { path: pathname, url } = useLocation();  if (!url) {    return null;  }  const search = url.substring(pathname.length);  // 현재 로케일을 결정하며, 제공되지 않은 경우 기본 로케일로 대체합니다.  const currentLocale = locale ?? defaultLocale;  // 경로에서 로케일 접두사를 제거하여 기본 경로를 구성합니다.  const pathWithoutLocale = getPathWithoutLocale(    pathname // 현재 URL 경로  );  /**   * middleware.prefixDefault가 true인 경우, 기본 로케일은 항상 접두사로 붙어야 합니다.   */  if (middleware.prefixDefault) {    // 로케일을 검증합니다.    if (!locale || !locales.includes(locale)) {      // 업데이트된 경로와 함께 기본 로케일로 리디렉션합니다.      return (        <Navigate          to={`/${defaultLocale}/${pathWithoutLocale}${search}`}          replace // 현재 히스토리 항목을 새 항목으로 교체합니다.        />      );    }    // IntlayerProvider로 자식을 감싸고 현재 로케일을 설정합니다.    return (      <IntlayerProvider locale={currentLocale}>{children}</IntlayerProvider>    );  } else {    /**     * middleware.prefixDefault가 false인 경우, 기본 로케일은 접두사로 붙지 않습니다.     * 현재 로케일이 유효하며 기본 로케일이 아님을 확인합니다.     */    if (      currentLocale.toString() !== defaultLocale.toString() &&      !locales        .filter(          (loc) => loc.toString() !== defaultLocale.toString() // 기본 로케일 제외        )        .includes(currentLocale) // 현재 로케일이 유효한 로케일 목록에 포함되어 있는지 확인    ) {      // 로케일 접두사가 없는 경로로 리디렉션      return <Navigate to={`${pathWithoutLocale}${search}`} replace />;    }    // IntlayerProvider로 children을 감싸고 현재 로케일을 설정    return (      <IntlayerProvider locale={currentLocale}>{children}</IntlayerProvider>    );  }};const RouterContent: FunctionalComponent<{  children: ComponentChildren;}> = ({ children }) => {  const { path } = useLocation();  if (!path) {    return null;  }  const pathLocale = path.split("/")[1] as Locales;  const isLocaleRoute = locales    .filter((locale) => middleware.prefixDefault || locale !== defaultLocale)    .some((locale) => locale.toString() === pathLocale);  if (isLocaleRoute) {    return <AppLocalized locale={pathLocale}>{children}</AppLocalized>;  }  return (    <AppLocalized      locale={!middleware.prefixDefault ? defaultLocale : undefined}    >      {children}    </AppLocalized>  );};/** * 로케일별 경로를 설정하는 라우터 컴포넌트입니다. * preact-iso를 사용하여 네비게이션을 관리하고 로컬라이즈된 컴포넌트를 렌더링합니다. */export const LocaleRouter: FunctionalComponent<{  children: ComponentChildren;}> = ({ children }) => (  <LocationProvider>    <RouterContent>{children}</RouterContent>  </LocationProvider>);

    그런 다음, 애플리케이션에서 LocaleRouter 컴포넌트를 사용할 수 있습니다:

    src/app.tsx
    import { LocaleRouter } from "./components/LocaleRouter";import type { FunctionalComponent } from "preact";tsx;// ... 5단계에서 정의한 AppContent 컴포넌트const App: FunctionalComponent = () => (  <LocaleRouter>    <AppContent />  </LocaleRouter>);export default App;

    병행하여, intLayerMiddlewarePlugin을 사용하여 애플리케이션에 서버 사이드 라우팅을 추가할 수도 있습니다. 이 플러그인은 URL을 기반으로 현재 로케일을 자동으로 감지하고 적절한 로케일 쿠키를 설정합니다. 로케일이 지정되지 않은 경우, 플러그인은 사용자의 브라우저 언어 설정을 기준으로 가장 적합한 로케일을 결정합니다. 로케일이 감지되지 않으면 기본 로케일로 리디렉션합니다.

    vite.config.ts
    import { defineConfig } from "vite";import preact from "@preact/preset-vite";import { intlayerPlugin, intLayerMiddlewarePlugin } from "vite-intlayer";// https://vitejs.dev/config/export default defineConfig({  plugins: [preact(), intlayerPlugin(), intLayerMiddlewarePlugin()],});

    (선택 사항) 8단계: 로케일이 변경될 때 URL 변경하기

    로케일이 변경될 때 URL을 변경하려면, useLocale 훅에서 제공하는 onLocaleChange 속성을 사용할 수 있습니다. 동시에, preact-isouseLocationroute를 사용하여 URL 경로를 업데이트할 수 있습니다.

    src/components/LocaleSwitcher.tsx
    import { useLocation, route } from "preact-iso";import {  Locales,  getHTMLTextDir,  getLocaleName,  getLocalizedUrl,} from "intlayer";import { useLocale } from "preact-intlayer";import type { FunctionalComponent } from "preact";const LocaleSwitcher: FunctionalComponent = () => {  const location = useLocation();  const { locale, availableLocales, setLocale } = useLocale({    onLocaleChange: (newLocale) => {      const currentFullPath = location.url; // preact-iso가 전체 URL을 제공합니다      // 업데이트된 로케일로 URL 구성      // 예시: /es/about?foo=bar      const pathWithLocale = getLocalizedUrl(currentFullPath, newLocale);      // URL 경로 업데이트      route(pathWithLocale, true); // true는 교체를 의미    },  });  return (    <div>      <button popovertarget="localePopover">{getLocaleName(locale)}</button>      <div id="localePopover" popover="auto">        {availableLocales.map((localeItem) => (          <a            href={getLocalizedUrl(location.url, localeItem)}            hreflang={localeItem}            aria-current={locale === localeItem ? "page" : undefined}            onClick={(e) => {              e.preventDefault();              setLocale(localeItem);              // 로케일 설정 후 프로그래밍 방식의 내비게이션은 onLocaleChange에서 처리됩니다.            }}            key={localeItem}          >            <span>              {/* 로케일 - 예: FR */}              {localeItem}            </span>            <span>              {/* 해당 로케일 내 언어 - 예: Français */}              {getLocaleName(localeItem, localeItem)}            </span>            <span dir={getHTMLTextDir(localeItem)} lang={localeItem}>              {/* 현재 로케일 내 언어 - 예: 현재 로케일이 Locales.SPANISH일 때 Francés */}              {getLocaleName(localeItem, locale)}            </span>            <span dir="ltr" lang={Locales.ENGLISH}>              {/* 영어로 된 언어명 - 예: French */}              {getLocaleName(localeItem, Locales.ENGLISH)}            </span>          </a>        ))}      </div>    </div>  );};export default LocaleSwitcher;

    문서 참조:

    아래는 추가 설명과 개선된 코드 예제가 포함된 업데이트된 9단계입니다:


    (선택 사항) 9단계: HTML 언어 및 방향 속성 변경하기

    애플리케이션이 다국어를 지원할 때, <html> 태그의 langdir 속성을 현재 로케일에 맞게 업데이트하는 것이 매우 중요합니다. 이렇게 하면 다음을 보장할 수 있습니다:

    • 접근성: 화면 낭독기 및 보조 기술은 올바른 lang 속성에 의존하여 콘텐츠를 정확하게 발음하고 해석합니다.
    • 텍스트 렌더링: dir (방향) 속성은 텍스트가 올바른 순서로 렌더링되도록 보장합니다(예: 영어는 왼쪽에서 오른쪽, 아랍어나 히브리어는 오른쪽에서 왼쪽). 이는 가독성에 필수적입니다.
    • SEO: 검색 엔진은 lang 속성을 사용하여 페이지의 언어를 판단하고, 검색 결과에서 적절한 현지화된 콘텐츠를 제공하는 데 도움을 줍니다.

    로케일이 변경될 때 이러한 속성을 동적으로 업데이트하면, 지원하는 모든 언어에서 일관되고 접근 가능한 사용자 경험을 보장할 수 있습니다.

    훅 구현하기

    HTML 속성을 관리하는 커스텀 훅을 만듭니다. 이 훅은 로케일 변경을 감지하여 속성을 적절히 업데이트합니다:

    src/hooks/useI18nHTMLAttributes.tsx
    import { useEffect } from "preact/hooks";import { useLocale } from "preact-intlayer";import { getHTMLTextDir } from "intlayer";/** * 현재 로케일에 따라 HTML <html> 요소의 `lang` 및 `dir` 속성을 업데이트합니다. * - `lang`: 브라우저와 검색 엔진에 페이지의 언어를 알립니다. * - `dir`: 올바른 읽기 순서(예: 영어는 'ltr', 아랍어는 'rtl')를 보장합니다. * * 이 동적 업데이트는 올바른 텍스트 렌더링, 접근성 및 SEO에 필수적입니다. */export const useI18nHTMLAttributes = () => {  const { locale } = useLocale();  useEffect(() => {    // 현재 로케일로 언어 속성을 업데이트합니다.    document.documentElement.lang = locale;    // 현재 로케일에 따라 텍스트 방향을 설정합니다.    document.documentElement.dir = getHTMLTextDir(locale);  }, [locale]);};

    애플리케이션에서 훅 사용하기

    로케일이 변경될 때마다 HTML 속성이 업데이트되도록 훅을 메인 컴포넌트에 통합하세요:

    src/app.tsx
    import type { FunctionalComponent } from "preact";import { IntlayerProvider } from "preact-intlayer"; // AppContent가 필요하면 useIntlayer는 이미 임포트됨import { useI18nHTMLAttributes } from "./hooks/useI18nHTMLAttributes";import "./app.css";// 5단계에서 정의한 AppContentconst AppWithHooks: FunctionalComponent = () => {  // 로케일에 따라 <html> 태그의 lang 및 dir 속성을 업데이트하기 위해 훅을 적용합니다.  useI18nHTMLAttributes();  // AppContent가 5단계에서의 주요 콘텐츠 표시 컴포넌트라고 가정합니다.  return <AppContent />;};const App: FunctionalComponent = () => (  <IntlayerProvider>    <AppWithHooks />  </IntlayerProvider>);export default App;

    이 변경 사항을 적용하면 애플리케이션은 다음을 수행합니다:

    • 언어(lang) 속성이 현재 로케일을 정확하게 반영하도록 하여 SEO 및 브라우저 동작에 중요합니다.
    • 로케일에 따라 텍스트 방향(dir)을 조정하여, 읽기 순서가 다른 언어에 대해 가독성과 사용성을 향상시킵니다.
    • 보조 기술이 최적으로 작동하기 위해 이러한 속성에 의존하므로, 보다 접근성 높은 경험을 제공합니다.

    (선택 사항) 10단계: 지역화된 링크 컴포넌트 생성

    애플리케이션의 내비게이션이 현재 로케일을 준수하도록 하려면, 커스텀 Link 컴포넌트를 만들 수 있습니다. 이 컴포넌트는 내부 URL에 자동으로 현재 언어 접두사를 붙입니다.

    이 동작은 여러 가지 이유로 유용합니다:

    • SEO 및 사용자 경험: 지역화된 URL은 검색 엔진이 언어별 페이지를 올바르게 색인화하도록 돕고, 사용자가 선호하는 언어로 된 콘텐츠를 제공합니다.
    • 일관성: 애플리케이션 전반에 걸쳐 지역화된 링크를 사용함으로써 내비게이션이 현재 로케일 내에서 유지되어 예상치 못한 언어 전환을 방지합니다.
    • 유지보수성: 로컬라이제이션 로직을 단일 컴포넌트에 집중시켜 URL 관리를 단순화합니다.

    preact-iso를 사용하는 Preact의 경우, 일반적으로 내비게이션에 표준 <a> 태그를 사용하며, preact-iso가 라우팅을 처리합니다. 클릭 시 프로그래밍 방식의 내비게이션이 필요하다면(예: 내비게이션 전에 작업 수행), useLocation에서 제공하는 route 함수를 사용할 수 있습니다. 다음은 URL을 로컬라이즈하는 커스텀 앵커 컴포넌트를 만드는 방법입니다:

    src/components/LocalizedLink.tsx
    import { getLocalizedUrl } from "intlayer";import { useLocale, useLocation, route } from "preact-intlayer"; // useLocation과 route가 preact-intlayer를 통해 preact-iso에서 재수출되는 경우 또는 직접 import하는 경우// 만약 재수출되지 않았다면, 직접 import 하세요: import { useLocation, route } from "preact-iso";import type { JSX } from "preact"; // HTMLAttributes 용import { forwardRef } from "preact/compat"; // ref 전달용export interface LocalizedLinkProps  extends JSX.HTMLAttributes<HTMLAnchorElement> {  href: string;  replace?: boolean; // 선택 사항: 히스토리 상태를 교체할지 여부}/** * 주어진 URL이 외부 링크인지 확인하는 유틸리티 함수. * URL이 http:// 또는 https:// 로 시작하면 외부 링크로 간주합니다. */export const checkIsExternalLink = (href?: string): boolean =>  /^https?:\/\//.test(href ?? "");/** * 현재 로케일에 따라 href 속성을 조정하는 커스텀 Link 컴포넌트. * 내부 링크의 경우, `getLocalizedUrl`을 사용하여 URL 앞에 로케일을 붙입니다 (예: /fr/about). * 이를 통해 내비게이션이 동일한 로케일 컨텍스트 내에서 유지되도록 합니다. * 표준 <a> 태그를 사용하지만 preact-iso의 `route`를 사용하여 클라이언트 측 내비게이션을 트리거할 수 있습니다. */export const LocalizedLink = forwardRef<HTMLAnchorElement, LocalizedLinkProps>(  ({ href, children, onClick, replace = false, ...props }, ref) => {    const { locale } = useLocale();    const location = useLocation(); // preact-iso에서 가져옴    const isExternalLink = checkIsExternalLink(href);    const hrefI18n =      href && !isExternalLink ? getLocalizedUrl(href, locale) : href;    const handleClick = (event: JSX.TargetedMouseEvent<HTMLAnchorElement>) => {      if (onClick) {        onClick(event);      }      if (        !isExternalLink &&        href && // href가 정의되어 있는지 확인        event.button === 0 && // 왼쪽 클릭        !event.metaKey &&        !event.ctrlKey &&        !event.shiftKey &&        !event.altKey && // 표준 수정키 확인        !props.target // 새 탭/창을 대상으로 하지 않음      ) {        event.preventDefault();        if (location.url !== hrefI18n) {          // URL이 다를 때만 네비게이션 수행          route(hrefI18n, replace); // preact-iso의 route 사용        }      }    };    return (      <a href={hrefI18n} ref={ref} onClick={handleClick} {...props}>        {children}      </a>    );  });

    작동 방식

    • 외부 링크 감지:
      헬퍼 함수 checkIsExternalLink는 URL이 외부 링크인지 여부를 판단합니다. 외부 링크는 변경되지 않고 그대로 둡니다.
    • 현재 로케일 가져오기:
      useLocale 훅은 현재 로케일을 제공합니다.
    • URL 현지화:
      내부 링크의 경우, getLocalizedUrl이 URL 앞에 현재 로케일을 붙입니다.
    • 클라이언트 사이드 내비게이션: handleClick 함수는 내부 링크인지 그리고 표준 내비게이션을 방지해야 하는지 확인합니다. 만약 그렇다면, preact-isoroute 함수(useLocation을 통해 얻거나 직접 임포트한)를 사용하여 클라이언트 사이드 내비게이션을 수행합니다. 이는 전체 페이지 리로드 없이 SPA와 같은 동작을 제공합니다.
    • 링크 반환:
      이 컴포넌트는 지역화된 URL과 커스텀 클릭 핸들러가 적용된 <a> 요소를 반환합니다.

    TypeScript 구성

    Intlayer는 모듈 확장을 사용하여 TypeScript의 이점을 활용하고 코드베이스를 더욱 견고하게 만듭니다.

    alt text

    alt text

    TypeScript 구성에 자동 생성된 타입이 포함되어 있는지 확인하세요.

    tsconfig.json
    {  // ... 기존 TypeScript 구성  "compilerOptions": {    // ...    "jsx": "react-jsx",    "jsxImportSource": "preact", // Preact 10+에 권장됨    // ...  },  "include": [    // ... 기존 TypeScript 구성    ".intlayer/**/*.ts", // 자동 생성된 타입 포함  ],}

    tsconfig.json이 Preact에 맞게 설정되어 있는지 확인하세요. 특히 jsxjsxImportSource 또는 구버전 Preact의 경우 jsxFactory/jsxFragmentFactorypreset-vite 기본값을 사용하지 않는다면 설정되어 있어야 합니다.

    Git 구성

    Intlayer가 생성한 파일들은 Git 저장소에 커밋하지 않도록 무시하는 것이 권장됩니다. 이렇게 하면 불필요한 파일 커밋을 방지할 수 있습니다.

    이를 위해 .gitignore 파일에 다음 지침을 추가할 수 있습니다:

    plaintext
    # Intlayer에서 생성된 파일 무시.intlayer

    VS 코드 확장 프로그램

    Intlayer와 함께 개발 경험을 향상시키기 위해 공식 Intlayer VS 코드 확장 프로그램을 설치할 수 있습니다.

    VS 코드 마켓플레이스에서 설치하기

    이 확장 프로그램은 다음을 제공합니다:

    • 번역 키에 대한 자동 완성 기능.
    • 누락된 번역에 대한 실시간 오류 감지.
    • 번역된 콘텐츠의 인라인 미리보기.
    • 번역을 쉽게 생성하고 업데이트할 수 있는 빠른 작업.

    확장 프로그램 사용 방법에 대한 자세한 내용은 Intlayer VS 코드 확장 프로그램 문서를 참조하세요.


    더 나아가기

    더 나아가려면, 비주얼 에디터를 구현하거나 CMS를 사용하여 콘텐츠를 외부화할 수 있습니다.


    문서 이력

    • 5.5.10 - 2025-06-29: 초기 이력
    Intlayer의 향후 출시 예정에 대한 알림을 받으세요