android源碼之WMS分析筆記

1.SystemServer:啟動(dòng)WindowManagerService

public static WindowManagerService main(final Context context,

final PowerManagerService pm, final DisplayManagerService dm,

final InputManagerService im, final Handler wmHandler,

final boolean haveInputMethods, final boolean showBootMsgs,

final boolean onlyCore) {

final WindowManagerService[] holder = new WindowManagerService[1];

wmHandler.runWithScissors(new Runnable() {

@Override

public void run() {

holder[0] = new WindowManagerService(context, pm, dm, im,

haveInputMethods, showBootMsgs, onlyCore);

}

}, 0);

return holder[0];

}

2.WindowManagerService的初始化

private WindowManagerService(Context context, PowerManagerService pm,

DisplayManagerService displayManager, InputManagerService inputManager,

boolean haveInputMethods, boolean showBootMsgs, boolean onlyCore) {

mContext = context;

mHaveInputMethods = haveInputMethods;

mAllowBootMessages = showBootMsgs;

mOnlyCore = onlyCore;

mLimitedAlphaCompositing = context.getResources().getBoolean(

com.android.internal.R.bool.config_sf_limitedAlpha);

/* 得到InputManagerService抑淫。輸入事件最終要分發(fā)給具有焦點(diǎn)的窗口嗤堰,由WMS負(fù)責(zé)與IMS交互

mInputManager = inputManager; // Must be before createDisplayContentLocked.

mDisplayManagerService = displayManager;

mHeadless = displayManager.isHeadless();

mDisplaySettings = new DisplaySettings(context);

mDisplaySettings.readSettingsLocked();

mPointerEventDispatcher = new PointerEventDispatcher(mInputManager.monitorInput(TAG));

mFxSession = new SurfaceSession();

/* 初始化DisplayContent列表寻馏。DisplayContent是Android4.2為支持多屏幕輸出所引入的一個(gè)

概念婉称。一個(gè)DisplayContent指代一塊屏幕柒莉,屏幕可以是手機(jī)自身的屏幕祭隔,也可以是基于Wi-FiDisplay

技術(shù)的虛擬屏幕[3]*/

mDisplayManager = (DisplayManager)context.getSystemService(Context.DISPLAY_SERVICE);

mDisplayManager.registerDisplayListener(this, null);

Display[] displays = mDisplayManager.getDisplays();

for (Display display : displays) {

createDisplayContentLocked(display);

}

mKeyguardDisableHandler = new KeyguardDisableHandler(mContext, mPolicy);

/* 電源管理模塊當(dāng)有關(guān)屏亮屏動(dòng)作的時(shí)候压储,需要WindowManagerPolicy去做相關(guān)的系統(tǒng)UI顯示動(dòng)作 */

mPowerManager = pm;

mPowerManager.setPolicy(mPolicy);

PowerManager pmc = (PowerManager)context.getSystemService(Context.POWER_SERVICE);

mScreenFrozenLock = pmc.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "SCREEN_FROZEN");

mScreenFrozenLock.setReferenceCounted(false);

mAppTransition = new AppTransition(context, mH);

mActivityManager = ActivityManagerNative.getDefault();

mBatteryStats = BatteryStatsService.getService();

mAppOps = (AppOpsManager)context.getSystemService(Context.APP_OPS_SERVICE);

mAppOps.startWatchingMode(AppOpsManager.OP_SYSTEM_ALERT_WINDOW, null,

new AppOpsManager.OnOpChangedInternalListener() {

@Override

public void onOpChanged(int op, String packageName) {

updateAppOpsState();

}

}

);

// 初始化全局系統(tǒng)設(shè)置的窗口切換動(dòng)畫的值

mWindowAnimationScale = Settings.Global.getFloat(context.getContentResolver(),

Settings.Global.WINDOW_ANIMATION_SCALE, mWindowAnimationScale);

mTransitionAnimationScale = Settings.Global.getFloat(context.getContentResolver(),

Settings.Global.TRANSITION_ANIMATION_SCALE, mTransitionAnimationScale);

setAnimatorDurationScale(Settings.Global.getFloat(context.getContentResolver(),

Settings.Global.ANIMATOR_DURATION_SCALE, mAnimatorDurationScale));

// Track changes to DevicePolicyManager state so we can enable/disable keyguard.

IntentFilter filter = new IntentFilter();

filter.addAction(DevicePolicyManager.ACTION_DEVICE_POLICY_MANAGER_STATE_CHANGED);

filter.addAction("jidou.app.action.AUTO_REBOOT");

