自定义滑动按钮为例图文剖析Android自定义View绘制

自定义View一直是横在Android开发者面前的一道坎。

一、View和ViewGroup的关系

从View和ViewGroup的关系来看,ViewGroup继承View。

View的子类,多是功能型的控件,提供绘制的样式,比如imageView,TextView等,而ViewGroup的子类,多用于管理控件的大小,位置,如LinearLayout,RelativeLayout等,从下图可以看出

从实际应用中看,他们又是组合关系,我们在布局中,常常是一个ViewGroup嵌套多个ViewGroup或View,而被嵌套的ViewGroup又会嵌套多个ViewGroup或View

如下

二、View的绘制流程

从View源码来看,主要关系三个方法:

1、measure():测量
     一个final方法,控制控件的大小
2、layout():布局
         用来控制自己的布局位置
          有相对性,只相对于自己的父类布局,不关心祖宗布局
3、draw():绘制
          用来控制控件的显示样式

流程:  流程 measure --> layout --> draw

对应于我们要实现的方法是

onMeasure()

onLayout()

onDraw()

实际绘制中,我们的思考顺序一般是这样的:

是否需要控制控件的大小-->是-->onMeasure()
(1)如果这个自定义view不是ViewGroup,onMeasure()方法调用setMeasureDeminsion(width,height):用来设置自己的大小
(2)如果是ViewGroup,onMeasure()方法调用 ,child.measure()测量孩子的大小,给出孩子的期望大小值,之后-->setMeasureDeminsion(width,height):用来设置自己的大小

是否需要控制控件的摆放位置-->是 -->onLayout ()

是否需要控制控件的样子-->是 -->onDraw ()-->canvas的绘制

下面是我绘制的流程图:

下面以自定义滑动按钮为例,说明自定义View的绘制流程

我们期待实现这样的效果:

拖动或点击按钮,开关向右滑动,变成

其中开关能随着手指的触摸滑动到相应位置,直到最后才固定在开关位置上

新建一个类继承自View,实现其两个构造方法

public class SwitchButtonView extends View { 

  public SwitchButtonView(Context context) {
    this(context, null);
  } 

  public SwitchButtonView(Context context, AttributeSet attrs) {
    super(context, attrs);
  } 

drawable资源中添加这两张图片

借此,我们可以用onMeasure()确定这个控件的大小

@Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 

    mSwitchButton = BitmapFactory.decodeResource(getResources(), R.drawable.switch_background);
    mSlideButton = BitmapFactory.decodeResource(getResources(), R.drawable.slide_button_background);
    setMeasuredDimension(mSwitchButton.getWidth(), mSwitchButton.getHeight());
  }

这个控件并不需要控制其摆放位置,略过onLayout();

接下来onDraw()确定其形状。

但我们需要根据我们点击控件的不同行为来确定形状,这需要重写onTouchEvent()

其中的逻辑是:

当按下时,触发事件MotionEvent.Action_Down,若此时状态为关:

(1)若手指触摸点(通过event.getX()得到)在按钮的 中线右侧,按钮向右滑动一段距离(event.getX()与开关控件一半宽度之差,具体看下文源码)

(2)若手指触摸点在按钮中线左侧,按钮依旧处于最左(即“关”的状态)。

若此时状态为开:

(1)若手指触摸点在按钮中线左侧,按钮向左滑动一段距离

(2)若手指触摸点在按钮中线右侧,按钮依旧处于最右(即“开”的状态)

当滑动时,触发时间MotionEvent.Action_MOVE,逻辑与按下时一致

注意,onTouchEvent()需要设置返回值 为 Return true,否则无法响应滑动事件

当手指收起时,若开关中线位于整个控件中线左侧,设置状态为关,反之,设置为开。

具体源码如下所示:(还对外提供了一个暴露此时开关状态的接口)

自定义View部分

package com.lian.switchtogglebutton; 

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View; 

/**
 * Created by lian on 2016/3/20.
 */
public class SwitchButtonView extends View { 

  private static final int STATE_NULL = 0;//默认状态
  private static final int STATE_DOWN = 1;
  private static final int STATE_MOVE = 2;
  private static final int STATE_UP = 3; 

  private Bitmap mSlideButton;
  private Bitmap mSwitchButton;
  private Paint mPaint = new Paint();
  private int buttonState = STATE_NULL;
  private float mDistance;
  private boolean isOpened = false;
  private onSwitchListener mListener; 

  public SwitchButtonView(Context context) {
    this(context, null);
  } 

  public SwitchButtonView(Context context, AttributeSet attrs) {
    super(context, attrs);
  } 

  @Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 

