Android 高级UI解密 :Paint滤镜 与 颜色过滤(矩阵变换)
Posted 鸽一门
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Android 高级UI解密 :Paint滤镜 与 颜色过滤(矩阵变换)相关的知识,希望对你有一定的参考价值。
若是曾经查看过系统UI的源码, 会发现其中使用了一些渲染效果,例如将图片加上黑白、怀旧的效果,生活中常用的逆天美颜相机,其中的原理就是使用了滤镜效果、颜色通道过滤。若还要深究其原理组成,便涉及到了高等数学里的矩阵变换,也就是android 中的颜色矩阵!此篇文章便来一探究竟如何实现滤镜和其原理组成。
(关于矩阵这一块,无需过度深究数学部分,此处为了充分理解渲染效果,只需了解大概原理,利用其API完成简单滤镜效果。)
其实滤镜效果就是对图像进行一定的过滤加工处理。例如PS软件中常见的滤镜效果:模糊、锐化、素描等等,以上功能便涉及到滤镜效果的矩阵。使用Paint设置滤镜效果,可分为以下两个方面:
- Alpha滤镜处理
- 颜色RGB的滤镜处理
以上两个方面正好对应Paint的两个重点API,分别是以下:
setMaskFilter(MaskFilter filter)
是基于整个画面来进行过滤。setColorFilter(ColorFilter filter)
是对每个像素的颜色进行过滤。
此篇文章分为以上两个方面来详细解析滤镜、颜色过滤的奥秘。
Android 高级UI解密 (一) :实例详解Paint 与 高级渲染
一. Alpha滤镜处理
Alpha就是对透明度的处理,涉及到MaskFilter这个类,它是一个抽象类:
MaskFilter是在绘制Alpha通道遮罩之前执行转换的对象的基类。 MaskFilter的子类可以通过Paint的setMaskFilter方法设置到画笔中。 而模糊遮罩滤镜BlurMaskFilter和浮雕遮罩滤镜EmbossMaskFilter是实现MaskFilter的子类。
1. 模糊遮罩滤镜BlurMaskFilter
见名识意,此滤镜类似于一种模糊效果。以构造方法中指定的半径模糊其边缘,另外还可指定模糊的风格,模糊其内部、外部、边框或者本身。
//构造方法
BlurMaskFilter (float radius, BlurMaskFilter.Blur style)
参数说明:
- radius:模糊区域半径;
- style:模糊的格式 (BlurMaskFilter.Blur 类型)
- INNER:模糊内部边框,外部不变;
- NORMAL:模糊内外边框;
- OUTER:内部不变,模糊外部;
- SOLID:在边界内部画实体,模糊外面;
注意:注意以上四种类型的解释差异,模糊内部和模糊内部边框是不同的!
- 内外部边框相关:INNER只是简单模糊其内部边框,图片外部呈现淡白色;而NORMAL是直接模糊内外边框,图片外部已经呈现图片边缘的背景色;
- 内外部相关:OUTER效果比较奇葩,图片内部空白,外部模糊成图片边缘的背景色;SOLID则是保持内部实体,外部模糊成图片边缘的背景色。
代码测试后的效果图如下:
2. 浮雕遮罩滤镜(EmbossMaskFilter)
EmbossMaskFilter(float[] direction, float ambient, float specular, float blurRadius)
参数说明:
- direction:指定光源方向的3个标量[x,y,z]的数组;
- ambient:指定环境光量强度[0,1];
- specular:指定镜面反射系数(例如8);
- blurRadius:指定模糊半径(例如3);
mPaint.setMaskFilter(new EmbossMaskFilter(new float[]400,100,100, 0.5f, 60, 80));
canvas.drawBitmap(bitmap, 400, 100, mPaint);
注意:查看EmbossMaskFilter类的构造方法源码,发现其真正创建对象是调用了native方法,因此这也表明google在android 的graphics包中准备了一系列的滤镜,也需要传入相应的参数,而其中参数的运算是非常复杂的,涉及到矩阵运算。
需要强调的是“矩阵运算”并非只是简单的公式计算,试想一块手机屏幕所含的像素点有多少,假设是1080P,若一张图片覆盖整个屏幕,需要处理每一个像素点,工作量是很大的,为了计算效率而采用了native方法,交由它来完成。
二. 颜色RGB的滤镜处理
滤镜的所有处理效果都是通过颜色矩阵的变换实现的,例如生活中常见的美颜相机,它实现的一些特效:高光、复古、黑白等滤镜。那么首先来了解何为矩阵?其中涉及到多阶矩阵,这里以二阶矩阵为例进行讲解。
1. 矩阵简析
(1)定义
(2)矩阵乘法
矩阵其实就相当于一个二维数组,而重点则在于矩阵之间的计算,特别是乘法计算与后续滤镜计算有关,乘法运算如下:
矩阵的乘法计算步骤如下:
- 将第一个矩阵A的第一行,与第二个矩阵B的第一列的数字分别相乘,得到的结果相加,最终的值做为结果矩阵的第(1,1)位置的值(即第一行,第一列)。
- 同样,A矩阵的第一行与B矩阵的第二列的数字分别相乘然后相加,结果做为结果矩阵第(1,2)位置的值(即第一行第二列)。
- 依次类推。
注意:矩阵A乘以矩阵B和矩阵B乘以矩阵A的结果是不一样的。
示例如下:
2. 色彩信息的矩阵表示
鲁迅曾经说过(并没有):矩阵运算对于Android像素处理的意义极大!
以上在重点强调矩阵中的乘法运算后,接下来将解密其奥妙。颜色的组成为ARGB,这里先不讨论Alpha透明度,以RGB为主。举个例子:美颜相机中的图片美白原理就是将红色、绿色、蓝色进行位移,可以获得不同的效果,而其中的计算则可以借助矩阵完成。
(1)四阶表示
ARGB的四阶表达式如下:
如果想将色彩(0,255,0,255)更改为半透明时,可以使用下面的的矩阵运算来表示:
其实颜色变换就是将矩阵看成一套数学模型,便于计算ARGB值。
(2)五阶矩阵
任何一个颜色都是三色素(红绿蓝)构成的,也就是RGB。例如黄色是由红色和绿色形成。
考虑下面这两个变换需求:
- 红色分量值更改为原来的2倍;
- 绿色分量增加100;
若要实现以上变换,四阶矩阵的乘法无法实现。根据以上ARGB四阶矩阵的运算规则,只能进行乘法运算,而无法进行加法运算,因此在四阶色彩变换矩阵上增加一个“哑元坐标”,来实现所列的矩阵运算,也就是“五阶矩阵”。过程如下图:
第一个矩阵中前四列中任然代表ARGB,而第五列则是分量值,即绿色需要加的100,200 = 1*100+100
。
3. 实例
(1)需求
通过矩阵变换讲一个图片、颜色块,过滤掉其中的红色、绿色,只留下蓝色。
(2)代码
查看以下代码,绘制出以下两个图形进行对比:
- 第一个矩形设置其ARGB颜色,整体偏红色;
- 重点是第二个矩形的颜色过滤器设置:创建其ColorMatrix 矩形变换对象,其中根据上部分公式讲解,结合过滤红色、绿色的需求。因此第一行第一列为R值为0,第二行第二列为G值为0,第三行第三列为B值为1,第四行第四列为A值为1,最后一列是分量值,皆为0。
//=====颜色RGB的滤镜处理===
mPaint.setColor(Color.argb(255,200,100,100));
canvas.drawRect(200, 200, 400, 400, mPaint);
canvas.translate(400,0);
//五阶矩阵,R、G为0,A、B为1,第五列为分量,不需要进行平移为0
ColorMatrix matrix = new ColorMatrix(new float[]
0,0,0,0,0,
0,0,0,0,0,
0,0,1,0,0,
0,0,0,1,0,
);
//设置颜色过滤器
mPaint.setColorFilter(new ColorMatrixColorFilter(matrix));
canvas.drawRect(200, 200, 400, 400, mPaint);
(3)效果展示
根据以上对比图实践成功,将第一个矩形中的颜色(#C86464)过滤,仅留下蓝色(#000064)。若纯色块对比不明显,难以理解“过滤”的概念,直接使用图片对比,将以上代码中的drawRect
改成drawBitmap
即可,效果如下:
注意:其滤镜的原理还是在于设置的颜色过滤器——矩阵变换,同理可只过滤掉红色、绿色、蓝色或任意组合,都可由矩阵变换完成。其中不仅可以修改ARGB值(乘法),同样可以修改五阶矩阵中代表分量值的第五列(加法),不同的修改方式可以形成各式滤镜效果。例如美图秀秀中的各种滤镜其原理是如此,内部包含大量的滤镜模板(库)。
4. 实践滤镜效果——色彩运算
以上简单的举个例子实践了矩阵变换,下面来总结归纳其矩阵运算,无非是以下两种:
- 色彩的平移运算(加法运算)
- 色彩的缩放运算(乘法运算)
以下代码实践5种滤镜效果来熟悉运用矩阵运算。
(1)反相效果 —— 曝光
常见的照相机中的曝光也就是矩阵运算中的反向,即设原先的ARGB值为100,200,250
,用最大值255减去原来的值,结果为155,55,5
,就是“曝光”。
矩阵运算解析:其余代码同上个代码示例相同,这里主要是矩阵运算方面的变化:反向效果涉及到用255减去原值,因此直接结合乘法与加法(分量值),可实现该结果!
代码示例如下:
//曝光效果
ColorMatrix matrix = new ColorMatrix(new float[]
-1,0,0,0,255,
0,-1,0,0,255,
0,0,-1,0,255,
0,0,0,1,0,
);
图片展示效果如下:
(2)美白效果 —– 颜色增强
矩阵运算解析:首先需要知道1f是图像原色,即不改变图像滤镜。若要增强颜色达到一种美白的效果,只需要将RGB值稍加增大即可。
代码如下:
//美白效果
ColorMatrix matrix = new ColorMatrix(new float[]
1.2f,0,0,0,0,
0,1.2f,0,0,0,
0,0,1.2f,0,0,
0,0,0,1.2f,0,
);
图片展示效果如下:
(3)黑白效果
去色原理:只要把RGB三通道的色彩信息设置成一样,即R=G=B,那么图像就变成了灰色,并且为了保证图像亮度不变,同一个通道中的 R+G+B=1。
例如 0.213+0.715+0.072 = 1
,三个数字是根据色彩光波频率及色彩心理学计算出来的。(人对色彩的感知是融合色彩显示和视觉心理成分的,例如你盯着一个纯色快看一段时间,再看向别的事物,此时你看的事物都是自带滤镜的)
矩阵运算解析:根据以上三个值设置RGB即可得到黑白图像的效果,但是需要将五阶矩阵中代表RGB的前四阶中代表各列中每一行的值都设置,A无需考虑,最后以列分量也无需考虑。
代码如下:
//黑白图片
ColorMatrix matrix = new ColorMatrix(new float[]
0.213f, 0.715f,0.072f,0,0,
0.213f, 0.715f,0.072f,0,0,
0.213f, 0.715f,0.072f,0,0,
0, 0, 0, 1f,0,
);
图片展示效果如下:
(4)色彩反射效果
何为反射效果?例如将图像中红色的成分替换成绿色的成分,绿色的成分替换成红色的。
//原始效果
ColorMatrix matrix = new ColorMatrix(new float[]
1f,0,0,0,0,
0,1f,0,0,0,
0,0,1f,0,0,
0,0,0,1f,0,
);
矩阵运算解析:以上是图像原始效果,即ARGB皆为1F,即使设置了此颜色过滤,图像并无任何改变。与此对比,要实现色彩反射效果,比如红色和绿色交换—-把第一行和第二行交换即可。
代码如下:
//发色效果(比如红色和绿色交换----把第一行和第二行交换)
ColorMatrix matrix = new ColorMatrix(new float[]
0,1f,0,0,0,
1f,0,0,0,0,
0,0,1f,0,0,
0,0,0,1f,0,
);
图片展示效果如下:
(5)复古效果
矩阵运算解析:这是美颜相机中常见的一款滤镜形式,矩阵中有特定的算法模板。
代码如下:
//复古风格
ColorMatrix matrix = new ColorMatrix(new float[]
1/2f,1/2f,1/2f,0,0,
1/3f,1/3f,1/3f,0,0,
1/4f,1/4f,1/4f,0,0,
0,0,0,1f,0,
);
图片展示效果如下:
三. ColorMatrix 相关详解
上一点讲解了多个有关矩阵变换的例子,对实践矩阵运算和其产生的效果稍有理解,如果涉及到美容相机或者图像处理的一些效果需求,绝大可能会用到矩阵运算,而矩阵运算肯定涉及到ColorMatrix,此部分内容来详细解析ColorMatrix。
ColorMatrix就是4x5矩阵,用于转换位图的RGB颜色和Alpha分量。 该矩阵可以作为单个数组传递,并按以下方式处理:
[ a, b, c, d, e,
f, g, h, i, j,
k, l, m, n, o,
p, q, r, s, t ]
当应用于颜色[R,G,B,A]时,每个值的范围是[0, 255],生成的颜色计算如下:
R’ = a*R + b*G + c*B + d*A + e;
G’ = f*R + g*G + h*B + i*A + j;
B’ = k*R + l*G + m*B + n*A + o;
A’ = p*R + q*G + r*B + s*A + t;
1、构造方法
(1)用指定的值数组创建一个新的Colormatrix
ColorMatrix matrix = new ColorMatrix(new float[]);
代码示例:
ColorMatrix matrix = new ColorMatrix(new float[]
1.2f,0,0,0,0,
0,1.2f,0,0,0,
0,0,1.2f,0,0,
0,0,0,1.2f,0,
);
(2)创建一个新的Colormatrix,后续设值
ColorMatrix matrix = new ColorMatrix();
float[] scr = ...;
matrix.set(src)
2.设置色彩的缩放函数(矩阵的乘法运算)
setScale(float rScale, float gScale, float bScale, float aScale)
API作用:设置此颜色矩阵按指定的值进行缩放。
参数:分别是设置R、G、B、A相乘的值。
注意:在上一部分的第四点中已经介绍过矩阵变换中两种重要运算 —— 乘法和加法,并且在以上示例中都是直接修改 4*5 数组矩阵。Colormatrix提供的此API可以轻易设置R、G、B、A需要相乘的值。另外追踪其源码实现也很简单,就是根据参数设置值与数组中对应的R、G、B、A相乘。
3.设置饱和度(矩阵的加法运算)
setSaturation(float sat)
API作用:设置矩阵以影响颜色的饱和度。
参数: sat参数指映射到灰色的值。 1代表原色,0代表灰色,>1则增加饱和度。
API源码
public void setSaturation(float sat)
reset();
float[] m = mArray;
final float invSat = 1 - sat;
final float R = 0.213f * invSat;
final float G = 0.715f * invSat;
final float B = 0.072f * invSat;
m[0] = R + sat; m[1] = G; m[2] = B;
m[5] = R; m[6] = G + sat; m[7] = B;
m[10] = R; m[11] = G; m[12] = B + sat;
源码剖析
有详细查看上一部分内容的读者,你会发现源码中这个三个特殊值很熟悉,它就是在讲解滤镜中黑白效果中有提到的去色原理:R+G+B=1
从而图片呈现灰色,同时考虑到色彩光波频率及色彩心理学,计算得出最佳值RGB最佳值:0.213+0.715+0.072 = 1
。因此在此基础之上,根据参数设置的值来修改RGB值,达到图像饱和度变化!
实例
下面实现一个简单的demo,在onDraw
方法中设置图像的颜色过滤器,设置饱和度为0,在onTouchEvent
方法中监听点击处理,每次触控其饱和度以0.3f 增加,查看图像变化效果:
(代码文末提供,在此不赘述)
效果分析
查看以上效果符合预期情况,最初设置参数为0,因此图像呈现出灰色,随着不断点击,饱和度依次增加,即RGB值逐渐增加,颜色恢复成原色,接着点击,参数值大于1,图像明显过饱和。
4.色彩旋转函数
setRotate(int axis, float degrees)
API作用:通过指定的值设置颜色轴上的旋转。
参数: axis代表旋转轴,0红色轴,1绿色,2蓝色;degrees代表旋转的度数。
注意: 类似于上图中的3D效果,例如这里指定围绕B轴选装,则B值不变,R、G值会随之改变,而且一圈360度旋转完会再次恢复到原始颜色图像。
实例
下面实现一个简单的demo,在onDraw
方法中设置图像的颜色过滤器,设置围绕R轴旋转,在onTouchEvent
方法中监听点击处理,每次触控其旋转度数以20f增加,查看图像变化效果:
(代码文末提供,在此不赘述)
效果分析
查看以上效果,根据设置是围绕R轴旋转,随着度数增加,图像渐渐呈现红色,最终又慢慢恢复成图像原色。既然将此颜色过滤器指定围绕R轴,随着旋转角度增加,达到某一个临界点,图像会逐渐过滤掉所有颜色,只剩下红色。继续增加,颜色接着改变,直至旋转到360度恢复成原图像。
5. ColorFilter的子类
ColorFilter类:一个颜色过滤器可以和Paint一起使用来修改用这个颜料绘制的每个像素的颜色。从它的名字也可知,为绘制设置颜色过滤。颜色过滤就是为绘制的内容设置统一的过滤规则。
Paint.setColorFilter(ColorFilter filter)
一般是通过Paint画笔设置其颜色过滤器,由于ColorFilter是抽象类,使用的是它的三个子类,如下:
(1)ColorMatrixColorFilter 色彩矩阵的颜色顾虑器
类作用:通过4x5彩色矩阵转换颜色的彩色滤镜。 这个滤镜可以用来改变像素的饱和度,从YUV转换到RGB等。
//构造函数
new ColorMatrixColorFilter(ColorMatrix matrix);
构造方法参数:就是**ColorMatrix**4x5矩阵,用于转换位图的RGB颜色和Alpha分量。
注意:本篇文章第二大部分颜色RGB的滤镜处理,使用的都是ColorMatrixColorFilter 色彩矩阵的颜色顾虑器,在此无需赘述。
(2) LightingColorFilter 光照颜色过滤器(过滤颜色和增强色彩)
类作用:可用于模拟简单照明效果的滤色器。 一个LightingColorFilter由两个参数定义,一个用于将源颜色(称为colorMultiply)和一个用于添加到源颜色(称为colorAdd)的颜色相乘。 这个彩色滤光片保持不变的alpha通道。
给定源颜色RGB,由此得出R’G’B’颜色:
R' = R * colorMultiply.R + colorAdd.R
G' = G * colorMultiply.G + colorAdd.G
B' = B * colorMultiply.B + colorAdd.B
//构造函数
new LightingColorFilter(int mul, int add);
构造方法参数: mul是与源RGB相乘的值;add是与源RGB相加的分量值。
注意:此方法就是结合了矩阵运算中的乘法和加法,更加简化。需要注意的是参数类型虽为int值,但规定为颜色值,即16进制的值,例如0x00ff00,说白了就是范围 [0,255]区间的颜色值。
实例
这里做一个简单的测试,设置LightingColorFilter 光照颜色过滤器的两个参数分别为0x00ff00,0x000000。0x00ff00就是一个绿色值(原谅色~),会与RGB源值相乘,而这里相加的值设置为0,不做修改。因此设置该光照颜色过滤器后,图像整体应该呈现出绿色。
......
mPaint.setColorFilter(new LightingColorFilter(0x00ff00, 0x000000));
canvas.drawBitmap(bitmap, null, new RectF(200, 200, 400, 400*bitmap.getHeight()/bitmap.getWidth()), mPaint);
效果如下:
效果分析
图片效果与理想效果相符,因此根据此API可轻易完成矩阵变换中的乘法和加法运算,实现需求效果。
(3) PorterDuffColorFilter 图形混合滤镜(图形学理论)
类作用:一种彩色滤光片,可用于使用单色和特定的Porter-Duff复合模式为源像素着色。 就是使用一个指定的颜色和一种指定的 PorterDuff.Mode 来与绘制对象进行合成。
//构造函数
new PorterDuffColorFilter(int color, PorterDuff.Mode mode);
构造方法参数: color 参数是指定的颜色;mode 参数是指定的 Mode,即PorterDuff.Mode。
注意: PorterDuffColorFilter 和ComposeShader两者都使用到了PorterDuff.Mode,其中一个是颜色过滤器,一个是着色器,而且PorterDuffColorFilter颜色过滤器只能指定一种颜色作为源,而不是一个 Bitmap。
文章小结
此篇文章的主要是研究Paint的两个重点API:从Alpha滤镜处理、颜色RGB的滤镜处理两个方面拓展开,其中涉及到了高数知识——矩阵运算,此篇为了研究颜色过滤原理稍作介绍,并实践展示了几个滤镜效果,学会API实际运用。
此篇文章是有关于有关Paint的高级使用,结合上一篇Paint的基本使用,Paint相关知识重点暂时介绍到这里,下一篇文章将开始归纳Canvas画布相关内容。
(代码整理中,后续会提供)
若有错误,欢迎指教~
以上是关于Android 高级UI解密 :Paint滤镜 与 颜色过滤(矩阵变换)的主要内容,如果未能解决你的问题,请参考以下文章
Android 高级UI解密 :Paint图形文字绘制 与 高级渲染
Android 高级UI解密 :Paint图形文字绘制 与 高级渲染
Android 高级UI解密 :PathMeasure截取片段 与 切线(新思路实现轨迹变换)
Android 高级UI解密 :PathMeasure截取片段 与 切线(新思路实现轨迹变换)