Android Xfermode 学习笔记
一、概述
Xfermode全名transfer-mode,其作用是实现两张图叠加时的混合效果。
网上流传的关于Xfermode最出名的图来源于AndroidSDK的samples中,名叫Xfermodes.java,效果如下:

二、体验
提炼出Xfermodes.java中的核心代码,自己写了个简单粗暴的demo试试水:
public class ImageViewXfermode extends ImageView {
public ImageViewXfermode(Context context) {
super(context);
init();
}
public ImageViewXfermode(Context context, AttributeSet attrs) {
super(context, attrs);
init();
}
public ImageViewXfermode(Context context, AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
init();
}
private void init() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
setLayerType(View.LAYER_TYPE_SOFTWARE, null);
}
}
@Override
protected void onDraw(Canvas canvas) {
int defaultWidth = dip2px(85); //xml里view的宽度是85dp
int defaultdHeight = dip2px(85); //xml里view的高度是85dp
if (canvas.getHeight() == defaultWidth && canvas.getHeight() == defaultdHeight) {
//拿到黄色圆形的bitmap
Bitmap bitcircle = Bitmap.createBitmap(getWidth(), getHeight(), Bitmap.Config.ARGB_8888);
Canvas canvascicle = new Canvas(bitcircle);
Paint paintcicle = new Paint(Paint.ANTI_ALIAS_FLAG);
paintcicle.setColor(0xFFFFCC44);
canvascicle.drawCircle(dip2px(30), dip2px(30), dip2px(25), paintcicle);
//拿到蓝色矩形的bitmap
Bitmap bitrect = Bitmap.createBitmap(defaultWidth, defaultdHeight, Config.ARGB_8888);
Canvas canvasrect = new Canvas(bitrect);
Paint paintrect = new Paint(Paint.ANTI_ALIAS_FLAG);
paintrect.setColor(0xFF66AAFF);
canvasrect.drawRect(dip2px(30), dip2px(30), dip2px(80), dip2px(80), paintrect);
Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
Xfermode xfermode = new PorterDuffXfermode(Mode.LIGHTEN);
//采用saveLayer,让后续canvas的绘制在自动创建的bitmap上
int cnt = canvas.saveLayer(0, 0, defaultWidth, defaultdHeight, null, Canvas.ALL_SAVE_FLAG);
//先画圆形,圆形是dest
canvas.drawBitmap(bitcircle, 0, 0, paint);
paint.setXfermode(xfermode);
//后画矩形,矩形是src
canvas.drawBitmap(bitrect, 0, 0, paint);
paint.setXfermode(null);
canvas.restoreToCount(cnt);
} else {
super.onDraw(canvas);
}
}
private int dip2px(float dip) {
float scale = getResources().getDisplayMetrics().density;
return (int)(dip * scale + 0.5f);
}
}
效果如下:

介绍一下几个关键点:
1、关于src和dest
先绘制到canvas上的是dest,后绘制的是src
2、关于硬件加速
在sdkversion>=11时,需要关闭硬件加速(第19行),否则 Mode.CLEAR 、 Mode.DARKEN 、 Mode.LIGHTEN 三种模式下绘制效果不正常
3、saveLayer的作用
Canvas.saveLayer在Canvas.save的基础上,额外自动分配了一个bitmap,使得saveLayer之后的所有绘制都在这个新分配的bitmap上完成。
如果把saveLayer去掉呢?效果就是蓝色矩形跟黄色圆形和灰色背景都进行了 Mode.LIGHTEN 操作:

