自定义View--刮刮卡

最终效果:

(1)先做简单的,效果图如下:

原理:最上层是一个(颜色值为#c0c0c0)bitmap,中间是一个画笔在画布上花下用户滑动的路径,最下层是一个背景图片的bitmap(见下图)

具体绘制的代码如下:

@Override
protected void onDraw(Canvas canvas) {
    canvas.drawBitmap(mOutterBitmap, 0, 0, null);//最上层
    mOutterPaint.setStyle(Paint.Style.STROKE);
    mOutterPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT));//Mode.DST_OUT改模式就类似橡皮檫,这个属性设置是关键
    mCanvas.drawPath(mPath, mOutterPaint);//中间层
    canvas.drawBitmap(mBitmap, 0, 0, null);//最下层
}

为什么中间层的画笔画了之后,会出现橡皮檫的效果,关键在于setXfermode()里面的参数,不同的参数效果对照下图:

,我们刚才说的中,src就是手指滑动的路径,DST就是最上面一个的遮罩。其实在onDraw()里面,drawPath与drawBitmap谁在前面都没有关系,你手滑动的路径会随时改变,所以这个不用争论什么,上面这种效果中,只要把两个drawBitmap的前后顺序弄清楚就可以了。作为背景的bitmap一定要在下面。

自定义view的代码:

public class GuaTwo extends View {
    private Path mPath;//手刮动的path,过程
    private Paint mOutterPaint;//绘制mPath的画笔
    private Canvas mCanvas;//临时画布
    private Bitmap mBitmap;//临时图片

    //记录用户path每次的开始坐标值
    private int mLastX;
    private int mLastY;

    private Bitmap mOutterBitmap;//图片遮罩,就是手刮动,要擦掉的那张图

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

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

    public GuaTwo(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        init();
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        //获得控件的宽高
        int width = getMeasuredWidth();
        int height = getMeasuredHeight();
        //初始化bitmap
        mBitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);
        mCanvas = new Canvas(mBitmap);

        //设置画笔属性
        setupOutPaint();
        mCanvas.drawColor(Color.parseColor("#c0c0c0"));
    }

    @Override
    protected void onDraw(Canvas canvas) {
        mOutterPaint.setStyle(Paint.Style.STROKE);
        mOutterPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT));//Mode.DST_OUT改模式就类似橡皮檫,这个属性设置是关键
        canvas.drawBitmap(mOutterBitmap, 0, 0, null);
        canvas.drawBitmap(mBitmap, 0, 0, null);
        mCanvas.drawPath(mPath, mOutterPaint);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        int action = event.getAction();
        int x = (int) event.getX();
        int y = (int) event.getY();

        switch (action) {
            case MotionEvent.ACTION_DOWN://按下
                //记录按下的时候的X和Y值,以便于之后移动的时候绘制
                mLastX = x;
                mLastY = y;
                mPath.moveTo(mLastX, mLastY);
                break;
            case MotionEvent.ACTION_MOVE://移动
                //拿到用户移动的X绝对值,Y轴绝对值
                int dx = Math.abs(x - mLastX);
                int dy = Math.abs(y - mLastY);
                //用户滑动超过3像素才会改变,这个可以不做,做只是为了避免很频繁的响应而已。
                if (dx > 3 || dy > 3) {
                    mPath.lineTo(x, y);
                }
                mLastX = x;
                mLastY = y;
                break;
        }
        invalidate();//刷新UI
        return true;
    }

    /**
     * 绘制path(也就是手刮动的path来绘制) 的画笔属性
     * 类似橡皮擦
     */
    private void setupOutPaint() {
        mOutterPaint.setColor(Color.RED);
        mOutterPaint.setAntiAlias(true);
        mOutterPaint.setDither(true);
        mOutterPaint.setStrokeJoin(Paint.Join.ROUND);//设置圆角
        mOutterPaint.setStrokeCap(Paint.Cap.ROUND);
        mOutterPaint.setStyle(Paint.Style.FILL);
        mOutterPaint.setStrokeWidth(60);//设置画笔宽度
    }
    /**
     * 初始化信息
     */
    private void init() {
        mOutterPaint = new Paint();
        mPath = new Path();
        mOutterBitmap = BitmapFactory.decodeResource(getResources(),
                R.drawable.image);
    }
}

