1 /* 2 * Copyright (C) 2010 Google Inc. 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not 5 * use this file except in compliance with the License. You may obtain a copy of 6 * 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, WITHOUT 12 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the 13 * License for the specific language governing permissions and limitations under 14 * the License. 15 */ 16 17 package com.android.inputmethod.latin; 18 19 import java.util.HashMap; 20 import java.util.Set; 21 import java.util.Map.Entry; 22 23 import android.content.ContentValues; 24 import android.content.Context; 25 import android.database.Cursor; 26 import android.database.sqlite.SQLiteDatabase; 27 import android.database.sqlite.SQLiteOpenHelper; 28 import android.database.sqlite.SQLiteQueryBuilder; 29 import android.os.AsyncTask; 30 import android.provider.BaseColumns; 31 import android.util.Log; 32 33 /** 34 * Stores new words temporarily until they are promoted to the user dictionary 35 * for longevity. Words in the auto dictionary are used to determine if it's ok 36 * to accept a word that's not in the main or user dictionary. Using a new word 37 * repeatedly will promote it to the user dictionary. 38 */ 39 public class AutoDictionary extends ExpandableDictionary { 40 // Weight added to a user picking a new word from the suggestion strip 41 static final int FREQUENCY_FOR_PICKED = 3; 42 // Weight added to a user typing a new word that doesn't get corrected (or is reverted) 43 static final int FREQUENCY_FOR_TYPED = 1; 44 // A word that is frequently typed and gets promoted to the user dictionary, uses this 45 // frequency. 46 static final int FREQUENCY_FOR_AUTO_ADD = 250; 47 // If the user touches a typed word 2 times or more, it will become valid. 48 private static final int VALIDITY_THRESHOLD = 2 * FREQUENCY_FOR_PICKED; 49 // If the user touches a typed word 4 times or more, it will be added to the user dict. 50 private static final int PROMOTION_THRESHOLD = 4 * FREQUENCY_FOR_PICKED; 51 52 private LatinIME mIme; 53 // Locale for which this auto dictionary is storing words 54 private String mLocale; 55 56 private HashMap<String,Integer> mPendingWrites = new HashMap<String,Integer>(); 57 private final Object mPendingWritesLock = new Object(); 58 59 private static final String DATABASE_NAME = "auto_dict.db"; 60 private static final int DATABASE_VERSION = 1; 61 62 // These are the columns in the dictionary 63 // TODO: Consume less space by using a unique id for locale instead of the whole 64 // 2-5 character string. 65 private static final String COLUMN_ID = BaseColumns._ID; 66 private static final String COLUMN_WORD = "word"; 67 private static final String COLUMN_FREQUENCY = "freq"; 68 private static final String COLUMN_LOCALE = "locale"; 69 70 /** Sort by descending order of frequency. */ 71 public static final String DEFAULT_SORT_ORDER = COLUMN_FREQUENCY + " DESC"; 72 73 /** Name of the words table in the auto_dict.db */ 74 private static final String AUTODICT_TABLE_NAME = "words"; 75 76 private static HashMap<String, String> sDictProjectionMap; 77 78 static { 79 sDictProjectionMap = new HashMap<String, String>(); 80 sDictProjectionMap.put(COLUMN_ID, COLUMN_ID); 81 sDictProjectionMap.put(COLUMN_WORD, COLUMN_WORD); 82 sDictProjectionMap.put(COLUMN_FREQUENCY, COLUMN_FREQUENCY); 83 sDictProjectionMap.put(COLUMN_LOCALE, COLUMN_LOCALE); 84 } 85 86 private static DatabaseHelper mOpenHelper = null; 87 88 public AutoDictionary(Context context, LatinIME ime, String locale) { 89 super(context); 90 mIme = ime; 91 mLocale = locale; 92 if (mOpenHelper == null) { 93 mOpenHelper = new DatabaseHelper(getContext()); 94 } 95 if (mLocale != null && mLocale.length() > 1) { 96 loadDictionary(); 97 } 98 } 99 100 @Override 101 public boolean isValidWord(CharSequence word) { 102 final int frequency = getWordFrequency(word); 103 return frequency >= VALIDITY_THRESHOLD; 104 } 105 106 @Override 107 public void close() { 108 flushPendingWrites(); 109 // Don't close the database as locale changes will require it to be reopened anyway 110 // Also, the database is written to somewhat frequently, so it needs to be kept alive 111 // throughout the life of the process. 112 // mOpenHelper.close(); 113 super.close(); 114 } 115 116 @Override 117 public void loadDictionaryAsync() { 118 // Load the words that correspond to the current input locale 119 Cursor cursor = query(COLUMN_LOCALE + "=?", new String[] { mLocale }); 120 try { 121 if (cursor.moveToFirst()) { 122 int wordIndex = cursor.getColumnIndex(COLUMN_WORD); 123 int frequencyIndex = cursor.getColumnIndex(COLUMN_FREQUENCY); 124 while (!cursor.isAfterLast()) { 125 String word = cursor.getString(wordIndex); 126 int frequency = cursor.getInt(frequencyIndex); 127 // Safeguard against adding really long words. Stack may overflow due 128 // to recursive lookup 129 if (word.length() < getMaxWordLength()) { 130 super.addWord(word, frequency); 131 } 132 cursor.moveToNext(); 133 } 134 } 135 } finally { 136 cursor.close(); 137 } 138 } 139 140 @Override 141 public void addWord(String word, int addFrequency) { 142 final int length = word.length(); 143 // Don't add very short or very long words. 144 if (length < 2 || length > getMaxWordLength()) return; 145 if (mIme.getCurrentWord().isAutoCapitalized()) { 146 // Remove caps before adding 147 word = Character.toLowerCase(word.charAt(0)) + word.substring(1); 148 } 149 int freq = getWordFrequency(word); 150 freq = freq < 0 ? addFrequency : freq + addFrequency; 151 super.addWord(word, freq); 152 153 if (freq >= PROMOTION_THRESHOLD) { 154 mIme.promoteToUserDictionary(word, FREQUENCY_FOR_AUTO_ADD); 155 freq = 0; 156 } 157 158 synchronized (mPendingWritesLock) { 159 // Write a null frequency if it is to be deleted from the db 160 mPendingWrites.put(word, freq == 0 ? null : new Integer(freq)); 161 } 162 } 163 164 /** 165 * Schedules a background thread to write any pending words to the database. 166 */ 167 public void flushPendingWrites() { 168 synchronized (mPendingWritesLock) { 169 // Nothing pending? Return 170 if (mPendingWrites.isEmpty()) return; 171 // Create a background thread to write the pending entries 172 new UpdateDbTask(getContext(), mOpenHelper, mPendingWrites, mLocale).execute(); 173 // Create a new map for writing new entries into while the old one is written to db 174 mPendingWrites = new HashMap<String, Integer>(); 175 } 176 } 177 178 /** 179 * This class helps open, create, and upgrade the database file. 180 */ 181 private static class DatabaseHelper extends SQLiteOpenHelper { 182 183 DatabaseHelper(Context context) { 184 super(context, DATABASE_NAME, null, DATABASE_VERSION); 185 } 186 187 @Override 188 public void onCreate(SQLiteDatabase db) { 189 db.execSQL("CREATE TABLE " + AUTODICT_TABLE_NAME + " (" 190 + COLUMN_ID + " INTEGER PRIMARY KEY," 191 + COLUMN_WORD + " TEXT," 192 + COLUMN_FREQUENCY + " INTEGER," 193 + COLUMN_LOCALE + " TEXT" 194 + ");"); 195 } 196 197 @Override 198 public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) { 199 Log.w("AutoDictionary", "Upgrading database from version " + oldVersion + " to " 200 + newVersion + ", which will destroy all old data"); 201 db.execSQL("DROP TABLE IF EXISTS " + AUTODICT_TABLE_NAME); 202 onCreate(db); 203 } 204 } 205 206 private Cursor query(String selection, String[] selectionArgs) { 207 SQLiteQueryBuilder qb = new SQLiteQueryBuilder(); 208 qb.setTables(AUTODICT_TABLE_NAME); 209 qb.setProjectionMap(sDictProjectionMap); 210 211 // Get the database and run the query 212 SQLiteDatabase db = mOpenHelper.getReadableDatabase(); 213 Cursor c = qb.query(db, null, selection, selectionArgs, null, null, 214 DEFAULT_SORT_ORDER); 215 return c; 216 } 217 218 /** 219 * Async task to write pending words to the database so that it stays in sync with 220 * the in-memory trie. 221 */ 222 private static class UpdateDbTask extends AsyncTask<Void, Void, Void> { 223 private final HashMap<String, Integer> mMap; 224 private final DatabaseHelper mDbHelper; 225 private final String mLocale; 226 227 public UpdateDbTask(Context context, DatabaseHelper openHelper, 228 HashMap<String, Integer> pendingWrites, String locale) { 229 mMap = pendingWrites; 230 mLocale = locale; 231 mDbHelper = openHelper; 232 } 233 234 @Override 235 protected Void doInBackground(Void... v) { 236 SQLiteDatabase db = mDbHelper.getWritableDatabase(); 237 // Write all the entries to the db 238 Set<Entry<String,Integer>> mEntries = mMap.entrySet(); 239 for (Entry<String,Integer> entry : mEntries) { 240 Integer freq = entry.getValue(); 241 db.delete(AUTODICT_TABLE_NAME, COLUMN_WORD + "=? AND " + COLUMN_LOCALE + "=?", 242 new String[] { entry.getKey(), mLocale }); 243 if (freq != null) { 244 db.insert(AUTODICT_TABLE_NAME, null, 245 getContentValues(entry.getKey(), freq, mLocale)); 246 } 247 } 248 return null; 249 } 250 251 private ContentValues getContentValues(String word, int frequency, String locale) { 252 ContentValues values = new ContentValues(4); 253 values.put(COLUMN_WORD, word); 254 values.put(COLUMN_FREQUENCY, frequency); 255 values.put(COLUMN_LOCALE, locale); 256 return values; 257 } 258 } 259 } 260