自定义控件(5)---PorterDuffXfermode图形过滤器

2024-08-31 23:32

本文主要是介绍自定义控件(5)---PorterDuffXfermode图形过滤器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

点击打开链接,下载项目代码。。。。。。。。。。。。

显示的是两个图形一圆一方通过一定的计算产生不同的组合效果,其中圆形是底部的目标图像,方形是上方的源图像。

setXfermode(Xfermode xfermode)

Xfermode国外有大神称之为过渡模式,这种翻译比较贴切但恐怕不易理解,大家也可以直接称之为图像混合模式,因为所谓的“过渡”其实就是图像混合的一种,这个方法跟我们上面讲到的setColorFilter蛮相似的,首先它与set一样没有公开的实现的方法:

同理可得其必然有一定的子类去实现一些方法供我们使用,查看API文档发现其果然有三个子类:AvoidXfermode, PixelXorXfermode和PorterDuffXfermode

PorterDuffXfermode

当大家看到上面API DEMO给出的效果时一定会觉得PorterDuffXfermode其实就是简单的图形交并集计算,比如重叠的部分删掉或者叠加等等,事实上呢!PorterDuffXfermode的计算绝非是根据于此!上面我们也说了PorterDuffXfermode的计算是要根据具体的Alpha值和RGB值的

PS:Src为源图像,意为将要绘制的图像;Dis为目标图像,意为我们将要把源图像绘制到的图像

PorterDuff.Mode.CLEAR     清除图像,很好理解不扯了。
PorterDuff.Mode.DARKEN    变暗
PorterDuff.Mode.DST   只绘制目标图像
PorterDuff.Mode.DST_ATOP  在源图像和目标图像相交的地方绘制目标图像而在不相交的地方绘制源图像
PorterDuff.Mode.DST_IN   只在源图像和目标图像相交的地方绘制目标图像
PorterDuff.Mode.DST_OUT  只在源图像和目标图像不相交的地方绘制目标图像
PorterDuff.Mode.DST_OVER   在源图像的上方绘制目标图像
PorterDuff.Mode.LIGHTEN 变亮
PorterDuff.Mode.MULTIPLY   正片叠底
PorterDuff.Mode.OVERLAY   叠加
PorterDuff.Mode.SCREEN  滤色
PorterDuff.Mode.SRC    显示源图  只绘制源图
PorterDuff.Mode.SRC_ATOP   在源图像和目标图像相交的地方绘制源图像,在不相交的地方绘制目标图像
PorterDuff.Mode.SRC_IN    只在源图像和目标图像相交的地方绘制源图像
PorterDuff.Mode.SRC_OUT    只在源图像和目标图像不相交的地方绘制源图像
PorterDuff.Mode.SRC_OVER    在目标图像的顶部绘制源图像
PorterDuff.Mode.XOR     在源图像和目标图像重叠之外的任何地方绘制他们,而在不重叠的地方不绘制任何内容
这篇博客就记录下DST_IN、DST_OUT 

下面的图片人物的实体部分大小一致(自己截取失误了)

图片从左到右依次是a3_mask  a3   


activity_main.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"android:layout_width="match_parent"android:layout_height="match_parent"android:background="#FFFFFF"android:orientation="vertical" ><com.aigestudio.customviewdemo.views.DisInViewandroid:id="@+id/main_cv"android:layout_width="match_parent"android:layout_height="match_parent" /></LinearLayout>

MeasureUtil

package com.aigestudio.customviewdemo.utils;import android.app.Activity;
import android.util.DisplayMetrics;/*** 测绘工具类*/
public final class MeasureUtil {/*** 获取屏幕尺寸* * @param activity*            Activity* @return 屏幕尺寸像素值,下标为0的值为宽,下标为1的值为高*/public static int[] getScreenSize(Activity activity) {DisplayMetrics metrics = new DisplayMetrics();activity.getWindowManager().getDefaultDisplay().getMetrics(metrics);return new int[] { metrics.widthPixels, metrics.heightPixels };}
}

MainActivity

package com.aigestudio.customviewdemo.activities;import android.app.Activity;
import android.os.Bundle;import com.aigestudio.customviewdemo.R;/*** 主界面* */
public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);}
}

DisInView


