Xfermodes的扩展应用- 图像擦除和还原效果

本文介绍如何使用Android图形API实现图像擦除与还原效果。利用Xfermodes中的XOR模式,结合触摸事件绘制遮罩,达到擦除显示的效果,并提供源码下载。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

最近在学习Android图形API,看到sdkdemo里的Xfermodes实例,于是结合之前学的做了一个简单的图像擦除和还原的效果,感觉比较好玩,分享下作为TestRoid第一篇^^。
Sdk中自带的例子如图所示,
 
是几种不同的Xfermodes效果,这些效果很显而易见,无非是一个黄色圆(Dst)和蓝色矩形(Src)的显示。比如SrcOver就是蓝色矩形显示在黄色圆前面;
SrcIn就是两个图形交集显示为Src也就是蓝色矩形的那部分。我主要是用里面的Xor效果来实现这里的例子,PorterDuff.Mode.XOR,即两块非透明区域重叠部分显示为透明。

如果单是做图像擦除的效果也就用不到这个东西咯,所以嘛,重点是在还原的效果。因此这个例子的思路是,建立一块与原图大小相同的mask bitmap,然后在这块mask上进行操作,这样就不会影响原图,这里我用了触摸画线的例子在mask上画点东西,然后通过Xermodes中的XOR模式与原图进行合成,得到我们想要看到的效果。是不是有点Photoshop的味道啦,hoho
原理图在这,随手画的,呵呵
 
废话不多说直接上效果图(OK,这个是用模拟器截得,有点卡=.=):
 


下面是一些代码片段:


初始化mask

  1. view plaincopy to clipboardprint?

  2.     private void initBmpMask()  
  3.     {  
  4.         if(mImgDesRect != null)  
  5.         {  
  6.             int w = (int) mImgDesRect.width();  
  7.             int h = (int) mImgDesRect.height();  
  8.             if(mEraseMaskBitmap == null)  
  9.             {  
  10.                 mEraseMaskBitmap = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888); //初始化一块和显示图片大小相同的bitmap  
  11.                 mEraseMaskBitmap.eraseColor(Color.TRANSPARENT);//设置为透明  
  12.             }  
  13.         }  
  14.       
  15.         if(mEraseMaskBitmap != null)  
  16.             mCanvas = new Canvas(mEraseMaskBitmap);  //得到要操作的mask  
  17.     }  
复制代码



初始
  1. 化画笔样式等:

  2. view plaincopy to clipboardprint?

  3.     private void initErasePaint(){  
  4.         mErasePaint = new Paint();  
  5.         mErasePaint.setAntiAlias(true);  
  6.         mErasePaint.setDither(true);  
  7.         mErasePaint.setColor(0xFF000000);  
  8.         mErasePaint.setStyle(Paint.Style.STROKE);  
  9.         mErasePaint.setStrokeJoin(Paint.Join.ROUND);  
  10.         mErasePaint.setStrokeCap(Paint.Cap.ROUND);  
  11.         mErasePaint.setStrokeWidth(20);  
  12.     }  
复制代码


响应触屏消息在mask上画图:
  1. view plaincopy to clipboardprint?

  2.     public boolean onTouchEvent(MotionEvent event) {  
  3.         int action = event.getAction();  
  4.         float x = event.getX() - mImgDesRect.left;  
  5.         float y = event.getY() - mImgDesRect.top;  
  6.         switch (action) {  
  7.         case MotionEvent.ACTION_DOWN:  
  8.             mErasePath.reset();  
  9.             mErasePath.moveTo(x, y);  
  10.             mX = x;  
  11.             mY = y;  
  12.             invalidate();  
  13.             break;  
  14.         case MotionEvent.ACTION_MOVE:  
  15.              float dx = Math.abs(x - mX);  
  16.              float dy = Math.abs(y - mY);  
  17.              if (dx >= TOUCH_TOLERANCE || dy >= TOUCH_TOLERANCE) {  
  18.                  mErasePath.quadTo(mX, mY, (x + mX)/2, (y + mY)/2);  
  19.                  mX = x;  
  20.                  mY = y;  
  21.                  // commit the path to offscreen  
  22.                  mCanvas.drawPath(mErasePath, mErasePaint);  
  23.              }  
  24.             invalidate(); //refresh  
  25.             break;  
  26.         case MotionEvent.ACTION_UP:  
  27.             mErasePath.lineTo(mX, mY);  
  28.             mCanvas.drawPath(mErasePath, mErasePaint);  
  29.             mErasePath.reset();  
  30.             invalidate();  
  31.             break;  
  32.         }  
  33.         return true;  
  34.     }
复制代码
切换擦除(erase)和还原(unerase)功能,只要改变Paint Xfermode
  1. view plaincopy to clipboardprint?

  2.     public void switchEraseMode(boolean bErase){  
  3.         if(mErasePaint == null)  
  4.             initErasePaint();  
  5.           
  6.         if(bErase)  
  7.             mErasePaint.setXfermode(null);  
  8.         else  
  9.             mErasePaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR));//清除样式  
  10.     }  
复制代码


在OnDraw中,把原图和mask通过XOR样式画到canvas上,就可以看到擦除的效果啦:
  1. view plaincopy to clipboardprint?

  2.     protected void onDraw(Canvas canvas) {  
  3.         super.onDraw(canvas);  
  4.       
  5.         if(mImgBitmap != null)  
  6.         {  
  7.             Paint paint = new Paint();  
  8.             int sc = canvas.saveLayer(mImgDesRect.left, mImgDesRect.top, mImgDesRect.right, mImgDesRect.bottom, null,     
  9.                     Canvas.MATRIX_SAVE_FLAG | Canvas.CLIP_SAVE_FLAG     
  10.                             | Canvas.HAS_ALPHA_LAYER_SAVE_FLAG     
  11.                             | Canvas.FULL_COLOR_LAYER_SAVE_FLAG     
  12.                             | Canvas.CLIP_TO_LAYER_SAVE_FLAG);     
  13.             paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.XOR));  
  14.             canvas.drawBitmap(mImgBitmap, new Rect(0, 0, mImgBitmap.getWidth(), mImgBitmap.getHeight()), mImgDesRect, null);      
  15.             canvas.drawBitmap(mEraseMaskBitmap, new Rect(0, 0, mEraseMaskBitmap.getWidth(), mEraseMaskBitmap.getHeight()), mImgDesRect, paint);  
  16.             paint.setXfermode(null);  
  17.             canvas.restoreToCount(sc);    
  18.         }  
  19.     }  
复制代码



好了,显示上擦除和“反擦”效果就完成了。
例程上传了,可以这里下载: http://download.youkuaiyun.com/source/3240693
里面顺便加上了把擦除效果保存下来的,/sdcard/out.png
注:为了视觉效果,例子里加上了一张背景图,让擦除后透明效果明显点。不过保存时这背景图是不会和原图一起保存的。(太懒了。。。)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值