android 画板之橡皮擦功能开发

记录一下橡皮擦功能开发。

讲一下原理:

橡皮擦功能要用到Paint类的一个属性:

paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_IN));

这句代码的意思是:

只在源图像和目标图像相交的地方绘制目标图像

不懂没关系,首先用一个paint来绘制线条,然后用另一个paint作为橡皮擦并设置上句代码的属性,然后就变成了橡皮擦。

哎哟妈呀,我到底在说些什么鬼。。。。

还是看代码吧。

看代码就秒懂的......

package com.example.testwhatever;

import android.app.Activity;
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.os.Bundle;
import android.util.DisplayMetrics;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup.LayoutParams;
import android.widget.Button;
import android.widget.LinearLayout;

public class CanvasActivity extends Activity {
	private int SCREEN_W;
	private int SCREEN_H;
	private int Pen = 1;
	private int Eraser = 2;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		LinearLayout layout = new LinearLayout(this);
		layout.setOrientation(LinearLayout.VERTICAL);
		LayoutParams params = new LayoutParams(LayoutParams.WRAP_CONTENT,
				LayoutParams.WRAP_CONTENT);
		Button paint = new Button(this);
		paint.setText("画笔");
		layout.addView(paint, params);
		Button eraser = new Button(this);
		eraser.setText("橡皮");
		layout.addView(eraser, params);
		final MyView myView = new MyView(this);
		layout.addView(myView);
		setContentView(layout);

		paint.setOnClickListener(new OnClickListener() {
			@Override
			public void onClick(View v) {
				myView.setMode(Pen);
			}
		});

		eraser.setOnClickListener(new OnClickListener() {
			@Override
			public void onClick(View v) {
				myView.setMode(Eraser);
			}
		});
	}
      //MyView就是自定义的画板
	class MyView extends View {
		private int mMode = 1;
		private Bitmap mBitmap;
		private Canvas mCanvas;
		private Paint mEraserPaint;
		private Paint mPaint;
		private Path mPath;
		private float mX, mY;
		private static final float TOUCH_TOLERANCE = 4;

		public MyView(Context context) {
			super(context);
			setFocusable(true);
			setScreenWH();
			initPaint();
		}

		private void setScreenWH() {
			DisplayMetrics dm = new DisplayMetrics();
			dm = this.getResources().getDisplayMetrics();
			int screenWidth = dm.widthPixels;
			int screenHeight = dm.heightPixels;
			SCREEN_W = screenWidth;
			SCREEN_H = screenHeight;
		}

		//设置绘制模式是“画笔”还是“橡皮擦”
		public void setMode(int mode){
			this.mMode = mode;
		}

		private void initPaint() {
			//画笔
			mPaint = new Paint();
			mPaint.setAntiAlias(true);
			mPaint.setStyle(Paint.Style.STROKE);
			mPaint.setStrokeCap(Paint.Cap.ROUND);
			mPaint.setStrokeJoin(Paint.Join.ROUND);
			mPaint.setColor(Color.BLACK);
			mPaint.setStrokeWidth(10);
			//橡皮擦
			mEraserPaint = new Paint();
			mEraserPaint.setAlpha(0);
			//这个属性是设置paint为橡皮擦重中之重
			//这是重点
			//下面这句代码是橡皮擦设置的重点
			mEraserPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_IN));
			//上面这句代码是橡皮擦设置的重点(重要的事是不是一定要说三遍)
			mEraserPaint.setAntiAlias(true);
			mEraserPaint.setDither(true);
			mEraserPaint.setStyle(Paint.Style.STROKE);
			mEraserPaint.setStrokeJoin(Paint.Join.ROUND);
			mEraserPaint.setStrokeWidth(30);

			mPath = new Path();

			mBitmap = Bitmap.createBitmap(SCREEN_W, SCREEN_H, Config.ARGB_8888);
			mCanvas = new Canvas(mBitmap);
		}

		@Override
		protected void onDraw(Canvas canvas) {
			if (mBitmap != null) {
				canvas.drawBitmap(mBitmap, 0, 0, mPaint);
			}
			super.onDraw(canvas);
		}

		private void touch_start(float x, float y) {
			mPath.reset();
			mPath.moveTo(x, y);
			mX = x;
			mY = y;
			//如果是“画笔”模式就用mPaint画笔进行绘制
			if (mMode == Pen) {
				mCanvas.drawPath(mPath, mPaint);
			}
			//如果是“橡皮擦”模式就用mEraserPaint画笔进行绘制
			if (mMode == Eraser) {
				mCanvas.drawPath(mPath, mEraserPaint);
			}

		}

		private void touch_move(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;
				if (mMode == Pen) {
					mCanvas.drawPath(mPath, mPaint);
				}
				if (mMode == Eraser) {
					mCanvas.drawPath(mPath, mEraserPaint);
				}
			}
		}

		private void touch_up() {
			mPath.lineTo(mX, mY);
			if (mMode == Pen) {
				mCanvas.drawPath(mPath, mPaint);
			}
			if (mMode == Eraser) {
				mCanvas.drawPath(mPath, mEraserPaint);
			}
		}

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

			switch (event.getAction()) {
			case MotionEvent.ACTION_DOWN:
				touch_start(x, y);
				invalidate();
				break;
			case MotionEvent.ACTION_MOVE:
				touch_move(x, y);
				invalidate();
				break;
			case MotionEvent.ACTION_UP:
				touch_up();
				invalidate();
				break;
			}
			return true;
		}
	}
}

  然后效果图是酱紫的:

其实就是,就是 ,就是那个画笔的paint和橡皮擦的paint两个图像相交,然后就会绘制相交的区域,相交的区域并不是绘制成白色,而是透明的。这都是因为橡皮擦的paint设置了那个很重要的属性:PorterDuff.Mode.DST_IN

为了证明相交区域真的绘制的是透明的,而不是白色,给MyView设置一张背景图,如果绘制的是白色,那么背景图也会被绘制成白色。如果是透明,那么背景图不会受影响。

在上述代码中的MyView的构造函数中加一句代码:setBackgroundResource(R.drawable.picture);

然后运行结果图是酱紫的:

所以这就是开发橡皮擦功能时为什么推荐用设置paint这个PorterDuff.Mode.DST_IN属性,而不是单纯的绘制白色了。

另外关于PorterDuff.Mode还有很多模式:

android.graphics.PorterDuff.Mode.SRC:只绘制源图像

android.graphics.PorterDuff.Mode.DST:只绘制目标图像

android.graphics.PorterDuff.Mode.DST_OVER:在源图像的顶部绘制目标图像

android.graphics.PorterDuff.Mode.DST_IN:只在源图像和目标图像相交的地方绘制目标图像

android.graphics.PorterDuff.Mode.DST_OUT:只在源图像和目标图像不相交的地方绘制目标图像

android.graphics.PorterDuff.Mode.DST_ATOP:在源图像和目标图像相交的地方绘制目标图像,在不相交的地方绘制源图像

android.graphics.PorterDuff.Mode.SRC_OVER:在目标图像的顶部绘制源图像

android.graphics.PorterDuff.Mode.SRC_IN:只在源图像和目标图像相交的地方绘制源图像

android.graphics.PorterDuff.Mode.SRC_OUT:只在源图像和目标图像不相交的地方绘制源图像

android.graphics.PorterDuff.Mode.SRC_ATOP:在源图像和目标图像相交的地方绘制源图像,在不相交的地方绘制目标图像

android.graphics.PorterDuff.Mode.XOR:在源图像和目标图像重叠之外的任何地方绘制他们,而在不重叠的地方不绘制任何内容

android.graphics.PorterDuff.Mode.LIGHTEN:获得每个位置上两幅图像中最亮的像素并显示

android.graphics.PorterDuff.Mode.DARKEN:获得每个位置上两幅图像中最暗的像素并显示

android.graphics.PorterDuff.Mode.MULTIPLY:将每个位置的两个像素相乘,除以255,然后使用该值创建一个新的像素进行显示。结果颜色=顶部颜色*底部颜色/255

android.graphics.PorterDuff.Mode.SCREEN:反转每个颜色,执行相同的操作(将他们相乘并除以255),然后再次反转。结果颜色=255-(((255-顶部颜色)*(255-底部颜色))/255)

可以每个模式都设置玩玩嘛,看看每个属性是什么效果~~~反正我是不会去玩这么无聊的东西的。我懒啊~~~O(∩_∩)O~~

时间: 2025-01-15 20:28:24

android 画板之橡皮擦功能开发的相关文章

和我一起开发Android应用(一)——开发一款安卓多功能背单词软件