mContext.registerReceiver(mBroadcastReceiver, filter);

mHoldingScreenWakeLock = pmc.newWakeLock(PowerManager.SCREEN_BRIGHT_WAKE_LOCK

| PowerManager.ON_AFTER_RELEASE, TAG);

mHoldingScreenWakeLock.setReferenceCounted(false);

//管理窗口動(dòng)畫對(duì)象

mAnimator = new WindowAnimator(this);

//實(shí)例化WindowManagerPolicy,實(shí)際就是初始化PhoneWindowManager

initPolicy(UiThread.getHandler());

// Add ourself to the Watchdog monitors.

Watchdog.getInstance().addMonitor(this);

SurfaceControl.openTransaction();

try {

createWatermarkInTransaction();

mFocusedStackFrame = new FocusedStackFrame(

getDefaultDisplayContentLocked().getDisplay(), mFxSession);

} finally {

SurfaceControl.closeTransaction();

}

}

3.初始化PhoneWindowManager

①.WindowManagerService對(duì)象創(chuàng)建的時(shí)候會(huì)通過(guò)PolicyManager的靜態(tài)方法makeNewWindowManager()得到類型為IPolicy的mPolicy全局變量;

②.在PolicyManager類加載中德频,會(huì)通過(guò)反射的方式創(chuàng)建類型為IPolicy的 sPolicy對(duì)象苍息,

③ IPolicy是一個(gè)接口文件,具體實(shí)現(xiàn)在為framework/base/policy/src/com/android/internal/policy/impl/Policy.java類,所以最終PolicyManager.makeNewWindowManager() 調(diào)用的就是Policy中的makeNewWindowManager(),返回的是一個(gè)PhoneWindowManager對(duì)象竞思。

④ initPolicy方法所做的動(dòng)作就是執(zhí)行PhoneWindowManager中的init方法表谊,做一些對(duì)象初始化動(dòng)作,以及廣播的監(jiān)聽盖喷,配置參數(shù)讀取等爆办。

4.PhoneWindowManager初始化完成之后,繼續(xù)回來(lái)WindowManagerServivce中传蹈,相繼執(zhí)行displayReady()押逼,systemReady()兩個(gè)函數(shù),然后等待客戶端請(qǐng)求添加窗口惦界。

5.接下來(lái)開始添加系統(tǒng)窗口挑格,以添加NavigationBar為例,在SystemServer中沾歪,資源加載完成后漂彤,啟動(dòng)系統(tǒng)級(jí)應(yīng)用SystemUI

SystemUI中有眾多的組件,繼承自SystemUI.java抽象類灾搏,這里不再贅述挫望,我直接查找NavigationBar的創(chuàng)建流程。

①.SystemUIService在其生命周期的onCreate函數(shù)中狂窑,通過(guò)反射的方式媳板,創(chuàng)建SystemBars組件的對(duì)象,然后調(diào)用其start()方法

②.在SystemBars的start方法中泉哈,實(shí)例化ServiceMonitor蛉幸,并調(diào)用它的start()方法,最終會(huì)走到SystemBars中onNoService中丛晦,通過(guò)反射的方式奕纫,初始化父類為BaseStatusBar的PhoneStatusBar對(duì)象,然后執(zhí)行該對(duì)象的start()方法烫沙。

③.在PhoneStatusBar中匹层,添加NavigationBar的邏輯代碼片段

//配置NavigationBar視圖

private void prepareNavigationBarView() {

mNavigationBarView.reorient();

mNavigationBarView.getRecentsButton().setOnClickListener(mRecentsClickListener);

mNavigationBarView.getRecentsButton().setOnTouchListener(mRecentsPreloadOnTouchListener);

mNavigationBarView.getHomeButton().setOnTouchListener(mHomeSearchActionListener);

mNavigationBarView.getSearchLight().setOnTouchListener(mHomeSearchActionListener);

mNavigationBarView.setBackgroundDrawable(new ColorDrawable(Color.TRANSPARENT));

//.....

}

//設(shè)置NavigationBar的窗口信息

private WindowManager.LayoutParams getNavigationBarLayoutParams() {

WindowManager.LayoutParams lp = new WindowManager.LayoutParams(

LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT,

WindowManager.LayoutParams.TYPE_NAVIGATION_BAR,

0

| WindowManager.LayoutParams.FLAG_TOUCHABLE_WHEN_WAKING

| WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE

| WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL

| WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH

| WindowManager.LayoutParams.FLAG_SPLIT_TOUCH,

PixelFormat.TRANSLUCENT);

// this will allow the navbar to run in an overlay on devices that support this

if (ActivityManager.isHighEndGfx()) {

lp.flags |= WindowManager.LayoutParams.FLAG_HARDWARE_ACCELERATED;

}

lp.setTitle("NavigationBar");

lp.windowAnimations = 0;

return lp;

}

