Android R WindowManagerService模块(5) 焦点窗口和InputWindows的更新

adjustForImeIfNeeded();
// Toast窗口
scheduleToastWindowsTimeoutIfNeededLocked(oldFocus, newFocus);

if (mode == UPDATE_FOCUS_PLACING_SURFACES) {
pendingLayoutChanges |= FINISH_LAYOUT_REDO_ANIM;
}
return true;
}

以上方法中:

    1. 通过findFocusedWindowIfNeeded()方法寻找焦点窗口;
    1. 根据焦点窗口的变化,更新Input Target窗口;
    1. 更新全局焦点窗口对象mCurrentFocus;
    1. 根据更新的不同阶段做不同处理。

下面看下如何寻找到焦点窗口。

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;
};

该方法中,将依次根据如下条件获得焦点窗口:

    1. 如果WindowState不能接收Input事件,则不能作为焦点窗口;
    1. 如果没有前台Activity,则当前WindowState作为焦点窗口返回;
    1. 如果前台Activity是不可获焦状态,则当前WindowState作为焦点窗口返回;
    1. 如果当前WindowState由ActivityRecord管理,且该Wi

你可能感兴趣的:(程序员,架构,移动开发,android)