Flutter笔记:手写一个简单的画板工具

这篇具有很好参考价值的文章主要介绍了Flutter笔记:手写一个简单的画板工具。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

Flutter笔记
手写一个简单的画板工具

作者李俊才 (jcLee95):https://blog.csdn.net/qq_28550263
邮箱 :291148484@163.com
本文地址:https://blog.csdn.net/qq_28550263/article/details/133418742



1. 任务介绍

在本文中,我们将一起开发一个基本的Flutter画板应用,用户可以在画板上自由绘制,选择不同的颜色来绘制线条。这个画板应用将允许用户通过点击颜色选择按钮来选择画笔的颜色,并提供鼠标光标支持以增强用户体验。

任务要求

  1. 创建一个Flutter应用,包含一个画板界面,初始时,画板上没有任何绘制内容。
  2. 实现颜色选择功能,用户可以点击应用栏中的颜色选择按钮,弹出颜色选择对话框,选择绘制颜色。
  3. 支持自由绘制功能,用户可以使用鼠标或触摸屏在画板上自由绘制线条。绘制时,使用所选的颜色。
  4. 当用户在画板上绘制时,应实时显示他们的绘制内容。
  5. 用户可以使用鼠标光标,绘制不同颜色的线条。

效果预览

Flutter笔记:手写一个简单的画板工具,Dart语言与Flutter框架开发笔记,前端、桌面端、移动端、UI、构建工具,Flutter,Dart,画板,绘图

预期收获

任务完成后,你将具备以下技能:

  • Flutter应用开发基础知识。
  • 实现用户界面中的颜色选择和绘制功能。
  • 处理用户输入和手势操作。
  • 自定义绘制,使用自定义绘制器和Canvas API。

这个任务将有助于您深入了解Flutter应用的开发,以及如何实现一个具有基本绘图功能的用户界面。

2. 知识点准备

本文将使用以下知识点,最终实现

  1. MouseRegion: MouseRegion 用于捕获鼠标事件,允许您指定鼠标在其内部时的行为,包括设置鼠标光标的外观(在这里使用了 SystemMouseCursors.click)。

  2. SystemMouseCursors: SystemMouseCursors 是一个用于指定不同系统鼠标光标样式的类。在这里,我们使用了 SystemMouseCursors.click,将鼠标光标设置为点击样式,以提供用户视觉反馈。

  3. GestureDetector: GestureDetector 用于捕获用户手势事件,如拖动手势(onPanUpdateonPanEnd)。它允许您检测用户的绘制动作并触发相应的回调。

  4. CustomPaint: CustomPaint 是一个用于自定义绘制的容器,它接受一个自定义的 CustomPainter 对象。在这里,它用于呈现用户的绘制。

  5. CustomPainter: CustomPainter 是一个抽象类,它包含了用于自定义绘制的方法。在这里,我们创建了 MyPainter 类来实现绘制用户绘制的笔画。

  6. RenderBox: RenderBox 是一个用于获取渲染对象的边界框和坐标的类。在这里,我们使用它来获取鼠标事件的本地坐标,并将其转换为相对于 CustomPaint 的坐标。

  7. Canvas: Canvas 是一个用于绘制 2D 图形的画布。我们使用它在 MyPainter 中绘制用户的笔画。

  8. PointModePointMode 类用于指定如何绘制点的枚举类。在画板应用中,它有两种可能的值:

    • PointMode.points: 这个模式用于绘制单个点。当用户在画板上点击并松开鼠标时,将使用此模式来绘制点,以实现单击绘制的效果。
    • PointMode.polygon: 这个模式用于绘制连接的线条。当用户在画板上拖动鼠标时,将使用此模式来绘制连接的线条,以实现绘制笔画的效果。

    PointMode 通过 canvas.drawPoints 方法在 CustomPainter 中使用,用于指定如何绘制用户的绘制。对于单击,我们使用 PointMode.points 来绘制点,而对于笔画,我们使用 PointMode.polygon 来绘制连接的线条。
    该类包含在dart:ui库中。

  9. flutter_colorpicker 模块:这是一个选择颜色的第三方模块,你可以参考其官方文档

3. 代码实现与效果

import 'dart:ui';
import 'package:flutter/material.dart';
import 'package:flutter_colorpicker/flutter_colorpicker.dart';

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

