1 /*
2  * Copyright (C) 2017 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License
15  */
16 
17 package com.android.settingslib.inputmethod;
18 
19 import android.content.ContentResolver;
20 import android.content.Context;
21 import android.content.SharedPreferences;
22 import android.content.res.Configuration;
23 import android.icu.text.ListFormatter;
24 import android.provider.Settings;
25 import android.provider.Settings.SettingNotFoundException;
26 import android.text.TextUtils;
27 import android.util.Log;
28 import android.view.inputmethod.InputMethodInfo;
29 import android.view.inputmethod.InputMethodSubtype;
30 
31 import androidx.annotation.NonNull;
32 import androidx.annotation.Nullable;
33 import androidx.preference.Preference;
34 import androidx.preference.PreferenceFragment;
35 import androidx.preference.PreferenceScreen;
36 import androidx.preference.TwoStatePreference;
37 
38 import com.android.internal.app.LocaleHelper;
39 
40 import java.util.HashMap;
41 import java.util.HashSet;
42 import java.util.List;
43 import java.util.Locale;
44 import java.util.Map;
45 
46 // TODO: Consolidate this with {@link InputMethodSettingValuesWrapper}.
47 public class InputMethodAndSubtypeUtil {
48 
49     private static final boolean DEBUG = false;
50     private static final String TAG = "InputMethdAndSubtypeUtl";
51 
52     private static final String SUBTYPE_MODE_KEYBOARD = "keyboard";
53     private static final char INPUT_METHOD_SEPARATER = ':';
54     private static final char INPUT_METHOD_SUBTYPE_SEPARATER = ';';
55     private static final int NOT_A_SUBTYPE_ID = -1;
56 
57     private static final TextUtils.SimpleStringSplitter sStringInputMethodSplitter
58             = new TextUtils.SimpleStringSplitter(INPUT_METHOD_SEPARATER);
59 
60     private static final TextUtils.SimpleStringSplitter sStringInputMethodSubtypeSplitter
61             = new TextUtils.SimpleStringSplitter(INPUT_METHOD_SUBTYPE_SEPARATER);
62 
63     // InputMethods and subtypes are saved in the settings as follows:
64     // ime0;subtype0;subtype1:ime1;subtype0:ime2:ime3;subtype0;subtype1
buildInputMethodsAndSubtypesString( final HashMap<String, HashSet<String>> imeToSubtypesMap)65     public static String buildInputMethodsAndSubtypesString(
66             final HashMap<String, HashSet<String>> imeToSubtypesMap) {
67         final StringBuilder builder = new StringBuilder();
68         for (final String imi : imeToSubtypesMap.keySet()) {
69             if (builder.length() > 0) {
70                 builder.append(INPUT_METHOD_SEPARATER);
71             }
72             final HashSet<String> subtypeIdSet = imeToSubtypesMap.get(imi);
73             builder.append(imi);
74             for (final String subtypeId : subtypeIdSet) {
75                 builder.append(INPUT_METHOD_SUBTYPE_SEPARATER).append(subtypeId);
76             }
77         }
78         return builder.toString();
79     }
80 
buildInputMethodsString(final HashSet<String> imiList)81     private static String buildInputMethodsString(final HashSet<String> imiList) {
82         final StringBuilder builder = new StringBuilder();
83         for (final String imi : imiList) {
84             if (builder.length() > 0) {
85                 builder.append(INPUT_METHOD_SEPARATER);
86             }
87             builder.append(imi);
88         }
89         return builder.toString();
90     }
91 
getInputMethodSubtypeSelected(ContentResolver resolver)92     private static int getInputMethodSubtypeSelected(ContentResolver resolver) {
93         try {
94             return Settings.Secure.getInt(resolver,
95                     Settings.Secure.SELECTED_INPUT_METHOD_SUBTYPE);
96         } catch (SettingNotFoundException e) {
97             return NOT_A_SUBTYPE_ID;
98         }
99     }
100 
isInputMethodSubtypeSelected(ContentResolver resolver)101     private static boolean isInputMethodSubtypeSelected(ContentResolver resolver) {
102         return getInputMethodSubtypeSelected(resolver) != NOT_A_SUBTYPE_ID;
103     }
104 
putSelectedInputMethodSubtype(ContentResolver resolver, int hashCode)105     private static void putSelectedInputMethodSubtype(ContentResolver resolver, int hashCode) {
106         Settings.Secure.putInt(resolver, Settings.Secure.SELECTED_INPUT_METHOD_SUBTYPE, hashCode);
107     }
108 
109     // Needs to modify InputMethodManageService if you want to change the format of saved string.
getEnabledInputMethodsAndSubtypeList( ContentResolver resolver)110     static HashMap<String, HashSet<String>> getEnabledInputMethodsAndSubtypeList(
111             ContentResolver resolver) {
112         final String enabledInputMethodsStr = Settings.Secure.getString(
113                 resolver, Settings.Secure.ENABLED_INPUT_METHODS);
114         if (DEBUG) {
115             Log.d(TAG, "--- Load enabled input methods: " + enabledInputMethodsStr);
116         }
117         return parseInputMethodsAndSubtypesString(enabledInputMethodsStr);
118     }
119 
parseInputMethodsAndSubtypesString( final String inputMethodsAndSubtypesString)120     public static HashMap<String, HashSet<String>> parseInputMethodsAndSubtypesString(
121             final String inputMethodsAndSubtypesString) {
122         final HashMap<String, HashSet<String>> subtypesMap = new HashMap<>();
123         if (TextUtils.isEmpty(inputMethodsAndSubtypesString)) {
124             return subtypesMap;
125         }
126         sStringInputMethodSplitter.setString(inputMethodsAndSubtypesString);
127         while (sStringInputMethodSplitter.hasNext()) {
128             final String nextImsStr = sStringInputMethodSplitter.next();
129             sStringInputMethodSubtypeSplitter.setString(nextImsStr);
130             if (sStringInputMethodSubtypeSplitter.hasNext()) {
131                 final HashSet<String> subtypeIdSet = new HashSet<>();
132                 // The first element is {@link InputMethodInfoId}.
133                 final String imiId = sStringInputMethodSubtypeSplitter.next();
134                 while (sStringInputMethodSubtypeSplitter.hasNext()) {
135                     subtypeIdSet.add(sStringInputMethodSubtypeSplitter.next());
136                 }
137                 subtypesMap.put(imiId, subtypeIdSet);
138             }
139         }
140         return subtypesMap;
141     }
142 
getDisabledSystemIMEs(ContentResolver resolver)143     private static HashSet<String> getDisabledSystemIMEs(ContentResolver resolver) {
144         HashSet<String> set = new HashSet<>();
145         String disabledIMEsStr = Settings.Secure.getString(
146                 resolver, Settings.Secure.DISABLED_SYSTEM_INPUT_METHODS);
147         if (TextUtils.isEmpty(disabledIMEsStr)) {
148             return set;
149         }
150         sStringInputMethodSplitter.setString(disabledIMEsStr);
151         while(sStringInputMethodSplitter.hasNext()) {
152             set.add(sStringInputMethodSplitter.next());
153         }
154         return set;
155     }
156 
saveInputMethodSubtypeList(PreferenceFragment context, ContentResolver resolver, List<InputMethodInfo> inputMethodInfos, boolean hasHardKeyboard)157     public static void saveInputMethodSubtypeList(PreferenceFragment context,
158             ContentResolver resolver, List<InputMethodInfo> inputMethodInfos,
159             boolean hasHardKeyboard) {
160         String currentInputMethodId = Settings.Secure.getString(resolver,
161                 Settings.Secure.DEFAULT_INPUT_METHOD);
162         final int selectedInputMethodSubtype = getInputMethodSubtypeSelected(resolver);
163         final HashMap<String, HashSet<String>> enabledIMEsAndSubtypesMap =
164                 getEnabledInputMethodsAndSubtypeList(resolver);
165         final HashSet<String> disabledSystemIMEs = getDisabledSystemIMEs(resolver);
166 
167         boolean needsToResetSelectedSubtype = false;
168         for (final InputMethodInfo imi : inputMethodInfos) {
169             final String imiId = imi.getId();
170             final Preference pref = context.findPreference(imiId);
171             if (pref == null) {
172                 continue;
173             }
174             // In the choose input method screen or in the subtype enabler screen,
175             // <code>pref</code> is an instance of TwoStatePreference.
176             final boolean isImeChecked = (pref instanceof TwoStatePreference) ?
177                     ((TwoStatePreference) pref).isChecked()
178                     : enabledIMEsAndSubtypesMap.containsKey(imiId);
179             final boolean isCurrentInputMethod = imiId.equals(currentInputMethodId);
180             final boolean systemIme = imi.isSystem();
181             if ((!hasHardKeyboard && InputMethodSettingValuesWrapper.getInstance(
182                     context.getActivity()).isAlwaysCheckedIme(imi))
183                     || isImeChecked) {
184                 if (!enabledIMEsAndSubtypesMap.containsKey(imiId)) {
185                     // imiId has just been enabled
186                     enabledIMEsAndSubtypesMap.put(imiId, new HashSet<>());
187                 }
188                 final HashSet<String> subtypesSet = enabledIMEsAndSubtypesMap.get(imiId);
189 
190                 boolean subtypePrefFound = false;
191                 final int subtypeCount = imi.getSubtypeCount();
192                 for (int i = 0; i < subtypeCount; ++i) {
193                     final InputMethodSubtype subtype = imi.getSubtypeAt(i);
194                     final String subtypeHashCodeStr = String.valueOf(subtype.hashCode());
195                     final TwoStatePreference subtypePref = (TwoStatePreference) context
196                             .findPreference(imiId + subtypeHashCodeStr);
197                     // In the Configure input method screen which does not have subtype preferences.
198                     if (subtypePref == null) {
199                         continue;
200                     }
201                     if (!subtypePrefFound) {
202                         // Once subtype preference is found, subtypeSet needs to be cleared.
203                         // Because of system change, hashCode value could have been changed.
204                         subtypesSet.clear();
205                         // If selected subtype preference is disabled, needs to reset.
206                         needsToResetSelectedSubtype = true;
207                         subtypePrefFound = true;
208                     }
209                     // Checking <code>subtypePref.isEnabled()</code> is insufficient to determine
210                     // whether the user manually enabled this subtype or not.  Implicitly-enabled
211                     // subtypes are also checked just as an indicator to users.  We also need to
212                     // check <code>subtypePref.isEnabled()</code> so that only manually enabled
213                     // subtypes can be saved here.
214                     if (subtypePref.isEnabled() && subtypePref.isChecked()) {
215                         subtypesSet.add(subtypeHashCodeStr);
216                         if (isCurrentInputMethod) {
217                             if (selectedInputMethodSubtype == subtype.hashCode()) {
218                                 // Selected subtype is still enabled, there is no need to reset
219                                 // selected subtype.
220                                 needsToResetSelectedSubtype = false;
221                             }
222                         }
223                     } else {
224                         subtypesSet.remove(subtypeHashCodeStr);
225                     }
226                 }
227             } else {
228                 enabledIMEsAndSubtypesMap.remove(imiId);
229                 if (isCurrentInputMethod) {
230                     // We are processing the current input method, but found that it's not enabled.
231                     // This means that the current input method has been uninstalled.
232                     // If currentInputMethod is already uninstalled, InputMethodManagerService will
233                     // find the applicable IME from the history and the system locale.
234                     if (DEBUG) {
235                         Log.d(TAG, "Current IME was uninstalled or disabled.");
236                     }
237                     currentInputMethodId = null;
238                 }
239             }
240             // If it's a disabled system ime, add it to the disabled list so that it
241             // doesn't get enabled automatically on any changes to the package list
242             if (systemIme && hasHardKeyboard) {
243                 if (disabledSystemIMEs.contains(imiId)) {
244                     if (isImeChecked) {
245                         disabledSystemIMEs.remove(imiId);
246                     }
247                 } else {
248                     if (!isImeChecked) {
249                         disabledSystemIMEs.add(imiId);
250                     }
251                 }
252             }
253         }
254 
255         final String enabledIMEsAndSubtypesString = buildInputMethodsAndSubtypesString(
256                 enabledIMEsAndSubtypesMap);
257         final String disabledSystemIMEsString = buildInputMethodsString(disabledSystemIMEs);
258         if (DEBUG) {
259             Log.d(TAG, "--- Save enabled inputmethod settings. :" + enabledIMEsAndSubtypesString);
260             Log.d(TAG, "--- Save disabled system inputmethod settings. :"
261                     + disabledSystemIMEsString);
262             Log.d(TAG, "--- Save default inputmethod settings. :" + currentInputMethodId);
263             Log.d(TAG, "--- Needs to reset the selected subtype :" + needsToResetSelectedSubtype);
264             Log.d(TAG, "--- Subtype is selected :" + isInputMethodSubtypeSelected(resolver));
265         }
266 
267         // Redefines SelectedSubtype when all subtypes are unchecked or there is no subtype
268         // selected. And if the selected subtype of the current input method was disabled,
269         // We should reset the selected input method's subtype.
270         if (needsToResetSelectedSubtype || !isInputMethodSubtypeSelected(resolver)) {
271             if (DEBUG) {
272                 Log.d(TAG, "--- Reset inputmethod subtype because it's not defined.");
273             }
274             putSelectedInputMethodSubtype(resolver, NOT_A_SUBTYPE_ID);
275         }
276 
277         Settings.Secure.putString(resolver,
278                 Settings.Secure.ENABLED_INPUT_METHODS, enabledIMEsAndSubtypesString);
279         if (disabledSystemIMEsString.length() > 0) {
280             Settings.Secure.putString(resolver, Settings.Secure.DISABLED_SYSTEM_INPUT_METHODS,
281                     disabledSystemIMEsString);
282         }
283         // If the current input method is unset, InputMethodManagerService will find the applicable
284         // IME from the history and the system locale.
285         Settings.Secure.putString(resolver, Settings.Secure.DEFAULT_INPUT_METHOD,
286                 currentInputMethodId != null ? currentInputMethodId : "");
287     }
288 
loadInputMethodSubtypeList(final PreferenceFragment context, final ContentResolver resolver, final List<InputMethodInfo> inputMethodInfos, final Map<String, List<Preference>> inputMethodPrefsMap)289     public static void loadInputMethodSubtypeList(final PreferenceFragment context,
290             final ContentResolver resolver, final List<InputMethodInfo> inputMethodInfos,
291             final Map<String, List<Preference>> inputMethodPrefsMap) {
292         final HashMap<String, HashSet<String>> enabledSubtypes =
293                 getEnabledInputMethodsAndSubtypeList(resolver);
294 
295         for (final InputMethodInfo imi : inputMethodInfos) {
296             final String imiId = imi.getId();
297             final Preference pref = context.findPreference(imiId);
298             if (pref instanceof TwoStatePreference) {
299                 final TwoStatePreference subtypePref = (TwoStatePreference) pref;
300                 final boolean isEnabled = enabledSubtypes.containsKey(imiId);
301                 subtypePref.setChecked(isEnabled);
302                 if (inputMethodPrefsMap != null) {
303                     for (final Preference childPref: inputMethodPrefsMap.get(imiId)) {
304                         childPref.setEnabled(isEnabled);
305                     }
306                 }
307                 setSubtypesPreferenceEnabled(context, inputMethodInfos, imiId, isEnabled);
308             }
309         }
310         updateSubtypesPreferenceChecked(context, inputMethodInfos, enabledSubtypes);
311     }
312 
setSubtypesPreferenceEnabled(final PreferenceFragment context, final List<InputMethodInfo> inputMethodProperties, final String id, final boolean enabled)313     private static void setSubtypesPreferenceEnabled(final PreferenceFragment context,
314             final List<InputMethodInfo> inputMethodProperties, final String id,
315             final boolean enabled) {
316         final PreferenceScreen preferenceScreen = context.getPreferenceScreen();
317         for (final InputMethodInfo imi : inputMethodProperties) {
318             if (id.equals(imi.getId())) {
319                 final int subtypeCount = imi.getSubtypeCount();
320                 for (int i = 0; i < subtypeCount; ++i) {
321                     final InputMethodSubtype subtype = imi.getSubtypeAt(i);
322                     final TwoStatePreference pref = (TwoStatePreference) preferenceScreen
323                             .findPreference(id + subtype.hashCode());
324                     if (pref != null) {
325                         pref.setEnabled(enabled);
326                     }
327                 }
328             }
329         }
330     }
331 
updateSubtypesPreferenceChecked(final PreferenceFragment context, final List<InputMethodInfo> inputMethodProperties, final HashMap<String, HashSet<String>> enabledSubtypes)332     private static void updateSubtypesPreferenceChecked(final PreferenceFragment context,
333             final List<InputMethodInfo> inputMethodProperties,
334             final HashMap<String, HashSet<String>> enabledSubtypes) {
335         final PreferenceScreen preferenceScreen = context.getPreferenceScreen();
336         for (final InputMethodInfo imi : inputMethodProperties) {
337             final String id = imi.getId();
338             if (!enabledSubtypes.containsKey(id)) {
339                 // There is no need to enable/disable subtypes of disabled IMEs.
340                 continue;
341             }
342             final HashSet<String> enabledSubtypesSet = enabledSubtypes.get(id);
343             final int subtypeCount = imi.getSubtypeCount();
344             for (int i = 0; i < subtypeCount; ++i) {
345                 final InputMethodSubtype subtype = imi.getSubtypeAt(i);
346                 final String hashCode = String.valueOf(subtype.hashCode());
347                 if (DEBUG) {
348                     Log.d(TAG, "--- Set checked state: " + "id" + ", " + hashCode + ", "
349                             + enabledSubtypesSet.contains(hashCode));
350                 }
351                 final TwoStatePreference pref = (TwoStatePreference) preferenceScreen
352                         .findPreference(id + hashCode);
353                 if (pref != null) {
354                     pref.setChecked(enabledSubtypesSet.contains(hashCode));
355                 }
356             }
357         }
358     }
359 
removeUnnecessaryNonPersistentPreference(final Preference pref)360     public static void removeUnnecessaryNonPersistentPreference(final Preference pref) {
361         final String key = pref.getKey();
362         if (pref.isPersistent() || key == null) {
363             return;
364         }
365         final SharedPreferences prefs = pref.getSharedPreferences();
366         if (prefs != null && prefs.contains(key)) {
367             prefs.edit().remove(key).apply();
368         }
369     }
370 
371     @NonNull
getSubtypeLocaleNameAsSentence(@ullable InputMethodSubtype subtype, @NonNull final Context context, @NonNull final InputMethodInfo inputMethodInfo)372     public static String getSubtypeLocaleNameAsSentence(@Nullable InputMethodSubtype subtype,
373             @NonNull final Context context, @NonNull final InputMethodInfo inputMethodInfo) {
374         if (subtype == null) {
375             return "";
376         }
377         final Locale locale = getDisplayLocale(context);
378         final CharSequence subtypeName = subtype.getDisplayName(context,
379                 inputMethodInfo.getPackageName(), inputMethodInfo.getServiceInfo()
380                         .applicationInfo);
381         return LocaleHelper.toSentenceCase(subtypeName.toString(), locale);
382     }
383 
384     @NonNull
getSubtypeLocaleNameListAsSentence( @onNull final List<InputMethodSubtype> subtypes, @NonNull final Context context, @NonNull final InputMethodInfo inputMethodInfo)385     public static String getSubtypeLocaleNameListAsSentence(
386             @NonNull final List<InputMethodSubtype> subtypes, @NonNull final Context context,
387             @NonNull final InputMethodInfo inputMethodInfo) {
388         if (subtypes.isEmpty()) {
389             return "";
390         }
391         final Locale locale = getDisplayLocale(context);
392         final int subtypeCount = subtypes.size();
393         final CharSequence[] subtypeNames = new CharSequence[subtypeCount];
394         for (int i = 0; i < subtypeCount; i++) {
395             subtypeNames[i] = subtypes.get(i).getDisplayName(context,
396                     inputMethodInfo.getPackageName(), inputMethodInfo.getServiceInfo()
397                             .applicationInfo);
398         }
399         return LocaleHelper.toSentenceCase(
400                 ListFormatter.getInstance(locale).format((Object[]) subtypeNames), locale);
401     }
402 
403     @NonNull
getDisplayLocale(@ullable final Context context)404     private static Locale getDisplayLocale(@Nullable final Context context) {
405         if (context == null) {
406             return Locale.getDefault();
407         }
408         if (context.getResources() == null) {
409             return Locale.getDefault();
410         }
411         final Configuration configuration = context.getResources().getConfiguration();
412         if (configuration == null) {
413             return Locale.getDefault();
414         }
415         final Locale configurationLocale = configuration.getLocales().get(0);
416         if (configurationLocale == null) {
417             return Locale.getDefault();
418         }
419         return configurationLocale;
420     }
421 
isValidNonAuxAsciiCapableIme(InputMethodInfo imi)422     public static boolean isValidNonAuxAsciiCapableIme(InputMethodInfo imi) {
423         if (imi.isAuxiliaryIme()) {
424             return false;
425         }
426         final int subtypeCount = imi.getSubtypeCount();
427         for (int i = 0; i < subtypeCount; ++i) {
428             final InputMethodSubtype subtype = imi.getSubtypeAt(i);
429             if (SUBTYPE_MODE_KEYBOARD.equalsIgnoreCase(subtype.getMode())
430                     && subtype.isAsciiCapable()) {
431                 return true;
432             }
433         }
434         return false;
435     }
436 }
437