Android开发之自定义刮刮卡实现代码

关于刮刮卡的实现效果不需要做太多解释,特别是在电商APP中,每当做活动的时候都会有它的身影存在,趁着美好周末,来实现下这个效果,也算是对零碎知识点的一个整合。

所涉及的知识点:

1、自定义View的一些流程
2、双缓冲绘图机制
3、Paint的绘图模式
4、触摸事件的一些流程
5、Bitmap的相关知识

实现思路:

其实非常简单,首先我们需要确定所要绘图的区域,然后对这块区域进行多层的绘图(背景层,前景层),然后去监听触摸事件,把手指触摸的区域的前景层给消除即可。

首先我们先来实现一个简单版的:

步骤:

1、绘制图片作为背景层
2、绘制一张和背景层大小一致的灰色图层作为前景层
3、监听手指的触摸区域,把对应区域的前景层消除

1、首先绘制图片作为背景层,这个太简单了,我们把资源文件转成Bitmap对象,然后利用onDraw(Canvas canvas)里的Canvas画出来即可。

//背景图
mBackGroundBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.background);
  @Override
  protected void onDraw(Canvas canvas) {
    //绘制背景层
    canvas.drawBitmap(mBackGroundBitmap, 0, 0, null);
  }

2、再来绘制一张和背景层大小一致的灰色图层作为前景层,这里我们需要用到绘图的双缓冲机制(这里的缓冲区指Bitmap对象)。

双缓冲机制:先将要绘制的图形以对象的形式存放在内存中,作为绘制缓冲区,然后在这个对象上进行一系列的操作,然后再将其绘制到屏幕,避免过多的操作使得在绘制的过程中出现屏幕闪烁现象。

    //背景图
    mBackGroundBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.background);
    //创建一个和背景图大小一致的Bitmap对象作为装载画布
    mForeGroundBitmap = Bitmap.createBitmap(mBackGroundBitmap.getWidth(), mBackGroundBitmap.getHeight(), Config.ARGB_8888);
    //与Canvas进行绑定
    mCanvas = new Canvas(mForeGroundBitmap);
    //涂成灰色
    mCanvas.drawColor(Color.GRAY);
  @Override
  protected void onDraw(Canvas canvas) {
    //绘制背景层
    canvas.drawBitmap(mBackGroundBitmap, 0, 0, null);
    //绘制前景层
    canvas.drawBitmap(mForeGroundBitmap, 0, 0, null);
  }

运行此时的代码,你会发现背景层已经和前景层融为一体(其实是2个图层,类似于PS里的图层叠加)

3、监听手指的触摸区域,把对应区域的前景层消除,这里我们需要用到一个技巧,在Paint画笔API中给我们提供了一个PorterDuffXfermode,它有点想数学里的交并集,是用来控制两个图像之间的混合显示模式。

在这里它会先去绘制DST层再绘制SRC层,那么对应着下来就是背景层(DST)和前景层(SRC),那么在这个图像我们怎么去选择模式呢?

这里我们需要取的是背景层的内容,也就是DST和 SRC的交集,然后内容区域显示DST,那么也就是DstIn模式,来看下关于画笔Paint的设置。

    mPaint = new Paint();
    mPaint.setAlpha(0);
    mPaint.setAntiAlias(true);
    mPaint.setStyle(Paint.Style.STROKE);
    mPaint.setStrokeCap(Paint.Cap.ROUND);
    mPaint.setStrokeJoin(Paint.Join.ROUND);
    mPaint.setStrokeWidth(80);
    mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_IN));

然后我们重写onTouchEvent在手指按下屏幕和滑动屏幕的时候利用Path去记录我们想要擦除的路径即可。

  @Override
  public boolean onTouchEvent(MotionEvent event) {
    switch (event.getAction()) {
      case MotionEvent.ACTION_DOWN:
        mLastX = (int) event.getX();
        mLastY = (int) event.getY();
        mPath.moveTo(mLastX, mLastY);
        break;
      case MotionEvent.ACTION_MOVE:
        mLastX = (int) event.getX();
        mLastY = (int) event.getY();
        mPath.lineTo(mLastX, mLastY);
        break;
      case MotionEvent.ACTION_UP:
        break;
      default:
        break;
    }

    mCanvas.drawPath(mPath, mPaint);
    invalidate();

    return true;
  }

