以前写as3的时候,遮罩效果一个mask属性就搞定了,真是方便。

转到android上以后,发现要实现类似的效果,可以使用Xfermode,android一共提供了三种:

AvoidXfermode;
PixelXorXfermode;
PorterDuffXfermode;

前两种已经不被推荐使用了(据说是因为不支持硬件加速,要生效得强制关闭硬件加速),就不细说了,主要说说第三种,一共提供了十六种效果(as3里也提供了类似,但是更加复杂的方法,所以对我而言还是比较熟悉的),如图所示:

但是要正确的在canvas上实现这些效果,还真是没那么容易,我也是研究了半天,终于实现了自己想要的效果,下面用一个例子说明下我的操作流程。

想要实现的效果是这样的:

简单分析一下,绘制一个圆形和一个矩形,计算好相应的坐标位置,然后使用SRC_IN进行混合就可以了,类似这样:

下面说一下我的操作流程:

1. 绘制border

2. 保存为单独层(canvas.saveLayer),特别注意这一步必须要有,否则无论如何出不来正常效果,起码我试了很久没有成功

3. 绘制填充的圆形,同时也是遮罩

4. 设置笔触的Xfermode为new PorterDuffXfermode(PorterDuff.Mode.SRC_IN)

5. 使用该笔触绘制矩形,进行混合

6. 恢复到canvas上

上代码,大家可以对照看一下各个步骤的具体代码

    @Override
protected void onDraw(Canvas canvas) {
float strokeWidth = getResources().getDimension(R.dimen.stroke_width);
int borderColor = getResources().getColor(R.color.carnation);
int fillColor = getResources().getColor(R.color.carnation_lighter);
int percentColor = getResources().getColor(R.color.carnation_light);
int width = getWidth();
int height = getHeight(); //border
Paint stroke = new Paint(Paint.ANTI_ALIAS_FLAG);
stroke.setStrokeWidth(strokeWidth);
stroke.setStyle(Paint.Style.STROKE);
stroke.setColor(borderColor);
canvas.drawOval(new RectF(strokeWidth/2,strokeWidth/2,width-strokeWidth/2,height-strokeWidth/2),stroke); //save as new layer
int save = canvas.saveLayer(0,0,width,height,null,Canvas.ALL_SAVE_FLAG); //fill background
Paint fill = new Paint(Paint.ANTI_ALIAS_FLAG);
fill.setStyle(Paint.Style.FILL);
fill.setColor(fillColor);
canvas.drawOval(new RectF(strokeWidth - 1, strokeWidth - 1, width - strokeWidth + 1, height - strokeWidth + 1), fill); //mix rect
fill.setColor(percentColor);
fill.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));
canvas.drawRect(0,(1-mPercent)*height,width,height,fill); //restore to canvas
canvas.restoreToCount(save); super.onDraw(canvas);
}

之前也查了不少文章,貌似没有看到多少着重说saveLayer的,还是我对照官方apidemos源码试出来的,希望对遇到同样疑问的朋友有所帮助!

2015/7/9 更新:

1. 要实现混合的两个图形,必须位于同一个layer上,经测试位于不同layer上是无法混合的,即使最后都绘制到了canvas上。

2. 不同的绘制顺序,可能有不同的效果,注意一下逻辑即可。

更新一段复杂点的例子

使用了两种混合方式SRC_IN和CLEAR,主要代码如下:

     @Override
protected void onDraw(Canvas canvas) {
int width = getWidth();
int height = getHeight(); float strokeWidth = DimenUtils.dp2px(4);
float pointRadius = DimenUtils.dp2px(4);
float gap = DimenUtils.dp2px(4);
float monthRadius = height * 0.2f;
float textSize = DimenUtils.dp2px(14); int color = getResources().getColor(R.color.carnation);
int lightColor = getResources().getColor(R.color.carnation_light);
int lighterColor = getResources().getColor(R.color.carnation_lighter); float degree = 360*mRate; Paint paint = new Paint(Paint.ANTI_ALIAS_FLAG); //new layer
int save = canvas.saveLayer(0, 0, width, height, null, Canvas.ALL_SAVE_FLAG); //draw percent
canvas.save();
float fillDistance = pointRadius+gap+strokeWidth/2;
canvas.translate(fillDistance,fillDistance);
RectF fillRect = new RectF(0,0,width-2*fillDistance,height-2*fillDistance);
paint.setColor(lightColor);
paint.setStyle(Paint.Style.FILL);
canvas.drawOval(fillRect,paint);
//mix rect
paint.setColor(color);
paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));
canvas.drawRect(0, (1 - mPercent) * (height-2*fillDistance), width-2*fillDistance, height-2*fillDistance, paint);
canvas.restore(); //border
paint.setXfermode(null);
paint.setStrokeWidth(strokeWidth);
paint.setStyle(Paint.Style.STROKE);
paint.setColor(lighterColor);
RectF borderRect = new RectF(pointRadius,pointRadius,width-pointRadius,height-pointRadius);
canvas.drawOval(borderRect, paint);
paint.setColor(color);
canvas.drawArc(borderRect,270,degree,false,paint);
//draw point
canvas.save();
paint.setStyle(Paint.Style.FILL);
canvas.translate(width/2,height/2);
canvas.rotate(degree);
canvas.drawCircle(0,pointRadius-height/2,pointRadius,paint);
canvas.restore(); //draw month
canvas.save();
canvas.translate(width*0.7f,height*0.8f);
paint.setColor(Color.BLACK);
paint.setStyle(Paint.Style.STROKE);
paint.setStrokeWidth(gap);
paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR));
canvas.drawCircle(0,0,monthRadius+gap/2,paint);
paint.setXfermode(null);
paint.setStyle(Paint.Style.FILL);
paint.setColor(getResources().getColor(R.color.orange));
canvas.drawCircle(0,0,monthRadius,paint);
canvas.restoreToCount(save);
}

