我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Android中如何自定义刮刮卡

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Android中如何自定义刮刮卡

小编给大家分享一下Android中如何自定义刮刮卡,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!

效果:

Android中如何自定义刮刮卡

Android中如何自定义刮刮卡

所涉及的知识点:

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

实现思路:

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

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

步骤:

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

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

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

再来绘制一张和背景层大小一致的灰色图层作为前景层,这里我们需要用到绘图的双缓冲机制(这里的缓冲区指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里的图层叠加)

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

Android中如何自定义刮刮卡

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

这里我们需要取的是背景层的内容,也就是DST和 class="lazy" data-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;  }

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

步骤:

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

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

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

Android中如何自定义刮刮卡

关于文字位置的确定

首先我们需要知道任何的控件在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];            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循环可能有的朋友会看的有点懵,没着急,我画一张图,你就能懂。

Android中如何自定义刮刮卡

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;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];            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();        }      }    }  };}

以上是“Android中如何自定义刮刮卡”这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注编程网行业资讯频道!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Android中如何自定义刮刮卡

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android中如何自定义刮刮卡

小编给大家分享一下Android中如何自定义刮刮卡,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!效果:所涉及的知识点:1、自定义View的一些流程2、双缓冲绘图机
2023-05-30

基于Android自定义控件实现刮刮乐效果

只是简单的实现了效果,界面没怎么做优化,不过那都是次要的啦!!相信大家都迫不及待的想看效果图了吧,其中主要的彩票视图类和橡皮擦类都是通过代码的方式构建视图,布局文件就一个主activity_main上代码!!主activity:packa
2022-06-06

如何在Android中自定义Dialog

本篇文章为大家展示了如何在Android中自定义Dialog,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。功能:android 提供给我们的只有2种Dialog 即 AlertDialog & Pr
2023-05-31

如何在Android中自定义ProgressBar

如何在Android中自定义ProgressBar?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。具体如下:
2023-05-30

Android如何自定义Dialog

本篇内容主要讲解“Android如何自定义Dialog”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Android如何自定义Dialog”吧!接下来看代码:/** * Created by o
2023-07-02

Android中如何自定义对话框

本文小编为大家详细介绍“Android中如何自定义对话框”,内容详细,步骤清晰,细节处理妥当,希望这篇“Android中如何自定义对话框”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。本文测试的harbor的版本是
2023-06-29

Android如何自定义view实现日历打卡签到

这篇文章主要介绍Android如何自定义view实现日历打卡签到,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!Android是什么Android是一种基于Linux内核的自由及开放源代码的操作系统,主要使用于移动设备
2023-06-14

Android如何实现自定义View中attrs.xml

这篇文章主要为大家展示了“Android如何实现自定义View中attrs.xml”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Android如何实现自定义View中attrs.xml”这篇文章
2023-05-30

如何在Android中自定义UI组件

如何在Android中自定义UI组件?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。Android开发自定义UI组件实现红色小球跟随手指移动要写实现自定义UI组件,要创建一个Ba
2023-06-15

如何在Android中自定义ContentProvider与ContentObserver

今天就跟大家聊聊有关如何在Android中自定义ContentProvider与ContentObserver,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。Android 中自定义C
2023-05-30

如何在Android项目中自定义title

这篇文章给大家介绍如何在Android项目中自定义title,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。首先编写title的布局文件,title.xml:
2023-05-31

Android如何自定义加载圈

本篇内容介绍了“Android如何自定义加载圈”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!
2023-07-02

android如何自定义对话框

这篇文章给大家分享的是有关android如何自定义对话框的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。1.实现效果 2.定义dialog.xml (res/layout/dialog.xml)
2023-06-22

Android自定义控件如何在XML文件中使用自定义属性

这篇文章主要为大家介绍了Android自定义控件之如何在XML文件中使用自定义属性示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2023-05-14

Android Compose自定义TextField如何实现自定义的输入框

这篇文章主要介绍Android Compose自定义TextField如何实现自定义的输入框,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!简单自定义BasicTextField示例1.代码 var text by r
2023-06-29

RHEL7中如何添加自定义网卡并更改网卡接口

这篇文章给大家分享的是有关RHEL7中如何添加自定义网卡并更改网卡接口的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。环境VMware Workstation Pro Red Hat Enterprise Linu
2023-06-27

Android自定义组件:2、如何实现和使用自定义组件、自定义属性

声明:本教程不收取任何费用,欢迎转载,尊重作者劳动成果,不得用于商业用途,侵权必究!!! 目录 一、前言 二、如何实现自定义组件 步骤1:写 attrs.xml 资源文件 1、attrs.xml 文件 和 R 文件对应关系 2、attrs.
2022-06-06

Android如何自定义视图属性

本文实例为大家介绍了Android自定义视图属性的方法,供大家参考,具体内容如下 1. 自定义一个自己的视图类继承自Viewpublic class MyView extends View {public MyView(Context co
2022-06-06

Android如何自定义评分控件

今天小编给大家分享一下Android如何自定义评分控件的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。自定义参数为了方便扩展,
2023-06-30

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录