Android高级UI学习记录(UI的绘制流程)

上篇文章是App的启动的流程这篇学习Activity中的布局文件是如何绘制的。先从onCreate的setContentView开始。先看下Activity的绘制过程

##Activity

public void setContentView(@LayoutRes int layoutResID) {

getWindow().setContentView(layoutResID);//getWindow拿到的是PhoneWindow

initWindowDecorActionBar();

}

##PhoneWindow

public void setContentView(int layoutResID) {

...

if (mContentParent == null) {

installDecor();//初始化DecorView

} else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {

mContentParent.removeAllViews();

}

...

}

private void installDecor() {//导航栏什么的在这个方法中设置

...

mContentParent = generateDecor(-1);//创建DecorView

...

mContentParent = generateLayout(mDecor);//设置DecorView布局

...

}

protected DecorView generateDecor(int featureId) {

...

return new DecorView(context, featureId, this, getAttributes());

}

protected ViewGroup generateLayout(DecorView decor) {

// 获取AndroidManifest.xml中指定的themes主题

TypedArray a = getWindowStyle();

...

if (a.getBoolean(R.styleable.Window_windowNoTitle, false)) {

requestFeature(FEATURE_NO_TITLE);

} else if (a.getBoolean(R.styleable.Window_windowActionBar, false)) { // Don't allow an action bar if there is no title. requestFeature(FEATURE_ACTION_BAR);

} if (a.getBoolean(R.styleable.Window_windowActionBarOverlay, false)) {

requestFeature(FEATURE_ACTION_BAR_OVERLAY);

} if (a.getBoolean(R.styleable.Window_windowActionModeOverlay, false)) {

requestFeature(FEATURE_ACTION_MODE_OVERLAY);

} if (a.getBoolean(R.styleable.Window_windowSwipeToDismiss, false)) {

requestFeature(FEATURE_SWIPE_TO_DISMISS);

} if (a.getBoolean(R.styleable.Window_windowFullscreen, false)) {

setFlags(FLAG_FULLSCREEN, FLAG_FULLSCREEN & (~getForcedWindowFlags()));

} if (a.getBoolean(R.styleable.Window_windowTranslucentStatus, false)) {

setFlags(FLAG_TRANSLUCENT_STATUS, FLAG_TRANSLUCENT_STATUS & (~getForcedWindowFlags()));

} if (a.getBoolean(R.styleable.Window_windowTranslucentNavigation, false)) {

setFlags(FLAG_TRANSLUCENT_NAVIGATION, FLAG_TRANSLUCENT_NAVIGATION & (~getForcedWindowFlags()));

} if (a.getBoolean(R.styleable.Window_windowOverscan, false)) {

setFlags(FLAG_LAYOUT_IN_OVERSCAN, FLAG_LAYOUT_IN_OVERSCAN&(~getForcedWindowFlags()));

} if (a.getBoolean(R.styleable.Window_windowShowWallpaper, false)) {

setFlags(FLAG_SHOW_WALLPAPER, FLAG_SHOW_WALLPAPER&(~getForcedWindowFlags()));

} if (a.getBoolean(R.styleable.Window_windowEnableSplitTouch, getContext().getApplicationInfo().targetSdkVersion >= android.os.Build.VERSION_CODES.HONEYCOMB)) {

setFlags(FLAG_SPLIT_TOUCH, FLAG_SPLIT_TOUCH&(~getForcedWindowFlags()));

}

a.getValue(R.styleable.Window_windowMinWidthMajor, mMinWidthMajor);

a.getValue(R.styleable.Window_windowMinWidthMinor, mMinWidthMinor);

if (DEBUG) Log.d(TAG, "Min width minor: " + mMinWidthMinor.coerceToString() + ", major: " + mMinWidthMajor.coerceToString());

if (a.hasValue(R.styleable.Window_windowFixedWidthMajor)) {

if (mFixedWidthMajor == null) mFixedWidthMajor = new TypedValue(); a.getValue(R.styleable.Window_windowFixedWidthMajor, mFixedWidthMajor);

} if (a.hasValue(R.styleable.Window_windowFixedWidthMinor)) {

if (mFixedWidthMinor == null) mFixedWidthMinor = new TypedValue(); a.getValue(R.styleable.Window_windowFixedWidthMinor, mFixedWidthMinor);

} if (a.hasValue(R.styleable.Window_windowFixedHeightMajor)) {

if (mFixedHeightMajor == null) mFixedHeightMajor = new TypedValue(); a.getValue(R.styleable.Window_windowFixedHeightMajor, mFixedHeightMajor);

} if (a.hasValue(R.styleable.Window_windowFixedHeightMinor)) {

if (mFixedHeightMinor == null) mFixedHeightMinor = new TypedValue(); a.getValue(R.styleable.Window_windowFixedHeightMinor, mFixedHeightMinor);

} if (a.getBoolean(R.styleable.Window_windowContentTransitions, false)) {

requestFeature(FEATURE_CONTENT_TRANSITIONS);

} if (a.getBoolean(R.styleable.Window_windowActivityTransitions, false)) {

requestFeature(FEATURE_ACTIVITY_TRANSITIONS);

} mIsTranslucent = a.getBoolean(R.styleable.Window_windowIsTranslucent, false);

...

if (!mForcedStatusBarColor) {

mStatusBarColor = a.getColor(R.styleable.Window_statusBarColor, 0xFF000000);

} if (!mForcedNavigationBarColor) {

mNavigationBarColor = a.getColor(R.styleable.Window_navigationBarColor, 0xFF000000); mNavigationBarDividerColor = a.getColor(R.styleable.Window_navigationBarDividerColor, 0x00000000);

}

...

int layoutResource; //DecorView 的布局文件

int features = getLocalFeatures();

// System.out.println("Features: 0x" + Integer.toHexString(features));

if ((features & (1 << FEATURE_SWIPE_TO_DISMISS)) != 0) {

layoutResource = R.layout.screen_swipe_dismiss; setCloseOnSwipeEnabled(true);

} else if ((features & ((1 << FEATURE_LEFT_ICON) | (1 << FEATURE_RIGHT_ICON))) != 0) {

if (mIsFloating) {

TypedValue res = new TypedValue(); getContext().getTheme().resolveAttribute( R.attr.dialogTitleIconsDecorLayout, res, true); layoutResource = res.resourceId;

} else {

layoutResource = R.layout.screen_title_icons;

}

// XXX Remove this once action bar supports these features. removeFeature(FEATURE_ACTION_BAR);

// System.out.println("Title Icons!");

} else if ((features & ((1 << FEATURE_PROGRESS) | (1 << FEATURE_INDETERMINATE_PROGRESS))) != 0 && (features & (1 << FEATURE_ACTION_BAR)) == 0) {

// Special case for a window with only a progress bar (and title).

// XXX Need to have a no-title version of embedded windows. layoutResource = R.layout.screen_progress;

// System.out.println("Progress!");

} else if ((features & (1 << FEATURE_CUSTOM_TITLE)) != 0) {

// Special case for a window with a custom title.

// If the window is floating, we need a dialog layout

if (mIsFloating) {

TypedValue res = new TypedValue();

getContext().getTheme().resolveAttribute( R.attr.dialogCustomTitleDecorLayout, res, true);

layoutResource = res.resourceId;

} else {

layoutResource = R.layout.screen_custom_title;

}

// XXX Remove this once action bar supports these

features. removeFeature(FEATURE_ACTION_BAR);

} else if ((features & (1 << FEATURE_NO_TITLE)) == 0) {

// If no other features and not embedded, only need a title.

// If the window is floating, we need a dialog layout

if (mIsFloating) {

TypedValue res = new TypedValue();

getContext().getTheme().resolveAttribute( R.attr.dialogTitleDecorLayout, res, true);

layoutResource = res.resourceId;

} else if ((features & (1 << FEATURE_ACTION_BAR)) != 0) {

layoutResource = a.getResourceId( R.styleable.Window_windowActionBarFullscreenDecorLayout, R.layout.screen_action_bar);

} else {

layoutResource = R.layout.screen_title;

}

// System.out.println("Title!");

} else if ((features & (1 << FEATURE_ACTION_MODE_OVERLAY)) != 0) {

layoutResource = R.layout.screen_simple_overlay_action_mode;//DecorView 的布局文件配置

} else {

// Embedded, so no decoration is needed.

layoutResource = R.layout.screen_simple;//DecorView 的布局文件配置

// System.out.println("Simple!");

}

mDecor.startChanging();

mDecor.onResourcesLoaded(mLayoutInflater, layoutResource);//将自己写的布局文件解析

ViewGroup contentParent = (ViewGroup)findViewById(ID_ANDROID_CONTENT);

if (contentParent == null) {

throw new RuntimeException("Window couldn't find content container view");

} if ((features & (1 << FEATURE_INDETERMINATE_PROGRESS)) != 0) {

ProgressBar progress = getCircularProgressBar(false);

if (progress != null) { progress.setIndeterminate(true);

}

} if ((features & (1 << FEATURE_SWIPE_TO_DISMISS)) != 0) {

registerSwipeCallbacks(contentParent);

}

...

return contentParent;

}

