通常情况下,Android实现自定义控件无非三种方式。

  Ⅰ、继承现有控件,对其控件的功能进行拓展。

  Ⅱ、将现有控件进行组合,实现功能更加强大控件。

  Ⅲ、重写View实现全新的控件

  上文说过了如何继承现有控件来自定义控件,这节我们来讨论第二个议题。怎么将控件组合来实现一个功能强大的自定义控件。

  先看看创建组合控件的好处吧,创建组合控件能够很好的创建具有组合功能的控件集合。那我们一般又是怎么做的了,一般我们来继承一个合适的ViewGroup,再为他创建一个新功能,从而就形成了一个新功能的控件。我们还会为这种控件指定一些新的属性,从而使他具有很好扩展性了。好了,废话说了这么多,下面,我们就以几乎每个app都有的控件——标题栏为例,来介绍组合控件的做法。

  首先,我来回答为什么要重用标题栏:

  Ⅰ、使应用程序拥有统一的风格。

  Ⅱ、重用标题栏,也是我们将来修改标题栏非常方便,真正实现"一次编写,到处运行"的效果,而不用大费周章的,每个页面都修改。

  Ⅲ、向调用者向外暴露调用接口,从而更加灵活的控制标题栏,使其功能更加的强大。

  那么,标题栏长成那个样子,请见下图:

  

  我们,先做一下简单的分析一下,这是一个自定义控件,应该像Android的原生控件一样,能够方便调用者设置控件的属性。因此,十分有必要为这个控件设置一些属性,为一个View提供一些自定义属性十分的简单,只需要在res资源目录下的values目录下创建一个attrs.xml属性文件,并在该文件定义你所需要的属性即可。这个自定义控件自定义属性如下:

  1. <declare-styleable name="titleBar">
  2. <attr name="title" format="string" />
  3. <attr name="titleTextSize" format="dimension" />
  4. <attr name="titleTextColor" format="color" />
  5. <attr name="titleLeftText" format="string" />
  6. <attr name="titleLeftBackground" format="color|reference" />
  7. <attr name="titleLeftTextColor" format="color" />
  8. <attr name="titleRightText" format="string" />
  9. <attr name="titleRightBackground" format="color|reference" />
  10. <attr name="titleRightTextColor" format="color" />
  11. </declare-styleable>

  我们用<declare-styleable>标签声明要使用的自定义属性,用name属性来确定引用的名称。用format来确定引用数据的格式。在这个自定义控件自定义属性对应列表如下:

  Ⅰ、title——对应标题的文字

  Ⅱ、titleTextSize——对应标题的文字大小

  Ⅲ、titleTextColor——对应标题的文本颜色

  Ⅳ、titleLeftText——对应左边按钮的文本

  Ⅴ、titleLeftBackground——对应左边按钮的背景

  Ⅵ、titleLeftTextColor——对应左边按钮的文字颜色

  Ⅶ、titleRightText——对应右边按钮的文本

  Ⅴ、titleRightBackground——对应右边按钮的背景

  Ⅵ、titleRightTextColor——对应右边按钮的文字颜色

  这里,需要指出的是左右按钮的背景,即可以是颜色类型,也可以对应为相应的图片,所以,我们可以用“|”来分隔不同的属性。

  好了,既然,有了自定义属性的定义了,我们就需要自定义一个TitleBar的控件,来获取这些定义好的属性值,上文,我们提到一般组合控件一般继承与ViewGroup控件,这里,我们方便起见,就继承与RelativeLayout。怎么获取属性值了,系统提供了TypedArray这样数据结构就能十分方便获取属性集了,获取属性的代码如下:

  1. private void initAttrs(AttributeSet attrs) {
  2. TypedArray ta = this.getContext().obtainStyledAttributes(attrs,
  3. R.styleable.titleBar);
  4. if (ta != null) {
  5. title = ta.getString(R.styleable.titleBar_title);
  6. titleTextSize = ta.getDimension(R.styleable.titleBar_titleTextSize,
  7. 16);
  8. titleTextColor = ta
  9. .getColor(R.styleable.titleBar_titleTextColor, 0);
  10. titleLeftText = ta.getString(R.styleable.titleBar_titleLeftText);
  11. titleLeftBackground = ta
  12. .getDrawable(R.styleable.titleBar_titleLeftBackground);
  13. titleLeftTextColor = ta.getColor(
  14. R.styleable.titleBar_titleLeftTextColor, 0);
  15. titleRightText = ta.getString(R.styleable.titleBar_titleRightText);
  16. titleRightBackground = ta
  17. .getDrawable(R.styleable.titleBar_titleRightBackground);
  18. titleRightTextColor = ta.getColor(
  19. R.styleable.titleBar_titleRightTextColor, 0);
  20. ta.recycle();
  21. }
  22. }

  这里,需要值得一提的是需要调用TypedArray的recycle方法将资源回收。

  既然,我们让这个组合控件有了属性以后,下面,我们要做的是将这个组合控件的按钮,文本框有机组合起来,组合的代码如下所示:

  1. private void initView() {
  2. leftButton = new Button(getContext());
  3. titleTextView = new TextView(getContext());
  4. rightButton = new Button(getContext());
  5.  
  6. leftButton.setTextColor(titleLeftTextColor);
  7. leftButton.setBackgroundDrawable(titleLeftBackground);
  8. leftButton.setText(titleLeftText);
  9.  
  10. rightButton.setTextColor(titleRightTextColor);
  11. rightButton.setBackgroundDrawable(titleRightBackground);
  12. rightButton.setText(titleRightText);
  13.  
  14. titleTextView.setText(title);
  15. titleTextView.setTextSize(titleTextSize);
  16. titleTextView.setTextColor(titleTextColor);
  17.  
  18. mLeftLayoutParams = new LayoutParams(LayoutParams.WRAP_CONTENT,
  19. LayoutParams.MATCH_PARENT);
  20. mLeftLayoutParams.addRule(RelativeLayout.ALIGN_PARENT_LEFT);
  21. addView(leftButton, mLeftLayoutParams);
  22.  
  23. mCenterLayoutParams = new LayoutParams(LayoutParams.WRAP_CONTENT,
  24. LayoutParams.MATCH_PARENT);
  25. mCenterLayoutParams.addRule(RelativeLayout.CENTER_IN_PARENT);
  26. addView(titleTextView, mCenterLayoutParams);
  27.  
  28. mRightLayoutParams = new LayoutParams(LayoutParams.WRAP_CONTENT,
  29. LayoutParams.MATCH_PARENT);
  30. mRightLayoutParams.addRule(RelativeLayout.ALIGN_PARENT_RIGHT);
  31. addView(rightButton, mRightLayoutParams);
  32. }

  我们看到上文定义一些属性,无非复制给了这些组合控件,使这个组合控件变得"有血有肉"了。

  这既然是一个自定义控件,是一个UI模版,应该每个调用者点击左右按钮,所实现的可能都不一样,我们应当所做就是向外暴露接口,让调用者灵活的控制这两个按钮。那么接口的定义如下:

  1. public interface ClickListener {
  2. void Click(int tag);
  3. }
  4.  
  5. private ClickListener listener;

  在模版方法中,为左、右按钮增加点击事件,调用接口的点击方法,代码如下所示:

  1. private void setListener() {
  2. leftButton.setOnClickListener(this);
  3. rightButton.setOnClickListener(this);
  4. }
  5.  
  6. @Override
  7. public void onClick(View v) {
  8. if (listener != null) {
  9. if (v == leftButton) {
  10. listener.Click(LEFT_BUTTON);
  11. } else if (v == rightButton) {
  12. listener.Click(RIGHT_BUTTON);
  13. }
  14. }
  15.  
  16. }

  在代码,我们有效判断是左边按钮点击了,还是右边按钮点击了。 

  有了这个模版方法中接口的定义之后,我们在外部调用这个回调代码如下:

  1. titleBar.setListener(new ClickListener() {
  2.  
  3. @Override
  4. public void Click(int tag) {
  5. switch (tag) {
  6. case TitleBar.LEFT_BUTTON:
  7. Toast.makeText(MainActivity.this, "左边按钮被点击了", 0).show();
  8. break;
  9. case TitleBar.RIGHT_BUTTON:
  10. Toast.makeText(MainActivity.this, "右边按钮被点击了", 0).show();
  11. break;
  12. default:
  13. break;
  14. }
  15. }
  16. });

  这样在外部,能够有效的控制左右按钮的点击事件了。

  做了这么多,就是希望能够有效调用这个组合控件,调用组合控件的代码如下:

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:custom="http://schemas.android.com/apk/res/com.example.test"
  3. xmlns:tools="http://schemas.android.com/tools"
  4. android:layout_width="match_parent"
  5. android:layout_height="match_parent"
  6. android:padding="5dp"
  7. tools:context=".MainActivity">
  8.  
  9. <!-- <include layout="@layout/topbar" /> -->
  10.  
  11. <com.example.test.TitleBar
  12. android:id="@+id/titleBar"
  13. android:layout_width="match_parent"
  14. android:layout_height="40dp"
  15. custom:titleLeftBackground="@drawable/blue_button"
  16. custom:titleLeftText="Back"
  17. custom:titleLeftTextColor="#FFFFFF"
  18. custom:titleRightBackground="@drawable/blue_button"
  19. custom:titleRightText="More"
  20. custom:titleRightTextColor="#FFFFFF"
  21. custom:title="自定义标题"
  22. custom:titleTextColor="#123412"
  23. custom:titleTextSize="10sp"/>
  24.  
  25. </RelativeLayout>

  这里,需要和大家交代的是,自定义控件与原生控件调用区别在于:

  Ⅰ、引用自定义控件必须引用它的完全类名。

  Ⅱ、引用自定义控件自定义属性时,必须要引用自定义的命名空间,引用方法如下:

  1. xmlns:custom="http://schemas.android.com/apk/res/com.example.test"
  2.  
  3.   这个控件,最终运行效果为:

  1.   这就是我封装标题栏,欢迎大家吐槽。

