polyline.dart 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201
  1. import 'dart:ui';
  2. import 'package:fis_measure/interfaces/date_types/point.dart';
  3. import 'package:fis_measure/interfaces/enums/items.dart';
  4. import 'package:fis_measure/interfaces/process/items/item_metas.dart';
  5. import 'package:fis_measure/interfaces/process/items/item.dart';
  6. import 'package:fis_measure/interfaces/process/items/terms.dart';
  7. import 'package:fis_measure/interfaces/process/items/types.dart';
  8. import 'package:fis_measure/interfaces/process/workspace/point_info.dart';
  9. import 'package:fis_measure/process/calcuators/curve.dart';
  10. import 'package:fis_measure/process/calcuators/calculator.dart';
  11. import 'package:fis_measure/process/items/item.dart';
  12. import 'package:fis_measure/process/items/item_feature.dart';
  13. import 'package:fis_measure/process/primitives/utils/auto_snap.dart';
  14. import 'package:fis_measure/utils/canvas.dart';
  15. import 'area_abstract.dart';
  16. /// 折线多边形
  17. class Polyline extends AreaItemAbstract with AutoSnapMixin {
  18. Polyline(ItemMeta meta, IMeasureItem? parent) : super(meta, parent);
  19. static Polyline createAreaPerimeter(ItemMeta meta, [IMeasureItem? parent]) {
  20. Polyline polygon = Polyline(meta, parent);
  21. polygon.calculator = AreaPerimeterCal(polygon);
  22. return polygon;
  23. }
  24. static Polyline createCurveLength(ItemMeta meta, [IMeasureItem? parent]) {
  25. Polyline polygon = Polyline(meta, parent);
  26. polygon.calculator = CurveLengthCal(polygon);
  27. polygon.isClosed = false;
  28. return polygon;
  29. }
  30. @override
  31. bool onExecuteMouse(PointInfo args) {
  32. if (state == ItemStates.finished) {
  33. if (args.pointType == PointInfoType.mouseDown) {
  34. state = ItemStates.waiting;
  35. }
  36. }
  37. if (state == ItemStates.waiting) {
  38. if (args.pointType == PointInfoType.mouseDown) {
  39. handleMouseDownWhileWaiting(args);
  40. }
  41. } else if (state == ItemStates.running) {
  42. if (feature == null) return false;
  43. if (args.pointType == PointInfoType.mouseUp) return false;
  44. final f = feature!;
  45. if (args.pointType == PointInfoType.mouseDown) {
  46. f.innerPoints.add(args);
  47. } else {
  48. f.innerPoints.last = args;
  49. }
  50. doCalculate();
  51. checkAutoSnap(args);
  52. // if (f.startPoint.almostEquals(f.endPoint)) {
  53. // doFeatureFinish();
  54. // }
  55. }
  56. return true;
  57. }
  58. DPoint lastStartPoint = DPoint(0, 0); // 上一个起点
  59. bool isFirstPointNeedOffset = false; // 第一个点是否需要施加偏移
  60. DPoint splineTouchStartPoint = DPoint(0, 0); // 轨迹触摸起始点
  61. bool isOtherPointNeedOffset = false; // 其他点是否需要施加偏移
  62. @override
  63. bool onExecuteTouch(PointInfo args) {
  64. if (state == ItemStates.finished) {
  65. if (args.pointType == PointInfoType.touchDown) {
  66. state = ItemStates.waiting;
  67. }
  68. }
  69. if (state == ItemStates.waiting) {
  70. if (isFirstPointNeedOffset) args.addOffset(0, -0.2);
  71. switch (args.pointType) {
  72. case PointInfoType.touchDown:
  73. handleTouchDownWhileWaiting(args);
  74. isFirstPointNeedOffset = false;
  75. break;
  76. case PointInfoType.touchUp:
  77. lastStartPoint = args; // 设置线段起点
  78. state = ItemStates.running;
  79. feature?.innerPoints.first = args;
  80. break; // 按下立即抬起无事发生
  81. case PointInfoType.touchMove:
  82. if (isMoveTargetOutOfRange(args)) return true;
  83. isFirstPointNeedOffset = true;
  84. feature?.innerPoints.first = args;
  85. break;
  86. default:
  87. break;
  88. }
  89. } else if (state == ItemStates.running) {
  90. if (feature == null) return false;
  91. DPoint newPoint =
  92. lastStartPoint.clone().addVector(args - splineTouchStartPoint);
  93. final f = feature!;
  94. if (args.pointType == PointInfoType.touchUp) {
  95. if (!isOtherPointNeedOffset) {
  96. f.innerPoints.last = args;
  97. lastStartPoint = args;
  98. doCalculate();
  99. } else {
  100. lastStartPoint = newPoint;
  101. }
  102. }
  103. if (args.pointType == PointInfoType.touchDown) {
  104. isOtherPointNeedOffset = false;
  105. splineTouchStartPoint = args;
  106. f.innerPoints.add(lastStartPoint);
  107. }
  108. if (args.pointType == PointInfoType.touchMove) {
  109. if (isMoveTargetOutOfRange(args)) return true;
  110. isOtherPointNeedOffset = true;
  111. f.innerPoints.last = newPoint;
  112. }
  113. doCalculate();
  114. PointInfo newPointInfo = isOtherPointNeedOffset
  115. ? PointInfo.fromOffset(newPoint.toOffset(), args.pointType)
  116. : args;
  117. checkAutoSnap(newPointInfo);
  118. }
  119. return true;
  120. }
  121. void handleMouseDownWhileWaiting(PointInfo args) {
  122. // TODO: 判断是否当前area
  123. // 转换为Area逻辑位置
  124. final point = args.toAreaLogicPoint();
  125. feature = PolylineFeature(this, point);
  126. if (args.hostVisualArea != null) {
  127. feature!.hostVisualArea = args.hostVisualArea;
  128. }
  129. state = ItemStates.running;
  130. }
  131. void handleTouchDownWhileWaiting(PointInfo args) {
  132. // TODO: 判断是否当前area
  133. final point = args.toAreaLogicPoint();
  134. feature = PolylineFeature.withOneStartPoint(this, point);
  135. if (args.hostVisualArea != null) {
  136. feature!.hostVisualArea = args.hostVisualArea;
  137. }
  138. }
  139. }
  140. class PolylineFeature extends AreaItemFeatureAbstract {
  141. PolylineFeature(AreaItemAbstract refItem, DPoint point) : super(refItem) {
  142. innerPoints.add(point.clone());
  143. innerPoints.add(point.clone());
  144. }
  145. PolylineFeature.withOneStartPoint(AreaItemAbstract refItem, DPoint point)
  146. : super(refItem) {
  147. innerPoints.add(point.clone());
  148. }
  149. @override
  150. void paint(Canvas canvas, Size size) {
  151. if (innerPoints.isEmpty) return;
  152. drawId(canvas, size);
  153. final startOffset = convert2ViewPoint(size, startPoint).toOffset();
  154. if (innerPoints.length == 1) {
  155. drawVertex(canvas, startOffset, true);
  156. return;
  157. } else {
  158. drawVertex(canvas, startOffset);
  159. }
  160. final len = innerPoints.length;
  161. for (var i = 1; i < len; i++) {
  162. final a = innerPoints[i - 1];
  163. final b = innerPoints[i];
  164. final offsetA = convert2ViewPoint(size, a).toOffset();
  165. final offsetB = convert2ViewPoint(size, b).toOffset();
  166. canvas.drawDashLine(offsetA, offsetB, 1, 10, paintPan);
  167. final isLast = len - i == 1;
  168. if (isLast) {
  169. drawVertex(canvas, offsetB, isActive);
  170. if (isClosed) {
  171. canvas.drawDashLine(offsetB, startOffset, 1, 10, paintLinePan);
  172. }
  173. } else {
  174. drawVertex(canvas, offsetB, false);
  175. }
  176. }
  177. }
  178. }