Android中自定义控件的步骤

Android开发中难免遇到需要自定义控件的需求,有些是产品的要求在Android标准控件库中没有满足要求的,有些是开发过程中没有代码的可复用,自己定义的。


一个好的自定义控件应当和Android本身提供的控件一样,封装了一系列的功能以供开发者使用,不仅具有完备的功能,也需要高效的使用内存和CPU。Android本身提供了
一些指标:
1. 应当遵守Android标准的规范(命名,可配置,事件处理等)。
2. 在XML布局中科配置控件的属性。
3. 对交互应当有合适的反馈,比如按下,点击等。
4. 具有兼容性, Android版本很多,应该具有广泛的适用性。


Android已经提供了一系列基础控件和xml属性来帮助你创建自定义控件。

1. View的子类

View在Android是最基础的几个控件之一, 所有的控件均继承自View,你也可以直接继承View也可以继承其他的控件比如ImageView等。


当然,你至少需要提供一个构造函数,其中Context和AttributeSet作为参数。 举例如下:

[java]  view plain  copy
  1. class PieChart extends View {  
  2.     public PieChart(Context context, AttributeSet attrs) {  
  3.         super(context, attrs);  
  4.     }  
  5. }  


2. 自定义属性

一个完美的自定义控件也可以添加xml来配置属性和风格。 要实现这一点,可按照下列步骤来做:
1) 添加自定义属性<declare-styleable>到xml文件中
2) 在xml的<declare-styleable>中,指定属性的值
3) 在view中获取xml中的值
4) 将获取的值应用到view中


下面继续举例说明:
添加<declare-styleable> 到你的程序中,习惯上一般是放在res/values/attrs.xml文件中,例如:
[java]  view plain  copy
  1. <resources>  
  2.    <declare-styleable name="PieChart">  
  3.        <attr name="showText" format="boolean" />  
  4.        <attr name="labelPosition" format="enum">  
  5.            <enum name="left" value="0"/>  
  6.            <enum name="right" value="1"/>  
  7.        </attr>  
  8.    </declare-styleable>  
  9. </resources>  



这段代码声明了两个自定义的属性 showText和labelPosition,他们属于一个自定义的实体PieChat。


一旦定义好了属性,就可以在xml中使用这些属性了,下面是一个简单的例子:


[java]  view plain  copy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.    xmlns:custom="http://schemas.android.com/apk/res/com.example.customviews">  
  4.  <com.example.customviews.charting.PieChart  
  5.      custom:showText="true"  
  6.      custom:labelPosition="left" />  
  7. </LinearLayout>  


可以看到和标准的Android的组件一样,唯一的差别在他们属于不同的命名空间,标准的组件的命名空间一般是http://schemas.android.com/apk/res/android,

而我们自定义的命名空间是http://schemas.android.com/apk/res/[your package name]。注意到xmlns:custom中的custom了吗?你可以使用任意的字符,但是
要和下面的控件的定义中的字符要保持一致。
另外一个需要注意的是, xml中的tag:com.example.customviews.charting.PieChart,需要的完整的包名,如果你的自定义控件是个内部类(好吧,这么奇葩),
也必须给全路径,假设PieChat有个内部类PieView,如果在XML中引用它,需要这样使用:com.example.customviews.charting.PieChart$PieView


3) 应用自定义的属性值
当View被创建的时候,可以通过AttributeSet读取所有的定义在xml中的属性,在构造函数中通过obtainStyledAttributes读取attrs,
该方法会返回一个TypeArray数组。通过TypeArray可以读取到已经定义在XML中的方法。下面的例子展示了读取上文中的xml属性值。

[java]  view plain  copy
  1. public PieChart(Context context, AttributeSet attrs) {  
  2.    super(context, attrs);  
  3.    TypedArray a = context.getTheme().obtainStyledAttributes(  
  4.         attrs,  
  5.         R.styleable.PieChart,  
  6.         00);  
  7.   
  8.    try {  
  9.        mShowText = a.getBoolean(R.styleable.PieChart_showText, false);  
  10.        mTextPos = a.getInteger(R.styleable.PieChart_labelPosition, 0);  
  11.    } finally {  
  12.        a.recycle();  
  13.    }  
  14. }  


需要强调的是, TypeArray使用完毕后需要销毁,不然会发生内存泄露。

4) 添加自定义的方法和事件
自定义属性很强大,但缺点也很明显,它只能在view初始化的时候被应用到控件中。 为了添加更加灵活的行为, 可以为每一个属性添加getter和setter
对。下面的代码段展示了PieChat的属性showText
[java]  view plain  copy
  1. public boolean isShowText() {  
  2.    return mShowText;  
  3. }  
  4.   
  5. public void setShowText(boolean showText) {  
  6.    mShowText = showText;  
  7.    invalidate();  
  8.    requestLayout();  
  9. }  


在setShowText中调用了invalidate()和requestLayout(), 保证了view能及时的更新。在你的自定义View中,如果有属性被改变并且需要立即生效时, 你也必须调用invalidate()这个方法。 这样系统会立即重新绘制view。 同样的,如果view的尺寸或者形状发生了变化,你也必须调用requestLayout(). 不然会引起 很多问题。

一般你也需要添加事件回调来和调用者沟通。 例如PieChat暴露了OnCurrentItemChanged来通知调用者pie chat发生了旋转。
在开发过程中,很容易忘记添加一些属性和事件,特别是作者是这个自定义View的唯一使用者的时候。为使View有更普遍的适用性,应当花些时间考虑的更加周全。
你最好是暴露所有的可能改变外观和行为的属性。当然这也对你提出了更高的要求,不然怎么进步呢。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值