Android开发中难免遇到需要自定义控件的需求,有些是产品的要求在Android标准控件库中没有满足要求的,有些是开发过程中没有代码的可复用,自己定义的。 一个好的自定义控件应当和Android本身提供的控件一样,封装了一系列的功能以供开发者使用,不仅具有完备的功能,也需要高效的使用内存和CPU。Android本身提供了一些指标:

应当遵守Android标准的规范(命名,可配置,事件处理等)。

在XML布局中科配置控件的属性。

对交互应当有合适的反馈,比如按下,点击等。

具有兼容性, Android版本很多,应该具有广泛的适用性。

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

1. View的子类

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

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

class PieChart extends View {

public PieChart(Context context, AttributeSet attrs) {

super(context, attrs);

}

}

2. 自定义属性

一个完美的自定义控件也可以添加xml来配置属性和风格。 要实现这一点,可按照下列步骤来做:

1) 添加自定义属性到xml文件中

2) 在xml的中,指定属性的值

3) 在view中获取xml中的值

4) 将获取的值应用到view中

下面继续举例说明:添加 到你的程序中,习惯上一般是放在res/values/attrs.xml文件中(可以参考sdk文件下的attrs文件 :sdk\platforms\android-19\data\res\values\attrs.xml

),例如:

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

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

xmlns:custom="http://schemas.android.com/apk/res/com.example.customviews">

custom:showText="true"

custom:labelPosition="left" />

可以看到和标准的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属性值。

public PieChart(Context context, AttributeSet attrs) {

super(context, attrs);

TypedArray a = context.getTheme().obtainStyledAttributes(

attrs,R.styleable.PieChart, 0, 0);

try {

mShowText = a.getBoolean(R.styleable.PieChart_showText, false);

mTextPos = a.getInteger(R.styleable.PieChart_labelPosition, 0);

} finally {

a.recycle();

}

}

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

4) 添加自定义的方法和事件

自定义属性很强大,但缺点也很明显,它只能在view初始化的时候被应用到控件中。 为了添加更加灵活的行为, 可以为每一个属性添加getter和setter对。下面的代码段展示了PieChat的属性showText

public boolean isShowText() {

return mShowText;

}

public void setShowText(boolean showText) {

mShowText = showText;

invalidate();

requestLayout();

}

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

一般你也需要添加事件回调来和调用者沟通。 例如PieChat暴露了OnCurrentItemChanged来通知调用者pie chat发生了旋转。在开发过程中,很容易忘记添加一些属性和事件,特别是作者是这个自定义View的唯一使用者的时候。为使View有更普遍的适用性,应当花些时间考虑的更加周全。最好是暴露所有的可能改变外观和行为的属性。

**实例展示:

**

Paste_Image.png

一、** 自定义属性: **res/values/attrs.xml文件

二、 LinearLayout**的子类 **

public class SettingItemView extends LinearLayout {

private ImageView mIv;

private boolean mChecked;

private boolean mIsVisable;

public SettingItemView(Context context) {

super(context);

}

public SettingItemView(Context context, AttributeSet attrs, int defStyle) {

super(context, attrs, defStyle);

}

public SettingItemView(Context context, AttributeSet attrs) {// 布局文件

super(context, attrs);

View view = View.inflate(context, R.layout.view_setting_item, this);// view_setting_item--View

// 拿到布局文件中的数据

TypedArray ta = context.obtainStyledAttributes(attrs,R.styleable.SettingItemView);

String des = ta.getString(R.styleable.SettingItemView_des);

int itemBg = ta.getInt(R.styleable.SettingItemView_itemBg, -1);// 获取枚举值

mChecked = ta.getBoolean(R.styleable.SettingItemView_checked, false);

mIsVisable = ta.getBoolean(R.styleable.SettingItemView_isVisiable,false);

TextView tv = (TextView) view.findViewById(R.id.tv_setting_item_des);

tv.setText(des);

switch (itemBg) {

case 0:

setBackgroundResource(R.drawable.iv_first_selector);

break;

case 1:

setBackgroundResource(R.drawable.iv_middle_selector);

break;

case 2:

setBackgroundResource(R.drawable.iv_last_selector);

break;

default:

setBackgroundResource(R.drawable.iv_first_selector);

break;

}

mIv = (ImageView) view.findViewById(R.id.iv_setting_item_checked);

setChecked();

setVisable();

// 让这个可点击

setClickable(true);

// 回收一下

ta.recycle();

}

private void setVisable() {

mIv.setVisibility(mIsVisable?View.VISIBLE:View.INVISIBLE);

}

private void setChecked() {

// if(mChecked){

// mIv.setImageResource(R.drawable.on);

// }else {

// mIv.setImageResource(R.drawable.off);

// }

mIv.setImageResource(mChecked ? R.drawable.on : R.drawable.off);

}

// 提供外面设置,是否被选择

public void setChecked(boolean isChecked){

this.mChecked=isChecked;

// 更新UI

setChecked();

}

public boolean isChecked(){

return this.mChecked;

}

public void toggle(){

this.mChecked=!this.mChecked;

// 更新UI

setChecked();

}

}

三、item文件

view_setting_item.xml

android:layout_width="match_parent"

android:layout_height="wrap_content"

android:paddingLeft="5dp"

android:paddingRight="5dp"

android:paddingTop="10dp"

android:paddingBottom="10dp" >

android:id="@+id/tv_setting_item_des"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:layout_centerVertical="true"

android:text="版本更新" />

