当系统提供的控件,无法满足我们的需求的时候,我们往往会 想到自定义控件,通过继承View来实现。当我们想像 系统提供的控件那样可以如android:text="hello",这样设置自定义控件中的文字显示去,这时我们该怎么办?
带着问题,我们会想到在继承View的构造函数中 有个AttributeSet这个属性,没错 这就是 突破口。
我们可以在attrs.xml中声明自己控件的属性,在布局xml文档中声明自己的命名空间,这时就可以对设置自己想要的值了,然后在AttributeSet这个属性中获取对应的值。好了不多说,我们来看下代码,一切尽在不言中:
工程下载地址:点击打开链接
第一步:
在/res/values下编写一个attrs.xml的文件,其中内容结构如下:
<?xml version="1.0" encoding="utf-8"?><resources><declare-styleable name="my_view"><attr name="text_color" format="color"></attr><attr name="bg_color" format="color"></attr><attr name="text" format="string"></attr></declare-styleable></resources>
其中要说的是format 格式,对于它的格式说明将在文章后面介绍。
第二步:
写一个自定义的Veiw,如下代码:
public class myView extends View{private Paint paint;private int bg_color,text_color,alpha;private String text;public myView(Context context, AttributeSet attrs) {super(context, attrs);paint = new Paint();TypedArray s = context.obtainStyledAttributes(attrs, R.styleable.my_view);//从xml那传来的一组值bg_color = s.getColor(R.styleable.my_view_bg_color, R.color.transparent);//得到attrs中对应属性的bg_color值text_color = s.getColor(R.styleable.my_view_text_color, android.R.color.black);//..text = s.getText(R.styleable.my_view_text).toString();//..}@Overridepublic void draw(Canvas canvas) {// TODO Auto-generated method stubsuper.draw(canvas);paint.setColor(bg_color);Rect r = new Rect(1, 80, 60, 140);canvas.drawRect(r, paint);paint.setColor(text_color);canvas.drawText(text, 10, 100, paint);}}
注释在重点地方已经写清楚了,也就值AttributeSetattrs的使用,通过它可以获取在XML中定义的值。
第三步:
在布局文档中声明你的View
<LinearLayout xmlns:android="/apk/res/android"xmlns:app="/apk/res/com.example.defineview"android:layout_width="fill_parent"android:layout_height="fill_parent"android:orientation="vertical" ><!-- 先声明命名空间 res/包名 以下就可以通过app:+attrs定义的属性,进行值设置 --><com.example.defineview.myViewandroid:layout_width="wrap_content"android:layout_height="wrap_content"app:text="nihao"app:text_color="#ff0000"app:bg_color="#00ff00"/></LinearLayout>
其中
xmlns:app="/apk/res/com.example.defineview"
是自己声明的命名空间,app是头自由命名,在res/+自己的包名
这时你就可以使用
app:text="nihao"
这样的属性了,注意text、bg_color这些属性是在attrs中设置好的。
好了这样就OK了我们来看下运行效果:
运行效果中是不是显示出了
app:text="nihao"app:text_color="#ff0000"app:bg_color="#00ff00"
的效果。
我们看下文件的目录结构
附录:attrs
在这里,需要补充attrs属性的相关知识,即Attr属性是如何在XML中定义的,自定义属性的Value值可以有10种类型以及其类型的组合值,其具体使用方法如下:
1.reference:参考某一资源ID。
(1)属性定义:
<declare-styleablename="名称">
<attrname="background"format="reference"/>
</declare-styleable>
(2)属性使用:
<ImageView
android:layout_width="42dip"
android:layout_height="42dip"
android:background="@drawable/图片ID" />
2.color:颜色值。
(1)属性定义:
<declare-styleablename="名称">
<attrname="textColor"format="color"/>
</declare-styleable>
(2)属性使用:
<TextView
android:layout_width="42dip"
android:layout_height="42dip"
android:textColor="#00FF00"/>
3.boolean:布尔值。
(1)属性定义:
<declare-styleablename="名称">
<attrname="focusable"format="boolean"/>
</declare-styleable>
(2)属性使用:
<Button
android:layout_width="42dip"
android:layout_height="42dip"
android:focusable="true"/>
4.dimension:尺寸值。
(1)属性定义:
<declare-styleablename="名称">
<attrname="layout_width"format="dimension"/>
</declare-styleable>
(2)属性使用:
<Button
android:layout_width="42dip"
android:layout_height="42dip"/>
5.float:浮点值。
(1)属性定义:
<declare-styleablename="AlphaAnimation">
<attrname="fromAlpha"format="float"/>
<attrname="toAlpha"format="float"/>
</declare-styleable>
(2)属性使用:
<alpha
android:fromAlpha="1.0"
android:toAlpha="0.7"/>
6.integer:整型值。
(1)属性定义:
<declare-styleablename="AnimatedRotateDrawable">
<attrname="visible"/>
<attrname="frameDuration"format="integer"/>
<attrname="framesCount"format="integer"/>
<attrname="pivotX"/>
<attrname="pivotY"/>
<attrname="drawable"/>
</declare-styleable>
(2)属性使用:
<animated-rotate
xmlns:android="/apk/res/android"
android:drawable="@drawable/图片ID"
android:pivotX="50%"
android:pivotY="50%"
android:framesCount="12"
android:frameDuration="100" />
7.string:字符串。
(1)属性定义:
<declare-styleablename="MapView">
<attrname="apiKey"format="string"/>
</declare-styleable>
(2)属性使用:
<com.google.android.maps.MapView
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:apiKey="0jOkQ80oD1JL9C6HAja99uGXCRiS2CGjKO_bc_g"
/>
8.fraction:百分数。
(1)属性定义:
<declare-styleablename="RotateDrawable">
<attrname="visible"/>
<attrname="fromDegrees"format="float"/>
<attrname="toDegrees"format="float"/>
<attrname="pivotX"format="fraction"/>
<attrname="pivotY"format="fraction"/>
<attrname="drawable"/>
</declare-styleable>
(2)属性使用:
<rotate
xmlns:android="/apk/res/android"
android:interpolator="@anim/动画ID"
android:fromDegrees="0"
android:toDegrees="360"
android:pivotX="200%"
android:pivotY="300%"
android:duration="5000"
android:repeatMode="restart"
android:repeatCount="infinite"/>
9.enum:枚举值。
(1)属性定义:
<declare-styleablename="名称">
<attrname="orientation">
<enumname="horizontal"value="0"/>
<enumname="vertical"value="1"/>
</attr>
</declare-styleable>
(2)属性使用:
<LinearLayout
xmlns:android="/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
>
</LinearLayout>
10.flag:位或运算。
(1)属性定义:
<declare-styleablename="名称">
<attrname="windowSoftInputMode">
<flagname="stateUnspecified"value="0"/>
<flagname="stateUnchanged"value="1"/>
<flagname="stateHidden"value="2"/>
<flagname="stateAlwaysHidden"value="3"/>
<flagname="stateVisible"value="4"/>
<flagname="stateAlwaysVisible"value="5"/>
<flagname="adjustUnspecified"value="0x00"/>
<flagname="adjustResize"value="0x10"/>
<flagname="adjustPan"value="0x20"/>
<flagname="adjustNothing"value="0x30"/>
</attr>
</declare-styleable>
(2)属性使用:
<activity
android:name=".StyleAndThemeActivity"
android:label="@string/app_name"
android:windowSoftInputMode="stateUnspecified|stateUnchanged|stateHidden">
<intent-filter>
<actionandroid:name="android.intent.action.MAIN"/>
<categoryandroid:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
注意:
属性定义时可以指定多种类型值。
(1)属性定义:
<declare-styleablename="名称">
<attrname="background"format="reference|color"/>
</declare-styleable>
(2)属性使用:
<ImageView
android:layout_width="42dip"
android:layout_height="42dip"
android:background="@drawable/图片ID|#00FF00"
/>