大家好,很久没有在博客园写过东西.前一段时间尝试学过一段时间cocos2d,但是由于学习资料有限,文档较少,一直难有进展,因此在刚刚过去的三周里,我又回到了我的老本行:java ,Android.在假期里我算是正式开发了我的第一个安卓应用: i背单词.经过三个星期的磕磕绊绊,终于写出一个较稳定的版本,现在这款应用已经成功上架GooglePlay,并且很快会在百度移动应用和安智网上架.当然由于这款软件只是由我个人维护,我也不奢求会有多大的下载量了呵呵,但我想有必要把这短时间开发应用的体会做一个总结

Android KitKat 4.4平台开发-添加USB ADB和MTP功能支持

ADB和MTP是Android基于USB实现的两个重要功能,极大地方便了用户在PC与Android设备之间的互操作,比如传输文件.安装应用.开发调试应用. 本文讲述如何在特定软硬件平台下支持Android ADB和MTP功能. Android版本: KitKat 4.4.2 Linux内核版本: 3.10 (Vendor Kernel) 硬件平台: Atmel SAMA5D3 SoC 针对Linux内核的更改 Merge Android Linux内核USB Gadget驱动到处理器厂商Linu

Android学习之高德地图的通用功能开发步骤(二)

周一又来了,我就接着上次的开发步骤(一)来吧,继续把高德地图的相关简单功能分享一下 上次写到了第六步,接着写第七步吧. 第七步:定位 + 地图选点 + 路径规划 + 实时导航 以下是我的这个功能NaviMapActivity的页面布局文件: 1 <?xml version="1.0" encoding="utf-8"?> 2 <RelativeLayout xmlns:android="http://schemas.android.com

Android画板的实现及demo

今天给大家带来Android画板功能的简单实现,以下是效果图: 以下是关键源码: import android.content.Context; import android.graphics.Canvas; import android.graphics.Color; import android.graphics.Paint; import android.graphics.Paint.Style; import android.graphics.PaintFlagsDrawFilter;

Android手机拨打电话的开发实例

一部手机最常用的功能就是打电话和发短信了,在Android开发中我们如何通过程序拨打电话呢?本文就给出一个用Android手机拨打电话的简单的实例. 下面是开发此实例的具体步骤: 一.新建一个Android工程,命名为phoneCallDemo. 二.设计程序的界面,打开main.xml把内容修改如下: XML/HTML代码 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:and

Android IOS WebRTC 音视频开发总结(三五)-- chatroulette介绍

本文主要从技术角度介绍chatroulette,文章来自博客园RTC.Blacker,支持原创,转载请说明出处. 很多人不知道或没用过chatroulette,下面先来张界面截图让大家有个整体了解: 特点介绍: 1.平均每天在线用户150万(对于一个没做过广告的网站来说这是一个很了不起的数字). 2.作者是一个17岁俄国高中生,11岁就开始学习编程. 3.为什么那么吸引人——因为在一个空白屏幕后面,有一批陌生人正在等待着与你聊天,人品好的话能见到各种肤色的美女. 现有技术: 1.技术原理其实很简

Android Studio入门(安装--&gt;开发调试)

写在前面的话:本文来源:http://blog.csdn.net/yanbober/article/details/45306483 目标:Android Studio新手–>下载安装配置–>零基础入门–>基本使用–>调试技能–>构建项目基础–>使用AS应对常规应用开发 下载安装 下载AS前先说下,AS安装包分为含SDK版本和不含SDK版本下载,如果你有SDK,那么完全可以下载不含SDK版本:不过下载了含SDK版本也没事,安装时选择自定义SDK也可以,安装后重新指定SD

第一章 Android系统移植与驱动开发概述

本书第一章首先简单概要地介绍了关于Android系统移植和驱动开发的相关内容. 所谓“移植”是指为特定的自己的设备,如手机定制Android的过程.自己开发一些程序(移植)装载在设备上,使得Android系统可以识别设备中的蓝牙,WiFi等,这样就可以使用这些功能了.这些程序大多是支持各种硬件设备的Linux驱动程序. 这一章中介绍了许多在进行Android系统移植和驱动开发前要了解的概念以及要做的准备.接下来主要说的是在Android系统移植过程中的主要工作. Android移植分为两部分:应

原生Android也能做Web开发了

原生Android也能做Web开发了 版权声明:转载必须注明本文转自严振杰的博客:http://blog.yanzhenjie.com 大家好,今天跟大家介绍一个让原生Android也可以做Web开发的开源项目--AndServer. 开源地址:https://github.com/yanzhenjie/AndServer AndServer是一个Android端的Web服务器,类似Apache或者Tomcat,但又有不同,它是一个普通的Android Library,Android项目Grad