1 /*
2  * Copyright (C) 2017 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.content.Context;
20 import android.database.Cursor;
21 import android.database.MatrixCursor;
22 import android.provider.UserDictionary;
23 import android.util.ArraySet;
24 
25 import androidx.annotation.VisibleForTesting;
26 import androidx.loader.content.CursorLoader;
27 
28 import java.util.Locale;
29 import java.util.Objects;
30 import java.util.Set;
31 
32 public class UserDictionaryCursorLoader extends CursorLoader {
33 
34     @VisibleForTesting
35     static final String[] QUERY_PROJECTION = {
36             UserDictionary.Words._ID,
37             UserDictionary.Words.WORD,
38             UserDictionary.Words.SHORTCUT
39     };
40 
41     // The index of the shortcut in the above array.
42     static final int INDEX_SHORTCUT = 2;
43 
44     // Either the locale is empty (means the word is applicable to all locales)
45     // or the word equals our current locale
46     private static final String QUERY_SELECTION =
47             UserDictionary.Words.LOCALE + "=?";
48     private static final String QUERY_SELECTION_ALL_LOCALES =
49             UserDictionary.Words.LOCALE + " is null";
50 
51 
52     // Locale can be any of:
53     // - The string representation of a locale, as returned by Locale#toString()
54     // - The empty string. This means we want a cursor returning words valid for all locales.
55     // - null. This means we want a cursor for the current locale, whatever this is.
56     // Note that this contrasts with the data inside the database, where NULL means "all
57     // locales" and there should never be an empty string. The confusion is called by the
58     // historical use of null for "all locales".
59     // TODO: it should be easy to make this more readable by making the special values
60     // human-readable, like "all_locales" and "current_locales" strings, provided they
61     // can be guaranteed not to match locales that may exist.
62     private final String mLocale;
63 
UserDictionaryCursorLoader(Context context, String locale)64     public UserDictionaryCursorLoader(Context context, String locale) {
65         super(context);
66         mLocale = locale;
67     }
68 
69     @Override
loadInBackground()70     public Cursor loadInBackground() {
71         final MatrixCursor result = new MatrixCursor(QUERY_PROJECTION);
72         final Cursor candidate;
73         if ("".equals(mLocale)) {
74             // Case-insensitive sort
75             candidate = getContext().getContentResolver().query(
76                     UserDictionary.Words.CONTENT_URI, QUERY_PROJECTION,
77                     QUERY_SELECTION_ALL_LOCALES, null,
78                     "UPPER(" + UserDictionary.Words.WORD + ")");
79         } else {
80             final String queryLocale = null != mLocale ? mLocale : Locale.getDefault().toString();
81             candidate = getContext().getContentResolver().query(UserDictionary.Words.CONTENT_URI,
82                     QUERY_PROJECTION, QUERY_SELECTION,
83                     new String[]{queryLocale}, "UPPER(" + UserDictionary.Words.WORD + ")");
84         }
85         final Set<Integer> hashSet = new ArraySet<>();
86         for (candidate.moveToFirst(); !candidate.isAfterLast(); candidate.moveToNext()) {
87             final int id = candidate.getInt(0);
88             final String word = candidate.getString(1);
89             final String shortcut = candidate.getString(2);
90             final int hash = Objects.hash(word, shortcut);
91             if (hashSet.contains(hash)) {
92                 continue;
93             }
94             hashSet.add(hash);
95             result.addRow(new Object[]{id, word, shortcut});
96         }
97         // The cursor needs to be closed after use, otherwise it will cause resource leakage
98         candidate.close();
99         return result;
100     }
101 }
102