1 /*
2 * Copyright (c) 2021-2022 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15 #ifdef SUPPORT_APP_PREFERRED_LANGUAGE
16 #include <regex>
17 #include "application_context.h"
18 #include "bundle_info.h"
19 #include "bundle_mgr_interface.h"
20 #include "iservice_registry.h"
21 #include "system_ability_definition.h"
22 #include "preferences_helper.h"
23 #endif
24 #include "i18n_hilog.h"
25 #include "locale_config.h"
26 #include "locale_info.h"
27 #include "locale_matcher.h"
28 #include "parameter.h"
29 #include "preferred_language.h"
30 #include "utils.h"
31 #include "vector"
32
33 namespace OHOS {
34 namespace Global {
35 namespace I18n {
36 const char *PreferredLanguage::RESOURCE_PATH_HEAD = "/data/accounts/account_0/applications/";
37 const char *PreferredLanguage::RESOURCE_PATH_TAILOR = "/assets/entry/resources.index";
38 const char *PreferredLanguage::RESOURCE_PATH_SPLITOR = "/";
39 const char *PreferredLanguage::PREFERRED_LANGUAGES = "persist.global.preferredLanguages";
40 const char *PreferredLanguage::APP_LANGUAGE_KEY = "app_language";
41 const char *PreferredLanguage::I18N_PREFERENCES_FILE_NAME = "/i18n";
42 const char *PreferredLanguage::DEFAULT_PREFERRED_LANGUAGE = "en-Latn-US";
43 std::unordered_set<std::string> PreferredLanguage::supportLanguageListExt = { "it", "ko", "th", "zz" };
44
GetPreferredLanguageList()45 std::vector<std::string> PreferredLanguage::GetPreferredLanguageList()
46 {
47 char preferredLanguageValue[CONFIG_LEN];
48 GetParameter(PREFERRED_LANGUAGES, "", preferredLanguageValue, CONFIG_LEN);
49 std::string systemLanguage = GetMatchedLanguage(LocaleConfig::GetSystemLanguage());
50 std::vector<std::string> list;
51 Split(preferredLanguageValue, ";", list);
52 list = FilterLanguages(list);
53 if (!list.size()) {
54 if (systemLanguage != "") {
55 list.push_back(systemLanguage);
56 }
57 return list;
58 }
59 if (list[0] == systemLanguage || systemLanguage.empty()) {
60 return list;
61 }
62 int systemLanguageIdx = -1;
63 for (size_t i = 0; i < list.size(); i++) {
64 if (list[i] == systemLanguage) {
65 systemLanguageIdx = (int)i;
66 }
67 }
68 if (systemLanguageIdx == -1) {
69 list.insert(list.begin(), systemLanguage);
70 } else {
71 for (size_t i = (size_t)systemLanguageIdx; i > 0; i--) {
72 list[i] = list[i - 1];
73 }
74 list[0] = systemLanguage;
75 }
76 return list;
77 }
78
FilterLanguages(std::vector<std::string> & preferredLanguagesList)79 std::vector<std::string> PreferredLanguage::FilterLanguages(std::vector<std::string>& preferredLanguagesList)
80 {
81 std::vector<std::string> matchedLanguagesList;
82 std::unordered_set<std::string> matchedSet;
83 for (auto& preferredLanguage : preferredLanguagesList) {
84 std::string matchedLanguage = GetMatchedLanguage(preferredLanguage);
85 if (matchedLanguage.empty()) {
86 HILOG_ERROR_I18N("FilterLanguages: the matching result of %{public}s is empty.",
87 preferredLanguage.c_str());
88 matchedLanguage = DEFAULT_PREFERRED_LANGUAGE;
89 }
90 if (matchedSet.find(matchedLanguage) == matchedSet.end()) {
91 matchedLanguagesList.push_back(matchedLanguage);
92 matchedSet.insert(matchedLanguage);
93 HILOG_ERROR_I18N("FilterLanguages: the matching result is %{public}s.", matchedLanguage.c_str());
94 }
95 }
96 return matchedLanguagesList;
97 }
98
GetMatchedLanguage(const std::string & language)99 std::string PreferredLanguage::GetMatchedLanguage(const std::string& language)
100 {
101 UErrorCode status = U_ZERO_ERROR;
102 icu::Locale locale = icu::Locale::forLanguageTag(language.c_str(), status);
103 if (U_FAILURE(status) || !IsValidLocaleTag(locale)) {
104 HILOG_ERROR_I18N("GetMatchedLanguage: %{public}s is an invalid locale.", language.c_str());
105 return "";
106 }
107 LocaleInfo* requestLocale = new LocaleInfo(language);
108 if (requestLocale == nullptr) {
109 HILOG_ERROR_I18N("GetMatchedLanguage: %{public}s failed to construct LocaleInfo.", language.c_str());
110 return "";
111 }
112 std::vector<LocaleInfo*> candidateLocales;
113 std::unordered_set<std::string> supportLanguageList = LocaleConfig::GetSystemLanguages();
114 supportLanguageList.insert(supportLanguageListExt.begin(), supportLanguageListExt.end());
115 for (auto& supportLanguage : supportLanguageList) {
116 LocaleInfo* supportLocaleInfo = new LocaleInfo(supportLanguage);
117 if (supportLocaleInfo == nullptr) {
118 HILOG_ERROR_I18N("GetMatchedLanguage: %{public}s failed to construct LocaleInfo.",
119 supportLanguage.c_str());
120 continue;
121 }
122 if (LocaleMatcher::Match(requestLocale, supportLocaleInfo)) {
123 candidateLocales.push_back(supportLocaleInfo);
124 } else {
125 delete supportLocaleInfo;
126 }
127 }
128 std::string matchedLanguage = LocaleMatcher::GetBestMatchedLocale(requestLocale, candidateLocales);
129 for (LocaleInfo* supportLocaleInfo : candidateLocales) {
130 delete supportLocaleInfo;
131 }
132 delete requestLocale;
133 return matchedLanguage;
134 }
135
GetFirstPreferredLanguage()136 std::string PreferredLanguage::GetFirstPreferredLanguage()
137 {
138 std::vector<std::string> preferredLanguageList = GetPreferredLanguageList();
139 if (preferredLanguageList.empty()) {
140 return "";
141 }
142 return preferredLanguageList[0];
143 }
144
145 #ifdef SUPPORT_APP_PREFERRED_LANGUAGE
GetI18nAppPreferences()146 std::shared_ptr<NativePreferences::Preferences> PreferredLanguage::GetI18nAppPreferences()
147 {
148 std::shared_ptr<AbilityRuntime::ApplicationContext> appContext = AbilityRuntime::ApplicationContext::GetInstance();
149 int defaultMode = appContext->GetArea();
150 appContext->SwitchArea(0);
151 std::string preferencesDirPath = appContext->GetPreferencesDir();
152 appContext->SwitchArea(defaultMode);
153 std::string i18nPreferencesFilePath = preferencesDirPath + I18N_PREFERENCES_FILE_NAME;
154 int status = 0;
155 NativePreferences::Options options(i18nPreferencesFilePath);
156 std::shared_ptr<NativePreferences::Preferences> preferences =
157 NativePreferences::PreferencesHelper::GetPreferences(options, status);
158 if (status != 0) {
159 HILOG_ERROR_I18N("PreferredLanguage::GetAppPreferredLanguage get i18n app preferences failed.");
160 return nullptr;
161 }
162 return preferences;
163 }
164
IsSetAppPreferredLanguage()165 bool PreferredLanguage::IsSetAppPreferredLanguage()
166 {
167 std::shared_ptr<NativePreferences::Preferences> preferences = GetI18nAppPreferences();
168 if (preferences == nullptr) {
169 HILOG_ERROR_I18N(
170 "PreferredLanguage::IsSetAppPreferredLanguage get i18n preferences failed, return system language.");
171 return false;
172 }
173 std::string res = preferences->GetString(PreferredLanguage::APP_LANGUAGE_KEY, "");
174 if (res.length() == 0 || res.compare("default") == 0) {
175 return false;
176 }
177 return true;
178 }
179
GetAppPreferredLanguage()180 std::string PreferredLanguage::GetAppPreferredLanguage()
181 {
182 std::shared_ptr<NativePreferences::Preferences> preferences = GetI18nAppPreferences();
183 if (preferences == nullptr) {
184 HILOG_ERROR_I18N(
185 "PreferredLanguage::GetAppPreferredLanguage get i18n preferences failed, return system language.");
186 return LocaleConfig::GetSystemLocale();
187 }
188 std::string res = preferences->GetString(PreferredLanguage::APP_LANGUAGE_KEY, "");
189 if (res.length() == 0 || res.compare("default") == 0) {
190 return LocaleConfig::GetSystemLocale();
191 }
192 return res;
193 }
194
SetAppPreferredLanguage(const std::string & language,I18nErrorCode & errCode)195 void PreferredLanguage::SetAppPreferredLanguage(const std::string &language, I18nErrorCode &errCode)
196 {
197 std::shared_ptr<AbilityRuntime::ApplicationContext> appContext = AbilityRuntime::ApplicationContext::GetInstance();
198 if (language.compare("default") != 0) {
199 appContext->SetLanguage(language);
200 }
201 std::shared_ptr<NativePreferences::Preferences> preferences = GetI18nAppPreferences();
202 if (preferences == nullptr) {
203 errCode = I18nErrorCode::FAILED;
204 HILOG_ERROR_I18N("PreferredLanguage::SetAppPreferredLanguage get i18n preferences failed.");
205 return;
206 }
207 int32_t status = preferences->PutString(PreferredLanguage::APP_LANGUAGE_KEY, language);
208 if (status != 0) {
209 errCode = I18nErrorCode::FAILED;
210 HILOG_ERROR_I18N(
211 "PreferredLanguage::SetAppPreferredLanguage set app language to i18n preferences failed.");
212 return;
213 }
214 preferences->Flush();
215 }
216 #endif
217
GetPreferredLocale()218 std::string PreferredLanguage::GetPreferredLocale()
219 {
220 std::string systemLocale = LocaleConfig::GetSystemLocale();
221 LocaleInfo systemLocaleInfo(systemLocale);
222 std::string systemRegion = systemLocaleInfo.GetRegion();
223 std::string preferredLanguageLocale = GetFirstPreferredLanguage();
224 LocaleInfo preferredLanguageLocaleInfo(preferredLanguageLocale);
225 std::string preferredLanguage = preferredLanguageLocaleInfo.GetLanguage();
226 std::string preferredLocale = preferredLanguage + "-" + systemRegion;
227 return preferredLocale;
228 }
229
IsValidLanguage(const std::string & language)230 bool PreferredLanguage::IsValidLanguage(const std::string &language)
231 {
232 std::string::size_type size = language.size();
233 if ((size != LANGUAGE_LEN) && (size != LANGUAGE_LEN + 1)) {
234 return false;
235 }
236 for (size_t i = 0; i < size; ++i) {
237 if ((language[i] > 'z') || (language[i] < 'a')) {
238 return false;
239 }
240 }
241 return true;
242 }
243
IsValidTag(const std::string & tag)244 bool PreferredLanguage::IsValidTag(const std::string &tag)
245 {
246 if (!tag.size()) {
247 return false;
248 }
249 std::vector<std::string> splits;
250 Split(tag, "-", splits);
251 if (!IsValidLanguage(splits[0])) {
252 return false;
253 }
254 return true;
255 }
256
Split(const std::string & src,const std::string & sep,std::vector<std::string> & dest)257 void PreferredLanguage::Split(const std::string &src, const std::string &sep, std::vector<std::string> &dest)
258 {
259 std::string::size_type begin = 0;
260 std::string::size_type end = src.find(sep);
261 while (end != std::string::npos) {
262 dest.push_back(src.substr(begin, end - begin));
263 begin = end + sep.size();
264 end = src.find(sep, begin);
265 }
266 if (begin != src.size()) {
267 dest.push_back(src.substr(begin));
268 }
269 }
270
AddPreferredLanguage(const std::string & language,int32_t index)271 I18nErrorCode PreferredLanguage::AddPreferredLanguage(const std::string &language, int32_t index)
272 {
273 if (!IsValidTag(language)) {
274 HILOG_ERROR_I18N("PreferredLanguage::AddPreferredLanguage %{public}s is not valid language tag.",
275 language.c_str());
276 return I18nErrorCode::INVALID_LANGUAGE_TAG;
277 }
278 std::vector<std::string> preferredLanguages;
279 I18nErrorCode status = I18nErrorCode::SUCCESS;
280 if (FindLanguage(language) == -1) {
281 // Case: language not in current preferred language list.
282 AddNonExistPreferredLanguage(language, index, preferredLanguages, status);
283 } else {
284 // Case: language in current preferred language list.
285 AddExistPreferredLanguage(language, index, preferredLanguages, status);
286 }
287 if (status != I18nErrorCode::SUCCESS) {
288 HILOG_ERROR_I18N("PreferredLanguage::AddPreferredLanguage failed.");
289 return status;
290 }
291 return SetPreferredLanguages(JoinPreferredLanguages(preferredLanguages));
292 }
293
RemovePreferredLanguage(int32_t index)294 I18nErrorCode PreferredLanguage::RemovePreferredLanguage(int32_t index)
295 {
296 std::vector<std::string> preferredLanguages = GetPreferredLanguageList();
297 if (preferredLanguages.size() == 1) {
298 HILOG_ERROR_I18N("PreferredLanguage::RemovePreferredLanguage can't remove the only language.");
299 return I18nErrorCode::REMOVE_PREFERRED_LANGUAGE_FAILED;
300 }
301 // valid index is [0, preferredLanguages.size() - 1] for Remove
302 int32_t validIndex = NormalizeIndex(index, preferredLanguages.size() - 1);
303 preferredLanguages.erase(preferredLanguages.begin() + validIndex);
304 // The first language in preferred language list is system language, therefor when first language changed
305 // in preferred language list, we need to reset system language.
306 if (validIndex == 0) {
307 if (LocaleConfig::SetSystemLanguage(preferredLanguages[0]) != I18nErrorCode::SUCCESS) {
308 HILOG_ERROR_I18N("PreferredLanguage::RemovePreferredLanguage update system language failed.");
309 return I18nErrorCode::REMOVE_PREFERRED_LANGUAGE_FAILED;
310 }
311 }
312 return SetPreferredLanguages(JoinPreferredLanguages(preferredLanguages));
313 }
314
AddNonExistPreferredLanguage(const std::string & language,int32_t index,std::vector<std::string> & preferredLanguages,I18nErrorCode & errCode)315 void PreferredLanguage::AddNonExistPreferredLanguage(const std::string& language, int32_t index,
316 std::vector<std::string> &preferredLanguages, I18nErrorCode &errCode)
317 {
318 // valid index is [0, GetPreferredLanguageList().size()] for add non-exist language.
319 int32_t validIndex = NormalizeIndex(index, GetPreferredLanguageList().size());
320 preferredLanguages = GetPreferredLanguageList();
321 preferredLanguages.insert(preferredLanguages.begin() + validIndex, language);
322 // The first language in preferred language list is system language, therefor when first language changed
323 // in preferred language list, we need to reset system language.
324 if (validIndex == 0) {
325 if (LocaleConfig::SetSystemLanguage(preferredLanguages[0]) != I18nErrorCode::SUCCESS) {
326 HILOG_ERROR_I18N("PreferredLanguage::AddNonExistPreferredLanguage update system language failed.");
327 errCode = I18nErrorCode::ADD_PREFERRED_LANGUAGE_NON_EXIST_FAILED;
328 return;
329 }
330 }
331 errCode = I18nErrorCode::SUCCESS;
332 }
333
AddExistPreferredLanguage(const std::string & language,int32_t index,std::vector<std::string> & preferredLanguages,I18nErrorCode & errCode)334 void PreferredLanguage::AddExistPreferredLanguage(const std::string& language, int32_t index,
335 std::vector<std::string> &preferredLanguages, I18nErrorCode &errCode)
336 {
337 // throw error when current index is same with target index.
338 // valid index is [0, GetPreferredLanguageList().size() - 1] for add exist language.
339 int32_t validIndex = NormalizeIndex(index, GetPreferredLanguageList().size() - 1);
340 int32_t languageIdx = FindLanguage(language);
341 if (languageIdx == validIndex) {
342 errCode = I18nErrorCode::ADD_PREFERRED_LANGUAGE_EXIST_FAILED;
343 return;
344 }
345 // Move language from languageIdx to validIdx.
346 preferredLanguages = GetPreferredLanguageList();
347 preferredLanguages.erase(preferredLanguages.begin() + languageIdx);
348 preferredLanguages.insert(preferredLanguages.begin() + validIndex, language);
349 // The first language in preferred language list is system language, therefor when first language changed
350 // in preferred language list, we need to reset system language.
351 if (languageIdx == 0 || validIndex == 0) {
352 if (LocaleConfig::SetSystemLanguage(preferredLanguages[0]) != I18nErrorCode::SUCCESS) {
353 HILOG_ERROR_I18N("PreferredLanguage::AddExistPreferredLanguage update system language failed.");
354 errCode = I18nErrorCode::ADD_PREFERRED_LANGUAGE_EXIST_FAILED;
355 return;
356 }
357 }
358 errCode = I18nErrorCode::SUCCESS;
359 }
360
NormalizeIndex(int32_t index,int32_t max)361 int32_t PreferredLanguage::NormalizeIndex(int32_t index, int32_t max)
362 {
363 if (index <= 0) {
364 return 0;
365 }
366 if (index >= max) {
367 return max;
368 }
369 return index;
370 }
371
372 // Query the index of language in system preferred language list.
FindLanguage(const std::string & language)373 int32_t PreferredLanguage::FindLanguage(const std::string &language)
374 {
375 std::vector<std::string> preferredLanguageList = GetPreferredLanguageList();
376 for (size_t i = 0; i < preferredLanguageList.size(); ++i) {
377 if (preferredLanguageList[i] == language) {
378 return static_cast<int32_t>(i);
379 }
380 }
381 return -1;
382 }
383
384 // Join preferred language tags to string with ';'
JoinPreferredLanguages(const std::vector<std::string> preferredLanguages)385 std::string PreferredLanguage::JoinPreferredLanguages(const std::vector<std::string> preferredLanguages)
386 {
387 std::string result = "";
388 for (size_t i = 0; i < preferredLanguages.size(); ++i) {
389 result += preferredLanguages[i];
390 result += ";";
391 }
392 // delete the last ';'
393 result.pop_back();
394 return result;
395 }
396
397 // Set PREFERRED_LANGUAGES system parameter with preferredLanguages.
SetPreferredLanguages(const std::string & preferredLanguages)398 I18nErrorCode PreferredLanguage::SetPreferredLanguages(const std::string &preferredLanguages)
399 {
400 // System parameter value's length can't beyong CONFIG_LEN
401 if (preferredLanguages.length() > CONFIG_LEN) {
402 HILOG_ERROR_I18N("PreferredLanguage::SetPreferredLanguage preferred language list is too long.");
403 return I18nErrorCode::UPDATE_SYSTEM_PREFERRED_LANGUAGE_FAILED;
404 }
405 if (SetParameter(PREFERRED_LANGUAGES, preferredLanguages.data()) != 0) {
406 HILOG_ERROR_I18N("PreferredLanguage::AddPreferredLanguage udpate preferred language param failed.");
407 return I18nErrorCode::UPDATE_SYSTEM_PREFERRED_LANGUAGE_FAILED;
408 }
409 return I18nErrorCode::SUCCESS;
410 }
411 } // namespace I18n
412 } // namespace Global
413 } // namespace OHOS