Flutter之简单绘图

Flutter中如果想要自定义绘制,那么你需要用到 CustomPaint 和 CustomPainter ; CustomPaint是Widget的子类。

const CustomPaint({
    Key key,
    this.painter,
    this.foregroundPainter,
    this.size = Size.zero,
    this.isComplex = false,
    this.willChange = false,
    Widget child,
  }) :super(key: key, child: child);

示例

import 'package:flutter/material.dart';

void main() => runApp(MyApp());

class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Flutter Demo',
      theme: ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: MyHomePage(title: 'Painter绘制直线'),
    );
  }
}

class MyHomePage extends StatefulWidget {
  MyHomePage({Key key, this.title}) : super(key: key);
  final String title;
  @override
  _MyHomePageState createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(widget.title),
      ),
      body: Center(
        child: CustomPaint(
          size: Size(300, 300),
          painter:MyPainter() ,
        ),
      )
    );
  }
}

class MyPainter extends CustomPainter {
  ///[定义画笔]
  Paint _paint = Paint()
  ..color = Colors.blueAccent //画笔颜色
  ..strokeCap = StrokeCap.round//画笔笔触类型
  ..isAntiAlias = true //是否启动抗锯齿
  ..style = PaintingStyle.fill //绘画风格,默认为填充
  ..strokeWidth = 5.0; //画笔的宽度

  @override
  void paint(Canvas canvas, Size size) {
     canvas.drawLine(Offset(20, 20), Offset(100,100), _paint);
  }

  @override
  bool shouldRepaint(CustomPainter oldDelegate) {
    return null;
  }
}

定义画笔

///[定义画笔]
  Paint _paint = Paint()
  ..color = Colors.blueAccent //画笔颜色
  ..strokeCap = StrokeCap.round//画笔笔触类型
  ..isAntiAlias = true //是否启动抗锯齿
  ..style = PaintingStyle.fill //绘画风格,默认为填充
  ..strokeWidth = 5.0; //画笔的宽度

绘制直线(drawLine)

使用给定的涂料在给定点之间绘制一条线。 该行被描边,此调用忽略[Paint.style]的值。p1和p2参数为两个点的坐标 , 在这两点之间绘制一条直线。

class MyPainter extends CustomPainter {
///[定义画笔]
Paint _paint = Paint()
..color = Colors.blueAccent //画笔颜色
..strokeCap = StrokeCap.round//画笔笔触类型
..isAntiAlias = true //是否启动抗锯齿
..style = PaintingStyle.fill //绘画风格,默认为填充
..strokeWidth = 5.0; //画笔的宽度

@override
void paint(Canvas canvas, Size size) {
 canvas.drawLine(Offset(20, 20), Offset(100,100), _paint);
}

@override
bool shouldRepaint(CustomPainter oldDelegate) {
return null;
}
}
image.png

绘制点(drawPoints)

绘制点也是非常的简单,3个参数分别为: PointMode枚举,坐标 list 和 paint。PointMode的枚举类型有三个,points(点),lines(线,隔点连接),polygon(线,相邻连接)

void drawPoints(PointMode pointMode, List points, Paint paint)
class MyPainter extends CustomPainter {
  
  ///[定义画笔]
  Paint _paint = Paint()
  ..color = Colors.blueAccent //画笔颜色
  ..strokeCap = StrokeCap.round//画笔笔触类型
  ..isAntiAlias = true //是否启动抗锯齿
  ..style = PaintingStyle.fill //绘画风格,默认为填充
  ..strokeWidth = 5.0; //画笔的宽度

  @override
  void paint(Canvas canvas, Size size) {
    ///PointMode的枚举类型有三个,points(点),lines(线,隔点连接),polygon(线,相邻连接)
     canvas.drawPoints(
       PointMode.points,
       [
         Offset(20.0,  40.0),
         Offset(100.0, 120.0),
         Offset(100.0, 220.0),
         Offset(200.0, 220.0),
         Offset(200.0, 120.0),
         Offset(280.0, 40.0),
         Offset(20,    40.0),
       ], 
       _paint
       );
  }

