1 /* 2 * Copyright (C) 2020 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 android.inputmethodservice; 18 19 import static android.inputmethodservice.InputMethodService.DEBUG; 20 21 import android.annotation.MainThread; 22 import android.annotation.NonNull; 23 import android.annotation.Nullable; 24 import android.os.Bundle; 25 import android.os.Handler; 26 import android.os.IBinder; 27 import android.os.Looper; 28 import android.os.RemoteException; 29 import android.util.Log; 30 import android.view.autofill.AutofillId; 31 import android.view.inputmethod.EditorInfo; 32 import android.view.inputmethod.InlineSuggestionsRequest; 33 import android.view.inputmethod.InlineSuggestionsResponse; 34 35 import com.android.internal.inputmethod.IInlineSuggestionsRequestCallback; 36 import com.android.internal.inputmethod.InlineSuggestionsRequestInfo; 37 38 import java.util.function.Consumer; 39 import java.util.function.Function; 40 import java.util.function.Supplier; 41 42 /** 43 * Manages the interaction with the autofill manager service for the inline suggestion sessions. 44 * 45 * <p> 46 * The class maintains the inline suggestion session with the autofill service. There is at most one 47 * active inline suggestion session at any given time. 48 * 49 * <p> 50 * The class receives the IME status change events (input start/finish, input view start/finish, and 51 * show input requested result), and send them through IPC to the {@link 52 * com.android.server.inputmethod.InputMethodManagerService}, which sends them to {@link 53 * com.android.server.autofill.InlineSuggestionSession} in the Autofill manager service. If there is 54 * no open inline suggestion session, no event will be send to autofill manager service. 55 * 56 * <p> 57 * All the methods are expected to be called from the main thread, to ensure thread safety. 58 */ 59 class InlineSuggestionSessionController { 60 private static final String TAG = "InlineSuggestionSessionController"; 61 62 private final Handler mMainThreadHandler = new Handler(Looper.getMainLooper(), null, true); 63 64 @NonNull 65 private final Function<Bundle, InlineSuggestionsRequest> mRequestSupplier; 66 @NonNull 67 private final Supplier<IBinder> mHostInputTokenSupplier; 68 @NonNull 69 private final Consumer<InlineSuggestionsResponse> mResponseConsumer; 70 71 /* The following variables track the IME status */ 72 @Nullable 73 private String mImeClientPackageName; 74 @Nullable 75 private AutofillId mImeClientFieldId; 76 private boolean mImeInputStarted; 77 private boolean mImeInputViewStarted; 78 79 @Nullable 80 private InlineSuggestionSession mSession; 81 InlineSuggestionSessionController( @onNull Function<Bundle, InlineSuggestionsRequest> requestSupplier, @NonNull Supplier<IBinder> hostInputTokenSupplier, @NonNull Consumer<InlineSuggestionsResponse> responseConsumer)82 InlineSuggestionSessionController( 83 @NonNull Function<Bundle, InlineSuggestionsRequest> requestSupplier, 84 @NonNull Supplier<IBinder> hostInputTokenSupplier, 85 @NonNull Consumer<InlineSuggestionsResponse> responseConsumer) { 86 mRequestSupplier = requestSupplier; 87 mHostInputTokenSupplier = hostInputTokenSupplier; 88 mResponseConsumer = responseConsumer; 89 } 90 91 /** 92 * Called upon IME receiving a create inline suggestion request. Must be called in the main 93 * thread to ensure thread safety. 94 */ 95 @MainThread onMakeInlineSuggestionsRequest(@onNull InlineSuggestionsRequestInfo requestInfo, @NonNull IInlineSuggestionsRequestCallback callback)96 void onMakeInlineSuggestionsRequest(@NonNull InlineSuggestionsRequestInfo requestInfo, 97 @NonNull IInlineSuggestionsRequestCallback callback) { 98 if (DEBUG) Log.d(TAG, "onMakeInlineSuggestionsRequest: " + requestInfo); 99 // Creates a new session for the new create request from Autofill. 100 if (mSession != null) { 101 mSession.invalidate(); 102 } 103 mSession = new InlineSuggestionSession(requestInfo, callback, mRequestSupplier, 104 mHostInputTokenSupplier, mResponseConsumer, this, mMainThreadHandler); 105 106 // If the input is started on the same view, then initiate the callback to the Autofill. 107 // Otherwise wait for the input to start. 108 if (mImeInputStarted && match(mSession.getRequestInfo())) { 109 mSession.makeInlineSuggestionRequestUncheck(); 110 // ... then update the Autofill whether the input view is started. 111 if (mImeInputViewStarted) { 112 try { 113 mSession.getRequestCallback().onInputMethodStartInputView(); 114 } catch (RemoteException e) { 115 Log.w(TAG, "onInputMethodStartInputView() remote exception:" + e); 116 } 117 } 118 } 119 } 120 121 /** 122 * Called from IME main thread before calling {@link InputMethodService#onStartInput(EditorInfo, 123 * boolean)}. This method should be quick as it makes a unblocking IPC. 124 */ 125 @MainThread notifyOnStartInput(@ullable String imeClientPackageName, @Nullable AutofillId imeFieldId)126 void notifyOnStartInput(@Nullable String imeClientPackageName, 127 @Nullable AutofillId imeFieldId) { 128 if (DEBUG) Log.d(TAG, "notifyOnStartInput: " + imeClientPackageName + ", " + imeFieldId); 129 if (imeClientPackageName == null || imeFieldId == null) { 130 return; 131 } 132 mImeInputStarted = true; 133 mImeClientPackageName = imeClientPackageName; 134 mImeClientFieldId = imeFieldId; 135 136 if (mSession != null) { 137 mSession.consumeInlineSuggestionsResponse(InlineSuggestionSession.EMPTY_RESPONSE); 138 // Initiates the callback to Autofill if there is a pending matching session. 139 // Otherwise updates the session with the Ime status. 140 if (!mSession.isCallbackInvoked() && match(mSession.getRequestInfo())) { 141 mSession.makeInlineSuggestionRequestUncheck(); 142 } else if (mSession.shouldSendImeStatus()) { 143 try { 144 mSession.getRequestCallback().onInputMethodStartInput(mImeClientFieldId); 145 } catch (RemoteException e) { 146 Log.w(TAG, "onInputMethodStartInput() remote exception:" + e); 147 } 148 } 149 } 150 } 151 152 /** 153 * Called from IME main thread after getting results from 154 * {@link InputMethodService#dispatchOnShowInputRequested(int, 155 * boolean)}. This method should be quick as it makes a unblocking IPC. 156 */ 157 @MainThread notifyOnShowInputRequested(boolean requestResult)158 void notifyOnShowInputRequested(boolean requestResult) { 159 if (DEBUG) Log.d(TAG, "notifyShowInputRequested"); 160 if (mSession != null && mSession.shouldSendImeStatus()) { 161 try { 162 mSession.getRequestCallback().onInputMethodShowInputRequested(requestResult); 163 } catch (RemoteException e) { 164 Log.w(TAG, "onInputMethodShowInputRequested() remote exception:" + e); 165 } 166 } 167 } 168 169 /** 170 * Called from IME main thread before calling 171 * {@link InputMethodService#onStartInputView(EditorInfo, 172 * boolean)} . This method should be quick as it makes a unblocking IPC. 173 */ 174 @MainThread notifyOnStartInputView()175 void notifyOnStartInputView() { 176 if (DEBUG) Log.d(TAG, "notifyOnStartInputView"); 177 mImeInputViewStarted = true; 178 if (mSession != null && mSession.shouldSendImeStatus()) { 179 try { 180 mSession.getRequestCallback().onInputMethodStartInputView(); 181 } catch (RemoteException e) { 182 Log.w(TAG, "onInputMethodStartInputView() remote exception:" + e); 183 } 184 } 185 } 186 187 /** 188 * Called from IME main thread before calling 189 * {@link InputMethodService#onFinishInputView(boolean)}. 190 * This method should be quick as it makes a unblocking IPC. 191 */ 192 @MainThread notifyOnFinishInputView()193 void notifyOnFinishInputView() { 194 if (DEBUG) Log.d(TAG, "notifyOnFinishInputView"); 195 mImeInputViewStarted = false; 196 if (mSession != null && mSession.shouldSendImeStatus()) { 197 try { 198 mSession.getRequestCallback().onInputMethodFinishInputView(); 199 } catch (RemoteException e) { 200 Log.w(TAG, "onInputMethodFinishInputView() remote exception:" + e); 201 } 202 } 203 } 204 205 /** 206 * Called from IME main thread before calling {@link InputMethodService#onFinishInput()}. This 207 * method should be quick as it makes a unblocking IPC. 208 */ 209 @MainThread notifyOnFinishInput()210 void notifyOnFinishInput() { 211 if (DEBUG) Log.d(TAG, "notifyOnFinishInput"); 212 mImeClientPackageName = null; 213 mImeClientFieldId = null; 214 mImeInputViewStarted = false; 215 mImeInputStarted = false; 216 if (mSession != null && mSession.shouldSendImeStatus()) { 217 try { 218 mSession.getRequestCallback().onInputMethodFinishInput(); 219 } catch (RemoteException e) { 220 Log.w(TAG, "onInputMethodFinishInput() remote exception:" + e); 221 } 222 } 223 } 224 225 /** 226 * Returns true if the current Ime focused field matches the session {@code requestInfo}. 227 */ 228 @MainThread match(@ullable InlineSuggestionsRequestInfo requestInfo)229 boolean match(@Nullable InlineSuggestionsRequestInfo requestInfo) { 230 return match(requestInfo, mImeClientPackageName, mImeClientFieldId); 231 } 232 233 /** 234 * Returns true if the current Ime focused field matches the {@code autofillId}. 235 */ 236 @MainThread match(@ullable AutofillId autofillId)237 boolean match(@Nullable AutofillId autofillId) { 238 return match(autofillId, mImeClientFieldId); 239 } 240 match( @ullable InlineSuggestionsRequestInfo inlineSuggestionsRequestInfo, @Nullable String imeClientPackageName, @Nullable AutofillId imeClientFieldId)241 private static boolean match( 242 @Nullable InlineSuggestionsRequestInfo inlineSuggestionsRequestInfo, 243 @Nullable String imeClientPackageName, @Nullable AutofillId imeClientFieldId) { 244 if (inlineSuggestionsRequestInfo == null || imeClientPackageName == null 245 || imeClientFieldId == null) { 246 return false; 247 } 248 return inlineSuggestionsRequestInfo.getComponentName().getPackageName().equals( 249 imeClientPackageName) && match(inlineSuggestionsRequestInfo.getAutofillId(), 250 imeClientFieldId); 251 } 252 match(@ullable AutofillId autofillId, @Nullable AutofillId imeClientFieldId)253 private static boolean match(@Nullable AutofillId autofillId, 254 @Nullable AutofillId imeClientFieldId) { 255 // The IME doesn't have information about the virtual view id for the child views in the 256 // web view, so we are only comparing the parent view id here. This means that for cases 257 // where there are two input fields in the web view, they will have the same view id 258 // (although different virtual child id), and we will not be able to distinguish them. 259 return autofillId != null && imeClientFieldId != null 260 && autofillId.getViewId() == imeClientFieldId.getViewId(); 261 } 262 } 263