}
// 为IME窗口进行调整
adjustForImeIfNeeded();
// Toast窗口
scheduleToastWindowsTimeoutIfNeededLocked(oldFocus, newFocus);
if (mode == UPDATE_FOCUS_PLACING_SURFACES) {
pendingLayoutChanges |= FINISH_LAYOUT_REDO_ANIM;
}
return true;
}
以上方法中:
-
- 通过findFocusedWindowIfNeeded()方法寻找焦点窗口;
-
- 根据焦点窗口的变化,更新Input Target窗口;
-
- 更新全局焦点窗口对象mCurrentFocus;
-
- 根据更新的不同阶段做不同处理。
下面看下如何寻找到焦点窗口。
1.4.DisplayContent#findFocusedWindowIfNeeded()
在这个方法中,会寻找到新的焦点窗口:
// frameworks/base/services/core/java/com/android/server/wm/DisplayContent.java
WindowState findFocusedWindowIfNeeded(int topFocusedDisplayId) {
return (mWmService.mPerDisplayFocusEnabled || topFocusedDisplayId == INVALID_DISPLAY)
? findFocusedWindow() : null;
}
当topFocusedDisplayId为INVALID_DISPLAY时,认为当前焦点display没有焦点窗口,需要寻找重新确认,所以又继续执行findFocusedWindow()方法寻找:
// frameworks/base/services/core/java/com/android/server/wm/DisplayContent.java
WindowState findFocusedWindow() {
mTmpWindow = null;
// 遍历WindowState
forAllWindows(mFindFocusedWindow, true /* traverseTopToBottom */);
if (mTmpWindow == null) {
return null;
}
return mTmpWindow;
}
该方法中,会遍历所有WindowState,然后将寻找到的WindowState赋值给mTmpWindow,并返回给WMS。接下来看下这个mFindFocusedWindow函数接口对象:
// frameworks/base/services/core/java/com/android/server/wm/DisplayContent.java
private final ToBooleanFunction mFindFocusedWindow = w -> {
// 当前处于前台的ActivityRecord
final ActivityRecord focusedApp = mFocusedApp;
// 如果窗口无法接收key事件,则不能作为焦点窗口,返回false
if (!w.canReceiveKeys()) {
return false;
}
final ActivityRecord activity = w.mActivityRecord;
// 如果前台没有Activity,则此次WindowState将作为焦点窗口返回
if (focusedApp == null) {
mTmpWindow = w;
return true;
}
// 如果前台Activity是不可获焦的,则此次WindowState将作为焦点窗口返回
if (!focusedApp.windowsAreFocusable()) {
mTmpWindow = w;
return true;
}
// 如果当前WindowState由ActivityRecord管理,且非StartingWindow,则当
if (activity != null && w.mAttrs.type != TYPE_APPLICATION_STARTING) {
if (focusedApp.compareTo(activity) > 0) {
mTmpWindow = null;
return true;
}
}
// 不满足以上条件,则此次WindowState将作为焦点窗口返回
mTmpWindow = w;
return true;
};
该方法中,将依次根据如下条件获得焦点窗口:
-
- 如果WindowState不能接收Input事件,则不能作为焦点窗口;
-
- 如果没有前台Activity,则当前WindowState作为焦点窗口返回