(2)刮刮卡效果:

刮刮卡其实就和上面的原理一样,差别就是最上面层是图片,最下面的是文字信息而已。

最下面的是文字的实现:drawText()替代掉上面讲解中的最下层的图片

  @Override
    protected void onDraw(Canvas canvas) {
        mOutterPaint.setStyle(Paint.Style.STROKE);
        mOutterPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT));//Mode.DST_OUT改模式就类似橡皮檫,这个属性设置是关键
//        canvas.drawBitmap(mOutterBitmap, 0, 0, null);
        canvas.drawText(mText,(getWidth()-mTextBound.width())/2,getHeight()/2-mTextBound.height()/2,mBackPaint);//把获奖信息放在正中间
        mCanvas.drawPath(mPath, mOutterPaint);
        canvas.drawBitmap(mBitmap, 0, 0, null);
    }

刮刮卡上面的图片:在onMeasure()里面去drawBitmap(你要的图片遮罩),之后在onDraw()里面canvas.drawBitmap(mBitmap,0,0,
null);

 @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        //获得控件的宽高
        int width = getMeasuredWidth();
        int height = getMeasuredHeight();
        //初始化bitmap
        mBitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);
        mCanvas = new Canvas(mBitmap);

        //设置画笔属性
        setupOutPaint();
        setUpBackPaint();
        mCanvas.drawRoundRect(new RectF(0, 0, width, height), 30, 30,
                mOutterPaint);//用mOutterPaint画圆角矩形
        mCanvas.drawBitmap(mOutterBitmap, null, new Rect(0, 0, width, height),
                null);//在刚刚画的圆角矩形上面再画一个bitmap图片,让图片大小和圆角矩形大小相关联
    }

我在学习的时候有一个疑问,我在onDraw()方法里面,直接加载canvas.drawBitmap(mBgBitmap,0,0,
null);其中mBgBitmap是直接BitmapFactory.decodeResource加载的一张图片,比如上面效果图中的“刮刮乐”几个字的图片,结果是怎么都达不到我们想要的刮奖的效果,为什么不能在onDraw()里面drawBitmap,而必须在onMeasure()里面的drawBitmap,这个问题让我有些混乱。

后来看了一些大神的博客,发现,我们这种效果是分了两个画布的,上面代码中的mCanvas和onDraw()中系统传入的canvas是两个不同的画布,mCanvas画布负责画手指滑动的路径(drawPath),我们要把该path效果画在mBitmap上面,这样才可以出现path像橡皮檫一样去擦掉mBitmap上面的东西这样的一种效果。也就是说,PorterDuff.Mode.DST_OUT 这个要出现相应的效果的要求是,你所画的Src和DST是在一个画布上,也就是上面提到的画布mCanvas。

所以,我刚刚说的疑问中,我把遮罩和画笔画在了不同的画布上,所以怎么擦拭都擦不掉。

