1 /* 2 * Copyright (C) 2007-2008 The Android Open Source Project 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.internal.view; 18 19 import static java.lang.annotation.RetentionPolicy.SOURCE; 20 21 import android.annotation.IntDef; 22 import android.annotation.Nullable; 23 import android.annotation.UnsupportedAppUsage; 24 import android.content.ComponentName; 25 import android.content.Intent; 26 import android.content.ServiceConnection; 27 import android.graphics.Matrix; 28 import android.os.IBinder; 29 import android.os.Parcel; 30 import android.os.Parcelable; 31 import android.os.UserHandle; 32 import android.view.InputChannel; 33 34 import java.lang.annotation.Retention; 35 36 /** 37 * Bundle of information returned by input method manager about a successful 38 * binding to an input method. 39 */ 40 public final class InputBindResult implements Parcelable { 41 42 @Retention(SOURCE) 43 @IntDef({ 44 ResultCode.SUCCESS_WITH_IME_SESSION, 45 ResultCode.SUCCESS_WAITING_IME_SESSION, 46 ResultCode.SUCCESS_WAITING_IME_BINDING, 47 ResultCode.SUCCESS_REPORT_WINDOW_FOCUS_ONLY, 48 ResultCode.ERROR_NULL, 49 ResultCode.ERROR_NO_IME, 50 ResultCode.ERROR_INVALID_PACKAGE_NAME, 51 ResultCode.ERROR_SYSTEM_NOT_READY, 52 ResultCode.ERROR_IME_NOT_CONNECTED, 53 ResultCode.ERROR_INVALID_USER, 54 ResultCode.ERROR_NULL_EDITOR_INFO, 55 ResultCode.ERROR_NOT_IME_TARGET_WINDOW, 56 ResultCode.ERROR_NO_EDITOR, 57 ResultCode.ERROR_DISPLAY_ID_MISMATCH, 58 ResultCode.ERROR_INVALID_DISPLAY_ID, 59 ResultCode.ERROR_INVALID_CLIENT, 60 }) 61 public @interface ResultCode { 62 /** 63 * Indicates that everything in this result object including {@link #method} is valid. 64 */ 65 int SUCCESS_WITH_IME_SESSION = 0; 66 /** 67 * Indicates that this is a temporary binding until the 68 * {@link android.inputmethodservice.InputMethodService} (IMS) establishes a valid session 69 * to {@link com.android.server.inputmethod.InputMethodManagerService} (IMMS). 70 * 71 * <p>Note that in this state the IMS is already bound to IMMS but the logical session 72 * is not yet established on top of the IPC channel.</p> 73 * 74 * <p>Some of fields such as {@link #channel} is not yet available.</p> 75 * 76 * @see android.inputmethodservice.InputMethodService##onCreateInputMethodSessionInterface() 77 **/ 78 int SUCCESS_WAITING_IME_SESSION = 1; 79 /** 80 * Indicates that this is a temporary binding until the 81 * {@link android.inputmethodservice.InputMethodService} (IMS) establishes a valid session 82 * to {@link com.android.server.inputmethod.InputMethodManagerService} (IMMS). 83 * 84 * <p>Note that in this state the IMMS has already initiated a connection to the IMS but 85 * the binding process is not completed yet.</p> 86 * 87 * <p>Some of fields such as {@link #channel} is not yet available.</p> 88 * @see android.content.ServiceConnection#onServiceConnected(ComponentName, IBinder) 89 */ 90 int SUCCESS_WAITING_IME_BINDING = 2; 91 /** 92 * Indicates that this is not intended for starting input but just for reporting window 93 * focus change from the application process. 94 * 95 * <p>All other fields do not have meaningful value.</p> 96 */ 97 int SUCCESS_REPORT_WINDOW_FOCUS_ONLY = 3; 98 /** 99 * Indicates somehow 100 * {@link 101 * com.android.server.inputmethod.InputMethodManagerService#startInputOrWindowGainedFocus} 102 * is trying to return null {@link InputBindResult}, which must never happen. 103 */ 104 int ERROR_NULL = 4; 105 /** 106 * Indicates that {@link com.android.server.inputmethod.InputMethodManagerService} 107 * recognizes no IME. 108 */ 109 int ERROR_NO_IME = 5; 110 /** 111 * Indicates that {@link android.view.inputmethod.EditorInfo#packageName} does not match 112 * the caller UID. 113 * 114 * @see android.view.inputmethod.EditorInfo#packageName 115 */ 116 int ERROR_INVALID_PACKAGE_NAME = 6; 117 /** 118 * Indicates that the system is still in an early stage of the boot process and any 3rd 119 * party application is not allowed to run. 120 * 121 * @see com.android.server.SystemService#PHASE_THIRD_PARTY_APPS_CAN_START 122 */ 123 int ERROR_SYSTEM_NOT_READY = 7; 124 /** 125 * Indicates that {@link com.android.server.inputmethod.InputMethodManagerService} tried to 126 * connect to an {@link android.inputmethodservice.InputMethodService} but failed. 127 * 128 * @see android.content.Context#bindServiceAsUser(Intent, ServiceConnection, int, UserHandle) 129 */ 130 int ERROR_IME_NOT_CONNECTED = 8; 131 /** 132 * Indicates that the caller is not the foreground user, does not have 133 * {@link android.Manifest.permission#INTERACT_ACROSS_USERS_FULL} permission, or the user 134 * specified in {@link android.view.inputmethod.EditorInfo#targetInputMethodUser} is not 135 * running. 136 */ 137 int ERROR_INVALID_USER = 9; 138 /** 139 * Indicates that the caller should have specified non-null 140 * {@link android.view.inputmethod.EditorInfo}. 141 */ 142 int ERROR_NULL_EDITOR_INFO = 10; 143 /** 144 * Indicates that the target window the client specified cannot be the IME target right now. 145 * 146 * <p>Due to the asynchronous nature of Android OS, we cannot completely avoid this error. 147 * The client should try to restart input when its {@link android.view.Window} is focused 148 * again.</p> 149 * 150 * @see com.android.server.wm.WindowManagerInternal#isInputMethodClientFocus(int, int, int) 151 */ 152 int ERROR_NOT_IME_TARGET_WINDOW = 11; 153 /** 154 * Indicates that focused view in the current window is not an editor. 155 */ 156 int ERROR_NO_EDITOR = 12; 157 /** 158 * Indicates that there is a mismatch in display ID between IME client and focused Window. 159 */ 160 int ERROR_DISPLAY_ID_MISMATCH = 13; 161 /** 162 * Indicates that current IME client is no longer allowed to access to the associated 163 * display. 164 */ 165 int ERROR_INVALID_DISPLAY_ID = 14; 166 /** 167 * Indicates that the client is not recognized by the system. 168 */ 169 int ERROR_INVALID_CLIENT = 15; 170 } 171 172 @ResultCode 173 public final int result; 174 175 /** 176 * The input method service. 177 */ 178 @UnsupportedAppUsage 179 public final IInputMethodSession method; 180 181 /** 182 * The input channel used to send input events to this IME. 183 */ 184 public final InputChannel channel; 185 186 /** 187 * The ID for this input method, as found in InputMethodInfo; null if 188 * no input method will be bound. 189 */ 190 public final String id; 191 192 /** 193 * Sequence number of this binding. 194 */ 195 public final int sequence; 196 197 @Nullable 198 private final float[] mActivityViewToScreenMatrixValues; 199 200 /** 201 * @return {@link Matrix} that corresponds to {@link #mActivityViewToScreenMatrixValues}. 202 * {@code null} if {@link #mActivityViewToScreenMatrixValues} is {@code null}. 203 */ 204 @Nullable getActivityViewToScreenMatrix()205 public Matrix getActivityViewToScreenMatrix() { 206 if (mActivityViewToScreenMatrixValues == null) { 207 return null; 208 } 209 final Matrix matrix = new Matrix(); 210 matrix.setValues(mActivityViewToScreenMatrixValues); 211 return matrix; 212 } 213 InputBindResult(@esultCode int _result, IInputMethodSession _method, InputChannel _channel, String _id, int _sequence, @Nullable Matrix activityViewToScreenMatrix)214 public InputBindResult(@ResultCode int _result, 215 IInputMethodSession _method, InputChannel _channel, String _id, int _sequence, 216 @Nullable Matrix activityViewToScreenMatrix) { 217 result = _result; 218 method = _method; 219 channel = _channel; 220 id = _id; 221 sequence = _sequence; 222 if (activityViewToScreenMatrix == null) { 223 mActivityViewToScreenMatrixValues = null; 224 } else { 225 mActivityViewToScreenMatrixValues = new float[9]; 226 activityViewToScreenMatrix.getValues(mActivityViewToScreenMatrixValues); 227 } 228 } 229 InputBindResult(Parcel source)230 InputBindResult(Parcel source) { 231 result = source.readInt(); 232 method = IInputMethodSession.Stub.asInterface(source.readStrongBinder()); 233 if (source.readInt() != 0) { 234 channel = InputChannel.CREATOR.createFromParcel(source); 235 } else { 236 channel = null; 237 } 238 id = source.readString(); 239 sequence = source.readInt(); 240 mActivityViewToScreenMatrixValues = source.createFloatArray(); 241 } 242 243 @Override toString()244 public String toString() { 245 return "InputBindResult{result=" + getResultString() + " method="+ method + " id=" + id 246 + " sequence=" + sequence 247 + " activityViewToScreenMatrix=" + getActivityViewToScreenMatrix() 248 + "}"; 249 } 250 251 /** 252 * Used to package this object into a {@link Parcel}. 253 * 254 * @param dest The {@link Parcel} to be written. 255 * @param flags The flags used for parceling. 256 */ 257 @Override writeToParcel(Parcel dest, int flags)258 public void writeToParcel(Parcel dest, int flags) { 259 dest.writeInt(result); 260 dest.writeStrongInterface(method); 261 if (channel != null) { 262 dest.writeInt(1); 263 channel.writeToParcel(dest, flags); 264 } else { 265 dest.writeInt(0); 266 } 267 dest.writeString(id); 268 dest.writeInt(sequence); 269 dest.writeFloatArray(mActivityViewToScreenMatrixValues); 270 } 271 272 /** 273 * Used to make this class parcelable. 274 */ 275 @UnsupportedAppUsage 276 public static final Parcelable.Creator<InputBindResult> CREATOR = 277 new Parcelable.Creator<InputBindResult>() { 278 @Override 279 public InputBindResult createFromParcel(Parcel source) { 280 return new InputBindResult(source); 281 } 282 283 @Override 284 public InputBindResult[] newArray(int size) { 285 return new InputBindResult[size]; 286 } 287 }; 288 289 @Override describeContents()290 public int describeContents() { 291 return channel != null ? channel.describeContents() : 0; 292 } 293 getResultString()294 public String getResultString() { 295 switch (result) { 296 case ResultCode.SUCCESS_WITH_IME_SESSION: 297 return "SUCCESS_WITH_IME_SESSION"; 298 case ResultCode.SUCCESS_WAITING_IME_SESSION: 299 return "SUCCESS_WAITING_IME_SESSION"; 300 case ResultCode.SUCCESS_WAITING_IME_BINDING: 301 return "SUCCESS_WAITING_IME_BINDING"; 302 case ResultCode.SUCCESS_REPORT_WINDOW_FOCUS_ONLY: 303 return "SUCCESS_REPORT_WINDOW_FOCUS_ONLY"; 304 case ResultCode.ERROR_NULL: 305 return "ERROR_NULL"; 306 case ResultCode.ERROR_NO_IME: 307 return "ERROR_NO_IME"; 308 case ResultCode.ERROR_NO_EDITOR: 309 return "ERROR_NO_EDITOR"; 310 case ResultCode.ERROR_INVALID_PACKAGE_NAME: 311 return "ERROR_INVALID_PACKAGE_NAME"; 312 case ResultCode.ERROR_SYSTEM_NOT_READY: 313 return "ERROR_SYSTEM_NOT_READY"; 314 case ResultCode.ERROR_IME_NOT_CONNECTED: 315 return "ERROR_IME_NOT_CONNECTED"; 316 case ResultCode.ERROR_INVALID_USER: 317 return "ERROR_INVALID_USER"; 318 case ResultCode.ERROR_NULL_EDITOR_INFO: 319 return "ERROR_NULL_EDITOR_INFO"; 320 case ResultCode.ERROR_NOT_IME_TARGET_WINDOW: 321 return "ERROR_NOT_IME_TARGET_WINDOW"; 322 case ResultCode.ERROR_DISPLAY_ID_MISMATCH: 323 return "ERROR_DISPLAY_ID_MISMATCH"; 324 case ResultCode.ERROR_INVALID_DISPLAY_ID: 325 return "ERROR_INVALID_DISPLAY_ID"; 326 case ResultCode.ERROR_INVALID_CLIENT: 327 return "ERROR_INVALID_CLIENT"; 328 default: 329 return "Unknown(" + result + ")"; 330 } 331 } 332 error(@esultCode int result)333 private static InputBindResult error(@ResultCode int result) { 334 return new InputBindResult(result, null, null, null, -1, null); 335 } 336 337 /** 338 * Predefined error object for {@link ResultCode#ERROR_NULL}. 339 */ 340 public static final InputBindResult NULL = error(ResultCode.ERROR_NULL); 341 /** 342 * Predefined error object for {@link ResultCode#NO_IME}. 343 */ 344 public static final InputBindResult NO_IME = error(ResultCode.ERROR_NO_IME); 345 /** 346 * Predefined error object for {@link ResultCode#NO_EDITOR}. 347 */ 348 public static final InputBindResult NO_EDITOR = error(ResultCode.ERROR_NO_EDITOR); 349 /** 350 * Predefined error object for {@link ResultCode#ERROR_INVALID_PACKAGE_NAME}. 351 */ 352 public static final InputBindResult INVALID_PACKAGE_NAME = 353 error(ResultCode.ERROR_INVALID_PACKAGE_NAME); 354 /** 355 * Predefined error object for {@link ResultCode#ERROR_NULL_EDITOR_INFO}. 356 */ 357 public static final InputBindResult NULL_EDITOR_INFO = error(ResultCode.ERROR_NULL_EDITOR_INFO); 358 /** 359 * Predefined error object for {@link ResultCode#ERROR_NOT_IME_TARGET_WINDOW}. 360 */ 361 public static final InputBindResult NOT_IME_TARGET_WINDOW = 362 error(ResultCode.ERROR_NOT_IME_TARGET_WINDOW); 363 /** 364 * Predefined error object for {@link ResultCode#ERROR_IME_NOT_CONNECTED}. 365 */ 366 public static final InputBindResult IME_NOT_CONNECTED = 367 error(ResultCode.ERROR_IME_NOT_CONNECTED); 368 /** 369 * Predefined error object for {@link ResultCode#ERROR_INVALID_USER}. 370 */ 371 public static final InputBindResult INVALID_USER = error(ResultCode.ERROR_INVALID_USER); 372 373 /** 374 * Predefined error object for {@link ResultCode#ERROR_DISPLAY_ID_MISMATCH}. 375 */ 376 public static final InputBindResult DISPLAY_ID_MISMATCH = 377 error(ResultCode.ERROR_DISPLAY_ID_MISMATCH); 378 379 /** 380 * Predefined error object for {@link ResultCode#ERROR_INVALID_DISPLAY_ID}. 381 */ 382 public static final InputBindResult INVALID_DISPLAY_ID = 383 error(ResultCode.ERROR_INVALID_DISPLAY_ID); 384 385 /** 386 * Predefined error object for {@link ResultCode#ERROR_INVALID_CLIENT}. 387 */ 388 public static final InputBindResult INVALID_CLIENT = error(ResultCode.ERROR_INVALID_CLIENT); 389 } 390