    mSwitchButton = BitmapFactory.decodeResource(getResources(), R.drawable.switch_background);
    mSlideButton = BitmapFactory.decodeResource(getResources(), R.drawable.slide_button_background);
    setMeasuredDimension(mSwitchButton.getWidth(), mSwitchButton.getHeight());
  } 

  @Override
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    if (mSwitchButton!= null){
      canvas.drawBitmap(mSwitchButton, 0, 0, mPaint);
    }
    //buttonState的值在onTouchEvent()中确定
    switch (buttonState){
      case STATE_DOWN:
      case STATE_MOVE:
        if (!isOpened){
          float middle = mSlideButton.getWidth() / 2f;
          if (mDistance > middle) {
            float max = mSwitchButton.getWidth() - mSlideButton.getWidth();
            float left = mDistance - middle;
            if (left >= max) {
              left = max;
            }
            canvas.drawBitmap(mSlideButton,left,0,mPaint);
          } 

          else { 

            canvas.drawBitmap(mSlideButton,0,0,mPaint);
          }
        }else{
          float middle = mSwitchButton.getWidth() - mSlideButton.getWidth() / 2f;
          if (mDistance < middle){
            float left = mDistance-mSlideButton.getWidth()/2f;
            float min = 0;
            if (left < 0){
              left = min;
            }
            canvas.drawBitmap(mSlideButton,left,0,mPaint);
          }else{
            canvas.drawBitmap(mSlideButton,mSwitchButton.getWidth()-mSlideButton.getWidth(),0,mPaint);
          }
        } 

        break; 

      case STATE_NULL:
      case STATE_UP:
        if (isOpened){
          Log.d("开关","开着的");
          canvas.drawBitmap(mSlideButton,mSwitchButton.getWidth()-mSlideButton.getWidth(),0,mPaint);
        }else{
          Log.d("开关","关着的");
          canvas.drawBitmap(mSlideButton,0,0,mPaint);
        }
        break; 

      default:
        break;
    } 

  } 

  @Override
  public boolean onTouchEvent(MotionEvent event) { 

    switch (event.getAction()){
      case MotionEvent.ACTION_DOWN:
        mDistance = event.getX();
        Log.d("DOWN","按下");
        buttonState = STATE_DOWN;
        invalidate();
        break; 

      case MotionEvent.ACTION_MOVE:
        buttonState = STATE_MOVE;
        mDistance = event.getX();
        Log.d("MOVE","移动");
        invalidate();
        break; 

      case MotionEvent.ACTION_UP:
        mDistance = event.getX();
        buttonState = STATE_UP;
        Log.d("UP","起开");
        if (mDistance >= mSwitchButton.getWidth() / 2f){
          isOpened = true;
        }else {
          isOpened = false;
        }
        if (mListener != null){
          mListener.onSwitchChanged(isOpened);
        }
        invalidate();
        break;
      default:
        break;
    } 

    return true;
  } 

  public void setOnSwitchListener(onSwitchListener listener){
    this.mListener = listener;
  } 

  public interface onSwitchListener{
    void onSwitchChanged(boolean isOpened);
  }
} 

DemoActivity:

package com.lian.switchtogglebutton; 

import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.widget.Toast; 

public class MainActivity extends AppCompatActivity { 

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main); 

    SwitchButtonView switchButtonView = (SwitchButtonView) findViewById(R.id.switchbutton);
    switchButtonView.setOnSwitchListener(new SwitchButtonView.onSwitchListener() {
      @Override
      public void onSwitchChanged(boolean isOpened) {
        if (isOpened) {
          Toast.makeText(MainActivity.this, "打开", Toast.LENGTH_SHORT).show();
        }else {
          Toast.makeText(MainActivity.this, "关闭", Toast.LENGTH_SHORT).show();
        }
      }
    });
  }
} 

布局:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
  xmlns:android="http://schemas.android.com/apk/res/android"
  xmlns:tools="http://schemas.android.com/tools"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:paddingBottom="@dimen/activity_vertical_margin"
  android:paddingLeft="@dimen/activity_horizontal_margin"
  android:paddingRight="@dimen/activity_horizontal_margin"
  android:paddingTop="@dimen/activity_vertical_margin"
  tools:context="com.lian.switchtogglebutton.MainActivity"> 

  <com.lian.switchtogglebutton.SwitchButtonView
    android:id="@+id/switchbutton"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    />
</RelativeLayout> 

以上就是本文的全部内容,希望对大家的学习有所帮助。

(0)

