ellipse.dart 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309
  1. import 'dart:ui';
  2. import 'dart:math' as math;
  3. import 'package:fis_measure/interfaces/date_types/point.dart';
  4. import 'package:fis_measure/interfaces/enums/items.dart';
  5. import 'package:fis_measure/interfaces/process/items/item.dart';
  6. import 'package:fis_measure/interfaces/process/items/item_metas.dart';
  7. import 'package:fis_measure/interfaces/process/workspace/point_info.dart';
  8. import 'package:fis_measure/process/calcuators/calculator.dart';
  9. import 'package:fis_measure/process/calcuators/curve.dart';
  10. import 'package:fis_measure/process/calcuators/formulas/general.dart';
  11. import 'package:fis_measure/utils/canvas.dart';
  12. import 'package:path_drawing/path_drawing.dart';
  13. import 'package:vid/us/vid_us_unit.dart';
  14. import 'area_abstract.dart';
  15. class Ellipse extends AreaItemAbstract {
  16. Ellipse(ItemMeta meta, IMeasureItem? parent) : super(meta, parent);
  17. @override
  18. bool onExecuteMouse(PointInfo args) {
  19. if (state == ItemStates.finished) {
  20. if (args.pointType == PointInfoType.mouseDown) {
  21. state = ItemStates.waiting;
  22. }
  23. }
  24. if (state == ItemStates.waiting) {
  25. if (args.pointType == PointInfoType.mouseDown) {
  26. handleMouseDownWhileWaiting(args);
  27. }
  28. } else if (state == ItemStates.running) {
  29. if (feature == null) return false;
  30. final f = feature! as EllipseFeature;
  31. final activeIndex = f.activeIndex;
  32. switch (args.pointType) {
  33. case PointInfoType.mouseMove:
  34. f.innerPoints[activeIndex] = args;
  35. f.adjustPoints(args);
  36. break;
  37. case PointInfoType.mouseDown:
  38. if (activeIndex == 1) {
  39. if (f.xAxisEnd == f.xAxisStart) {
  40. break;
  41. }
  42. f.adjustPoints(args);
  43. f.activeIndex = 2;
  44. } else if (activeIndex == 2) {
  45. doFeatureFinish();
  46. }
  47. break;
  48. default:
  49. return false;
  50. }
  51. doCalculate();
  52. return true;
  53. }
  54. return false;
  55. }
  56. @override
  57. bool onExecuteTouch(PointInfo args) {
  58. return true;
  59. }
  60. void handleMouseDownWhileWaiting(PointInfo args) {
  61. // TODO: 判断是否当前area
  62. // 转换为Area逻辑位置
  63. final point = args.toAreaLogicPoint();
  64. feature = EllipseFeature(this, point);
  65. if (args.hostVisualArea != null) {
  66. feature!.hostVisualArea = args.hostVisualArea;
  67. }
  68. feature!.activeIndex = 1;
  69. state = ItemStates.running;
  70. }
  71. static Ellipse createAreaPerimeter(ItemMeta meta, [IMeasureItem? parent]) {
  72. final ellipse = Ellipse(meta, parent);
  73. ellipse.calculator = AreaPerimeterEllipseCal(ellipse);
  74. return ellipse;
  75. }
  76. static Ellipse createVolume(ItemMeta meta, [IMeasureItem? parent]) {
  77. final ellipse = Ellipse(meta, parent);
  78. ellipse.calculator = _EllipseVolumeCal(ellipse);
  79. return ellipse;
  80. }
  81. }
  82. class EllipseFeature extends AreaItemFeatureAbstract {
  83. EllipseFeature(Ellipse refItem, DPoint point) : super(refItem) {
  84. innerPoints.add(point.clone());
  85. innerPoints.add(point.clone());
  86. innerPoints.add(point.clone());
  87. innerPoints.add(point.clone());
  88. }
  89. /// 质心
  90. DPoint get centroid {
  91. final x = (xAxisStart.x + xAxisEnd.x) * 0.5;
  92. final y = (xAxisStart.y + xAxisEnd.y) * 0.5;
  93. return DPoint(x, y);
  94. }
  95. DPoint get xAxisStart => innerPoints[0];
  96. DPoint get xAxisEnd => innerPoints[1];
  97. DPoint get yAxisStart => innerPoints[2];
  98. DPoint get yAxisEnd => innerPoints[3];
  99. @override
  100. void paint(Canvas canvas, Size size) {
  101. if (innerPoints.isEmpty) return;
  102. var idText = '$id.${refItem.briefAnnotation}';
  103. drawId(canvas, size, idText);
  104. final xStartOffset = convert2ViewPoint(size, xAxisStart).toOffset();
  105. if (activeIndex < 1) {
  106. drawVertex(canvas, xStartOffset, true);
  107. return;
  108. } else {
  109. drawVertex(canvas, xStartOffset);
  110. }
  111. final xEndOffset = convert2ViewPoint(size, xAxisEnd).toOffset();
  112. canvas.drawDashLine(xStartOffset, xEndOffset, 1, 10, paintPan);
  113. final yStartOffset = convert2ViewPoint(size, innerPoints[2]).toOffset();
  114. final yEndOffset = convert2ViewPoint(size, innerPoints[3]).toOffset();
  115. final centroidOffset = convert2ViewPoint(size, centroid).toOffset();
  116. canvas.drawDashLine(yStartOffset, yEndOffset, 1, 10, paintPan);
  117. // canvas.drawDashLine(yStartOffset, xEndOffset, 1, 10, paintPan);
  118. // canvas.drawDashLine(yEndOffset, xEndOffset, 1, 10, paintPan);
  119. final radiusX = getRadiusX(size);
  120. final radiusY = getRadiusY(size);
  121. canvas.save();
  122. final path = Path();
  123. path.moveTo(xStartOffset.dx, xStartOffset.dy);
  124. // 以质心为原点旋转到水平方向
  125. canvas.translate(centroidOffset.dx, centroidOffset.dy);
  126. final rotateRad = (xEndOffset - xStartOffset).direction;
  127. canvas.rotate(rotateRad);
  128. canvas.translate(-centroidOffset.dx, -centroidOffset.dy);
  129. path.addOval(
  130. Rect.fromCenter(
  131. center: centroidOffset,
  132. width: radiusX * 2,
  133. height: radiusY * 2,
  134. ),
  135. );
  136. canvas.drawPath(
  137. dashPath(
  138. path,
  139. dashArray: CircularIntervalList<double>(<double>[2.0, 10.0]),
  140. ),
  141. paintPan,
  142. );
  143. canvas.restore();
  144. if (activeIndex == 1) {
  145. drawVertex(canvas, xEndOffset, true);
  146. drawVertex(canvas, yStartOffset);
  147. drawVertex(canvas, yEndOffset);
  148. } else if (activeIndex == 2) {
  149. drawVertex(canvas, xEndOffset);
  150. drawVertex(canvas, yEndOffset);
  151. drawVertex(canvas, yStartOffset, isActive);
  152. }
  153. }
  154. /// X轴半径
  155. double getRadiusX([Size? fitSize]) {
  156. if (innerPoints.length < 2) {
  157. return 0;
  158. }
  159. var p1 = xAxisStart;
  160. var p2 = xAxisEnd;
  161. if (fitSize != null) {
  162. p1 = p1.scale2Size(fitSize);
  163. p2 = p2.scale2Size(fitSize);
  164. }
  165. return (p2 - p1).length / 2;
  166. }
  167. /// Y轴半径
  168. double getRadiusY([Size? fitSize]) {
  169. if (activeIndex > 1) {
  170. var p = yAxisStart;
  171. var p2 = yAxisEnd;
  172. var c = centroid;
  173. if (fitSize != null) {
  174. p = p.scale2Size(fitSize);
  175. p2 = p2.scale2Size(fitSize);
  176. c = c.scale2Size(fitSize);
  177. }
  178. return (p - p2).length / 2;
  179. } else {
  180. return getRadiusX(fitSize);
  181. }
  182. }
  183. /// 圆周长
  184. double getCircumference([Size? fitSize]) {
  185. double a = getRadiusX(fitSize), b = getRadiusY(fitSize);
  186. if (a < b) {
  187. a = b;
  188. b = a;
  189. }
  190. return 2 * math.pi * b + 4 * (a - b);
  191. }
  192. /// 面积
  193. double getArea([Size? fitSize]) {
  194. return math.pi * getRadiusX(fitSize) * getRadiusY(fitSize);
  195. }
  196. /// 计算Y轴坐标点
  197. void adjustPoints(DPoint point) {
  198. if (activeIndex < 1) return;
  199. final refSize = refItem.application.displaySize;
  200. // const refSize= Size(1000, 1000);
  201. // const restoreSize = Size(1 / 1000, 1 / 1000);
  202. final restoreSize = Size(1 / refSize.width, 1 / refSize.height);
  203. final p = point.scale2Size(refSize);
  204. final p1 = xAxisStart.scale2Size(refSize);
  205. final p2 = xAxisEnd.scale2Size(refSize);
  206. final double radius;
  207. if (activeIndex == 2) {
  208. radius = GeneralFormulas.distance2Line(p1, p2, p);
  209. } else {
  210. radius = getRadiusX(refSize);
  211. }
  212. final crossPoints = _findCrossPoints(p2, p1, radius);
  213. final p3Index = _findNearest(p, crossPoints);
  214. innerPoints[2] = crossPoints[p3Index].scale2Size(restoreSize);
  215. innerPoints[3] = crossPoints[1 - p3Index].scale2Size(restoreSize);
  216. }
  217. static List<DPoint> _findCrossPoints(DPoint a, DPoint b, distance) {
  218. final dx = b.x - a.x;
  219. final dy = b.y - a.y;
  220. final len = (b - a).length;
  221. /// 单位长度轴上距离增加量
  222. final udx = dx / len;
  223. final udy = dy / len;
  224. final px = -udy;
  225. final py = udx;
  226. final centroidX = (a.x + b.x) * 0.5;
  227. final centroidY = (a.y + b.y) * 0.5;
  228. final centroid = DPoint(centroidX, centroidY);
  229. final x1 = centroid.x + px * distance;
  230. final y1 = centroid.y + py * distance;
  231. final out1 = DPoint(x1, y1);
  232. final x2 = centroid.x - px * distance;
  233. final y2 = centroid.y - py * distance;
  234. final out2 = DPoint(x2, y2);
  235. return [out1, out2];
  236. }
  237. static int _findNearest(DPoint p, List<DPoint> source) {
  238. int rst = 0;
  239. double minLen = (source[0] - p).length;
  240. for (var i = 1; i < source.length; i++) {
  241. final len = (source[i] - p).length;
  242. if (len < minLen) {
  243. minLen = len;
  244. rst = i;
  245. }
  246. }
  247. return rst;
  248. }
  249. }
  250. class _EllipseVolumeCal extends Calculator<Ellipse, double> {
  251. _EllipseVolumeCal(Ellipse ref) : super(ref);
  252. @override
  253. void calculate() {
  254. if (ref.feature == null) return;
  255. final feature = ref.feature! as EllipseFeature;
  256. final viewport = feature.hostVisualArea!.viewport!;
  257. final size = viewport.convertBoundary;
  258. double xAxis = feature.getRadiusX(size);
  259. double yAxis = feature.getRadiusY(size);
  260. var a = math.max(xAxis, yAxis);
  261. var b = math.min(xAxis, yAxis);
  262. var value = math.pi / 6.0 * a * b * b;
  263. updateFloatValue(value, unit: VidUsUnit.cm3, useRound: true);
  264. }
  265. }