关于PorterDuff.Mode的一点理解

写过自定义view的小伙伴们可能了解或者是用到过PorterDuff.Mode,第一次接触这个东西的时候,就感觉这玩意儿挺魔性,可以玩一把。于是乎就查找了一些关于PorterDuff.Mode的资料以及一些小Demo,印象最深的应该还是刮奖的那个。废话不多说,下面就说说我对PorterDuff.Mode的理解和体会,仅代表个人观点,如有错误希望大家指正。

    1. PorterDuff.Mode是什么?
      在android SDK Paint类中有一个很重要的方法setXfermode,这个方法用于设置图像的过渡模式,所谓过渡是指图像的饱和度、颜色值等参数的计算结果的图像表现。在SDK中Xfermode有三个子类:AvoidXfermode, PixelXorXfermode和PorterDuffXfermode,前两个类在API 16被遗弃了,所以这里不作介绍。PorterDuffXfermode类主要用于图形合成时的图像过渡模式计算,其概念来自于1984年在ACM SIGGRAPH计算机图形学出版物上发表了“Compositing digital images(合成数字图像)”的Tomas Porter和Tom Duff,合成图像的概念极大地推动了图形图像学的发展,PorterDuffXfermode类名就来源于这俩人的名字组合PorterDuff。下面是android SDK中PorterDuff的Mode枚举类型定义。
public enum Mode {
    /** [0, 0] */
    CLEAR       (0),
    /** [Sa, Sc] */
    SRC         (1),
    /** [Da, Dc] */
    DST         (2),
    /** [Sa + (1 - Sa)*Da, Rc = Sc + (1 - Sa)*Dc] */
    SRC_OVER    (3),
    /** [Sa + (1 - Sa)*Da, Rc = Dc + (1 - Da)*Sc] */
    DST_OVER    (4),
    /** [Sa * Da, Sc * Da] */
    SRC_IN      (5),
    /** [Sa * Da, Sa * Dc] */
    DST_IN      (6),
    /** [Sa * (1 - Da), Sc * (1 - Da)] */
    SRC_OUT     (7),
    /** [Da * (1 - Sa), Dc * (1 - Sa)] */
    DST_OUT     (8),
    /** [Da, Sc * Da + (1 - Sa) * Dc] */
    SRC_ATOP    (9),
    /** [Sa, Sa * Dc + Sc * (1 - Da)] */
    DST_ATOP    (10),
    /** [Sa + Da - 2 * Sa * Da, Sc * (1 - Da) + (1 - Sa) * Dc] */
    XOR         (11),
    /** [Sa + Da - Sa*Da, Sc*(1 - Da) + Dc*(1 - Sa) + min(Sc, Dc)] */
    DARKEN      (12),
    /** [Sa + Da - Sa*Da, Sc*(1 - Da) + Dc*(1 - Sa) + max(Sc, Dc)] */
    LIGHTEN     (13),
    /** [Sa * Da, Sc * Dc] */
    MULTIPLY    (14),
    /** [Sa + Da - Sa * Da, Sc + Dc - Sc * Dc] */
    SCREEN      (15),
    /** Saturate(S + D) */
    ADD         (16),
    OVERLAY     (17);
    Mode(int nativeInt) {
        this.nativeInt = nativeInt;
    }
    /**
     * @hide
     */
    public final int nativeInt;
}  

上面代码中每种模式的注释都说明了该模式的alpha通道和颜色值的计算方式,要理解各个模式的计算方式需要先弄明白公式中各个元素的具体含义:

Sa:全称为Source alpha,表示源图的Alpha通道;
Sc:全称为Source color,表示源图的颜色;
Da:全称为Destination alpha,表示目标图的Alpha通道;
Dc:全称为Destination color,表示目标图的颜色.