##R.layout.screen_simple_overlay_action_mode.xml 

<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" 

 android:layout_width="match_parent" 

 android:layout_height="match_parent" 

 android:fitsSystemWindows="true">

 <FrameLayout android:id="@android:id/content" //用于添加我们自己写的布局文件

 android:layout_width="match_parent"

 android:layout_height="match_parent"

 android:foregroundInsidePadding="false" 

 android:foregroundGravity="fill_horizontal|top"

 android:foreground="?android:attr/windowContentOverlay" /> 

 <ViewStub android:id="@+id/action_mode_bar_stub" 

 android:inflatedId="@+id/action_mode_bar" 

 android:layout="@layout/action_mode_bar"

 android:layout_width="match_parent" 

 android:layout_height="wrap_content" 

 android:theme="?attr/actionBarTheme" /> 

</FrameLayout> 

##R.layout.screen_simple.xml

 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" 

 android:layout_width="match_parent"

 android:layout_height="match_parent" 

 android:fitsSystemWindows="true" 

 android:orientation="vertical"> 

 <ViewStub

 android:id="@+id/action_mode_bar_stub" 

 android:inflatedId="@+id/action_mode_bar" 

 android:layout="@layout/action_mode_bar" 

 android:layout_width="match_parent"

 android:layout_height="wrap_content" 

 android:theme="?attr/actionBarTheme" /> 

 <FrameLayout android:id="@android:id/content"  //用于添加我们自己写的布局文件

 android:layout_width="match_parent" 

 android:layout_height="match_parent"

 android:foregroundInsidePadding="false" 

 android:foregroundGravity="fill_horizontal|top" 

 android:foreground="?android:attr/windowContentOverlay" /> 

