Android 自己定义控件实现刮刮卡效果 真的就仅仅是刮刮卡么
转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/40162163 , 本文出自:【张鸿洋的博客】
非常久以前也过一个html5的刮刮卡效果~~上次看到有人写Android的刮刮卡效果~~于是乎产生了本篇博客~~此类样例也比較多了,大家能够百度看看~只是还是通过本样例,带大家发掘一下。里面隐含的知识~
1、Xfermode以及PorterDuff
假设大家还记得,以前在博客:完美实现图片圆角和圆形 简介过圆角的实现原理也是基于这个。
首先我们看一下官方的样例。非常好的展示了16种Mode的效果:
注:先绘制的Dst。再绘制的Src。
好了,看了这个图。我来问大家几个问题:
问题1、假设我想实现圆形图片。怎么实现?
答:先绘制我们的图片,然后在上面绘制一个圆。最后生成的效果就是圆形图片。等等,怎么就生成了,请看上面的SrcIn这样的模式;
先绘制的Dst。然后设置DstIn,然后绘制Src;最后效果是留下了二者交集且是Dst的部分;以下我们把我们的答案带进去。
先绘制图片,然后设置DstIn。然后绘制圆形。最后效果是留下了二者交集且是图片的部分。嗯,交集是什么。圆形;圆形内容是什么,图片;搜噶,有点感觉了。
----
等等,我还有有个思路。先绘制圆形,然后设置SrcIn,再绘制我们的图片;也能生成我们的圆形图片。
我们来看看:
SrcIn最终保留的依旧是交集,可是显示为后绘制的,也就是我们的图片,搜噶。这样也能够。
问题2、假设我想实现圆角图片,怎么实现?
答:擦。看了上面的答案。你还没思路么。
把绘制圆形,改成绘制圆角矩形。请问你还有什么问的,额,。。木有了。
嗯。把问题1的圆形改成圆角,依照同样的绘制过程就实现了我们的圆角图片了。
问题3、这和我们的刮刮卡有毛线关系?
答:怎么没有关系,,,你先绘制刮奖层,然后设置DST_OUT。然后把用户手触摸的线条绘制上去。用户触摸到刮奖层的部分(交集部分)会被消除。也是就说刮奖层被我们擦掉了~
这不就是刮奖么。等等,奖呢?
奖无非就是文本,或者图片,提前绘制一下。然后在其上绘制刮奖层。设置DST_OUT,然后把用户触摸绘制上去。这样消失以后就能看到背后的奖了~~~对了,如今还有个app叫脱什么妹子衣服。先绘制妹子,然后绘制衣服,然后擦~~这个和刮奖像不像,额,我什么都没说。
搜噶,经过上面的3个问题。大家应该明确了,什么圆角。圆形,刮刮卡。事实上原理就这么简单,,,
2、简易画板的实现
我们的刮刮卡须要掌握画图,当然了这里不要求你有美术天分,会瞎涂鸦就能够了~~
以下開始我们的一个简易的画板。事实上就是能够在上面画点线条,当然你也能够签个名。我们的View的叫做GuaGuaKa:
1、初步GuaGuaKa
package com.zhy.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.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View; public class GuaGuaKa extends View
{ /**
* 绘制线条的Paint,即用户手指绘制Path
*/
private Paint mOutterPaint = new Paint();
/**
* 记录用户绘制的Path
*/
private Path mPath = new Path();
/**
* 内存中创建的Canvas
*/
private Canvas mCanvas;
/**
* mCanvas绘制内容在其上
*/
private Bitmap mBitmap; private int mLastX;
private int mLastY; public GuaGuaKa(Context context)
{
this(context, null);
} public GuaGuaKa(Context context, AttributeSet attrs)
{
this(context, attrs, 0);
} public GuaGuaKa(Context context, AttributeSet attrs, int defStyle)
{
super(context, attrs, defStyle);
init();
} private void init()
{
mPath = new Path(); } @Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
{
super.onMeasure(widthMeasureSpec, heightMeasureSpec); int width = getMeasuredWidth();
int height = getMeasuredHeight();
// 初始化bitmap
mBitmap = Bitmap.createBitmap(width, height, Config.ARGB_8888);
mCanvas = new Canvas(mBitmap);
// 设置画笔
mOutterPaint.setColor(Color.RED);
mOutterPaint.setAntiAlias(true);
mOutterPaint.setDither(true);
mOutterPaint.setStyle(Paint.Style.STROKE);
mOutterPaint.setStrokeJoin(Paint.Join.ROUND); // 圆角
mOutterPaint.setStrokeCap(Paint.Cap.ROUND); // 圆角
// 设置画笔宽度
mOutterPaint.setStrokeWidth(20);
} @Override
protected void onDraw(Canvas canvas)
{
drawPath();
canvas.drawBitmap(mBitmap, 0, 0, null); } /**
* 绘制线条
*/
private void drawPath()
{
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:
mLastX = x;
mLastY = y;
mPath.moveTo(mLastX, mLastY);
break;
case MotionEvent.ACTION_MOVE: int dx = Math.abs(x - mLastX);
int dy = Math.abs(y - mLastY); if (dx > 3 || dy > 3)
mPath.lineTo(x, y); mLastX = x;
mLastY = y;
break;
} invalidate();
return true;
} }
代码量比較少。我们在内存中搞了一个mCanvas,创建了一个mBitmap。然后通过mCanvas使用我们预先设置的mOuterPaint在我们的mBitmap上绘制mPath;
mPath里面的数据怎么搞呢?就是onTouchEvent里面不断的moveTo。lineTo就好了~~代码还是非常任意的
最后,注意我们绘制内存上的mBitmap上面,然后我们通过view的canvas。把我们的mBitmap展现。咦。怎么有点双缓冲的感脚。
好了,如今你就能够在我们的画板上肆掠了:
以下看布局文件以及执行效果:
2、布局文件及执行效果
布局文件:
<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" > <com.zhy.view.GuaGuaKa
android:layout_width="match_parent"
android:layout_height="match_parent" /> </RelativeLayout>
执行效果:
看到我浑厚的字体没有,等以后写不动程序了,我就去当书法家~
好了。我们的简易画板完毕以后。我们開始考虑正题,一步一步逼近我们的刮刮板,如今我们准备这样做。首先在背后绘制一张图片,然后绘制一个遮盖层。然后我们绘画的过程就是擦除遮盖层。
3、擦除的第一次实现
鉴于非常多朋友的意见。我决定这次的图片用风景图,远离xxx , 感谢seven提供的图片~
1、绘制遮盖层
事实上遮盖层就是一个颜色:
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)
{
super.onMeasure(widthMeasureSpec, heightMeasureSpec); int width = getMeasuredWidth();
int height = getMeasuredHeight();
// 初始化bitmap
mBitmap = Bitmap.createBitmap(width, height, Config.ARGB_8888);
mCanvas = new Canvas(mBitmap);
setUpOutPaint();
//绘制这改成
mCanvas.drawColor(Color.parseColor("#c0c0c0"));
}
和上面贴的代码就多了最后一行,另外我们的paint的设置抽取出去了~
2、drawPath
文章起初的原理最终要用上了。我们在绘制Path的时候,须要设置一个模式,这里是DST_OUT ,想想有点小激动~
private void drawPath()
{ mOutterPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT));
mCanvas.drawPath(mPath, mOutterPaint);
}
3、onDraw
onDraw里面也要做一点改动
@Override
protected void onDraw(Canvas canvas)
{
canvas.drawBitmap(mBackBitmap, 0, 0, null);
drawPath();
canvas.drawBitmap(mBitmap, 0, 0, null);
}
好了。到此完毕。事实上就加入了几行代码,就完毕了我们简易画板到刮刮卡的转变;
以下看效果:
有没有拨开云雾见天明的感觉~~
到此我们的刮刮卡的原理,以及初步的实现结束了~~还是非常easy的~接下来就是兴许的完好工作
4、刮刮卡的完好
我们准备把奖项改为字体。将字体绘制在屏幕的中间;
那么直接把上例绘制图片改为绘制字体即可了,只是多了一个绘制字体画笔的设置:
有变化的代码:
private Paint mBackPint = new Paint();
private Rect mTextBound = new Rect();
private String mText = "500,0000,000";
/**
* 初始化canvas的绘制用的画笔
*/
private void setUpBackPaint()
{
mBackPint.setStyle(Style.FILL);
mBackPint.setTextScaleX(2f);
mBackPint.setColor(Color.DKGRAY);
mBackPint.setTextSize(22);
mBackPint.getTextBounds(mText, 0, mText.length(), mTextBound);
} @Override
protected void onDraw(Canvas canvas)
{
// canvas.drawBitmap(mBackBitmap, 0, 0, null);
//绘制奖项
canvas.drawText(mText, getWidth() / 2 - mTextBound.width() / 2,
getHeight() / 2 + mTextBound.height() / 2, mBackPint); drawPath();
canvas.drawBitmap(mBitmap, 0, 0, null);
}
以下看看我中了多钱:
好了,到此已经全然实现了。大家依照样例,结合自己需求改动即可,里面所涉及的原理相信已经解释清楚了;对了。差点忘了,刮刮卡一般都有一个功能,当你挂了几乎相同的时候。涂层会自己主动清除。以下我们尝试加入该功能。
5、统计刮开区域已占的百分比
我们在ACTION_UP的时候即可计算。首先我们还是给大家灌输下计算的原理。假设大家用心看了,应该知道我们全部的操作基本都在mBitmap,如今我们获得mBItmap上全部的像素点的数据。统计被清除的区域(被清除的像素为0);最后与我们图片的总像素数做个除法元算,就能够拿到我们清除的百分比了。
只是,计算可能会是一个耗时的操作,详细速度跟图片大小有关,所以我们决定使用异步的方式去计算:
/**
* 统计擦除区域任务
*/
private Runnable mRunnable = new Runnable()
{
private int[] mPixels; @Override
public void run()
{ int w = getWidth();
int h = getHeight(); float wipeArea = 0;
float totalArea = w * h; Bitmap bitmap = mBitmap; mPixels = new int[w * h]; /**
* 拿到全部的像素信息
*/
bitmap.getPixels(mPixels, 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 (mPixels[index] == 0)
{
wipeArea++;
}
}
} /**
* 依据所占百分比。进行一些操作
*/
if (wipeArea > 0 && totalArea > 0)
{
int percent = (int) (wipeArea * 100 / totalArea);
Log.e("TAG", percent + ""); if (percent > 70)
{
isComplete = true;
postInvalidate();
}
}
} };
有了这个任务,我们在ACTION_UP的时候即可调用:
case MotionEvent.ACTION_UP:
new Thread(mRunnable).start();
break;
注意任务结束,会把一个isComplete设置为true。当为true时。我们直接展现刮奖区
@Override
protected void onDraw(Canvas canvas)
{
drawBackText(canvas); if (!isComplete)
{
drawPath();
canvas.drawBitmap(mBitmap, 0, 0, null);
} }
到此刮奖区的计算就结束了。
以下在演示前。我做了一些简单的美化,详细大家到时候看源代码就能够。
到此我们的刮刮卡制作就结束了,另外假设大家希望再完好,能够把里面非常多常量设置成变量,加入对外的set方法。或者抽取成自己定义属性,在布局文件进行定义都能够~~~
有一点须要说明一下,对于我们刮刮卡这个案例。我们布局文件假设宽高设置为wrap_content。也会占满屏幕,主要是由于我认为刮刮卡这个view不是必需wrap_content;可是假设你希望支持wrp_content,能够參考Android 自己定义View (一) 。
---------------------------------------------------------------------------------------------------------
我建了一个QQ群,方便大家交流。
群号:55032675
----------------------------------------------------------------------------------------------------------
博主部分视频已经上线,假设你不喜欢枯燥的文本,请猛戳(初录。期待您的支持):
Android 自己定义控件实现刮刮卡效果 真的就仅仅是刮刮卡么的更多相关文章
- Android自己定义控件系列五:自己定义绚丽水波纹效果
尊重原创!转载请注明出处:http://blog.csdn.net/cyp331203/article/details/41114551 今天我们来利用Android自己定义控件实现一个比較有趣的效果 ...
- Android自己定义控件:进度条的四种实现方式
前三种实现方式代码出自: http://stormzhang.com/openandroid/2013/11/15/android-custom-loading/ (源代码下载)http://down ...
- android 自己定义控件
Android自己定义View实现非常easy 继承View,重写构造函数.onDraw.(onMeasure)等函数. 假设自己定义的View须要有自己定义的属性.须要在values下建立attrs ...
- Android自己定义控件皮肤
Android自己定义控件皮肤 对于Android的自带控件,其外观仅仅能说中规中矩,而我们平时所示Android应用中,一个简单的button都做得十分美观.甚至于很多button在按下时的外观都有 ...
- android 自己定义控件属性(TypedArray以及attrs解释)
近期在捣鼓android 自己定义控件属性,学到了TypedArray以及attrs.在这当中看了一篇大神博客Android 深入理解Android中的自己定义属性.我就更加深入学习力一番.我就沿着这 ...
- Android自己定义控件之应用程序首页轮播图
如今基本上大多数的Android应用程序的首页都有轮播图.就是像下图这种(此图为转载的一篇博文中的图.拿来直接用了): 像这种组件我相信大多数的应用程序都会使用到,本文就是自己定义一个这种组件,能够动 ...
- Android自己定义控件(状态提示图表)
[工匠若水 http://blog.csdn.net/yanbober 转载烦请注明出处.尊重分享成果] 1 背景 前面分析那么多系统源代码了.也该暂停下来歇息一下,趁昨晚闲着看见一个有意思的需求就操 ...
- Android自己定义控件系列二:自己定义开关button(一)
这一次我们将会实现一个完整纯粹的自己定义控件,而不是像之前的组合控件一样.拿系统的控件来实现.计划分为三部分:自己定义控件的基本部分,自己定义控件的触摸事件的处理和自己定义控件的自己定义属性: 以下就 ...
- Android 自己定义控件开发入门(二)
上一次我们讲了一堆实现自己定义控件的理论基础.列举了View类一些能够重写的方法,我们对这些方法的重写是我们继承View类来派生自己定义控件的关键 我通过一个最简单的样例给大家展示了这一个过程,不管是 ...
- Android自己定义控件
今天我们来讲一下 Android中自己定义控件的介绍,在Android中, 我们一般写xml都是用的是单个的控件来完毕的 ,但是.往往在一些项目中.单个控件有时是满足不了的.故此我们能够自己定义控件 ...
随机推荐
- FormsAuthentication权限管理
通常我们在做访问权限管理的时候会把用户正确登录后的基本信息保存在Session中然后用户每次请求页面或接口数据的时候代上会话状态即能拿到Session中存储的基本信息Session的原理,也就是在服务 ...
- CentOS 7.4 下搭建 Elasticsearch 6.3 搜索群集
上个月 13 号,Elasticsearch 6.3 如约而至,该版本和以往版本相比,新增了很多新功能,其中最令人瞩目的莫过于集成了 X-Pack 模块.而在最新的 X-Pack 中 Elastics ...
- html5前端杂记
首先是css的一些知识 毕竟自己懂得不多,但是一看资料.感觉似曾相识 <style> .red-text { color: red; } </style>//这里是css样式的 ...
- Angular——$http
基本介绍 $http用于向服务端发起异步请求,同时还支持多种快捷方式如$http.get().$http.post().$http.jsonp.$hhtp也是属于内置服务的一种,这里特意提出来写一篇用 ...
- 富士康的盈利秒杀99%的A股公司:3星|《三联生活周刊》2018年10期
三联生活周刊·最美的数学:天才为何成群到来(2018年10期) 本期专题是数学和成都,我都跳过去没看.其他内容也还有点意思. 总体评价3星. 以下是本期一些内容的摘抄,#号后面是kindle电子版中的 ...
- CAD得到布局名
js代码如下: var database = mxOcx.GetDatabase(); var sRet = null; //返回数据库中的布局字典 var spLayoutDictionary = ...
- 梦想CAD控件事件COM接口知识点
一.鼠标事件 _DMxDrawXEvents::MouseEvent 控件中的鼠标事件. 参数 说明 LONG lType 事件类型,1鼠标移动,2是鼠标左键按下,3是鼠标右键按下,4是鼠标左键双击 ...
- JavaScipt30(第十个案例)(主要知识点:选中一个数组中间相连部分进行操作的一种思路)
承接上文,第九个案例就不说了,是控制台的一些东西,一般用的很少,了解下就行了,想用的时候再翻api.这是第10个案例: 需要实现的效果是:点击一个checkbox,然后按下shift点击另一个chec ...
- freemarker使用map替换ftl中相关值
ftl文件demo01.ftl <html> <head> <title>Welcome!</title> </head> <body ...
- checkbox prop无效问题
因为bootstrap插件问题,需要先获取input的上级元素,然后添加checked $("input[name='checkInput']").parent().addClas ...