/* * Copyright (C) 2006-2008 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.server; import com.android.internal.os.HandlerCaller; import com.android.internal.view.IInputContext; import com.android.internal.view.IInputMethod; import com.android.internal.view.IInputMethodCallback; import com.android.internal.view.IInputMethodClient; import com.android.internal.view.IInputMethodManager; import com.android.internal.view.IInputMethodSession; import com.android.internal.view.InputBindResult; import com.android.server.status.IconData; import com.android.server.status.StatusBarService; import org.xmlpull.v1.XmlPullParserException; import android.app.AlertDialog; import android.content.ComponentName; import android.content.ContentResolver; import android.content.Context; import android.content.DialogInterface; import android.content.IntentFilter; import android.content.DialogInterface.OnCancelListener; import android.content.Intent; import android.content.ServiceConnection; import android.content.pm.PackageManager; import android.content.pm.ResolveInfo; import android.content.pm.ServiceInfo; import android.content.res.Resources; import android.content.res.TypedArray; import android.database.ContentObserver; import android.net.Uri; import android.os.Binder; import android.os.Handler; import android.os.IBinder; import android.os.IInterface; import android.os.Message; import android.os.Parcel; import android.os.RemoteException; import android.os.ServiceManager; import android.provider.Settings; import android.text.TextUtils; import android.util.Log; import android.util.PrintWriterPrinter; import android.util.Printer; import android.view.IWindowManager; import android.view.WindowManager; import android.view.inputmethod.DefaultInputMethod; import android.view.inputmethod.InputBinding; import android.view.inputmethod.InputMethod; import android.view.inputmethod.InputMethodInfo; import android.view.inputmethod.InputMethodManager; import android.view.inputmethod.EditorInfo; import java.io.FileDescriptor; import java.io.IOException; import java.io.PrintWriter; import java.util.ArrayList; import java.util.HashMap; import java.util.List; /** * This class provides a system service that manages input methods. */ public class InputMethodManagerService extends IInputMethodManager.Stub implements ServiceConnection, Handler.Callback { static final boolean DEBUG = false; static final String TAG = "InputManagerService"; static final int MSG_SHOW_IM_PICKER = 1; static final int MSG_UNBIND_INPUT = 1000; static final int MSG_BIND_INPUT = 1010; static final int MSG_SHOW_SOFT_INPUT = 1020; static final int MSG_HIDE_SOFT_INPUT = 1030; static final int MSG_ATTACH_TOKEN = 1040; static final int MSG_CREATE_SESSION = 1050; static final int MSG_START_INPUT = 2000; static final int MSG_RESTART_INPUT = 2010; static final int MSG_UNBIND_METHOD = 3000; static final int MSG_BIND_METHOD = 3010; final Context mContext; final Handler mHandler; final SettingsObserver mSettingsObserver; final StatusBarService mStatusBar; final IBinder mInputMethodIcon; final IconData mInputMethodData; final IWindowManager mIWindowManager; final HandlerCaller mCaller; final InputBindResult mNoBinding = new InputBindResult(null, null, -1); // All known input methods. mMethodMap also serves as the global // lock for this class. final ArrayList mMethodList = new ArrayList(); final HashMap mMethodMap = new HashMap(); final TextUtils.SimpleStringSplitter mStringColonSplitter = new TextUtils.SimpleStringSplitter(':'); class SessionState { final ClientState client; final IInputMethod method; final IInputMethodSession session; @Override public String toString() { return "SessionState{uid " + client.uid + " pid " + client.pid + " method " + Integer.toHexString( System.identityHashCode(method)) + " session " + Integer.toHexString( System.identityHashCode(session)) + "}"; } SessionState(ClientState _client, IInputMethod _method, IInputMethodSession _session) { client = _client; method = _method; session = _session; } } class ClientState { final IInputMethodClient client; final IInputContext inputContext; final int uid; final int pid; final InputBinding binding; boolean sessionRequested; SessionState curSession; @Override public String toString() { return "ClientState{" + Integer.toHexString( System.identityHashCode(this)) + " uid " + uid + " pid " + pid + "}"; } ClientState(IInputMethodClient _client, IInputContext _inputContext, int _uid, int _pid) { client = _client; inputContext = _inputContext; uid = _uid; pid = _pid; binding = new InputBinding(null, inputContext.asBinder(), uid, pid); } } final HashMap mClients = new HashMap(); /** * Id of the currently selected input method. */ String mCurMethodId; /** * The current binding sequence number, incremented every time there is * a new bind performed. */ int mCurSeq; /** * The client that is currently bound to an input method. */ ClientState mCurClient; /** * The attributes last provided by the current client. */ EditorInfo mCurAttribute; /** * The input method ID of the input method service that we are currently * connected to or in the process of connecting to. */ String mCurId; /** * Set to true if our ServiceConnection is currently actively bound to * a service (whether or not we have gotten its IBinder back yet). */ boolean mHaveConnection; /** * Set if the client has asked for the input method to be shown. */ boolean mShowRequested; /** * Set if we last told the input method to show itself. */ boolean mInputShown; /** * The Intent used to connect to the current input method. */ Intent mCurIntent; /** * The token we have made for the currently active input method, to * identify it in the future. */ IBinder mCurToken; /** * If non-null, this is the input method service we are currently connected * to. */ IInputMethod mCurMethod; /** * Have we called mCurMethod.bindInput()? */ boolean mBoundToMethod; /** * Currently enabled session. Only touched by service thread, not * protected by a lock. */ SessionState mEnabledSession; /** * True if the screen is on. The value is true initially. */ boolean mScreenOn = true; AlertDialog.Builder mDialogBuilder; AlertDialog mSwitchingDialog; InputMethodInfo[] mIms; CharSequence[] mItems; class SettingsObserver extends ContentObserver { SettingsObserver(Handler handler) { super(handler); ContentResolver resolver = mContext.getContentResolver(); resolver.registerContentObserver(Settings.Secure.getUriFor( Settings.Secure.DEFAULT_INPUT_METHOD), false, this); } @Override public void onChange(boolean selfChange) { synchronized (mMethodMap) { updateFromSettingsLocked(); } } } class ScreenOnOffReceiver extends android.content.BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { if (intent.getAction().equals(Intent.ACTION_SCREEN_ON)) { mScreenOn = true; } else if (intent.getAction().equals(Intent.ACTION_SCREEN_OFF)) { mScreenOn = false; } else { Log.e(TAG, "Unexpected intent " + intent); } // Inform the current client of the change in active status try { if (mCurClient != null && mCurClient.client != null) { mCurClient.client.setActive(mScreenOn); } } catch (RemoteException e) { Log.e(TAG, "Got RemoteException sending 'screen on/off' notification", e); } } } class PackageReceiver extends android.content.BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { synchronized (mMethodMap) { buildInputMethodListLocked(mMethodList, mMethodMap); InputMethodInfo curIm = null; String curInputMethodId = Settings.Secure.getString(context .getContentResolver(), Settings.Secure.DEFAULT_INPUT_METHOD); final int N = mMethodList.size(); if (curInputMethodId != null) { for (int i=0; i enabled = getEnabledInputMethodListLocked(); if (enabled != null && enabled.size() > 0) { changed = true; curIm = enabled.get(0); curInputMethodId = curIm.getId(); Log.i(TAG, "Switching to: " + curInputMethodId); Settings.Secure.putString(mContext.getContentResolver(), Settings.Secure.DEFAULT_INPUT_METHOD, curInputMethodId); } else if (curIm != null) { changed = true; curIm = null; curInputMethodId = ""; Log.i(TAG, "Unsetting current input method"); Settings.Secure.putString(mContext.getContentResolver(), Settings.Secure.DEFAULT_INPUT_METHOD, curInputMethodId); } } } else if (curIm == null) { // We currently don't have a default input method... is // one now available? List enabled = getEnabledInputMethodListLocked(); if (enabled != null && enabled.size() > 0) { changed = true; curIm = enabled.get(0); curInputMethodId = curIm.getId(); Log.i(TAG, "New default input method: " + curInputMethodId); Settings.Secure.putString(mContext.getContentResolver(), Settings.Secure.DEFAULT_INPUT_METHOD, curInputMethodId); } } if (changed) { updateFromSettingsLocked(); } } } } class MethodCallback extends IInputMethodCallback.Stub { final IInputMethod mMethod; MethodCallback(IInputMethod method) { mMethod = method; } public void finishedEvent(int seq, boolean handled) throws RemoteException { } public void sessionCreated(IInputMethodSession session) throws RemoteException { onSessionCreated(mMethod, session); } } public InputMethodManagerService(Context context, StatusBarService statusBar) { mContext = context; mHandler = new Handler(this); mIWindowManager = IWindowManager.Stub.asInterface( ServiceManager.getService(Context.WINDOW_SERVICE)); mCaller = new HandlerCaller(context, new HandlerCaller.Callback() { public void executeMessage(Message msg) { handleMessage(msg); } }); IntentFilter packageFilt = new IntentFilter(); packageFilt.addAction(Intent.ACTION_PACKAGE_ADDED); packageFilt.addAction(Intent.ACTION_PACKAGE_CHANGED); packageFilt.addAction(Intent.ACTION_PACKAGE_REMOVED); packageFilt.addAction(Intent.ACTION_PACKAGE_RESTARTED); packageFilt.addDataScheme("package"); mContext.registerReceiver(new PackageReceiver(), packageFilt); IntentFilter screenOnOffFilt = new IntentFilter(); screenOnOffFilt.addAction(Intent.ACTION_SCREEN_ON); screenOnOffFilt.addAction(Intent.ACTION_SCREEN_OFF); mContext.registerReceiver(new ScreenOnOffReceiver(), screenOnOffFilt); buildInputMethodListLocked(mMethodList, mMethodMap); final String enabledStr = Settings.Secure.getString( mContext.getContentResolver(), Settings.Secure.ENABLED_INPUT_METHODS); Log.i(TAG, "Enabled input methods: " + enabledStr); if (enabledStr == null) { Log.i(TAG, "Enabled input methods has not been set, enabling all"); InputMethodInfo defIm = null; StringBuilder sb = new StringBuilder(256); final int N = mMethodList.size(); for (int i=0; i 0) sb.append(':'); sb.append(imi.getId()); if (defIm == null && imi.getIsDefaultResourceId() != 0) { try { Resources res = mContext.createPackageContext( imi.getPackageName(), 0).getResources(); if (res.getBoolean(imi.getIsDefaultResourceId())) { defIm = imi; Log.i(TAG, "Selected default: " + imi.getId()); } } catch (PackageManager.NameNotFoundException ex) { } catch (Resources.NotFoundException ex) { } } } if (defIm == null && N > 0) { defIm = mMethodList.get(0); Log.i(TAG, "No default found, using " + defIm.getId()); } Settings.Secure.putString(mContext.getContentResolver(), Settings.Secure.ENABLED_INPUT_METHODS, sb.toString()); if (defIm != null) { Settings.Secure.putString(mContext.getContentResolver(), Settings.Secure.DEFAULT_INPUT_METHOD, defIm.getId()); } } mStatusBar = statusBar; mInputMethodData = IconData.makeIcon("ime", null, 0, 0, 0); mInputMethodIcon = statusBar.addIcon(mInputMethodData, null); statusBar.setIconVisibility(mInputMethodIcon, false); mSettingsObserver = new SettingsObserver(mHandler); updateFromSettingsLocked(); } @Override public boolean onTransact(int code, Parcel data, Parcel reply, int flags) throws RemoteException { try { return super.onTransact(code, data, reply, flags); } catch (RuntimeException e) { // The input method manager only throws security exceptions, so let's // log all others. if (!(e instanceof SecurityException)) { Log.e(TAG, "Input Method Manager Crash", e); } throw e; } } public void systemReady() { } public List getInputMethodList() { synchronized (mMethodMap) { return new ArrayList(mMethodList); } } public List getEnabledInputMethodList() { synchronized (mMethodMap) { return getEnabledInputMethodListLocked(); } } List getEnabledInputMethodListLocked() { final ArrayList res = new ArrayList(); final String enabledStr = Settings.Secure.getString( mContext.getContentResolver(), Settings.Secure.ENABLED_INPUT_METHODS); if (enabledStr != null) { final TextUtils.SimpleStringSplitter splitter = mStringColonSplitter; splitter.setString(enabledStr); while (splitter.hasNext()) { InputMethodInfo info = mMethodMap.get(splitter.next()); if (info != null) { res.add(info); } } } return res; } public void addClient(IInputMethodClient client, IInputContext inputContext, int uid, int pid) { synchronized (mMethodMap) { mClients.put(client.asBinder(), new ClientState(client, inputContext, uid, pid)); } } public void removeClient(IInputMethodClient client) { synchronized (mMethodMap) { mClients.remove(client.asBinder()); } } void executeOrSendMessage(IInterface target, Message msg) { if (target.asBinder() instanceof Binder) { mCaller.sendMessage(msg); } else { handleMessage(msg); msg.recycle(); } } void unbindCurrentInputLocked() { if (mCurClient != null) { if (DEBUG) Log.v(TAG, "unbindCurrentInputLocked: client = " + mCurClient.client.asBinder()); if (mBoundToMethod) { mBoundToMethod = false; if (mCurMethod != null) { executeOrSendMessage(mCurMethod, mCaller.obtainMessageO( MSG_UNBIND_INPUT, mCurMethod)); } } executeOrSendMessage(mCurClient.client, mCaller.obtainMessageIO( MSG_UNBIND_METHOD, mCurSeq, mCurClient.client)); mCurClient.sessionRequested = false; // Call setActive(false) on the old client try { mCurClient.client.setActive(false); } catch (RemoteException e) { Log.e(TAG, "Got RemoteException sending setActive(false) notification", e); } mCurClient = null; } } InputBindResult attachNewInputLocked(boolean initial, boolean needResult) { if (!mBoundToMethod) { executeOrSendMessage(mCurMethod, mCaller.obtainMessageOO( MSG_BIND_INPUT, mCurMethod, mCurClient.binding)); mBoundToMethod = true; } final SessionState session = mCurClient.curSession; if (initial) { executeOrSendMessage(session.method, mCaller.obtainMessageOO( MSG_START_INPUT, session, mCurAttribute)); } else { executeOrSendMessage(session.method, mCaller.obtainMessageOO( MSG_RESTART_INPUT, session, mCurAttribute)); } if (mShowRequested) { showCurrentInputLocked(); } return needResult ? new InputBindResult(session.session, mCurId, mCurSeq) : null; } InputBindResult startInputLocked(IInputMethodClient client, EditorInfo attribute, boolean initial, boolean needResult) { // If no method is currently selected, do nothing. if (mCurMethodId == null) { return mNoBinding; } ClientState cs = mClients.get(client.asBinder()); if (cs == null) { throw new IllegalArgumentException("unknown client " + client.asBinder()); } try { if (!mIWindowManager.inputMethodClientHasFocus(cs.client)) { // Check with the window manager to make sure this client actually // has a window with focus. If not, reject. This is thread safe // because if the focus changes some time before or after, the // next client receiving focus that has any interest in input will // be calling through here after that change happens. Log.w(TAG, "Starting input on non-focused client " + cs.client + " (uid=" + cs.uid + " pid=" + cs.pid + ")"); return null; } } catch (RemoteException e) { } if (mCurClient != cs) { // If the client is changing, we need to switch over to the new // one. unbindCurrentInputLocked(); if (DEBUG) Log.v(TAG, "switching to client: client = " + cs.client.asBinder()); // If the screen is on, inform the new client it is active if (mScreenOn) { try { cs.client.setActive(mScreenOn); } catch (RemoteException e) { Log.e(TAG, "Got RemoteException sending setActive notification", e); } } } // Bump up the sequence for this client and attach it. mCurSeq++; if (mCurSeq <= 0) mCurSeq = 1; mCurClient = cs; mCurAttribute = attribute; // Check if the input method is changing. if (mCurId != null && mCurId.equals(mCurMethodId)) { if (cs.curSession != null) { // Fast case: if we are already connected to the input method, // then just return it. return attachNewInputLocked(initial, needResult); } if (mHaveConnection) { if (mCurMethod != null && !cs.sessionRequested) { cs.sessionRequested = true; if (DEBUG) Log.v(TAG, "Creating new session for client " + cs); executeOrSendMessage(mCurMethod, mCaller.obtainMessageOO( MSG_CREATE_SESSION, mCurMethod, new MethodCallback(mCurMethod))); } return new InputBindResult(null, mCurId, mCurSeq); } } InputMethodInfo info = mMethodMap.get(mCurMethodId); if (info == null) { throw new IllegalArgumentException("Unknown id: " + mCurMethodId); } if (mCurToken != null) { try { mIWindowManager.removeWindowToken(mCurToken); } catch (RemoteException e) { } mCurToken = null; } if (mHaveConnection) { mContext.unbindService(this); mHaveConnection = false; } clearCurMethod(); mCurIntent = new Intent(InputMethod.SERVICE_INTERFACE); mCurIntent.setComponent(info.getComponent()); if (mContext.bindService(mCurIntent, this, Context.BIND_AUTO_CREATE)) { mHaveConnection = true; mCurId = info.getId(); mCurToken = new Binder(); try { mIWindowManager.addWindowToken(mCurToken, WindowManager.LayoutParams.TYPE_INPUT_METHOD); } catch (RemoteException e) { } return new InputBindResult(null, mCurId, mCurSeq); } else { mCurIntent = null; Log.e(TAG, "Failure connecting to input method service: " + mCurIntent); } return null; } public InputBindResult startInput(IInputMethodClient client, EditorInfo attribute, boolean initial, boolean needResult) { synchronized (mMethodMap) { final long ident = Binder.clearCallingIdentity(); try { return startInputLocked(client, attribute, initial, needResult); } finally { Binder.restoreCallingIdentity(ident); } } } public void finishInput(IInputMethodClient client) { } public void onServiceConnected(ComponentName name, IBinder service) { synchronized (mMethodMap) { if (mCurIntent != null && name.equals(mCurIntent.getComponent())) { mCurMethod = IInputMethod.Stub.asInterface(service); if (mCurClient != null) { executeOrSendMessage(mCurMethod, mCaller.obtainMessageOO( MSG_ATTACH_TOKEN, mCurMethod, mCurToken)); if (mCurClient != null) { if (DEBUG) Log.v(TAG, "Creating first session while with client " + mCurClient); executeOrSendMessage(mCurMethod, mCaller.obtainMessageOO( MSG_CREATE_SESSION, mCurMethod, new MethodCallback(mCurMethod))); } } } } } void onSessionCreated(IInputMethod method, IInputMethodSession session) { synchronized (mMethodMap) { if (mCurMethod == method) { if (mCurClient != null) { mCurClient.curSession = new SessionState(mCurClient, method, session); mCurClient.sessionRequested = false; InputBindResult res = attachNewInputLocked(true, true); if (res.method != null) { executeOrSendMessage(mCurClient.client, mCaller.obtainMessageOO( MSG_BIND_METHOD, mCurClient.client, res)); } } } } } void clearCurMethod() { if (mCurMethod != null) { for (ClientState cs : mClients.values()) { cs.sessionRequested = false; cs.curSession = null; } mCurMethod = null; } } public void onServiceDisconnected(ComponentName name) { synchronized (mMethodMap) { if (DEBUG) Log.v(TAG, "Service disconnected: " + name + " mCurIntent=" + mCurIntent); if (mCurMethod != null && mCurIntent != null && name.equals(mCurIntent.getComponent())) { clearCurMethod(); mShowRequested = mInputShown; mInputShown = false; if (mCurClient != null) { executeOrSendMessage(mCurClient.client, mCaller.obtainMessageIO( MSG_UNBIND_METHOD, mCurSeq, mCurClient.client)); } } } } public void updateStatusIcon(int iconId, String iconPackage) { if (iconId == 0) { Log.d(TAG, "hide the small icon for the input method"); mStatusBar.setIconVisibility(mInputMethodIcon, false); } else { Log.d(TAG, "show a small icon for the input method"); if (iconPackage != null && iconPackage .equals(InputMethodManager.BUILDIN_INPUTMETHOD_PACKAGE)) { iconPackage = null; } mInputMethodData.iconId = iconId; mInputMethodData.iconPackage = iconPackage; mStatusBar.updateIcon(mInputMethodIcon, mInputMethodData, null); mStatusBar.setIconVisibility(mInputMethodIcon, true); } } void updateFromSettingsLocked() { String id = Settings.Secure.getString(mContext.getContentResolver(), Settings.Secure.DEFAULT_INPUT_METHOD); if (id != null) { try { setInputMethodLocked(id); } catch (IllegalArgumentException e) { Log.w(TAG, "Unknown input method from prefs: " + id, e); } } } void setInputMethodLocked(String id) { InputMethodInfo info = mMethodMap.get(id); if (info == null) { throw new IllegalArgumentException("Unknown id: " + mCurMethodId); } if (id.equals(mCurMethodId)) { return; } final long ident = Binder.clearCallingIdentity(); try { mCurMethodId = id; Settings.Secure.putString(mContext.getContentResolver(), Settings.Secure.DEFAULT_INPUT_METHOD, id); Intent intent = new Intent(Intent.ACTION_INPUT_METHOD_CHANGED); intent.putExtra("input_method_id", id); mContext.sendBroadcast(intent); unbindCurrentInputLocked(); } finally { Binder.restoreCallingIdentity(ident); } } public void showSoftInput(IInputMethodClient client) { synchronized (mMethodMap) { if (mCurClient == null || client == null || mCurClient.client.asBinder() != client.asBinder()) { try { // We need to check if this is the current client with // focus in the window manager, to allow this call to // be made before input is started in it. if (!mIWindowManager.inputMethodClientHasFocus(client)) { Log.w(TAG, "Ignoring showSoftInput of: " + client); return; } } catch (RemoteException e) { } } showCurrentInputLocked(); } } void showCurrentInputLocked() { mShowRequested = true; if (!mInputShown) { if (mCurMethod != null) { executeOrSendMessage(mCurMethod, mCaller.obtainMessageO( MSG_SHOW_SOFT_INPUT, mCurMethod)); mInputShown = true; } } } public void hideSoftInput(IInputMethodClient client) { synchronized (mMethodMap) { if (mCurClient == null || client == null || mCurClient.client.asBinder() != client.asBinder()) { try { // We need to check if this is the current client with // focus in the window manager, to allow this call to // be made before input is started in it. if (!mIWindowManager.inputMethodClientHasFocus(client)) { Log.w(TAG, "Ignoring hideSoftInput of: " + client); return; } } catch (RemoteException e) { } } hideCurrentInputLocked(); } } void hideCurrentInputLocked() { if (mInputShown && mCurMethod != null) { executeOrSendMessage(mCurMethod, mCaller.obtainMessageO( MSG_HIDE_SOFT_INPUT, mCurMethod)); } mInputShown = false; mShowRequested = false; } public void windowGainedFocus(IInputMethodClient client, boolean viewHasFocus, int softInputMode, boolean first, int windowFlags) { synchronized (mMethodMap) { if (DEBUG) Log.v(TAG, "windowGainedFocus: " + client.asBinder() + " viewHasFocus=" + viewHasFocus + " softInputMode=#" + Integer.toHexString(softInputMode) + " first=" + first + " flags=#" + Integer.toHexString(windowFlags)); if (mCurClient == null || client == null || mCurClient.client.asBinder() != client.asBinder()) { try { // We need to check if this is the current client with // focus in the window manager, to allow this call to // be made before input is started in it. if (!mIWindowManager.inputMethodClientHasFocus(client)) { Log.w(TAG, "Ignoring focus gain of: " + client); return; } } catch (RemoteException e) { } } switch (softInputMode&WindowManager.LayoutParams.SOFT_INPUT_MASK_STATE) { case WindowManager.LayoutParams.SOFT_INPUT_STATE_UNSPECIFIED: if (!viewHasFocus || (softInputMode & WindowManager.LayoutParams.SOFT_INPUT_MASK_ADJUST) != WindowManager.LayoutParams.SOFT_INPUT_ADJUST_RESIZE) { if ((windowFlags&WindowManager.LayoutParams .FLAG_ALT_FOCUSABLE_IM) == 0) { // There is no focus view, and this window will // be behind any soft input window, then hide the // soft input window if it is shown. hideCurrentInputLocked(); } } break; case WindowManager.LayoutParams.SOFT_INPUT_STATE_UNCHANGED: // Do nothing. break; case WindowManager.LayoutParams.SOFT_INPUT_STATE_HIDDEN: hideCurrentInputLocked(); break; case WindowManager.LayoutParams.SOFT_INPUT_STATE_FIRST_VISIBLE: if (first && !viewHasFocus && (windowFlags & WindowManager.LayoutParams.FLAG_RESTORED_STATE) == 0) { showCurrentInputLocked(); } break; case WindowManager.LayoutParams.SOFT_INPUT_STATE_VISIBLE: if (viewHasFocus) { showCurrentInputLocked(); } break; } } } public void showInputMethodPickerFromClient(IInputMethodClient client) { synchronized (mMethodMap) { if (mCurClient == null || client == null || mCurClient.client.asBinder() != client.asBinder()) { Log.w(TAG, "Ignoring showInputMethodDialogFromClient of: " + client); } mHandler.sendEmptyMessage(MSG_SHOW_IM_PICKER); } } public void setInputMethod(IBinder token, String id) { synchronized (mMethodMap) { if (mCurToken == null || mCurToken != token) { Log.w(TAG, "Ignoring setInputMethod of token: " + token); } setInputMethodLocked(id); } } public void hideMySoftInput(IBinder token) { synchronized (mMethodMap) { if (mCurToken == null || mCurToken != token) { Log.w(TAG, "Ignoring hideInputMethod of token: " + token); } if (mInputShown && mCurMethod != null) { executeOrSendMessage(mCurMethod, mCaller.obtainMessageO( MSG_HIDE_SOFT_INPUT, mCurMethod)); } mInputShown = false; mShowRequested = false; } } void setEnabledSessionInMainThread(SessionState session) { if (mEnabledSession != session) { if (mEnabledSession != null) { try { if (DEBUG) Log.v(TAG, "Disabling: " + mEnabledSession); mEnabledSession.method.setSessionEnabled( mEnabledSession.session, false); } catch (RemoteException e) { } } mEnabledSession = session; try { if (DEBUG) Log.v(TAG, "Enabling: " + mEnabledSession); session.method.setSessionEnabled( session.session, true); } catch (RemoteException e) { } } } public boolean handleMessage(Message msg) { HandlerCaller.SomeArgs args; switch (msg.what) { case MSG_SHOW_IM_PICKER: showInputMethodMenu(); return true; // --------------------------------------------------------- case MSG_UNBIND_INPUT: try { ((IInputMethod)msg.obj).unbindInput(); } catch (RemoteException e) { // There is nothing interesting about the method dying. } return true; case MSG_BIND_INPUT: args = (HandlerCaller.SomeArgs)msg.obj; try { ((IInputMethod)args.arg1).bindInput((InputBinding)args.arg2); } catch (RemoteException e) { } return true; case MSG_SHOW_SOFT_INPUT: try { ((IInputMethod)msg.obj).showSoftInput(); } catch (RemoteException e) { } return true; case MSG_HIDE_SOFT_INPUT: try { ((IInputMethod)msg.obj).hideSoftInput(); } catch (RemoteException e) { } return true; case MSG_ATTACH_TOKEN: args = (HandlerCaller.SomeArgs)msg.obj; try { ((IInputMethod)args.arg1).attachToken((IBinder)args.arg2); } catch (RemoteException e) { } return true; case MSG_CREATE_SESSION: args = (HandlerCaller.SomeArgs)msg.obj; try { ((IInputMethod)args.arg1).createSession( (IInputMethodCallback)args.arg2); } catch (RemoteException e) { } return true; // --------------------------------------------------------- case MSG_START_INPUT: args = (HandlerCaller.SomeArgs)msg.obj; try { SessionState session = (SessionState)args.arg1; setEnabledSessionInMainThread(session); session.method.startInput((EditorInfo)args.arg2); } catch (RemoteException e) { } return true; case MSG_RESTART_INPUT: args = (HandlerCaller.SomeArgs)msg.obj; try { SessionState session = (SessionState)args.arg1; setEnabledSessionInMainThread(session); session.method.restartInput((EditorInfo)args.arg2); } catch (RemoteException e) { } return true; // --------------------------------------------------------- case MSG_UNBIND_METHOD: try { ((IInputMethodClient)msg.obj).onUnbindMethod(msg.arg1); } catch (RemoteException e) { // There is nothing interesting about the last client dying. } return true; case MSG_BIND_METHOD: args = (HandlerCaller.SomeArgs)msg.obj; try { ((IInputMethodClient)args.arg1).onBindMethod( (InputBindResult)args.arg2); } catch (RemoteException e) { Log.w(TAG, "Client died receiving input method " + args.arg2); } return true; } return false; } void buildInputMethodListLocked(ArrayList list, HashMap map) { list.clear(); map.clear(); PackageManager pm = mContext.getPackageManager(); Object[][] buildin = {{ DefaultInputMethod.class.getName(), DefaultInputMethod.getMetaInfo()}}; List services = pm.queryIntentServices( new Intent(InputMethod.SERVICE_INTERFACE), PackageManager.GET_META_DATA); for (int i = 0; i < services.size(); ++i) { ResolveInfo ri = services.get(i); ServiceInfo si = ri.serviceInfo; ComponentName compName = new ComponentName(si.packageName, si.name); if (!android.Manifest.permission.BIND_INPUT_METHOD.equals( si.permission)) { Log.w(TAG, "Skipping input method " + compName + ": it does not require the permission " + android.Manifest.permission.BIND_INPUT_METHOD); continue; } if (DEBUG) Log.d(TAG, "Checking " + compName); /* Built-in input methods are not currently supported... this will * need to be reworked to bring them back (all input methods must * now be published in a manifest). */ /* if (compName.getPackageName().equals( InputMethodManager.BUILDIN_INPUTMETHOD_PACKAGE)) { // System build-in input methods; String inputMethodName = null; int kbType = 0; String skbName = null; for (int j = 0; j < buildin.length; ++j) { Object[] obj = buildin[j]; if (compName.getClassName().equals(obj[0])) { InputMethodMetaInfo imp = (InputMethodMetaInfo) obj[1]; inputMethodName = imp.inputMethodName; } } InputMethodMetaInfo p = new InputMethodMetaInfo(compName, inputMethodName, ""); list.add(p); if (DEBUG) { Log.d(TAG, "Found a build-in input method " + p); } continue; } */ try { InputMethodInfo p = new InputMethodInfo(mContext, ri); list.add(p); map.put(p.getId(), p); if (DEBUG) { Log.d(TAG, "Found a third-party input method " + p); } } catch (XmlPullParserException e) { Log.w(TAG, "Unable to load input method " + compName, e); } catch (IOException e) { Log.w(TAG, "Unable to load input method " + compName, e); } } } // ---------------------------------------------------------------------- public void showInputMethodMenu() { if (DEBUG) Log.v(TAG, "Show switching menu"); hideInputMethodMenu(); final Context context = mContext; final PackageManager pm = context.getPackageManager(); String lastInputMethodId = Settings.Secure.getString(context .getContentResolver(), Settings.Secure.DEFAULT_INPUT_METHOD); if (DEBUG) Log.v(TAG, "Current IME: " + lastInputMethodId); final List immis = getEnabledInputMethodList(); int N = (immis == null ? 0 : immis.size()); mItems = new CharSequence[N]; mIms = new InputMethodInfo[N]; for (int i = 0; i < N; ++i) { InputMethodInfo property = immis.get(i); mItems[i] = property.loadLabel(pm); mIms[i] = property; } int checkedItem = 0; for (int i = 0; i < N; ++i) { if (mIms[i].getId().equals(lastInputMethodId)) { checkedItem = i; break; } } AlertDialog.OnClickListener adocl = new AlertDialog.OnClickListener() { public void onClick(DialogInterface dialog, int which) { hideInputMethodMenu(); } }; TypedArray a = context.obtainStyledAttributes(null, com.android.internal.R.styleable.DialogPreference, com.android.internal.R.attr.alertDialogStyle, 0); mDialogBuilder = new AlertDialog.Builder(context) .setTitle(com.android.internal.R.string.select_input_method) .setOnCancelListener(new OnCancelListener() { public void onCancel(DialogInterface dialog) { hideInputMethodMenu(); } }) .setIcon(a.getDrawable( com.android.internal.R.styleable.DialogPreference_dialogTitle)); a.recycle(); mDialogBuilder.setSingleChoiceItems(mItems, checkedItem, new AlertDialog.OnClickListener() { public void onClick(DialogInterface dialog, int which) { synchronized (mMethodMap) { InputMethodInfo im = mIms[which]; hideInputMethodMenu(); setInputMethodLocked(im.getId()); } } }); synchronized (mMethodMap) { mSwitchingDialog = mDialogBuilder.create(); mSwitchingDialog.getWindow().setType( WindowManager.LayoutParams.TYPE_INPUT_METHOD_DIALOG); mSwitchingDialog.show(); } } void hideInputMethodMenu() { if (DEBUG) Log.v(TAG, "Hide switching menu"); synchronized (mMethodMap) { if (mSwitchingDialog != null) { mSwitchingDialog.dismiss(); mSwitchingDialog = null; } mDialogBuilder = null; mItems = null; mIms = null; } } @Override protected void dump(FileDescriptor fd, PrintWriter pw, String[] args) { if (mContext.checkCallingPermission("android.permission.DUMP") != PackageManager.PERMISSION_GRANTED) { pw.println("Permission Denial: can't dump InputMethodManager from from pid=" + Binder.getCallingPid() + ", uid=" + Binder.getCallingUid()); return; } synchronized (mMethodMap) { final Printer p = new PrintWriterPrinter(pw); p.println("Current Input Method Manager state:"); int N = mMethodList.size(); p.println(" Input Methods:"); for (int i=0; i