在日常开发过程中,图片的特效处理是一个很常见的需求。除了颜色特效,还有就是外形的特效,比如圆角图片,圆形图片等。今天我们就来学习下图片的外形特效相关的知识
Paint有一个专门用于处理图片外形特效的API
//Paint.class
public Xfermode setXfermode(Xfermode xfermode) {
long xfermodeNative = 0;
if (xfermode != null)
xfermodeNative = xfermode.native_instance;
nSetXfermode(mNativePaint, xfermodeNative);
mXfermode = xfermode;
return xfermode;
}
在Android的SDK中Xfermode只有一个子类:PorterDuffXfermode
PorterDuffXfermode控制图像的混合模式,影响的是2个图层交集区域的显示方式。
//PorterDuff.class
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 (16),
/** [Sa + Da - Sa*Da,
Sc*(1 - Da) + Dc*(1 - Sa) + max(Sc, Dc)] */
LIGHTEN (17),
/** [Sa * Da, Sc * Dc] */
MULTIPLY (13),
/** [Sa + Da - Sa * Da, Sc + Dc - Sc * Dc] */
SCREEN (14),
/** Saturate(S + D) */
ADD (12),
OVERLAY (15);
Mode(int nativeInt) {
this.nativeInt = nativeInt;
}
/**
* @hide
*/
public final int nativeInt;
}
这么多种,分别是什么效果呢?看看下面的效果图就明白了。
其中需要说明的是Dst表示底层图层,也可以理解为遮罩层,而Src则是原图
原理很简单,就是通过一个和图片大小一样的带有圆角的图层和图片图层叠加起来
(1)初始化画布Canvas和画笔Paint
Bitmap bitmap = Bitmap.createBitmap(src.getWidth(), src.getHeight(), Bitmap.Config.ARGB_8888);
Canvas canvas = new Canvas(bitmap);
Paint paint = new Paint();
paint.setAntiAlias(true);
(2)在画布上画一个带圆角的图层,图层尺寸和原图一样
//这里80为圆角的半径
canvas.drawRoundRect(new RectF(0, 0, src.getWidth(), src.getHeight()),80,80, paint);
(3)为Paint设置图形混合模式SRC_IN,SRC_IN模式从上面的模式效果图中我们可以看出叠加后显示的是重叠部分后面一张图的部分
paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));
(4)绘制图片,并把新的Bitmap对象设置给ImageView
canvas.drawBitmap(src, 0, 0, paint);
imageView.setImageBitmap(bitmap)
通过以上的简单几步,我们就得到了一个圆角图片的效果。
上面我们采用的混合显示模式是SRC_IN,如果我们采用的是SRC_OUT呢?我们得到的是什么效果?
如果你理解了上面效果示意图所展示的效果,我想这个问题应该不难。SRC_OUT模式就是原图去掉叠加的部分,在我们的例子中也就是四个圆角了。
其他的模式如果不太理解,大家可以按照上面的例子试试,看看实际的效果,可以加深理解。
Shader又被称之为着色器、渲染器,主要用来实现一系列的渐变、渲染效果。Android中一共有5种Shader,分别是BitmapShader(位图Shder)、LinearGradient(线性Shader)、RadialGradient(光束Shader)、SweepGradient(梯度Shader)、ComposeShader(混合Shader)
此外,Shader还提供了几种填充的模式
//Shader.class
public enum TileMode {
/**
* replicate the edge color if the shader draws outside of its
* original bounds
*/
CLAMP (0),
/**
* repeat the shader's image horizontally and vertically
*/
REPEAT (1),
/**
* repeat the shader's image horizontally and vertically, alternating
* mirror images so that adjacent images always seam
*/
MIRROR (2);
TileMode(int nativeInt) {
this.nativeInt = nativeInt;
}
final int nativeInt;
}
BitmapShader(位图Shder)跟其他4种Shader不一样,它产生的是一个图像,有点类似PS中的图像填充渐变。它的作用就是通过Paint对画布进行指定的Bitmap填充。
BitmapShader一种常用的情景就是生成圆形图片
原理就是把要绘制的图片利用BitmapShader填充到圆形的画布上
(1)初始化画布Canvas和画笔Paint
Bitmap bitmap = Bitmap.createBitmap(src.getWidth(), src.getHeight(), Bitmap.Config.ARGB_8888);
Canvas canvas = new Canvas(bitmap);
Paint paint = new Paint();
paint.setAntiAlias(true);
(2)利用Bitmap生成BitmapShader,模式为拉伸模式
BitmapShader shader = new BitmapShader(src, Shader.TileMode.CLAMP,Shader.TileMode.CLAMP);
(3)将BitmapShader设置给Paint,便于后面绘制
paint.setShader(shader);
(4)利用设置好的Paint,在圆形的画布上填充图片,也就是Canvas要调用drawCircle方法
//这里圆形的半径我们用图片边长的一半
float radius = Math.min(src.getWidth()/2, src.getHeight()/2);
//这里圆形的圆点我们设在原图的中间
canvas.drawCircle(src.getWidth()/2, src.getHeight()/2, radius, paint);
通过以上简单的几步,我们就获得了图片的圆形效果
图里面上面的图是我们画出来的圆形图,下面的是开源库CircleImageView实现的圆形图,可见效果是一样的,难道CircleImageView也是用这个原理实现的?让我们一探究竟。
public class CircleImageView extends ImageView {
...
@Override
protected void onDraw(Canvas canvas) {
if (mDisableCircularTransformation) {
super.onDraw(canvas);
return;
}
if (mBitmap == null) {
return;
}
if (mCircleBackgroundColor != Color.TRANSPARENT) {
canvas.drawCircle(mDrawableRect.centerX(), mDrawableRect.centerY(), mDrawableRadius, mCircleBackgroundPaint);
}
//这里就是绘制圆形图的地方
canvas.drawCircle(mDrawableRect.centerX(), mDrawableRect.centerY(), mDrawableRadius, mBitmapPaint);
if (mBorderWidth > 0) {
canvas.drawCircle(mBorderRect.centerX(), mBorderRect.centerY(), mBorderRadius, mBorderPaint);
}
}
private void setup() {
...
//找到了,可见所采用的方法也是BitmapShader
mBitmapShader = new BitmapShader(mBitmap, Shader.TileMode.CLAMP, Shader.TileMode.CLAMP);
mBitmapPaint.setAntiAlias(true);
mBitmapPaint.setShader(mBitmapShader);
...
}
...
}
从上面CircleImageView的源码中我们可以看出,CircleImageView也是通过BitmapShader来实现图片的圆形特效的。有兴趣的朋友可以找来研究一下。
其他的填充渐变效果从字面意思就能看出来,大家有兴趣可以自己写个例子看看效果。