simpson.dart 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404
  1. // ignore_for_file: non_constant_identifier_names
  2. import 'package:fis_measure/interfaces/process/items/item_metas.dart';
  3. import 'package:fis_measure/interfaces/process/items/terms.dart';
  4. import 'package:fis_measure/process/primitives/combos/simpson.dart';
  5. import 'package:fis_measure/process/primitives/multi_method/multi_simpson_path.dart';
  6. import 'package:fis_measure/process/primitives/simpson_path.dart';
  7. import 'package:vid/us/vid_us_unit.dart';
  8. import 'formulas/cardiac.dart';
  9. import 'lv_study.dart';
  10. import 'dart:math' as math;
  11. class LvSimpsonCal extends LvStudyCalculatorBase<LvStudySimpson> {
  12. LvSimpsonCal(super.ref) {
  13. //
  14. }
  15. @override
  16. void calculate() {
  17. if (ref.feature == null) return;
  18. final feature = ref.feature!;
  19. restoreVals();
  20. feature.updateStringValue(
  21. ItemOutputMeta(
  22. ref.meta.description, ref.meta.description, VidUsUnit.None),
  23. "",
  24. );
  25. for (var childItem in ref.childItems) {
  26. _clacChildItem(childItem as MultiSimpsonPath);
  27. }
  28. for (var output in ref.meta.outputs) {
  29. switch (output.name) {
  30. case MeasureTerms.LVEDV:
  31. updateLVEDV();
  32. break;
  33. case MeasureTerms.LVESV:
  34. updateLVESV();
  35. break;
  36. case MeasureTerms.SV:
  37. updateSV();
  38. break;
  39. case MeasureTerms.EF:
  40. updateEF();
  41. break;
  42. case MeasureTerms.FS:
  43. updatePercentFS();
  44. break;
  45. case MeasureTerms.SI:
  46. updateSI();
  47. break;
  48. case MeasureTerms.LVEDVI:
  49. updateLVEDVI();
  50. break;
  51. case MeasureTerms.LVESVI:
  52. updateLVESVI();
  53. break;
  54. }
  55. }
  56. // _updateA2cGroup();
  57. // _updateA4cGroup();
  58. }
  59. @override
  60. void updateLVEDV() {
  61. final value = _clacLvedvNew(ref.a2cLvedv, ref.a4cLvedv);
  62. if (value != null) {
  63. updateFloatValueByName(MeasureTerms.LVEDV, value, unit: VidUsUnit.cm3);
  64. v.lvedv = value;
  65. }
  66. }
  67. @override
  68. void updateLVESV() {
  69. final value = _clacLvedvNew(ref.a2cLvesv, ref.a4cLvesv);
  70. if (value != null) {
  71. updateFloatValueByName(MeasureTerms.LVESV, value, unit: VidUsUnit.cm3);
  72. v.lvesv = value;
  73. }
  74. }
  75. double? _clacLvedv(MultiSimpsonPath path1, MultiSimpsonPath path2) {
  76. final feature1 = findChildFeature(path1);
  77. final feature2 = findChildFeature(path2);
  78. if (feature1 == null || feature2 == null) {
  79. return null;
  80. }
  81. feature1 as SimpsonPathFeature;
  82. feature2 as SimpsonPathFeature;
  83. final distance1 = feature1.centerLineLength;
  84. final distance2 = feature2.centerLineLength;
  85. int index = 0;
  86. List<double> aDiameters =
  87. List.generate(SimpsonPath.splitterCount, (index) => 0);
  88. index = 0;
  89. feature1.horizontalSplitterLegths.forEach((key, value) {
  90. aDiameters[index++] = value;
  91. });
  92. List<double> bDiameters =
  93. List.generate(SimpsonPath.splitterCount, (index) => 0);
  94. index = 0;
  95. feature2.horizontalSplitterLegths.forEach((key, value) {
  96. bDiameters[index++] = value;
  97. });
  98. final longDiameter = (distance1 + distance2) / 2.0;
  99. final lvesv = CardiacFormulas.lvSimsonVolume(
  100. longDiameter,
  101. aDiameters,
  102. bDiameters,
  103. SimpsonPath.splitterCount,
  104. );
  105. return lvesv;
  106. }
  107. double? _clacLvedvNew(MultiSimpsonPath path1, MultiSimpsonPath path2) {
  108. final feature1 = findChildFeature(path1);
  109. final feature2 = findChildFeature(path2);
  110. if (feature1 == null || feature2 == null) {
  111. return null;
  112. }
  113. feature1 as SimpsonPathFeature;
  114. feature2 as SimpsonPathFeature;
  115. final distance1 = feature1.centerLineLength;
  116. final distance2 = feature2.centerLineLength;
  117. int index = 0;
  118. List<double> aDiameters =
  119. List.generate(SimpsonPath.splitterCount, (index) => 0);
  120. index = 0;
  121. feature1.horizontalSplitterLegths.forEach((key, value) {
  122. aDiameters[index++] = value;
  123. });
  124. List<double> bDiameters =
  125. List.generate(SimpsonPath.splitterCount, (index) => 0);
  126. index = 0;
  127. feature2.horizontalSplitterLegths.forEach((key, value) {
  128. bDiameters[index++] = value;
  129. });
  130. double sum = 0;
  131. for (var i = 0; i < SimpsonPath.splitterCount; i++) {
  132. sum += aDiameters[i] * bDiameters[i];
  133. }
  134. final maxL = math.max(distance1, distance2);
  135. final value = maxL * 3.1415926 / (4 * 20) * sum;
  136. return value;
  137. }
  138. void _clacChildItem(MultiSimpsonPath childItem) {
  139. final feature = findChildFeature(childItem);
  140. if (feature == null) {
  141. return;
  142. }
  143. feature as SimpsonPathFeature;
  144. // final outputs = childItem.meta.outputs;
  145. // TODO:
  146. final itemName = childItem.meta.description.split(' ').first;
  147. final outputs = [
  148. ItemOutputMeta("Area", "$itemName LVAs", VidUsUnit.mm2),
  149. ItemOutputMeta("L", "$itemName LVLs", VidUsUnit.mm),
  150. ];
  151. for (var output in outputs) {
  152. switch (output.name) {
  153. case "Area":
  154. feature.updateFloatValue(output, feature.area, VidUsUnit.cm2);
  155. break;
  156. case "L":
  157. feature.updateFloatValue(
  158. output, feature.centerLineLength, VidUsUnit.cm);
  159. break;
  160. default:
  161. }
  162. }
  163. }
  164. void _updateA2cGroup() {
  165. _updateLVEDV_A2C();
  166. _updateLVESV_A2C();
  167. }
  168. void _updateA4cGroup() {
  169. _updateLVEDV_A2C();
  170. _updateLVESV_A2C();
  171. }
  172. void _updateLVEDV_A2C() {
  173. final feature = findChildFeature(ref.a2cLvedv);
  174. if (feature == null) {
  175. return;
  176. }
  177. feature as SimpsonPathFeature;
  178. final value = _SimpsonFormulas.lvedv(
  179. feature.area,
  180. feature.centerLineLength,
  181. );
  182. ref.feature!.updateFloatValue(
  183. ItemOutputMeta("LVEDV(A2C Simp)", "LVEDV(A2C Simp)", VidUsUnit.ml),
  184. value,
  185. VidUsUnit.ml,
  186. );
  187. }
  188. void _updateLVESV_A2C() {
  189. final feature = findChildFeature(ref.a2cLvesv);
  190. if (feature == null) {
  191. return;
  192. }
  193. feature as SimpsonPathFeature;
  194. final value = _SimpsonFormulas.lvedv(
  195. feature.area,
  196. feature.centerLineLength,
  197. );
  198. ref.feature!.updateFloatValue(
  199. ItemOutputMeta("LVESV(A2C Simp)", "LVESV(A2C Simp)", VidUsUnit.ml),
  200. value,
  201. VidUsUnit.ml,
  202. );
  203. }
  204. void _updateLVEDV_A4C() {}
  205. void _updateLVESV_A4C() {}
  206. }
  207. class LvSingleSimpsonCal extends LvStudyCalculatorBase<LvStudySingleSimpson> {
  208. LvSingleSimpsonCal(super.ref) {
  209. //
  210. }
  211. @override
  212. void calculate() {
  213. if (ref.feature == null) return;
  214. final feature = ref.feature!;
  215. restoreVals();
  216. feature.updateStringValue(
  217. ItemOutputMeta(
  218. ref.meta.description, ref.meta.description, VidUsUnit.None),
  219. "",
  220. );
  221. _clacChildItem(ref.lvedv);
  222. for (var output in ref.meta.outputs) {
  223. switch (output.name) {
  224. case MeasureTerms.LVEDV:
  225. updateLVEDV();
  226. break;
  227. case MeasureTerms.LVESV:
  228. updateLVESV();
  229. break;
  230. case MeasureTerms.SV:
  231. updateSV();
  232. break;
  233. case MeasureTerms.EF:
  234. updateEF();
  235. break;
  236. case MeasureTerms.SI:
  237. updateSI();
  238. break;
  239. case MeasureTerms.LVEDVI:
  240. updateLVEDVI();
  241. break;
  242. case MeasureTerms.LVESVI:
  243. updateLVESVI();
  244. break;
  245. }
  246. }
  247. /**
  248. *
  249. <OutPut Id="LVEDV" IsEnabled="False" CalculationIDs="LVEDV(BP Simp)"/>
  250. <OutPut Id="LVESV" IsEnabled="False" CalculationIDs="LVESV(BP Simp)"/>
  251. <OutPut Id="SV" IsEnabled="False" CalculationIDs="SV(Mod Simp)"/>
  252. <OutPut Id="EF" IsEnabled="False" CalculationIDs="EF(Mod Simp)"/>
  253. <OutPut Id="SI" IsEnabled="False" CalculationIDs="SI(Mod Simp)"/>
  254. <OutPut Id="LVEDVI" IsEnabled="False" CalculationIDs="LVEDVI(Mod Simp)"/>
  255. <OutPut Id="LVESVI" IsEnabled="False" CalculationIDs="LVESVI(Mod Simp)"/>
  256. <OutPut Id="CO" IsEnabled="False" CalculationIDs="CO(Mod Simp)"/>
  257. <OutPut Id="CI" IsEnabled="False" CalculationIDs="CI(Mod Simp)"/>
  258. */
  259. }
  260. @override
  261. void updateLVEDV() {
  262. final value = _calcLvedvNew(ref.lvedv);
  263. if (value != null) {
  264. updateFloatValueByName(MeasureTerms.LVEDV, value, unit: VidUsUnit.cm3);
  265. v.lvedv = value;
  266. }
  267. }
  268. @override
  269. void updateLVESV() {
  270. final value = _calcLvedvNew(ref.lvesv);
  271. if (value != null) {
  272. updateFloatValueByName(MeasureTerms.LVESV, value, unit: VidUsUnit.cm3);
  273. v.lvesv = value;
  274. }
  275. }
  276. double? _calcLvedv(MultiSimpsonPath item) {
  277. if (item.feature == null) {
  278. return null;
  279. }
  280. final feature = item.feature!;
  281. double longDiameter = roundDouble(feature.centerLineLength);
  282. int index = 0;
  283. List<double> bDiameters =
  284. List.generate(SimpsonPath.splitterCount, (index) => 0);
  285. feature.horizontalSplitterLegths.forEach((key, value) {
  286. bDiameters[index++] = value;
  287. });
  288. return CardiacFormulas.lvSimsonVolume(
  289. longDiameter,
  290. bDiameters,
  291. bDiameters,
  292. SimpsonPath.splitterCount,
  293. );
  294. }
  295. double? _calcLvedvNew(MultiSimpsonPath item) {
  296. final feature = findChildFeature(item);
  297. if (feature == null) {
  298. return null;
  299. }
  300. feature as SimpsonPathFeature;
  301. double longDiameter = roundDouble(feature.centerLineLength);
  302. int index = 0;
  303. List<double> bDiameters =
  304. List.generate(SimpsonPath.splitterCount, (index) => 0);
  305. feature.horizontalSplitterLegths.forEach((key, value) {
  306. bDiameters[index++] = value;
  307. });
  308. double sum = 0;
  309. for (var i = 0; i < SimpsonPath.splitterCount; i++) {
  310. sum += math.pow(bDiameters[i], 2).toDouble();
  311. }
  312. var value = longDiameter * 3.1415926 / (4 * 20) * sum;
  313. return value;
  314. return CardiacFormulas.lvSimsonVolume(
  315. longDiameter,
  316. bDiameters,
  317. bDiameters,
  318. SimpsonPath.splitterCount,
  319. );
  320. }
  321. void _clacChildItem(MultiSimpsonPath childItem) {
  322. final feature = findChildFeature(childItem);
  323. if (feature == null) {
  324. return;
  325. }
  326. feature as SimpsonPathFeature;
  327. // final outputs = childItem.meta.outputs;
  328. // TODO:
  329. final itemName = childItem.meta.description.split(' ').first;
  330. final outputs = [
  331. ItemOutputMeta("Area", "$itemName LVAs", VidUsUnit.mm2),
  332. ItemOutputMeta("L", "$itemName LVLs", VidUsUnit.mm),
  333. ];
  334. for (var output in outputs) {
  335. switch (output.name) {
  336. case "Area":
  337. feature.updateFloatValue(output, feature.area, VidUsUnit.cm2);
  338. break;
  339. case "L":
  340. feature.updateFloatValue(
  341. output, feature.centerLineLength, VidUsUnit.cm);
  342. break;
  343. default:
  344. }
  345. }
  346. }
  347. }
  348. abstract class _SimpsonFormulas {
  349. static double lvedv(double area, double distance) {
  350. final value = 0.85 * math.pow(area, 2) / distance;
  351. return value;
  352. }
  353. }