1 /* 2 * Copyright (C) 2013 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.inputmethod.latin.userdictionary; 18 19 import com.android.inputmethod.latin.R; 20 21 import android.app.ListFragment; 22 import android.content.ContentResolver; 23 import android.content.Context; 24 import android.content.Intent; 25 import android.database.Cursor; 26 import android.os.Build; 27 import android.os.Bundle; 28 import android.provider.UserDictionary; 29 import android.text.TextUtils; 30 import android.view.LayoutInflater; 31 import android.view.Menu; 32 import android.view.MenuInflater; 33 import android.view.MenuItem; 34 import android.view.View; 35 import android.view.ViewGroup; 36 import android.widget.AlphabetIndexer; 37 import android.widget.ListAdapter; 38 import android.widget.ListView; 39 import android.widget.SectionIndexer; 40 import android.widget.SimpleCursorAdapter; 41 import android.widget.TextView; 42 43 import java.util.Locale; 44 45 // Caveat: This class is basically taken from 46 // packages/apps/Settings/src/com/android/settings/inputmethod/UserDictionarySettings.java 47 // in order to deal with some devices that have issues with the user dictionary handling 48 49 public class UserDictionarySettings extends ListFragment { 50 51 public static final boolean IS_SHORTCUT_API_SUPPORTED = 52 Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN; 53 54 private static final String[] QUERY_PROJECTION_SHORTCUT_UNSUPPORTED = 55 { UserDictionary.Words._ID, UserDictionary.Words.WORD}; 56 private static final String[] QUERY_PROJECTION_SHORTCUT_SUPPORTED = 57 { UserDictionary.Words._ID, UserDictionary.Words.WORD, UserDictionary.Words.SHORTCUT}; 58 private static final String[] QUERY_PROJECTION = 59 IS_SHORTCUT_API_SUPPORTED ? 60 QUERY_PROJECTION_SHORTCUT_SUPPORTED : QUERY_PROJECTION_SHORTCUT_UNSUPPORTED; 61 62 // The index of the shortcut in the above array. 63 private static final int INDEX_SHORTCUT = 2; 64 65 private static final String[] ADAPTER_FROM_SHORTCUT_UNSUPPORTED = { 66 UserDictionary.Words.WORD, 67 }; 68 69 private static final String[] ADAPTER_FROM_SHORTCUT_SUPPORTED = { 70 UserDictionary.Words.WORD, UserDictionary.Words.SHORTCUT 71 }; 72 73 private static final String[] ADAPTER_FROM = IS_SHORTCUT_API_SUPPORTED ? 74 ADAPTER_FROM_SHORTCUT_SUPPORTED : ADAPTER_FROM_SHORTCUT_UNSUPPORTED; 75 76 private static final int[] ADAPTER_TO_SHORTCUT_UNSUPPORTED = { 77 android.R.id.text1, 78 }; 79 80 private static final int[] ADAPTER_TO_SHORTCUT_SUPPORTED = { 81 android.R.id.text1, android.R.id.text2 82 }; 83 84 private static final int[] ADAPTER_TO = IS_SHORTCUT_API_SUPPORTED ? 85 ADAPTER_TO_SHORTCUT_SUPPORTED : ADAPTER_TO_SHORTCUT_UNSUPPORTED; 86 87 // Either the locale is empty (means the word is applicable to all locales) 88 // or the word equals our current locale 89 private static final String QUERY_SELECTION = 90 UserDictionary.Words.LOCALE + "=?"; 91 private static final String QUERY_SELECTION_ALL_LOCALES = 92 UserDictionary.Words.LOCALE + " is null"; 93 94 private static final String DELETE_SELECTION_WITH_SHORTCUT = UserDictionary.Words.WORD 95 + "=? AND " + UserDictionary.Words.SHORTCUT + "=?"; 96 private static final String DELETE_SELECTION_WITHOUT_SHORTCUT = UserDictionary.Words.WORD 97 + "=? AND " + UserDictionary.Words.SHORTCUT + " is null OR " 98 + UserDictionary.Words.SHORTCUT + "=''"; 99 private static final String DELETE_SELECTION_SHORTCUT_UNSUPPORTED = 100 UserDictionary.Words.WORD + "=?"; 101 102 private static final int OPTIONS_MENU_ADD = Menu.FIRST; 103 104 private Cursor mCursor; 105 106 protected String mLocale; 107 108 @Override onCreate(Bundle savedInstanceState)109 public void onCreate(Bundle savedInstanceState) { 110 super.onCreate(savedInstanceState); 111 getActivity().getActionBar().setTitle(R.string.edit_personal_dictionary); 112 } 113 114 @Override onCreateView( LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)115 public View onCreateView( 116 LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { 117 return inflater.inflate( 118 R.layout.user_dictionary_preference_list_fragment, container, false); 119 } 120 121 @Override onActivityCreated(Bundle savedInstanceState)122 public void onActivityCreated(Bundle savedInstanceState) { 123 super.onActivityCreated(savedInstanceState); 124 125 final Intent intent = getActivity().getIntent(); 126 final String localeFromIntent = 127 null == intent ? null : intent.getStringExtra("locale"); 128 129 final Bundle arguments = getArguments(); 130 final String localeFromArguments = 131 null == arguments ? null : arguments.getString("locale"); 132 133 final String locale; 134 if (null != localeFromArguments) { 135 locale = localeFromArguments; 136 } else if (null != localeFromIntent) { 137 locale = localeFromIntent; 138 } else { 139 locale = null; 140 } 141 142 mLocale = locale; 143 // WARNING: The following cursor is never closed! TODO: don't put that in a member, and 144 // make sure all cursors are correctly closed. Also, this comes from a call to 145 // Activity#managedQuery, which has been deprecated for a long time (and which FORBIDS 146 // closing the cursor, so take care when resolving this TODO). We should either use a 147 // regular query and close the cursor, or switch to a LoaderManager and a CursorLoader. 148 mCursor = createCursor(locale); 149 TextView emptyView = (TextView) getView().findViewById(android.R.id.empty); 150 emptyView.setText(R.string.user_dict_settings_empty_text); 151 152 final ListView listView = getListView(); 153 listView.setAdapter(createAdapter()); 154 listView.setFastScrollEnabled(true); 155 listView.setEmptyView(emptyView); 156 157 setHasOptionsMenu(true); 158 // Show the language as a subtitle of the action bar 159 getActivity().getActionBar().setSubtitle( 160 UserDictionarySettingsUtils.getLocaleDisplayName(getActivity(), mLocale)); 161 } 162 163 @Override onResume()164 public void onResume() { 165 super.onResume(); 166 ListAdapter adapter = getListView().getAdapter(); 167 if (adapter != null && adapter instanceof MyAdapter) { 168 // The list view is forced refreshed here. This allows the changes done 169 // in UserDictionaryAddWordFragment (update/delete/insert) to be seen when 170 // user goes back to this view. 171 MyAdapter listAdapter = (MyAdapter) adapter; 172 listAdapter.notifyDataSetChanged(); 173 } 174 } 175 176 @SuppressWarnings("deprecation") createCursor(final String locale)177 private Cursor createCursor(final String locale) { 178 // Locale can be any of: 179 // - The string representation of a locale, as returned by Locale#toString() 180 // - The empty string. This means we want a cursor returning words valid for all locales. 181 // - null. This means we want a cursor for the current locale, whatever this is. 182 // Note that this contrasts with the data inside the database, where NULL means "all 183 // locales" and there should never be an empty string. The confusion is called by the 184 // historical use of null for "all locales". 185 // TODO: it should be easy to make this more readable by making the special values 186 // human-readable, like "all_locales" and "current_locales" strings, provided they 187 // can be guaranteed not to match locales that may exist. 188 if ("".equals(locale)) { 189 // Case-insensitive sort 190 return getActivity().managedQuery(UserDictionary.Words.CONTENT_URI, QUERY_PROJECTION, 191 QUERY_SELECTION_ALL_LOCALES, null, 192 "UPPER(" + UserDictionary.Words.WORD + ")"); 193 } 194 final String queryLocale = null != locale ? locale : Locale.getDefault().toString(); 195 return getActivity().managedQuery(UserDictionary.Words.CONTENT_URI, QUERY_PROJECTION, 196 QUERY_SELECTION, new String[] { queryLocale }, 197 "UPPER(" + UserDictionary.Words.WORD + ")"); 198 } 199 createAdapter()200 private ListAdapter createAdapter() { 201 return new MyAdapter(getActivity(), R.layout.user_dictionary_item, mCursor, 202 ADAPTER_FROM, ADAPTER_TO); 203 } 204 205 @Override onListItemClick(ListView l, View v, int position, long id)206 public void onListItemClick(ListView l, View v, int position, long id) { 207 final String word = getWord(position); 208 final String shortcut = getShortcut(position); 209 if (word != null) { 210 showAddOrEditDialog(word, shortcut); 211 } 212 } 213 214 @Override onCreateOptionsMenu(Menu menu, MenuInflater inflater)215 public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) { 216 if (!UserDictionarySettings.IS_SHORTCUT_API_SUPPORTED) { 217 final Locale systemLocale = getResources().getConfiguration().locale; 218 if (!TextUtils.isEmpty(mLocale) && !mLocale.equals(systemLocale.toString())) { 219 // Hide the add button for ICS because it doesn't support specifying a locale 220 // for an entry. This new "locale"-aware API has been added in conjunction 221 // with the shortcut API. 222 return; 223 } 224 } 225 MenuItem actionItem = 226 menu.add(0, OPTIONS_MENU_ADD, 0, R.string.user_dict_settings_add_menu_title) 227 .setIcon(R.drawable.ic_menu_add); 228 actionItem.setShowAsAction( 229 MenuItem.SHOW_AS_ACTION_IF_ROOM | MenuItem.SHOW_AS_ACTION_WITH_TEXT); 230 } 231 232 @Override onOptionsItemSelected(MenuItem item)233 public boolean onOptionsItemSelected(MenuItem item) { 234 if (item.getItemId() == OPTIONS_MENU_ADD) { 235 showAddOrEditDialog(null, null); 236 return true; 237 } 238 return false; 239 } 240 241 /** 242 * Add or edit a word. If editingWord is null, it's an add; otherwise, it's an edit. 243 * @param editingWord the word to edit, or null if it's an add. 244 * @param editingShortcut the shortcut for this entry, or null if none. 245 */ showAddOrEditDialog(final String editingWord, final String editingShortcut)246 private void showAddOrEditDialog(final String editingWord, final String editingShortcut) { 247 final Bundle args = new Bundle(); 248 args.putInt(UserDictionaryAddWordContents.EXTRA_MODE, null == editingWord 249 ? UserDictionaryAddWordContents.MODE_INSERT 250 : UserDictionaryAddWordContents.MODE_EDIT); 251 args.putString(UserDictionaryAddWordContents.EXTRA_WORD, editingWord); 252 args.putString(UserDictionaryAddWordContents.EXTRA_SHORTCUT, editingShortcut); 253 args.putString(UserDictionaryAddWordContents.EXTRA_LOCALE, mLocale); 254 android.preference.PreferenceActivity pa = 255 (android.preference.PreferenceActivity)getActivity(); 256 pa.startPreferencePanel(UserDictionaryAddWordFragment.class.getName(), 257 args, R.string.user_dict_settings_add_dialog_title, null, null, 0); 258 } 259 getWord(final int position)260 private String getWord(final int position) { 261 if (null == mCursor) return null; 262 mCursor.moveToPosition(position); 263 // Handle a possible race-condition 264 if (mCursor.isAfterLast()) return null; 265 266 return mCursor.getString( 267 mCursor.getColumnIndexOrThrow(UserDictionary.Words.WORD)); 268 } 269 getShortcut(final int position)270 private String getShortcut(final int position) { 271 if (!IS_SHORTCUT_API_SUPPORTED) return null; 272 if (null == mCursor) return null; 273 mCursor.moveToPosition(position); 274 // Handle a possible race-condition 275 if (mCursor.isAfterLast()) return null; 276 277 return mCursor.getString( 278 mCursor.getColumnIndexOrThrow(UserDictionary.Words.SHORTCUT)); 279 } 280 deleteWord(final String word, final String shortcut, final ContentResolver resolver)281 public static void deleteWord(final String word, final String shortcut, 282 final ContentResolver resolver) { 283 if (!IS_SHORTCUT_API_SUPPORTED) { 284 resolver.delete(UserDictionary.Words.CONTENT_URI, DELETE_SELECTION_SHORTCUT_UNSUPPORTED, 285 new String[] { word }); 286 } else if (TextUtils.isEmpty(shortcut)) { 287 resolver.delete( 288 UserDictionary.Words.CONTENT_URI, DELETE_SELECTION_WITHOUT_SHORTCUT, 289 new String[] { word }); 290 } else { 291 resolver.delete( 292 UserDictionary.Words.CONTENT_URI, DELETE_SELECTION_WITH_SHORTCUT, 293 new String[] { word, shortcut }); 294 } 295 } 296 297 private static class MyAdapter extends SimpleCursorAdapter implements SectionIndexer { 298 private AlphabetIndexer mIndexer; 299 300 private ViewBinder mViewBinder = new ViewBinder() { 301 302 @Override 303 public boolean setViewValue(final View v, final Cursor c, final int columnIndex) { 304 if (!IS_SHORTCUT_API_SUPPORTED) { 305 // just let SimpleCursorAdapter set the view values 306 return false; 307 } 308 if (columnIndex == INDEX_SHORTCUT) { 309 final String shortcut = c.getString(INDEX_SHORTCUT); 310 if (TextUtils.isEmpty(shortcut)) { 311 v.setVisibility(View.GONE); 312 } else { 313 ((TextView)v).setText(shortcut); 314 v.setVisibility(View.VISIBLE); 315 } 316 v.invalidate(); 317 return true; 318 } 319 320 return false; 321 } 322 }; 323 MyAdapter(final Context context, final int layout, final Cursor c, final String[] from, final int[] to)324 public MyAdapter(final Context context, final int layout, final Cursor c, 325 final String[] from, final int[] to) { 326 super(context, layout, c, from, to, 0 /* flags */); 327 328 if (null != c) { 329 final String alphabet = context.getString(R.string.user_dict_fast_scroll_alphabet); 330 final int wordColIndex = c.getColumnIndexOrThrow(UserDictionary.Words.WORD); 331 mIndexer = new AlphabetIndexer(c, wordColIndex, alphabet); 332 } 333 setViewBinder(mViewBinder); 334 } 335 336 @Override getPositionForSection(final int section)337 public int getPositionForSection(final int section) { 338 return null == mIndexer ? 0 : mIndexer.getPositionForSection(section); 339 } 340 341 @Override getSectionForPosition(final int position)342 public int getSectionForPosition(final int position) { 343 return null == mIndexer ? 0 : mIndexer.getSectionForPosition(position); 344 } 345 346 @Override getSections()347 public Object[] getSections() { 348 return null == mIndexer ? null : mIndexer.getSections(); 349 } 350 } 351 } 352 353