  @override
  bool shouldRepaint(CustomPainter oldDelegate) {
    return null;
  }
}

PointMode改为points

image.png

PointMode改为polygon, PointMode改为polygon,相邻点互相连接
image.png

PointMode改为lines。 PointMode为lines时,两个点相互连接,也就是说第一个和第二个点连接,第三个跟第四个连接,如果最后只有一个点就舍弃不连接了,在我们的例子中有7个点,所以图中只有三条连线。

image.png

绘制圆drawCircle

参数分别为:圆心的坐标、半径和paint即可。圆形是否填充或描边(或两者)由Paint.style控制。

void drawCircle(Offset c, double radius, Paint paint)
class MyPainter extends CustomPainter {
  
  ///[定义画笔]
  Paint _paint = Paint()
  ..color = Colors.blueAccent //画笔颜色
  ..strokeCap = StrokeCap.round//画笔笔触类型
  ..isAntiAlias = true //是否启动抗锯齿
  ..style = PaintingStyle.stroke //绘画风格,默认为不填充
  ..strokeWidth = 5.0; //画笔的宽度

  @override
  void paint(Canvas canvas, Size size) {
    //绘制圆 参数(圆心,半径,画笔)
    canvas.drawCircle(Offset(140,80), 80,_paint..color = Colors.green);
  }

  @override
  bool shouldRepaint(CustomPainter oldDelegate) {
    return null;
  }
}
image.png

PaintStyle.fill 填充

image.png

绘制椭圆drawOval

制一个轴对称的椭圆形,参数为一个矩形和画笔paint.

void drawOval(Rect rect, Paint paint)
class MyPainter extends CustomPainter {
  
  ///[定义画笔]
  Paint _paint = Paint()
  ..color = Colors.blueAccent //画笔颜色
  ..strokeCap = StrokeCap.round//画笔笔触类型
  ..isAntiAlias = true //是否启动抗锯齿
  ..style = PaintingStyle.fill //绘画风格,默认为填充
  ..strokeWidth = 5.0; //画笔的宽度

  @override
  void paint(Canvas canvas, Size size) {
    //使用左上和右下角坐标来确定矩形的大小和位置,椭圆是在这个矩形之中内切的
    Rect rect = Rect.fromPoints(Offset(100.0, 40.0), Offset(220.0, 100.0));
    canvas.drawOval(rect, _paint..color=Colors.green);
  }
}

Rect也有多种构建方式: fromLTWH(double left, double top, double width, double height) 使用矩形左边的X坐标、矩形顶部的Y坐标矩形的宽高来确定矩形的大小和位置

@override
  void paint(Canvas canvas, Size size) {
    //使用矩形左边的X坐标、矩形顶部的Y坐标矩形的宽高来确定矩形的大小和位置
    Rect rect = Rect.fromLTWH(100, 20, 100, 60);
    canvas.drawOval(rect, _paint..color=Colors.green);
  }

image.png

fromLTRB(double left, double top, double right, double bottom) 使用矩形左边的X坐标、矩形顶部的Y坐标、矩形右边的X坐标、矩形底部的Y坐标来确定矩形的大小和位置

@override
  void paint(Canvas canvas, Size size) {
    //使用矩形左边的X坐标、矩形顶部的Y坐标、矩形右边的X坐标、矩形底部的Y坐标来确定矩形的大小和位置
    Rect rect = Rect.fromLTRB(30, 30, 160, 140);
    canvas.drawOval(rect, _paint..color=Colors.green);
  }
image.png

fromCircle({ Offset center, double radius }) 使用圆的圆心点坐标和半径和确定外切矩形的大小和位置

@override
  void paint(Canvas canvas, Size size) {
    //使用圆的圆心点坐标和半径和确定外切矩形的大小和位置
    Rect rect = Rect.fromCircle(center: Offset(100, 40),radius: 60);
    canvas.drawOval(rect, _paint..color=Colors.green);
  }
image.png

绘制圆弧drawArc

首先还是需要Rect来确认圆弧的位置,还需要开始的弧度、结束的弧度、是否使用中心点绘制(圆弧是否向中心闭合)、以及paint.