class MyApp extends StatelessWidget {
  const MyApp({super.key});

  
  Widget build(BuildContext context) {
    return const MaterialApp(
      home: DrawingBoard(),
    );
  }
}

/// 画板应用的主页 [StatefulWidget]。
class DrawingBoard extends StatefulWidget {
  const DrawingBoard({super.key});

  
  State<DrawingBoard> createState() => _DrawingBoardState();
}

/// [DrawingBoard] 的状态类,负责管理用户绘制的功能和界面。
class _DrawingBoardState extends State<DrawingBoard> {
  List<DrawObject> drawObjects = [];
  Color selectedColor = Colors.black;
  bool isFirstDraw = true;

  /// 显示颜色选择对话框的方法。
  void selectColor() {
    showDialog(
      context: context,
      builder: (BuildContext context) {
        Color newColor = selectedColor;
        return AlertDialog(
          title: const Text('选择颜色'),
          content: SingleChildScrollView(
            child: ColorPicker(
              pickerColor: selectedColor,
              onColorChanged: (color) {
                newColor = color;
              },
            ),
          ),
          actions: <Widget>[
            TextButton(
              onPressed: () {
                setState(() {
                  selectedColor = newColor;
                });
                Navigator.of(context).pop();
              },
              child: const Text('确定'),
            ),
          ],
        );
      },
    );
  }

  
  Widget build(BuildContext context) {
    // 创建一个鼠标光标,使用Icons.draw_outlined图标
    const mouseCursor = SystemMouseCursors.click;

    return Scaffold(
      appBar: AppBar(
        title: const Text('Jack Lee 的画板'),
        actions: [
          IconButton(
            icon: const Icon(Icons.color_lens),
            onPressed: () {
              selectColor();
            },
          ),
        ],
      ),
      body: MouseRegion(
        cursor: mouseCursor, // 使用自定义光标
        child: GestureDetector(
          onPanUpdate: (details) {
            setState(() {
              RenderBox renderBox = context.findRenderObject() as RenderBox;
              final offset = renderBox.globalToLocal(details.localPosition);
              if (isFirstDraw) {
                drawObjects.add(DrawObject([], selectedColor));
                isFirstDraw = false;
              }
              if (drawObjects.isNotEmpty) {
                drawObjects.last.points.add(offset);
              }
            });
          },
          onPanEnd: (details) {
            setState(() {
              drawObjects.add(DrawObject([], selectedColor));
              isFirstDraw = true;
            });
          },
          child: CustomPaint(
            painter: MyPainter(drawObjects),
            size: Size.infinite,
          ),
        ),
      ),
    );
  }
}

/// 自定义画板的绘制器 [CustomPainter]。
class MyPainter extends CustomPainter {
  final List<DrawObject> drawObjects;

  MyPainter(this.drawObjects);

  
  void paint(Canvas canvas, Size size) {
    for (final drawObject in drawObjects) {
      final paint = Paint()
        ..color = drawObject.color
        ..strokeCap = StrokeCap.round
        ..strokeWidth = 5.0;

      if (drawObject.points.length > 1) {
        for (int i = 0; i < drawObject.points.length - 1; i++) {
          if (drawObject.points[i] != null &&
              drawObject.points[i + 1] != null) {
            canvas.drawLine(
                drawObject.points[i]!, drawObject.points[i + 1]!, paint);
          }
        }
      } else if (drawObject.points.isNotEmpty) {
        canvas.drawPoints(
            PointMode.points, drawObject.points.cast<Offset>(), paint);
      }
    }
  }

  
  bool shouldRepaint(CustomPainter oldDelegate) {
    return true;
  }
}

/// 用户绘制的对象类,包括点和颜色。
class DrawObject {
  final List<Offset?> points;
  final Color color;

  DrawObject(this.points, this.color);
}

效果如图所示:

Flutter笔记:手写一个简单的画板工具,Dart语言与Flutter框架开发笔记,前端、桌面端、移动端、UI、构建工具,Flutter,Dart,画板,绘图文章来源地址https://www.toymoban.com/news/detail-729957.html