4、如果不saveLayer
有一种变通的方法,在dest对应bitmap的canvas上绘制src对应的bitmap,这样的目的与saveLayer是一致的,不在当前canvas上直接绘图:
@Override
protected void onDraw(Canvas canvas) {
int defaultWidth = dip2px(85); //xml里view的宽度是85dp
int defaultdHeight = dip2px(85); //xml里view的高度是85dp if (canvas.getHeight() == defaultWidth && canvas.getHeight() == defaultdHeight) {
//拿到黄色圆形的bitmap
Bitmap bitcircle = Bitmap.createBitmap(getWidth(), getHeight(), Bitmap.Config.ARGB_8888);
Canvas canvascicle = new Canvas(bitcircle);
Paint paintcicle = new Paint(Paint.ANTI_ALIAS_FLAG);
paintcicle.setColor(0xFFFFCC44);
canvascicle.drawCircle(dip2px(30), dip2px(30), dip2px(25), paintcicle); //拿到蓝色矩形的bitmap
Bitmap bitrect = Bitmap.createBitmap(defaultWidth, defaultdHeight, Config.ARGB_8888);
Canvas canvasrect = new Canvas(bitrect);
Paint paintrect = new Paint(Paint.ANTI_ALIAS_FLAG);
paintrect.setColor(0xFF66AAFF);
canvasrect.drawRect(dip2px(30), dip2px(30), dip2px(80), dip2px(80), paintrect); Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
Xfermode xfermode = new PorterDuffXfermode(Mode.LIGHTEN); //设置setXfermode
paint.setXfermode(xfermode);
//在圆形的canvas上画矩形,先画圆形,圆形是dest,后画矩形,矩形是src
canvascicle.drawBitmap(bitrect, 0, 0, paint);
paint.setXfermode(null);
//将圆形canvas上画出来的结果绘制到canvas上
canvas.drawBitmap(bitcircle, 0.0f, 0.0f, paint);
} else {
super.onDraw(canvas);
}
}
[转载请保留本文地址:http://www.cnblogs.com/goagent/p/5326438.html]
三、更多玩法——圆形ImageView
利用Xfermode的特性也可以做出圆形的ImageView来。先画原图,再画圆形,采用 Mode.DST_IN 即可:
@Override
protected void onDraw(Canvas canvas) {
int defaultWidth = dip2px(85); //xml里view的宽度是85dp
int defaultdHeight = dip2px(85); //xml里view的高度是85dp
Drawable drawable = getDrawable(); if (canvas.getHeight() == defaultWidth && canvas.getHeight() == defaultdHeight && drawable instanceof BitmapDrawable) {
//setBackgroundColor(Color.TRANSPARENT);
//拿到原图的bitmap
Bitmap bitimg = Bitmap.createBitmap(getWidth(), getHeight(), Bitmap.Config.ARGB_8888);
Canvas canvasimg = new Canvas(bitimg);
Matrix matrix = new Matrix();
matrix.setScale(defaultWidth * 1.0f / drawable.getIntrinsicWidth(), defaultdHeight * 1.0f / drawable.getIntrinsicHeight());
Paint paintimg = new Paint(Paint.ANTI_ALIAS_FLAG);
canvasimg.drawBitmap(((BitmapDrawable)drawable).getBitmap(), matrix, paintimg); //拿到圆形的bitmap
Bitmap bitcircle = Bitmap.createBitmap(defaultWidth, defaultdHeight, Config.ARGB_8888);
Canvas canvascircle = new Canvas(bitcircle);
Paint paintcircle = new Paint(Paint.ANTI_ALIAS_FLAG);
paintcircle.setColor(0xFF66AAFF);
canvascircle.drawCircle(dip2px(85 / 2.0f), dip2px(85 / 2.0f), dip2px(85 / 2.0f), paintcircle); Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG);
Xfermode xfermode = new PorterDuffXfermode(Mode.DST_IN); //采用saveLayer,让后续canvas的绘制在自动创建的bitmap上
int cnt = canvas.saveLayer(0, 0, defaultWidth, defaultdHeight, null, Canvas.ALL_SAVE_FLAG);
//先画原图,原图是dest
canvas.drawBitmap(bitimg, 0, 0, paint);
paint.setXfermode(xfermode);
//后画圆形,圆形是src
canvas.drawBitmap(bitcircle, 0, 0, paint);
paint.setXfermode(null);
canvas.restoreToCount(cnt);
} else {
super.onDraw(canvas);
}
}