接下来我们来实现一个完整版的刮刮卡:

步骤:

1、绘制中奖信息作为背景层
2、绘制一张和中奖信息同等大小的刮奖封面作为前景层
3、监听手指的触摸区域,把对应区域的前景层消除
4、在消除大部分区域的时候,讲中奖信息完整展示

步骤1、2、3和前面大体一致,这里我就不详细说了,来讲一下需要注意的几个点:

1、在绘制中奖信息(文本)的时候,如何确定绘制的位置:

关于文字位置的确定

首先我们需要知道任何的控件在Android的布局中外层都是一个矩形的,A代表刮刮卡绘制区域,B代表中奖信息绘制区域,所以在这里我们绘制文本信息的起始点应该是A布局宽的一半减去B布局宽的一半,同理,高也应该是A布局高的一半减去B布局高的一半,这里我们把B布局,也就是文字控件的大小信息用一个Rect对象来存储,而这里的A布局即为Bitmap背景图的大小。

    //文字画笔
    mTextPaint = new Paint();
    mTextPaint.setAntiAlias(true);
    mTextPaint.setColor(Color.GREEN);
    mTextPaint.setStyle(Paint.Style.FILL);
    mTextPaint.setTextSize(30);
    mTextPaint.getTextBounds(mText, 0, mText.length(), mRect);
@Override
  protected void onDraw(Canvas canvas) {
    canvas.drawText(mText, mBitmap.getWidth() / 2 - mRect.width() / 2, mBitmap.getHeight() / 2 + mRect.height() / 2, mTextPaint);
  }

这样我们就绘制好了背景层的中奖信息,再来就是前景层,和上面一样我们利用资源文件转Bitmap对象然后绑定Canvas并绘制上刮刮卡图案

    //通过资源文件创建Bitmap对象
    mBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.background);
    //新建同等大小的Bitmap对象
    mForeBitmap = Bitmap.createBitmap(mBitmap.getWidth(), mBitmap.getHeight(), Bitmap.Config.ARGB_8888);
    //双缓冲,装载画布
    mForeCanvas = new Canvas(mForeBitmap);
    mForeCanvas.drawBitmap(mBitmap, 0, 0, null);

剩下的利用Path来记录用户手指触摸路径就是一样的了,这里我们额外来添加一个功能,使得当用户在刮刮卡上刮的区域范围超过50%后,自动消除刮刮卡前景层。

我们通过Bitmap的getPixels方法就可以拿到Bitmap的像素信息,由于这里涉及到了计算,这是个耗时操作,所以这里我们开启一个子线程来执行任务

private Runnable mRunnable = new Runnable() {
    int[] pixels;

    @Override
    public void run() {

      int w = mForeBitmap.getWidth();
      int h = mForeBitmap.getHeight();

      float wipeArea = 0;
      float totalArea = w * h;

      pixels = new int[w * h];
      /**
       * pixels   接收位图颜色值的数组
       * offset   写入到pixels[]中的第一个像素索引值
       * stride   pixels[]中的行间距个数值(必须大于等于位图宽度)。可以为负数
       * x      从位图中读取的第一个像素的x坐标值。
       * y      从位图中读取的第一个像素的y坐标值
       * width    从每一行中读取的像素宽度
       * height    读取的行数
       */
      mForeBitmap.getPixels(pixels, 0, w, 0, 0, w, h);

      for (int i = 0; i < w; i++) {
        for (int j = 0; j < h; j++) {
          int index = i + j * w;
          if (pixels[index] == 0) {
            wipeArea++;
          }
        }
      }

      if (wipeArea > 0 && totalArea > 0) {
        int percent = (int) (wipeArea * 100 / totalArea);
        if (percent > 50) {
          isClear = true;
          postInvalidate();
        }
      }

    }
  };

首先我们声明一个数组来记录像素点信息,数组的大小即为像素总数的大小也就是Bitmap的宽高,然后我们在onTouchEvent里的ACTION_UP中去计算被擦除的像素值,这里的for循环可能有的朋友会看的有点懵,没着急,我画一张图,你就能懂。

Bitmap像素点

我们第一层for循环i指的是Bitmap的宽,第二次层for循环j指的是Bitmap的高,那么index=i+jw,假设这个Bitmap的像素大小是3*3,那么index的值就是0,3,6,1,4,7,2,5,8,是不是有感觉了?我们遍历像素点是按照纵向下来的,当pixels的值为0的时候,证明已经是被用户擦除掉的像素点。

