背景:
我们在开发过程,常常在布局中引入一些定义好的View,这些view需要一些属性值,这些值是怎么来的?我们是否可以自己定义一些自己的属性值,答案是可以的。
这时候我们可以借助declare-styleable来实现,一般这个declare-styleable出现在自定义view比较多。
declare-styleable是什么?
它是一种资源定义,通过xml管理
declare-styleable如何使用?
1.xml资源的新建
declare-styleable是一种资源,所以在项目中应该出现资源文件夹下values,在资源文件夹下新建一个自己的declare-styleable的xml管理文件,attr,
<?xml version="1.0" encoding="utf-8"?>
<resources>
<declare-styleable name="CircleImage">
<attr name="hasBorder" format="boolean"/>
<attr name="borderWidth" format="dimension"/>
<attr name="borderColor" format="color"/>
</declare-styleable>
</resources>
根节点是resources。
declare-styleable节点有一个自己的name,name下方是attr 的数组节点,我们可以定义我们想要的参数和类型。
<attr name="borderWidth" format="dimension"/>
name=参数名,format=类型
format都有哪些类型?
color:颜色值 dimension:dimen值 integer:整型 boolean:布尔型 string:字符串 enum:枚举 flags:标签数组,value int型, float:浮点 fraction:百分数 reference:引用类型,需要搭配资源类型
二、小试牛刀
1.自定义数据类型
<declare-styleable name="testAttr">
<attr name="colors" format="color" />
<attr name="age" format="integer" />
<attr name="mheights" format="float" />
<attr name="name" format="string" />
<attr name="flags" format="flags">
<flag name="f1" value="1" />
<flag name="f2" value="2" />
<flag name="f3" value="3" />
</attr>
<attr name="enums" format="enum">
<enum name="a" value="97" />
<enum name="b" value="98" />
<enum name="c" value="99" />
<enum name="d" value="100" />
</attr>
<attr name="sex" format="boolean" />
<attr name="size" format="reference|dimension" />
<attr name="length" format="dimension" />
<attr name="per" format="fraction"/>
</declare-styleable>
以上我们定义了一组styleable,name为testAttr
2.在layout进行引用
<com.example.wiik.testdemo.view.TestAttrView
xmlns:app="http://schemas.android.com/apk/res-auto"
android:layout_width="match_parent"
android:layout_height="match_parent"
app:age="10"
app:colors="@color/color_222222"
app:enums="c"
app:flags="f1"
app:mheights="180.0"
app:length="@dimen/android_public_space_23px"
app:name="zhangshan"
app:sex="true"
app:size="@dimen/android_public_space_20dp"
app:per="80%"
/>
如果我们需要引用资源自定义,需要引进:
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:加自定义名称(app)
如果该类型是enum或者flags,该值已定义好了,直接选择就行,且这两个参数的值只能是int类型否则会报错
error: invalid value 'true' for <flag>; must be an integer.
error: invalid value 'true' for <enum>; must be an integer.
这个和我们经常用的一些官方的View设置参数很像,
<attr name="visibility">
<enum name="visible" value="0" />
<enum name="invisible" value="1" />
<enum name="gone" value="2" />
</attr>
<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 申明了aatr的类型,在使用中,也应该指定类型,否则会报错
这里面特别说明的地方是:fraction
fraction是百分比类型,分为base和pbase,
app:per="80%p"是pbase意思,
app:per="80%"是base
以上两种会在解析的时候进行详细介绍
3.解析
private void initAttrs(Context context, AttributeSet attrs) {
TypedArray typed = context.obtainStyledAttributes(attrs, R.styleable.testAttr);
if (typed == null)
return;
int color = typed.getColor(R.styleable.testAttr_colors, 0);
boolean sex = typed.getBoolean(R.styleable.testAttr_sex, false);
int age = typed.getInt(R.styleable.testAttr_age, 0);
float height = typed.getFloat(R.styleable.testAttr_mheights, 0);
String name = typed.getString(R.styleable.testAttr_name);
// int flag = typed.getInt(R.styleable.testAttr_flags, 0);
int enums = typed.getInt(R.styleable.testAttr_enums, -1);
float size = typed.getDimension(R.styleable.testAttr_size, -1);
float length = typed.getDimension(R.styleable.testAttr_length, -1);
float per = typed.getFraction(R.styleable.testAttr_per, 1, 1, 0);
int type = typed.getType(R.styleable.testAttr_flags);
String typeMsg = "";
if (type == TypedValue.TYPE_STRING) {
typeMsg = "TYPE_STRING";
} else if (type == TypedValue.TYPE_FLOAT) {
typeMsg = "TYPE_FLOAT";
} else if (type == TypedValue.TYPE_INT_HEX) {
typeMsg = "TYPE_INT_HEX";
}
//资源释放
typed.recycle();
}
3.1解析一般先获取TypedArray
TypedArray typed = context.obtainStyledAttributes(attrs, R.styleable.testAttr);
然后根据自己的定义类型,获取相应的值。
我们定义的attr索引名称为=styleable_name
比如我们的styleable name=testAttr
所以age的索引key=testAttr_age(R.styleable.testAttr_age)
最后我们获取的值如下
特别说明:
1、这里特别介绍一下,使用完TypedArray一定要释放掉,因为这些都是资源,如果不释放,长时间占用会导致OOM
2、关于 typed.getFraction()的使用
public float getFraction(@StyleableRes int index, int base, int pbase, float defValue)
index:name的索引
base:base倍数
pbase:parent 倍数
defValue:默认值
- base,返回的百分比为属性值乘以 base。
- pbase,返回的百分比为属性值乘以 pbase。
- base 和 pbase 同时设置只会有一个生效,因为上面 return 中只能使用一个参数。
- base 表示百分比资源的基值,返回结果为 nn% * base 的结果值。
- pbase 表示 %p 形态百分比资源的基值,返回结果为 nn%p * pbase 的结果值。
测试:
app:per="100%p"
app:per="100%"
3.关于flags和enum的值
这两个child value只能是int,否则会报错
error: invalid value 'true' for <flag>; must be an integer.
error: invalid value 'true' for <enum>; must be an integer.
4.精度丢失
typed.getFraction存在精度丢失情况,并不是你设置80%,就返回0.8,目前测试存在精度丢失,返回是0.79999..,如果对精度要求比较准确的,谨慎使用fraction来作为标识。