2018-09-26【自定义图片验证码】

Android自定义View之随机生成图片验证码,开发中我们会经常需要随机生成图片验证码,但是这个是其次,主要还是想总结一些自定义View的开发过程以及一些需要注意的地方。

一、先总结下自定义View的步骤: 

1、自定义View的属性 

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

3、重写onMesure 

4、重写onDraw 

其中onMesure方法不一定要重写,但大部分情况下还是需要重写的

二、View 的几个构造函数 

1、public CustomView(Context context) 

—>java代码直接new一个CustomView实例的时候,会调用这个只有一个参数的构造函数; 

2、public CustomView(Context context, AttributeSet attrs) 

—>在默认的XML布局文件中创建的时候调用这个有两个参数的构造函数。AttributeSet类型的参数负责把XML布局文件中所自定义的属性通过AttributeSet带入到View内; 

3、public CustomView(Context context, AttributeSet attrs, int defStyleAttr) 

—>构造函数中第三个参数是默认的Style,这里的默认的Style是指它在当前Application或者Activity所用的Theme中的默认Style,且只有在明确调用的时候才会调用 

4、public CustomView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) 

—>该构造函数是在API21的时候才添加上的

三、下面我们就开始来看看代码啦 

1、自定义View的属性,首先在res/values/ 下建立一个attr.xml , 在里面定义我们的需要用到的属性以及声明相对应属性的取值类型

<?xml version="1.0" encoding="utf-8"?>

<resources>

<attr name="text" format="string" />

<attr name="textColor" format="color" />

<attr name="textSize" format="dimension" />

<attr name="bgColor" format="color" />

<declare-styleable name="CustomView">

  <attr name="text" />

  <attr name="textColor" />

  <attr name="textSize" />

  <attr name="bgColor" />

</declare-styleable>

</resources>

我们定义了字体,字体颜色,字体大小以及字体的背景颜色4个属性,format是值该属性的取值类型,format取值类型总共有10种,包括:string,color,demension,integer,enum,reference,float,boolean,fraction和flag。

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

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"

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

android:layout_width="match_parent"

android:layout_height="match_parent">

<com.per.customview01.view.CustomView

  android:layout_width="wrap_content"

  android:layout_height="wrap_content"

  android:layout_centerInParent="true"

  android:padding="10dp"

  custom:bgColor="#FF27FF28"

  custom:text="J2RdWQG"

  custom:textColor="#ff0000"

  custom:textSize="36dp" />

</RelativeLayout>

一定要引入xmlns:custom=”http://schemas.android.com/apk/res-auto”,Android Studio中我们可以使用res-atuo命名空间,就不用在添加自定义View全类名。

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

/**

  * 文本

  */

private String mText;

/**

  * 文本的颜色

  */

private int mTextColor;

/**

  * 文本的大小

  */

private int mTextSize;

/**

  * 文本的背景颜色

  */

private int mBgCplor;

private Rect mBound;

private Paint mPaint;

public CustomView(Context context) {

  this(context, null);

}

public CustomView(Context context, AttributeSet attrs) {

  this(context, attrs, 0);

}

public CustomView(Context context, AttributeSet attrs, int defStyleAttr) {

  super(context, attrs, defStyleAttr);

  /**

  * 获得我们所定义的自定义样式属性

  */

  TypedArray a = context.getTheme().obtainStyledAttributes(attrs, R.styleable.CustomView, defStyleAttr, 0);

  for (int i = 0; i < a.getIndexCount(); i++) {

  int attr = a.getIndex(i);

  switch (attr) {

    case R.styleable.CustomView_text:

    mText = a.getString(attr);

    break;

    case R.styleable.CustomView_textColor:

    // 默认文本颜色设置为黑色

    mTextColor = a.getColor(R.styleable.CustomView_textColor, Color.BLACK);

    break;

    case R.styleable.CustomView_bgColor:

    // 默认文本背景颜色设置为蓝色

    mBgCplor = a.getColor(R.styleable.CustomView_bgColor, Color.BLUE);

    break;

    case R.styleable.CustomView_textSize:

    // 默认设置为16sp,TypeValue也可以把sp转化为px

    mTextSize = a.getDimensionPixelSize(attr, (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_SP, 16, getResources().getDisplayMetrics()));

    break;

  }

  }

  a.recycle();

  // 获得绘制文本的宽和高

  mPaint = new Paint();

  mPaint.setTextSize(mTextSize);

  mBound = new Rect();

  mPaint.getTextBounds(mText, 0, mText.length(), mBound);

}

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

