美文网首页Android开发Flutter 入门与实战Flutter圈子
Flutter 通过拖拽交互调整曲线控制点绘制图形

Flutter 通过拖拽交互调整曲线控制点绘制图形

作者: 岛上码农 | 来源:发表于2022-08-04 20:36 被阅读0次

前言

上一篇我们通过Listener获取触控点的位置作为贝塞尔曲线的控制点,实现曲线的交互式绘制。不过,上一篇有个缺陷,控制点绘制完成后只能撤销,没法修改,如果要调整绘制的图形的话会非常麻烦,这一篇我们来实现控制点的拖拽式移动,动态调整位置来调整绘制的图形。

实现逻辑

上一篇的主要代码我们不做更改,主要是需要实现控制点的拖拽式移动,移动过程中动态绘制新的曲线。不过由于绘制过程中不能同时移动点,因此需要有个完成绘制的控制,完成绘制后才支持拖拽控制点。拖拽控制点实现这里有两个主要逻辑:

  1. 控制点的命中判断:即拖拽开始时判断哪个控制点被命中,需要移动。
  2. 监听触控位置的移动过程:移动过程中动态绘制新的图形,以便直接看到对应的效果。

控制点的命中判断在完成绘制后,首先需要监听触控按下事件,看看触控点是否覆盖了某个控制点的触控响应范围,同时对于距离较近的点,可能会同时命中多个点的触控响应范围,这个时候需要取距离最近的那个点。对于触控范围,我们定义为每个触控点的周边10个像素点。命中触控点的实现代码如下:

int checkPointToMove(Offset pressedPoint) {
    // 控制点非空才查找
    if (points.isNotEmpty) {
      var pointsToCheck = <Offset>[];
      final maxDistance = 10.0;
      // 查找触控响应范围内的控制点
      for (Offset p in points) {
        if ((p.dx - pressedPoint.dx).abs() < maxDistance &&
            (p.dy - pressedPoint.dy).abs() < maxDistance) {
          pointsToCheck.add(p);
        }
      }
      
      // 未找到
      if (pointsToCheck.length == 0) {
        return -1;
      } else if (pointsToCheck.length == 1) {
        // 只有一个点,直接返回
        return points.indexOf(pointsToCheck[0]);
      } else {
        // 有多个点命中,找到距离最近的点返回
        Offset point = pointsToCheck[0];
        var distance = distanceBetween(pointsToCheck[0], pressedPoint);
        for (int i = 1; i < pointsToCheck.length; i++) {
          var newDistance = distanceBetween(pointsToCheck[i], pressedPoint);
          if (newDistance < distance) {
            point = pointsToCheck[i];
            distance = newDistance;
          }
        }

        return points.indexOf(point);
      }
    }

    return -1;
  }

移动过程的处理就比较简单了,我们已经找到了命中的控制点,那就在触控位置移动监听响应方法onPointerMove中更新控制点位置,并重新绘制即可,代码如下,其中indexOfPointToMove是一个状态变量,即找到的控制点下标:

onPointerMove: ((event) {
  if (indexOfPointToMove != -1) {
    points[indexOfPointToMove] = event.localPosition;
    setState(() {});
  }
}),

应用

逻辑完成了,我们就来做一个绘制应用吧。我们尝试来绘制一个粽子的线条画看看。下面是调整前后的对比效果以及调整过程的动图,可以看到,调整后的还是更像粽子一些。


拖拽调整

总结

本篇介绍了如何通过拖拽调整贝塞尔曲线绘制的控制点来调整图形的绘制,实际上很多绘图都可能用到拖拽式的控制点位的调整,比如电子围栏的设置。实际上主要的代码是判断触控位置命中了具体哪个控制点。

相关文章

  • Flutter 通过拖拽交互调整曲线控制点绘制图形

    前言 上一篇我们通过Listener获取触控点的位置作为贝塞尔曲线的控制点,实现曲线的交互式绘制。不过,上一篇有个...

  • Flutter 仿QQ消息拖拽(贝塞尔曲线实现)

    Flutter 仿qq消息拖拽效果通过自定义view实现,效果如下 如果要实现最终的效果,首先绘制拖拽连接处的线(...

  • PS教程|曲线功能为照片除黄除黑

    01原图,偏黄,人像不突出 02新建曲线调整图层,加控制点,箭头可调整输入输出值 03单击曲线目标调整工具后,在图...

  • 自定义View-2Path贝塞尔曲线

    Path贝塞尔曲线 贝塞尔曲线用途广泛 QQ消息小红点拖拽效果炫酷的下拉控件翻书效果 一阶曲线是没有控制点的,只有...

  • 通过交互绘制贝塞尔曲线

    前言 之前几篇我们介绍了贝塞尔曲线的原理、绘制曲线和动效实现,这些都是代码预设好的,如果我们要根据需要自行绘制曲线...

  • matplotlib快速入门

    1. 折线图 1.1 绘制一条折线 图形如下: 1.2绘制两条折线的方法: 图形: 1.3 绘制多条曲线 1.4 ...

  • 自定义View(七)Path 贝塞尔曲线

    简单点说,贝塞尔曲线在Android上就是用来画曲线的。贝塞尔曲线是按阶分的,曲线有数据点和控制点两个重要参数绘制...

  • NURBS曲线

    NURBS曲线组成 控制点控制点负责控制曲线的整体轮廓,走势。 权重控制点将影响该点附近的曲线轨迹,该点的权重越大...

  • WebGL

    WebGL 使用户 绘制和渲染3D图形使用户 通过页面与三维图形交互技术栈: HTML、HTML5、JavaScr...

  • 这一篇让你彻底搞懂贝塞尔曲线的原理

    贝塞尔曲线介绍 我们在前面讲了绘制自定义曲线,而实际开发过程还会遇到更复杂的图形绘制,比如下面的这些图形: 这时候...

网友评论

    本文标题:Flutter 通过拖拽交互调整曲线控制点绘制图形

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