multiple_trace.dart 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  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.dart';
  5. import 'package:fis_measure/interfaces/process/items/item_metas.dart';
  6. import 'package:fis_measure/interfaces/process/workspace/point_info.dart';
  7. import 'package:fis_measure/process/calcuators/trace.dart';
  8. import 'package:fis_measure/process/items/item.dart';
  9. import 'package:fis_measure/process/items/item_feature.dart';
  10. import 'package:fis_measure/utils/canvas.dart';
  11. /// 手势轨迹图形
  12. class MultiTrace extends TraceItemAbstract {
  13. MultiTrace(ItemMeta meta, IMeasureItem? parent) : super(meta, parent);
  14. @override
  15. bool onExecuteMouse(PointInfo args) {
  16. if (state == ItemStates.finished) {
  17. if (args.pointType == PointInfoType.mouseDown) {
  18. state = ItemStates.waiting;
  19. }
  20. }
  21. if (state == ItemStates.waiting) {
  22. if (args.pointType == PointInfoType.mouseDown) {
  23. handleMouseDownWhileWaiting(args);
  24. }
  25. } else if (state == ItemStates.running) {
  26. if (args.pointType == PointInfoType.mouseUp) return false;
  27. feature?.adopt(args);
  28. doCalculate();
  29. if (args.pointType == PointInfoType.mouseDown) {
  30. doFeatureFinish();
  31. }
  32. }
  33. return true;
  34. }
  35. @override
  36. void doFeatureFinish() {
  37. super.doFeatureFinish();
  38. }
  39. void handleMouseDownWhileWaiting(PointInfo args) {
  40. // TODO: 判断是否当前area
  41. // 转换为Area逻辑位置
  42. feature = MultiTraceFeature(this);
  43. if (args.hostVisualArea != null) {
  44. feature!.hostVisualArea = args.hostVisualArea;
  45. }
  46. final point = args.toAreaLogicPoint();
  47. feature!.adopt(point);
  48. state = ItemStates.running;
  49. }
  50. PointInfo? startPoint;
  51. @override
  52. bool onExecuteTouch(PointInfo args) {
  53. if (state == ItemStates.finished) {
  54. if (args.pointType == PointInfoType.touchDown) {
  55. state = ItemStates.waiting;
  56. }
  57. }
  58. if (state == ItemStates.waiting) {
  59. switch (args.pointType) {
  60. case PointInfoType.touchDown:
  61. startPoint = args; // 设置线段起点
  62. break;
  63. case PointInfoType.touchUp:
  64. break; // 按下立即抬起无事发生
  65. case PointInfoType.touchMove:
  66. handleMouseDownWhileWaiting(startPoint!); // 通过设置的起点开始一个绘制事件
  67. break;
  68. default:
  69. break;
  70. }
  71. } else if (state == ItemStates.running) {
  72. if (args.pointType == PointInfoType.touchUp) {
  73. doFeatureFinish();
  74. }
  75. if (args.pointType == PointInfoType.touchMove) {
  76. feature?.adopt(args);
  77. doCalculate();
  78. }
  79. }
  80. return true;
  81. }
  82. static MultiTrace createTrace(
  83. ItemMeta meta, [
  84. IMeasureItem? parent,
  85. ]) {
  86. MultiTrace trace = MultiTrace(meta, parent);
  87. trace.calculator = TraceCal(trace);
  88. return trace;
  89. }
  90. }
  91. class MultiTraceFeature extends TraceItemFeatureAbstract {
  92. MultiTraceFeature(TraceItemAbstract refItem) : super(refItem);
  93. final greenPen = Paint()
  94. ..color = const Color.fromARGB(255, 0, 255, 0)
  95. ..isAntiAlias = true
  96. ..strokeWidth = 1
  97. ..style = PaintingStyle.stroke;
  98. final dashLinePan = Paint()
  99. ..color = const Color.fromARGB(255, 255, 255, 0)
  100. ..isAntiAlias = false
  101. ..strokeWidth = 1
  102. ..style = PaintingStyle.stroke;
  103. DPoint furthestPoint = DPoint(0, 0);
  104. @override
  105. void paint(Canvas canvas, Size size) {
  106. final double areaTop = hostVisualArea!.displayRegion.top * size.height;
  107. final double areaBottom =
  108. hostVisualArea!.displayRegion.bottom * size.height;
  109. if (innerPoints.isEmpty) return;
  110. double maxDistance = 0;
  111. drawId(canvas, size);
  112. final points = innerPoints.map((e) => convert2ViewPoint(size, e)).toList();
  113. final startOffset = convert2ViewPoint(size, startPoint);
  114. final endOffset = convert2ViewPoint(size, endPoint);
  115. canvas.drawDashLine(Offset(startOffset.x, areaTop),
  116. Offset(startOffset.x, areaBottom), 3, 3, dashLinePan);
  117. canvas.drawDashLine(Offset(endOffset.x, areaTop),
  118. Offset(endOffset.x, areaBottom), 3, 3, dashLinePan);
  119. if (points.length > 1) {
  120. final Path path = Path();
  121. path.moveTo(startOffset.x, startOffset.y);
  122. for (var i = 1; i < points.length; i++) {
  123. final point = points[i];
  124. path.lineTo(point.x, point.y);
  125. final distance = (point.y - startOffset.y).abs();
  126. if (distance > maxDistance) {
  127. maxDistance = distance;
  128. furthestPoint = point;
  129. }
  130. }
  131. drawCrossVertex(canvas, furthestPoint.toOffset());
  132. canvas.drawPath(
  133. path,
  134. greenPen,
  135. );
  136. }
  137. }
  138. }
  139. abstract class TraceItemFeatureAbstract extends MeasureItemFeature {
  140. TraceItemFeatureAbstract(TraceItemAbstract refItem) : super(refItem);
  141. @override
  142. TraceItemAbstract get refItem => super.refItem as TraceItemAbstract;
  143. DPoint get startPoint => innerPoints.first;
  144. DPoint get endPoint => innerPoints.last;
  145. bool ifRightSide = true;
  146. /// 接收新坐标
  147. void adopt(DPoint point) {
  148. if (innerPoints.isEmpty) {
  149. innerPoints.add(point);
  150. }
  151. if (point.x > startPoint.x) {
  152. handleChangeSide(point);
  153. if (point.x < innerPoints.last.x) {
  154. clearRight(point.x);
  155. } else {
  156. innerPoints.add(point);
  157. }
  158. } else {
  159. handleChangeSide(point);
  160. if (point.x > innerPoints.last.x) {
  161. clearLeft(point.x);
  162. } else {
  163. innerPoints.add(point);
  164. }
  165. }
  166. }
  167. void clearRight(double X) {
  168. if (innerPoints.isEmpty) return;
  169. for (var i = innerPoints.length - 1; i >= 0; i--) {
  170. if (innerPoints[i].x >= X) {
  171. innerPoints.removeAt(i);
  172. }
  173. }
  174. }
  175. void clearLeft(double X) {
  176. if (innerPoints.isEmpty) return;
  177. for (var i = innerPoints.length - 1; i >= 0; i--) {
  178. if (innerPoints[i].x <= X) {
  179. innerPoints.removeAt(i);
  180. }
  181. }
  182. }
  183. void handleChangeSide(point) {
  184. if (ifRightSide) {
  185. if (point.x < startPoint.x) {
  186. ifRightSide = false;
  187. innerPoints.clear();
  188. innerPoints.add(point);
  189. }
  190. } else {
  191. if (point.x > startPoint.x) {
  192. ifRightSide = true;
  193. innerPoints.clear();
  194. innerPoints.add(point);
  195. }
  196. }
  197. }
  198. }
  199. abstract class TraceItemAbstract extends MeasureItem<TraceItemFeatureAbstract> {
  200. TraceItemAbstract(ItemMeta meta, IMeasureItem? parent) : super(meta, parent);
  201. }