当被擦除的区域超出50%,我们就在onDraw里去控制不让canvas绘制前景图即可。

  @Override
  protected void onDraw(Canvas canvas) {
    canvas.drawText(mText, mForeBitmap.getWidth() / 2 - mRect.width() / 2, mForeBitmap.getHeight() / 2 + mRect.height() / 2, mTextPaint);
    if (!isClear) {
      canvas.drawBitmap(mForeBitmap, 0, 0, null);
    }
  }

下面贴一下完整版的代码:

package com.lcw.view;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Path;
import android.graphics.PorterDuff;
import android.graphics.PorterDuffXfermode;
import android.graphics.Rect;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;

/**
 * 刮刮卡(完善版)
 * Create by: chenwei.li
 * Date: 2017/7/22
 * Time: 下午7:25
 */

public class ScratchCardView2 extends View {

  //处理文字
  private String mText = "恭喜您中奖啦!!";
  private Paint mTextPaint;
  private Rect mRect;

  //处理图层
  private Paint mForePaint;
  private Path mPath;

  private Bitmap mBitmap;//加载资源文件
  private Canvas mForeCanvas;//前景图Canvas
  private Bitmap mForeBitmap;//前景图Bitmap

  //记录位置
  private int mLastX;
  private int mLastY;

  private volatile boolean isClear;//标志是否被清除

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

  public ScratchCardView2(Context context, AttributeSet attrs) {
    this(context, attrs, 0);
  }

  public ScratchCardView2(Context context, AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    init();
  }

  private void init() {

    mRect = new Rect();
    mPath = new Path();

    //文字画笔
    mTextPaint = new Paint();
    mTextPaint.setAntiAlias(true);
    mTextPaint.setColor(Color.GREEN);
    mTextPaint.setStyle(Paint.Style.FILL);
    mTextPaint.setTextSize(30);
    mTextPaint.getTextBounds(mText, 0, mText.length(), mRect);

    //擦除画笔
    mForePaint = new Paint();
    mForePaint.setAntiAlias(true);
    mForePaint.setAlpha(0);
    mForePaint.setStrokeCap(Paint.Cap.ROUND);
    mForePaint.setStrokeJoin(Paint.Join.ROUND);
    mForePaint.setStyle(Paint.Style.STROKE);
    mForePaint.setStrokeWidth(30);
    mForePaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_IN));

    //通过资源文件创建Bitmap对象
    mBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.background);
    mForeBitmap = Bitmap.createBitmap(mBitmap.getWidth(), mBitmap.getHeight(), Bitmap.Config.ARGB_8888);
    //双缓冲,装载画布
    mForeCanvas = new Canvas(mForeBitmap);
    mForeCanvas.drawBitmap(mBitmap, 0, 0, null);

  }

  @Override
  protected void onDraw(Canvas canvas) {
    canvas.drawText(mText, mForeBitmap.getWidth() / 2 - mRect.width() / 2, mForeBitmap.getHeight() / 2 + mRect.height() / 2, mTextPaint);
    if (!isClear) {
      canvas.drawBitmap(mForeBitmap, 0, 0, null);
    }
  }

  @Override
  public boolean onTouchEvent(MotionEvent event) {
    switch (event.getAction()) {
      case MotionEvent.ACTION_DOWN:
        mLastX = (int) event.getX();
        mLastY = (int) event.getY();
        mPath.moveTo(mLastX, mLastY);
        break;
      case MotionEvent.ACTION_MOVE:
        mLastX = (int) event.getX();
        mLastY = (int) event.getY();
        mPath.lineTo(mLastX, mLastY);
        break;
      case MotionEvent.ACTION_UP:
        new Thread(mRunnable).start();
        break;
      default:
        break;
    }

    mForeCanvas.drawPath(mPath, mForePaint);
    invalidate();
    return true;
  }

  /**
   * 开启子线程计算被擦除的像素点
   */
  private Runnable mRunnable = new Runnable() {
    int[] pixels;

    @Override
    public void run() {

      int w = mForeBitmap.getWidth();
      int h = mForeBitmap.getHeight();

      float wipeArea = 0;
      float totalArea = w * h;

      pixels = new int[w * h];
      /**
       * pixels   接收位图颜色值的数组
       * offset   写入到pixels[]中的第一个像素索引值
       * stride   pixels[]中的行间距个数值(必须大于等于位图宽度)。可以为负数
       * x      从位图中读取的第一个像素的x坐标值。
       * y      从位图中读取的第一个像素的y坐标值
       * width    从每一行中读取的像素宽度
       * height    读取的行数
       */
      mForeBitmap.getPixels(pixels, 0, w, 0, 0, w, h);

      for (int i = 0; i < w; i++) {
        for (int j = 0; j < h; j++) {
          int index = i + j * w;
          if (pixels[index] == 0) {
            wipeArea++;
          }
        }
      }

      if (wipeArea > 0 && totalArea > 0) {
        int percent = (int) (wipeArea * 100 / totalArea);
        if (percent > 50) {
          isClear = true;
          postInvalidate();
        }
      }

    }
  };
}

