expo-localization
, react-i18next
, i18next
, @react-native-async-storage/async-storage
라이브러리를 설치합니다.npx expo install expo-localization react-i18next i18next @react-native-async-storage/async-storage
expo-localization
을 import 하여 기기의 현재 언어 설정을 가져오는 코드를 작성할 수 있습니다.import React from "react";
import { View, Text } from "react-native";
import { getLocales } from "expo-localization";
const App = () => {
const deviceLanguage = getLocales()[0].languageCode;
return (
<View
style={{
flex: 1,
justifyContent: "center",
alignItems: "center",
}}
>
<Text style={{ fontSize: 20 }}>현재 기기 언어: {deviceLanguage}</Text>
</View>
);
};
export default App;
getLocales()
는 기기의 import React from "react";
import { View, Text, Button } from "react-native";
import { initReactI18next, useTranslation } from "react-i18next";
import { getLocales } from "expo-localization";
import i18n from "i18next";
const resources = {
en: {
translation: {
welcome: "Welcome to the app!",
},
},
ko: {
translation: {
welcome: "앱에 오신 것을 환영합니다!",
},
},
ar: {
translation: {
welcome: "مرحبًا بك في التطبيق!",
},
},
};
i18n.use(initReactI18next).init({
resources,
lng: getLocales()[0].languageCode || "ko",
fallbackLng: {
"en-*": ["en"],
"ko-*": ["ko"],
"ar-*": ["ar"],
default: ["en"],
},
interpolation: {
escapeValue: false,
},
react: {
useSuspense: false,
},
});
const App = () => {
const { t, i18n } = useTranslation();
return (
<View
style={{
flex: 1,
justifyContent: "center",
alignItems: "center",
gap: 20,
}}
>
<Text>{t("welcome")}</Text>
<Button title="English" onPress={() => i18n.changeLanguage("en")} />
<Button title="Korean" onPress={() => i18n.changeLanguage("ko")} />
<Button title="Arabic" onPress={() => i18n.changeLanguage("ar")} />
</View>
);
};
export default App;
// 컴포넌트 밖에 번역 데이터 정의
const resources = {
en: {
translation: {
welcome: "Welcome to the app!",
},
},
ko: {
translation: {
welcome: "앱에 오신 것을 환영합니다!",
},
},
ar: {
translation: {
welcome: "مرحبًا بك في التطبيق!",
},
},
};
// ...
getLocales()[0].languageCode
를 사용해 사용자의 기기 언어를 기본 언어로 설정합니다.
fallbackLng을 통해 언어가 지정되지 않은 경우 대페할 기본 언어를 설정할 수 있습니다.import i18n from "i18next";
import { initReactI18next } from "react-i18next";
i18n.use(initReactI18next).init({
resources,
lng: getLocales()[0].languageCode || "ko",
fallbackLng: {
"en-*": ["en"],
"ko-*": ["ko"],
"ar-*": ["ar"],
default: ["en"],
},
interpolation: {
escapeValue: false,
},
react: {
useSuspense: false,
},
});
useTranslation
훅을 사용하여 번역을 사용할 수 있습니다. `import React from "react";
import { View, Text, Button } from "react-native";
import { initReactI18next, useTranslation } from "react-i18next";
import { getLocales } from "expo-localization";
import i18n from "i18next";
//... i18n 초기화 코드
const App = () => {
const { t, i18n } = useTranslation();
return (
<View
style={{
flex: 1,
justifyContent: "center",
alignItems: "center",
gap: 20,
}}
>
<Text>{t("welcome")}</Text>
<Button title="English" onPress={() => i18n.changeLanguage("en")} />
<Button title="Korean" onPress={() => i18n.changeLanguage("ko")} />
<Button title="Arabic" onPress={() => i18n.changeLanguage("ar")} />
</View>
);
};
export default App;
mkdir -p src/i18n/locales/{en-US,ko-KR,ar-SA} && \
touch src/i18n/index.ts \
src/i18n/locales/en-US/translations.json \
src/i18n/locales/ko-KR/translations.json \
src/i18n/locales/ar-SA/translations.json
📦 src
┣ 📂 i18n
┃ ┣ 📂 locales
┃ ┃ ┣ 📂 ar-SA
┃ ┃ ┃ ┗ 📜 translations.json
┃ ┃ ┣ 📂 en-US
┃ ┃ ┃ ┗ 📜 translations.json
┃ ┃ ┗ 📂 ko-KR
┃ ┃ ┃ ┗ 📜 translations.json
┗ ┗ 📜 index.ts
{
"welcome": "Welcome to the app!"
}
{
"welcome": "앱에 오신 것을 환영합니다!"
}
{
"welcome": "مرحبًا بك في التطبيق!"
}
src/i18n/index.ts
파일에 번역 데이터를 불러오는 코드를 작성합니다.import i18n from "i18next";
import { initReactI18next } from "react-i18next";
import { getLocales } from "expo-localization";
import translationEn from "./locales/en-US/translations.json";
import translationKo from "./locales/ko-KR/translations.json";
import translationAr from "./locales/ar-SA/translations.json";
const resources = {
en: { translation: translationEn },
ko: { translation: translationKo },
ar: { translation: translationAr },
};
const initI18n = async () => {
i18n.use(initReactI18next).init({
resources,
lng: getLocales()[0].languageCode || "ko",
fallbackLng: {
"en-*": ["en"],
"ko-*": ["ko"],
"ar-*": ["ar"],
default: ["en"],
},
interpolation: {
escapeValue: false,
},
react: {
useSuspense: false,
},
});
};
initI18n();
export default i18n;
src/i18n/index.ts
파일을 import하여 사용합니다. 이제 번역 데이터를 관리하기 편리해졌습니다. /app/_layout.tsx
파일에 적용해서 프로젝트 전체에서 반영되도록 할 수 있습니다.import "@/src/i18n";
// ... App 컴포넌트 밖의 i18n 초기화 코드 제거
AsyncStorage
를 사용하여 언어 설정을 저장하면 됩니다. AsyncStorage
는 앱의 데이터를 비동기적으로 저장하고 불러올 수 있는 기능을 제공합니다.
우리는 /src/i18n/index.ts
파일에서 AsyncStorage
를 사용하여 언어 설정을 저장하고 불러오는 코드를 작성합니다.ko-KR
, 영어는 en-US
형식으로 저장합니다. 하지만 현재 데이터는 ko
, en
, ar
형식으로 저장되어 있으므로, 기기에서 언어 설정을 불러온 후 언어 코드만 추출하여 저장합니다. 오류를 대비해 기본 언어는 en
으로 설정합니다.import i18n from "i18next";
import { initReactI18next } from "react-i18next";
import * as Localization from "expo-localization";
import AsyncStorage from "@react-native-async-storage/async-storage";
import translationEn from "./locales/en-US/translations.json";
import translationKo from "./locales/ko-KR/translations.json";
import translationAr from "./locales/ar-SA/translations.json";
const resources = {
en: { translation: translationEn },
ko: { translation: translationKo },
ar: { translation: translationAr },
};
const LANGUAGE_KEY = "@app_language";
const initI18n = async () => {
try {
const savedLanguage = await AsyncStorage.getItem(LANGUAGE_KEY);
let selectedLanguage = savedLanguage;
if (!selectedLanguage) {
const deviceLocales = Localization.getLocales();
const deviceLocale = deviceLocales[0]?.languageTag || "en-US";
const languageCode = deviceLocale.split("-")[0];
if (languageCode in resources) {
selectedLanguage = languageCode;
} else {
selectedLanguage = "en";
}
}
await i18n.use(initReactI18next).init({
resources,
lng: selectedLanguage,
fallbackLng: {
"en-*": ["en"],
"ko-*": ["ko"],
"ar-*": ["ar"],
default: ["en"],
},
interpolation: {
escapeValue: false,
},
react: {
useSuspense: false,
},
});
if (!savedLanguage) {
await AsyncStorage.setItem(LANGUAGE_KEY, selectedLanguage);
}
} catch (error) {
await i18n.use(initReactI18next).init({
resources,
lng: "en",
fallbackLng: "en",
interpolation: {
escapeValue: false,
},
react: {
useSuspense: false,
},
});
}
};
initI18n();
export default i18n;
/app/index.tsx
파일에서 언어 변경 버튼을 누를 때마다 AsyncStorage에 언어 설정을 저장합니다.import React from "react";
import "@/src/i18n";
import { View, Text, Button } from "react-native";
import { useTranslation } from "react-i18next";
import AsyncStorage from "@react-native-async-storage/async-storage";
const App = () => {
const { t, i18n } = useTranslation();
const changeLanguage = async (language: "en" | "ko" | "ar") => {
i18n.changeLanguage(language);
await AsyncStorage.setItem("@app_language", language);
};
return (
<View
style={{
flex: 1,
justifyContent: "center",
alignItems: "center",
gap: 20,
}}
>
<Text>{t("welcome")}</Text>
<Button title="English" onPress={() => changeLanguage("en")} />
<Button title="Korean" onPress={() => changeLanguage("ko")} />
<Button title="Arabic" onPress={() => changeLanguage("ar")} />
</View>
);
};
export default App;
cd ios && pod install && cd ..
npx expo run:ios
expo-localization
라이브러리를 사용하여 기기의 언어 설정을 가져오고, i18next
라이브러리를 사용하여 다국어 번역을 구현하는 방법을 알아보았습니다.
또한 번역 데이터를 별도 파일로 분리하여 관리하고, AsyncStorage를 사용하여 언어 설정을 저장하는 방법을 알아보았습니다. 다국어 언어 팩을 지원하는 것은 앱을 글로벌하게 서비스하고자 하는 경우 필수적입니다. 앱을 사용하는 사용자들이 편리하게 앱을 사용할 수 있도록 다국어 언어 팩을 지원하는 것을 고려해보세요.