Android 自定义View 三板斧之二——组合现有控件的更多相关文章

  1. Android自定义View(CustomCalendar-定制日历控件)

    转载请标明出处: http://blog.csdn.net/xmxkf/article/details/54020386 本文出自:[openXu的博客] 目录: 1分析 2自定义属性 3onMeas ...

  2. Android自定义View(三、深入解析控件测量onMeasure)

    转载请标明出处: http://blog.csdn.net/xmxkf/article/details/51490283 本文出自:[openXu的博客] 目录: onMeasure什么时候会被调用 ...

  3. Android 自定义支持快速搜索筛选的选择控件(一)

    Android 自定义支持快速搜索筛选的选择控件 项目中遇到选择控件选项过多,需要快速查找匹配的情况. 做了简单的Demo,效果图如下: 源码地址:https://github.com/whieenz ...

  4. Android 自定义View 三板斧之一——继承现有控件

    通常情况下,Android实现自定义控件无非三种方式. Ⅰ.继承现有控件,对其控件的功能进行拓展. Ⅱ.将现有控件进行组合,实现功能更加强大控件. Ⅲ.重写View实现全新的控件 本文重点讨论继承现有 ...

  5. Android 自定义View 三板斧之三——重写View来实现全新控件

    通常情况下,Android实现自定义控件无非三种方式. Ⅰ.继承现有控件,对其控件的功能进行拓展. Ⅱ.将现有控件进行组合,实现功能更加强大控件. Ⅲ.重写View实现全新的控件 本文来讨论最难的一种 ...

  6. Android自定义View学习(二)

    绘制顺序 参考:HenCoder Android 开发进阶:自定义 View 1-5 绘制顺序 绘制过程 包括 背景 主体(onDraw()) 子 View(dispatchDraw()) 滑动边缘渐 ...

  7. Android 自定义 HorizontalScrollView 打造再多图片(控件)也不怕 OOM 的横向滑动效果

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/38140505 自从Gallery被谷歌废弃以后,Google推荐使用ViewPa ...

  8. Android自定义View(RollWeekView-炫酷的星期日期选择控件)

    转载请标明出处: http://blog.csdn.net/xmxkf/article/details/53420889 本文出自:[openXu的博客] 目录: 1分析 2定义控件布局 3定义Cus ...

  9. Android自定义View(LimitScrollerView-仿天猫广告栏上下滚动效果)

    转载请标明出处: http://blog.csdn.net/xmxkf/article/details/53303872 本文出自:[openXu的博客] 1分析 2定义组合控件布局 3继承最外层控件 ...