当Alpha通道的值为1时,图像完全可见;当Alpha通道值为0时,图像完全不可见;当Alpha通道的值介于0和1之间时,图像只有一部分可见。Alpha通道描述的是图像的形状,而不是透明度。

  • 2.什么是源图什么是目标图?
    源图:其实就是你即将要绘制在当前画布上的图。
    目标图:就是你在进行下一步绘制之前当前画布对应绘制区域的图;
    以下是Google官方Sample中给的16种模式下合成的效果图,事实上Google在这儿是挖了一个坑的(两次绘制的图像都是从最左顶点开始,绘制图形的宽高也写的是不同的,具体详细解释请戳:http://blog.csdn.net/iispring/article/details/50472485),真实的合成样子并不完全是这样的;
    image.png

    这才是真实的合成效果(蓝色的绘制起点是从黄色的圆心处):
    image.png
  • 3.PorterDuff.Mode的用法以及一个刮奖小Demo
package com.ebanswers.iqcore;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
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.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;

/**
 * Created by air on 2017/8/23.
 */

public class GuaJiangView extends View {

   private Bitmap mBitmap;
    private Bitmap mDstBitmap;
    private Canvas mCanvas;
    private Path mPath;
    private Paint paint;

    public GuaJiangView(Context context) {
        this(context, null);
    }

    public GuaJiangView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public GuaJiangView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        initView();
    }

    private void initView() {
        //创建一个bitmap(一张图片)
        mBitmap = BitmapFactory.decodeResource(getResources(), R.drawable.picture);
        //根据上面的bitmap创建相同大小的bitmap,此时改bitmap所有的像素点的颜色值全为0,即全透明的一个bitmap
        mDstBitmap = Bitmap.createBitmap(mBitmap.getWidth(), mBitmap.getHeight(), Bitmap.Config.ARGB_8888);
        //创建一个Path对象,后面根据手的touch事件设置path
        mPath = new Path();
        paint = new Paint();
        paint.setAntiAlias(true);//抗锯齿
        paint.setDither(true);//防止抖动
        paint.setAlpha(0);//设置Alpha为0
        paint.setStyle(Paint.Style.STROKE);//设置风格为描边
        paint.setStrokeWidth(50);//设置描边的宽度
        paint.setStrokeCap(Paint.Cap.ROUND);//设置笔触为圆角
        paint.setStrokeJoin(Paint.Join.ROUND);//设置绘制转折的的地方为圆角
        mCanvas = new Canvas(mDstBitmap);//创建一个画布在透明的bitmap上进行绘制
        mCanvas.drawColor(Color.GRAY);//将透明的bitmap绘制成灰色的,所有像素点的色值对应灰色的值
    }

    @Override
    protected void onDraw(Canvas canvas) {
        //第一步绘制一个图片
        canvas.drawBitmap(mBitmap, 0, 0, null);
        //在灰色的bitmap上绘制路径
        drawTouchPath();
        //将灰色的bitmap绘制到原画布上
        canvas.drawBitmap(mDstBitmap, 0, 0, null);
    }

    /**
     * 关键点,绘制手指的touch路径
     */
    private void drawTouchPath() {
        /*
         * 手指触摸后绘制触摸路径,在执行此操作时候
         * 第一次mDstBitmap全部像素色值是灰色(mCanvas.drawColor(Color.GRAY))----目标图(当前画布上的图)
         * 将要绘制的路径mPath----源图(将要绘制的图)
         * 模式为:PorterDuff.Mode.DST_IN(这里DST_IN的IN即是数学中的交集的意思,指针对mPath和mDstBitmap的交集进行合成,实际上就是mPath)
         * 该模式的算法[Sa * Da, Sa * Dc]
         * 目标图(灰色)的透明度通道为1即Da为1,Dc为灰色的色值
         * 源图(画笔设置了Alpha为0),即(mPath绘制的区域)Sa为0
         * 在mPath绘制的区域内目标图Da =1,Dc = Color.GRAY,源图Sa = 0;
         * 则[Sa * Da, Sa * Dc] = [0*1,0*Color.GRAY]=[0,0]
         * 即mPath区域内是全透明的
         */
        paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_IN));
        mCanvas.drawPath(mPath, paint);
    }


    /**
     * 根据touch事件记录手指的触摸路径
     * @param event
     * @return
     */
    @Override
    public boolean onTouchEvent(MotionEvent event) {

        float x = event.getX();
        float y = event.getY();

        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                mPath.reset();
                mPath.moveTo(x, y);
                break;
            case MotionEvent.ACTION_MOVE:
                mPath.lineTo(x, y);
                break;
        }
        invalidate();
        return true;
    }
}  
  • 4.解释已经在代码中进行注释了,运行结果图如下:

image.png

中奖了一个妹纸;
最后还要感谢一下各位,获益匪浅:
http://blog.csdn.net/iispring/article/details/50472485
//www.greatytc.com/p/d11892bbe055

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,817评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,329评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,354评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,498评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,600评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,829评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,979评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,722评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,189评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,519评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,654评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,329评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,940评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,762评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,993评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,382评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,543评论 2 349

推荐阅读更多精彩内容

  • 在Paint中有很多的属性可以设置,比如可以设置阴影,颜色过滤等等,这些会产生不同的奇妙效果,今天就对各种属性探索...
    Jwennnnnnnnnn阅读 2,352评论 0 2
  • 做过图形图像处理coding的Android程序员一定用过或了解过PorterDuff.Mode这个枚举变量中的某...
    Alexyz123阅读 62,318评论 27 154
  • 位图图像和图像蒙板就像Quartz中的任何图形图元。 Quartz中的图像和图像蒙板都由CGImageRef数据类...
    权宜平和阅读 1,657评论 0 3
  • 目录 [Paint 之 Font](#Paint 之 Font) 姿势点 在Android中设置数字类型的参数时如...
    北疆_阅读 629评论 0 2
  • 信息时代,擅长逻辑思考、线性分析的知识工作者占据主导地位。然后,随着物质的富裕和工作全球化,世界正在发生巨大的变化...
    小喜悦阅读 1,572评论 0 0