void drawArc(Rect rect, double startAngle, double sweepAngle, bool useCenter, Paint paint)
class MyPainter extends CustomPainter {
  
  ///[定义画笔]
  Paint _paint = Paint()
  ..color = Colors.blueAccent //画笔颜色
  ..strokeCap = StrokeCap.round//画笔笔触类型
  ..isAntiAlias = true //是否启动抗锯齿
  ..style = PaintingStyle.fill //绘画风格,默认为填充
  ..strokeWidth = 5.0; //画笔的宽度

  @override
  void paint(Canvas canvas, Size size) {
    // Rect来确认圆弧的位置,还需要开始的弧度、结束的弧度、是否使用中心点绘制、以及paint弧度
    Rect rect = Rect.fromCircle(center: Offset(100, 40),radius:80);
    canvas.drawArc(rect,0.0,0.8,false,_paint);
  }

  @override
  bool shouldRepaint(CustomPainter oldDelegate) {
    return null;
  }
}

圆弧向中心点闭合 将useCenter改成true,圆弧向中心点闭合了

@override
  void paint(Canvas canvas, Size size) {
    // Rect来确认圆弧的位置,还需要开始的弧度、结束的弧度、是否使用中心点绘制、以及paint弧度
    const PI = 3.1415926;
    Rect rect2 = Rect.fromCircle(center: Offset(100.0, 50.0), radius: 80.0);
    canvas.drawArc(rect2, 0.0, PI / 2, true, _paint);
  }
image.png

绘制圆角矩形drawDRRect

class MyPainter extends CustomPainter {
  
  ///[定义画笔]
  Paint _paint = Paint()
  ..color = Colors.blueAccent //画笔颜色
  ..strokeCap = StrokeCap.round//画笔笔触类型
  ..isAntiAlias = true //是否启动抗锯齿
  ..style = PaintingStyle.stroke //绘画风格,默认为填充
  ..strokeWidth = 5.0; //画笔的宽度

  @override
  void paint(Canvas canvas, Size size) {
     //用Rect构建一个边长50,中心点坐标为100,100的矩形
    Rect rect = Rect.fromCircle(center: Offset(140.0, 50.0), radius: 50.0);
    //根据上面的矩形,构建一个圆角矩形
    RRect rrect = RRect.fromRectAndRadius(rect, Radius.circular(20.0));
    canvas.drawRRect(rrect, _paint);
  }

  @override
  bool shouldRepaint(CustomPainter oldDelegate) {
    return null;
  }
}
image.png

绘制路径drawPath

绘制路径,首先需要一个要绘制的路径path,然后就是这个paint了。

方法名            作用
moveTo    将路径起始点移动到指定的位置
relativeMoveTo    相对于当前位置移动到
lineTo    从当前位置连接指定点
relativeLineTo    相对当前位置连接到
arcTo    曲线
conicTo    贝塞尔曲线
add**    添加其他图形,如addArc,在路径是添加圆弧
contains    路径上是否包括某点
transfor    给路径做matrix4变换
combine    结合两个路径
close    关闭路径,连接路径的起始点
reset    重置路径,恢复到默认状态
class MyPainter extends CustomPainter {
  
  ///[定义画笔]
  Paint _paint = Paint()
  ..color = Colors.blueAccent //画笔颜色
  ..strokeCap = StrokeCap.round//画笔笔触类型
  ..isAntiAlias = true //是否启动抗锯齿
  ..style = PaintingStyle.stroke //绘画风格,默认为填充
  ..strokeWidth = 5.0; //画笔的宽度

  @override
  void paint(Canvas canvas, Size size) {
      
   //新建了一个path,然后将路径起始点移动到坐标(100,100)的位置
    Path path = new Path()..moveTo(100.0, 100.0);
    path.lineTo(200.0, 200.0);
    canvas.drawPath(path, _paint);
  }

  @override
  bool shouldRepaint(CustomPainter oldDelegate) {
    return null;
  }
}

绘制多个路径

@override
  void paint(Canvas canvas, Size size) {
      
   Path path = new Path()..moveTo(100.0, 100.0);

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

推荐阅读更多精彩内容