• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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     if (appContext == nullptr) {
150         HILOG_ERROR_I18N("PreferredLanguage::GetAppPreferredLanguage: Get app context failed.");
151         return nullptr;
152     }
153     int defaultMode = appContext->GetArea();
154     appContext->SwitchArea(0);
155     std::string preferencesDirPath = appContext->GetPreferencesDir();
156     appContext->SwitchArea(defaultMode);
157     std::string i18nPreferencesFilePath = preferencesDirPath + I18N_PREFERENCES_FILE_NAME;
158     int status = 0;
159     NativePreferences::Options options(i18nPreferencesFilePath);
160     std::shared_ptr<NativePreferences::Preferences> preferences =
161         NativePreferences::PreferencesHelper::GetPreferences(options, status);
162     if (status != 0) {
163         HILOG_ERROR_I18N("PreferredLanguage::GetAppPreferredLanguage get i18n app preferences failed.");
164         return nullptr;
165     }
166     return preferences;
167 }
168 
IsSetAppPreferredLanguage()169 bool PreferredLanguage::IsSetAppPreferredLanguage()
170 {
171     std::shared_ptr<NativePreferences::Preferences> preferences = GetI18nAppPreferences();
172     if (preferences == nullptr) {
173         HILOG_ERROR_I18N(
174             "PreferredLanguage::IsSetAppPreferredLanguage get i18n preferences failed, return system language.");
175         return false;
176     }
177     std::string res = preferences->GetString(PreferredLanguage::APP_LANGUAGE_KEY, "");
178     if (res.length() == 0 || res.compare("default") == 0) {
179         return false;
180     }
181     return true;
182 }
183 
GetAppPreferredLanguage()184 std::string PreferredLanguage::GetAppPreferredLanguage()
185 {
186     std::shared_ptr<NativePreferences::Preferences> preferences = GetI18nAppPreferences();
187     if (preferences == nullptr) {
188         HILOG_ERROR_I18N(
189             "PreferredLanguage::GetAppPreferredLanguage get i18n preferences failed, return system language.");
190         return LocaleConfig::GetEffectiveLanguage();
191     }
192     std::string res = preferences->GetString(PreferredLanguage::APP_LANGUAGE_KEY, "");
193     if (res.length() == 0 || res.compare("default") == 0) {
194         return LocaleConfig::GetEffectiveLanguage();
195     }
196     return res;
197 }
198 
SetAppPreferredLanguage(const std::string & language,I18nErrorCode & errCode)199 void PreferredLanguage::SetAppPreferredLanguage(const std::string &language, I18nErrorCode &errCode)
200 {
201     std::shared_ptr<AbilityRuntime::ApplicationContext> appContext = AbilityRuntime::ApplicationContext::GetInstance();
202     if (appContext != nullptr && language.compare("default") != 0) {
203         appContext->SetLanguage(language);
204     }
205     std::shared_ptr<NativePreferences::Preferences> preferences = GetI18nAppPreferences();
206     if (preferences == nullptr) {
207         errCode = I18nErrorCode::FAILED;
208         HILOG_ERROR_I18N("PreferredLanguage::SetAppPreferredLanguage get i18n preferences failed.");
209         return;
210     }
211     int32_t status = preferences->PutString(PreferredLanguage::APP_LANGUAGE_KEY, language);
212     if (status != 0) {
213         errCode = I18nErrorCode::FAILED;
214         HILOG_ERROR_I18N(
215             "PreferredLanguage::SetAppPreferredLanguage set app language to i18n preferences failed.");
216         return;
217     }
218     preferences->Flush();
219 }
220 #endif
221 
GetPreferredLocale()222 std::string PreferredLanguage::GetPreferredLocale()
223 {
224     std::string systemLocale = LocaleConfig::GetSystemLocale();
225     LocaleInfo systemLocaleInfo(systemLocale);
226     std::string systemRegion = systemLocaleInfo.GetRegion();
227     std::string preferredLanguageLocale = GetFirstPreferredLanguage();
228     LocaleInfo preferredLanguageLocaleInfo(preferredLanguageLocale);
229     std::string preferredLanguage = preferredLanguageLocaleInfo.GetLanguage();
230     std::string preferredLocale = preferredLanguage + "-" + systemRegion;
231     return preferredLocale;
232 }
233 
IsValidLanguage(const std::string & language)234 bool PreferredLanguage::IsValidLanguage(const std::string &language)
235 {
236     std::string::size_type size = language.size();
237     if ((size != LANGUAGE_LEN) && (size != LANGUAGE_LEN + 1)) {
238         return false;
239     }
240     for (size_t i = 0; i < size; ++i) {
241         if ((language[i] > 'z') || (language[i] < 'a')) {
242             return false;
243         }
244     }
245     return true;
246 }
247 
IsValidTag(const std::string & tag)248 bool PreferredLanguage::IsValidTag(const std::string &tag)
249 {
250     if (!tag.size()) {
251         return false;
252     }
253     std::vector<std::string> splits;
254     Split(tag, "-", splits);
255     if (!IsValidLanguage(splits[0])) {
256         return false;
257     }
258     return true;
259 }
260 
Split(const std::string & src,const std::string & sep,std::vector<std::string> & dest)261 void PreferredLanguage::Split(const std::string &src, const std::string &sep, std::vector<std::string> &dest)
262 {
263     std::string::size_type begin = 0;
264     std::string::size_type end = src.find(sep);
265     while (end != std::string::npos) {
266         dest.push_back(src.substr(begin, end - begin));
267         begin = end + sep.size();
268         end = src.find(sep, begin);
269     }
270     if (begin != src.size()) {
271         dest.push_back(src.substr(begin));
272     }
273 }
274 
AddPreferredLanguage(const std::string & language,int32_t index)275 I18nErrorCode PreferredLanguage::AddPreferredLanguage(const std::string &language, int32_t index)
276 {
277     if (!IsValidTag(language)) {
278         HILOG_ERROR_I18N("PreferredLanguage::AddPreferredLanguage %{public}s is not valid language tag.",
279             language.c_str());
280         return I18nErrorCode::INVALID_LANGUAGE_TAG;
281     }
282     std::vector<std::string> preferredLanguages;
283     I18nErrorCode status = I18nErrorCode::SUCCESS;
284     if (FindLanguage(language) == -1) {
285         // Case: language not in current preferred language list.
286         AddNonExistPreferredLanguage(language, index, preferredLanguages, status);
287     } else {
288         // Case: language in current preferred language list.
289         AddExistPreferredLanguage(language, index, preferredLanguages, status);
290     }
291     if (status != I18nErrorCode::SUCCESS) {
292         HILOG_ERROR_I18N("PreferredLanguage::AddPreferredLanguage failed.");
293         return status;
294     }
295     return SetPreferredLanguages(JoinPreferredLanguages(preferredLanguages));
296 }
297 
RemovePreferredLanguage(int32_t index)298 I18nErrorCode PreferredLanguage::RemovePreferredLanguage(int32_t index)
299 {
300     std::vector<std::string> preferredLanguages = GetPreferredLanguageList();
301     if (preferredLanguages.size() == 1) {
302         HILOG_ERROR_I18N("PreferredLanguage::RemovePreferredLanguage can't remove the only language.");
303         return I18nErrorCode::REMOVE_PREFERRED_LANGUAGE_FAILED;
304     }
305     // valid index is [0, preferredLanguages.size() - 1] for Remove
306     int32_t validIndex = NormalizeIndex(index, preferredLanguages.size() - 1);
307     preferredLanguages.erase(preferredLanguages.begin() + validIndex);
308     // The first language in preferred language list is system language, therefor when first language changed
309     // in preferred language list, we need to reset system language.
310     if (validIndex == 0) {
311         if (LocaleConfig::SetSystemLanguage(preferredLanguages[0]) != I18nErrorCode::SUCCESS) {
312             HILOG_ERROR_I18N("PreferredLanguage::RemovePreferredLanguage update system language failed.");
313             return I18nErrorCode::REMOVE_PREFERRED_LANGUAGE_FAILED;
314         }
315     }
316     return SetPreferredLanguages(JoinPreferredLanguages(preferredLanguages));
317 }
318 
AddNonExistPreferredLanguage(const std::string & language,int32_t index,std::vector<std::string> & preferredLanguages,I18nErrorCode & errCode)319 void PreferredLanguage::AddNonExistPreferredLanguage(const std::string& language, int32_t index,
320     std::vector<std::string> &preferredLanguages, I18nErrorCode &errCode)
321 {
322     // valid index is [0, GetPreferredLanguageList().size()] for add non-exist language.
323     int32_t validIndex = NormalizeIndex(index, GetPreferredLanguageList().size());
324     preferredLanguages = GetPreferredLanguageList();
325     preferredLanguages.insert(preferredLanguages.begin() + validIndex, language);
326     // The first language in preferred language list is system language, therefor when first language changed
327     // in preferred language list, we need to reset system language.
328     if (validIndex == 0) {
329         if (LocaleConfig::SetSystemLanguage(preferredLanguages[0]) != I18nErrorCode::SUCCESS) {
330             HILOG_ERROR_I18N("PreferredLanguage::AddNonExistPreferredLanguage update system language failed.");
331             errCode = I18nErrorCode::ADD_PREFERRED_LANGUAGE_NON_EXIST_FAILED;
332             return;
333         }
334     }
335     errCode = I18nErrorCode::SUCCESS;
336 }
337 
AddExistPreferredLanguage(const std::string & language,int32_t index,std::vector<std::string> & preferredLanguages,I18nErrorCode & errCode)338 void PreferredLanguage::AddExistPreferredLanguage(const std::string& language, int32_t index,
339     std::vector<std::string> &preferredLanguages, I18nErrorCode &errCode)
340 {
341     // throw error when current index is same with target index.
342     // valid index is [0, GetPreferredLanguageList().size() - 1] for add exist language.
343     int32_t validIndex = NormalizeIndex(index, GetPreferredLanguageList().size() - 1);
344     int32_t languageIdx = FindLanguage(language);
345     if (languageIdx == validIndex) {
346         errCode = I18nErrorCode::ADD_PREFERRED_LANGUAGE_EXIST_FAILED;
347         return;
348     }
349     // Move language from languageIdx to validIdx.
350     preferredLanguages = GetPreferredLanguageList();
351     preferredLanguages.erase(preferredLanguages.begin() + languageIdx);
352     preferredLanguages.insert(preferredLanguages.begin() + validIndex, language);
353     // The first language in preferred language list is system language, therefor when first language changed
354     // in preferred language list, we need to reset system language.
355     if (languageIdx == 0 || validIndex == 0) {
356         if (LocaleConfig::SetSystemLanguage(preferredLanguages[0]) != I18nErrorCode::SUCCESS) {
357             HILOG_ERROR_I18N("PreferredLanguage::AddExistPreferredLanguage update system language failed.");
358             errCode = I18nErrorCode::ADD_PREFERRED_LANGUAGE_EXIST_FAILED;
359             return;
360         }
361     }
362     errCode = I18nErrorCode::SUCCESS;
363 }
364 
NormalizeIndex(int32_t index,int32_t max)365 int32_t PreferredLanguage::NormalizeIndex(int32_t index, int32_t max)
366 {
367     if (index <= 0) {
368         return 0;
369     }
370     if (index >= max) {
371         return max;
372     }
373     return index;
374 }
375 
376 // Query the index of language in system preferred language list.
FindLanguage(const std::string & language)377 int32_t PreferredLanguage::FindLanguage(const std::string &language)
378 {
379     std::vector<std::string> preferredLanguageList = GetPreferredLanguageList();
380     for (size_t i = 0; i < preferredLanguageList.size(); ++i) {
381         if (preferredLanguageList[i] == language) {
382             return static_cast<int32_t>(i);
383         }
384     }
385     return -1;
386 }
387 
388 // Join preferred language tags to string with ';'
JoinPreferredLanguages(const std::vector<std::string> preferredLanguages)389 std::string PreferredLanguage::JoinPreferredLanguages(const std::vector<std::string> preferredLanguages)
390 {
391     std::string result = "";
392     for (size_t i = 0; i < preferredLanguages.size(); ++i) {
393         result += preferredLanguages[i];
394         result += ";";
395     }
396     // delete the last ';'
397     result.pop_back();
398     return result;
399 }
400 
401 // Set PREFERRED_LANGUAGES system parameter with preferredLanguages.
SetPreferredLanguages(const std::string & preferredLanguages)402 I18nErrorCode PreferredLanguage::SetPreferredLanguages(const std::string &preferredLanguages)
403 {
404     // System parameter value's length can't beyong CONFIG_LEN
405     if (preferredLanguages.length() > CONFIG_LEN) {
406         HILOG_ERROR_I18N("PreferredLanguage::SetPreferredLanguage preferred language list is too long.");
407         return I18nErrorCode::UPDATE_SYSTEM_PREFERRED_LANGUAGE_FAILED;
408     }
409     if (SetParameter(PREFERRED_LANGUAGES, preferredLanguages.data()) != 0) {
410         HILOG_ERROR_I18N("PreferredLanguage::AddPreferredLanguage udpate preferred language param failed.");
411         return I18nErrorCode::UPDATE_SYSTEM_PREFERRED_LANGUAGE_FAILED;
412     }
413     return I18nErrorCode::SUCCESS;
414 }
415 } // namespace I18n
416 } // namespace Global
417 } // namespace OHOS