总的来说,就是mCanvas=
new
Canvas(mBitmap); 表示用指定的mBitmap构造一个画布(mCanvas)来绘制。那么后面在画布mCanvas上面画的东西都会画在mBitmap上面,所以mCanvas.drawBitmap(mOutterBitmap,
xxxx);效果就是把一个图片mOutterBitmap画在mBitmap上面,最后再把mBitmap画在屏幕上(canvas.drawBitmap(mBitmap,0,0,
null);这里面也包含了把path画在屏幕上(因为path画在临时图片mBitmap上面,mBitmap再画在屏幕上,所以path就画在了屏幕上)

自定义view的代码:

public class GuaTwo extends View {
    private Path mPath;//手刮动的path,过程
    private Paint mOutterPaint;//绘制mPath的画笔
    private Canvas mCanvas;
    private Bitmap mBitmap;

    //记录用户path每次的开始坐标值
    private int mLastX;
    private int mLastY;

    private Bitmap mOutterBitmap;//图片遮罩,就是手刮动,要擦掉的那张图
    private String mText;//刮奖文本信息
    private Rect mTextBound;
    private Paint mBackPaint;//刮奖信息的画笔

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

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

    public GuaTwo(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        init();
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        //获得控件的宽高
        int width = getMeasuredWidth();
        int height = getMeasuredHeight();
        //初始化bitmap
        mBitmap = Bitmap.createBitmap(width, height, Bitmap.Config.ARGB_8888);
        mCanvas = new Canvas(mBitmap);//用指定的位图构造一个画布来绘制。

        //设置画笔属性
        setupOutPaint();
        setUpBackPaint();

//        mCanvas.drawColor(Color.parseColor("#c0c0c0"));
        mCanvas.drawRoundRect(new RectF(0, 0, width, height), 30, 30,
                mOutterPaint);//用mOutterPaint画圆角矩形
        mCanvas.drawBitmap(mOutterBitmap, null, new Rect(0, 0, width, height),
                null);//在刚刚画的圆角矩形上面再画一个bitmap图片,让图片大小和圆角矩形大小相关联
    }

    @Override
    protected void onDraw(Canvas canvas) {
        mOutterPaint.setStyle(Paint.Style.STROKE);
        mOutterPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT));//Mode.DST_OUT改模式就类似橡皮檫,这个属性设置是关键
        canvas.drawText(mText, (getWidth() - mTextBound.width()) / 2, getHeight() / 2 - mTextBound.height() / 2, mBackPaint);//把获奖信息放在正中间
        mCanvas.drawPath(mPath, mOutterPaint);
        canvas.drawBitmap(mBitmap, 0, 0, null);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        int action = event.getAction();
        int x = (int) event.getX();
        int y = (int) event.getY();

        switch (action) {
            case MotionEvent.ACTION_DOWN://按下
                //记录按下的时候的X和Y值,以便于之后移动的时候绘制
                mLastX = x;
                mLastY = y;
                mPath.moveTo(mLastX, mLastY);
                break;
            case MotionEvent.ACTION_MOVE://移动
                //拿到用户移动的X绝对值,Y轴绝对值
                int dx = Math.abs(x - mLastX);
                int dy = Math.abs(y - mLastY);
                //用户滑动超过3像素才会改变,这个可以不做,做只是为了避免很频繁的相应而已。
                if (dx > 3 || dy > 3) {
                    mPath.lineTo(x, y);
                }
                mLastX = x;
                mLastY = y;
                break;
        }
        invalidate();//刷新UI
        return true;
    }
    private void setUpBackPaint() {
        mBackPaint.setColor(Color.RED);
        mBackPaint.setStyle(Paint.Style.FILL);
        mBackPaint.setTextSize(60);
        //获得当前画笔绘制文本的宽和高
        mBackPaint.getTextBounds(mText, 0, mText.length(), mTextBound);
    }
    /**
     * 绘制path(也就是手刮动的path来绘制) 的画笔属性
     * 类似橡皮擦
     */
    private void setupOutPaint() {
        mOutterPaint.setColor(Color.RED);
        mOutterPaint.setAntiAlias(true);
        mOutterPaint.setDither(true);
        mOutterPaint.setStrokeJoin(Paint.Join.ROUND);//设置圆角
        mOutterPaint.setStrokeCap(Paint.Cap.ROUND);
        mOutterPaint.setStyle(Paint.Style.FILL);
        mOutterPaint.setStrokeWidth(60);//设置画笔宽度
    }
    /**
     * 初始化信息
     */
    private void init() {
        mOutterPaint = new Paint();
        mPath = new Path();
        mOutterBitmap = BitmapFactory.decodeResource(getResources(),
                R.drawable.fg_guaguaka);
        mText = "谢谢惠顾!";
        mTextBound = new Rect();
        mBackPaint = new Paint();
    }
}

更加完善的源码下载:http://www.imooc.com/video/5148,包含了自定义属性(刮刮乐获奖信息的字体、颜色、大小)

时间: 2024-10-30 00:40:53

自定义View--刮刮卡的相关文章

【Android - View】之自定义View实现“刮刮卡”效果

首先来介绍一下这个自定义View: (1)这个自定义View的名字叫做 GuaguakaView ,继承自View类: (2)这个View实现了很多电商项目中的"刮刮卡"的效果,即用户可以刮开覆盖层,查看自己是否中奖: (3)用户可以设置覆盖层的图片以及显示的文本内容和字体大小等参数: (4)用户可以设置一个阈值,当刮开的面积大于这个阈值时,就会自动清除所有覆盖物. 接下来简单介绍一下在这个自定义View中用到的技术点: (1)自定义属性:在 /res/values/attr.xml 

