1 /*
2  * Copyright (C) 2011 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.settings.inputmethod;
18 
19 import android.app.settings.SettingsEnums;
20 import android.content.Context;
21 import android.content.DialogInterface;
22 import android.content.pm.ApplicationInfo;
23 import android.os.Bundle;
24 import android.provider.Settings;
25 import android.util.Log;
26 import android.view.textservice.SpellCheckerInfo;
27 import android.view.textservice.SpellCheckerSubtype;
28 import android.view.textservice.TextServicesManager;
29 import android.widget.CompoundButton;
30 import android.widget.CompoundButton.OnCheckedChangeListener;
31 
32 import androidx.appcompat.app.AlertDialog;
33 import androidx.preference.Preference;
34 import androidx.preference.Preference.OnPreferenceChangeListener;
35 import androidx.preference.PreferenceScreen;
36 
37 import com.android.settings.R;
38 import com.android.settings.SettingsActivity;
39 import com.android.settings.SettingsPreferenceFragment;
40 import com.android.settings.widget.SettingsMainSwitchBar;
41 
42 public class SpellCheckersSettings extends SettingsPreferenceFragment
43         implements OnCheckedChangeListener, OnPreferenceChangeListener {
44     private static final String TAG = SpellCheckersSettings.class.getSimpleName();
45     private static final boolean DBG = false;
46 
47     private static final String KEY_SPELL_CHECKER_LANGUAGE = "spellchecker_language";
48     private static final String KEY_DEFAULT_SPELL_CHECKER = "default_spellchecker";
49     private static final int ITEM_ID_USE_SYSTEM_LANGUAGE = 0;
50 
51     private SettingsMainSwitchBar mSwitchBar;
52     private Preference mSpellCheckerLanaguagePref;
53     private AlertDialog mDialog = null;
54     private SpellCheckerInfo mCurrentSci;
55     private SpellCheckerInfo[] mEnabledScis;
56     private TextServicesManager mTsm;
57 
58     @Override
getMetricsCategory()59     public int getMetricsCategory() {
60         return SettingsEnums.INPUTMETHOD_SPELL_CHECKERS;
61     }
62 
63     @Override
onCreate(final Bundle icicle)64     public void onCreate(final Bundle icicle) {
65         super.onCreate(icicle);
66 
67         addPreferencesFromResource(R.xml.spellchecker_prefs);
68         mSpellCheckerLanaguagePref = findPreference(KEY_SPELL_CHECKER_LANGUAGE);
69 
70         mTsm = (TextServicesManager) getSystemService(Context.TEXT_SERVICES_MANAGER_SERVICE);
71         mCurrentSci = mTsm.getCurrentSpellChecker();
72         mEnabledScis = mTsm.getEnabledSpellCheckers();
73         populatePreferenceScreen();
74     }
75 
populatePreferenceScreen()76     private void populatePreferenceScreen() {
77         final SpellCheckerPreference pref = new SpellCheckerPreference(getPrefContext(),
78                 mEnabledScis);
79         pref.setTitle(R.string.default_spell_checker);
80         final int count = (mEnabledScis == null) ? 0 : mEnabledScis.length;
81         if (count > 0) {
82             pref.setSummary("%s");
83         } else {
84             pref.setSummary(R.string.spell_checker_not_selected);
85         }
86         pref.setKey(KEY_DEFAULT_SPELL_CHECKER);
87         pref.setOnPreferenceChangeListener(this);
88         getPreferenceScreen().addPreference(pref);
89     }
90 
91     @Override
onResume()92     public void onResume() {
93         super.onResume();
94         mSwitchBar = ((SettingsActivity) getActivity()).getSwitchBar();
95         mSwitchBar.setTitle(getContext().getString(R.string.spell_checker_primary_switch_title));
96         mSwitchBar.show();
97         mSwitchBar.addOnSwitchChangeListener(this);
98         updatePreferenceScreen();
99     }
100 
101     @Override
onPause()102     public void onPause() {
103         super.onPause();
104         mSwitchBar.removeOnSwitchChangeListener(this);
105     }
106 
107     @Override
onCheckedChanged(final CompoundButton buttonView, final boolean isChecked)108     public void onCheckedChanged(final CompoundButton buttonView, final boolean isChecked) {
109         Settings.Secure.putInt(getContentResolver(), Settings.Secure.SPELL_CHECKER_ENABLED,
110                 isChecked ? 1 : 0);
111         updatePreferenceScreen();
112     }
113 
updatePreferenceScreen()114     private void updatePreferenceScreen() {
115         mCurrentSci = mTsm.getCurrentSpellChecker();
116         final boolean isSpellCheckerEnabled = mTsm.isSpellCheckerEnabled();
117         mSwitchBar.setChecked(isSpellCheckerEnabled);
118 
119         final SpellCheckerSubtype currentScs;
120         if (mCurrentSci != null) {
121             currentScs = mTsm.getCurrentSpellCheckerSubtype(
122                     false /* allowImplicitlySelectedSubtype */);
123         } else {
124             currentScs = null;
125         }
126         mSpellCheckerLanaguagePref.setSummary(getSpellCheckerSubtypeLabel(mCurrentSci, currentScs));
127 
128         final PreferenceScreen screen = getPreferenceScreen();
129         final int count = screen.getPreferenceCount();
130         for (int index = 0; index < count; index++) {
131             final Preference preference = screen.getPreference(index);
132             preference.setEnabled(isSpellCheckerEnabled);
133             if (preference instanceof SpellCheckerPreference) {
134                 final SpellCheckerPreference pref = (SpellCheckerPreference) preference;
135                 pref.setSelected(mCurrentSci);
136                 pref.setEnabled(mEnabledScis != null);
137             }
138         }
139         mSpellCheckerLanaguagePref.setEnabled(isSpellCheckerEnabled && mCurrentSci != null);
140     }
141 
getSpellCheckerSubtypeLabel(final SpellCheckerInfo sci, final SpellCheckerSubtype subtype)142     private CharSequence getSpellCheckerSubtypeLabel(final SpellCheckerInfo sci,
143             final SpellCheckerSubtype subtype) {
144         if (sci == null) {
145             return getString(R.string.spell_checker_not_selected);
146         }
147         if (subtype == null) {
148             return getString(com.android.settingslib.R
149                     .string.use_system_language_to_select_input_method_subtypes);
150         }
151         return subtype.getDisplayName(
152                 getActivity(), sci.getPackageName(), sci.getServiceInfo().applicationInfo);
153     }
154 
155     @Override
onPreferenceTreeClick(Preference preference)156     public boolean onPreferenceTreeClick(Preference preference) {
157         if (KEY_SPELL_CHECKER_LANGUAGE.equals(preference.getKey())) {
158             writePreferenceClickMetric(preference);
159             showChooseLanguageDialog();
160             return true;
161         }
162         return super.onPreferenceTreeClick(preference);
163     }
164 
165     @Override
onPreferenceChange(Preference preference, Object newValue)166     public boolean onPreferenceChange(Preference preference, Object newValue) {
167         final SpellCheckerInfo sci = (SpellCheckerInfo) newValue;
168         final boolean isSystemApp =
169                 (sci.getServiceInfo().applicationInfo.flags & ApplicationInfo.FLAG_SYSTEM) != 0;
170         if (isSystemApp) {
171             changeCurrentSpellChecker(sci);
172             return true;
173         } else {
174             showSecurityWarnDialog(sci);
175             return false;
176         }
177     }
178 
convertSubtypeIndexToDialogItemId(final int index)179     private static int convertSubtypeIndexToDialogItemId(final int index) {
180         return index + 1;
181     }
182 
convertDialogItemIdToSubtypeIndex(final int item)183     private static int convertDialogItemIdToSubtypeIndex(final int item) {
184         return item - 1;
185     }
186 
showChooseLanguageDialog()187     private void showChooseLanguageDialog() {
188         if (mDialog != null && mDialog.isShowing()) {
189             mDialog.dismiss();
190         }
191         final SpellCheckerInfo currentSci = mTsm.getCurrentSpellChecker();
192         if (currentSci == null) {
193             // This can happen in some situations.  One example is that the package that the current
194             // spell checker belongs to was uninstalled or being in background.
195             return;
196         }
197         final SpellCheckerSubtype currentScs = mTsm.getCurrentSpellCheckerSubtype(
198                 false /* allowImplicitlySelectedSubtype */);
199         final AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
200         builder.setTitle(R.string.phone_language);
201         final int subtypeCount = currentSci.getSubtypeCount();
202         final CharSequence[] items = new CharSequence[subtypeCount + 1 /* default */];
203         items[ITEM_ID_USE_SYSTEM_LANGUAGE] = getSpellCheckerSubtypeLabel(currentSci, null);
204         int checkedItemId = ITEM_ID_USE_SYSTEM_LANGUAGE;
205         for (int index = 0; index < subtypeCount; ++index) {
206             final SpellCheckerSubtype subtype = currentSci.getSubtypeAt(index);
207             final int itemId = convertSubtypeIndexToDialogItemId(index);
208             items[itemId] = getSpellCheckerSubtypeLabel(currentSci, subtype);
209             if (subtype.equals(currentScs)) {
210                 checkedItemId = itemId;
211             }
212         }
213         builder.setSingleChoiceItems(items, checkedItemId, new AlertDialog.OnClickListener() {
214             @Override
215             public void onClick(final DialogInterface dialog, final int item) {
216                 final int subtypeId;
217                 if (item == ITEM_ID_USE_SYSTEM_LANGUAGE) {
218                     subtypeId = SpellCheckerSubtype.SUBTYPE_ID_NONE;
219                 } else {
220                     final int index = convertDialogItemIdToSubtypeIndex(item);
221                     subtypeId = currentSci.getSubtypeAt(index).hashCode();
222                 }
223 
224                 Settings.Secure.putInt(getContentResolver(),
225                         Settings.Secure.SELECTED_SPELL_CHECKER_SUBTYPE, subtypeId);
226 
227                 if (DBG) {
228                     final SpellCheckerSubtype subtype = mTsm.getCurrentSpellCheckerSubtype(
229                             true /* allowImplicitlySelectedSubtype */);
230                     Log.d(TAG, "Current spell check locale is "
231                             + subtype == null ? "null" : subtype.getLocale());
232                 }
233                 dialog.dismiss();
234                 updatePreferenceScreen();
235             }
236         });
237         mDialog = builder.create();
238         mDialog.show();
239     }
240 
showSecurityWarnDialog(final SpellCheckerInfo sci)241     private void showSecurityWarnDialog(final SpellCheckerInfo sci) {
242         if (mDialog != null && mDialog.isShowing()) {
243             mDialog.dismiss();
244         }
245         final AlertDialog.Builder builder = new AlertDialog.Builder(getActivity());
246         builder.setTitle(android.R.string.dialog_alert_title);
247         builder.setMessage(getString(R.string.spellchecker_security_warning,
248                 sci.loadLabel(getPackageManager())));
249         builder.setCancelable(true);
250         builder.setPositiveButton(android.R.string.ok, new DialogInterface.OnClickListener() {
251             @Override
252             public void onClick(final DialogInterface dialog, final int which) {
253                 changeCurrentSpellChecker(sci);
254             }
255         });
256         builder.setNegativeButton(android.R.string.cancel, new DialogInterface.OnClickListener() {
257             @Override
258             public void onClick(final DialogInterface dialog, final int which) {
259             }
260         });
261         mDialog = builder.create();
262         mDialog.show();
263     }
264 
changeCurrentSpellChecker(final SpellCheckerInfo sci)265     private void changeCurrentSpellChecker(final SpellCheckerInfo sci) {
266         Settings.Secure.putString(getContentResolver(), Settings.Secure.SELECTED_SPELL_CHECKER,
267                 sci.getId());
268         // Reset the spell checker subtype
269         Settings.Secure.putInt(getContentResolver(), Settings.Secure.SELECTED_SPELL_CHECKER_SUBTYPE,
270                 SpellCheckerSubtype.SUBTYPE_ID_NONE);
271         if (DBG) {
272             Log.d(TAG, "Current spell check is " + mTsm.getCurrentSpellChecker().getId());
273         }
274         updatePreferenceScreen();
275     }
276 }
277