• 前言
  • SystemUI功能介绍
  • SystemUI主要视图
  • PhoneStatusBarView
  • PanelHolder
  • keyguard_bouncer
  • 总结


前言

Android ROM开发过程中,难免会涉及到对SystemUI的修改,之前做过一些这方面的工作,现在整理下,准备按照如下章节介绍SystemUI.借此对SystemUI做下整体性回顾。
-SystemUI之功能介绍和UI布局实现
- SystemUI之启动流程
- SystemUI之常见需求/Bug整理
- SystemUI之为我所用
该系列的博文都会基于Android L的代码分析。本篇主写SystemUI界面构成以及代码的整体框架。

SystemUI功能介绍

手机中的下拉状态栏,锁屏,通知以及最近打开任务列表等功能都是SystemUI实现的。Android L对SystemUI做的更新比较大,体现在

- Notification 和 Quick Settings 被合并在一个界面中

- KeyGuard和SystemUI联系更紧密了,KeyGuard作为一个java lib库供SystemUI调用

- Recents App采用了叠加立体式显示效果,界面更加绚丽。

SystemUI的代码结构如下图:

安卓system用户su 安卓systemui_安卓system用户su


主要功能点对应的界面如下图所示:

安卓system用户su 安卓systemui_加载_02

SystemUI主要视图

SystemUI的根视图是在PhoneStatusBar的makeStatusBarView()方法中加载的。


<code class="hljs cs has-numbering"><span class="hljs-keyword">protected</span> PhoneStatusBarView <span class="hljs-title">makeStatusBarView</span>() {    mStatusBarWindow = (StatusBarWindowView) View.inflate(context,
                    R.layout.super_status_bar, <span class="hljs-keyword">null</span>);
}</code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li></ul>


通过对super_status_bar.xml文件的认识,就能了解SystemUI的大体视图构成。下面贴出省略后的该文件代码片段:

<code class="hljs avrasm has-numbering"><!-- This is the combined status bar / notification panel window. --><<span class="hljs-keyword">com</span><span class="hljs-preprocessor">.android</span><span class="hljs-preprocessor">.systemui</span><span class="hljs-preprocessor">.statusbar</span><span class="hljs-preprocessor">.phone</span><span class="hljs-preprocessor">.StatusBarWindowView</span>>
    <<span class="hljs-keyword">com</span><span class="hljs-preprocessor">.android</span><span class="hljs-preprocessor">.systemui</span><span class="hljs-preprocessor">.statusbar</span><span class="hljs-preprocessor">.BackDropView</span>
            android:id=<span class="hljs-string">"@+id/backdrop"</span>>
        <ImageView android:id=<span class="hljs-string">"@+id/backdrop_back"</span> />
        <ImageView android:id=<span class="hljs-string">"@+id/backdrop_front"</span>/>
    </<span class="hljs-keyword">com</span><span class="hljs-preprocessor">.android</span><span class="hljs-preprocessor">.systemui</span><span class="hljs-preprocessor">.statusbar</span><span class="hljs-preprocessor">.BackDropView</span>>

    <<span class="hljs-keyword">com</span><span class="hljs-preprocessor">.android</span><span class="hljs-preprocessor">.systemui</span><span class="hljs-preprocessor">.statusbar</span><span class="hljs-preprocessor">.ScrimView</span> android:id=<span class="hljs-string">"@+id/scrim_behind"</span>
        />

    <<span class="hljs-keyword">com</span><span class="hljs-preprocessor">.android</span><span class="hljs-preprocessor">.systemui</span><span class="hljs-preprocessor">.statusbar</span><span class="hljs-preprocessor">.AlphaOptimizedView</span>
        android:id=<span class="hljs-string">"@+id/heads_up_scrim"</span>/>

    <include layout=<span class="hljs-string">"@layout/status_bar"</span>/>

    <FrameLayout android:id=<span class="hljs-string">"@+id/brightness_mirror"</span>>
        <FrameLayout         android:background=<span class="hljs-string">"@drawable/brightness_mirror_background"</span>>
        <include layout=<span class="hljs-string">"@layout/quick_settings_brightness_dialog"</span>/>
        </FrameLayout>
    </FrameLayout>

    <<span class="hljs-keyword">com</span><span class="hljs-preprocessor">.android</span><span class="hljs-preprocessor">.systemui</span><span class="hljs-preprocessor">.statusbar</span><span class="hljs-preprocessor">.phone</span><span class="hljs-preprocessor">.PanelHolder</span>
        android:id=<span class="hljs-string">"@+id/panel_holder"</span>>
        <include layout=<span class="hljs-string">"@layout/status_bar_expanded"</span>/>
    </<span class="hljs-keyword">com</span><span class="hljs-preprocessor">.android</span><span class="hljs-preprocessor">.systemui</span><span class="hljs-preprocessor">.statusbar</span><span class="hljs-preprocessor">.phone</span><span class="hljs-preprocessor">.PanelHolder</span>>

    <<span class="hljs-keyword">com</span><span class="hljs-preprocessor">.android</span><span class="hljs-preprocessor">.systemui</span><span class="hljs-preprocessor">.statusbar</span><span class="hljs-preprocessor">.ScrimView</span>   android:id=<span class="hljs-string">"@+id/scrim_in_front"</span>/>