相关推荐

  • Android自定义View之继承TextView绘制背景

    本文实例为大家分享了TextView绘制背景的方法,供大家参考,具体内容如下 效果: 实现流程: 1.初始化:对画笔进行设置 mPaintIn = new Paint(); mPaintIn.setAntiAlias(true); mPaintIn.setDither(true); mPaintIn.setStyle(Paint.Style.FILL); mPaintIn.setColor(getResources().getColor(R.color.colorPrimary)); mPain

  • Android使用自定义View绘制渐隐渐现动画

    实现了一个有趣的小东西:使用自定义View绘图,一边画线,画出的线条渐渐变淡,直到消失.效果如下图所示: 用属性动画或者渐变填充(Shader)可以做到一笔一笔的变化,但要想一笔渐变(手指不抬起边画边渐隐),没在Android中找到现成的API可用.所以,自己做了一个. 基本的想法是这样的: 在View的onTouchEvent中记录触摸点,生成一条一条的线LineElement,放在一个List中.给每个LineElement配置一个Paint实例. 在onDraw中绘制线段. 变换LineE

  • Android自定义view绘制圆环占比动画

    一.实现效果图 二.核心代码 1.自定义MyProgressView.java package com.czhappy.effectdemo.view; import android.content.Context; import android.content.res.TypedArray; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.Canvas;

  • Android自定义View实现shape图形绘制

    概述 之前曾写过一篇文章介绍了Android中drawable使用Shape资源,通过定义drawable中的shape资源能够绘制简单的图形效果,如矩形,椭圆形,线形和圆环等.后来我在项目中正好遇到这样一个需求,要在特定的位置上显示一条垂直的虚线.正当我胸有成竹的把上面的资源文件放入进去的时候,我才发现它并不能符合我的要求.使用shape画出的垂直虚线,其实就是将一条水平的线,旋转90度.但这样做的弊端就是,该View有效区域为旋转90度后与原来位置相重合的区域,还不能随意的改动,这样的效果根

  • Android自定义View实现绘制虚线的方法详解

    前言 说实话当第一次看到这个需求的时候,第一反应就是Canvas只有drawLine方法,并没有drawDashLine方法啊!这咋整啊,难道要我自己做个遍历不断的drawLine?不到1秒,我就放弃这个想法了,因为太恶心了.方法肯定是有的,只不过我不知道而已. 绘制方法 最简单的方法是利用ShapeDrawable,比如说你想用虚线要隔开两个控件,就可以在这两个控件中加个View,然后给它个虚线背景. 嗯,理论上就是这样子的,实现上也很简单. <!-- drawable 文件 --> <

  • Android开发使用自定义View将圆角矩形绘制在Canvas上的方法

    本文实例讲述了Android开发使用自定义View将圆角矩形绘制在Canvas上的方法.分享给大家供大家参考,具体如下: 前几天,公司一个项目中,头像图片需要添加圆角,这样UI效果会更好看,于是写了一个小的demo进行圆角的定义,该处主要是使用BitmapShader进行了渲染(如果要将一张图片裁剪成椭圆或圆形显示在屏幕上,也可以使用BitmapShader来完成). BitmapShader类完成渲染图片的基本步骤如下: 1.创建BitmapShader类的对象 /** * Call this

  • Android自定义View绘制的方法及过程(二)

    上一篇<Android 自定义View(一) Paint.Rect.Canvas介绍>讲了最基础的如何自定义一个View,以及View用到的一些工具类.下面讲下View绘制的方法及过程 public class MyView extends View { private String TAG = "--------MyView"; private int width, height; public MyView(Context context, AttributeSet a

  • Android编程开发之在Canvas中利用Path绘制基本图形(圆形,矩形,椭圆,三角形等)

    本文实例讲述了Android编程开发之在Canvas中利用Path绘制基本图形的方法.分享给大家供大家参考,具体如下: 在Android中绘制基本的集合图形,本程序就是自定义一个View组件,程序重写该View组件的onDraw(Canvase)方法,然后在该Canvas上绘制大量的基本的集合图形. 直接上代码: 1.自定义的View组件代码: package com.infy.configuration; import android.content.Context; import andro

  • Android使用Canvas绘制圆形进度条效果

    前言 Android自定义控件经常会用到Canvas绘制2D图形,在优化自己自定义控件技能之前,必须熟练掌握Canvas绘图机制.本文从以下三个方面对Canvas绘图机制进行讲解: 画布Canvas 画笔Paint 示例圆形进度条 画布Canvas 首先,来看一下Android官网对Canvas类的定义: The Canvas class holds the "draw" calls.To draw something, you need 4 basic components: A B

  • Android自定义View绘制随机生成图片验证码

    本篇文章讲的是Android自定义View之随机生成图片验证码,开发中我们会经常需要随机生成图片验证码,但是这个是其次,主要还是想总结一些自定义View的开发过程以及一些需要注意的地方. 按照惯例先看看效果图: 一.先总结下自定义View的步骤: 1.自定义View的属性 2.在View的构造方法中获得我们自定义的属性 3.重写onMesure 4.重写onDraw 其中onMesure方法不一定要重写,但大部分情况下还是需要重写的 二.View 的几个构造函数 1.public CustomV

  • Android自定义View系列之Path绘制仿支付宝支付成功动画

    前言 使用支付宝付款时,我们可以看到成功或者失败都会有个动画提示,如果我们需要做这样的效果的话,当然,你可以让设计师给你做个GIF,但是我们知道图像比较耗内存的,我们自己可以用代码实现还是代码实现好点吧. 效果 实现方法 首先我们需要了解PathMeasure这个类,这个类我们可以理解为用来管理Path.我们主要看几个方法. PathMeasure(): 构造方法 ,实例化一个对象 PathMeasure(Path path,boolean isClosed):传入Path对象和是否闭合,pat

随机推荐