【原】使用Xfermode正确的绘制出遮罩效果的更多相关文章

  1. iOS: 如何正确的绘制1像素的线

    iOS 绘制1像素的线 一.Point Vs Pixel iOS中当我们使用Quartz,UIKit,CoreAnimation等框架时,所有的坐标系统采用Point来衡量.系统在实际渲染到设置时会帮 ...

  2. 移动端和PC端弹出遮罩层后,页面禁止滚动的解决方法及探究

    PC端解决方案 pc端的解决思路就是在弹出遮罩层的时候取消已经存在的滚动条,达到无法滚动的效果. 也就是说给body添加overflow:hidden属性即可,IE6.7下不会生效,需要给html增加 ...

  3. jquery特效(7)—弹出遮罩层且内容居中

    上周写了几个小特效,其中有个点击按钮弹出遮罩层的特效,下面来看最终实现的效果: 由于是测试的程序,所以我未加关闭的按钮. 一.主体程序 <!DOCTYPE html> <html&g ...

  4. Css动画形式弹出遮罩层,内容区上下左右居中于不定宽高的容器中

    <!DOCTYPE html> <html> <head> </head> <body id="body"> <! ...

  5. jquery制作弹出层带遮罩效果,点击阴影部分层消失

    jquery制作弹出层带遮罩效果,点击阴影部分层消失. 整体还是比较简单的. HTML代码很简单 <a href="#" class="big-link" ...

  6. 使用 HTML5 Canvas 绘制出惊艳的水滴效果

    HTML5 在不久前正式成为推荐标准,标志着全新的 Web 时代已经来临.在众多 HTML5 特性中,Canvas 元素用于在网页上绘制图形,该元素标签强大之处在于可以直接在 HTML 上进行图形操作 ...

  7. 基于jQuery向下弹出遮罩图片相册

    今天给大家分享一款基于jQuery向下弹出遮罩图片相册.单击相册图片时,一个遮罩层从上到下动画出现.然后弹出显示图片.这款插件适用浏览器:IE8.360.FireFox.Chrome.Safari.O ...

  8. Cocos2d-x--Box2D绘制出两个矩形框的解决方案

    一个简单的Demo,只是在程序窗口绘制出一个矩形 找到以下代码,注释掉其中一句 效果:

  9. js弹出对话框,遮罩效果,

    刚刚来到实习单位,我跟着廖哥做项目.然后他分配给我一个小小的任务,实现起来总的效果如下: 然后,但我们单击显示数目这个链接的时候,就会弹出一个又遮罩效果的对话框,如下图: 当我们在对话框中再点击里面的 ...

随机推荐

  1. BOM and Event Source

    EventSource: var test = function(){ var btn = event.srcElement; var str += btn.type + ":" ...

  2. win7 打印机共享

    1.在工具->文件夹选项->查看,将"使用简单文件共享"前面的勾勾去掉2.在控制面板->用户帐号,将guest帐户启用3.运行"gpedit.msc&q ...

  3. 【Java EE 学习 22 下】【单线程下载】【单线程断点下载】【多线程下载】

    一.文件下载简述 1.使用浏览器从网页上下载文件,Servlet需要增加一些响应头信息 (1)response.setContentType("application/force-downl ...

  4. mac

    command+R 刷新页面 command+shift+C 我的电脑 finder->应用程序 -> 实用程序 -> 终端 打开 cmd command+F3 显示桌面 comma ...

  5. <更新日期03-31-2016> 复利计算5.0 <已改进>

    作业要求: 1.客户说:帮我开发一个复利计算软件. 完成复利公式计算程序,并成功PUSH到github上. 客户提出: 2.如果按照单利计算,本息又是多少呢? 3.假如30年之后要筹措到300万元的养 ...

  6. STM32解密STM32F103芯片解密STM32F103R6单片机破解多少钱?

    STM32解密STM32F103芯片解密STM32F103R6单片机破解多少钱? STM32F系列单片机芯片解密型号: STM32F100  |  STM32F101  |  STM32F102  | ...

  7. BZOJ 2844 albus就是要第一个出场 ——高斯消元 线性基

    [题目分析] 高斯消元求线性基. 题目本身不难,但是两种维护线性基的方法引起了我的思考. void gauss(){ k=n; F(i,1,n){ F(j,i+1,n) if (a[j]>a[i ...

  8. 2012 Multi-University #7

    最短路+拆点 A As long as Binbin loves Sangsang 题意:从1走到n,每次都是LOVE,问到n时路径是连续多个"LOVE"的最短距离.秀恩爱不想吐槽. 分析:在普通的最 ...

  9. Python for Infomatics 第13章 网页服务一(译)

    注:文章原文为Dr. Charles Severance 的 <Python for Informatics>.文中代码用3.4版改写,并在本机测试通过. 一旦利用程序通过HTTP协议获得 ...

  10. PHP遍历、删除文件夹中的所有文件

    <?php header("Content-type:text/html;charset=utf-8"); /** * getDirFile 遍历文件夹中的所有文件 * @p ...