private void addNavigationBar() {

if (DEBUG) Log.v(TAG, "addNavigationBar: about to add " + mNavigationBarView);

if (mNavigationBarView == null) return;

prepareNavigationBarView();

if (mNavigationBarView.isAttachedToWindow()) return;

mWindowManager.addView(mNavigationBarView, getNavigationBarLayoutParams());

}

分析代碼可知,NavigationBar是一個(gè)類型為WindowManager.LayoutParams.TYPE_NAVIGATION_BAR的視圖窗口锌蓄,通過(guò)WindowManager添加到視圖系統(tǒng)中升筏。

④繼續(xù)走,mWindowManager是在BaseStatusBar中初始化的瘸爽,通過(guò)源碼跟蹤您访,我們發(fā)現(xiàn)得到的其實(shí)為WindowManagerImpl.java,

//在源碼core/java/android/app/ContextImpl.java中

registerService(WINDOW_SERVICE, new ServiceFetcher() {

Display mDefaultDisplay;

public Object getService(ContextImpl ctx) {

Display display = ctx.mDisplay;

if (display == null) {

if (mDefaultDisplay == null) {

DisplayManager dm = (DisplayManager)ctx.getOuterContext().

getSystemService(Context.DISPLAY_SERVICE);

mDefaultDisplay = dm.getDisplay(Display.DEFAULT_DISPLAY);

}

display = mDefaultDisplay;

}

return new WindowManagerImpl(display);

}});

所以mWindowManager.addView相當(dāng)于執(zhí)行了如下代碼。

public void addView(View view, ViewGroup.LayoutParams params) {

mGlobal.addView(view, params, mDisplay, mParentWindow);

}

這里的mDisplay為默認(rèn)的蝶糯,mParentWindow對(duì)象為NULL洋只。

mGlobal為WindowManagerGlobal對(duì)象的實(shí)例,于是添加視圖走到了該類的addView方法中,代碼如下:

public void addView(View view, ViewGroup.LayoutParams params, Display display, Window parentWindow) {

if (view == null) { //不允許添加空對(duì)象的視圖

throw new IllegalArgumentException("view must not be null");

}

if (display == null) {

throw new IllegalArgumentException("display must not be null");

}

if (!(params instanceof WindowManager.LayoutParams)) {

throw new IllegalArgumentException("Params must be WindowManager.LayoutParams");

}

final WindowManager.LayoutParams wparams = (WindowManager.LayoutParams)params;

if (parentWindow != null) { //添加NavigationBar時(shí)识虚,這里我們的parentWindow是為空

parentWindow.adjustLayoutParamsForSubWindow(wparams);

}

ViewRootImpl root;

View panelParentView = null;

synchronized (mLock) {

// Start watching for system property changes.

if (mSystemPropertyUpdater == null) {

mSystemPropertyUpdater = new Runnable() {

@Override public void run() {

synchronized (mLock) {

for (int i = mRoots.size() - 1; i >= 0; --i) {

mRoots.get(i).loadSystemProperties();

}

}

}

};

SystemProperties.addChangeCallback(mSystemPropertyUpdater);

}

int index = findViewLocked(view, false);

if (index >= 0) {

if (mDyingViews.contains(view)) {

// Don't wait for MSG_DIE to make it's way through root's queue.

mRoots.get(index).doDie();

} else {

throw new IllegalStateException("View " + view

+ " has already been added to the window manager.");

}

// The previous removeView() had not completed executing. Now it has.

}

// If this is a panel window, then find the window it is being

// attached to for future reference.

if (wparams.type >= WindowManager.LayoutParams.FIRST_SUB_WINDOW &&

wparams.type <= WindowManager.LayoutParams.LAST_SUB_WINDOW) {

final int count = mViews.size();

for (int i = 0; i < count; i++) {

if (mRoots.get(i).mWindow.asBinder() == wparams.token) {

panelParentView = mViews.get(i);

}

}

}

root = new ViewRootImpl(view.getContext(), display);

view.setLayoutParams(wparams);

mViews.add(view);

mRoots.add(root);

mParams.add(wparams);

}

// do this last because it fires off messages to start doing things

try {

root.setView(view, wparams, panelParentView);

} catch (RuntimeException e) {

// BadTokenException or InvalidDisplayException, clean up.

synchronized (mLock) {

final int index = findViewLocked(view, false);

if (index >= 0) {

removeViewLocked(index, true);

}

}

throw e;

}

}