到了这里,关于Flutter笔记:手写一个简单的画板工具的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • flutter的引擎,Dart语言概括

    Dart是谷歌开发的, 类型安全的 , 面向对象 的编程语言,被应用于 Web、服务器、移动应用和物联网 等领域。 dart是谷歌在2011年推出的编程语言。谷歌希望使用dart来取代JavaScript。谷歌是一个颠覆式创新公司,谷歌退出golang是为了取代java,c++。谷歌退出flutter就是为了取代R

    2023年04月22日
    浏览(32)
  • Flutter Dart语言(04)库操作

    该系列教程主要是为有一定语言基础 C/C++的程序员,快速学习一门新语言所采用的方法,属于在C/C++基础上扩展新语言的模式。 引入代码如下所示: 一般从官方网站:Page 1 | Top packages中 搜索需要的第三方库,打开项目中的配置文件,名为:pubspec.yaml,找到dependencies选项,这

    2024年02月14日
    浏览(27)
  • Unity 实现简单画板画画功能(笔记)

    创建一个plan 摄像机调成正交摄像机 创建三个cube,附上材质 再创建一个cube,找个色板, 创建一个脚本,挂在非失活物体上 创建一个slider,放在旁边 Unity 画板

    2024年02月15日
    浏览(29)
  • flutter手写一个底部导航栏

    使用了一下flutter提供的导航栏,NavigationBar,不过感觉使用起来不是很方便。 譬如说: 不能直接使用图片资源,需要中间加几层转换把图片转化成icon 文本大小及颜色不太好控制 状态栏的上边来一个横线也没有相应的样式,等等。 因此想来想去还是自己写一个吧。 效果如下

    2024年02月12日
    浏览(31)
  • Flutter笔记:发布一个模块 scale_design - (移动端)设计师尺寸适配工具

    Flutter笔记 发布一个模块scale_design 设计师尺寸适配工具与常用组件库 作者 : 李俊才 (jcLee95):https://blog.csdn.net/qq_28550263 邮箱 : 291148484@163.com 本文地址 :https://blog.csdn.net/qq_28550263/article/details/134210226 模块地址 :https://pub.dev/packages/scale_design 仓库地址 :https://github.com/jack

    2024年02月05日
    浏览(28)
  • uniapp - 实现微信小程序电子签名板,横屏手写姓名签名专用写字画板(详细运行示例,一键复制开箱即用)

    实现了在uniapp项目中,微信小程序平台流畅的写字签名板(也可以绘图)功能源码,复制粘贴,改改样式几分钟即可搞定! 支持自动横屏、持预览,真机运行测试非常流畅不卡顿。 如下代码所示。

    2024年02月16日
    浏览(121)
  • 【Flutter 面试题】Dart是什么?Dart和Flutter有什么关系?

    【Flutter 面试题】Dart是什么?Dart和Flutter有什么关系? 👏🏻 正在学 Flutter 的同学,你好! 😊 本专栏是解决 Flutter 面试过程中可能出现的问题,而进行汇总整理的。一个问题一篇文章,尽可能详细解答,满足面试需求。 🔍 想解决开发中的零散问题?碎片化教程 👉 Flutte

    2024年02月22日
    浏览(33)
  • 【Flutter】Flutter Dart 获取当前时间戳

    在日常的软件开发中,我们经常需要获取当前的时间戳。无论是用于日志记录,还是用于生成唯一标识符,或者是用于时间同步,时间戳都发挥着重要的作用。 本文将详细介绍如何在 Flutter 和 Dart 中获取当前时间戳。 通过阅读本文,你将掌握以下知识: 了解时间戳的重要性

    2024年02月12日
    浏览(27)
  • Flutter学习2 - Dart

    Dart语言特点: Dart 2.0 开始便是强类型语言,而且是静态类型的(可以类比Java,C#等) 强类型语言的优点:所有类型的检查必须在编译的时候完成 前端开发的 js 语言是弱类型的语言 强类型的语言需要在定义的时候指定类型,如果不指定,编译器也可以在赋值的时候推断出变

    2024年02月21日
    浏览(32)
  • 用C语言构建一个手写数字识别神经网络

    (原理和程序基本框架请参见前一篇 \\\"用C语言构建了一个简单的神经网路\\\") 1.准备训练和测试数据集 从http://yann.lecun.com/exdb/mnist/下载手写数字训练数据集, 包括图像数据train-images-idx3-ubyte.gz 和标签数据 train-labels-idx1-ubyte.gz. 分别将他们解压后放在本地文件夹中,解压后文件名

    2024年02月14日
    浏览(26)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包