【Android】attr、style和theme

一、Attr

属性,风格样式的最小单元;

Attr 的定义

在自定义 View 的时候,在 res/attrs.xml 文件中声明属性,而Android 系统的属性也是以同样的方式定义的。
比如 layout_width 属性对应到框架中的 attr 如下:

<declare-styleable name="ViewGroup_Layout">
    <attr name="layout_width" format="dimension">
        <enum name="fill_parent" value="-1" />
        <enum name="match_parent" value="-1" />
        <enum name="wrap_content" value="-2" />
    </attr>
    ...
</declare-styleable>

attr 的 format 有以下几种格式,可以进行或运算:
color、reference、boolean、dimension、enum、flag、float、fraction、integer、string
这里着重说一下 enum 是枚举值,而 flag 可以进行或运算,属性值可以叠加使用,
reference 用在一些可以设置引用值的情况,引用 res 资源
fraction 是百分数的意思

二、Style

风格,它是一系列Attr的集合,用以定义一个View的样式;
style 是定义在 res/styles.xml 文件中的,在控件中使用时只需要style="@style/style_name"就可以使用样式了。
我们在自定义 View 时通过
·obtainStyledAttributes( AttributeSet set, @StyleableRes int[] attrs, @AttrRes int defStyleAttr, @StyleRes int defStyleRes)
方法获取自定义的 attr ,其中第一个参数表示是 xml 文件解析后得到的属性集合,attrs 是 View 声明的属性集,后两个用作指定默认的 Style,表示如果 set 中没有你想获得的属性,但如果你指定了默认 Style,它会去从该默认的 Style 里面找你想要的属性。defStyleAttr 和 defStyleRes 功能一样,指定的资源形式不同,前者表示一个默认的指向一个 style 风格的 attr 属性,而后者你可以直接传入一个 style 风格的 id。

三、Theme

主题,它与Style作用一样,都是一系列属性的集合,不同于Style作用于一个单独View,而它是作用于Activity上或是整个应用;与 Window 有关的属性作用于 Window,与 View 有关的属性作用于 Activity 的所有 View 或者整个应用的所有 View,如果想要改变 Window 的属性,那么继承相应 Theme,重写属性值,如果想要改变 View 相关样式,在 Theme 中重写属性作用于整个 Activity 或整个应用,或者定义 style 作用于单个 View。
Theme 的实质也是 Style,Theme 的定义格式与 Style 的基本一致,Theme 需要设置到 AndroidManifest.xml 的 <application> 或者 <activity> 标签下,设置后,被设置的 Activity 或整个应用下所有的 View 都可以使用该 <style> 里面的属性了。
style 除过有 name 属性以外还可以有 parent 属性,一般在 style 中使用 parent 字段的继承适用于继承系统平台现有定义的 style,想要继承自己实现的 style 一般不会通过parent字段来实现,而是通过指定格式的 name 字段来实现。
Theme 中以 windowXXX 开头的属性就不适用于 View,但是不会报错,只是 View 会忽略这些不适合自己的属性,应用适合自己的属性。

四、Theme 兼容性处理

Google官方提供Android Support Library package来保证高版本SDK的向下兼容。

  • v4 Support Library
    此包用在API lever 4(即Android 1.6)及更高版本之上。它包含了较多的内容,使用非常广泛,例如:Fragment,NotificationCompat,LoadBroadcastManager,ViewPager,PageTabStrip,Loader,FileProvider 等。
  • v7 Support Libraries
    此包是针对API level 7(即Android 2.1)及以上版本而设计的,但是v7是要依赖v4这个包的,v7支持了Action Bar以及一些Theme的兼容。
    v7 appcompat library 是包含在 v7 Support Libraries里面的一个包,正是此包增加了Action Bar 用户界面的设计模式,并加入了对material design 的支持,是我们使用最多的一个兼容包。

Android 系统有三个基本主题:

  • android:Theme
  • android:Theme.Holo
  • android:Theme.Material
  • Theme.AppCompat