⑤ 這里創(chuàng)建一個(gè)ViewRootImlp對(duì)象,然后調(diào)用了它的setView方法肢扯,傳遞三個(gè)參數(shù),一個(gè)正在添加的視圖view担锤,一個(gè)視圖的相關(guān)參數(shù)蔚晨,一個(gè)父類容器view,這里為空

⑥ 接著往下肛循,可以看到setView中铭腕,有這樣一句話

//.......

try {

mOrigWindowType = mWindowAttributes.type;

mAttachInfo.mRecomputeGlobalAttributes = true;

collectViewAttributes();

res = mWindowSession.addToDisplay(mWindow, mSeq, mWindowAttributes,

getHostVisibility(), mDisplay.getDisplayId(),

mAttachInfo.mContentInsets, mInputChannel);

} catch (RemoteException e) {

mAdded = false;

mView = null;

mAttachInfo.mRootView = null;

mInputChannel = null;

mFallbackEventHandler.setView(null);

unscheduleTraversals();

setAccessibilityFocus(null, null);

throw new RuntimeException("Adding window failed", e);

} finally {

if (restore) {

attrs.restore();

}

}

//.......

⑦ mWindowSession.addToDisplay實(shí)際就是調(diào)用了Session.java中的addToDisplay,

@Override

public int addToDisplay(IWindow window, int seq, WindowManager.LayoutParams attrs,

int viewVisibility, int displayId, Rect outContentInsets,

InputChannel outInputChannel) {

return mService.addWindow(this, window, seq, attrs, viewVisibility, displayId,

outContentInsets, outInputChannel);

}

最終addToDisplay被執(zhí)行到了WindowManagerService的addWindow函數(shù)中

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末多糠,一起剝皮案震驚了整個(gè)濱河市累舷,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌夹孔,老刑警劉巖被盈,帶你破解...
    沈念sama閱讀 216,496評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異搭伤,居然都是意外死亡只怎,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,407評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門怜俐,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)身堡,“玉大人,你說(shuō)我怎么就攤上這事拍鲤√眩” “怎么了?”我有些...
    開封第一講書人閱讀 162,632評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵殿漠,是天一觀的道長(zhǎng)赴精。 經(jīng)常有香客問(wèn)我佩捞,道長(zhǎng)绞幌,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,180評(píng)論 1 292
  • 正文 為了忘掉前任一忱,我火速辦了婚禮莲蜘,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘帘营。我一直安慰自己票渠,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,198評(píng)論 6 388
  • 文/花漫 我一把揭開白布芬迄。 她就那樣靜靜地躺著问顷,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上杜窄,一...
    開封第一講書人閱讀 51,165評(píng)論 1 299
  • 那天肠骆,我揣著相機(jī)與錄音,去河邊找鬼塞耕。 笑死蚀腿,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的扫外。 我是一名探鬼主播莉钙,決...
    沈念sama閱讀 40,052評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼筛谚!你這毒婦竟也來(lái)了磁玉?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,910評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤驾讲,失蹤者是張志新(化名)和其女友劉穎蜀涨,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蝎毡,經(jīng)...
    沈念sama閱讀 45,324評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡厚柳,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,542評(píng)論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了沐兵。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片别垮。...
    茶點(diǎn)故事閱讀 39,711評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖扎谎,靈堂內(nèi)的尸體忽然破棺而出碳想,到底是詐尸還是另有隱情,我是刑警寧澤毁靶,帶...
    沈念sama閱讀 35,424評(píng)論 5 343
  • 正文 年R本政府宣布胧奔,位于F島的核電站,受9級(jí)特大地震影響预吆,放射性物質(zhì)發(fā)生泄漏龙填。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,017評(píng)論 3 326
  • 文/蒙蒙 一拐叉、第九天 我趴在偏房一處隱蔽的房頂上張望岩遗。 院中可真熱鬧,春花似錦凤瘦、人聲如沸宿礁。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,668評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)梆靖。三九已至控汉,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間返吻,已是汗流浹背暇番。 一陣腳步聲響...
    開封第一講書人閱讀 32,823評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留思喊,地道東北人壁酬。 一個(gè)月前我還...
    沈念sama閱讀 47,722評(píng)論 2 368
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像恨课,于是被迫代替她去往敵國(guó)和親舆乔。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,611評(píng)論 2 353

推薦閱讀更多精彩內(nèi)容