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 ...
随机推荐
- 【腾讯Bugly干货分享】Android性能优化典范——第6季
本文来自于腾讯bugly开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/580d91208d80e49771f0a07c 导语 这里是Android性能优 ...
- 相关query挖掘
1.何为相关query 我通常也把相关query称为相似query,搜索日志中一个用户在短时间内的一系列搜索词被称为相关query.相关就是两个query间有一定的关系,反映了用户在当时的需求.本文就 ...
- node之path模块
node之path模块 原文链接 //引用该模块 var path = require("path"); 1.路径解析,得到规范化的路径格式 对window系统,目录分隔为'', ...
- Python 小白的新手教程(一)
本文是 python 入门级别的基础知识,包括数据类型和变量.输入输出.字符串和编码.list tuple dict set .条件判断.循环.函数.切片 迭代 列表生成器 生成器 迭代器等. 参考课 ...
- 一步步构造自己的vue2.0+webpack环境
前面vue2.0和webpack都已经有接触了些(vue.js入门,webpack入门之简单例子跑起来),现在开始学习如何构造自己的vue2.0+webpack环境. 1.首先新建一个目录vue-wk ...
- Asp.net中存储过程拖拽至dbml文件中,提示无法获得返回值
Asp.net中存储过程拖拽至dbml文件中,提示无法获得返回值,去属性表中设置这时候会提示你去属性表中更改返回类型. 其实存储过程返回的也是一张表,只不过有时候存储过程有点复杂或者写法不规范的话不能 ...
- C可变参数的函数
我们实现一个简单的printf函数(可变参数) #include <stdio.h> #include <stdarg.h> void myprintf(const char ...
- Vue1.0 的技术栈
vuejs概述 Vue.js是用于构建交互式的Web界面的库.它提供了MVVM数据绑定和一个可组合的组件系统,具有简单.灵活的API. 结合node.js 可以实现前后端开发从物理上的分离.使前端负责 ...
- C++服务器开发之笔记三
为什么需要原子性操作? 我们考虑一个例子:(1)x++这个常见的运算符在内存中是怎样操作的?从内存中读x的值到寄存器中,对寄存器加1,再把新值写回x所处的内存地址 若是有两个线程同时对同一个变量++, ...
- Chrome在302重定向的时候对原请求产生2次请求的问题说明
这个问题应该确确实实是一个Chrome的BUG,我在自己的编程环境中发现,并在多个服务器,多个编程语言的运行环境,以及多个浏览器下都测试过,都看到有2次请求出现.为了证明不是自己环境的问题,我也特意去 ...