android:id="@+id/iv_setting_item_checked"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:layout_alignParentRight="true"

android:layout_centerVertical="true"

android:src="@drawable/on" />

四、布局文件

xmlns:custom="http://schemas.android.com/apk/res/com.fanfy"

android:layout_width="match_parent"

android:layout_height="match_parent"

android:orientation="vertical" >

android:id="@+id/siv_version"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:layout_marginTop="20dp"

custom:checked="true"

custom:des="版本更新"

custom:isVisiable="true"

custom:itemBg="first" />

android:id="@+id/siv_call_sms_safe"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

custom:checked="true"

custom:des="版本更新"

customisVisiable="true"

custom:itemBg="last" />

android:id="@+id/siv_address"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

android:layout_marginTop="10dp"

custom:checked="false"

custom:des="版本更新"

custom:isVisiable="true"

custom:itemBg="first" />

android:id="@+id/siv_style_address"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

custom:checked="true"

custom:des="版本更新"

custom:isVisiable="false"

custom:itemBg="middle" />

android:id="@+id/siv_style_address"

android:layout_width="wrap_content"

android:layout_height="wrap_content"

custom:checked="true"

custom:des="版本更新"

custom:isVisiable="false"

custom:itemBg="last" />

android中设置控件的搞,Android中如何自定义控件相关推荐

  1. android线程改变布局,Android线程中设置控件的值提示报错的解决方法

    本文实例讲述了Android线程中设置控件的值提示报错的解决方法.分享给大家供大家参考,具体如下: 在Android线程中设置控件的值一般会与Handler联合使用,如下: package com.y ...

  2. android 动态设置padding,Android动态设置控件大小以及设定margin以及padding值

    http://www.aichengxu.com/Java/73893.htm Android动态设置控件大小以及设定margin以及padding值,有需要的朋友可以参考下. 一.概述 在andro ...

  3. 怎么在ASP.NET中写HTML,如何:在 ASP.NET 网页中设置控件的 HTML 属性

    如何:在 ASP.NET 网页中设置控件的 HTML 属性 10/22/2014 本文内容 更新:2007 年 11 月 这些示例显示如何向页面中的元素添加 HTML 属性.第一个示例演示如何以声明方 ...

  4. Android动态设置控件宽高和padding

    在开发过程中我们经常需要动态设置控件的宽高和padding.以TextView为例.如下: 布局代码 TextView的默认宽高为包裹文字,添加了一个背景颜色设置参数后更直观看到效果 <?xml ...

  5. Android中设置控件可见与不可见详解

    通常控件的可见与不可见分为三种情况. 第一种 gone 表示不可见并且不占用空间 第二种 visible 表示可见 第三种 invisible 表示不可见但是占用空间 可见与不可见的表现形式有两种. ...

  6. android设置控件的坐标,Android中使PopupWindow显示在指定控件的上下左右!

    **1.View中的方法getLocationOnScreen(int[] location):获取该控件在屏幕中的绝对坐标并将坐标保存在数组中:如下图所示,A为屏幕的原点,R,C点即为view的绝对 ...

  7. android中设置控件获得焦点

    android中,要使控件获得焦点,需要先setFocus,再requestFocus. 以Button为例: btn.setFocusable(true); btn.setFocusableInTo ...

  8. android中设置控件获得焦点 (转)

    android中,要使控件获得焦点,需要先setFocus,再requestFocus. 以Button为例:                 btn.setFocusable(true);      ...

  9. android 自定义控件 焦点,android中设置控件获得焦点

    android中,要使控件获得焦点,需要先setFocus,再requestFocus. 以Button为例: btn.setFocusable(true); btn.setFocusableInTo ...

最新文章

  1. 高分求FP-tree算法用Delphi实现
  2. java 克隆_Java实现对象克隆的方法
  3. QT的QUiLoader类的使用
  4. H.264的码率控制算法
  5. java plus in_mybatis plus in方法使用详解
  6. 如何使用阿里云打造离线下载服务器?
  7. 软件配置--ubuntu16.04
  8. 关于单机winform数据库 数据存储
  9. STM32H7 at070tn92 LTDC SDRAM 调试笔记
  10. 联想微型计算机开机黑屏什么原因,联想笔记本电源键亮但黑屏怎么办
  11. 什么是蜘蛛池?有什么作用?
  12. JavaEE项目的三层架构
  13. 云主机被挖矿解决方法
  14. dubbo中标签的使用
  15. Perl之正则表达式基础(一)
  16. Digital Square HDU - 4394 dfs + 剪枝
  17. 【实验技术笔记】利用重组载体做基因过表达(pCDH载体)
  18. rewind java_Java ShortBuffer rewind()用法及代码示例
  19. Exynos_4412——轮询与中断
  20. 计算机系统集成双代号网络图讲解,系统集成项目管理工程师双代号时标网络图知识解读...

热门文章

  1. django跳转到其他页面的方式
  2. Spring 框架 IOC 与 DI 的总结
  3. Madagascar的自定义浮点型函数--对数函数
  4. __attribute__((regparm(number))):函数通过number个寄存器传递参数,否则通过堆栈
  5. 用 cairo 实现跨平台图形
  6. C++异常处理(一) - iStack
  7. python美化输出模块_美化输出结果之Python模块PrettyTable介绍
  8. vscode remote 第三方库_还能这么玩?用VsCode画类图、流程图、时序图、...不要太爽
  9. java分布式架构的发展
  10. JavaScript HTML DOM 事件(笔记)