2011-03-18 05:49:06 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2011 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.compat;
|
|
|
|
|
2011-04-08 11:57:13 +01:00
|
|
|
import com.android.inputmethod.deprecated.LanguageSwitcherProxy;
|
2011-03-29 02:45:49 +01:00
|
|
|
import com.android.inputmethod.latin.LatinIME;
|
|
|
|
import com.android.inputmethod.latin.SubtypeSwitcher;
|
|
|
|
import com.android.inputmethod.latin.Utils;
|
|
|
|
|
2011-03-18 05:49:06 +00:00
|
|
|
import android.content.Context;
|
|
|
|
import android.os.IBinder;
|
2011-03-29 02:45:49 +01:00
|
|
|
import android.text.TextUtils;
|
2011-03-18 05:49:06 +00:00
|
|
|
import android.util.Log;
|
|
|
|
import android.view.inputmethod.InputMethodInfo;
|
|
|
|
import android.view.inputmethod.InputMethodManager;
|
|
|
|
|
|
|
|
import java.lang.reflect.Method;
|
2011-03-23 21:01:05 +00:00
|
|
|
import java.util.ArrayList;
|
2011-03-25 22:56:10 +00:00
|
|
|
import java.util.Collections;
|
2011-03-18 05:49:06 +00:00
|
|
|
import java.util.HashMap;
|
|
|
|
import java.util.List;
|
2011-03-29 02:45:49 +01:00
|
|
|
import java.util.Locale;
|
2011-03-18 05:49:06 +00:00
|
|
|
import java.util.Map;
|
|
|
|
|
|
|
|
// TODO: Override this class with the concrete implementation if we need to take care of the
|
|
|
|
// performance.
|
|
|
|
public class InputMethodManagerCompatWrapper {
|
|
|
|
private static final String TAG = InputMethodManagerCompatWrapper.class.getSimpleName();
|
|
|
|
private static final Method METHOD_getCurrentInputMethodSubtype =
|
|
|
|
CompatUtils.getMethod(InputMethodManager.class, "getCurrentInputMethodSubtype");
|
|
|
|
private static final Method METHOD_getEnabledInputMethodSubtypeList =
|
|
|
|
CompatUtils.getMethod(InputMethodManager.class, "getEnabledInputMethodSubtypeList",
|
|
|
|
InputMethodInfo.class, boolean.class);
|
|
|
|
private static final Method METHOD_getShortcutInputMethodsAndSubtypes =
|
|
|
|
CompatUtils.getMethod(InputMethodManager.class, "getShortcutInputMethodsAndSubtypes");
|
|
|
|
private static final Method METHOD_setInputMethodAndSubtype =
|
|
|
|
CompatUtils.getMethod(
|
|
|
|
InputMethodManager.class, "setInputMethodAndSubtype", IBinder.class,
|
|
|
|
String.class, InputMethodSubtypeCompatWrapper.CLASS_InputMethodSubtype);
|
2011-03-25 22:56:10 +00:00
|
|
|
private static final Method METHOD_switchToLastInputMethod = CompatUtils.getMethod(
|
|
|
|
InputMethodManager.class, "switchToLastInputMethod", IBinder.class);
|
2011-03-18 05:49:06 +00:00
|
|
|
|
|
|
|
private static final InputMethodManagerCompatWrapper sInstance =
|
|
|
|
new InputMethodManagerCompatWrapper();
|
|
|
|
|
2011-04-08 11:57:13 +01:00
|
|
|
public static final boolean SUBTYPE_SUPPORTED;
|
|
|
|
|
|
|
|
static {
|
|
|
|
// This static initializer guarantees that METHOD_getShortcutInputMethodsAndSubtypes is
|
|
|
|
// already instantiated.
|
|
|
|
SUBTYPE_SUPPORTED = METHOD_getShortcutInputMethodsAndSubtypes != null;
|
|
|
|
}
|
|
|
|
|
2011-03-29 02:45:49 +01:00
|
|
|
// For the compatibility, IMM will create dummy subtypes if subtypes are not found.
|
|
|
|
// This is required to be false if the current behavior is broken. For now, it's ok to be true.
|
2011-04-20 03:29:26 +01:00
|
|
|
public static final boolean FORCE_ENABLE_VOICE_EVEN_WITH_NO_VOICE_SUBTYPES =
|
|
|
|
!InputMethodServiceCompatWrapper.CAN_HANDLE_ON_CURRENT_INPUT_METHOD_SUBTYPE_CHANGED;
|
2011-03-29 02:45:49 +01:00
|
|
|
private static final String VOICE_MODE = "voice";
|
|
|
|
private static final String KEYBOARD_MODE = "keyboard";
|
|
|
|
|
2011-03-18 05:49:06 +00:00
|
|
|
private InputMethodManager mImm;
|
2011-04-08 11:57:13 +01:00
|
|
|
private LanguageSwitcherProxy mLanguageSwitcherProxy;
|
2011-03-29 02:45:49 +01:00
|
|
|
private String mLatinImePackageName;
|
2011-04-08 11:57:13 +01:00
|
|
|
|
2011-03-18 05:49:06 +00:00
|
|
|
private InputMethodManagerCompatWrapper() {
|
|
|
|
}
|
|
|
|
|
|
|
|
public static InputMethodManagerCompatWrapper getInstance(Context context) {
|
|
|
|
if (sInstance.mImm == null) {
|
|
|
|
sInstance.init(context);
|
|
|
|
}
|
|
|
|
return sInstance;
|
|
|
|
}
|
|
|
|
|
|
|
|
private synchronized void init(Context context) {
|
|
|
|
mImm = (InputMethodManager) context.getSystemService(
|
|
|
|
Context.INPUT_METHOD_SERVICE);
|
2011-03-29 02:45:49 +01:00
|
|
|
if (context instanceof LatinIME) {
|
|
|
|
mLatinImePackageName = context.getPackageName();
|
|
|
|
}
|
2011-04-08 11:57:13 +01:00
|
|
|
mLanguageSwitcherProxy = LanguageSwitcherProxy.getInstance();
|
2011-03-18 05:49:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public InputMethodSubtypeCompatWrapper getCurrentInputMethodSubtype() {
|
2011-04-08 11:57:13 +01:00
|
|
|
if (!SUBTYPE_SUPPORTED) {
|
|
|
|
return new InputMethodSubtypeCompatWrapper(
|
|
|
|
0, 0, mLanguageSwitcherProxy.getInputLocale().toString(), KEYBOARD_MODE, "");
|
|
|
|
}
|
2011-03-29 02:45:49 +01:00
|
|
|
Object o = CompatUtils.invoke(mImm, null, METHOD_getCurrentInputMethodSubtype);
|
|
|
|
return new InputMethodSubtypeCompatWrapper(o);
|
2011-03-18 05:49:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public List<InputMethodSubtypeCompatWrapper> getEnabledInputMethodSubtypeList(
|
2011-03-23 21:01:05 +00:00
|
|
|
InputMethodInfoCompatWrapper imi, boolean allowsImplicitlySelectedSubtypes) {
|
2011-04-08 11:57:13 +01:00
|
|
|
if (!SUBTYPE_SUPPORTED) {
|
2011-04-14 03:42:08 +01:00
|
|
|
String[] languages = mLanguageSwitcherProxy.getEnabledLanguages(
|
|
|
|
allowsImplicitlySelectedSubtypes);
|
2011-04-08 11:57:13 +01:00
|
|
|
List<InputMethodSubtypeCompatWrapper> subtypeList =
|
|
|
|
new ArrayList<InputMethodSubtypeCompatWrapper>();
|
|
|
|
for (String lang: languages) {
|
|
|
|
subtypeList.add(new InputMethodSubtypeCompatWrapper(0, 0, lang, KEYBOARD_MODE, ""));
|
|
|
|
}
|
|
|
|
return subtypeList;
|
|
|
|
}
|
2011-03-18 05:49:06 +00:00
|
|
|
Object retval = CompatUtils.invoke(mImm, null, METHOD_getEnabledInputMethodSubtypeList,
|
2011-03-23 21:01:05 +00:00
|
|
|
(imi != null ? imi.getInputMethodInfo() : null), allowsImplicitlySelectedSubtypes);
|
2011-03-29 02:45:49 +01:00
|
|
|
if (retval == null || !(retval instanceof List) || ((List<?>)retval).isEmpty()) {
|
2011-04-20 03:29:26 +01:00
|
|
|
if (!FORCE_ENABLE_VOICE_EVEN_WITH_NO_VOICE_SUBTYPES) {
|
2011-03-29 02:45:49 +01:00
|
|
|
// Returns an empty list
|
|
|
|
return Collections.emptyList();
|
|
|
|
}
|
|
|
|
// Creates dummy subtypes
|
2011-04-18 08:06:31 +01:00
|
|
|
@SuppressWarnings("unused")
|
2011-03-29 02:45:49 +01:00
|
|
|
List<InputMethodSubtypeCompatWrapper> subtypeList =
|
|
|
|
new ArrayList<InputMethodSubtypeCompatWrapper>();
|
|
|
|
InputMethodSubtypeCompatWrapper keyboardSubtype = getLastResortSubtype(KEYBOARD_MODE);
|
|
|
|
InputMethodSubtypeCompatWrapper voiceSubtype = getLastResortSubtype(VOICE_MODE);
|
|
|
|
if (keyboardSubtype != null) {
|
|
|
|
subtypeList.add(keyboardSubtype);
|
|
|
|
}
|
|
|
|
if (voiceSubtype != null) {
|
|
|
|
subtypeList.add(voiceSubtype);
|
|
|
|
}
|
|
|
|
return subtypeList;
|
|
|
|
}
|
2011-03-19 04:22:28 +00:00
|
|
|
return CompatUtils.copyInputMethodSubtypeListToWrapper((List<?>)retval);
|
2011-03-18 05:49:06 +00:00
|
|
|
}
|
|
|
|
|
2011-03-29 02:45:49 +01:00
|
|
|
private InputMethodInfoCompatWrapper getLatinImeInputMethodInfo() {
|
|
|
|
if (TextUtils.isEmpty(mLatinImePackageName))
|
|
|
|
return null;
|
|
|
|
return Utils.getInputMethodInfo(this, mLatinImePackageName);
|
|
|
|
}
|
|
|
|
|
|
|
|
@SuppressWarnings("unused")
|
|
|
|
private InputMethodSubtypeCompatWrapper getLastResortSubtype(String mode) {
|
2011-04-20 03:29:26 +01:00
|
|
|
if (VOICE_MODE.equals(mode) && !FORCE_ENABLE_VOICE_EVEN_WITH_NO_VOICE_SUBTYPES)
|
2011-03-29 02:45:49 +01:00
|
|
|
return null;
|
|
|
|
Locale inputLocale = SubtypeSwitcher.getInstance().getInputLocale();
|
|
|
|
if (inputLocale == null)
|
|
|
|
return null;
|
|
|
|
return new InputMethodSubtypeCompatWrapper(0, 0, inputLocale.toString(), mode, "");
|
|
|
|
}
|
|
|
|
|
2011-03-23 21:01:05 +00:00
|
|
|
public Map<InputMethodInfoCompatWrapper, List<InputMethodSubtypeCompatWrapper>>
|
2011-03-18 05:49:06 +00:00
|
|
|
getShortcutInputMethodsAndSubtypes() {
|
|
|
|
Object retval = CompatUtils.invoke(mImm, null, METHOD_getShortcutInputMethodsAndSubtypes);
|
2011-03-29 02:45:49 +01:00
|
|
|
if (retval == null || !(retval instanceof Map) || ((Map<?, ?>)retval).isEmpty()) {
|
2011-04-20 03:29:26 +01:00
|
|
|
if (!FORCE_ENABLE_VOICE_EVEN_WITH_NO_VOICE_SUBTYPES) {
|
2011-03-29 02:45:49 +01:00
|
|
|
// Returns an empty map
|
|
|
|
return Collections.emptyMap();
|
|
|
|
}
|
|
|
|
// Creates dummy subtypes
|
2011-04-18 08:06:31 +01:00
|
|
|
@SuppressWarnings("unused")
|
2011-03-29 02:45:49 +01:00
|
|
|
InputMethodInfoCompatWrapper imi = getLatinImeInputMethodInfo();
|
|
|
|
InputMethodSubtypeCompatWrapper voiceSubtype = getLastResortSubtype(VOICE_MODE);
|
|
|
|
if (imi != null && voiceSubtype != null) {
|
|
|
|
Map<InputMethodInfoCompatWrapper, List<InputMethodSubtypeCompatWrapper>>
|
|
|
|
shortcutMap =
|
|
|
|
new HashMap<InputMethodInfoCompatWrapper,
|
|
|
|
List<InputMethodSubtypeCompatWrapper>>();
|
|
|
|
List<InputMethodSubtypeCompatWrapper> subtypeList =
|
|
|
|
new ArrayList<InputMethodSubtypeCompatWrapper>();
|
|
|
|
subtypeList.add(voiceSubtype);
|
|
|
|
shortcutMap.put(imi, subtypeList);
|
|
|
|
return shortcutMap;
|
|
|
|
} else {
|
|
|
|
return Collections.emptyMap();
|
|
|
|
}
|
|
|
|
}
|
2011-03-23 21:01:05 +00:00
|
|
|
Map<InputMethodInfoCompatWrapper, List<InputMethodSubtypeCompatWrapper>> shortcutMap =
|
|
|
|
new HashMap<InputMethodInfoCompatWrapper, List<InputMethodSubtypeCompatWrapper>>();
|
2011-03-18 05:49:06 +00:00
|
|
|
final Map<?, ?> retvalMap = (Map<?, ?>)retval;
|
2011-03-23 21:01:05 +00:00
|
|
|
for (Object key : retvalMap.keySet()) {
|
2011-03-18 05:49:06 +00:00
|
|
|
if (!(key instanceof InputMethodInfo)) {
|
|
|
|
Log.e(TAG, "Class type error.");
|
|
|
|
return null;
|
|
|
|
}
|
2011-03-23 21:01:05 +00:00
|
|
|
shortcutMap.put(new InputMethodInfoCompatWrapper((InputMethodInfo)key),
|
|
|
|
CompatUtils.copyInputMethodSubtypeListToWrapper(retvalMap.get(key)));
|
2011-03-18 05:49:06 +00:00
|
|
|
}
|
|
|
|
return shortcutMap;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void setInputMethodAndSubtype(
|
|
|
|
IBinder token, String id, InputMethodSubtypeCompatWrapper subtype) {
|
2011-04-18 08:06:31 +01:00
|
|
|
if (subtype != null && subtype.hasOriginalObject()) {
|
|
|
|
CompatUtils.invoke(mImm, null, METHOD_setInputMethodAndSubtype,
|
|
|
|
token, id, subtype.getOriginalObject());
|
|
|
|
}
|
2011-03-18 05:49:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public boolean switchToLastInputMethod(IBinder token) {
|
2011-03-29 02:45:49 +01:00
|
|
|
if (SubtypeSwitcher.getInstance().isDummyVoiceMode()) {
|
|
|
|
return true;
|
|
|
|
}
|
2011-03-25 22:56:10 +00:00
|
|
|
return (Boolean)CompatUtils.invoke(mImm, false, METHOD_switchToLastInputMethod, token);
|
2011-03-18 05:49:06 +00:00
|
|
|
}
|
|
|
|
|
2011-03-23 21:01:05 +00:00
|
|
|
public List<InputMethodInfoCompatWrapper> getEnabledInputMethodList() {
|
2011-03-18 05:49:06 +00:00
|
|
|
if (mImm == null) return null;
|
2011-03-23 21:01:05 +00:00
|
|
|
List<InputMethodInfoCompatWrapper> imis = new ArrayList<InputMethodInfoCompatWrapper>();
|
|
|
|
for (InputMethodInfo imi : mImm.getEnabledInputMethodList()) {
|
|
|
|
imis.add(new InputMethodInfoCompatWrapper(imi));
|
|
|
|
}
|
|
|
|
return imis;
|
2011-03-18 05:49:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public void showInputMethodPicker() {
|
|
|
|
if (mImm == null) return;
|
|
|
|
mImm.showInputMethodPicker();
|
|
|
|
}
|
|
|
|
}
|