InputMethodAndSubtypeEnabler.java revision f0ae329740eb66663b6f099ca24a13ea7572418c
14a5f889f80b683446e498f244d0eadfd979ca5d0satok/*
24a5f889f80b683446e498f244d0eadfd979ca5d0satok * Copyright (C) 2010 The Android Open Source Project
34a5f889f80b683446e498f244d0eadfd979ca5d0satok *
44a5f889f80b683446e498f244d0eadfd979ca5d0satok * Licensed under the Apache License, Version 2.0 (the "License");
54a5f889f80b683446e498f244d0eadfd979ca5d0satok * you may not use this file except in compliance with the License.
64a5f889f80b683446e498f244d0eadfd979ca5d0satok * You may obtain a copy of the License at
74a5f889f80b683446e498f244d0eadfd979ca5d0satok *
84a5f889f80b683446e498f244d0eadfd979ca5d0satok *      http://www.apache.org/licenses/LICENSE-2.0
94a5f889f80b683446e498f244d0eadfd979ca5d0satok *
104a5f889f80b683446e498f244d0eadfd979ca5d0satok * Unless required by applicable law or agreed to in writing, software
114a5f889f80b683446e498f244d0eadfd979ca5d0satok * distributed under the License is distributed on an "AS IS" BASIS,
124a5f889f80b683446e498f244d0eadfd979ca5d0satok * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
134a5f889f80b683446e498f244d0eadfd979ca5d0satok * See the License for the specific language governing permissions and
144a5f889f80b683446e498f244d0eadfd979ca5d0satok * limitations under the License.
154a5f889f80b683446e498f244d0eadfd979ca5d0satok */
164a5f889f80b683446e498f244d0eadfd979ca5d0satok
170417e4094713c5f4dac700b645000d0959bf62fasatokpackage com.android.settings.inputmethod;
180417e4094713c5f4dac700b645000d0959bf62fasatok
190417e4094713c5f4dac700b645000d0959bf62fasatokimport com.android.settings.R;
200417e4094713c5f4dac700b645000d0959bf62fasatokimport com.android.settings.SettingsPreferenceFragment;
214a5f889f80b683446e498f244d0eadfd979ca5d0satok
224a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.app.AlertDialog;
234a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.content.Context;
244a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.content.DialogInterface;
254a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.content.pm.PackageManager;
264a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.content.res.Configuration;
274a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.os.Bundle;
284a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.preference.CheckBoxPreference;
294a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.preference.Preference;
304a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.preference.PreferenceCategory;
314a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.preference.PreferenceScreen;
32c88a7ff1efd10374974e45768bde1658cc1d8483satokimport android.text.TextUtils;
334a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.view.inputmethod.InputMethodInfo;
344a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.view.inputmethod.InputMethodManager;
354a5f889f80b683446e498f244d0eadfd979ca5d0satokimport android.view.inputmethod.InputMethodSubtype;
364a5f889f80b683446e498f244d0eadfd979ca5d0satok
374a5f889f80b683446e498f244d0eadfd979ca5d0satokimport java.util.ArrayList;
38c88a7ff1efd10374974e45768bde1658cc1d8483satokimport java.util.HashMap;
394a5f889f80b683446e498f244d0eadfd979ca5d0satokimport java.util.List;
404a5f889f80b683446e498f244d0eadfd979ca5d0satok
414a5f889f80b683446e498f244d0eadfd979ca5d0satokpublic class InputMethodAndSubtypeEnabler extends SettingsPreferenceFragment {
424a5f889f80b683446e498f244d0eadfd979ca5d0satok    private AlertDialog mDialog = null;
43c88a7ff1efd10374974e45768bde1658cc1d8483satok    private boolean mHaveHardKeyboard;
441c58c15e7a84b582b7d4d20c26824cabce80bce9satok    final private HashMap<String, List<Preference>> mInputMethodAndSubtypePrefsMap =
451c58c15e7a84b582b7d4d20c26824cabce80bce9satok            new HashMap<String, List<Preference>>();
461c58c15e7a84b582b7d4d20c26824cabce80bce9satok    final private HashMap<String, CheckBoxPreference> mSubtypeAutoSelectionCBMap =
471c58c15e7a84b582b7d4d20c26824cabce80bce9satok            new HashMap<String, CheckBoxPreference>();
481c58c15e7a84b582b7d4d20c26824cabce80bce9satok    private InputMethodManager mImm;
49c88a7ff1efd10374974e45768bde1658cc1d8483satok    private List<InputMethodInfo> mInputMethodProperties;
50c88a7ff1efd10374974e45768bde1658cc1d8483satok    private String mInputMethodId;
514a5f889f80b683446e498f244d0eadfd979ca5d0satok
524a5f889f80b683446e498f244d0eadfd979ca5d0satok    @Override
534a5f889f80b683446e498f244d0eadfd979ca5d0satok    public void onCreate(Bundle icicle) {
544a5f889f80b683446e498f244d0eadfd979ca5d0satok        super.onCreate(icicle);
551c58c15e7a84b582b7d4d20c26824cabce80bce9satok        mImm = (InputMethodManager) getSystemService(Context.INPUT_METHOD_SERVICE);
564a5f889f80b683446e498f244d0eadfd979ca5d0satok        Configuration config = getResources().getConfiguration();
574a5f889f80b683446e498f244d0eadfd979ca5d0satok        mHaveHardKeyboard = (config.keyboard == Configuration.KEYBOARD_QWERTY);
58649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa
59649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa        // Input method id should be available from an Intent when this preference is launched as a
60649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa        // single Activity (see InputMethodAndSubtypeEnablerActivity). It should be available
61649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa        // from a preference argument when the preference is launched as a part of the other
62649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa        // Activity (like a right pane of 2-pane Settings app)
639f7320132b7a5e7def378ae754c8c708759a8670Tadashi G. Takaoka        mInputMethodId = getActivity().getIntent().getStringExtra(
649f7320132b7a5e7def378ae754c8c708759a8670Tadashi G. Takaoka                android.provider.Settings.EXTRA_INPUT_METHOD_ID);
65649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa        if (mInputMethodId == null && (getArguments() != null)) {
66649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa            final String inputMethodId =
67649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa                    getArguments().getString(android.provider.Settings.EXTRA_INPUT_METHOD_ID);
68649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa            if (inputMethodId != null) {
69649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa                mInputMethodId = inputMethodId;
70649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa            }
71649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa        }
72649b9f19e1193547f339239afa32d569f43358fcDaisuke Miyakawa
734a5f889f80b683446e498f244d0eadfd979ca5d0satok        onCreateIMM();
744a5f889f80b683446e498f244d0eadfd979ca5d0satok        setPreferenceScreen(createPreferenceHierarchy());
754a5f889f80b683446e498f244d0eadfd979ca5d0satok    }
764a5f889f80b683446e498f244d0eadfd979ca5d0satok
774a5f889f80b683446e498f244d0eadfd979ca5d0satok    @Override
784a5f889f80b683446e498f244d0eadfd979ca5d0satok    public void onResume() {
794a5f889f80b683446e498f244d0eadfd979ca5d0satok        super.onResume();
800417e4094713c5f4dac700b645000d0959bf62fasatok        InputMethodAndSubtypeUtil.loadInputMethodSubtypeList(
811c58c15e7a84b582b7d4d20c26824cabce80bce9satok                this, getContentResolver(), mInputMethodProperties, mInputMethodAndSubtypePrefsMap);
821c58c15e7a84b582b7d4d20c26824cabce80bce9satok        updateAutoSelectionCB();
834a5f889f80b683446e498f244d0eadfd979ca5d0satok    }
844a5f889f80b683446e498f244d0eadfd979ca5d0satok
854a5f889f80b683446e498f244d0eadfd979ca5d0satok    @Override
864a5f889f80b683446e498f244d0eadfd979ca5d0satok    public void onPause() {
874a5f889f80b683446e498f244d0eadfd979ca5d0satok        super.onPause();
881c58c15e7a84b582b7d4d20c26824cabce80bce9satok        // Clear all subtypes of all IMEs to make sure
891c58c15e7a84b582b7d4d20c26824cabce80bce9satok        clearImplicitlyEnabledSubtypes(null);
900417e4094713c5f4dac700b645000d0959bf62fasatok        InputMethodAndSubtypeUtil.saveInputMethodSubtypeList(this, getContentResolver(),
915fd39cafe1de792300e9e3dd60258a1ce5079f73satok                mInputMethodProperties, mHaveHardKeyboard);
924a5f889f80b683446e498f244d0eadfd979ca5d0satok    }
934a5f889f80b683446e498f244d0eadfd979ca5d0satok
944a5f889f80b683446e498f244d0eadfd979ca5d0satok    @Override
954a5f889f80b683446e498f244d0eadfd979ca5d0satok    public boolean onPreferenceTreeClick(
964a5f889f80b683446e498f244d0eadfd979ca5d0satok            PreferenceScreen preferenceScreen, Preference preference) {
974a5f889f80b683446e498f244d0eadfd979ca5d0satok
984a5f889f80b683446e498f244d0eadfd979ca5d0satok        if (preference instanceof CheckBoxPreference) {
994a5f889f80b683446e498f244d0eadfd979ca5d0satok            final CheckBoxPreference chkPref = (CheckBoxPreference) preference;
1001c58c15e7a84b582b7d4d20c26824cabce80bce9satok
1011c58c15e7a84b582b7d4d20c26824cabce80bce9satok            for (String imiId: mSubtypeAutoSelectionCBMap.keySet()) {
1021c58c15e7a84b582b7d4d20c26824cabce80bce9satok                if (mSubtypeAutoSelectionCBMap.get(imiId) == chkPref) {
1031c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    // We look for the first preference item in subtype enabler.
1041c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    // The first item is used for turning on/off subtype auto selection.
1051c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    // We are in the subtype enabler and trying selecting subtypes automatically.
1061c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    setSubtypeAutoSelectionEnabled(imiId, chkPref.isChecked());
1071c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    return super.onPreferenceTreeClick(preferenceScreen, preference);
1081c58c15e7a84b582b7d4d20c26824cabce80bce9satok                }
1091c58c15e7a84b582b7d4d20c26824cabce80bce9satok            }
1101c58c15e7a84b582b7d4d20c26824cabce80bce9satok
1114a5f889f80b683446e498f244d0eadfd979ca5d0satok            final String id = chkPref.getKey();
1124a5f889f80b683446e498f244d0eadfd979ca5d0satok            if (chkPref.isChecked()) {
1134a5f889f80b683446e498f244d0eadfd979ca5d0satok                InputMethodInfo selImi = null;
1144a5f889f80b683446e498f244d0eadfd979ca5d0satok                final int N = mInputMethodProperties.size();
1154a5f889f80b683446e498f244d0eadfd979ca5d0satok                for (int i = 0; i < N; i++) {
1164a5f889f80b683446e498f244d0eadfd979ca5d0satok                    InputMethodInfo imi = mInputMethodProperties.get(i);
1174a5f889f80b683446e498f244d0eadfd979ca5d0satok                    if (id.equals(imi.getId())) {
1184a5f889f80b683446e498f244d0eadfd979ca5d0satok                        selImi = imi;
1199cd11a9aa5ac74ca89432655d019f68d789bc405satok                        if (InputMethodAndSubtypeUtil.isSystemIme(imi)) {
1209cd11a9aa5ac74ca89432655d019f68d789bc405satok                            InputMethodAndSubtypeUtil.setSubtypesPreferenceEnabled(
1219cd11a9aa5ac74ca89432655d019f68d789bc405satok                                    this, mInputMethodProperties, id, true);
1224a5f889f80b683446e498f244d0eadfd979ca5d0satok                            // This is a built-in IME, so no need to warn.
1234a5f889f80b683446e498f244d0eadfd979ca5d0satok                            return super.onPreferenceTreeClick(preferenceScreen, preference);
1244a5f889f80b683446e498f244d0eadfd979ca5d0satok                        }
1254a5f889f80b683446e498f244d0eadfd979ca5d0satok                        break;
1264a5f889f80b683446e498f244d0eadfd979ca5d0satok                    }
1274a5f889f80b683446e498f244d0eadfd979ca5d0satok                }
1284a5f889f80b683446e498f244d0eadfd979ca5d0satok                if (selImi == null) {
1294a5f889f80b683446e498f244d0eadfd979ca5d0satok                    return super.onPreferenceTreeClick(preferenceScreen, preference);
1304a5f889f80b683446e498f244d0eadfd979ca5d0satok                }
1314a5f889f80b683446e498f244d0eadfd979ca5d0satok                chkPref.setChecked(false);
1324a5f889f80b683446e498f244d0eadfd979ca5d0satok                if (mDialog == null) {
1334a5f889f80b683446e498f244d0eadfd979ca5d0satok                    mDialog = (new AlertDialog.Builder(getActivity()))
1344a5f889f80b683446e498f244d0eadfd979ca5d0satok                            .setTitle(android.R.string.dialog_alert_title)
1354a5f889f80b683446e498f244d0eadfd979ca5d0satok                            .setIcon(android.R.drawable.ic_dialog_alert)
1364a5f889f80b683446e498f244d0eadfd979ca5d0satok                            .setCancelable(true)
1374a5f889f80b683446e498f244d0eadfd979ca5d0satok                            .setPositiveButton(android.R.string.ok,
1384a5f889f80b683446e498f244d0eadfd979ca5d0satok                                    new DialogInterface.OnClickListener() {
1394a5f889f80b683446e498f244d0eadfd979ca5d0satok                                        public void onClick(DialogInterface dialog, int which) {
1404a5f889f80b683446e498f244d0eadfd979ca5d0satok                                            chkPref.setChecked(true);
1419cd11a9aa5ac74ca89432655d019f68d789bc405satok                                            InputMethodAndSubtypeUtil.setSubtypesPreferenceEnabled(
1429cd11a9aa5ac74ca89432655d019f68d789bc405satok                                                    InputMethodAndSubtypeEnabler.this,
1439cd11a9aa5ac74ca89432655d019f68d789bc405satok                                                    mInputMethodProperties, id, true);
1444a5f889f80b683446e498f244d0eadfd979ca5d0satok                                        }
1454a5f889f80b683446e498f244d0eadfd979ca5d0satok
1464a5f889f80b683446e498f244d0eadfd979ca5d0satok                            })
1474a5f889f80b683446e498f244d0eadfd979ca5d0satok                            .setNegativeButton(android.R.string.cancel,
1484a5f889f80b683446e498f244d0eadfd979ca5d0satok                                    new DialogInterface.OnClickListener() {
1494a5f889f80b683446e498f244d0eadfd979ca5d0satok                                        public void onClick(DialogInterface dialog, int which) {
1504a5f889f80b683446e498f244d0eadfd979ca5d0satok                                        }
1514a5f889f80b683446e498f244d0eadfd979ca5d0satok
1524a5f889f80b683446e498f244d0eadfd979ca5d0satok                            })
1534a5f889f80b683446e498f244d0eadfd979ca5d0satok                            .create();
1544a5f889f80b683446e498f244d0eadfd979ca5d0satok                } else {
1554a5f889f80b683446e498f244d0eadfd979ca5d0satok                    if (mDialog.isShowing()) {
1564a5f889f80b683446e498f244d0eadfd979ca5d0satok                        mDialog.dismiss();
1574a5f889f80b683446e498f244d0eadfd979ca5d0satok                    }
1584a5f889f80b683446e498f244d0eadfd979ca5d0satok                }
1594a5f889f80b683446e498f244d0eadfd979ca5d0satok                mDialog.setMessage(getResources().getString(
1604a5f889f80b683446e498f244d0eadfd979ca5d0satok                        R.string.ime_security_warning,
1614a5f889f80b683446e498f244d0eadfd979ca5d0satok                        selImi.getServiceInfo().applicationInfo.loadLabel(getPackageManager())));
1624a5f889f80b683446e498f244d0eadfd979ca5d0satok                mDialog.show();
1634a5f889f80b683446e498f244d0eadfd979ca5d0satok            } else {
1649cd11a9aa5ac74ca89432655d019f68d789bc405satok                InputMethodAndSubtypeUtil.setSubtypesPreferenceEnabled(
1659cd11a9aa5ac74ca89432655d019f68d789bc405satok                        this, mInputMethodProperties, id, false);
16674a5414b3c4473106ab2c043d52d8b55a9e4d0c5satok                updateAutoSelectionCB();
1674a5f889f80b683446e498f244d0eadfd979ca5d0satok            }
1684a5f889f80b683446e498f244d0eadfd979ca5d0satok        }
1694a5f889f80b683446e498f244d0eadfd979ca5d0satok        return super.onPreferenceTreeClick(preferenceScreen, preference);
1704a5f889f80b683446e498f244d0eadfd979ca5d0satok    }
1714a5f889f80b683446e498f244d0eadfd979ca5d0satok
1724a5f889f80b683446e498f244d0eadfd979ca5d0satok    @Override
1734a5f889f80b683446e498f244d0eadfd979ca5d0satok    public void onDestroy() {
1744a5f889f80b683446e498f244d0eadfd979ca5d0satok        super.onDestroy();
1754a5f889f80b683446e498f244d0eadfd979ca5d0satok        if (mDialog != null) {
1764a5f889f80b683446e498f244d0eadfd979ca5d0satok            mDialog.dismiss();
1774a5f889f80b683446e498f244d0eadfd979ca5d0satok            mDialog = null;
1784a5f889f80b683446e498f244d0eadfd979ca5d0satok        }
1794a5f889f80b683446e498f244d0eadfd979ca5d0satok    }
1804a5f889f80b683446e498f244d0eadfd979ca5d0satok
1814a5f889f80b683446e498f244d0eadfd979ca5d0satok    private void onCreateIMM() {
1824a5f889f80b683446e498f244d0eadfd979ca5d0satok        InputMethodManager imm = (InputMethodManager) getSystemService(
1834a5f889f80b683446e498f244d0eadfd979ca5d0satok                Context.INPUT_METHOD_SERVICE);
1844a5f889f80b683446e498f244d0eadfd979ca5d0satok
1854a5f889f80b683446e498f244d0eadfd979ca5d0satok        // TODO: Change mInputMethodProperties to Map
1864a5f889f80b683446e498f244d0eadfd979ca5d0satok        mInputMethodProperties = imm.getInputMethodList();
1874a5f889f80b683446e498f244d0eadfd979ca5d0satok    }
1884a5f889f80b683446e498f244d0eadfd979ca5d0satok
1894a5f889f80b683446e498f244d0eadfd979ca5d0satok    private PreferenceScreen createPreferenceHierarchy() {
1904a5f889f80b683446e498f244d0eadfd979ca5d0satok        // Root
191f0ae329740eb66663b6f099ca24a13ea7572418csatok        final PreferenceScreen root = getPreferenceManager().createPreferenceScreen(getActivity());
192f0ae329740eb66663b6f099ca24a13ea7572418csatok        final Context context = getActivity();
1934a5f889f80b683446e498f244d0eadfd979ca5d0satok
1944a5f889f80b683446e498f244d0eadfd979ca5d0satok        int N = (mInputMethodProperties == null ? 0 : mInputMethodProperties.size());
195c88a7ff1efd10374974e45768bde1658cc1d8483satok
1964a5f889f80b683446e498f244d0eadfd979ca5d0satok        for (int i = 0; i < N; ++i) {
197c88a7ff1efd10374974e45768bde1658cc1d8483satok            final InputMethodInfo imi = mInputMethodProperties.get(i);
1985a813ba7f99f1c797ead4a58fdf05f91296158a4Ken Wakasa            final int subtypeCount = imi.getSubtypeCount();
1995a813ba7f99f1c797ead4a58fdf05f91296158a4Ken Wakasa            if (subtypeCount <= 1) continue;
2001c58c15e7a84b582b7d4d20c26824cabce80bce9satok            final String imiId = imi.getId();
2011c58c15e7a84b582b7d4d20c26824cabce80bce9satok            // Add this subtype to the list when no IME is specified or when the IME of this
202c88a7ff1efd10374974e45768bde1658cc1d8483satok            // subtype is the specified IME.
203c88a7ff1efd10374974e45768bde1658cc1d8483satok            if (!TextUtils.isEmpty(mInputMethodId) && !mInputMethodId.equals(imiId)) {
204c88a7ff1efd10374974e45768bde1658cc1d8483satok                continue;
205c88a7ff1efd10374974e45768bde1658cc1d8483satok            }
206f0ae329740eb66663b6f099ca24a13ea7572418csatok            PreferenceCategory keyboardSettingsCategory = new PreferenceCategory(context);
2074a5f889f80b683446e498f244d0eadfd979ca5d0satok            root.addPreference(keyboardSettingsCategory);
2084a5f889f80b683446e498f244d0eadfd979ca5d0satok            PackageManager pm = getPackageManager();
209c88a7ff1efd10374974e45768bde1658cc1d8483satok            CharSequence label = imi.loadLabel(pm);
210c88a7ff1efd10374974e45768bde1658cc1d8483satok
2111c58c15e7a84b582b7d4d20c26824cabce80bce9satok            keyboardSettingsCategory.setTitle(label);
212c88a7ff1efd10374974e45768bde1658cc1d8483satok            keyboardSettingsCategory.setKey(imiId);
2131c58c15e7a84b582b7d4d20c26824cabce80bce9satok            // TODO: Use toggle Preference if images are ready.
214f0ae329740eb66663b6f099ca24a13ea7572418csatok            CheckBoxPreference autoCB = new CheckBoxPreference(context);
2151c58c15e7a84b582b7d4d20c26824cabce80bce9satok            autoCB.setTitle(R.string.use_system_language_to_select_input_method_subtypes);
2161c58c15e7a84b582b7d4d20c26824cabce80bce9satok            mSubtypeAutoSelectionCBMap.put(imiId, autoCB);
2171c58c15e7a84b582b7d4d20c26824cabce80bce9satok            keyboardSettingsCategory.addPreference(autoCB);
2181c58c15e7a84b582b7d4d20c26824cabce80bce9satok
219f0ae329740eb66663b6f099ca24a13ea7572418csatok            PreferenceCategory activeInputMethodsCategory = new PreferenceCategory(context);
2201c58c15e7a84b582b7d4d20c26824cabce80bce9satok            activeInputMethodsCategory.setTitle(R.string.active_input_method_subtypes);
2211c58c15e7a84b582b7d4d20c26824cabce80bce9satok            root.addPreference(activeInputMethodsCategory);
2224a5f889f80b683446e498f244d0eadfd979ca5d0satok
223c88a7ff1efd10374974e45768bde1658cc1d8483satok            ArrayList<Preference> subtypePreferences = new ArrayList<Preference>();
2245a813ba7f99f1c797ead4a58fdf05f91296158a4Ken Wakasa            if (subtypeCount > 0) {
2255a813ba7f99f1c797ead4a58fdf05f91296158a4Ken Wakasa                for (int j = 0; j < subtypeCount; ++j) {
226f0ae329740eb66663b6f099ca24a13ea7572418csatok                    final InputMethodSubtype subtype = imi.getSubtypeAt(j);
227f0ae329740eb66663b6f099ca24a13ea7572418csatok                    final CharSequence subtypeLabel = subtype.getDisplayName(context,
228f0ae329740eb66663b6f099ca24a13ea7572418csatok                            imi.getPackageName(), imi.getServiceInfo().applicationInfo);
229f0ae329740eb66663b6f099ca24a13ea7572418csatok                    final CheckBoxPreference chkbxPref = new CheckBoxPreference(context);
230c88a7ff1efd10374974e45768bde1658cc1d8483satok                    chkbxPref.setKey(imiId + subtype.hashCode());
2314a5f889f80b683446e498f244d0eadfd979ca5d0satok                    chkbxPref.setTitle(subtypeLabel);
2321c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    activeInputMethodsCategory.addPreference(chkbxPref);
233c88a7ff1efd10374974e45768bde1658cc1d8483satok                    subtypePreferences.add(chkbxPref);
2344a5f889f80b683446e498f244d0eadfd979ca5d0satok                }
2351c58c15e7a84b582b7d4d20c26824cabce80bce9satok                mInputMethodAndSubtypePrefsMap.put(imiId, subtypePreferences);
2364a5f889f80b683446e498f244d0eadfd979ca5d0satok            }
2374a5f889f80b683446e498f244d0eadfd979ca5d0satok        }
2384a5f889f80b683446e498f244d0eadfd979ca5d0satok        return root;
2394a5f889f80b683446e498f244d0eadfd979ca5d0satok    }
2401c58c15e7a84b582b7d4d20c26824cabce80bce9satok
2411c58c15e7a84b582b7d4d20c26824cabce80bce9satok    private boolean isNoSubtypesExplicitlySelected(String imiId) {
2421c58c15e7a84b582b7d4d20c26824cabce80bce9satok        boolean allSubtypesOff = true;
2431c58c15e7a84b582b7d4d20c26824cabce80bce9satok        final List<Preference> subtypePrefs = mInputMethodAndSubtypePrefsMap.get(imiId);
2441c58c15e7a84b582b7d4d20c26824cabce80bce9satok        for (Preference subtypePref: subtypePrefs) {
2451c58c15e7a84b582b7d4d20c26824cabce80bce9satok            if (subtypePref instanceof CheckBoxPreference
2461c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    && ((CheckBoxPreference)subtypePref).isChecked()) {
2471c58c15e7a84b582b7d4d20c26824cabce80bce9satok                allSubtypesOff = false;
2481c58c15e7a84b582b7d4d20c26824cabce80bce9satok                break;
2491c58c15e7a84b582b7d4d20c26824cabce80bce9satok            }
2501c58c15e7a84b582b7d4d20c26824cabce80bce9satok        }
2511c58c15e7a84b582b7d4d20c26824cabce80bce9satok        return allSubtypesOff;
2521c58c15e7a84b582b7d4d20c26824cabce80bce9satok    }
2531c58c15e7a84b582b7d4d20c26824cabce80bce9satok
2541c58c15e7a84b582b7d4d20c26824cabce80bce9satok    private void setSubtypeAutoSelectionEnabled(String imiId, boolean autoSelectionEnabled) {
25574a5414b3c4473106ab2c043d52d8b55a9e4d0c5satok        CheckBoxPreference autoSelectionCB = mSubtypeAutoSelectionCBMap.get(imiId);
25674a5414b3c4473106ab2c043d52d8b55a9e4d0c5satok        if (autoSelectionCB == null) return;
25774a5414b3c4473106ab2c043d52d8b55a9e4d0c5satok        autoSelectionCB.setChecked(autoSelectionEnabled);
2581c58c15e7a84b582b7d4d20c26824cabce80bce9satok        final List<Preference> subtypePrefs = mInputMethodAndSubtypePrefsMap.get(imiId);
2591c58c15e7a84b582b7d4d20c26824cabce80bce9satok        for (Preference subtypePref: subtypePrefs) {
2601c58c15e7a84b582b7d4d20c26824cabce80bce9satok            if (subtypePref instanceof CheckBoxPreference) {
2611c58c15e7a84b582b7d4d20c26824cabce80bce9satok                // When autoSelectionEnabled is true, all subtype prefs need to be disabled with
2621c58c15e7a84b582b7d4d20c26824cabce80bce9satok                // implicitly checked subtypes. In case of false, all subtype prefs need to be
2631c58c15e7a84b582b7d4d20c26824cabce80bce9satok                // enabled.
2641c58c15e7a84b582b7d4d20c26824cabce80bce9satok                subtypePref.setEnabled(!autoSelectionEnabled);
2651c58c15e7a84b582b7d4d20c26824cabce80bce9satok                if (autoSelectionEnabled) {
2661c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    ((CheckBoxPreference)subtypePref).setChecked(false);
2671c58c15e7a84b582b7d4d20c26824cabce80bce9satok                }
2681c58c15e7a84b582b7d4d20c26824cabce80bce9satok            }
2691c58c15e7a84b582b7d4d20c26824cabce80bce9satok        }
2701c58c15e7a84b582b7d4d20c26824cabce80bce9satok        if (autoSelectionEnabled) {
2711c58c15e7a84b582b7d4d20c26824cabce80bce9satok            InputMethodAndSubtypeUtil.saveInputMethodSubtypeList(this, getContentResolver(),
2721c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    mInputMethodProperties, mHaveHardKeyboard);
2731c58c15e7a84b582b7d4d20c26824cabce80bce9satok            setCheckedImplicitlyEnabledSubtypes(imiId);
2741c58c15e7a84b582b7d4d20c26824cabce80bce9satok        }
2751c58c15e7a84b582b7d4d20c26824cabce80bce9satok    }
2761c58c15e7a84b582b7d4d20c26824cabce80bce9satok
2771c58c15e7a84b582b7d4d20c26824cabce80bce9satok    private void setCheckedImplicitlyEnabledSubtypes(String targetImiId) {
2781c58c15e7a84b582b7d4d20c26824cabce80bce9satok        updateImplicitlyEnabledSubtypes(targetImiId, true);
2791c58c15e7a84b582b7d4d20c26824cabce80bce9satok    }
2801c58c15e7a84b582b7d4d20c26824cabce80bce9satok
2811c58c15e7a84b582b7d4d20c26824cabce80bce9satok    private void clearImplicitlyEnabledSubtypes(String targetImiId) {
2821c58c15e7a84b582b7d4d20c26824cabce80bce9satok        updateImplicitlyEnabledSubtypes(targetImiId, false);
2831c58c15e7a84b582b7d4d20c26824cabce80bce9satok    }
2841c58c15e7a84b582b7d4d20c26824cabce80bce9satok
2851c58c15e7a84b582b7d4d20c26824cabce80bce9satok    private void updateImplicitlyEnabledSubtypes(String targetImiId, boolean check) {
2861c58c15e7a84b582b7d4d20c26824cabce80bce9satok        // When targetImiId is null, apply to all subtypes of all IMEs
2871c58c15e7a84b582b7d4d20c26824cabce80bce9satok        for (InputMethodInfo imi: mInputMethodProperties) {
2881c58c15e7a84b582b7d4d20c26824cabce80bce9satok            String imiId = imi.getId();
2891c58c15e7a84b582b7d4d20c26824cabce80bce9satok            if (targetImiId != null && !targetImiId.equals(imiId)) continue;
2901c58c15e7a84b582b7d4d20c26824cabce80bce9satok            final CheckBoxPreference autoCB = mSubtypeAutoSelectionCBMap.get(imiId);
2911c58c15e7a84b582b7d4d20c26824cabce80bce9satok            // No need to update implicitly enabled subtypes when the user has unchecked the
2921c58c15e7a84b582b7d4d20c26824cabce80bce9satok            // "subtype auto selection".
2931c58c15e7a84b582b7d4d20c26824cabce80bce9satok            if (autoCB == null || !autoCB.isChecked()) continue;
2941c58c15e7a84b582b7d4d20c26824cabce80bce9satok            final List<Preference> subtypePrefs = mInputMethodAndSubtypePrefsMap.get(imiId);
2951c58c15e7a84b582b7d4d20c26824cabce80bce9satok            final List<InputMethodSubtype> implicitlyEnabledSubtypes =
2961c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    mImm.getEnabledInputMethodSubtypeList(imi, true);
2971c58c15e7a84b582b7d4d20c26824cabce80bce9satok            if (subtypePrefs == null || implicitlyEnabledSubtypes == null) continue;
2981c58c15e7a84b582b7d4d20c26824cabce80bce9satok            for (Preference subtypePref: subtypePrefs) {
2991c58c15e7a84b582b7d4d20c26824cabce80bce9satok                if (subtypePref instanceof CheckBoxPreference) {
3001c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    CheckBoxPreference cb = (CheckBoxPreference)subtypePref;
3011c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    cb.setChecked(false);
3021c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    if (check) {
3031c58c15e7a84b582b7d4d20c26824cabce80bce9satok                        for (InputMethodSubtype subtype: implicitlyEnabledSubtypes) {
3041c58c15e7a84b582b7d4d20c26824cabce80bce9satok                            String implicitlyEnabledSubtypePrefKey = imiId + subtype.hashCode();
3051c58c15e7a84b582b7d4d20c26824cabce80bce9satok                            if (cb.getKey().equals(implicitlyEnabledSubtypePrefKey)) {
3061c58c15e7a84b582b7d4d20c26824cabce80bce9satok                                cb.setChecked(true);
3071c58c15e7a84b582b7d4d20c26824cabce80bce9satok                                break;
3081c58c15e7a84b582b7d4d20c26824cabce80bce9satok                            }
3091c58c15e7a84b582b7d4d20c26824cabce80bce9satok                        }
3101c58c15e7a84b582b7d4d20c26824cabce80bce9satok                    }
3111c58c15e7a84b582b7d4d20c26824cabce80bce9satok                }
3121c58c15e7a84b582b7d4d20c26824cabce80bce9satok            }
3131c58c15e7a84b582b7d4d20c26824cabce80bce9satok        }
3141c58c15e7a84b582b7d4d20c26824cabce80bce9satok    }
3151c58c15e7a84b582b7d4d20c26824cabce80bce9satok
3161c58c15e7a84b582b7d4d20c26824cabce80bce9satok    private void updateAutoSelectionCB() {
3171c58c15e7a84b582b7d4d20c26824cabce80bce9satok        for (String imiId: mInputMethodAndSubtypePrefsMap.keySet()) {
3181c58c15e7a84b582b7d4d20c26824cabce80bce9satok            setSubtypeAutoSelectionEnabled(imiId, isNoSubtypesExplicitlySelected(imiId));
3191c58c15e7a84b582b7d4d20c26824cabce80bce9satok        }
3201c58c15e7a84b582b7d4d20c26824cabce80bce9satok        setCheckedImplicitlyEnabledSubtypes(null);
3211c58c15e7a84b582b7d4d20c26824cabce80bce9satok    }
3224a5f889f80b683446e498f244d0eadfd979ca5d0satok}
323