Android 自定义View 实现刮刮卡效果

主要思想: 将一个view设计成多层:背景层,含中奖信息等: 遮盖层,用于刮奖,使用关联一个Bitmap的Canvas 在该Bitmap上,使用它的canvas.drawPath的api来处理 手势滑动(类似刮奖的动作) 使用paint.setXfermode 来进行消除手势滑动区域 /** * author : stone * email : [email protected] * time : 15/7/28 11 01 */ public class GuaView extends Vie

【2014年最后的分享啦】Android实现自定义刮刮卡效果View

一.简介: 今天是2014年最后一天啦,首先在这里,我祝福大家在新的2015年都一个个的新健康,新收入,新顺利,新如意!!! 上一偏,我介绍了用Xfermode实现自定义圆角和椭圆图片view的博文<Android实现自定义圆形.圆角和椭圆ImageView(使用Xfermode图形渲染方法)>, 今天我们来看看如何实现电商app里常用到的刮刮卡效果的view组件,其实原理和实现圆角图片的差不多,都是使用Xfermode渲染模式来实现的. (老规矩,源码在博文最后给出哈) 基本原理步骤是这样的

Android 刮刮卡自定义view

import android.content.Context; import android.graphics.Bitmap; import android.graphics.Bitmap.Config; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Paint; import android.graphics.Path; import android.graphics

Android 自定义控件实现刮刮卡效果 真的就只是刮刮卡么

很久以前也过一个html5的刮刮卡效果~~上次看到有人写Android的刮刮卡效果~~于是乎产生了本篇博客~~此类例子也比较多了,大家可以百度看看~不过还是通过本例子,带大家发掘一下,里面隐含的知识~ 1.Xfermode以及PorterDuff 如果大家还记得,曾经在博客:完美实现图片圆角和圆形 简单介绍过圆角的实现原理也是基于这个. 首先我们看一下官方的例子,很好的展示了16种Mode的效果: 注:先绘制的Dst,再绘制的Src. 好了,看了这个图,我来问大家几个问题: 问题1.如果我想实现

Android 刮刮卡实现效果

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

Android 刮刮卡

好久没有写博客了,今天有正好有时间,接下来为大家带来一款电商中要使用的刮刮卡项目,刮刮卡相信大家小时候都玩过,那个时候,可能大家使用的是小纸片的,现在也可以带大家重温小时候的美好时光,今天将这个刮刮卡的项目分享给大家,希望对大家有学习和工作上的帮助. 一:先上效果图 二:原理分析 1.创建一张你想要的bitmap大小,然后根据bitmap的大小,绘制一张画布在上面,然后重写onDraw()方法 2.在刮开面积的不断扩张中,计算扩张面的面积大小,当面积超过60%的时候,就清楚图层 3.下面的刮刮卡

封装了一个H5刮刮卡插件

下班后,闲着无事,刚好近期在学习画布相关知识,就写了个demo练一下手.高手路过,多多指点! 简单介绍一下页面整体结构: 1 <div class="wrap"> 2 <canvas class="page can" id="can" width="640" height="1136"></canvas> 3 <div class="page index&

Android 自己定义控件实现刮刮卡效果 真的就仅仅是刮刮卡么

转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/40162163 , 本文出自:[张鸿洋的博客] 非常久以前也过一个html5的刮刮卡效果~~上次看到有人写Android的刮刮卡效果~~于是乎产生了本篇博客~~此类样例也比較多了,大家能够百度看看~只是还是通过本样例,带大家发掘一下.里面隐含的知识~ 1.Xfermode以及PorterDuff 假设大家还记得,以前在博客:完美实现图片圆角和圆形 简介过圆角的实现原理也是基于这个

PorterDuffXfermode 实现刮刮卡效果

今天闲来无事做,就想着巩固一下 PorterDuffXfermode 的使用吧,先是做了圆形和圆角矩形头像效果,然后就做了个刮刮卡效果.如下图所示 当然 PorterDuffXfermode 的详细使用在这里我就不在强调了,可以去 Google 一下,下面主要来介绍一下效果的实现原理. 其实这个效果十分的简单,主要就是 PorterDuffXfermode 模式的设置和 Paint 的属性设置,下面来看看核心代码 mBitmap = BitmapFactory.decodeResource(ge