<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=".MainActivity" > <!-- <com.example.rubbler.Text_Rubbler android:id="@+id/rubbler" android:layout_width="200dp" android:layout_height="200dp" android:layout_centerHorizontal="true" android:layout_centerVertical="true" android:gravity="center" android:text="@string/hello_world" /> --> <com.example.rubbler.Image_Rubbler android:id="@+id/rubbler" android:background="@drawable/girl1" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_centerHorizontal="true" android:layout_centerVertical="true" android:gravity="center" /> </RelativeLayout>
package com.example.rubbler; import com.example.rubbler.R; import android.os.Bundle; import android.app.Activity; import android.view.Menu; public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); //Text_Rubbler rubbler = (Text_Rubbler)this.findViewById(R.id.rubbler); Image_Rubbler rubbler = (Image_Rubbler) this.findViewById(R.id.rubbler); rubbler.beginRubbler(0xffff0000, 40, 50); } @Override public boolean onCreateOptionsMenu(Menu menu) { // Inflate the menu; this adds items to the action bar if it is present. getMenuInflater().inflate(R.menu.main, menu); return true; } }
package com.example.rubbler; 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.PorterDuff; import android.graphics.PorterDuffXfermode; import android.util.AttributeSet; import android.view.MotionEvent; import android.view.animation.AlphaAnimation; import android.view.animation.Animation; import android.view.animation.LinearInterpolator; import android.widget.ImageView; public class Image_Rubbler extends ImageView { private float TOUCH_TOLERANCE; //填充距离,使线条更自然,柔和,值越小,越柔和。 // private final int bgColor; private Bitmap mBitmap; private Canvas mCanvas; private Paint mPaint; private Path mPath; private float mX, mY; private boolean isDraw = false; public Image_Rubbler(Context context) { super(context); } public Image_Rubbler(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); // bgColor = attrs.getAttributeIntValue("http://schemas.android.com/apk/res/android", "textColor", 0xFFFFFF); // System.out.println("Color:"+bgColor); } public Image_Rubbler(Context context, AttributeSet attrs) { super(context, attrs); // bgColor = attrs.getAttributeIntValue("http://schemas.android.com/apk/res/android", "textColor", 0xFFFFFF); // System.out.println(bgColor); // System.out.println(attrs.getAttributeValue("http://schemas.android.com/apk/res/android", "layout_width")); } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); if (isDraw) { mCanvas.drawPath(mPath, mPaint); canvas.drawBitmap(mBitmap, 0, 0, null); } } /** * 开启檫除功能 * @param bgColor 覆盖的背景颜色 * @param paintStrokeWidth 触点(橡皮)宽度 * @param touchTolerance 填充距离,值越小,越柔和。 */ public void beginRubbler(final int bgColor , final int paintStrokeWidth , float touchTolerance) { TOUCH_TOLERANCE = touchTolerance; //设置画笔 mPaint = new Paint(); //mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT)); //或者 mPaint.setAlpha(0); mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_IN)); mPaint.setAntiAlias(true); mPaint.setDither(true); mPaint.setStyle(Paint.Style.STROKE); mPaint.setStrokeJoin(Paint.Join.ROUND); //前圆角 mPaint.setStrokeCap(Paint.Cap.ROUND); //后圆角 mPaint.setStrokeWidth(paintStrokeWidth); //笔宽 //痕迹 mPath = new Path(); //覆盖 // if (getLayoutParams().width == LayoutParams.FILL_PARENT) { // // } int width = this.getResources().getDrawable(R.drawable.girl1).getIntrinsicWidth(); int height = this.getResources().getDrawable(R.drawable.girl1).getIntrinsicHeight(); mBitmap = Bitmap.createBitmap(width, height, Config.ARGB_8888); mCanvas = new Canvas(mBitmap); mCanvas.drawColor(bgColor); isDraw = true; } @Override public boolean onTouchEvent(MotionEvent event) { if (!isDraw) { return true; } switch (event.getAction()) { case MotionEvent.ACTION_DOWN: //触点按下 // touchDown(event.getRawX(),event.getRawY()); touchDown(event.getX(), event.getY()); invalidate(); break; case MotionEvent.ACTION_MOVE: //触点移动 touchMove(event.getX(), event.getY()); invalidate(); break; case MotionEvent.ACTION_UP: //触点弹起 touchUp(event.getX(), event.getY()); invalidate(); break; default: break; } return true; } private void touchDown(float x , float y) { mPath.reset(); mPath.moveTo(x, y); mX = x; mY = y; } private void touchMove(float x , float y) { float dx = Math.abs(x - mX); float dy = Math.abs(y - mY); if (dx >= TOUCH_TOLERANCE || dy >= TOUCH_TOLERANCE) { mPath.quadTo(mX, mY, (x + mX) / 2, (y + mY) / 2); mX = x; mY = y; } } private void touchUp(float x , float y) { mPath.lineTo(x, y); mCanvas.drawPath(mPath, mPaint); mPath.reset(); int w = mBitmap.getWidth(); int h = mBitmap.getHeight(); int[] pixels = new int[w * h]; try { /* 参数 pixels 接收位图颜色值的数组 offset 写入到pixels[]中的第一个像素索引值 stride pixels[]中的行间距个数值(必须大于等于位图宽度)。可以为负数 x 从位图中读取的第一个像素的x坐标值。 y 从位图中读取的第一个像素的y坐标值 width 从每一行中读取的像素宽度 height 读取的行数 */ mBitmap.getPixels(pixels, 0, w, 0, 0, w, h); int rubblerCount = 0; int bitmapCount = pixels.length; for (int i = 0; i < bitmapCount; i++) { if (pixels[i] != 0xffff0000) { rubblerCount++; } } if (rubblerCount > bitmapCount * 0.3) { setFlickerAnimation(this); Paint paint = new Paint(); paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR)); mCanvas.drawPaint(paint); isDraw = false; } } catch (Exception e) { e.printStackTrace(); } } //闪烁 private void setFlickerAnimation(ImageView iv_chat_head) { final Animation animation = new AlphaAnimation(1, 0); // Change alpha from fully visible to invisible animation.setDuration(50); // duration - half a second animation.setInterpolator(new LinearInterpolator()); // do not alter animation rate // animation.setRepeatCount(Animation.INFINITE); // Repeat animation infinitely // animation.setRepeatCount(2); // Repeat animation infinitely animation.setRepeatMode(Animation.REVERSE); // iv_chat_head.setAnimation(animation); } }
package com.example.rubbler; 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.PorterDuff; import android.graphics.PorterDuffXfermode; import android.util.AttributeSet; import android.view.MotionEvent; import android.widget.TextView; public class Text_Rubbler extends TextView { private float TOUCH_TOLERANCE; //填充距离,使线条更自然,柔和,值越小,越柔和。 // private final int bgColor; private Bitmap mBitmap; private Canvas mCanvas; private Paint mPaint; private Path mPath; private float mX, mY; private boolean isDraw = false; public Text_Rubbler(Context context) { super(context); } public Text_Rubbler(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); // bgColor = attrs.getAttributeIntValue("http://schemas.android.com/apk/res/android", "textColor", 0xFFFFFF); // System.out.println("Color:"+bgColor); } public Text_Rubbler(Context context, AttributeSet attrs) { super(context, attrs); // bgColor = attrs.getAttributeIntValue("http://schemas.android.com/apk/res/android", "textColor", 0xFFFFFF); // System.out.println(bgColor); // System.out.println(attrs.getAttributeValue("http://schemas.android.com/apk/res/android", "layout_width")); } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); if (isDraw) { mCanvas.drawPath(mPath, mPaint); // mCanvas.drawPoint(mX, mY, mPaint); canvas.drawBitmap(mBitmap, 0, 0, null); } } /** * 开启檫除功能 * @param bgColor 覆盖的背景颜色 * @param paintStrokeWidth 触点(橡皮)宽度 * @param touchTolerance 填充距离,值越小,越柔和。 */ public void beginRubbler(final int bgColor , final int paintStrokeWidth , float touchTolerance) { TOUCH_TOLERANCE = touchTolerance; //设置画笔 mPaint = new Paint(); // mPaint.setAlpha(0); //画笔划过的痕迹就变成透明色了 mPaint.setColor(Color.BLACK); //此处不能为透明色 mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_OUT)); //或者 // mPaint.setAlpha(0); // mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_IN)); mPaint.setAntiAlias(true); mPaint.setDither(true); mPaint.setStyle(Paint.Style.STROKE); mPaint.setStrokeJoin(Paint.Join.ROUND); //前圆角 mPaint.setStrokeCap(Paint.Cap.ROUND); //后圆角 mPaint.setStrokeWidth(paintStrokeWidth); //笔宽 //痕迹 mPath = new Path(); ; //覆盖 // if (getLayoutParams().width == LayoutParams.FILL_PARENT) { // // } mBitmap = Bitmap.createBitmap(getLayoutParams().width, getLayoutParams().height, Config.ARGB_8888); mCanvas = new Canvas(mBitmap); mCanvas.drawColor(bgColor); isDraw = true; } @Override public boolean onTouchEvent(MotionEvent event) { if (!isDraw) { return true; } switch (event.getAction()) { case MotionEvent.ACTION_DOWN: //触点按下 // touchDown(event.getRawX(),event.getRawY()); touchDown(event.getX(), event.getY()); invalidate(); break; case MotionEvent.ACTION_MOVE: //触点移动 touchMove(event.getX(), event.getY()); invalidate(); break; case MotionEvent.ACTION_UP: //触点弹起 touchUp(event.getX(), event.getY()); invalidate(); break; default: break; } return true; } private void touchDown(float x , float y) { mPath.reset(); mPath.moveTo(x, y); mX = x; mY = y; } private void touchMove(float x , float y) { float dx = Math.abs(x - mX); float dy = Math.abs(y - mY); if (dx >= TOUCH_TOLERANCE || dy >= TOUCH_TOLERANCE) { mPath.quadTo(mX, mY, (x + mX) / 2, (y + mY) / 2); mX = x; mY = y; } } private void touchUp(float x , float y) { mPath.lineTo(x, y); mCanvas.drawPath(mPath, mPaint); mPath.reset(); } }
相关推荐
【Android 刮刮乐源码】是一个非常有趣的Android应用程序开发项目,它允许用户在屏幕上模拟刮刮乐彩票的游戏体验。这个小程序展示了Android平台上的图形处理和用户交互设计能力,是学习和理解Android游戏开发的一个...
在Android开发中,"刮刮乐"效果是一种常见的交互设计,通常用于抽奖或者隐藏信息的展示,用户可以通过滑动来揭示下面的内容。本资源提供的"Android 刮刮乐 图片(内附GIF效果)"应该包含一个实现刮刮乐功能的Android库...
这个"Android刮刮乐Demo下载"提供了实现此类功能的源代码示例。通过学习和理解这个Demo,开发者可以了解如何在Android应用中集成刮刮乐效果,为用户提供互动性强、娱乐性强的体验。 刮刮乐通常涉及到以下几个关键...
在Android开发中,"Android刮刮乐"是一个常见的交互效果,通常用于增加用户参与度和趣味性,类似于现实生活中刮彩票的体验。这种效果可以应用于各种场景,如促销活动、抽奖游戏或者隐藏信息的揭示等。下面将详细介绍...
Android 刮刮乐效果,开发环境:Android studio 3.1.3 博客地址:https://blog.csdn.net/huangliniqng/article/details/80722972
在Android平台上开发一款刮刮乐...综上所述,创建一个Android刮刮乐应用需要掌握Android图形绘制、触摸事件处理、动画制作以及性能优化等多个方面知识。通过不断实践和调试,可以打造出一款既有趣又实用的刮刮乐应用。
这个"android刮刮乐"项目旨在创建一个用户界面,让用户能够通过触摸屏幕来揭示隐藏的图像,就如同真实的刮刮乐彩票一样。下面将详细介绍实现这样一个功能所需的关键知识点。 首先,我们需要了解Android应用的基本...
【Android 刮刮乐 Demo】是一个基于Android平台的模拟淘宝抽奖功能的应用程序源代码。这个Demo旨在实现一种用户交互体验,让用户在手机屏幕上通过类似刮奖的动作来揭示隐藏的信息,以此增加互动性和趣味性。以下将...
这个"Android 刮刮乐开奖"项目,作为一个毕业设计示例,旨在帮助学生深入理解和实践Android应用程序的开发过程。以下是该项目涉及的一些关键知识点: 1. **Android Studio**:作为Android开发的主要集成开发环境...
下面我们将深入探讨Android刮刮乐应用的相关知识点。 1. **用户界面设计**: - Android刮刮乐应用的核心是其用户界面(UI),它需要设计一个具有刮擦效果的区域。这个区域通常由两个图层组成:底层是隐藏的内容,...
【Android 刮刮乐源码】是一个专门为 Android 平台设计的应用程序开发项目,它旨在模仿现实生活中常见的物理刮奖效果。在这款应用中,用户可以通过触摸屏幕来模拟刮开覆盖层,揭晓隐藏的内容,就像真实的刮刮乐彩票...
通过以上步骤,一个基本的Android刮刮乐效果就可以实现了。当然,还可以根据需求进一步定制,比如添加声音效果、振动反馈、限制刮擦速度等,以提升整体的用户体验。在实际项目中,可以根据项目的具体需求和设计来...
在Android应用开发中,签名和刮刮乐功能是两个独立但相关的主题。首先,我们来详细探讨一下Android签名的功能和重要性,然后我们会转向刮刮乐功能的实现。 ### Android签名 Android签名是确保应用程序安全性和完整...
在Android平台上,开发一款“刮刮乐”应用是一项有趣且具有挑战性的任务。这个应用的核心功能是模拟现实世界中的刮刮乐彩票,让用户通过手指滑动屏幕来揭示隐藏的区域,以达到抽奖或揭秘信息的目的。下面将详细介绍...
【安卓刮刮乐相关——Android刮刮乐效果源码解析】 在移动应用开发中,有时候我们需要为用户设计一些趣味性的互动元素,刮刮乐效果就是其中之一。这种效果常见于彩票应用、游戏或某些营销活动中,用户通过“刮开”...
只是简单的实现了效果,界面没怎么做优化,不过...import android.app.Activity; import android.os.Bundle; import android.view.Menu; import android.view.View; import android.view.View.OnClickListener; import
通过以上技术,我们可以创建一个功能完善的Android刮刮乐应用。开发者可以根据需求调整细节,如刮擦声音、颜色变化、刮擦形状等,以满足不同场景的需求。这个Demo不仅可以作为学习Android图形绘制和自定义View的实践...