android:Theme是所有主题的超级父类。所有的主题都是直接继承或者间接继承它。
android:Theme.Holo 从 Api 11开始可以使用。android:Theme.Material 从 Api 21开始可以使用。
如果要在不同版本的系统上用各自的主题,比如在4.0之下的系统用android:Theme,4.0至5.0的系统用Holo主题,5.0及之后的系统使用Material Design,那我们需要建不同的value-vX目录。在各自的目录中的style继承相应的系统主题。在运行是系统就会根据平台版本使用相应的主题。
如果只创建一个 value 文件的话,那么系统的版本可能会不支持 value 中继承的主题,系统就会根据 App 指定的 targetSdkVersion 自动设置主题,如果设置的 targetSdkVersion 还是高于系统的版本,系统就设置为支持的最高系统 sdk 版本的主题。
具体是怎么选择的呢?我们来看 Resources 类中的源码

 public static int selectDefaultTheme(int curTheme, int targetSdkVersion) {
        return selectSystemTheme(curTheme, targetSdkVersion,
                com.android.internal.R.style.Theme,
                com.android.internal.R.style.Theme_Holo,
                com.android.internal.R.style.Theme_DeviceDefault,
                com.android.internal.R.style.Theme_DeviceDefault_Light_DarkActionBar);
    }

    /** @hide */
    public static int selectSystemTheme(int curTheme, int targetSdkVersion, int orig, int holo,
            int dark, int deviceDefault) {
        if (curTheme != 0) {
            return curTheme;
        }
        if (targetSdkVersion < Build.VERSION_CODES.HONEYCOMB) {
            return orig;
        }
        if (targetSdkVersion < Build.VERSION_CODES.ICE_CREAM_SANDWICH) {
            return holo;
        }
        if (targetSdkVersion < Build.VERSION_CODES.CUR_DEVELOPMENT) {
            return dark;
        }
        return deviceDefault;
    }

我们可以看到当 targetSdkVersion 低于11时,选择 Theme 主题、当 targetSdkVersion 低于 14 时,选择 Theme.Holo 主题,当 targetSdkVersion 高于 14 时,选择 Theme.DeviceDefault,而 Theme.DeviceDefault 继承自 Theme.Material。
如果我们想在 5.0 以下的系统中使用 Material 主题,只需要让我们的系统主题继承Theme.AppCompat即可。当然需要导入support-v7 appcompat 包,可以在sdk\extras\android\support\v7\appcompat 查看到。使用了 Theme.AppCompat 之后,就不受 targetSdkVersion 影响了。
AppCompatActivity(包括 ActionBarActivity)的子 Activity 必须使用 Theme.AppCompat 或 Theme.AppCompat 的子主题,如果不是,编译时不会给出任何警告,但运行时会抛出异常。

五、Android 系统预制的 Theme 和 Style

我们想要知道系统的 Theme 有哪些属性可以使用,系统为我们提供了哪些 Style 可以使用,可以到 sdk\platforms\android-24\data\res\values 文件夹下去查看 styles.xml 和 theme.xml 文件中去查看,以 Theme 为例,一个完整的主题需要定义颜色、Text 样式、Button 样式、List 属性、Window 属性、Dialog 属性、AlertDialog 属性、Panel 属性、ScrollBar 属性、文字选中属性、Widget 样式、Preference (设置类界面)样式、Search Widget 样式、ActionBar 样式等。我们可以发现这些属性的值有些是具体的值,有些是引用的 android:color、android:drawable 等系统资源,有些是引用的系统中所定义的样式 android:style,有些则是直接引用系统主题中所定义的其它属性?android:attr
1、我们可以根据这些属性修改系统提供的控件样式和窗体样式;
2、我们可以引用系统主题的属性值;

六、Android应用资源拓展语法

Theme 和 Style 的实质都属于 res 资源,和其它资源类型如 drawable、layout、dimen 等的引用方式是一样的。
Android 中资源在Java文件中引用的语法定义如下:

[包名.]R.<资源类型.><资源名>
//当资源在当前应用中则包名可以省略,当为系统的资源则可换为 android.

Android 资源在XML文件中引用的语法定义如下:

@[包名:]<资源类型/><资源名>
//当资源在当前应用中则包名可以省略,当为系统的资源则换为 android.

Android系统预制资源在XML文件中引用的特殊语法定义如下:

//可以引用系统所有资源,public & private
@*android:type/name
//只能引用系统public的资源
@android:type/name

Android在XML文件中引用当前主题属性的语法定义如下:
资源值允许引用当前主题中的属性的值,这个属性值只能在style资源和XML中使用,随着当前主题的切换该值也在变换

?[<包名>:][<资源类型>/]<资源名>
//如果是本应用中的attr使用,则可以省去<package_name>部分。

Android在XML文件中创建或者引用资源语法定义如下:

//在R.java的type内部类中添加一条静态常量id资源标识符,如果标示符(包括系统资源)已经存在则表示引用该标示符。
@+type/name

//在R.java中寻找已经定义的标识符,如果找不到则提示失败错误,一般在xml中定义有先后关系。
@type/name

Android在XML文件中xmlns语法定义如下:

xmlns:namespace-prefix=http://schemas.android.com/apk/res/应用程序包路径

参考:
Attr、Style和Theme详解
Android主题和样式之系统篇(上)
Android主题和样式之系统篇(下)
Android开发之Theme、Style探索及源码浅析
Android关于Theme.AppCompat相关问题的深入分析
Android的系统主题总结

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

推荐阅读更多精彩内容