application.dart 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249
  1. import 'package:fis_measure/interfaces/process/items/item.dart';
  2. import 'package:fis_measure/interfaces/process/items/item_feature.dart';
  3. import 'package:fis_measure/interfaces/process/items/measure_terms.dart';
  4. import 'package:fis_measure/interfaces/process/visuals/visual_area.dart';
  5. import 'package:fis_measure/interfaces/process/visuals/visual.dart';
  6. import 'package:fis_measure/interfaces/process/viewports/viewport.dart';
  7. import 'package:fis_measure/interfaces/process/modes/mode.dart';
  8. import 'package:fis_common/event/event_type.dart';
  9. import 'package:fis_measure/interfaces/process/workspace/application.dart';
  10. import 'package:fis_measure/interfaces/process/workspace/point_info.dart';
  11. import 'package:fis_measure/process/primitives/ellipse.dart';
  12. import 'package:fis_measure/process/primitives/location.dart';
  13. import 'package:fis_measure/process/primitives/straightline.dart';
  14. import 'package:flutter/painting.dart';
  15. import 'package:vid/us/vid_us_image.dart';
  16. import 'package:vid/us/vid_us_probe.dart';
  17. import 'package:vid/us/vid_us_unit.dart';
  18. import '../visual/visual.dart';
  19. class Application implements IApplication {
  20. late VidUsProbe _probe;
  21. VidUsImage? _frame;
  22. List<IVisual>? _visuals;
  23. IMeasureItem? _activeItem;
  24. bool _canMeasure = false;
  25. Size _displaySize = Size.zero;
  26. final Set<IMeasureItem> _items = {};
  27. Application(VidUsProbe probe) {
  28. _probe = probe;
  29. }
  30. @override
  31. bool get canMeasure => _canMeasure;
  32. @override
  33. set canMeasure(bool value) {
  34. if (value != _canMeasure) {
  35. _canMeasure = value;
  36. _doCanMeasureChanged();
  37. }
  38. }
  39. bool get isProbeConvex => probe.type == VidUsProbeType.Convex;
  40. @override
  41. VidUsProbe get probe => _probe;
  42. @override
  43. String get applicationName => _probe.application.applicationName;
  44. @override
  45. String get categoryName => _probe.application.applicationCategoryName;
  46. @override
  47. IMode get currentMode => currentVisualArea.mode;
  48. @override
  49. IViewPort get currentViewPort => currentVisualArea.viewport!;
  50. @override
  51. IVisual get currentVisual => visuals.firstWhere((e) => e.activeArea != null);
  52. @override
  53. IVisualArea get currentVisualArea => currentVisual.activeArea!;
  54. @override
  55. VidUsImage? get frameData => _frame;
  56. @override
  57. List<IVisual> get visuals => _visuals!;
  58. @override
  59. Set<IMeasureItem> get items => _items;
  60. @override
  61. Size get displaySize => _displaySize;
  62. @override
  63. set displaySize(Size value) {
  64. if (value != _displaySize) {
  65. _displaySize = value;
  66. }
  67. }
  68. @override
  69. List<IMode> get avaliableModes {
  70. final modes = <IMode>[];
  71. for (var visual in visuals) {
  72. modes.addAll(visual.modes);
  73. }
  74. return modes;
  75. }
  76. @override
  77. IMeasureItem? get activeItem => _activeItem;
  78. set activeItem(IMeasureItem? value) {
  79. if (value != _activeItem) {
  80. _activeItem?.featureUpdated.removeListener(_onActiveItemFeatureUpdated);
  81. _activeItem = value;
  82. if (_activeItem != null) {
  83. _items.add(_activeItem!);
  84. _activeItem!.featureUpdated.addListener(_onActiveItemFeatureUpdated);
  85. }
  86. activeItemChanged.emit(this, value);
  87. _doMeasureRerender();
  88. }
  89. }
  90. @override
  91. var currentModeChanged = FEventHandler<IMode>();
  92. @override
  93. var visualAreaChanged = FEventHandler<IVisualArea>();
  94. @override
  95. var canMeasureChanged = FEventHandler<bool>();
  96. @override
  97. var activeItemChanged = FEventHandler<IMeasureItem?>();
  98. @override
  99. var measureRerenderReady = FEventHandler<void>();
  100. @override
  101. void loadFrame(VidUsImage frame) {
  102. bool frameLoaded = _frame != null;
  103. _frame = frame;
  104. if (!frameLoaded && canMeasure) {
  105. _loadVisuals();
  106. }
  107. }
  108. @override
  109. PointInfo createPointInfo(Offset offset, PointInfoType type) {
  110. if (frameData == null) {
  111. throw NullThrownError();
  112. }
  113. final width = displaySize.width;
  114. final height = displaySize.height;
  115. final x = offset.dx / width;
  116. final y = offset.dy / height;
  117. final percentOffset = Offset(x, y);
  118. final info = PointInfo.fromOffset(percentOffset, type);
  119. info.hostVisualArea = visuals.first.visualAreas.first; // TODO: 找区域
  120. activeItem?.execute(info);
  121. return info;
  122. }
  123. @override
  124. void switchItemByName(String name) {
  125. // TODO: create from map
  126. if (name == MeasureTerms.Distance) {
  127. activeItem = StraightLine.createDistance(
  128. ItemMeta(
  129. MeasureTerms.Distance,
  130. {
  131. "Description": MeasureTerms.Distance,
  132. "BriefDescription": "D",
  133. "Unit": VidUsUnit.cm,
  134. },
  135. ),
  136. );
  137. return;
  138. }
  139. if (name == MeasureTerms.Perimeter) {
  140. activeItem = Ellipse(
  141. ItemMeta(
  142. MeasureTerms.Perimeter,
  143. {
  144. "Description": MeasureTerms.Perimeter,
  145. "BriefDescription": MeasureTerms.Perimeter,
  146. "Unit": VidUsUnit.cm,
  147. },
  148. ),
  149. null,
  150. );
  151. return;
  152. }
  153. if (name == MeasureTerms.Area) {
  154. activeItem = Ellipse(
  155. ItemMeta(
  156. MeasureTerms.Area,
  157. {
  158. "Description": MeasureTerms.Area,
  159. "BriefDescription": MeasureTerms.Area,
  160. "Unit": VidUsUnit.cm2,
  161. },
  162. ),
  163. null,
  164. );
  165. return;
  166. }
  167. if (name == MeasureTerms.Depth) {
  168. final Location Function(ItemMeta, [IMeasureItem?]) fn = isProbeConvex
  169. ? Location.createTissueConvexDepth
  170. : Location.createTissueDepth;
  171. activeItem = fn(
  172. ItemMeta(
  173. MeasureTerms.Depth,
  174. {
  175. "Description": MeasureTerms.Depth,
  176. "BriefDescription": MeasureTerms.Depth,
  177. "Unit": VidUsUnit.cm,
  178. },
  179. ),
  180. null,
  181. );
  182. return;
  183. }
  184. }
  185. void _doMeasureRerender() {
  186. measureRerenderReady.emit(this, null);
  187. }
  188. void _doCanMeasureChanged() {
  189. canMeasureChanged.emit(this, canMeasure);
  190. _clear();
  191. if (canMeasure) {
  192. if (frameData != null) {
  193. _loadVisuals();
  194. }
  195. }
  196. }
  197. void _onActiveItemFeatureUpdated(Object sender, IMeasureItemFeature? e) {
  198. _doMeasureRerender();
  199. }
  200. void _loadVisuals() {
  201. _clearVisuals();
  202. for (final data in frameData!.visuals) {
  203. _visuals!.add(VisualBase(data));
  204. }
  205. }
  206. void _clear() {
  207. for (var item in items) {
  208. item.clear();
  209. }
  210. _clearVisuals();
  211. }
  212. void _clearVisuals() {
  213. _visuals = [];
  214. }
  215. }