目录
一、简介
1、为什么要有画布操作?
二、Canvas的基本操作
1、Translate(平移)
2、Scale(缩放)
3、Rotate(旋转)
4、skew(错切)
5、save(快照)和restore(回滚)
6、clipXXX(裁剪)
6.1、clipRect
6.2、clipPath
7、saveLayer(图层)
三、总结
Canvas类中的方法大致可以分为三类:
前面几篇介绍了drawXXX系列的方法,本篇继续分析scale、rotate、clipXXX 等对画布进行操作的方法。
画布操作可以帮助我们用更加容易理解的方式制作图形。
例如: 从坐标原点为起点,绘制一个长度为20dp,与水平线夹角为30度的线段怎么做?
按照我们通常的想法,就是先使用三角函数计算出线段结束点的坐标,然后调用drawLine即可。
然而利用对画布的操作可以用更加简单的方式实现。
假设我们先绘制一个长度为20dp的水平线,然后将这条水平线旋转30度,则最终看起来效果是相同的,而且不用进行三角函数计算。
合理的使用画布操作可以帮助你用更容易理解的方式创作你想要的效果,这也是画布操作存在的原因。
PS: 所有的画布操作都只影响后续的绘制,对之前已经绘制过的内容没有影响。
Canvas的操作一般分为四类:translate(平移)、scale(缩放)、rotate(旋转)、skew(错切)。
在分析之前先明确下几个基本概念:在Android中,默认的坐标零点位于屏幕左上角,向下为Y轴正方向,向右为X轴正方向,反之为负。
canvas类提供的方法:
public void translate(float dx, float dy)
参数说明:
先在屏幕画一个矩形:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
}
效果图:
在将Canvas进行平移操作后绘制同样坐标的矩形,为了对比效果重新设置画笔颜色:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
//向X轴方向移动200,向Y轴方向移动200
canvas.translate(200,200);
mPaint.setColor(Color.GRAY);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
}
效果:
然后在将Canvas进行同样的平移操作:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
//向X轴方向移动200,向Y轴方向移动200
canvas.translate(200,200);
mPaint.setColor(Color.GRAY);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
//向X轴方向移动200,向Y轴方向移动200
canvas.translate(200,200);
mPaint.setColor(Color.RED);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
}
效果:
从效果上看,两次translate 进行了叠加,绘制第二个矩形的时候画布已经偏移了(400,400);即:Translate是基于当前位置移动,而不是每次基于屏幕左上角的(0,0)点移动。
Canvas提供了两个方法进行scale:
public void scale (float sx, float sy)
public final void scale (float sx, float sy, float px, float py)
参数说明:
仔细观察会发现scale是以坐标原点为中心,沿着坐标轴方向进行的。如下:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
//以原点(0,0)为中心,x、y坐标分别缩小0.5
canvas.scale(0.5f,0.5f);
mPaint.setColor(Color.GRAY);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
}
效果图:
当(sx,sy)取值大于1时是按比例放大,a小于1时是按比例缩小;取值小于0时,则会翻转后进行缩放,如下:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
mBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.you2);
canvas.translate(300,300);
canvas.drawBitmap(mBitmap,0,0,null);
canvas.scale(-0.5f,-0.5f);
canvas.drawBitmap(mBitmap,0,0,null);
}
效果图:
下面看第二个方法:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
//以原点(800,800)为中心,x、y坐标分别缩小0.5
canvas.scale(0.5f,0.5f,800,800);
mPaint.setColor(Color.GRAY);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
}
效果图:
以上就是以红点(800,800)为原点的进行缩放,其实通过源码能看出它的实现原理:
/**
* Preconcat the current matrix with the specified scale.
*
* @param sx The amount to scale in X
* @param sy The amount to scale in Y
* @param px The x-coord for the pivot point (unchanged by the scale)
* @param py The y-coord for the pivot point (unchanged by the scale)
*/
public final void scale(float sx, float sy, float px, float py) {
if (sx == 1.0f && sy == 1.0f) return;
translate(px, py);
scale(sx, sy);
translate(-px, -py);
}
可以看出第二个方法是通过先将Canvas平移一段距离(px,py)出进行缩放后在平移相同的距离回来(-px,-py),但是由于叠加效果,实际上最后的平移的距离是经过缩放的。下面按照源码的原理实现下:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
canvas.drawPoint(800,800,mPaint);
canvas.translate(800,800);
canvas.scale(0.5f,0.5f);
canvas.translate(-800,-800);
mPaint.setColor(Color.GRAY);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
}
看下效果图,发现效果和scale(0.5f,0.5f,800,800)相同:
同样,Canvas提供了两个方法进行rotate:
public void rotate (float degrees)
public final void rotate (float degrees, float px, float py)
参数说明:
看下实现代码:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
mBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.you2);
canvas.drawBitmap(mBitmap,0,0,null);
canvas.rotate(30);
canvas.drawBitmap(mBitmap,0,0,null);
}
效果图:
第二个方法:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
mBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.you2);
canvas.drawBitmap(mBitmap,0,0,null);
canvas.drawPoint(500,500,mPaint);
canvas.rotate(180,500,500);
canvas.drawBitmap(mBitmap,0,0,null);
}
效果图:
同理,旋转的操作也是可以叠加的。
错切是特殊类型的线性变换。Canvas为skew提供了一个方法:
public void skew (float sx, float sy)
参数说明:
(x,y)坐标经过skew变换后:
看下实现代码:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
mBitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.you2);
canvas.drawBitmap(mBitmap,0,0,null);
canvas.skew(0.5f,0.5f);
canvas.drawBitmap(mBitmap,0,0,null);
}
效果图:
同理,错切也是可叠加的,不过请注意:调用次序不同绘制结果也会不同。
Q: 为什存在快照与回滚?
A:画布的操作是不可逆的叠加的,很多画布操作会影响后续的步骤。所以会对画布的一些状态进行保存和回滚。即:
注意:save和restore要配对使用(restore可以比save少,但不能多),如果restore调用次数比save多,会引发Error。save和restore之间,往往夹杂的是对Canvas的特殊操作。
如同上面Translate中的两个平移操作在没有使用save和restore的情况下是叠加的,效果图如下:
在代码中上save和restore操作,代码如下,效果图在上面右边。可以看出此时平移的操作并没有叠加,因为Canvas的状态被回滚到上次save的时候。
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
canvas.save();
//向X轴方向移动200,向Y轴方向移动200
canvas.translate(200,200);
mPaint.setColor(Color.GRAY);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
canvas.restore();
//向X轴方向移动200,向Y轴方向移动200
canvas.translate(200,200);
mPaint.setColor(Color.RED);
canvas.drawRect(new RectF(0,0,400,400), mPaint);
}
Canvas提供的save方法有两种,如下:
// 保存全部状态
public int save ()
// 根据saveFlags参数保存一部分状态
public int save (int saveFlags)
参数说明:
saveFlags取值和其意义如下:
FLAG | 意义 |
---|---|
ALL_SAVE_FLAG | 保存所有的标识 |
MATRIX_SAVE_FLAG | 仅保存canvas的matrix数组 |
CLIP_SAVE_FLAG | 仅保存canvas的当前大小 |
注意:save (int saveFlags)在官方文档中已经被标记过时不推荐使用了,用save()方法代替。
Canvas提供了三种裁剪的方式:
对于clipRect,Canvas提供了多个重载方法,如下:
public boolean clipRect (Rect rect)
public boolean clipRect (RectF rect)
public boolean clipRect (int left, int top, int right, int bottom)
public boolean clipRect (float left, float top, float right, float bottom)
public boolean clipRect (Rect rect, Region.Op op)
public boolean clipRect (RectF rect, Region.Op op)
public boolean clipRect (float left, float top, float right, float bottom, Region.Op op)
从上面这些重载方法可以看出,clipRect函数的参数主要是两个部分:
裁剪区域(Rect,或者传入的Rect的四个坐标):这部分参数很好理解,直接看代码。
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
canvas.drawColor(Color.GRAY);
canvas.clipRect(new RectF(100,100,400,400));
canvas.drawColor(Color.RED);
}
效果图:红色部分是裁剪后的区域。
Region.Op:从源码中可以看出其是一个枚举类,
public enum Op {
DIFFERENCE(0),
INTERSECT(1),
UNION(2),
XOR(3),
REVERSE_DIFFERENCE(4),
REPLACE(5);
Op(int nativeInt) {
this.nativeInt = nativeInt;
}
/**
* @hide
*/
public final int nativeInt;
}
Region是区域的意思。而Region.op里面的值其实表示就是传入的区域与当前区域的组合方式:
如图表示:
Canvas提供两个clipPath重载方法:
public boolean clipPath(Path path,Region.Op op)
public boolean clipPath(Path path)
上面介绍过Region.Op参数的意义,简单实现下:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
Path path = new Path();
path.addCircle(400, 400, 200, Path.Direction.CW);
canvas.clipPath(path);
canvas.drawColor(Color.RED);
}
效果图:
ps:clipPath有明显的锯齿,没有找到好的办法解决,所以使用时注意下。
saveLayer可以为canvas创建一个新的透明图层,在新的图层上绘制,并不会直接绘制到屏幕上,而会在restore之后,绘制到上一个图层或者屏幕上(如果没有上一个图层)。为什么会需要一个新的图层,例如在处理xfermode的时候,原canvas上的图(包括背景)会影响src和dst的合成,这个时候,使用一个新的透明图层是一个很好的选择。又例如需要当前绘制的图形都带有一定的透明度,那么创建一个带有透明度的图层,也是一个方便的选择。可以把这些图层看做是一层一层的玻璃板,在每层的玻璃板上绘制内容,然后把这些玻璃板叠在一起看就是最终效果。如图:
在canvas中提供了以下方法来实现图层操作:
public int saveLayer(RectF bounds, Paint paint)
public int saveLayer(RectF bounds, Paint paint, int saveFlags)
public int saveLayer(float left, float top, float right, float bottom, Paint paint)
public int saveLayer(float left, float top, float right, float bottom, Paint paint,int saveFlags)
//saveLayerAlpha可以开启一个带有透明度的图层
public int saveLayerAlpha(RectF bounds, int alpha)
public int saveLayerAlpha(RectF bounds, int alpha, int saveFlags)
public int saveLayerAlpha(float left, float top, float right, float bottom, int alpha)
public int saveLayerAlpha(float left, float top, float right, float bottom, int alpha, int saveFlags)
参数说明:
MATRIX_SAVE_FLAG,
CLIP_SAVE_FLAG,
HAS_ALPHA_LAYER_SAVE_FLAG,
FULL_COLOR_LAYER_SAVE_FLAG,
CLIP_TO_LAYER_SAVE_,
ALL_SAVE_FLAG。//保存全部
ps:在官方文档中表示saveFlags参数的六个值前面五个都被标记过时不推荐使用了,默认使用ALL_SAVE_FLAG。
代码:
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
mPaint.setColor(Color.BLUE);
canvas.drawCircle(100, 100, 100, mPaint);
mPaint.setColor(Color.RED);
canvas.saveLayerAlpha(100, 100, 200, 200, 120, Canvas.ALL_SAVE_FLAG);
canvas.drawCircle(200, 200, 100, mPaint);
canvas.restore();
}
效果图:
本文介绍一些操作画布的方法。合理的使用画布操作可以更容易实现想要的效果。
祝:工作顺利!
参考资料:
http://www.gcssloop.com/customview/Canvas_Convert
https://blog.csdn.net/tianjian4592/article/details/45234419
https://www.jianshu.com/p/afa06f716ca6