View的自定义

转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/24252901

很多的Android入门程序猿来说对于Android自定义View,可能都是比较恐惧的,但是这又是高手进阶的必经之路,所有准备在自定义View上面花一些功夫,多写一些文章。先总结下自定义View的步骤:

1、自定义View的属性

2、在View的构造方法中获得我们自定义的属性

[ 3、重写onMesure ]

4、重写onDraw

我把3用[]标出了,所以说3不一定是必须的,当然了大部分情况下还是需要重写的。

1、自定义View的属性,首先在res/values/  下建立一个attrs.xml , 在里面定义我们的属性和声明我们的整个样式。

[html] view plaincopy

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <resources>
  3. <attr name="titleText" format="string" />
  4. <attr name="titleTextColor" format="color" />
  5. <attr name="titleTextSize" format="dimension" />
  6. <declare-styleable name="CustomTitleView">
  7. <attr name="titleText" />
  8. <attr name="titleTextColor" />
  9. <attr name="titleTextSize" />
  10. </declare-styleable>
  11. </resources>

我们定义了字体,字体颜色,字体大小3个属性,format是值该属性的取值类型:

一共有:string,color,demension,integer,enum,reference,float,boolean,fraction,flag;不清楚的可以google一把。

然后在布局中声明我们的自定义View

[objc] view plaincopy

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools"
  3. xmlns:custom="http://schemas.android.com/apk/res/com.example.customview01"
  4. android:layout_width="match_parent"
  5. android:layout_height="match_parent" >
  6. <com.example.customview01.view.CustomTitleView
  7. android:layout_width="200dp"
  8. android:layout_height="100dp"
  9. custom:titleText="3712"
  10. custom:titleTextColor="#ff0000"
  11. custom:titleTextSize="40sp" />
  12. </RelativeLayout>

一定要引入 xmlns:custom="http://schemas.android.com/apk/res/com.example.customview01"我们的命名空间,后面的包路径指的是项目的package

2、在View的构造方法中,获得我们的自定义的样式

[java] view plaincopy

  1. /**
  2. * 文本
  3. */
  4. private String mTitleText;
  5. /**
  6. * 文本的颜色
  7. */
  8. private int mTitleTextColor;
  9. /**
  10. * 文本的大小
  11. */
  12. private int mTitleTextSize;
  13. /**
  14. * 绘制时控制文本绘制的范围
  15. */
  16. private Rect mBound;
  17. private Paint mPaint;
  18. public CustomTitleView(Context context, AttributeSet attrs)
  19. {
  20. this(context, attrs, 0);
  21. }
  22. public CustomTitleView(Context context)
  23. {
  24. this(context, null);
  25. }
  26. /**
  27. * 获得我自定义的样式属性
  28. *
  29. * @param context
  30. * @param attrs
  31. * @param defStyle
  32. */
  33. public CustomTitleView(Context context, AttributeSet attrs, int defStyle)
  34. {
  35. super(context, attrs, defStyle);
  36. /**
  37. * 获得我们所定义的自定义样式属性
  38. */
  39. TypedArray a = context.getTheme().obtainStyledAttributes(attrs, R.styleable.CustomTitleView, defStyle, 0);
  40. int n = a.getIndexCount();
  41. for (int i = 0; i < n; i++)
  42. {
  43. int attr = a.getIndex(i);
  44. switch (attr)
  45. {
  46. case R.styleable.CustomTitleView_titleText:
  47. mTitleText = a.getString(attr);
  48. break;
  49. case R.styleable.CustomTitleView_titleTextColor:
  50. // 默认颜色设置为黑色
  51. mTitleTextColor = a.getColor(attr, Color.BLACK);
  52. break;
  53. case R.styleable.CustomTitleView_titleTextSize:
  54. // 默认设置为16sp,TypeValue也可以把sp转化为px
  55. mTitleTextSize = a.getDimensionPixelSize(attr, (int) TypedValue.applyDimension(
  56. TypedValue.COMPLEX_UNIT_SP, 16, getResources().getDisplayMetrics()));
  57. break;
  58. }
  59. }
  60. a.recycle();
  61. /**
  62. * 获得绘制文本的宽和高
  63. */
  64. mPaint = new Paint();
  65. mPaint.setTextSize(mTitleTextSize);
  66. // mPaint.setColor(mTitleTextColor);
  67. mBound = new Rect();
  68. mPaint.getTextBounds(mTitleText, 0, mTitleText.length(), mBound);
  69. }

我们重写了3个构造方法,默认的布局文件调用的是两个参数的构造方法,所以记得让所有的构造调用我们的三个参数的构造,我们在三个参数的构造中获得自定义属性。

3、我们重写onDraw,onMesure调用系统提供的:

