Merge "Move startInput for WINDOW_FOCUS_GAIN to background thread"
This commit is contained in:
@@ -92,7 +92,10 @@ import java.util.Comparator;
|
||||
import java.util.List;
|
||||
import java.util.Map;
|
||||
import java.util.Objects;
|
||||
import java.util.concurrent.CancellationException;
|
||||
import java.util.concurrent.CompletableFuture;
|
||||
import java.util.concurrent.CountDownLatch;
|
||||
import java.util.concurrent.ExecutionException;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
/**
|
||||
@@ -421,6 +424,13 @@ public final class InputMethodManager {
|
||||
int mCursorCandStart;
|
||||
int mCursorCandEnd;
|
||||
|
||||
/**
|
||||
* Initial startInput with {@link StartInputReason.WINDOW_FOCUS_GAIN} is executed
|
||||
* in a background thread. Later, if there is an actual startInput it will wait on
|
||||
* main thread till the background thread completes.
|
||||
*/
|
||||
private CompletableFuture<Void> mWindowFocusGainFuture;
|
||||
|
||||
/**
|
||||
* The instance that has previously been sent to the input method.
|
||||
*/
|
||||
@@ -1598,6 +1608,18 @@ public final class InputMethodManager {
|
||||
boolean startInputInner(@StartInputReason int startInputReason,
|
||||
@Nullable IBinder windowGainingFocus, @StartInputFlags int startInputFlags,
|
||||
@SoftInputModeFlags int softInputMode, int windowFlags) {
|
||||
if (startInputReason != StartInputReason.WINDOW_FOCUS_GAIN
|
||||
&& mWindowFocusGainFuture != null) {
|
||||
try {
|
||||
mWindowFocusGainFuture.get();
|
||||
} catch (ExecutionException | InterruptedException e) {
|
||||
// do nothing
|
||||
} catch (CancellationException e) {
|
||||
// window no longer has focus.
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
final View view;
|
||||
synchronized (mH) {
|
||||
view = mServedView;
|
||||
@@ -1947,31 +1969,38 @@ public final class InputMethodManager {
|
||||
}
|
||||
}
|
||||
|
||||
if (checkFocusNoStartInput(forceNewFocus)) {
|
||||
// We need to restart input on the current focus view. This
|
||||
// should be done in conjunction with telling the system service
|
||||
// about the window gaining focus, to help make the transition
|
||||
// smooth.
|
||||
if (startInputInner(StartInputReason.WINDOW_FOCUS_GAIN, rootView.getWindowToken(),
|
||||
startInputFlags, softInputMode, windowFlags)) {
|
||||
return;
|
||||
}
|
||||
final boolean forceNewFocus1 = forceNewFocus;
|
||||
final int startInputFlags1 = startInputFlags;
|
||||
if (mWindowFocusGainFuture != null) {
|
||||
mWindowFocusGainFuture.cancel(false/* mayInterruptIfRunning */);
|
||||
}
|
||||
mWindowFocusGainFuture = CompletableFuture.runAsync(() -> {
|
||||
if (checkFocusNoStartInput(forceNewFocus1)) {
|
||||
// We need to restart input on the current focus view. This
|
||||
// should be done in conjunction with telling the system service
|
||||
// about the window gaining focus, to help make the transition
|
||||
// smooth.
|
||||
if (startInputInner(StartInputReason.WINDOW_FOCUS_GAIN, rootView.getWindowToken(),
|
||||
startInputFlags1, softInputMode, windowFlags)) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
// For some reason we didn't do a startInput + windowFocusGain, so
|
||||
// we'll just do a window focus gain and call it a day.
|
||||
synchronized (mH) {
|
||||
try {
|
||||
if (DEBUG) Log.v(TAG, "Reporting focus gain, without startInput");
|
||||
mService.startInputOrWindowGainedFocus(
|
||||
StartInputReason.WINDOW_FOCUS_GAIN_REPORT_ONLY, mClient,
|
||||
rootView.getWindowToken(), startInputFlags, softInputMode, windowFlags,
|
||||
null, null, 0 /* missingMethodFlags */,
|
||||
rootView.getContext().getApplicationInfo().targetSdkVersion);
|
||||
} catch (RemoteException e) {
|
||||
throw e.rethrowFromSystemServer();
|
||||
// For some reason we didn't do a startInput + windowFocusGain, so
|
||||
// we'll just do a window focus gain and call it a day.
|
||||
synchronized (mH) {
|
||||
try {
|
||||
if (DEBUG) Log.v(TAG, "Reporting focus gain, without startInput");
|
||||
mService.startInputOrWindowGainedFocus(
|
||||
StartInputReason.WINDOW_FOCUS_GAIN_REPORT_ONLY, mClient,
|
||||
rootView.getWindowToken(), startInputFlags1, softInputMode, windowFlags,
|
||||
null, null, 0 /* missingMethodFlags */,
|
||||
rootView.getContext().getApplicationInfo().targetSdkVersion);
|
||||
} catch (RemoteException e) {
|
||||
throw e.rethrowFromSystemServer();
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
/** @hide */
|
||||
@@ -1986,6 +2015,10 @@ public final class InputMethodManager {
|
||||
// If the mCurRootView is losing window focus, release the strong reference to it
|
||||
// so as not to prevent it from being garbage-collected.
|
||||
mCurRootView = null;
|
||||
if (mWindowFocusGainFuture != null) {
|
||||
mWindowFocusGainFuture.cancel(false /* mayInterruptIfRunning */);
|
||||
mWindowFocusGainFuture = null;
|
||||
}
|
||||
} else {
|
||||
if (DEBUG) {
|
||||
Log.v(TAG, "Ignoring onPreWindowFocus()."
|
||||
|
||||
Reference in New Issue
Block a user