源码下载:

这里附上源码地址:源码下载 https://github.com/Lichenwei-Dev/ScratchCardView

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

(0)

相关推荐

  • Android App中实现简单的刮刮卡抽奖效果的实例详解

    主要思想: 将一个view设计成多层:背景层,含中奖信息等: 遮盖层,用于刮奖,使用关联一个Bitmap的Canvas 在该Bitmap上,使用它的canvas.drawPath的api来处理 手势滑动(类似刮奖的动作) 使用paint.setXfermode 来进行消除手势滑动区域 public class GuaView extends View { private Bitmap mBitmap; //遮盖的图层 private Canvas mCanvas; //绘制遮盖图层 privat

  • Android刮刮卡效果实现代码

    本文实例为大家分享了Android刮刮卡效果,供大家参考,具体内容如下 android实现底层一张图片,上层一个遮罩层,触摸滑动按手指滑动路径实现去除遮罩效果,类似于抽奖的刮刮卡一样,不多说先上张效果图: 直接上代码: XfermodeView.java /** * Created by 57 on 2016-4-21. */ public class XfermodeView extends View{ private Bitmap mBgBitmap,mFgBitmap; private P

  • 简单实现Android刮刮卡效果

    本文实例为大家分享了Android仿刮刮卡效果展示的具体代码,供大家参考,具体内容如下 一.Xfermode 通过使用Xfermode将绘制的图形的像素和Canvas上对应位置的像素按照一定的规则进行混合,形成新的像素,再更新到Canvas中形成最终的图形,使用的时候都是通过Paint.setXfermode来实现. 二.混合模式分类 PorterDuff则是用于描述数字图像合成的基本手法,通过组合使用Porter-Duff操作,可完成任意2D图像的合成. public class Porter

  • Android刮刮卡实现原理与代码讲解

    实现刮刮卡我们可以Get到哪些技能? * 圆形圆角图片的实现原理 * 双缓冲技术绘图 * Bitmap获取像素值数据 * 获取绘制文本的长宽 * 自定义View的掌握 * 获取屏幕密度 * TypeValue.applyDemension * Canvas的一些绘制方法 * Paint的一些常用的属性 * Path的一些方法 刮刮卡的实现原理图 这里用到了13中模式中的DstOut这种模式. 对于这幅图而言,首先绘制Dst,设置xfermode,再绘制Src. 刮刮卡的实现原理步骤 1.绘制显示

  • Android刮刮卡功能具体实现代码

    今天整理之前的代码,忽然看到之前自己写的一个刮刮卡,整理下以便以后使用,同时分享给需要的朋友,如有错误,还请多多指正. 实现的步骤,其实就是徒手画三个图层叠加在一起,最上层是绘制需要的问题,就是以上所述的"骚年,刮我吧",第二层就是覆盖宽高的灰层,第三层是结果层,多的不啰嗦了,具体实现如下,附上详细注释. /** * * created by zero on 2016-9-9 * * 刮刮卡 * */ public class ScratchView extends View { pu

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

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

  • Android开发之自定义CheckBox

    要实现的效果如下 考虑到关键是动画效果,所以直接继承View.不过CheckBox的超类CompoundButton实现了Checkable接口,这一点值得借鉴. 下面记录一下遇到的问题,并从源码的角度解决. 问题一: 支持 wrap_content 由于是直接继承自View,wrap_content需要进行特殊处理. View measure流程的MeasureSpec: /** * A MeasureSpec encapsulates the layout requirements pass

  • Android开发实现自定义水平滚动的容器示例

    本文实例讲述了Android开发实现自定义水平滚动的容器.分享给大家供大家参考,具体如下: public class HorizontalScrollView extends ViewGroup { //手势 private GestureDetector mGestureDetector; private HorizontalScroller mScroller; private int curID; //快速滑动 private boolean isFlying; //--回调函数-----

  • Android开发中自定义ProgressBar控件的方法示例

    本文实例讲述了Android开发中自定义ProgressBar控件的方法.分享给大家供大家参考,具体如下: 很简单,首先加载Drawable,在onMeasure设置好其区域大小, 然后使用canvas.clipRect绘图 public class ProgressView extends ImageView { private Drawable maskDraw; /** * 加载的进度 0-100 */ private int mProcess = 20; public ProgressV

  • Android开发之自定义view实现通讯录列表A~Z字母提示效果【附demo源码下载】

    本文实例讲述了Android开发之自定义view实现通讯录列表A~Z字母提示效果.分享给大家供大家参考,具体如下: 开发工具:eclipse 运行环境:htc G9 android2.3.3 话不多说,先看效果图 其实左右边的A~Z是一个自定义的View,它直接覆盖在ListView上. MyLetterListView: public class MyLetterListView extends View { OnTouchingLetterChangedListener onTouching

  • Android开发实现自定义新闻加载页面功能实例

    本文实例讲述了Android开发实现自定义新闻加载页面功能.分享给大家供大家参考,具体如下: 一.概述: 1.效果演示: 2.说明:在新闻页面刚加载的时候,一般会出现五种状态 未知状态(STATE_UNKNOW).空状态(STATE_EMPTY).加载中(STATE_LOADING).错误(STATE_ERROT).成功(STATE_SUCCESS) 因为每个Detail页面都会出现,所以我们可以把他们封装成一个LoadPage的自定义view,可以复用 二.实现: 1.首先的定义三个布局,为什

  • Android开发之自定义View(视图)用法详解

    本文实例讲述了Android开发之自定义View(视图)用法.分享给大家供大家参考,具体如下: View类是Android的一个超类,这个类几乎包含了所有的屏幕类型.每一个View都有一个用于绘图的画布,这个画布可以进行任意扩展.在游戏开发中往往需要自定义视图(View),这个画布的功能更能满足我们在游戏开发中的需要.在Android中,任何一个View类都只需重写onDraw 方法来实现界面显示,自定义的视图可以是复杂的3D实现,也可以是非常简单的文本形式等. 为了实现自定义View,需要创建

  • Android开发使用自定义view实现ListView下拉的视差特效功能

    本文实例讲述了Android开发使用自定义view实现ListView下拉的视差特效功能.分享给大家供大家参考,具体如下: 一.概述: 现在流型的APP如微信朋友圈,QQ空间,微博个人展示都有视差特效的影子. 如图:下拉图片会产生图片拉升的效果,放手后图片有弹回到原处: 那我们如何实现呢? 1)重写ListView控件: 2)重写里面的overScrollBy方法 3)在松手后执行值动画 二.具体实现: 1.创建ParallaListView 自定义ListView public class P

  • Android开发实现自定义Toast、LayoutInflater使用其他布局示例

    本文实例讲述了Android开发实现自定义Toast.LayoutInflater使用其他布局.分享给大家供大家参考,具体如下: 内容: 1.自定义样式toast 2.再活动中添加其他布局 实现效果: 步骤: 一.自定义View 引用zidingyixml文件 生成一个布局对象 二.采用Toast 的addView() 方法将该对象添加到Toast对象中 三.显示:Toast.show() 具体实现方法: public class MainActivity extends Activity {

  • Android开发之自定义星星评分控件RatingBar用法示例

    本文实例讲述了Android开发之自定义星星评分控件RatingBar用法.分享给大家供大家参考,具体如下: 星级评分条RatingBar类似于SeekBar.ProgressBar'等等都可以自定义样式 它的主要用途就比如淘宝.景点 满意度等 这里给出两种自定义效果 如图所示 第一种是通过RatingBar获得分数 第二个是通过RatingBar动态调节控件属性(透明度) 由于RatngBar使用简单 自定义样式方法和 https://www.jb51.net/article/158338.h

随机推荐