menu_button_group.dart 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380
  1. import 'package:fis_common/index.dart';
  2. import 'package:fis_i18n/i18n.dart';
  3. import 'package:fis_jsonrpc/rpc.dart';
  4. import 'package:fis_lib_qrcode/qr_flutter.dart';
  5. import 'package:fis_measure/interfaces/process/player/play_controller.dart';
  6. import 'package:fis_measure/interfaces/process/standard_line/calibration.dart';
  7. import 'package:fis_measure/interfaces/process/workspace/application.dart';
  8. import 'package:fis_measure/interfaces/process/workspace/measure_3d_view_controller.dart';
  9. import 'package:fis_measure/process/workspace/measure_3d_view_controller.dart';
  10. import 'package:fis_measure/process/workspace/measure_data_controller.dart';
  11. import 'package:fis_measure/process/workspace/measure_handler.dart';
  12. import 'package:fis_measure/process/workspace/rpc_helper.dart';
  13. import 'package:fis_measure/utils/prompt_box.dart';
  14. import 'package:fis_measure/view/loadding/loadding.dart';
  15. import 'package:fis_measure/view/measure/operate_type_change_button.dart';
  16. import 'package:fis_measure/view/paint/ai_patint_controller.dart';
  17. import 'package:fis_measure/view/player/controller.dart';
  18. import 'package:fis_ui/base_define/page.dart';
  19. import 'package:fis_ui/index.dart';
  20. import 'package:fis_ui/widgets/functional/text_tooltip.dart';
  21. import 'package:flutter/material.dart';
  22. import 'package:flutter/services.dart';
  23. import 'package:get/get.dart';
  24. import '../measure/switch_loop_button.dart';
  25. // import 'package:url_launcher/url_launcher.dart';
  26. class FMenuButtonGroup extends FStatefulWidget {
  27. final VoidCallback capturePng;
  28. const FMenuButtonGroup({
  29. Key? key,
  30. required this.capturePng,
  31. required this.businessParent,
  32. }) : super(key: key);
  33. final FPage businessParent;
  34. @override
  35. FState<FMenuButtonGroup> createState() => _FMenuButtonGroupState();
  36. }
  37. class _FMenuButtonGroupState extends FState<FMenuButtonGroup> {
  38. double get devicePixelRatio => MediaQuery.of(context).devicePixelRatio;
  39. // 单个按钮高度
  40. double get buttonHeight => 30 / devicePixelRatio;
  41. // 按钮外边距
  42. double get buttonMargin => 10 / devicePixelRatio;
  43. // 单个按钮总高度 = Content(42)+Margin(10)
  44. // static const double singleButtonHeight = buttonHeight + buttonMargin;
  45. // 按钮组宽度
  46. double get buttonGroupWidth => 60 / devicePixelRatio;
  47. // 按钮组容器底部内边距 30 + 30
  48. double get groupContainerPadding => buttonGroupWidth / 2;
  49. /// 获取当前组件最大高度
  50. double maxGroupHeight = Get.height - 190;
  51. final application = Get.find<IApplication>();
  52. /// 数据
  53. final measureData = Get.find<MeasureDataController>();
  54. final playerController = Get.find<IPlayerController>() as VidPlayerController;
  55. final measure3DViewController = Get.find<Measure3DViewController>();
  56. late final measureHandler = Get.find<MeasureHandler>();
  57. late final aiPatintController = Get.find<AiPatintController>();
  58. late bool canShowAI = [
  59. DiagnosisConclusionEnum.Benign,
  60. DiagnosisConclusionEnum.Malignant,
  61. DiagnosisConclusionEnum.BenignAndMalignant
  62. ].contains(measureData.diagnosisConclusion);
  63. double groupHeight = 0;
  64. bool isExpanded = true;
  65. bool ifShowModeButton = false;
  66. bool ifShowVidModeButton = true;
  67. get isShell => measure3DViewController.isShell;
  68. get exist3DData => measure3DViewController.exist3DData;
  69. @override
  70. void initState() {
  71. super.initState();
  72. groupHeight = maxGroupHeight;
  73. measure3DViewController.updatePlayerMode.addListener(_onModeChanged);
  74. }
  75. @override
  76. void dispose() {
  77. measure3DViewController.updatePlayerMode.removeListener(_onModeChanged);
  78. super.dispose();
  79. }
  80. /// 模式改变触发更新
  81. void _onModeChanged(Object s, MeasureMode mode) {
  82. switch (mode) {
  83. case MeasureMode.vidMode:
  84. setState(() {
  85. ifShowModeButton = false;
  86. ifShowVidModeButton = true;
  87. groupHeight = maxGroupHeight;
  88. isExpanded = true;
  89. });
  90. break;
  91. case MeasureMode.carotid2DMode:
  92. setState(() {
  93. ifShowModeButton = true;
  94. ifShowVidModeButton = false;
  95. groupHeight = maxGroupHeight;
  96. isExpanded = true;
  97. });
  98. break;
  99. case MeasureMode.carotid3DMode:
  100. setState(() {
  101. ifShowModeButton = false;
  102. ifShowVidModeButton = false;
  103. });
  104. break;
  105. }
  106. }
  107. List<FWidget> buildTitleButtonList() {
  108. return [
  109. _buildTitleButton(
  110. FIcons.fis_quash,
  111. i18nBook.common.revoke.t,
  112. () => application.undoRecord(),
  113. ),
  114. _buildTitleButton(
  115. FIcons.fis_purge,
  116. i18nBook.measure.clear.t,
  117. () => application.clearRecords(),
  118. ),
  119. OperateTypeChangeButton(businessParent: widget.businessParent),
  120. _buildTitleButton(
  121. measureHandler.fullScreenState
  122. ? FIcons.fis_full_screen_reduction
  123. : FIcons.fis_full_screen,
  124. measureHandler.fullScreenState
  125. ? i18nBook.common.cancel.t + i18nBook.measure.fullScreen.t
  126. : i18nBook.measure.fullScreen.t,
  127. () {
  128. measureHandler.fullScreenState = !measureHandler.fullScreenState;
  129. setState(() {});
  130. },
  131. ),
  132. _buildTitleButton(
  133. FIcons.fis_save,
  134. i18nBook.common.save.t,
  135. widget.capturePng,
  136. ),
  137. _buildTitleButton(
  138. FIcons.fis_share,
  139. i18nBook.remedical.share.t,
  140. () async {
  141. var itemCurrentImage = await measureData.shareImage.call(
  142. measureData.itemCurrentImage,
  143. );
  144. Get.dialog(_buildShareQRCode(itemCurrentImage));
  145. },
  146. ),
  147. //第三方图像需要校准线 并且能够测量
  148. if (application.isThirdPart && measureHandler.canMeasureDrawing) ...[
  149. _buildTitleButton(
  150. FIcons.device,
  151. i18nBook.remedical.calibrationLine.t,
  152. () {
  153. Get.find<IStandardLineCalibrationController>().enterEditMode();
  154. },
  155. ),
  156. ],
  157. //AI结果隐藏按钮
  158. if (canShowAI) ...[
  159. _buildTitleButton(
  160. !aiPatintController.state.ifShowAi
  161. ? FIcons.fis_eye_display
  162. : FIcons.fis_eye_hidden,
  163. !aiPatintController.state.ifShowAi
  164. ? i18nBook.measure.showAi.t
  165. : i18nBook.measure.hideAi.t,
  166. () {
  167. aiPatintController.state.ifShowAi =
  168. !aiPatintController.state.ifShowAi;
  169. setState(() {});
  170. },
  171. ),
  172. ],
  173. //进入颈动脉3D模式
  174. if (isShell && exist3DData && ifShowVidModeButton) ...[
  175. _buildTitleButton(
  176. FIcons.three_dimensional,
  177. i18nBook.measure.carotid3DMode.t,
  178. () {
  179. measure3DViewController.changeModeTo3DMode();
  180. },
  181. ),
  182. ],
  183. //从颈动脉2D返回到【测量】或【3D】
  184. if (isShell && exist3DData && ifShowModeButton) ...[
  185. _buildTitleButton(
  186. FIcons.three_dimensional,
  187. i18nBook.measure.carotid3DMode.t,
  188. () {
  189. measure3DViewController.backTo3DMode();
  190. },
  191. ),
  192. _buildTitleButton(
  193. Icons.play_circle_outline_rounded,
  194. i18nBook.measure.videoMode.t,
  195. () {
  196. measure3DViewController.backToVidMode();
  197. playerController.resetCurrentFrame();
  198. },
  199. ),
  200. ],
  201. SwitchPlayLoopButton(businessParent: widget.businessParent),
  202. ];
  203. }
  204. @override
  205. FWidget build(BuildContext context) {
  206. return FRow(
  207. children: [
  208. FExpanded(child: FContainer()),
  209. FColumn(
  210. verticalDirection: VerticalDirection.up,
  211. mainAxisAlignment: MainAxisAlignment.end,
  212. children: [
  213. QuickFWidget(
  214. Transform.translate(
  215. offset: Offset(0, -buttonGroupWidth / 2),
  216. child: FAnimatedContainer(
  217. duration: const Duration(milliseconds: 300),
  218. padding: EdgeInsets.only(
  219. top: groupContainerPadding / 2,
  220. bottom: groupContainerPadding / 2,
  221. ),
  222. decoration: BoxDecoration(
  223. borderRadius: BorderRadius.only(
  224. bottomLeft: Radius.circular(buttonGroupWidth / 2),
  225. bottomRight: Radius.circular(buttonGroupWidth / 2),
  226. ),
  227. color: Colors.grey.withOpacity(0.6),
  228. ),
  229. clipBehavior: Clip.antiAlias,
  230. width: buttonGroupWidth,
  231. height: groupHeight == groupContainerPadding
  232. ? groupHeight
  233. : null,
  234. constraints: BoxConstraints(
  235. maxHeight: Get.height - (190 / devicePixelRatio),
  236. ),
  237. child: FListView(
  238. shrinkWrap: true,
  239. children: buildTitleButtonList(),
  240. ))),
  241. ),
  242. _buildDropDownButton(),
  243. // const FSizedBox(height: 10),
  244. ],
  245. )
  246. ],
  247. );
  248. }
  249. FWidget _buildTitleButton(IconData icon, String title, Function onClick) {
  250. return FTextTooltip(
  251. position: DisplayPosition.left,
  252. textStyle: const TextStyle(fontSize: 16, color: Colors.white),
  253. height: 36 / devicePixelRatio,
  254. message: title,
  255. margin: const EdgeInsets.all(0),
  256. decoration: BoxDecoration(
  257. color: Colors.black.withOpacity(1.0),
  258. borderRadius: BorderRadius.circular(4),
  259. ),
  260. child: FContainer(
  261. height: buttonHeight,
  262. margin: EdgeInsets.only(
  263. top: buttonMargin,
  264. ),
  265. child: FInkWell(
  266. onTap: () => onClick.call(),
  267. child: Icon(
  268. icon,
  269. color: Colors.white,
  270. ),
  271. ),
  272. ),
  273. );
  274. }
  275. FWidget _buildDropDownButton() {
  276. return FInkWell(
  277. onHover: (isHover) {
  278. if (isHover && !isExpanded) {
  279. setState(() {
  280. groupHeight = maxGroupHeight;
  281. isExpanded = true;
  282. });
  283. }
  284. },
  285. onTap: () {
  286. if (isExpanded) {
  287. setState(() {
  288. groupHeight = groupContainerPadding;
  289. isExpanded = false;
  290. });
  291. } else {
  292. setState(() {
  293. groupHeight = maxGroupHeight;
  294. isExpanded = true;
  295. });
  296. }
  297. },
  298. child: FContainer(
  299. decoration: BoxDecoration(
  300. borderRadius:
  301. BorderRadius.all(Radius.circular(buttonGroupWidth / 2)),
  302. color: Colors.white,
  303. ),
  304. height: buttonGroupWidth,
  305. width: buttonGroupWidth,
  306. child: FColumn(
  307. mainAxisAlignment: MainAxisAlignment.center,
  308. children: [
  309. if (isExpanded) ...[
  310. // if (i18nBook.isCurrentChinese) ...[
  311. // FText(
  312. // i18nBook.measure.collapseGroup.t,
  313. // style: const TextStyle(fontSize: 12),
  314. // ),
  315. // ],
  316. const FIcon(Icons.arrow_drop_up)
  317. ] else ...[
  318. // if (i18nBook.isCurrentChinese) ...[
  319. // FText(
  320. // i18nBook.measure.expandGroup.t,
  321. // style: const TextStyle(fontSize: 12),
  322. // ),
  323. // ],
  324. const FIcon(Icons.arrow_drop_down)
  325. ]
  326. ],
  327. ),
  328. ));
  329. }
  330. FWidget _buildShareQRCode(String itemCurrentImage) {
  331. return FSimpleDialog(
  332. title: FText(
  333. i18nBook.remedical.share.t,
  334. style: const TextStyle(
  335. color: Colors.white,
  336. fontSize: 18,
  337. ),
  338. ),
  339. isDefault: false,
  340. children: [
  341. FContainer(
  342. width: 360,
  343. child: FContainer(
  344. padding: const EdgeInsets.only(top: 20, bottom: 20),
  345. child: QRCodeWithLogo(itemCurrentImage,
  346. codeStatement: i18nBook.remedical.scanQrCodeToShareImage.t,
  347. operationStatement: i18nBook.remedical.copyLink.t,
  348. operationSuccessCallback: () {
  349. final rpcHelper = Get.find<RPCHelper>();
  350. if (FPlatform.isWindows || FPlatform.isMacOS) {
  351. rpcHelper.rpc.platform.copyToClipboard(itemCurrentImage);
  352. } else {
  353. Clipboard.setData(ClipboardData(text: itemCurrentImage));
  354. }
  355. PromptBox.toast(i18nBook.remedical.copySuccess.t);
  356. }),
  357. ),
  358. ),
  359. ],
  360. );
  361. }
  362. }