一开始一个参数的构造方法和两个参数的构造方法是这样的:

public CustomView(Context context) {

  super(context);

}

public CustomView(Context context, AttributeSet attrs) {

  super(context, attrs);

}

有一点要注意的是super应该改成this,然后让一个参数的构造方法引用两个参数的构造方法,两个参数的构造方法引用三个参数的构造方法,代码如下:

public CustomView(Context context) {

  this(context, null);

}

public CustomView(Context context, AttributeSet attrs) {

  this(context, attrs, 0);

}

4、重写onDraw,onMesure方法

@Override

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

  super.onMeasure(widthMeasureSpec, heightMeasureSpec);

}

@Override

protected void onDraw(Canvas canvas) {

  super.onDraw(canvas);

  mPaint.setColor(mBgCplor);

  canvas.drawRect(0, 0, getMeasuredWidth(), getMeasuredHeight(), mPaint);

  mPaint.setColor(mTextColor);

  canvas.drawText(mText, getWidth() / 2 - mBound.width() / 2, getHeight() / 2 + mBound.height() / 2, mPaint);

}

View的绘制流程是从ViewRoot的performTravarsals方法开始的,经过measure、layout和draw三个过程才能最终将一个View绘制出来,其中:

•测量——onMeasure():用来测量View的宽和高来决定View的大小

•布局——onLayout():用来确定View在父容器ViewGroup中的放置位置

•绘制——onDraw():负责将View绘制在屏幕上 

来看下此时的效果图

细心的朋友会发现,在上面的布局文件中,我们是把宽和高设置为wrap_content的,可是这个效果图怎么看都不是我们想要的,这是因为系统帮我们测量的高度和宽度默认是MATCH_PARNET,当我们设置明确的宽度和高度时,系统帮我们测量的结果就是我们设置的结果,这个是对的。但是除了设置明确的宽度和高度,不管我们设置为WRAP_CONTENT还是MATCH_PARENT,系统帮我们测量的结果就是MATCH_PARENT,所以,当我们设置了WRAP_CONTENT时,我们需要自己进行测量,也就是说我们需要重写onMesure方法

下面是我们重写onMeasure代码:

@Override

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

  super.onMeasure(widthMeasureSpec, heightMeasureSpec);

  int widthMode = MeasureSpec.getMode(widthMeasureSpec);

  int widthSize = MeasureSpec.getSize(widthMeasureSpec);

  int heighMode = MeasureSpec.getMode(heightMeasureSpec);

  int heighSize = MeasureSpec.getSize(heightMeasureSpec);

  setMeasuredDimension(widthMode == MeasureSpec.EXACTLY ? widthSize : getPaddingLeft() + getPaddingRight() + mBound.width(), heighMode == MeasureSpec.EXACTLY ? heighSize : getPaddingTop() + getPaddingBottom() + mBound.height());

}

MeasureSpec封装了父布局传递给子布局的布局要求,MeasureSpec的specMode一共有三种模式: 

(1)EXACTLY(完全):一般是设置了明确的值或者是MATCH_PARENT,父元素决定了子元素的大小,子元素将被限定在给定的范围里而忽略它本身大小; 

(2)AT_MOST(至多):表示子元素至多达到给定的一个最大值,一般为WARP_CONTENT;

现在这个是我们想要的结果了吧,回归到主题,今天讲的是自定义View之随机生成图片验证码,现在把自定义View的部分完成了,我把完整的代码贴出来

package com.per.customview01.view;

import android.content.Context;

import android.content.res.TypedArray;

import android.graphics.Canvas;

import android.graphics.Color;

import android.graphics.Paint;

import android.graphics.Rect;

import android.util.AttributeSet;

import android.util.Log;

import android.util.TypedValue;

import android.view.View;

import com.per.customview01.R;

import java.util.Random;

/**

* @author: adan

* @description:

* @projectName: CustomView01

* @date: 2016-06-12

* @time: 10:26

*/