[转载请保留本文地址:http://www.cnblogs.com/goagent/p/5326438.html]
Android Xfermode 学习笔记的更多相关文章
- Android自动化学习笔记:编写MonkeyRunner脚本的几种方式
---------------------------------------------------------------------------------------------------- ...
- Android自动化学习笔记之MonkeyRunner:官方介绍和简单实例
---------------------------------------------------------------------------------------------------- ...
- android开发学习笔记000
使用书籍:<疯狂android讲义>——李刚著,2011年7月出版 虽然现在已2014,可我挑来跳去,还是以这本书开始我的android之旅吧. “疯狂源自梦想,技术成就辉煌.” 让我这个 ...
- Android动画学习笔记-Android Animation
Android动画学习笔记-Android Animation 3.0以前,android支持两种动画模式,tween animation,frame animation,在android3.0中 ...
- Android 数字签名学习笔记
Android 数字签名学习笔记 在Android系统中,所有安装到系统的应用程序都必有一个数字证书,此数字证书用于标识应用程序的作者和在应用程序之间建立信任关系,如果一个permission的pro ...
- Android:日常学习笔记(9)———探究持久化技术
Android:日常学习笔记(9)———探究持久化技术 引入持久化技术 什么是持久化技术 持久化技术就是指将那些内存中的瞬时数据保存到存储设备中,保证即使在手机或电脑关机的情况下,这些数据仍然不会丢失 ...
- Android:日常学习笔记(9)———探究广播机制
Android:日常学习笔记(9)———探究广播机制 引入广播机制 Andorid广播机制 广播是任何应用均可接收的消息.系统将针对系统事件(例如:系统启动或设备开始充电时)传递各种广播.通过将 In ...
- Android:日常学习笔记(8)———开发微信聊天界面
Android:日常学习笔记(8)———开发微信聊天界面 只做Nine-Patch图片 Nine-Patch是一种被特殊处理过的PNG图片,能够指定哪些区域可以被拉升,哪些区域不可以.
- Android:日常学习笔记(8)———探究UI开发(5)
Android:日常学习笔记(8)———探究UI开发(5) ListView控件的使用 ListView概述 A view that shows items in a vertically scrol ...
随机推荐
- Akka.NET v1.0 已发布,支持Mono
Akka.NET 是Java/Scala 流行框架Akka的一个 .NET 开源移植.可用于构建高并发,分布式和容错事件驱动的应用在 .NET 和 Mono 平台之上.Akka.NET 经过一年多的努 ...
- 让MacBook识别noppoo mini 84
让MacBook识别noppoo mini 84 noppoo默认是没有mac驱动的,需要下载一个IOUSBHIDDriverDescriptorOverride否则无法noppoo的键位是识别错误的 ...
- BCL中String.Join的实现
在开发中,有时候会遇到需要把一个List对象中的某个字段用一个分隔符拼成一个字符串的情况.比如在SQL语句的in条件中,我们通常需要把List<int>这样的对象转换为“1,2,3”这样的 ...
- js中几种实用的跨域方法原理详解(转)
今天研究js跨域问题的时候发现一篇好博,非常详细地讲解了js几种跨域方法的原理,特分享一下. 原博地址:http://www.cnblogs.com/2050/p/3191744.html 下面正文开 ...
- 微软MVP攻略 (如何成为MVP?一个SQL Server MVP的经验之谈)
一.本文所涉及的内容(Contents) 本文所涉及的内容(Contents) 初衷 什么是微软MVP? 成为微软MVP的条件? 如何成为微软MVP? (一) 申请时间划分 (二) 前期准备 (三) ...
- 数组去重 JS
我说的数组去重是这样的: var arr = ['f', 'a', 'b', 'd', 'e', 'g'] ; var str='f'; 去除arr中的str 最简单的是遍历arr与str做比较, ...
- MVC5 网站开发之二 创建项目
昨天对项目的思路大致理了一下,今天先把解决方案建立起来.整个解决包含Ninesky.Web.Ninesky.Core,Ninesky.DataLibrary等3个项目.Ninesky.Web是web应 ...
- 计算机程序的思维逻辑 (39) - 剖析LinkedList
上节我们介绍了ArrayList,ArrayList随机访问效率很高,但插入和删除性能比较低,我们提到了同样实现了List接口的LinkedList,它的特点与ArrayList几乎正好相反,本节我们 ...
- 浅谈系列之 javascript原型与对象
在我学习与使用javascript三个月中,我一直对javascript的继承关系以及prototype理解不清,导致很多时候为什么这么用说不出个所以然来.截止到本周为止,通过之前的学习以及自己的再学 ...
- Swift 必备开发库 (高级篇) (转)
1.CryptoSwift swift加密库, 支持md5,sha1,sha224,sha256... github地址: https://github.com/krzyzanowskim/Crypt ...