随机推荐

  1. UISegmentedControl 控件

    一.创建 UISegmentedControl* mySegmentedControl = [[UISegmentedControl alloc]initWithItems:nil]; 是不是很奇怪没 ...

  2. 在 JQuery Mobile 中实现瀑布流图库布局

    先来看在Windows系统的1080P显示器中显示的效果: 这个整合方式几乎没有现存的实例,是自己总结出来的方法,在此记录下来. 首先访问Masonry官网下载masonry.pkgd.min.js: ...

  3. 去除inline-block之间的间距

    a标签的父容器添加: font-size: 0; -webkit-text-size-adjust:none;

  4. intel显卡笔记本恢复屏幕亮度调整功能

    更新Intel显卡驱动后不能修改屏幕亮度,可以在注册表里面搜索featuretestcontrol,将f000修改为ffff,重启后就可以通过Fn+F4/F5调整屏幕亮度了. 注:此方法适用于带有in ...

  5. First class ,6 examples anlaysisi

    http://www.fgm.cc/learn/ First class ,6 examples anlaysisi <!DOCTYPE html> <!-- To change t ...

  6. 根据juery CSS点击一个标签弹出一个遮罩层的简单示例

    <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xhtml"> <head> <m ...

  7. 文件操作总结 (Path,Directory,File)

    Path类就是对字符串的操作,与实际的文件没有任何关系 属性: Path.GetFileName("路径"), //获取文件名带后缀: Path.GetFileNameWithou ...

  8. iOS基础之网络请求相关

    1.AFNetwork二次封装方法一: #import <Foundation/Foundation.h> @interface BeeNetworkManager : NSObject ...

  9. UI基础之UITextField相关

    UITextField *textF = [[UITextField alloc] init]; 1.字体相关 textF.text = @"文本框文字"; textF.textC ...

  10. Java 反射工具类封装

    封装了常用的反射相关方法 public class ReflectUtil { /** * 通过类路径获取Class * * @author LHY <br> * Description ...