一、概述
相信你已经知道,Android 可使用 XML 标签语言进行界面的定义。每个标签中有一个一个的属性,这些属性有相应的属性值。例如:
1 2 3 4 5 6 7
| <cn.neillee.composedmenu.RotatingArcMenu android:id="@+id/ram2" android:layout_width="wrap_content" android:layout_height="wrap_content" android:background="?attr/barBgColor" app:position="left_bottom" app:radius="150dp"/>
|
有两点可以注意到:
二、详细介绍
2.1 自定义属性
这里介绍
1 2
| app:position="left_bottom" app:radius="150dp"
|
的使用。
自定义属性常见于自定义的 View 中,让我们还是以概述中的代码作为例子。自定义属性及其属性值在 /values/attr.xml
中有如下定义:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| <?xml version="1.0" encoding="utf-8"?> <resources> <attr name="position"> <enum name="left_top" value="0"/> <enum name="left_bottom" value="1"/> <enum name="right_top" value="2"/> <enum name="right_bottom" value="3"/> </attr> <attr name="radius" format="dimension"/> <declare-styleable name="ComposedMenu"> <attr name="position"/> <attr name="radius"/> </declare-styleable> </resources>
|
在该文件中,枚举了 position
有四个属性值,并为每个属性值提供了不同的 value 值作区分。而 radius
定义为 dimension
引用,表示其值须为 dimension
类型的值。类似地,还有如有其他类型,参考文章 【Android】Android自定义属性,attr format取值类型
接下来如何获取自定义属性值就成了关键。我在自定义 ViewGroup 中使用如下代码获取到用户在 layout 文件中,自定义 ViewGroup 标签下使用到的值。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| private static final int LEFT_TOP = 0; private static final int LEFT_BOTTOM = 1; private static final int RIGHT_TOP = 2; private static final int RIGHT_BOTTOM = 3; protected static final int DEFAULT_RADIUS = 150; protected static final int DEFAULT_POSITION = RIGHT_BOTTOM; protected int mRadius; ... public RotatingArcMenu(Context context, AttributeSet attrs, int defStyleAttr) { TypedArray a = context.getTheme().obtainStyledAttributes(attrs, R.styleable.ComposedMenu, defStyleAttr, 0); int pos = a.getInt(R.styleable.ComposedMenu_position, DEFAULT_POSITION); mRadius = (int) a.getDimension(R.styleable.ComposedMenu_radius, TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, DEFAULT_RADIUS, getResources().getDisplayMetrics())); a.recycle();// 使用完后记得回收 }
|
2.2 获取系统属性
这里介绍 android:background="?attr/barBgColor"
属性的定义与获取。
我之所以如此定义,是为了使得 背景色 能够随着应用的主题切换而变化,最简单的例子就是 夜间模式。
首先,我在 values/attr.xml
文件中对 barBgColor
进行定义:
1 2 3 4 5
| <?xml version="1.0" encoding="utf-8"?> <resources> ... <attr name="barBgColor" format="color"/> </resources>
|
其次,在 values/styles.xml
中对该属性值进行了定义:
1 2 3 4 5 6 7 8 9
| <resources> <style name="AppDayTheme" parent="Theme.AppCompat.Light.NoActionBar"> <item name="barBgColor">@color/ZHIHUBlue</item> </style> <style name="AppNightTheme" parent="Theme.AppCompat.Light.NoActionBar"> <item name="barBgColor">#263238</item> </style> </resources>
|
最后,我在控件的属性中对该属性值进行了使用
1 2 3 4 5 6 7
| <cn.neillee.composedmenu.RotatingArcMenu android:id="@+id/ram2" android:layout_width="wrap_content" android:layout_height="wrap_content" android:background="?attr/barBgColor" app:position="left_bottom" app:radius="150dp"/>
|
需要说明的是,可以不再控件属性中使用自定义的属性值,可在代码中进行获取与设置。如:
1 2 3
| TypedValue typedValue = new TypedValue(); mContext.getTheme().resolveAttribute(R.attr.barBgColor, typedValue, true); mRotatingArcMenu.setBackgroundColor(typedValue.data);
|
这里使用到了 TypedValue
这个类。
2.3 关于 TypedValue
详细介绍见官网:
https://developer.android.com/reference/android/util/TypedValue.html
此处我仅对我在上一小节中的使用做简单介绍。在使用过程中,我观察到在不同主题下,TypedValue 读取到的值如下:
在 DayTheme 中,由于我定义的是 @color/ZHIHUBlue
,对color的引用即 resourceId。因此,typedValue.resourceId 有值,且直接等效于 R.color.ZHIHUBlue
,typedValue.type=TYPE_INT_COLOR_RGB8,表示 typedValue.data 有值,为8位的颜色代码值(#rrggbb)。
在 NightTheme 中,由于我定义的是 #263238
,颜色代码值。因此,TypedValue.resourceId 无值,但 typedValue.type=TYPE_INT_COLOR_RGB8,表示 typedValue.data 有值,为8位的颜色代码(#rrggbb)。