package com.aigestudio.customviewdemo.views;import android.app.Activity;
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.PorterDuff;
import android.graphics.PorterDuffXfermode;
import android.util.AttributeSet;
import android.view.View;import com.aigestudio.customviewdemo.R;
import com.aigestudio.customviewdemo.utils.MeasureUtil;
/*** 测试DisIn模式的View*/
public class DisInView extends View {private Paint mPaint;// 画笔private Bitmap bitmapDis, bitmapSrc;// 位图private PorterDuffXfermode porterDuffXfermode;// 图形混合模式private int x, y;// 位图绘制时左上角的起点坐标private int screenW, screenH;// 屏幕尺寸public DisInView(Context context) {this(context, null);}public DisInView(Context context, AttributeSet attrs) {super(context, attrs);// 实例化混合模式porterDuffXfermode = new PorterDuffXfermode(PorterDuff.Mode.DST_IN);// 初始化画笔initPaint();// 初始化资源initRes(context);}/*** 初始化画笔*/private void initPaint() {// 实例化画笔mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);}/*** 初始化资源*/private void initRes(Context context) {// 目标位图bitmapDis = BitmapFactory.decodeResource(context.getResources(), R.drawable.a3);// 源位图bitmapSrc = BitmapFactory.decodeResource(context.getResources(), R.drawable.a3_mask);// 获取包含屏幕尺寸的数组int[] screenSize = MeasureUtil.getScreenSize((Activity) context);// 获取屏幕尺寸screenW = screenSize[0];screenH = screenSize[1];/** 计算位图绘制时左上角的坐标使其位于屏幕中心* 屏幕坐标x轴向左偏移位图一半的宽度* 屏幕坐标y轴向上偏移位图一半的高度*/x = screenW / 2 - bitmapDis.getWidth() / 2;y = screenH / 2 - bitmapDis.getHeight() / 2;}@Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);//先绘制一层白色canvas.drawColor(Color.RED);/** 将绘制操作保存到新的图层(更官方的说法应该是离屏缓存)* (float left, float top, float right, float bottom, Paint paint, int saveFlags)*/int sc = canvas.saveLayer(0, 0, screenW, screenH, null, Canvas.ALL_SAVE_FLAG);// 先绘制dis目标图canvas.drawBitmap(bitmapDis, x, y, mPaint);// 设置混合模式mPaint.setXfermode(porterDuffXfermode);// 再绘制src源图canvas.drawBitmap(bitmapSrc, x, y, mPaint);// 还原混合模式mPaint.setXfermode(null);// 还原画布--给一个人物头像canvas.restoreToCount(sc);}
}



DisOutView


package com.aigestudio.customviewdemo.views;import android.app.Activity;
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.PorterDuff;
import android.graphics.PorterDuffXfermode;
import android.util.AttributeSet;
import android.view.View;import com.aigestudio.customviewdemo.R;
import com.aigestudio.customviewdemo.utils.MeasureUtil;
/*** 测试DisOut模式的View*/
public class DisOutView extends View {private Paint mPaint;// 画笔private Bitmap bitmapSrc;// 位图private PorterDuffXfermode porterDuffXfermode;// 图形混合模式private int x, y;// 位图绘制时左上角的起点坐标private int screenW, screenH;// 屏幕尺寸public DisOutView(Context context) {this(context, null);}public DisOutView(Context context, AttributeSet attrs) {super(context, attrs);// 实例化混合模式porterDuffXfermode = new PorterDuffXfermode(PorterDuff.Mode.DST_OUT);// 初始化画笔initPaint();// 初始化资源initRes(context);}/*** 初始化画笔*/private void initPaint() {// 实例化画笔mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);}/*** 初始化资源*/private void initRes(Context context) {// 获取位图bitmapSrc = BitmapFactory.decodeResource(context.getResources(), R.drawable.a3_mask);// 获取包含屏幕尺寸的数组int[] screenSize = MeasureUtil.getScreenSize((Activity) context);// 获取屏幕尺寸screenW = screenSize[0];screenH = screenSize[1];/** 计算位图绘制时左上角的坐标使其位于屏幕中心* 屏幕坐标x轴向左偏移位图一半的宽度* 屏幕坐标y轴向上偏移位图一半的高度*/x = screenW / 2 - bitmapSrc.getWidth() / 2;y = screenH / 2 - bitmapSrc.getHeight() / 2;}@Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);//先绘制一层白色canvas.drawColor(Color.BLUE);/** 将绘制操作保存到新的图层(更官方的说法应该是离屏缓存)我们将在1/3中学习到Canvas的全部用法这里就先follow me*/int sc = canvas.saveLayer(0, 0, screenW, screenH, null, Canvas.ALL_SAVE_FLAG);// 先绘制一层颜色红色背景canvas.drawColor(Color.YELLOW);// 设置混合模式mPaint.setXfermode(porterDuffXfermode);// 再绘制src源图canvas.drawBitmap(bitmapSrc, x, y, mPaint);// 还原混合模式mPaint.setXfermode(null);// 还原画布canvas.restoreToCount(sc);}
}

+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

源图像在运算时,只是在源图像所在区域与对应区域的目标图像做运算。所以目标图像与源图像不相交的地方是不会参与运算的!这一点非常重要!
     * 不相交的地方不会参与运算,所以不相交的地方的图像也不会是脏数据,也不会被更新,所以不相交地方的图像也永远显示的是目标图像。