</LinearLayout>

DecorView的布局文件可以看出其实我们写的布局文件其实是通过id为content的布局进行添加的

那接下来看看是如何解析我们写的布局的

##DecorView

void onResourcesLoaded(LayoutInflater inflater, int layoutResource) {//将View树

...

mDecorCaptionView = createDecorCaptionView(inflater);

final View root = inflater.inflate(layoutResource, null);

if (mDecorCaptionView != null) {

    if (mDecorCaptionView.getParent() == null) {

    addView(mDecorCaptionView, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));

    }

    mDecorCaptionView.addView(root, new ViewGroup.MarginLayoutParams(MATCH_PARENT, MATCH_PARENT));

} else {

// Put it below the color views.

addView(root, 0, new ViewGroup.LayoutParams(MATCH_PARENT, MATCH_PARENT));

}

mContentRoot = (ViewGroup) root;

initializeElevation();

}

private DecorCaptionView createDecorCaptionView(LayoutInflater inflater) {//DecorCaptionView 布局文件在这个方法中添加

DecorCaptionView decorCaptionView = null;

for (int i = getChildCount() - 1; i >= 0 && decorCaptionView == null; i--) {

...

decorCaptionView = inflateDecorCaptionView(inflater);

...

return decorCaptionView;

}

private DecorCaptionView inflateDecorCaptionView(LayoutInflater inflater) {

final Context context = getContext();

// We make a copy of the inflater, so it has the right context associated with it.

inflater = inflater.from(context);

final DecorCaptionView view = (DecorCaptionView) inflater.inflate(R.layout.decor_caption, null);

setDecorCaptionShade(context, view); return view;

}

##R.layout.decor_caption.xml

<com.android.internal.widget.DecorCaptionView

xmlns:android="http://schemas.android.com/apk/res/android"

android:orientation="vertical"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:descendantFocusability="beforeDescendants" >

<LinearLayout android:id="@+id/caption"

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:gravity="end"

android:background="@drawable/decor_caption_title"

android:focusable="false"

android:descendantFocusability="blocksDescendants" >

<Button android:id="@+id/maximize_window"

android:layout_width="32dp"

android:layout_height="32dp"

android:layout_margin="5dp"

android:padding="4dp"

android:layout_gravity="center_vertical|end"

android:contentDescription="@string/maximize_button_text"

android:background="@drawable/decor_maximize_button_dark" />

<Button android:id="@+id/close_window"

android:layout_width="32dp"

android:layout_height="32dp"

android:layout_margin="5dp"

android:padding="4dp"

android:layout_gravity="center_vertical|end"

android:contentDescription="@string/close_button_text"

android:background="@drawable/decor_close_button_dark" />

</LinearLayout>

</com.android.internal.widget.DecorCaptionView>

这里是将view树这里也是将整个窗口的内容添加到DecorView 的根目录包括title和content,并将整个窗口内容赋值给mContentRoot

Activity的绘制流程到这里就结束了接下来看看AppCampatActivity。同样从setContentView开始

##AppCompatActivity

public void setContentView(@LayoutRes int layoutResID) {

        this.getDelegate().setContentView(layoutResID);

    }

public AppCompatDelegate getDelegate() {

        if (this.mDelegate == null) {

            this.mDelegate = AppCompatDelegate.create(this, this);

        }

        return this.mDelegate;

    }

##AppCompatDelegate

    public abstract void setContentView(@LayoutRes int var1);

##AppCompatDelegateImpl

public void setContentView(int resId) {

        this.ensureSubDecor();//创建DecorView 和 ViewGroup

//获取id为16908290的容器

        ViewGroup contentParent = (ViewGroup)this.mSubDecor.findViewById(16908290);

        contentParent.removeAllViews();

//通过布局加载器解析自己写的xml布局文件转换为View树,并把解析出来的View树添加到contentParent(ViewGroup)布局容器中

        LayoutInflater.from(this.mContext).inflate(resId, contentParent);

        this.mOriginalWindowCallback.onContentChanged();

    }

private void ensureSubDecor() {

        if (!this.mSubDecorInstalled) {

            this.mSubDecor = this.createSubDecor();

            CharSequence title = this.getTitle();//设置title

            if (!TextUtils.isEmpty(title)) {

                if (this.mDecorContentParent != null) {

                    this.mDecorContentParent.setWindowTitle(title);

                } else if (this.peekSupportActionBar() != null) {

                    this.peekSupportActionBar().setWindowTitle(title);

                } else if (this.mTitleView != null) {

                    this.mTitleView.setText(title);

                }

            }

            ...

        }

    }

private ViewGroup createSubDecor() {

        ...

        this.mWindow.getDecorView();

        LayoutInflater inflater = LayoutInflater.from(this.mContext);

        ViewGroup subDecor = null;

            if (!this.mWindowNoTitle) {//根据主题加载不同布局

                if (this.mIsFloating) {

                    subDecor = (ViewGroup)inflater.inflate(layout.abc_dialog_title_material, (ViewGroup)null);

                    this.mHasActionBar = this.mOverlayActionBar = false;

                } else if (this.mHasActionBar) {

                    ...

                    subDecor = (ViewGroup)LayoutInflater.from((Context)themedContext).inflate(layout.abc_screen_toolbar, (ViewGroup)null);

                    ...

                }

            } else {

                if (this.mOverlayActionMode) {

                    subDecor = (ViewGroup)inflater.inflate(layout.abc_screen_simple_overlay_action_mode, (ViewGroup)null);

                } else {

                    subDecor = (ViewGroup)inflater.inflate(layout.abc_screen_simple, (ViewGroup)null);

                }

                ...

            }

            if (subDecor == null) {

                throw new IllegalArgumentException("AppCompat does not support the current theme features: { windowActionBar: " + this.mHasActionBar + ", windowActionBarOverlay: " + this.mOverlayActionBar + ", android:windowIsFloating: " + this.mIsFloating + ", windowActionModeOverlay: " + this.mOverlayActionMode + ", windowNoTitle: " + this.mWindowNoTitle + " }");

            } else {

                if (this.mDecorContentParent == null) {

                    this.mTitleView = (TextView)subDecor.findViewById(id.title);

                }

                ViewUtils.makeOptionalFitsSystemWindows(subDecor);//这里用放射调用View的makeOptionalFitsSystemWindows方法该方法为设置Flag

                ContentFrameLayout contentView = (ContentFrameLayout)subDecor.findViewById(id.action_bar_activity_content);

                ViewGroup windowContentView = (ViewGroup)this.mWindow.findViewById(16908290);

                if (windowContentView != null) {

                    while(windowContentView.getChildCount() > 0) {

                        View child = windowContentView.getChildAt(0);

                        windowContentView.removeViewAt(0);

                        contentView.addView(child);

                    }

                    windowContentView.setId(-1);

                    contentView.setId(16908290);

                    if (windowContentView instanceof FrameLayout) {

                        ((FrameLayout)windowContentView).setForeground((Drawable)null);

                    }

                }

                this.mWindow.setContentView(subDecor);//这里mWindow的setContentView具体实现为PhoneWindow的setContentView

                ...

                return subDecor;

            }

        }

    }

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

        // Note: FEATURE_CONTENT_TRANSITIONS may be set in the process of installing the window

        // decor, when theme attributes and the like are crystalized. Do not check the feature

        // before this happens.

        if (mContentParent == null) {

            installDecor();//这里和activity的调用是一样的就不继续了

        } else if (!hasFeature(FEATURE_CONTENT_TRANSITIONS)) {

            mContentParent.removeAllViews();

        }

        if (hasFeature(FEATURE_CONTENT_TRANSITIONS)) {

            view.setLayoutParams(params);

            final Scene newScene = new Scene(mContentParent, view);

            transitionTo(newScene);

        } else {

            mContentParent.addView(view, params);

        }

        mContentParent.requestApplyInsets();

        final Callback cb = getCallback();

        if (cb != null && !isDestroyed()) {

            cb.onContentChanged();

        }

        mContentParentExplicitlySet = true;

    }

AppComapActivity只是在Acitivity的installDecor()调用逻辑之前多了一些配置支持Material design风格

到这里UI绘制流程就结束了。

备注:文中Android源码版本9.0

作者:Dean_Xu

原创博客,请注明转载处....

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,254评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,875评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,682评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,896评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,015评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,152评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,208评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,962评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,388评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,700评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,867评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,551评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,186评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,901评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,689评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,757评论 2 351

推荐阅读更多精彩内容