[java] view plaincopy

  1. @Override
  2. protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
  3. {
  4. super.onMeasure(widthMeasureSpec, heightMeasureSpec);
  5. }
  6. @Override
  7. protected void onDraw(Canvas canvas)
  8. {
  9. mPaint.setColor(Color.YELLOW);
  10. canvas.drawRect(0, 0, getMeasuredWidth(), getMeasuredHeight(), mPaint);
  11. mPaint.setColor(mTitleTextColor);
  12. canvas.drawText(mTitleText, getWidth() / 2 - mBound.width() / 2, getHeight() / 2 + mBound.height() / 2, mPaint);
  13. }

此时的效果是:

是不是觉得还不错,基本已经实现了自定义View。但是此时如果我们把布局文件的宽和高写成wrap_content,会发现效果并不是我们的预期:

系统帮我们测量的高度和宽度都是MATCH_PARNET,当我们设置明确的宽度和高度时,系统帮我们测量的结果就是我们设置的结果,当我们设置为WRAP_CONTENT,或者MATCH_PARENT系统帮我们测量的结果就是MATCH_PARENT的长度。

所以,当设置了WRAP_CONTENT时,我们需要自己进行测量,即重写onMesure方法”:

重写之前先了解MeasureSpec的specMode,一共三种类型:

EXACTLY:一般是设置了明确的值或者是MATCH_PARENT

AT_MOST:表示子布局限制在一个最大值内,一般为WARP_CONTENT

UNSPECIFIED:表示子布局想要多大就多大,很少使用

下面是我们重写onMeasure代码:

[java] view plaincopy

  1. @Override
  2. protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
  3. {
  4. int widthMode = MeasureSpec.getMode(widthMeasureSpec);
  5. int widthSize = MeasureSpec.getSize(widthMeasureSpec);
  6. int heightMode = MeasureSpec.getMode(heightMeasureSpec);
  7. int heightSize = MeasureSpec.getSize(heightMeasureSpec);
  8. int width;
  9. int height ;
  10. if (widthMode == MeasureSpec.EXACTLY)
  11. {
  12. width = widthSize;
  13. } else
  14. {
  15. mPaint.setTextSize(mTitleTextSize);
  16. mPaint.getTextBounds(mTitle, 0, mTitle.length(), mBounds);
  17. float textWidth = mBounds.width();
  18. int desired = (int) (getPaddingLeft() + textWidth + getPaddingRight());
  19. width = desired;
  20. }
  21. if (heightMode == MeasureSpec.EXACTLY)
  22. {
  23. height = heightSize;
  24. } else
  25. {
  26. mPaint.setTextSize(mTitleTextSize);
  27. mPaint.getTextBounds(mTitle, 0, mTitle.length(), mBounds);
  28. float textHeight = mBounds.height();
  29. int desired = (int) (getPaddingTop() + textHeight + getPaddingBottom());
  30. height = desired;
  31. }
  32. setMeasuredDimension(width, height);
  33. }

现在我们修改下布局文件:

[html] view plaincopy

  1. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  2. xmlns:tools="http://schemas.android.com/tools"
  3. xmlns:custom="http://schemas.android.com/apk/res/com.example.customview01"
  4. android:layout_width="match_parent"
  5. android:layout_height="match_parent" >
  6. <com.example.customview01.view.CustomTitleView
  7. android:layout_width="wrap_content"
  8. android:layout_height="wrap_content"
  9. custom:titleText="3712"
  10. android:padding="10dp"
  11. custom:titleTextColor="#ff0000"
  12. android:layout_centerInParent="true"
  13. custom:titleTextSize="40sp" />
  14. </RelativeLayout>

现在的效果是:

完全复合我们的预期,现在我们可以对高度、宽度进行随便的设置了,基本可以满足我们的需求。

当然了,这样下来我们这个自定义View与TextView相比岂不是没什么优势,所有我们觉得给自定义View添加一个事件:

在构造中添加:

[java] view plaincopy

  1. this.setOnClickListener(new OnClickListener()
  2. {
  3. @Override
  4. public void onClick(View v)
  5. {
  6. mTitleText = randomText();
  7. postInvalidate();
  8. }
  9. });

[java] view plaincopy

  1. private String randomText()
  2. {
  3. Random random = new Random();
  4. Set<Integer> set = new HashSet<Integer>();
  5. while (set.size() < 4)
  6. {
  7. int randomInt = random.nextInt(10);
  8. set.add(randomInt);
  9. }
  10. StringBuffer sb = new StringBuffer();
  11. for (Integer i : set)
  12. {
  13. sb.append("" + i);
  14. }
  15. return sb.toString();
  16. }

下面再来运行:

我们添加了一个点击事件,每次让它随机生成一个4位的随机数,有兴趣的可以在onDraw中添加一点噪点,然后改写为验证码,是不是感觉很不错。

时间: 2024-10-06 00:39:06

View的自定义的相关文章

qt model/view 架构自定义模型之QStringListModel

# -*- coding: utf-8 -*- # python:2.x #QStringListModel #QStringListModel 是最简单的模型类,具备向视图提供字符串数据的能力. #QStringListModel 是一个可编辑的模型,可以为组件提供一系列字符串作为数据. #看作是封装了 QStringList 的模型.QStringList 是一种很常用的数据类型,实际上 #是一个字符串列表(也就是 QList<QString>).既然是列表,它也就是线性的数据结构, #因

Dialog详解(包括进度条、PopupWindow、自定义view、自定义样式的对话框)

Android中提供了多种对话框,在实际应用中我们可能会需要修改这些已有的对话框.本实例就是从实际出发,展现了andorid中大部分对话框,代码中用了一个对话框管理类来做封装,其中还定义了对话框的动画.自定义样式等等. 主布局文件(全是button) <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.co

安卓学习-界面-View的自定义

android的所有UI控件,都是基于View的,因此特意重点学习了下这个,为后面学习其他控件打下基础. http://www.360doc.com/content/14/0102/12/12050012_342019150.shtml 重新时常用覆盖的方法 package com.example.ddddddd; import android.content.Context; import android.graphics.Canvas; import android.graphics.Rec

t model/view 架构自定义模型之QFileSystemModel

# -*- coding: utf-8 -*- # python:2.x #QFileSystemModel """ Qt  内置了两种模型:QStandardItemModel 和 QFileSystemModel .QStandardItemModel 是一种多用途的模型,能够让列表.表格. 树等视图显示不同的数据结构.这种模型会将数据保存起来.试想一下, 列表和表格所要求 的数据结构肯定是不一样的:前者是一维的,后者是二维的.因此,模型需要保存有实际数 据,当视图是列表

Android群英传笔记系列三 view的自定义:实现一个模拟下载

1.实现效果:动态显示进度(分别显示了整个的动态改变的过程,然后完成后,弹出一个对话框)       2.实现过程:可以分为绘制一个圆,圆弧和文本三部分,然后在MainAcitivity中通过线程模拟下载进度. a.定义一个类继承至view,然后添加其构造函数,记得一定要添加含有Attributset参数的构造函数; b.定义和初始化一些数据: private int mCircleXY; private int mWidth; private float mRadius; private Pa

Android自定义View之自定义复合控件--标题栏

1 实现效果 2 例子代码讲解 2.1 attrs.xml <?xml version="1.0" encoding="utf-8"?> <resources> <declare-styleable name="TopBar"> <attr name="title" format="string" /> <attr name="titleTex

Android 自定义View修炼-自定义加载进度动画LoadingImageView

一.概述 本自定义View,是加载进度动画的自定义View,继承于ImageView来实现,主要实现蒙层加载进度的加载进度效果. 支持水平左右加载和垂直上下加载四个方向,同时也支持自定义蒙层进度颜色. 直接看下面的效果图吧. 二.效果图 废话不说,先来看看效果图吧~~ 三.实现原理方案 1.自定义View-XCLoadingImageView,继承ImageVIew来实现,这样就不用自己再处理drawable和测量的工作内容. 2.根据蒙层颜色创建一个蒙层bitmap,然后根据这个bitmap来

手把手带你画一个漂亮蜂窝view Android自定义view

上一篇做了一个水波纹view  不知道大家有没有动手试试呢点击打开链接 这个效果做起来好像没什么意义,如果不加监听回调 图片就能直接替代.写这篇博客的目的是锻炼一下思维能力,以更好的面多各种自定义view需求. 转载请注明出处:http://blog.csdn.net/wingichoy/article/details/50554058 本文是和代码同步写的.也就是说在写文章的时候才敲的代码.这样会显得文章有些许混乱.但是我想这样记录下来,一个自定义view的真正的制作过程,是一点一点,一步步跟

Android 自定义View修炼-自定义弹幕效果View

一.概述 现在有个很流行的效果就是弹幕效果,满屏幕的文字从右到左飘来飘去.看的眼花缭乱,看起来还蛮cool的 现在就是来实现这一的一个效果,大部分的都是从右向左移动漂移,本文的效果中也支持从左向右的漂移移动 效果. 二.效果图 废话不说,先来看看效果图吧~~ 三.实现原理方案 1.自定义ViewGroup-XCDanmuView,继承RelativeLayout来实现,当然也可以继承其他三大布局类哈 2.初始化若干个TextView(弹幕的item View,这里以TextView 为例,当然也