public class CustomView extends View {

private static final char[] CHARS = {'0', '1', '2', '3', '4', '5', '6',

  '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j',

  'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w',

  'x', 'y', 'z', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J',

  'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W',

  'X', 'Y', 'Z'};

/**

  * 初始化生成随机数的类

  */

private Random mRandom = new Random();

/**

  * 初始化可变字符串

  */

private StringBuffer sb = new StringBuffer();

/**

  * 文本

  */

private String mText;

/**

  * 文本的颜色

  */

private int mTextColor;

/**

  * 文本的大小

  */

private int mTextSize;

/**

  * 文本的背景颜色

  */

private int mBgCplor;

private Rect mBound;

private Paint mPaint;

public CustomView(Context context) {

  this(context, null);

}

public CustomView(Context context, AttributeSet attrs) {

  this(context, attrs, 0);

}

public CustomView(Context context, AttributeSet attrs, int defStyleAttr) {

  super(context, attrs, defStyleAttr);

  /**

  * 获得我们所定义的自定义样式属性

  */

  TypedArray a = context.getTheme().obtainStyledAttributes(attrs, R.styleable.CustomView, defStyleAttr, 0);

  for (int i = 0; i < a.getIndexCount(); i++) {

  int attr = a.getIndex(i);

  switch (attr) {

    case R.styleable.CustomView_text:

    mText = a.getString(attr);

    break;

    case R.styleable.CustomView_textColor:

    // 默认文本颜色设置为黑色

    mTextColor = a.getColor(R.styleable.CustomView_textColor, Color.BLACK);

    break;

    case R.styleable.CustomView_bgColor:

    // 默认文本背景颜色设置为蓝色

    mBgCplor = a.getColor(R.styleable.CustomView_bgColor, Color.BLUE);

    break;

    case R.styleable.CustomView_textSize:

    // 默认设置为16sp,TypeValue也可以把sp转化为px

    mTextSize = a.getDimensionPixelSize(attr, (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_SP, 16, getResources().getDisplayMetrics()));

    break;

  }

  }

  a.recycle();

  // 获得绘制文本的宽和高

  mPaint = new Paint();

  mPaint.setTextSize(mTextSize);

  mBound = new Rect();

  mPaint.getTextBounds(mText, 0, mText.length(), mBound);

  this.setOnClickListener(new OnClickListener() {

  @Override

  public void onClick(View v) {

    mText = createCode();

    mTextColor = randomColor();

    mBgCplor = randomColor();

    //View重新调用一次draw过程,以起到界面刷新的作用

    postInvalidate();

  }

  });

}

/**

  * 生成验证码

  */

public String createCode() {

  sb.delete(0, sb.length()); // 使用之前首先清空内容

  for (int i = 0; i < 6; i++) {

  sb.append(CHARS[mRandom.nextInt(CHARS.length)]);

  }

  Log.e("生成验证码", sb.toString());

  return sb.toString();

}

/**

  * 随机颜色

  */

private int randomColor() {

  sb.delete(0, sb.length()); // 使用之前首先清空内容

  String haxString;

  for (int i = 0; i < 3; i++) {

  haxString = Integer.toHexString(mRandom.nextInt(0xFF));

  if (haxString.length() == 1) {

    haxString = "0" + haxString;

  }

  sb.append(haxString);

  }

  Log.e("随机颜色", "#" + sb.toString());

  return Color.parseColor("#" + sb.toString());

}

@Override

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

  super.onMeasure(widthMeasureSpec, heightMeasureSpec);

  int widthMode = MeasureSpec.getMode(widthMeasureSpec);

  int widthSize = MeasureSpec.getSize(widthMeasureSpec);

  int heighMode = MeasureSpec.getMode(heightMeasureSpec);

  int heighSize = MeasureSpec.getSize(heightMeasureSpec);

  setMeasuredDimension(widthMode == MeasureSpec.EXACTLY ? widthSize : getPaddingLeft() + getPaddingRight() + mBound.width(), heighMode == MeasureSpec.EXACTLY ? heighSize : getPaddingTop() + getPaddingBottom() + mBound.height());

}

@Override

protected void onDraw(Canvas canvas) {

  super.onDraw(canvas);

  mPaint.setColor(mBgCplor);

  canvas.drawRect(0, 0, getMeasuredWidth(), getMeasuredHeight(), mPaint);

  mPaint.setColor(mTextColor);

  canvas.drawText(mText, getWidth() / 2 - mBound.width() / 2, getHeight() / 2 + mBound.height() / 2, mPaint);

}

}

我们添加了一个点击事件,每一次点击View我都让它把生成的验证码和字体颜色以及字体背景颜色打印出来

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容