</<span class="hljs-keyword">com</span><span class="hljs-preprocessor">.android</span><span class="hljs-preprocessor">.systemui</span><span class="hljs-preprocessor">.statusbar</span><span class="hljs-preprocessor">.phone</span><span class="hljs-preprocessor">.StatusBarWindowView</span>></code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li><li>13</li><li>14</li><li>15</li><li>16</li><li>17</li><li>18</li><li>19</li><li>20</li><li>21</li><li>22</li><li>23</li><li>24</li><li>25</li><li>26</li><li>27</li><li>28</li><li>29</li><li>30</li></ul>


上述view从layout看主要分为三块。

- include进来的status_bar布局,也就是PhoneStatusBarView。 
 - PanelHolder 
 - ScrimView

安卓system用户su 安卓systemui_Android_03


其实这里还漏掉了一个重要的view—-keyguard_bouncer,它不是直接在layout布局里加入的,只有用户设置锁屏保护后才可见。至于ScrimView我在开发中没有修改过,具体作用不甚清楚,在此不做介绍了,以免误人子弟。

PhoneStatusBarView

PhoneStatusBarView主要用来显示系统状态、通知等,主要包括 notification icons 和 status bar icons

下面是PhoneStatusBarView的view 树形图:

安卓system用户su 安卓systemui_android_04

PanelHolder

PanelHolder是用户下拉 status bar 后得到的 view。它主要包含 QuickSettings 和 Notification panel 两个部分。PanelHolder是一个继承自FrameLayout的自定义view,它的内容是通过include status_bar_expanded.xml进行填充的。PanelHolder的布局比较复杂,为了提高view的重用性大量的使用了include标签。下面是PanelHolder的view树形图, 只给出了了主要的view:

安卓system用户su 安卓systemui_安卓system用户su_05

keyguard_bouncer

先来看看keyguard_bouncer是个什么样。

安卓system用户su 安卓systemui_android_06


需要注意的是keyguard_bouncer view 有多种形式,这里设置的是图案解锁,如果设置的是密码解锁keyguard_bouncer就会以数字键盘的形式显示出来。但无论是哪种解锁模式,都是在KeyguardBouncer类中加载进来的。

<code class="hljs java has-numbering"><span class="hljs-javadoc">/** * A class which manages the bouncer on the lockscreen. */</span><span class="hljs-keyword">public</span> <span class="hljs-class"><span class="hljs-keyword">class</span> <span class="hljs-title">KeyguardBouncer</span> {</span>
    <span class="hljs-keyword">private</span> ViewGroup mRoot;
    <span class="hljs-keyword">private</span> ViewGroup mContainer;

    <span class="hljs-keyword">private</span> <span class="hljs-keyword">void</span> <span class="hljs-title">inflateView</span>() {
        mRoot = (ViewGroup) LayoutInflater.from(mContext).inflate(R.layout.keyguard_bouncer, <span class="hljs-keyword">null</span>);
        mContainer.addView(mRoot, mContainer.getChildCount());
    }
}</code><ul style="display: block;" class="pre-numbering"><li>1</li><li>2</li><li>3</li><li>4</li><li>5</li><li>6</li><li>7</li><li>8</li><li>9</li><li>10</li><li>11</li><li>12</li></ul>


关于keyguard_bouncer在什么时候被加载的,加载的具体过程如何,后续在分析,本篇着重分析keyguard_bouncer视图的构成。

下面是keyguard_bouncer的view 树形图:

安卓system用户su 安卓systemui_加载_07

总结

以上对SystemUI的主要视图做了介绍,SystemUI的布局还是很复杂的,上述只对主要的视图从大的方向上做了分析,以后碰到具体的SystemUI view显示问题,可以先定位出问题View属于哪个大的分类,然后结合图例给出的id缩小定位范围。