美文网首页
Canvas中drawxxx方法的使用

Canvas中drawxxx方法的使用

作者: 夏广成 | 来源:发表于2017-06-26 11:20 被阅读51次

一:Canvas.drawxxx()系列方法以及Paint常见使用

1:移动坐标系

canvas.translate(width/2,height/2); //可以将坐标系从左上角移动到指定位置
cavas.scale(1,-1);//反转y坐标轴,反转之后向上为正,向下为负。

2:drawRect(float left, float top, float right, float bottom, Paint paint)

画矩形传入的是左上点和右下点的坐标。drawRect(左上x,左上y,右下x,右下y,paint);

3:drawPoint(float x, float y, Paint paint)

在画点的时候,需要paint.setStrokeWidth();不然点看不到。

(1):画多个点

 float [] pts={30,100,100,100,170,100,240,100,310,100};
private void init() {
    paint=new Paint(Paint.ANTI_ALIAS_FLAG);
    paint.setColor(Color.parseColor("#2bb8ed"));
    paint.setStrokeWidth(50);
    paint.setStrokeCap(Paint.Cap.ROUND);
}

@Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    canvas.drawPoints(pts,paint);
//        canvas.drawPoints(pts,2,6,paint);
}

(2):画多个点同时偏移

 float [] pts={30,100,100,100,170,100,240,100,310,100};
    private void init() {
        paint=new Paint(Paint.ANTI_ALIAS_FLAG);
        paint.setColor(Color.parseColor("#2bb8ed"));
        paint.setStrokeWidth(50);
        paint.setStrokeCap(Paint.Cap.ROUND);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
//        canvas.drawPoints(pts,paint);
        canvas.drawPoints(pts,2,6,paint);
    }

使用的数组还是之前的数组,就是在drawPoints的参数上做了调整。该方法第二个参数表示从第几个数组元素开始绘制,由于数组的元素是两个点构成一个坐标,因此偏移两个位置,即一个点。第三个参数表示一共使用几个数组元素,在本段代码中最后将原始数组进行了截取,相当于使用了下面这个新数组进行绘制。

 float [] pts={100,100,170,100,240,100};

如果我们将参数调整为

canvas.drawPoints(pts,1,7,paint);

将会显示如下,虽然取了7个元素,但是最后一个元素没有配对的y坐标,无法绘制出来。


4:drawOval(float left, float top, float right, float bottom, @NonNull Paint paint);

前两个参数可以认为是椭圆外切矩形的左上点坐标。
接下来的两个参数可以认为是椭圆外切矩形的右下点坐标。

5:画线drawLine(float startX, float startY, float stopX, float stopY,@NonNull Paint paint)

我们通过代码画了三条线,分别设置了不同的Cap,三条线除了颜色不一样和cap不一样之外,其他的都一致。

 private void init() {
    paint=new Paint(Paint.ANTI_ALIAS_FLAG);
    paint.setColor(Color.RED);
    paint.setStrokeWidth(50);
    paint.setStrokeCap(Paint.Cap.BUTT);

    paint1=new Paint(Paint.ANTI_ALIAS_FLAG);
    paint1.setColor(Color.GRAY);
    paint1.setStrokeWidth(50);
    paint1.setStrokeCap(Paint.Cap.ROUND);

    paint2=new Paint(Paint.ANTI_ALIAS_FLAG);
    paint2.setColor(Color.BLUE);
    paint2.setStrokeWidth(50);
    paint2.setStrokeCap(Paint.Cap.SQUARE);
}
@Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    canvas.drawLine(30,30,30,630,paint);
    canvas.drawLine(130,30,130,630,paint1);
    canvas.drawLine(230,30,230,630,paint2);
    }

但是从效果图中,我们看到的三条线的长度好像并不一致。从代码中我们知道我们设置的起始点y坐标是30,但是只有第一条线满足了,后面两条线不满足,长度为600,也只有第一条满足了,后面两条未满足。因为cap是指当画笔画完之后,再给任务画一个帽子,这个帽子是多出来的。

和cap类似的还有一个join

