/* * Copyright (C) 2013 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.internal.inputmethod; import android.app.AppOpsManager; import android.content.ContentResolver; import android.content.Context; import android.content.pm.ApplicationInfo; import android.content.pm.PackageManager; import android.content.pm.PackageManager.NameNotFoundException; import android.content.res.Resources; import android.provider.Settings; import android.provider.Settings.SettingNotFoundException; import android.text.TextUtils; import android.util.Pair; import android.util.Slog; import android.view.inputmethod.InputMethodInfo; import android.view.inputmethod.InputMethodSubtype; import android.view.textservice.SpellCheckerInfo; import android.view.textservice.TextServicesManager; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Locale; /** * InputMethodManagerUtils contains some static methods that provides IME informations. * This methods are supposed to be used in both the framework and the Settings application. */ public class InputMethodUtils { public static final boolean DEBUG = false; public static final int NOT_A_SUBTYPE_ID = -1; public static final String SUBTYPE_MODE_ANY = null; public static final String SUBTYPE_MODE_KEYBOARD = "keyboard"; public static final String SUBTYPE_MODE_VOICE = "voice"; private static final String TAG = "InputMethodUtils"; private static final Locale ENGLISH_LOCALE = new Locale("en"); private static final String NOT_A_SUBTYPE_ID_STR = String.valueOf(NOT_A_SUBTYPE_ID); private static final String TAG_ENABLED_WHEN_DEFAULT_IS_NOT_ASCII_CAPABLE = "EnabledWhenDefaultIsNotAsciiCapable"; private static final String TAG_ASCII_CAPABLE = "AsciiCapable"; /** * Used in {@link #getFallbackLocaleForDefaultIme(ArrayList, Context)} to find the fallback IMEs * that are mainly used until the system becomes ready. Note that {@link Locale} in this array * is checked with {@link Locale#equals(Object)}, which means that {@code Locale.ENGLISH} * doesn't automatically match {@code Locale("en", "IN")}. */ private static final Locale[] SEARCH_ORDER_OF_FALLBACK_LOCALES = { Locale.ENGLISH, // "en" Locale.US, // "en_US" Locale.UK, // "en_GB" }; private InputMethodUtils() { // This utility class is not publicly instantiable. } // ---------------------------------------------------------------------- // Utilities for debug public static String getStackTrace() { final StringBuilder sb = new StringBuilder(); try { throw new RuntimeException(); } catch (RuntimeException e) { final StackTraceElement[] frames = e.getStackTrace(); // Start at 1 because the first frame is here and we don't care about it for (int j = 1; j < frames.length; ++j) { sb.append(frames[j].toString() + "\n"); } } return sb.toString(); } public static String getApiCallStack() { String apiCallStack = ""; try { throw new RuntimeException(); } catch (RuntimeException e) { final StackTraceElement[] frames = e.getStackTrace(); for (int j = 1; j < frames.length; ++j) { final String tempCallStack = frames[j].toString(); if (TextUtils.isEmpty(apiCallStack)) { // Overwrite apiCallStack if it's empty apiCallStack = tempCallStack; } else if (tempCallStack.indexOf("Transact(") < 0) { // Overwrite apiCallStack if it's not a binder call apiCallStack = tempCallStack; } else { break; } } } return apiCallStack; } // ---------------------------------------------------------------------- public static boolean isSystemIme(InputMethodInfo inputMethod) { return (inputMethod.getServiceInfo().applicationInfo.flags & ApplicationInfo.FLAG_SYSTEM) != 0; } /** * @deprecated Use {@link Locale} returned from * {@link #getFallbackLocaleForDefaultIme(ArrayList)} instead. */ @Deprecated public static boolean isSystemImeThatHasEnglishKeyboardSubtype(InputMethodInfo imi) { if (!isSystemIme(imi)) { return false; } return containsSubtypeOf(imi, ENGLISH_LOCALE.getLanguage(), SUBTYPE_MODE_KEYBOARD); } public static Locale getFallbackLocaleForDefaultIme(final ArrayList imis, final Context context) { for (final Locale fallbackLocale : SEARCH_ORDER_OF_FALLBACK_LOCALES) { for (int i = 0; i < imis.size(); ++i) { final InputMethodInfo imi = imis.get(i); if (isSystemIme(imi) && imi.isDefault(context) && containsSubtypeOf(imi, fallbackLocale, false /* ignoreCountry */, SUBTYPE_MODE_KEYBOARD)) { return fallbackLocale; } } } return null; } private static boolean isSystemAuxilialyImeThatHasAutomaticSubtype(InputMethodInfo imi) { if (!isSystemIme(imi)) { return false; } if (!imi.isAuxiliaryIme()) { return false; } final int subtypeCount = imi.getSubtypeCount(); for (int i = 0; i < subtypeCount; ++i) { final InputMethodSubtype s = imi.getSubtypeAt(i); if (s.overridesImplicitlyEnabledSubtype()) { return true; } } return false; } public static Locale getSystemLocaleFromContext(final Context context) { try { return context.getResources().getConfiguration().locale; } catch (Resources.NotFoundException ex) { return null; } } public static ArrayList getDefaultEnabledImes( Context context, boolean isSystemReady, ArrayList imis) { // OK to store null in fallbackLocale because isImeThatHasSubtypeOf() is null-tolerant. final Locale fallbackLocale = getFallbackLocaleForDefaultIme(imis, context); if (!isSystemReady) { final ArrayList retval = new ArrayList<>(); for (int i = 0; i < imis.size(); ++i) { final InputMethodInfo imi = imis.get(i); // TODO: We should check isAsciiCapable instead of relying on fallbackLocale. if (isSystemIme(imi) && imi.isDefault(context) && isImeThatHasSubtypeOf(imi, fallbackLocale, false /* ignoreCountry */, SUBTYPE_MODE_KEYBOARD)) { retval.add(imi); } } return retval; } // OK to store null in fallbackLocale because isImeThatHasSubtypeOf() is null-tolerant. final Locale systemLocale = getSystemLocaleFromContext(context); // TODO: Use LinkedHashSet to simplify the code. final ArrayList retval = new ArrayList<>(); boolean systemLocaleKeyboardImeFound = false; // First, try to find IMEs with taking the system locale country into consideration. for (int i = 0; i < imis.size(); ++i) { final InputMethodInfo imi = imis.get(i); if (!isSystemIme(imi) || !imi.isDefault(context)) { continue; } final boolean isSystemLocaleKeyboardIme = isImeThatHasSubtypeOf(imi, systemLocale, false /* ignoreCountry */, SUBTYPE_MODE_KEYBOARD); // TODO: We should check isAsciiCapable instead of relying on fallbackLocale. // TODO: Use LinkedHashSet to simplify the code. if (isSystemLocaleKeyboardIme || isImeThatHasSubtypeOf(imi, fallbackLocale, false /* ignoreCountry */, SUBTYPE_MODE_ANY)) { retval.add(imi); } systemLocaleKeyboardImeFound |= isSystemLocaleKeyboardIme; } // System locale country doesn't match any IMEs, try to find IMEs in a country-agnostic // way. if (!systemLocaleKeyboardImeFound) { for (int i = 0; i < imis.size(); ++i) { final InputMethodInfo imi = imis.get(i); if (!isSystemIme(imi) || !imi.isDefault(context)) { continue; } if (isImeThatHasSubtypeOf(imi, fallbackLocale, false /* ignoreCountry */, SUBTYPE_MODE_KEYBOARD)) { // IMEs that have fallback locale are already added in the previous loop. We // don't need to add them again here. // TODO: Use LinkedHashSet to simplify the code. continue; } if (isImeThatHasSubtypeOf(imi, systemLocale, true /* ignoreCountry */, SUBTYPE_MODE_ANY)) { retval.add(imi); } } } // If one or more auxiliary input methods are available, OK to stop populating the list. for (int i = 0; i < retval.size(); ++i) { if (retval.get(i).isAuxiliaryIme()) { return retval; } } for (int i = 0; i < imis.size(); ++i) { final InputMethodInfo imi = imis.get(i); if (isSystemAuxilialyImeThatHasAutomaticSubtype(imi)) { retval.add(imi); } } return retval; } public static boolean isImeThatHasSubtypeOf(final InputMethodInfo imi, final Locale locale, final boolean ignoreCountry, final String mode) { if (locale == null) { return false; } return containsSubtypeOf(imi, locale, ignoreCountry, mode); } /** * @deprecated Use {@link #isSystemIme(InputMethodInfo)} and * {@link InputMethodInfo#isDefault(Context)} and * {@link #isImeThatHasSubtypeOf(InputMethodInfo, Locale, boolean, String))} instead. */ @Deprecated public static boolean isValidSystemDefaultIme( boolean isSystemReady, InputMethodInfo imi, Context context) { if (!isSystemReady) { return false; } if (!isSystemIme(imi)) { return false; } if (imi.getIsDefaultResourceId() != 0) { try { if (imi.isDefault(context) && containsSubtypeOf( imi, context.getResources().getConfiguration().locale.getLanguage(), SUBTYPE_MODE_ANY)) { return true; } } catch (Resources.NotFoundException ex) { } } if (imi.getSubtypeCount() == 0) { Slog.w(TAG, "Found no subtypes in a system IME: " + imi.getPackageName()); } return false; } public static boolean containsSubtypeOf(final InputMethodInfo imi, final Locale locale, final boolean ignoreCountry, final String mode) { final int N = imi.getSubtypeCount(); for (int i = 0; i < N; ++i) { final InputMethodSubtype subtype = imi.getSubtypeAt(i); if (ignoreCountry) { final Locale subtypeLocale = new Locale(getLanguageFromLocaleString( subtype.getLocale())); if (!subtypeLocale.getLanguage().equals(locale.getLanguage())) { continue; } } else { // TODO: Use {@link Locale#toLanguageTag()} and // {@link Locale#forLanguageTag(languageTag)} instead. if (!TextUtils.equals(subtype.getLocale(), locale.toString())) { continue; } } if (mode == SUBTYPE_MODE_ANY || TextUtils.isEmpty(mode) || mode.equalsIgnoreCase(subtype.getMode())) { return true; } } return false; } /** * @deprecated Use {@link #containsSubtypeOf(InputMethodInfo, Locale, boolean, String)} instead. */ @Deprecated public static boolean containsSubtypeOf(InputMethodInfo imi, String language, String mode) { final int N = imi.getSubtypeCount(); for (int i = 0; i < N; ++i) { final InputMethodSubtype subtype = imi.getSubtypeAt(i); if (!subtype.getLocale().startsWith(language)) { continue; } if (mode == SUBTYPE_MODE_ANY || TextUtils.isEmpty(mode) || mode.equalsIgnoreCase(subtype.getMode())) { return true; } } return false; } public static ArrayList getSubtypes(InputMethodInfo imi) { ArrayList subtypes = new ArrayList(); final int subtypeCount = imi.getSubtypeCount(); for (int i = 0; i < subtypeCount; ++i) { subtypes.add(imi.getSubtypeAt(i)); } return subtypes; } public static ArrayList getOverridingImplicitlyEnabledSubtypes( InputMethodInfo imi, String mode) { ArrayList subtypes = new ArrayList(); final int subtypeCount = imi.getSubtypeCount(); for (int i = 0; i < subtypeCount; ++i) { final InputMethodSubtype subtype = imi.getSubtypeAt(i); if (subtype.overridesImplicitlyEnabledSubtype() && subtype.getMode().equals(mode)) { subtypes.add(subtype); } } return subtypes; } public static InputMethodInfo getMostApplicableDefaultIME(List enabledImes) { if (enabledImes == null || enabledImes.isEmpty()) { return null; } // We'd prefer to fall back on a system IME, since that is safer. int i = enabledImes.size(); int firstFoundSystemIme = -1; while (i > 0) { i--; final InputMethodInfo imi = enabledImes.get(i); if (InputMethodUtils.isSystemImeThatHasEnglishKeyboardSubtype(imi) && !imi.isAuxiliaryIme()) { return imi; } if (firstFoundSystemIme < 0 && InputMethodUtils.isSystemIme(imi) && !imi.isAuxiliaryIme()) { firstFoundSystemIme = i; } } return enabledImes.get(Math.max(firstFoundSystemIme, 0)); } public static boolean isValidSubtypeId(InputMethodInfo imi, int subtypeHashCode) { return getSubtypeIdFromHashCode(imi, subtypeHashCode) != NOT_A_SUBTYPE_ID; } public static int getSubtypeIdFromHashCode(InputMethodInfo imi, int subtypeHashCode) { if (imi != null) { final int subtypeCount = imi.getSubtypeCount(); for (int i = 0; i < subtypeCount; ++i) { InputMethodSubtype ims = imi.getSubtypeAt(i); if (subtypeHashCode == ims.hashCode()) { return i; } } } return NOT_A_SUBTYPE_ID; } private static ArrayList getImplicitlyApplicableSubtypesLocked( Resources res, InputMethodInfo imi) { final List subtypes = InputMethodUtils.getSubtypes(imi); final String systemLocale = res.getConfiguration().locale.toString(); if (TextUtils.isEmpty(systemLocale)) return new ArrayList(); final String systemLanguage = res.getConfiguration().locale.getLanguage(); final HashMap applicableModeAndSubtypesMap = new HashMap(); final int N = subtypes.size(); for (int i = 0; i < N; ++i) { // scan overriding implicitly enabled subtypes. InputMethodSubtype subtype = subtypes.get(i); if (subtype.overridesImplicitlyEnabledSubtype()) { final String mode = subtype.getMode(); if (!applicableModeAndSubtypesMap.containsKey(mode)) { applicableModeAndSubtypesMap.put(mode, subtype); } } } if (applicableModeAndSubtypesMap.size() > 0) { return new ArrayList(applicableModeAndSubtypesMap.values()); } for (int i = 0; i < N; ++i) { final InputMethodSubtype subtype = subtypes.get(i); final String locale = subtype.getLocale(); final String mode = subtype.getMode(); final String language = getLanguageFromLocaleString(locale); // When system locale starts with subtype's locale, that subtype will be applicable // for system locale. We need to make sure the languages are the same, to prevent // locales like "fil" (Filipino) being matched by "fi" (Finnish). // // For instance, it's clearly applicable for cases like system locale = en_US and // subtype = en, but it is not necessarily considered applicable for cases like system // locale = en and subtype = en_US. // // We just call systemLocale.startsWith(locale) in this function because there is no // need to find applicable subtypes aggressively unlike // findLastResortApplicableSubtypeLocked. // // TODO: This check is broken. It won't take scripts into account and doesn't // account for the mandatory conversions performed by Locale#toString. if (language.equals(systemLanguage) && systemLocale.startsWith(locale)) { final InputMethodSubtype applicableSubtype = applicableModeAndSubtypesMap.get(mode); // If more applicable subtypes are contained, skip. if (applicableSubtype != null) { if (systemLocale.equals(applicableSubtype.getLocale())) continue; if (!systemLocale.equals(locale)) continue; } applicableModeAndSubtypesMap.put(mode, subtype); } } final InputMethodSubtype keyboardSubtype = applicableModeAndSubtypesMap.get(SUBTYPE_MODE_KEYBOARD); final ArrayList applicableSubtypes = new ArrayList( applicableModeAndSubtypesMap.values()); if (keyboardSubtype != null && !keyboardSubtype.containsExtraValueKey(TAG_ASCII_CAPABLE)) { for (int i = 0; i < N; ++i) { final InputMethodSubtype subtype = subtypes.get(i); final String mode = subtype.getMode(); if (SUBTYPE_MODE_KEYBOARD.equals(mode) && subtype.containsExtraValueKey( TAG_ENABLED_WHEN_DEFAULT_IS_NOT_ASCII_CAPABLE)) { applicableSubtypes.add(subtype); } } } if (keyboardSubtype == null) { InputMethodSubtype lastResortKeyboardSubtype = findLastResortApplicableSubtypeLocked( res, subtypes, SUBTYPE_MODE_KEYBOARD, systemLocale, true); if (lastResortKeyboardSubtype != null) { applicableSubtypes.add(lastResortKeyboardSubtype); } } return applicableSubtypes; } private static List getEnabledInputMethodSubtypeList( Context context, InputMethodInfo imi, List enabledSubtypes, boolean allowsImplicitlySelectedSubtypes) { if (allowsImplicitlySelectedSubtypes && enabledSubtypes.isEmpty()) { enabledSubtypes = InputMethodUtils.getImplicitlyApplicableSubtypesLocked( context.getResources(), imi); } return InputMethodSubtype.sort(context, 0, imi, enabledSubtypes); } /** * Returns the language component of a given locale string. * TODO: Use {@link Locale#toLanguageTag()} and {@link Locale#forLanguageTag(languageTag)} */ public static String getLanguageFromLocaleString(String locale) { final int idx = locale.indexOf('_'); if (idx < 0) { return locale; } else { return locale.substring(0, idx); } } /** * If there are no selected subtypes, tries finding the most applicable one according to the * given locale. * @param subtypes this function will search the most applicable subtype in subtypes * @param mode subtypes will be filtered by mode * @param locale subtypes will be filtered by locale * @param canIgnoreLocaleAsLastResort if this function can't find the most applicable subtype, * it will return the first subtype matched with mode * @return the most applicable subtypeId */ public static InputMethodSubtype findLastResortApplicableSubtypeLocked( Resources res, List subtypes, String mode, String locale, boolean canIgnoreLocaleAsLastResort) { if (subtypes == null || subtypes.size() == 0) { return null; } if (TextUtils.isEmpty(locale)) { locale = res.getConfiguration().locale.toString(); } final String language = getLanguageFromLocaleString(locale); boolean partialMatchFound = false; InputMethodSubtype applicableSubtype = null; InputMethodSubtype firstMatchedModeSubtype = null; final int N = subtypes.size(); for (int i = 0; i < N; ++i) { InputMethodSubtype subtype = subtypes.get(i); final String subtypeLocale = subtype.getLocale(); final String subtypeLanguage = getLanguageFromLocaleString(subtypeLocale); // An applicable subtype should match "mode". If mode is null, mode will be ignored, // and all subtypes with all modes can be candidates. if (mode == null || subtypes.get(i).getMode().equalsIgnoreCase(mode)) { if (firstMatchedModeSubtype == null) { firstMatchedModeSubtype = subtype; } if (locale.equals(subtypeLocale)) { // Exact match (e.g. system locale is "en_US" and subtype locale is "en_US") applicableSubtype = subtype; break; } else if (!partialMatchFound && language.equals(subtypeLanguage)) { // Partial match (e.g. system locale is "en_US" and subtype locale is "en") applicableSubtype = subtype; partialMatchFound = true; } } } if (applicableSubtype == null && canIgnoreLocaleAsLastResort) { return firstMatchedModeSubtype; } // The first subtype applicable to the system locale will be defined as the most applicable // subtype. if (DEBUG) { if (applicableSubtype != null) { Slog.d(TAG, "Applicable InputMethodSubtype was found: " + applicableSubtype.getMode() + "," + applicableSubtype.getLocale()); } } return applicableSubtype; } public static boolean canAddToLastInputMethod(InputMethodSubtype subtype) { if (subtype == null) return true; return !subtype.isAuxiliary(); } public static void setNonSelectedSystemImesDisabledUntilUsed( PackageManager packageManager, List enabledImis) { if (DEBUG) { Slog.d(TAG, "setNonSelectedSystemImesDisabledUntilUsed"); } final String[] systemImesDisabledUntilUsed = Resources.getSystem().getStringArray( com.android.internal.R.array.config_disabledUntilUsedPreinstalledImes); if (systemImesDisabledUntilUsed == null || systemImesDisabledUntilUsed.length == 0) { return; } // Only the current spell checker should be treated as an enabled one. final SpellCheckerInfo currentSpellChecker = TextServicesManager.getInstance().getCurrentSpellChecker(); for (final String packageName : systemImesDisabledUntilUsed) { if (DEBUG) { Slog.d(TAG, "check " + packageName); } boolean enabledIme = false; for (int j = 0; j < enabledImis.size(); ++j) { final InputMethodInfo imi = enabledImis.get(j); if (packageName.equals(imi.getPackageName())) { enabledIme = true; break; } } if (enabledIme) { // enabled ime. skip continue; } if (currentSpellChecker != null && packageName.equals(currentSpellChecker.getPackageName())) { // enabled spell checker. skip if (DEBUG) { Slog.d(TAG, packageName + " is the current spell checker. skip"); } continue; } ApplicationInfo ai = null; try { ai = packageManager.getApplicationInfo(packageName, PackageManager.GET_DISABLED_UNTIL_USED_COMPONENTS); } catch (NameNotFoundException e) { Slog.w(TAG, "NameNotFoundException: " + packageName, e); } if (ai == null) { // No app found for packageName continue; } final boolean isSystemPackage = (ai.flags & ApplicationInfo.FLAG_SYSTEM) != 0; if (!isSystemPackage) { continue; } setDisabledUntilUsed(packageManager, packageName); } } private static void setDisabledUntilUsed(PackageManager packageManager, String packageName) { final int state = packageManager.getApplicationEnabledSetting(packageName); if (state == PackageManager.COMPONENT_ENABLED_STATE_DEFAULT || state == PackageManager.COMPONENT_ENABLED_STATE_ENABLED) { if (DEBUG) { Slog.d(TAG, "Update state(" + packageName + "): DISABLED_UNTIL_USED"); } packageManager.setApplicationEnabledSetting(packageName, PackageManager.COMPONENT_ENABLED_STATE_DISABLED_UNTIL_USED, 0); } else { if (DEBUG) { Slog.d(TAG, packageName + " is already DISABLED_UNTIL_USED"); } } } public static CharSequence getImeAndSubtypeDisplayName(Context context, InputMethodInfo imi, InputMethodSubtype subtype) { final CharSequence imiLabel = imi.loadLabel(context.getPackageManager()); return subtype != null ? TextUtils.concat(subtype.getDisplayName(context, imi.getPackageName(), imi.getServiceInfo().applicationInfo), (TextUtils.isEmpty(imiLabel) ? "" : " - " + imiLabel)) : imiLabel; } /** * Returns true if a package name belongs to a UID. * *

This is a simple wrapper of {@link AppOpsManager#checkPackage(int, String)}.

* @param appOpsManager the {@link AppOpsManager} object to be used for the validation. * @param uid the UID to be validated. * @param packageName the package name. * @return {@code true} if the package name belongs to the UID. */ public static boolean checkIfPackageBelongsToUid(final AppOpsManager appOpsManager, final int uid, final String packageName) { try { appOpsManager.checkPackage(uid, packageName); return true; } catch (SecurityException e) { return false; } } /** * Utility class for putting and getting settings for InputMethod * TODO: Move all putters and getters of settings to this class. */ public static class InputMethodSettings { // The string for enabled input method is saved as follows: // example: ("ime0;subtype0;subtype1;subtype2:ime1:ime2;subtype0") private static final char INPUT_METHOD_SEPARATER = ':'; private static final char INPUT_METHOD_SUBTYPE_SEPARATER = ';'; private final TextUtils.SimpleStringSplitter mInputMethodSplitter = new TextUtils.SimpleStringSplitter(INPUT_METHOD_SEPARATER); private final TextUtils.SimpleStringSplitter mSubtypeSplitter = new TextUtils.SimpleStringSplitter(INPUT_METHOD_SUBTYPE_SEPARATER); private final Resources mRes; private final ContentResolver mResolver; private final HashMap mMethodMap; private final ArrayList mMethodList; private String mEnabledInputMethodsStrCache; private int mCurrentUserId; private int[] mCurrentProfileIds = new int[0]; private static void buildEnabledInputMethodsSettingString( StringBuilder builder, Pair> pair) { String id = pair.first; ArrayList subtypes = pair.second; builder.append(id); // Inputmethod and subtypes are saved in the settings as follows: // ime0;subtype0;subtype1:ime1;subtype0:ime2:ime3;subtype0;subtype1 for (String subtypeId: subtypes) { builder.append(INPUT_METHOD_SUBTYPE_SEPARATER).append(subtypeId); } } public InputMethodSettings( Resources res, ContentResolver resolver, HashMap methodMap, ArrayList methodList, int userId) { setCurrentUserId(userId); mRes = res; mResolver = resolver; mMethodMap = methodMap; mMethodList = methodList; } public void setCurrentUserId(int userId) { if (DEBUG) { Slog.d(TAG, "--- Swtich the current user from " + mCurrentUserId + " to " + userId); } // IMMS settings are kept per user, so keep track of current user mCurrentUserId = userId; } public void setCurrentProfileIds(int[] currentProfileIds) { synchronized (this) { mCurrentProfileIds = currentProfileIds; } } public boolean isCurrentProfile(int userId) { synchronized (this) { if (userId == mCurrentUserId) return true; for (int i = 0; i < mCurrentProfileIds.length; i++) { if (userId == mCurrentProfileIds[i]) return true; } return false; } } public List getEnabledInputMethodListLocked() { return createEnabledInputMethodListLocked( getEnabledInputMethodsAndSubtypeListLocked()); } public List>> getEnabledInputMethodAndSubtypeHashCodeListLocked() { return createEnabledInputMethodAndSubtypeHashCodeListLocked( getEnabledInputMethodsAndSubtypeListLocked()); } public List getEnabledInputMethodSubtypeListLocked( Context context, InputMethodInfo imi, boolean allowsImplicitlySelectedSubtypes) { List enabledSubtypes = getEnabledInputMethodSubtypeListLocked(imi); if (allowsImplicitlySelectedSubtypes && enabledSubtypes.isEmpty()) { enabledSubtypes = InputMethodUtils.getImplicitlyApplicableSubtypesLocked( context.getResources(), imi); } return InputMethodSubtype.sort(context, 0, imi, enabledSubtypes); } public List getEnabledInputMethodSubtypeListLocked( InputMethodInfo imi) { List>> imsList = getEnabledInputMethodsAndSubtypeListLocked(); ArrayList enabledSubtypes = new ArrayList(); if (imi != null) { for (Pair> imsPair : imsList) { InputMethodInfo info = mMethodMap.get(imsPair.first); if (info != null && info.getId().equals(imi.getId())) { final int subtypeCount = info.getSubtypeCount(); for (int i = 0; i < subtypeCount; ++i) { InputMethodSubtype ims = info.getSubtypeAt(i); for (String s: imsPair.second) { if (String.valueOf(ims.hashCode()).equals(s)) { enabledSubtypes.add(ims); } } } break; } } } return enabledSubtypes; } // At the initial boot, the settings for input methods are not set, // so we need to enable IME in that case. public void enableAllIMEsIfThereIsNoEnabledIME() { if (TextUtils.isEmpty(getEnabledInputMethodsStr())) { StringBuilder sb = new StringBuilder(); final int N = mMethodList.size(); for (int i = 0; i < N; i++) { InputMethodInfo imi = mMethodList.get(i); Slog.i(TAG, "Adding: " + imi.getId()); if (i > 0) sb.append(':'); sb.append(imi.getId()); } putEnabledInputMethodsStr(sb.toString()); } } public List>> getEnabledInputMethodsAndSubtypeListLocked() { ArrayList>> imsList = new ArrayList>>(); final String enabledInputMethodsStr = getEnabledInputMethodsStr(); if (TextUtils.isEmpty(enabledInputMethodsStr)) { return imsList; } mInputMethodSplitter.setString(enabledInputMethodsStr); while (mInputMethodSplitter.hasNext()) { String nextImsStr = mInputMethodSplitter.next(); mSubtypeSplitter.setString(nextImsStr); if (mSubtypeSplitter.hasNext()) { ArrayList subtypeHashes = new ArrayList(); // The first element is ime id. String imeId = mSubtypeSplitter.next(); while (mSubtypeSplitter.hasNext()) { subtypeHashes.add(mSubtypeSplitter.next()); } imsList.add(new Pair>(imeId, subtypeHashes)); } } return imsList; } public void appendAndPutEnabledInputMethodLocked(String id, boolean reloadInputMethodStr) { if (reloadInputMethodStr) { getEnabledInputMethodsStr(); } if (TextUtils.isEmpty(mEnabledInputMethodsStrCache)) { // Add in the newly enabled input method. putEnabledInputMethodsStr(id); } else { putEnabledInputMethodsStr( mEnabledInputMethodsStrCache + INPUT_METHOD_SEPARATER + id); } } /** * Build and put a string of EnabledInputMethods with removing specified Id. * @return the specified id was removed or not. */ public boolean buildAndPutEnabledInputMethodsStrRemovingIdLocked( StringBuilder builder, List>> imsList, String id) { boolean isRemoved = false; boolean needsAppendSeparator = false; for (Pair> ims: imsList) { String curId = ims.first; if (curId.equals(id)) { // We are disabling this input method, and it is // currently enabled. Skip it to remove from the // new list. isRemoved = true; } else { if (needsAppendSeparator) { builder.append(INPUT_METHOD_SEPARATER); } else { needsAppendSeparator = true; } buildEnabledInputMethodsSettingString(builder, ims); } } if (isRemoved) { // Update the setting with the new list of input methods. putEnabledInputMethodsStr(builder.toString()); } return isRemoved; } private List createEnabledInputMethodListLocked( List>> imsList) { final ArrayList res = new ArrayList(); for (Pair> ims: imsList) { InputMethodInfo info = mMethodMap.get(ims.first); if (info != null) { res.add(info); } } return res; } private List>> createEnabledInputMethodAndSubtypeHashCodeListLocked( List>> imsList) { final ArrayList>> res = new ArrayList>>(); for (Pair> ims : imsList) { InputMethodInfo info = mMethodMap.get(ims.first); if (info != null) { res.add(new Pair>(info, ims.second)); } } return res; } private void putEnabledInputMethodsStr(String str) { Settings.Secure.putStringForUser( mResolver, Settings.Secure.ENABLED_INPUT_METHODS, str, mCurrentUserId); mEnabledInputMethodsStrCache = str; if (DEBUG) { Slog.d(TAG, "putEnabledInputMethodStr: " + str); } } public String getEnabledInputMethodsStr() { mEnabledInputMethodsStrCache = Settings.Secure.getStringForUser( mResolver, Settings.Secure.ENABLED_INPUT_METHODS, mCurrentUserId); if (DEBUG) { Slog.d(TAG, "getEnabledInputMethodsStr: " + mEnabledInputMethodsStrCache + ", " + mCurrentUserId); } return mEnabledInputMethodsStrCache; } private void saveSubtypeHistory( List> savedImes, String newImeId, String newSubtypeId) { StringBuilder builder = new StringBuilder(); boolean isImeAdded = false; if (!TextUtils.isEmpty(newImeId) && !TextUtils.isEmpty(newSubtypeId)) { builder.append(newImeId).append(INPUT_METHOD_SUBTYPE_SEPARATER).append( newSubtypeId); isImeAdded = true; } for (Pair ime: savedImes) { String imeId = ime.first; String subtypeId = ime.second; if (TextUtils.isEmpty(subtypeId)) { subtypeId = NOT_A_SUBTYPE_ID_STR; } if (isImeAdded) { builder.append(INPUT_METHOD_SEPARATER); } else { isImeAdded = true; } builder.append(imeId).append(INPUT_METHOD_SUBTYPE_SEPARATER).append( subtypeId); } // Remove the last INPUT_METHOD_SEPARATER putSubtypeHistoryStr(builder.toString()); } private void addSubtypeToHistory(String imeId, String subtypeId) { List> subtypeHistory = loadInputMethodAndSubtypeHistoryLocked(); for (Pair ime: subtypeHistory) { if (ime.first.equals(imeId)) { if (DEBUG) { Slog.v(TAG, "Subtype found in the history: " + imeId + ", " + ime.second); } // We should break here subtypeHistory.remove(ime); break; } } if (DEBUG) { Slog.v(TAG, "Add subtype to the history: " + imeId + ", " + subtypeId); } saveSubtypeHistory(subtypeHistory, imeId, subtypeId); } private void putSubtypeHistoryStr(String str) { if (DEBUG) { Slog.d(TAG, "putSubtypeHistoryStr: " + str); } Settings.Secure.putStringForUser( mResolver, Settings.Secure.INPUT_METHODS_SUBTYPE_HISTORY, str, mCurrentUserId); } public Pair getLastInputMethodAndSubtypeLocked() { // Gets the first one from the history return getLastSubtypeForInputMethodLockedInternal(null); } public String getLastSubtypeForInputMethodLocked(String imeId) { Pair ime = getLastSubtypeForInputMethodLockedInternal(imeId); if (ime != null) { return ime.second; } else { return null; } } private Pair getLastSubtypeForInputMethodLockedInternal(String imeId) { List>> enabledImes = getEnabledInputMethodsAndSubtypeListLocked(); List> subtypeHistory = loadInputMethodAndSubtypeHistoryLocked(); for (Pair imeAndSubtype : subtypeHistory) { final String imeInTheHistory = imeAndSubtype.first; // If imeId is empty, returns the first IME and subtype in the history if (TextUtils.isEmpty(imeId) || imeInTheHistory.equals(imeId)) { final String subtypeInTheHistory = imeAndSubtype.second; final String subtypeHashCode = getEnabledSubtypeHashCodeForInputMethodAndSubtypeLocked( enabledImes, imeInTheHistory, subtypeInTheHistory); if (!TextUtils.isEmpty(subtypeHashCode)) { if (DEBUG) { Slog.d(TAG, "Enabled subtype found in the history: " + subtypeHashCode); } return new Pair(imeInTheHistory, subtypeHashCode); } } } if (DEBUG) { Slog.d(TAG, "No enabled IME found in the history"); } return null; } private String getEnabledSubtypeHashCodeForInputMethodAndSubtypeLocked(List>> enabledImes, String imeId, String subtypeHashCode) { for (Pair> enabledIme: enabledImes) { if (enabledIme.first.equals(imeId)) { final ArrayList explicitlyEnabledSubtypes = enabledIme.second; final InputMethodInfo imi = mMethodMap.get(imeId); if (explicitlyEnabledSubtypes.size() == 0) { // If there are no explicitly enabled subtypes, applicable subtypes are // enabled implicitly. // If IME is enabled and no subtypes are enabled, applicable subtypes // are enabled implicitly, so needs to treat them to be enabled. if (imi != null && imi.getSubtypeCount() > 0) { List implicitlySelectedSubtypes = getImplicitlyApplicableSubtypesLocked(mRes, imi); if (implicitlySelectedSubtypes != null) { final int N = implicitlySelectedSubtypes.size(); for (int i = 0; i < N; ++i) { final InputMethodSubtype st = implicitlySelectedSubtypes.get(i); if (String.valueOf(st.hashCode()).equals(subtypeHashCode)) { return subtypeHashCode; } } } } } else { for (String s: explicitlyEnabledSubtypes) { if (s.equals(subtypeHashCode)) { // If both imeId and subtypeId are enabled, return subtypeId. try { final int hashCode = Integer.valueOf(subtypeHashCode); // Check whether the subtype id is valid or not if (isValidSubtypeId(imi, hashCode)) { return s; } else { return NOT_A_SUBTYPE_ID_STR; } } catch (NumberFormatException e) { return NOT_A_SUBTYPE_ID_STR; } } } } // If imeId was enabled but subtypeId was disabled. return NOT_A_SUBTYPE_ID_STR; } } // If both imeId and subtypeId are disabled, return null return null; } private List> loadInputMethodAndSubtypeHistoryLocked() { ArrayList> imsList = new ArrayList>(); final String subtypeHistoryStr = getSubtypeHistoryStr(); if (TextUtils.isEmpty(subtypeHistoryStr)) { return imsList; } mInputMethodSplitter.setString(subtypeHistoryStr); while (mInputMethodSplitter.hasNext()) { String nextImsStr = mInputMethodSplitter.next(); mSubtypeSplitter.setString(nextImsStr); if (mSubtypeSplitter.hasNext()) { String subtypeId = NOT_A_SUBTYPE_ID_STR; // The first element is ime id. String imeId = mSubtypeSplitter.next(); while (mSubtypeSplitter.hasNext()) { subtypeId = mSubtypeSplitter.next(); break; } imsList.add(new Pair(imeId, subtypeId)); } } return imsList; } private String getSubtypeHistoryStr() { if (DEBUG) { Slog.d(TAG, "getSubtypeHistoryStr: " + Settings.Secure.getStringForUser( mResolver, Settings.Secure.INPUT_METHODS_SUBTYPE_HISTORY, mCurrentUserId)); } return Settings.Secure.getStringForUser( mResolver, Settings.Secure.INPUT_METHODS_SUBTYPE_HISTORY, mCurrentUserId); } public void putSelectedInputMethod(String imeId) { if (DEBUG) { Slog.d(TAG, "putSelectedInputMethodStr: " + imeId + ", " + mCurrentUserId); } Settings.Secure.putStringForUser( mResolver, Settings.Secure.DEFAULT_INPUT_METHOD, imeId, mCurrentUserId); } public void putSelectedSubtype(int subtypeId) { if (DEBUG) { Slog.d(TAG, "putSelectedInputMethodSubtypeStr: " + subtypeId + ", " + mCurrentUserId); } Settings.Secure.putIntForUser(mResolver, Settings.Secure.SELECTED_INPUT_METHOD_SUBTYPE, subtypeId, mCurrentUserId); } public String getDisabledSystemInputMethods() { return Settings.Secure.getStringForUser( mResolver, Settings.Secure.DISABLED_SYSTEM_INPUT_METHODS, mCurrentUserId); } public String getSelectedInputMethod() { if (DEBUG) { Slog.d(TAG, "getSelectedInputMethodStr: " + Settings.Secure.getStringForUser( mResolver, Settings.Secure.DEFAULT_INPUT_METHOD, mCurrentUserId) + ", " + mCurrentUserId); } return Settings.Secure.getStringForUser( mResolver, Settings.Secure.DEFAULT_INPUT_METHOD, mCurrentUserId); } public boolean isSubtypeSelected() { return getSelectedInputMethodSubtypeHashCode() != NOT_A_SUBTYPE_ID; } private int getSelectedInputMethodSubtypeHashCode() { try { return Settings.Secure.getIntForUser( mResolver, Settings.Secure.SELECTED_INPUT_METHOD_SUBTYPE, mCurrentUserId); } catch (SettingNotFoundException e) { return NOT_A_SUBTYPE_ID; } } public boolean isShowImeWithHardKeyboardEnabled() { return Settings.Secure.getIntForUser(mResolver, Settings.Secure.SHOW_IME_WITH_HARD_KEYBOARD, 0, mCurrentUserId) == 1; } public void setShowImeWithHardKeyboard(boolean show) { Settings.Secure.putIntForUser(mResolver, Settings.Secure.SHOW_IME_WITH_HARD_KEYBOARD, show ? 1 : 0, mCurrentUserId); } public int getCurrentUserId() { return mCurrentUserId; } public int getSelectedInputMethodSubtypeId(String selectedImiId) { final InputMethodInfo imi = mMethodMap.get(selectedImiId); if (imi == null) { return NOT_A_SUBTYPE_ID; } final int subtypeHashCode = getSelectedInputMethodSubtypeHashCode(); return getSubtypeIdFromHashCode(imi, subtypeHashCode); } public void saveCurrentInputMethodAndSubtypeToHistory( String curMethodId, InputMethodSubtype currentSubtype) { String subtypeId = NOT_A_SUBTYPE_ID_STR; if (currentSubtype != null) { subtypeId = String.valueOf(currentSubtype.hashCode()); } if (canAddToLastInputMethod(currentSubtype)) { addSubtypeToHistory(curMethodId, subtypeId); } } public HashMap> getExplicitlyOrImplicitlyEnabledInputMethodsAndSubtypeListLocked(Context context) { HashMap> enabledInputMethodAndSubtypes = new HashMap>(); for (InputMethodInfo imi: getEnabledInputMethodListLocked()) { enabledInputMethodAndSubtypes.put( imi, getEnabledInputMethodSubtypeListLocked(context, imi, true)); } return enabledInputMethodAndSubtypes; } } }