2012-11-09 09:21:41 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2012 The Android Open Source Project
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package com.android.inputmethod.latin;
|
|
|
|
|
2014-10-23 09:37:32 +00:00
|
|
|
import static com.android.inputmethod.latin.common.Constants.Subtype.KEYBOARD_MODE;
|
2012-11-09 09:21:41 +00:00
|
|
|
|
|
|
|
import android.content.Context;
|
2012-11-28 04:28:18 +00:00
|
|
|
import android.content.SharedPreferences;
|
2014-11-06 05:27:47 +00:00
|
|
|
import android.inputmethodservice.InputMethodService;
|
|
|
|
import android.os.AsyncTask;
|
2014-04-27 22:39:00 +00:00
|
|
|
import android.os.Build;
|
2012-11-09 09:21:41 +00:00
|
|
|
import android.os.IBinder;
|
2013-01-07 09:40:59 +00:00
|
|
|
import android.preference.PreferenceManager;
|
2013-04-17 05:27:32 +00:00
|
|
|
import android.util.Log;
|
2012-11-09 09:21:41 +00:00
|
|
|
import android.view.inputmethod.InputMethodInfo;
|
|
|
|
import android.view.inputmethod.InputMethodManager;
|
|
|
|
import android.view.inputmethod.InputMethodSubtype;
|
|
|
|
|
2014-11-06 05:45:32 +00:00
|
|
|
import com.android.inputmethod.annotations.UsedForTesting;
|
2012-11-09 09:21:41 +00:00
|
|
|
import com.android.inputmethod.compat.InputMethodManagerCompatWrapper;
|
2013-07-22 03:43:37 +00:00
|
|
|
import com.android.inputmethod.latin.settings.Settings;
|
2013-07-24 01:37:07 +00:00
|
|
|
import com.android.inputmethod.latin.utils.AdditionalSubtypeUtils;
|
2014-11-07 20:48:59 +00:00
|
|
|
import com.android.inputmethod.latin.utils.LanguageOnSpacebarUtils;
|
2013-07-24 01:37:07 +00:00
|
|
|
import com.android.inputmethod.latin.utils.SubtypeLocaleUtils;
|
2012-11-09 09:21:41 +00:00
|
|
|
|
|
|
|
import java.util.Collections;
|
2013-06-11 08:33:58 +00:00
|
|
|
import java.util.HashMap;
|
2014-11-06 04:16:20 +00:00
|
|
|
import java.util.HashSet;
|
2012-11-09 09:21:41 +00:00
|
|
|
import java.util.List;
|
2014-11-06 04:16:20 +00:00
|
|
|
import java.util.Locale;
|
2014-11-06 05:27:47 +00:00
|
|
|
import java.util.Map;
|
2014-11-06 04:16:20 +00:00
|
|
|
import java.util.Set;
|
2012-11-09 09:21:41 +00:00
|
|
|
|
2014-10-22 05:04:07 +00:00
|
|
|
import javax.annotation.Nonnull;
|
|
|
|
|
2012-11-09 09:21:41 +00:00
|
|
|
/**
|
|
|
|
* Enrichment class for InputMethodManager to simplify interaction and add functionality.
|
|
|
|
*/
|
2014-08-14 00:09:06 +00:00
|
|
|
// non final for easy mocking.
|
|
|
|
public class RichInputMethodManager {
|
2012-11-09 09:21:41 +00:00
|
|
|
private static final String TAG = RichInputMethodManager.class.getSimpleName();
|
2014-11-06 05:27:47 +00:00
|
|
|
private static final boolean DEBUG = false;
|
2012-11-09 09:21:41 +00:00
|
|
|
|
|
|
|
private RichInputMethodManager() {
|
|
|
|
// This utility class is not publicly instantiable.
|
|
|
|
}
|
|
|
|
|
|
|
|
private static final RichInputMethodManager sInstance = new RichInputMethodManager();
|
|
|
|
|
2014-10-15 14:23:30 +00:00
|
|
|
private Context mContext;
|
2012-11-09 09:21:41 +00:00
|
|
|
private InputMethodManagerCompatWrapper mImmWrapper;
|
2014-03-03 08:04:46 +00:00
|
|
|
private InputMethodInfoCache mInputMethodInfoCache;
|
2014-11-06 05:45:32 +00:00
|
|
|
private RichInputMethodSubtype mCurrentRichInputMethodSubtype;
|
2014-11-06 05:27:47 +00:00
|
|
|
private InputMethodInfo mShortcutInputMethodInfo;
|
|
|
|
private InputMethodSubtype mShortcutSubtype;
|
2012-11-09 09:21:41 +00:00
|
|
|
|
2013-04-17 05:27:32 +00:00
|
|
|
private static final int INDEX_NOT_FOUND = -1;
|
|
|
|
|
2012-11-09 09:21:41 +00:00
|
|
|
public static RichInputMethodManager getInstance() {
|
|
|
|
sInstance.checkInitialized();
|
|
|
|
return sInstance;
|
|
|
|
}
|
|
|
|
|
2013-01-07 09:40:59 +00:00
|
|
|
public static void init(final Context context) {
|
2014-07-04 07:36:54 +00:00
|
|
|
sInstance.initInternal(context);
|
2012-11-28 04:28:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private boolean isInitialized() {
|
|
|
|
return mImmWrapper != null;
|
2012-11-09 09:21:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private void checkInitialized() {
|
2012-11-28 04:28:18 +00:00
|
|
|
if (!isInitialized()) {
|
2012-11-09 09:21:41 +00:00
|
|
|
throw new RuntimeException(TAG + " is used before initialization");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-04 07:36:54 +00:00
|
|
|
private void initInternal(final Context context) {
|
2012-11-28 04:28:18 +00:00
|
|
|
if (isInitialized()) {
|
|
|
|
return;
|
|
|
|
}
|
2012-11-09 09:21:41 +00:00
|
|
|
mImmWrapper = new InputMethodManagerCompatWrapper(context);
|
2014-10-15 14:23:30 +00:00
|
|
|
mContext = context;
|
2014-03-03 08:04:46 +00:00
|
|
|
mInputMethodInfoCache = new InputMethodInfoCache(
|
|
|
|
mImmWrapper.mImm, context.getPackageName());
|
2012-11-28 04:28:18 +00:00
|
|
|
|
|
|
|
// Initialize additional subtypes.
|
2013-07-24 01:37:07 +00:00
|
|
|
SubtypeLocaleUtils.init(context);
|
2014-11-21 03:49:04 +00:00
|
|
|
final InputMethodSubtype[] additionalSubtypes = getAdditionalSubtypes();
|
2014-11-07 20:21:10 +00:00
|
|
|
mImmWrapper.mImm.setAdditionalInputMethodSubtypes(
|
|
|
|
getInputMethodIdOfThisIme(), additionalSubtypes);
|
|
|
|
|
|
|
|
// Initialize the current input method subtype and the shortcut IME.
|
|
|
|
refreshSubtypeCaches();
|
2014-07-04 07:36:54 +00:00
|
|
|
}
|
|
|
|
|
2014-11-21 03:49:04 +00:00
|
|
|
public InputMethodSubtype[] getAdditionalSubtypes() {
|
|
|
|
final SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(mContext);
|
2013-01-08 09:20:31 +00:00
|
|
|
final String prefAdditionalSubtypes = Settings.readPrefAdditionalSubtypes(
|
2014-11-21 03:49:04 +00:00
|
|
|
prefs, mContext.getResources());
|
2014-07-04 07:36:54 +00:00
|
|
|
return AdditionalSubtypeUtils.createAdditionalSubtypesArray(prefAdditionalSubtypes);
|
2012-11-09 09:21:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public InputMethodManager getInputMethodManager() {
|
|
|
|
checkInitialized();
|
|
|
|
return mImmWrapper.mImm;
|
|
|
|
}
|
|
|
|
|
2013-05-09 06:53:31 +00:00
|
|
|
public List<InputMethodSubtype> getMyEnabledInputMethodSubtypeList(
|
|
|
|
boolean allowsImplicitlySelectedSubtypes) {
|
2014-03-03 08:04:46 +00:00
|
|
|
return getEnabledInputMethodSubtypeList(
|
|
|
|
getInputMethodInfoOfThisIme(), allowsImplicitlySelectedSubtypes);
|
2013-05-09 06:53:31 +00:00
|
|
|
}
|
|
|
|
|
2012-11-09 09:21:41 +00:00
|
|
|
public boolean switchToNextInputMethod(final IBinder token, final boolean onlyCurrentIme) {
|
2013-04-17 05:27:32 +00:00
|
|
|
if (mImmWrapper.switchToNextInputMethod(token, onlyCurrentIme)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
// Was not able to call {@link InputMethodManager#switchToNextInputMethodIBinder,boolean)}
|
|
|
|
// because the current device is running ICS or previous and lacks the API.
|
|
|
|
if (switchToNextInputSubtypeInThisIme(token, onlyCurrentIme)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return switchToNextInputMethodAndSubtype(token);
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean switchToNextInputSubtypeInThisIme(final IBinder token,
|
|
|
|
final boolean onlyCurrentIme) {
|
|
|
|
final InputMethodManager imm = mImmWrapper.mImm;
|
|
|
|
final InputMethodSubtype currentSubtype = imm.getCurrentInputMethodSubtype();
|
2013-05-09 06:53:31 +00:00
|
|
|
final List<InputMethodSubtype> enabledSubtypes = getMyEnabledInputMethodSubtypeList(
|
|
|
|
true /* allowsImplicitlySelectedSubtypes */);
|
2013-04-17 05:27:32 +00:00
|
|
|
final int currentIndex = getSubtypeIndexInList(currentSubtype, enabledSubtypes);
|
|
|
|
if (currentIndex == INDEX_NOT_FOUND) {
|
|
|
|
Log.w(TAG, "Can't find current subtype in enabled subtypes: subtype="
|
2013-07-24 01:37:07 +00:00
|
|
|
+ SubtypeLocaleUtils.getSubtypeNameForLogging(currentSubtype));
|
2012-11-09 09:21:41 +00:00
|
|
|
return false;
|
|
|
|
}
|
2013-04-17 05:27:32 +00:00
|
|
|
final int nextIndex = (currentIndex + 1) % enabledSubtypes.size();
|
|
|
|
if (nextIndex <= currentIndex && !onlyCurrentIme) {
|
|
|
|
// The current subtype is the last or only enabled one and it needs to switch to
|
|
|
|
// next IME.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
final InputMethodSubtype nextSubtype = enabledSubtypes.get(nextIndex);
|
|
|
|
setInputMethodAndSubtype(token, nextSubtype);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean switchToNextInputMethodAndSubtype(final IBinder token) {
|
|
|
|
final InputMethodManager imm = mImmWrapper.mImm;
|
|
|
|
final List<InputMethodInfo> enabledImis = imm.getEnabledInputMethodList();
|
2014-03-03 08:04:46 +00:00
|
|
|
final int currentIndex = getImiIndexInList(getInputMethodInfoOfThisIme(), enabledImis);
|
2013-04-17 05:27:32 +00:00
|
|
|
if (currentIndex == INDEX_NOT_FOUND) {
|
|
|
|
Log.w(TAG, "Can't find current IME in enabled IMEs: IME package="
|
2014-03-03 08:04:46 +00:00
|
|
|
+ getInputMethodInfoOfThisIme().getPackageName());
|
2013-04-17 05:27:32 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
final InputMethodInfo nextImi = getNextNonAuxiliaryIme(currentIndex, enabledImis);
|
2013-06-11 08:33:58 +00:00
|
|
|
final List<InputMethodSubtype> enabledSubtypes = getEnabledInputMethodSubtypeList(nextImi,
|
|
|
|
true /* allowsImplicitlySelectedSubtypes */);
|
2013-04-17 05:27:32 +00:00
|
|
|
if (enabledSubtypes.isEmpty()) {
|
|
|
|
// The next IME has no subtype.
|
|
|
|
imm.setInputMethod(token, nextImi.getId());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
final InputMethodSubtype firstSubtype = enabledSubtypes.get(0);
|
|
|
|
imm.setInputMethodAndSubtype(token, nextImi.getId(), firstSubtype);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static int getImiIndexInList(final InputMethodInfo inputMethodInfo,
|
|
|
|
final List<InputMethodInfo> imiList) {
|
|
|
|
final int count = imiList.size();
|
|
|
|
for (int index = 0; index < count; index++) {
|
|
|
|
final InputMethodInfo imi = imiList.get(index);
|
|
|
|
if (imi.equals(inputMethodInfo)) {
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return INDEX_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This method mimics {@link InputMethodManager#switchToNextInputMethod(IBinder,boolean)}.
|
|
|
|
private static InputMethodInfo getNextNonAuxiliaryIme(final int currentIndex,
|
|
|
|
final List<InputMethodInfo> imiList) {
|
|
|
|
final int count = imiList.size();
|
|
|
|
for (int i = 1; i < count; i++) {
|
|
|
|
final int nextIndex = (currentIndex + i) % count;
|
|
|
|
final InputMethodInfo nextImi = imiList.get(nextIndex);
|
|
|
|
if (!isAuxiliaryIme(nextImi)) {
|
|
|
|
return nextImi;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return imiList.get(currentIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copied from {@link InputMethodInfo}. See how auxiliary of IME is determined.
|
|
|
|
private static boolean isAuxiliaryIme(final InputMethodInfo imi) {
|
|
|
|
final int count = imi.getSubtypeCount();
|
|
|
|
if (count == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
for (int index = 0; index < count; index++) {
|
|
|
|
final InputMethodSubtype subtype = imi.getSubtypeAt(index);
|
|
|
|
if (!subtype.isAuxiliary()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2012-11-09 09:21:41 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-03-03 08:04:46 +00:00
|
|
|
private static class InputMethodInfoCache {
|
|
|
|
private final InputMethodManager mImm;
|
|
|
|
private final String mImePackageName;
|
|
|
|
|
2014-11-07 20:48:59 +00:00
|
|
|
private InputMethodInfo mCachedThisImeInfo;
|
|
|
|
private final HashMap<InputMethodInfo, List<InputMethodSubtype>>
|
|
|
|
mCachedSubtypeListWithImplicitlySelected;
|
|
|
|
private final HashMap<InputMethodInfo, List<InputMethodSubtype>>
|
|
|
|
mCachedSubtypeListOnlyExplicitlySelected;
|
2014-03-03 08:04:46 +00:00
|
|
|
|
|
|
|
public InputMethodInfoCache(final InputMethodManager imm, final String imePackageName) {
|
|
|
|
mImm = imm;
|
|
|
|
mImePackageName = imePackageName;
|
2014-11-07 20:48:59 +00:00
|
|
|
mCachedSubtypeListWithImplicitlySelected = new HashMap<>();
|
|
|
|
mCachedSubtypeListOnlyExplicitlySelected = new HashMap<>();
|
2014-03-03 08:04:46 +00:00
|
|
|
}
|
|
|
|
|
2014-11-07 20:48:59 +00:00
|
|
|
public synchronized InputMethodInfo getInputMethodOfThisIme() {
|
|
|
|
if (mCachedThisImeInfo != null) {
|
|
|
|
return mCachedThisImeInfo;
|
2014-03-03 08:04:46 +00:00
|
|
|
}
|
|
|
|
for (final InputMethodInfo imi : mImm.getInputMethodList()) {
|
|
|
|
if (imi.getPackageName().equals(mImePackageName)) {
|
2014-11-07 20:48:59 +00:00
|
|
|
mCachedThisImeInfo = imi;
|
2014-03-03 08:04:46 +00:00
|
|
|
return imi;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
throw new RuntimeException("Input method id for " + mImePackageName + " not found.");
|
|
|
|
}
|
|
|
|
|
2014-11-07 20:48:59 +00:00
|
|
|
public synchronized List<InputMethodSubtype> getEnabledInputMethodSubtypeList(
|
|
|
|
final InputMethodInfo imi, final boolean allowsImplicitlySelectedSubtypes) {
|
|
|
|
final HashMap<InputMethodInfo, List<InputMethodSubtype>> cache =
|
|
|
|
allowsImplicitlySelectedSubtypes
|
|
|
|
? mCachedSubtypeListWithImplicitlySelected
|
|
|
|
: mCachedSubtypeListOnlyExplicitlySelected;
|
|
|
|
final List<InputMethodSubtype> cachedList = cache.get(imi);
|
|
|
|
if (cachedList != null) {
|
|
|
|
return cachedList;
|
|
|
|
}
|
|
|
|
final List<InputMethodSubtype> result = mImm.getEnabledInputMethodSubtypeList(
|
|
|
|
imi, allowsImplicitlySelectedSubtypes);
|
|
|
|
cache.put(imi, result);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-03-03 08:04:46 +00:00
|
|
|
public synchronized void clear() {
|
2014-11-07 20:48:59 +00:00
|
|
|
mCachedThisImeInfo = null;
|
|
|
|
mCachedSubtypeListWithImplicitlySelected.clear();
|
|
|
|
mCachedSubtypeListOnlyExplicitlySelected.clear();
|
2014-03-03 08:04:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-09 09:21:41 +00:00
|
|
|
public InputMethodInfo getInputMethodInfoOfThisIme() {
|
2014-11-07 20:48:59 +00:00
|
|
|
return mInputMethodInfoCache.getInputMethodOfThisIme();
|
2012-11-09 09:21:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public String getInputMethodIdOfThisIme() {
|
2014-03-03 08:04:46 +00:00
|
|
|
return getInputMethodInfoOfThisIme().getId();
|
2012-11-09 09:21:41 +00:00
|
|
|
}
|
|
|
|
|
2012-11-19 02:25:30 +00:00
|
|
|
public boolean checkIfSubtypeBelongsToThisImeAndEnabled(final InputMethodSubtype subtype) {
|
2015-02-10 22:54:38 +00:00
|
|
|
return checkIfSubtypeBelongsToList(subtype,
|
|
|
|
getEnabledInputMethodSubtypeList(
|
|
|
|
getInputMethodInfoOfThisIme(),
|
|
|
|
true /* allowsImplicitlySelectedSubtypes */));
|
2012-11-19 02:25:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public boolean checkIfSubtypeBelongsToThisImeAndImplicitlyEnabled(
|
|
|
|
final InputMethodSubtype subtype) {
|
|
|
|
final boolean subtypeEnabled = checkIfSubtypeBelongsToThisImeAndEnabled(subtype);
|
2015-02-10 22:54:38 +00:00
|
|
|
final boolean subtypeExplicitlyEnabled = checkIfSubtypeBelongsToList(subtype,
|
|
|
|
getMyEnabledInputMethodSubtypeList(false /* allowsImplicitlySelectedSubtypes */));
|
2012-11-19 02:25:30 +00:00
|
|
|
return subtypeEnabled && !subtypeExplicitlyEnabled;
|
2012-11-09 09:21:41 +00:00
|
|
|
}
|
|
|
|
|
2012-11-19 02:25:30 +00:00
|
|
|
private static boolean checkIfSubtypeBelongsToList(final InputMethodSubtype subtype,
|
|
|
|
final List<InputMethodSubtype> subtypes) {
|
2013-04-17 05:27:32 +00:00
|
|
|
return getSubtypeIndexInList(subtype, subtypes) != INDEX_NOT_FOUND;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static int getSubtypeIndexInList(final InputMethodSubtype subtype,
|
|
|
|
final List<InputMethodSubtype> subtypes) {
|
|
|
|
final int count = subtypes.size();
|
|
|
|
for (int index = 0; index < count; index++) {
|
|
|
|
final InputMethodSubtype ims = subtypes.get(index);
|
2012-11-19 02:25:30 +00:00
|
|
|
if (ims.equals(subtype)) {
|
2013-04-17 05:27:32 +00:00
|
|
|
return index;
|
2012-11-09 09:21:41 +00:00
|
|
|
}
|
|
|
|
}
|
2013-04-17 05:27:32 +00:00
|
|
|
return INDEX_NOT_FOUND;
|
2012-11-09 09:21:41 +00:00
|
|
|
}
|
|
|
|
|
2014-11-07 20:21:10 +00:00
|
|
|
public void onSubtypeChanged(@Nonnull final InputMethodSubtype newSubtype) {
|
|
|
|
updateCurrentSubtype(newSubtype);
|
|
|
|
updateShortcutIme();
|
2014-11-06 05:45:32 +00:00
|
|
|
if (DEBUG) {
|
2014-11-07 20:21:10 +00:00
|
|
|
Log.w(TAG, "onSubtypeChanged: " + mCurrentRichInputMethodSubtype.getNameForLogging());
|
2014-11-06 05:45:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static RichInputMethodSubtype sForcedSubtypeForTesting = null;
|
|
|
|
|
|
|
|
@UsedForTesting
|
2014-11-07 20:21:10 +00:00
|
|
|
static void forceSubtype(@Nonnull final InputMethodSubtype subtype) {
|
2014-11-06 05:45:32 +00:00
|
|
|
sForcedSubtypeForTesting = new RichInputMethodSubtype(subtype);
|
|
|
|
}
|
|
|
|
|
2014-11-07 20:21:10 +00:00
|
|
|
@Nonnull
|
2014-11-06 05:45:32 +00:00
|
|
|
public Locale[] getCurrentSubtypeLocales() {
|
|
|
|
if (null != sForcedSubtypeForTesting) {
|
|
|
|
return sForcedSubtypeForTesting.getLocales();
|
|
|
|
}
|
|
|
|
return getCurrentSubtype().getLocales();
|
|
|
|
}
|
|
|
|
|
2014-11-07 20:21:10 +00:00
|
|
|
@Nonnull
|
2014-11-06 05:45:32 +00:00
|
|
|
public RichInputMethodSubtype getCurrentSubtype() {
|
|
|
|
if (null != sForcedSubtypeForTesting) {
|
|
|
|
return sForcedSubtypeForTesting;
|
|
|
|
}
|
|
|
|
return mCurrentRichInputMethodSubtype;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public String getCombiningRulesExtraValueOfCurrentSubtype() {
|
|
|
|
return SubtypeLocaleUtils.getCombiningRulesExtraValue(getCurrentSubtype().getRawSubtype());
|
|
|
|
}
|
|
|
|
|
2012-11-09 09:21:41 +00:00
|
|
|
public boolean hasMultipleEnabledIMEsOrSubtypes(final boolean shouldIncludeAuxiliarySubtypes) {
|
|
|
|
final List<InputMethodInfo> enabledImis = mImmWrapper.mImm.getEnabledInputMethodList();
|
|
|
|
return hasMultipleEnabledSubtypes(shouldIncludeAuxiliarySubtypes, enabledImis);
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean hasMultipleEnabledSubtypesInThisIme(
|
|
|
|
final boolean shouldIncludeAuxiliarySubtypes) {
|
2014-03-03 08:04:46 +00:00
|
|
|
final List<InputMethodInfo> imiList = Collections.singletonList(
|
|
|
|
getInputMethodInfoOfThisIme());
|
2012-11-09 09:21:41 +00:00
|
|
|
return hasMultipleEnabledSubtypes(shouldIncludeAuxiliarySubtypes, imiList);
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean hasMultipleEnabledSubtypes(final boolean shouldIncludeAuxiliarySubtypes,
|
|
|
|
final List<InputMethodInfo> imiList) {
|
|
|
|
// Number of the filtered IMEs
|
|
|
|
int filteredImisCount = 0;
|
|
|
|
|
|
|
|
for (InputMethodInfo imi : imiList) {
|
|
|
|
// We can return true immediately after we find two or more filtered IMEs.
|
|
|
|
if (filteredImisCount > 1) return true;
|
2013-06-11 08:33:58 +00:00
|
|
|
final List<InputMethodSubtype> subtypes = getEnabledInputMethodSubtypeList(imi, true);
|
2012-11-09 09:21:41 +00:00
|
|
|
// IMEs that have no subtypes should be counted.
|
|
|
|
if (subtypes.isEmpty()) {
|
|
|
|
++filteredImisCount;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
int auxCount = 0;
|
|
|
|
for (InputMethodSubtype subtype : subtypes) {
|
|
|
|
if (subtype.isAuxiliary()) {
|
|
|
|
++auxCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
final int nonAuxCount = subtypes.size() - auxCount;
|
|
|
|
|
|
|
|
// IMEs that have one or more non-auxiliary subtypes should be counted.
|
|
|
|
// If shouldIncludeAuxiliarySubtypes is true, IMEs that have two or more auxiliary
|
|
|
|
// subtypes should be counted as well.
|
|
|
|
if (nonAuxCount > 0 || (shouldIncludeAuxiliarySubtypes && auxCount > 1)) {
|
|
|
|
++filteredImisCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (filteredImisCount > 1) {
|
|
|
|
return true;
|
|
|
|
}
|
2013-05-09 06:53:31 +00:00
|
|
|
final List<InputMethodSubtype> subtypes = getMyEnabledInputMethodSubtypeList(true);
|
2012-11-09 09:21:41 +00:00
|
|
|
int keyboardCount = 0;
|
|
|
|
// imm.getEnabledInputMethodSubtypeList(null, true) will return the current IME's
|
|
|
|
// both explicitly and implicitly enabled input method subtype.
|
|
|
|
// (The current IME should be LatinIME.)
|
|
|
|
for (InputMethodSubtype subtype : subtypes) {
|
|
|
|
if (KEYBOARD_MODE.equals(subtype.getMode())) {
|
|
|
|
++keyboardCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return keyboardCount > 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
public InputMethodSubtype findSubtypeByLocaleAndKeyboardLayoutSet(final String localeString,
|
|
|
|
final String keyboardLayoutSetName) {
|
2014-03-03 08:04:46 +00:00
|
|
|
final InputMethodInfo myImi = getInputMethodInfoOfThisIme();
|
2012-11-09 09:21:41 +00:00
|
|
|
final int count = myImi.getSubtypeCount();
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
final InputMethodSubtype subtype = myImi.getSubtypeAt(i);
|
2013-07-24 01:37:07 +00:00
|
|
|
final String layoutName = SubtypeLocaleUtils.getKeyboardLayoutSetName(subtype);
|
2012-11-09 09:21:41 +00:00
|
|
|
if (localeString.equals(subtype.getLocale())
|
|
|
|
&& keyboardLayoutSetName.equals(layoutName)) {
|
|
|
|
return subtype;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2012-10-02 09:40:06 +00:00
|
|
|
public void setInputMethodAndSubtype(final IBinder token, final InputMethodSubtype subtype) {
|
|
|
|
mImmWrapper.mImm.setInputMethodAndSubtype(
|
2014-03-03 08:04:46 +00:00
|
|
|
token, getInputMethodIdOfThisIme(), subtype);
|
2012-10-02 09:40:06 +00:00
|
|
|
}
|
|
|
|
|
2012-11-09 09:21:41 +00:00
|
|
|
public void setAdditionalInputMethodSubtypes(final InputMethodSubtype[] subtypes) {
|
|
|
|
mImmWrapper.mImm.setAdditionalInputMethodSubtypes(
|
2014-03-03 08:04:46 +00:00
|
|
|
getInputMethodIdOfThisIme(), subtypes);
|
|
|
|
// Clear the cache so that we go read the {@link InputMethodInfo} of this IME and list of
|
|
|
|
// subtypes again next time.
|
2014-11-07 20:21:10 +00:00
|
|
|
refreshSubtypeCaches();
|
2013-06-11 08:33:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private List<InputMethodSubtype> getEnabledInputMethodSubtypeList(final InputMethodInfo imi,
|
|
|
|
final boolean allowsImplicitlySelectedSubtypes) {
|
2014-11-07 20:48:59 +00:00
|
|
|
return mInputMethodInfoCache.getEnabledInputMethodSubtypeList(
|
2013-06-11 08:33:58 +00:00
|
|
|
imi, allowsImplicitlySelectedSubtypes);
|
|
|
|
}
|
|
|
|
|
2014-11-07 20:21:10 +00:00
|
|
|
public void refreshSubtypeCaches() {
|
2014-03-03 08:04:46 +00:00
|
|
|
mInputMethodInfoCache.clear();
|
2014-11-07 20:21:10 +00:00
|
|
|
updateCurrentSubtype(mImmWrapper.mImm.getCurrentInputMethodSubtype());
|
|
|
|
updateShortcutIme();
|
2012-11-09 09:21:41 +00:00
|
|
|
}
|
2014-04-27 22:39:00 +00:00
|
|
|
|
|
|
|
public boolean shouldOfferSwitchingToNextInputMethod(final IBinder binder,
|
|
|
|
boolean defaultValue) {
|
2014-06-13 03:12:17 +00:00
|
|
|
// Use the default value instead on Jelly Bean MR2 and previous where
|
|
|
|
// {@link InputMethodManager#shouldOfferSwitchingToNextInputMethod} isn't yet available
|
|
|
|
// and on KitKat where the API is still just a stub to return true always.
|
|
|
|
if (Build.VERSION.SDK_INT <= Build.VERSION_CODES.KITKAT) {
|
2014-04-27 22:39:00 +00:00
|
|
|
return defaultValue;
|
|
|
|
}
|
|
|
|
return mImmWrapper.shouldOfferSwitchingToNextInputMethod(binder);
|
|
|
|
}
|
2014-11-06 04:16:20 +00:00
|
|
|
|
|
|
|
public boolean isSystemLocaleSameAsLocaleOfAllEnabledSubtypesOfEnabledImes() {
|
|
|
|
final Locale systemLocale = mContext.getResources().getConfiguration().locale;
|
|
|
|
final Set<InputMethodSubtype> enabledSubtypesOfEnabledImes = new HashSet<>();
|
|
|
|
final InputMethodManager inputMethodManager = getInputMethodManager();
|
|
|
|
final List<InputMethodInfo> enabledInputMethodInfoList =
|
|
|
|
inputMethodManager.getEnabledInputMethodList();
|
|
|
|
for (final InputMethodInfo info : enabledInputMethodInfoList) {
|
|
|
|
final List<InputMethodSubtype> enabledSubtypes =
|
|
|
|
inputMethodManager.getEnabledInputMethodSubtypeList(
|
|
|
|
info, true /* allowsImplicitlySelectedSubtypes */);
|
|
|
|
if (enabledSubtypes.isEmpty()) {
|
|
|
|
// An IME with no subtypes is found.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
enabledSubtypesOfEnabledImes.addAll(enabledSubtypes);
|
|
|
|
}
|
|
|
|
for (final InputMethodSubtype subtype : enabledSubtypesOfEnabledImes) {
|
|
|
|
if (!subtype.isAuxiliary() && !subtype.getLocale().isEmpty()
|
|
|
|
&& !systemLocale.equals(SubtypeLocaleUtils.getSubtypeLocale(subtype))) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2014-11-06 05:27:47 +00:00
|
|
|
|
2014-11-07 20:21:10 +00:00
|
|
|
private void updateCurrentSubtype(@Nonnull final InputMethodSubtype subtype) {
|
2014-12-11 04:41:30 +00:00
|
|
|
mCurrentRichInputMethodSubtype = new RichInputMethodSubtype(subtype);
|
2014-11-07 20:21:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private void updateShortcutIme() {
|
2014-11-06 05:27:47 +00:00
|
|
|
if (DEBUG) {
|
|
|
|
Log.d(TAG, "Update shortcut IME from : "
|
|
|
|
+ (mShortcutInputMethodInfo == null
|
|
|
|
? "<null>" : mShortcutInputMethodInfo.getId()) + ", "
|
|
|
|
+ (mShortcutSubtype == null ? "<null>" : (
|
|
|
|
mShortcutSubtype.getLocale() + ", " + mShortcutSubtype.getMode())));
|
|
|
|
}
|
2014-11-07 20:48:59 +00:00
|
|
|
final RichInputMethodSubtype richSubtype = mCurrentRichInputMethodSubtype;
|
|
|
|
final boolean implicitlyEnabledSubtype = checkIfSubtypeBelongsToThisImeAndImplicitlyEnabled(
|
|
|
|
richSubtype.getRawSubtype());
|
|
|
|
final Locale systemLocale = mContext.getResources().getConfiguration().locale;
|
|
|
|
LanguageOnSpacebarUtils.onSubtypeChanged(
|
|
|
|
richSubtype, implicitlyEnabledSubtype, systemLocale);
|
|
|
|
LanguageOnSpacebarUtils.setEnabledSubtypes(getMyEnabledInputMethodSubtypeList(
|
|
|
|
true /* allowsImplicitlySelectedSubtypes */));
|
|
|
|
|
2014-11-06 05:27:47 +00:00
|
|
|
// TODO: Update an icon for shortcut IME
|
|
|
|
final Map<InputMethodInfo, List<InputMethodSubtype>> shortcuts =
|
|
|
|
getInputMethodManager().getShortcutInputMethodsAndSubtypes();
|
|
|
|
mShortcutInputMethodInfo = null;
|
|
|
|
mShortcutSubtype = null;
|
|
|
|
for (final InputMethodInfo imi : shortcuts.keySet()) {
|
|
|
|
final List<InputMethodSubtype> subtypes = shortcuts.get(imi);
|
|
|
|
// TODO: Returns the first found IMI for now. Should handle all shortcuts as
|
|
|
|
// appropriate.
|
|
|
|
mShortcutInputMethodInfo = imi;
|
|
|
|
// TODO: Pick up the first found subtype for now. Should handle all subtypes
|
|
|
|
// as appropriate.
|
|
|
|
mShortcutSubtype = subtypes.size() > 0 ? subtypes.get(0) : null;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (DEBUG) {
|
|
|
|
Log.d(TAG, "Update shortcut IME to : "
|
|
|
|
+ (mShortcutInputMethodInfo == null
|
|
|
|
? "<null>" : mShortcutInputMethodInfo.getId()) + ", "
|
|
|
|
+ (mShortcutSubtype == null ? "<null>" : (
|
|
|
|
mShortcutSubtype.getLocale() + ", " + mShortcutSubtype.getMode())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-07 20:21:10 +00:00
|
|
|
public void switchToShortcutIme(final InputMethodService context) {
|
2014-11-06 05:27:47 +00:00
|
|
|
if (mShortcutInputMethodInfo == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
final String imiId = mShortcutInputMethodInfo.getId();
|
|
|
|
switchToTargetIME(imiId, mShortcutSubtype, context);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void switchToTargetIME(final String imiId, final InputMethodSubtype subtype,
|
|
|
|
final InputMethodService context) {
|
|
|
|
final IBinder token = context.getWindow().getWindow().getAttributes().token;
|
|
|
|
if (token == null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
final InputMethodManager imm = getInputMethodManager();
|
|
|
|
new AsyncTask<Void, Void, Void>() {
|
|
|
|
@Override
|
|
|
|
protected Void doInBackground(Void... params) {
|
|
|
|
imm.setInputMethodAndSubtype(token, imiId, subtype);
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}.executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR);
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isShortcutImeReady() {
|
|
|
|
if (mShortcutInputMethodInfo == null) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (mShortcutSubtype == null) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2012-11-09 09:21:41 +00:00
|
|
|
}
|