自定义View-绘制基础

HenCoder 原文

  • 关键点 自定义绘制方法的重写,其中最常用的是onDraw
  • 绘制的关键是Canvas的使用
    • Canvas的绘制类方法: drawXXX() (关键参数:Paint)
    • Canvas的辅助类方法: 范围裁切和集合变换
  • 控制绘制的前后顺序来控制遮盖关系。

Canvas.drawXXX() 和 Paint基础

  1. Paint类的几个最常用方法:

    • Paint.setStyle(Style style) 设置绘制模式:空心实心
      • Paint.Style.STROKE 划线模式 空心
      • Paint.Style.FILL 填充模式 实心
      • Paint.Style.FILL_AND_STROKE 既划线又填充
    • Paint.setColor(int color) 设置颜色
    • Paint.setStrokeWidth(float width) 设置线条宽度
    • Paint.setTextSize(float textSize) 设置字体大小
    • Paint.setAntiAlias(boolean open) 是否抗锯齿
      • 抗锯齿只是将边缘像素模糊化,避免由于色值差异过大,导致边缘过于尖锐。
  2. drawColor(@ColorInt int color) 用于绘制之前设置底色 或者绘制之后设置蒙版

    • drawColor(Color.BLACK) 纯黑色
    • drawColor(Color.parse("#88880000") 半透明浅红色
    • drawRGB(100,200,100)
  3. drawCircle(float centerX,float centY,float radius,Paint paint) 画圆

    • centerX,centerY 圆心点 radius 圆半径 paint 绘制的画笔

    • 在Android里,每一个View都有自己的坐标系。一个View的(x,y)指的是离他左上角x像素的位置,数值方向y像素的位置。

    • 画一个红色圆

      paint.setColor(Color.RED)
      canvas.drawCircle(300,300,200,paint) 先设置画笔颜色,再绘制
      
    • 画一个空心圆

      paint.setStyle(Paint.Style.STROKE) 
      canvas.drawCircle(300,300,200,paint) 设置画笔为划线模式
      
    • 画一个圆环

       paint.setStyle(Paint.Style.STROKE) 
       // 或者 paint.setStyle(Paint.Style.FILL_AND_STROKE)
       paint.setStrokeWidth(20) //设置线条宽度20
       canvas.drawCircle(300,300,200,paint) 
      
  4. drawRect(float left,float top, float right, float bottom, Paint paint) 绘制矩形

    • left , top 矩形左上角坐标 right,bottom 矩形右下角坐标

    • drawRect(RectF rectF,Paint paint),drawRect(Rect rect,Paint paint) 重载实现

    • 画有填充的矩形

      paint.setStyle(Style.FILL);
      canvas.drawRect(100,100,500,500,paint);
      
    • 画矩形边

      paint.setStyle(Style.STROKE);
      canvas.drawRect(100,100,500,500,paint);
      
  5. drawPoint(float x,float y,Paint paint) 画点

    • x,y点的坐标

    • 点的大小通过设置线条粗细来设置

    • 点的形状(方形或者圆形)可以通过paint.setStrokeCap(cap),ROUND 圆形,SQUARE 或者BUTT 方形 (Cap : 罩,盖)

    • 圆形的点

      paint.setStrokeWidth(20) ;//点的粗细
      paint.setStrokeCap(Paint.Cap.ROUND); // 点的形状
      canvas.drawPoint(50,50,paint);
      
    • 方形的点

      paint.setStrokeWidth(20);
      paint.setStrokeCap(Paint.Cap.SQUARE); 
      canvas.drawPoint(50,50,paint);
      
  6. drawOval(float left, float top, float right, float bottom , Paint paint) 话椭圆

    • left,top,right,bottom 椭圆左上、右下界点坐标

    • 实心椭圆

      paint.setStyle(Style.FILL);
      canvas.drawOval(50,50,350,200,paint); 
      
    • 空心椭圆

      paint.setStyle(Style.STROKE);
      canvas.drawOval(50,50,350,200,paint);      
      
  7. drawLine(float startX, float startY, float stopX, float stopY) 绘制直线

    • startX, startY , stopX, stopY 起点、终点坐标

    • 直线不是封闭图形,所以setStyle对其没有影响

      canvas.drawLine(200,200,500,500,paint);
      
  8. drawRoundRect(float left, float top, float right, float bottom,float rx, float ry,Paint paint) 绘制圆角矩形

    • left,top,right,bottom 矩形边界坐标,rx,ry 圆角的横向和纵向半径

      canvas.drawRoundRect(100,100,500,300,50,50,paint);
      
  9. drawArc(float left, float top, float right, float bottom, float startAngle,float sweepAngle, boolean useCenter, Paint paint) 绘制弧形或者扇形

    • 使用绘制椭圆来绘制。left,top,right,bottom 椭圆的边界点,

    • startAngle 弧形的起始角度, sweepAngle 弧形的划过角度, x 轴的正向,即正右的方向,是 0 度的位置;顺时针为正角度,逆时针为负角度

    • useCenter 是否连接“圆心”,不连接“圆心”,就是弧形,连接“圆心”,就是扇形。

    • 起始角度,旋转的方向,是否连接圆心画弧3要素

       paint.setStyle(Style.FILL); // 填充模式
       canvas.drawArc(200,100,800,500,-110,100,true,paint); //连接圆心,扇形(连接圆弧和圆形的图像)
       canvas.drawArc(200,100,800,500,20,140,flase,paint);// 不连接圆心,但填充模式会生成封闭区域,会连接起点和终点,所以成填充的“弧形”
       paint.setStyle(Style.STROKE); // 划线模式
       canvas.drawArc(200,100,800,500,180,60,false,paint); // 绘制弧形,“只有圆弧”
      
      弧形
  10. drawPath(path) 路径绘制

  • 描述路径
    • addXXX() 添加子图形

    • addCircle(float x, float y, float radius, Direction dir)

      • x,y,radius 圆的圆心坐标和圆的半径

      • dir 顺时针(CW: Clockwise)还是逆时针(counter-clockwise) 这个在图形自相交的情况下会影响填充的判断

        path.addCircle(300,300,200,Path.Direction.CW);
           ...
        canvas.drawPath(path,paint);
        
      • 如果只需要单纯的画一个圆,那么直接用cavas.drawCirle()就可以实现,没必要搞成这么复杂

    • addOval(float left , float top, float right, float bottom, Direction dir) / addOval(RectF oval,Direction dir) 添加椭圆

    • addRect(float left , float top, float right, float bottom, Direction dir) / addRect(RectF rect, Direction dir) 添加椭圆

    • addRoundRect(RectF rectf, float rx, float ry,Direction dir) / addRoundRect(float left , float top, float right, float bottom, float rx, float ry,Direction dir) 添加圆角矩形

    • addPath() 添加另外一个path

      • xxxTo() 画线
        • lineTo(float x, float y) / rLineTo(float x, float y) 从当前位置画直线到x,y位置

          • r(relative) 相对坐标

            paint.setStyle(Style.STROKE) ; //设置画线模式
            path.lineTo(100,100);
            canvas.drawPath(path,paint);
            
        • quardTo(float x1, float y1,float x2 ,float y2) / rQuardTo(float x1, float y1 , float x2, float y2) 二次贝塞尔曲线

        • cubicTO(float x1, float y1,float x2 ,float y2, float x3, float y3) / rCubicTO(float x1, float y1,float x2 ,float y2, float x3, float y3) 三次贝塞尔曲线

        • moveTo(float x, float y) 从当前停笔位置,移动画笔到指定位置,然后再开始绘制

        • arcTo(RectF oval, float startAngle, float sweepAngle, boolen forceMoveTo) / arcTo(float left, float top, float right , float bottom , float startAngle, float sweeoAngle) / arcTo(RectF oval, float startAngle, float sweepAngle) 画弧形

          • 比之drawArc()方法,缺少了userCenter,因为此处是明确画弧形,而userCenter为trues时绘制的是扇形。 forceMoveTo 表示是从当前位置直接跳到指定位置开始绘制弧形还是 从当前位置画一条直线到指定位置,然后再开始绘制弧形,即是否有拖动的痕迹。
          • addArc()是使用forceMoveTo = true的简化版arcTo()
      • close() 直线连接起底而最终的终点,封闭子图形
    • Path.setFillType(fillType) 辅助计算 -- 自相交时的填充绘制

      • FillType 四种取值

        • EVEN_ODD
        • WINDING (默认值)
        • INVERSE_EVEN_ODD (第1种反模式)
        • INVERSE_WINDING (第2种反模式)
      • EVEN_ODD -- even-odd rule 奇偶原则

        • 区域内的任意一点向任意方向发射射线,如果与图形相交的次数为奇数,则认为区域在图形的内部,区域被填充颜色;如果为偶数,则为图形内部,则不填充颜色。

          奇偶原则
      • WINDING 缠绕原则

        • 每个图形都有其绘制的方向(CW 顺时针,CCW 逆时针)
        • 区域内的任意一点向任意方向发射射线,起始交叉数为0,如果与顺时针方向的线条相交则交叉数增加1,如果与逆时针方向的线条相交则交叉数减去1,最终的交叉数如果不是0,那么区域在图形的内部,区域被填充颜色,如果为0,区域在图形的外部,区域不填充颜色。
        缠绕原则

        对比
  1. drawBitmap(Bitmap bitmap, float left, float top, paint() 画bitmap

    • 把bitmap图形绘制到left,top位置。
    • 重载方法
      • drawBitmap(Bitmap bitmap, Rect src, RectF dst, Paint paint)
      • drawBitmap(Bitmap bitmap, Rect src, Rect dst, Paint paint)
      • drawBitmap(Bitmap bitmap, Matrix matrix, Paint paint)
  2. drawText(String text, float x,float y, Paint paint) 绘制文字

    • text 需要绘制的文字, x,y 绘制的起始点坐标,文字是在起始点坐标之上进行绘制
    • paint.setTextSize() 设置绘制的文打字大小
    • paint.setStyle(Style.FILL) 设置填充样式

HenCoder 原文

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,132评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,802评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,566评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,858评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,867评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,695评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,064评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,705评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,915评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,677评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,796评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,432评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,041评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,992评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,223评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,185评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,535评论 2 343