package com.example.porterduffmodesrcin;import android.annotation.SuppressLint;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.PorterDuff;
import android.graphics.PorterDuffXfermode;
import android.graphics.RectF;
import android.util.AttributeSet;
import android.view.View;public class MyView extends View {private int width = 400;private int height = 400;private Bitmap dstBmp;private Bitmap srcBmp;private Paint mPaint;/*** 构造函数* * @param context* @param attrs*/public MyView(Context context, AttributeSet attrs) {super(context, attrs);dstBmp = makeDst(width, height);srcBmp = makeSrc(width, height);mPaint = new Paint();}/*** 源图像在运算时,只是在源图像所在区域与对应区域的目标图像做运算。所以目标图像与源图像不相交的地方是不会参与运算的!这一点非常重要!* 不相交的地方不会参与运算,所以不相交的地方的图像也不会是脏数据,也不会被更新,所以不相交地方的图像也永远显示的是目标图像。*/@SuppressLint("DrawAllocation")@Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);int layerID = canvas.saveLayer(0, 0, width, height, mPaint,Canvas.ALL_SAVE_FLAG);canvas.drawBitmap(dstBmp, 0, 0, mPaint);mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));canvas.drawBitmap(srcBmp, width / 2, height / 2, mPaint);mPaint.setXfermode(null);canvas.restoreToCount(layerID);}static Bitmap makeDst(int w, int h) {Bitmap bm = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888);Canvas c = new Canvas(bm);Paint p = new Paint(Paint.ANTI_ALIAS_FLAG);p.setColor(0xFFFFCC44);c.drawOval(new RectF(0, 0, w, h), p);return bm;}static Bitmap makeSrc(int w, int h) {Bitmap bm = Bitmap.createBitmap(w, h, Bitmap.Config.ARGB_8888);Canvas c = new Canvas(bm);Paint p = new Paint(Paint.ANTI_ALIAS_FLAG);p.setColor(0xFF66AAFF);c.drawRect(0, 0, w, h, p);return bm;}
}


这篇关于自定义控件(5)---PorterDuffXfermode图形过滤器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1125305

相关文章

深度解析Spring Boot拦截器Interceptor与过滤器Filter的区别与实战指南

《深度解析SpringBoot拦截器Interceptor与过滤器Filter的区别与实战指南》本文深度解析SpringBoot中拦截器与过滤器的区别,涵盖执行顺序、依赖关系、异常处理等核心差异,并... 目录Spring Boot拦截器(Interceptor)与过滤器(Filter)深度解析:区别、实现

Java实现自定义table宽高的示例代码

《Java实现自定义table宽高的示例代码》在桌面应用、管理系统乃至报表工具中,表格(JTable)作为最常用的数据展示组件,不仅承载对数据的增删改查,还需要配合布局与视觉需求,而JavaSwing... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

一文详解Java Stream的sorted自定义排序

《一文详解JavaStream的sorted自定义排序》Javastream中的sorted方法是用于对流中的元素进行排序的方法,它可以接受一个comparator参数,用于指定排序规则,sorte... 目录一、sorted 操作的基础原理二、自定义排序的实现方式1. Comparator 接口的 Lam

如何自定义一个log适配器starter

《如何自定义一个log适配器starter》:本文主要介绍如何自定义一个log适配器starter的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录需求Starter 项目目录结构pom.XML 配置LogInitializer实现MDCInterceptor

使用Python和Matplotlib实现可视化字体轮廓(从路径数据到矢量图形)

《使用Python和Matplotlib实现可视化字体轮廓(从路径数据到矢量图形)》字体设计和矢量图形处理是编程中一个有趣且实用的领域,通过Python的matplotlib库,我们可以轻松将字体轮廓... 目录背景知识字体轮廓的表示实现步骤1. 安装依赖库2. 准备数据3. 解析路径指令4. 绘制图形关键

Druid连接池实现自定义数据库密码加解密功能

《Druid连接池实现自定义数据库密码加解密功能》在现代应用开发中,数据安全是至关重要的,本文将介绍如何在​​Druid​​连接池中实现自定义的数据库密码加解密功能,有需要的小伙伴可以参考一下... 目录1. 环境准备2. 密码加密算法的选择3. 自定义 ​​DruidDataSource​​ 的密码解密3

spring-gateway filters添加自定义过滤器实现流程分析(可插拔)

《spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔)》:本文主要介绍spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔),本文通过实例图... 目录需求背景需求拆解设计流程及作用域逻辑处理代码逻辑需求背景公司要求,通过公司网络代理访问的请求需要做请

WinForms中主要控件的详细使用教程

《WinForms中主要控件的详细使用教程》WinForms(WindowsForms)是Microsoft提供的用于构建Windows桌面应用程序的框架,它提供了丰富的控件集合,可以满足各种UI设计... 目录一、基础控件1. Button (按钮)2. Label (标签)3. TextBox (文本框

Spring Boot拦截器Interceptor与过滤器Filter深度解析(区别、实现与实战指南)

《SpringBoot拦截器Interceptor与过滤器Filter深度解析(区别、实现与实战指南)》:本文主要介绍SpringBoot拦截器Interceptor与过滤器Filter深度解析... 目录Spring Boot拦截器(Interceptor)与过滤器(Filter)深度解析:区别、实现与实

9个SpringBoot中的自带实用过滤器使用详解

《9个SpringBoot中的自带实用过滤器使用详解》在SpringBoot应用中,过滤器(Filter)是处理HTTP请求和响应的重要组件,SpringBoot自带了许多实用的过滤器,如字符编码,跨... 目录1. CharacterEncodingFilter - 字符编码过滤器功能和配置手动配置示例2