public enum Join {  
    /** 
     * The outer edges of a join meet at a sharp angle 
     */  
    MITER   (0),  
    /** 
     * The outer edges of a join meet in a circular arc. 
     */  
    ROUND   (1),  
    /** 
     * The outer edges of a join meet with a straight line 
     */  
    BEVEL   (2);  
      
    private Join(int nativeInt) {  
        this.nativeInt = nativeInt;  
    }  
    final int nativeInt;  
}
图片来自gcsSloop

6:drawRoundRect(float left, float top, float right, float bottom, float rx, float ry, Paint paint)

圆角矩形

7:drawArc(float left, float top, float right, float bottom, float startAngle, float sweepAngle, boolean useCenter, Paint paint)

扇形或者弧形或者弧线。是否填充全在于paint.setStyle(Paint.Style.);

三点钟方向为0度。

顺时针为正方向。

useCenter表示在两边是否用线连接到圆心。如果连接就是弧形,不连接就是扇形。

 private void init() {
    paint=new Paint(Paint.ANTI_ALIAS_FLAG);
    paint.setColor(Color.RED);
    paint.setStrokeWidth(10);
    paint.setStrokeCap(Paint.Cap.BUTT);
    paint.setStyle(Paint.Style.STROKE);

    paint1=new Paint(Paint.ANTI_ALIAS_FLAG);
    paint1.setColor(Color.GRAY);
    paint1.setStrokeWidth(50);
    paint1.setStrokeCap(Paint.Cap.ROUND);

    paint2=new Paint(Paint.ANTI_ALIAS_FLAG);
    paint2.setColor(Color.BLUE);
    paint2.setStrokeWidth(10);
    paint2.setStrokeCap(Paint.Cap.SQUARE);
}
@Override
protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);
    canvas.drawArc(0, 0, 200, 200, 50, 100, false, paint); 
    canvas.drawArc(0, 200, 300, 400, -360, 100, true, paint); 
    canvas.drawArc(0, 500, 400, 700, -150, 100, false, paint1); 
    canvas.drawArc(400, 500, 500, 600, -150, 100, true, paint1); 

    }

8:drawPath

path有两种方法,一类是直接描述路径,一类是辅助设置或计算

(1):addCircle/addOval/addArc/addRect/addRoundRect。

最后一个参数dir表示方向,有两个选项,一个是cr顺时针,一个是ccr逆时针。

lineTo/rLineTo 前者是绝对坐标。后者是相对坐标。都是画直线。
使用path画粗线的时候,比如直方图的粗线,一定要把style设置为stroke。

画线的方向就是方向。

quadTo/rQuadTo 画二次贝塞尔曲线。


二阶贝塞尔曲线图片来自网络

cubicTo/rCubicTo 画三次贝塞尔曲线


三阶贝塞尔曲线图片来自网络 四阶贝塞尔曲线图片来自网络 五阶贝塞尔曲线图片来自网络

moveTo/rMoveTo 移动到目标位置开始画起点

moveTo :移动到下一次操作的起点位置
setLastPoint 设置之前操作的最后一个点位置。这会取消之前的终点位置。如图所示:

图片来自gcsSloop
那么在3d旋转的时候,能不能通过setLastPoint来旋转一个点达到效果呢?
重点
arcTo 只画弧形不画扇形。forceMoveTo参数表示是否留下画笔的移动痕迹。

addArc()表示forceMoveTo=true的简化版arcTo。

close()封闭当前图形。终点到起点的连线。当画笔为fill的时候,会自动封闭。

在画心形的时候,不要抬笔,也就是不要将forceMoveTo设置为true,因为true就以为着是一个新的If true, always begin a new contour with the arc。那么在封闭的时候,只能封闭当前的contour,而不能封闭之前的contour了。

 path.arcTo(200,100,400,300,150, 210,false);
 path.arcTo(400,100,600,300,180, 210,false);
 path.lineTo(400, 442);

(2):辅助设置或计算 Path.setFillType(Path.FillType ft)


图片来自hencoder

参考:
gcsSloop的博客
Hencoder的博客

相关文章

网友评论

      本文标题:Canvas中drawxxx方法的使用

      本文链接:https://www.haomeiwen.com/subject/urwdcxtx.html