heart_rate.dart 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412
  1. import 'dart:async';
  2. import 'dart:convert';
  3. import 'package:flutter/material.dart';
  4. import 'package:get/get.dart';
  5. import 'package:vitalapp/managers/interfaces/data_convert.dart';
  6. import 'package:vitalapp/managers/interfaces/models/device.dart';
  7. import 'package:vitalapp/pages/medical/controllers/heart.dart';
  8. import 'package:vitalapp/pages/medical/widgets/device_status.dart';
  9. import 'package:vitalapp/pages/medical/widgets/device_status_position.dart';
  10. import 'package:vitalapp/pages/medical/widgets/ecg_view/index.dart';
  11. import 'package:vnote_device_plugin/consts/types.dart';
  12. import 'package:vnote_device_plugin/devices/heart.dart';
  13. import 'package:vitalapp/pages/check/widgets/exam_configurable/exam_card.dart';
  14. import 'package:vitalapp/pages/medical/controller.dart';
  15. import 'package:vitalapp/pages/medical/models/worker.dart';
  16. import 'package:vitalapp/pages/medical/widgets/side_bar.dart';
  17. import 'dart:ui' as ui;
  18. import 'dart:typed_data';
  19. import 'package:fis_common/logger/logger.dart';
  20. import 'package:vnote_device_plugin/models/exams/heart.dart';
  21. class HeartRate extends StatefulWidget {
  22. const HeartRate({
  23. super.key,
  24. });
  25. @override
  26. State<HeartRate> createState() => _HeartRateState();
  27. }
  28. class _HeartRateState extends State<HeartRate> {
  29. var controller = Get.find<MedicalController>();
  30. late HeartDeviceController heart;
  31. late HeartDeviceWorker worker;
  32. final dataConvertManager = Get.find<IDataConvertManager>();
  33. WorkerStatus _connectStatus = WorkerStatus.connecting;
  34. List<int> ecgPoint = [];
  35. int errorCount = 0;
  36. bool isConnectFail = false;
  37. late String _heart =
  38. controller.diagnosisDataValue['Heart']?['HEART']?.toString() ?? '';
  39. late String _assess = controller.diagnosisDataValue['Heart']?['ASSESS'] ?? '';
  40. late final String _ecg = controller.diagnosisDataValue['Heart']?['ECG'] ?? '';
  41. late final String _ecgPoint =
  42. controller.diagnosisDataValue['Heart']?['ECG_POINT'] ?? '';
  43. late final heartRate = controller.diagnosisDataValue['Heart']?['HEART'] ?? 0;
  44. String _deviceError = '';
  45. @override
  46. void initState() {
  47. WidgetsBinding.instance.addPostFrameCallback((timeStamp) {
  48. initHeart();
  49. });
  50. print(controller.diagnosisDataValue['Heart']?['ECG_POINT']);
  51. super.initState();
  52. }
  53. Future<void> connect() async {
  54. await worker.connect();
  55. }
  56. Future<void> disconnect() async {
  57. if (!(worker == null)) {
  58. await worker.disconnect();
  59. releaseListeners();
  60. }
  61. }
  62. void releaseListeners() {
  63. worker.connectErrorEvent.removeListener(_onConnectFail);
  64. worker.connectedEvent.removeListener(_onConnectSuccess);
  65. // worker.successEvent.removeListener(_onSuccess);
  66. worker.disconnectedEvent.removeListener(_onDisconnected);
  67. worker.hrValueUpdateEvent.removeListener(_onHrValueUpdate);
  68. worker.ecgValueUpdateEvent.removeListener(_onEcgValueUpdate);
  69. worker.resultReceivedEvent.removeListener(_onRCesultReceived);
  70. worker.errorEvent.removeListener(_onError);
  71. }
  72. @override
  73. void dispose() {
  74. disconnect();
  75. super.dispose();
  76. }
  77. void loadListeners() {
  78. // worker.successEvent.addListener(_onSuccess);
  79. worker.connectErrorEvent.addListener(_onConnectFail);
  80. worker.connectedEvent.addListener(_onConnectSuccess);
  81. worker.disconnectedEvent.addListener(_onDisconnected);
  82. worker.hrValueUpdateEvent.addListener(_onHrValueUpdate);
  83. worker.ecgValueUpdateEvent.addListener(_onEcgValueUpdate);
  84. worker.resultReceivedEvent.addListener(_onRCesultReceived);
  85. worker.errorEvent.addListener(_onError);
  86. worker.connect();
  87. }
  88. Future<void> currentDevice() async {
  89. DeviceModel? device = await controller.getDevice(DeviceTypes.HEART);
  90. if (device == null) {
  91. _connectStatus = WorkerStatus.unboundDevice;
  92. setState(() {});
  93. return;
  94. }
  95. heart = HeartDeviceController(device.model, device.mac);
  96. worker = heart.worker;
  97. _connectStatus = heart.connectStatus;
  98. loadListeners();
  99. }
  100. Future<void> initHeart() async {
  101. currentDevice();
  102. await initData();
  103. }
  104. Future<void> initData() async {
  105. // await Future.delayed(const Duration(milliseconds: 800));
  106. await controller.readCachedRecord();
  107. if (controller.diagnosisDataValue['Heart'] == null) {
  108. controller.diagnosisDataValue['Heart'] = {};
  109. }
  110. setState(() {});
  111. }
  112. void _onHrValueUpdate(_, int e) {
  113. logger.i('心率更新:$e');
  114. if (_deviceError.isNotEmpty) {
  115. _deviceError = '';
  116. }
  117. setState(() {
  118. _heart = e.toString();
  119. });
  120. }
  121. void _onError(_, String e) {
  122. logger.i('心率更新失败:$e');
  123. _deviceError = e;
  124. ecgPoint = [];
  125. try {
  126. EcgViewController ecgViewController = Get.find<EcgViewController>();
  127. ecgPoint = [];
  128. ecgViewController.reset();
  129. } catch (e) {}
  130. setState(() {});
  131. }
  132. void _onRCesultReceived(_, HeartExamResult e) async {
  133. logger.i('心率更新接收结果:${e.heartRate}');
  134. print(ecgPoint.toString());
  135. // _heart = e.toString();
  136. _assess = e.analysis.first;
  137. _heart = e.heartRate.toString();
  138. controller.diagnosisDataValue['Heart']?['HEART'] = e.heartRate.toString();
  139. controller.diagnosisDataValue['Heart']?['ASSESS'] =
  140. dataConvertManager.heartRateConversion(int.parse(heartRate));
  141. try {
  142. EcgViewController ecgViewController = Get.find<EcgViewController>();
  143. controller.diagnosisDataValue['Heart']?['ECG_POINT'] =
  144. jsonEncode(ecgViewController.allPoints);
  145. controller.diagnosisDataValue['Heart']?['ECG'] =
  146. await ecgViewController.getFullDataImageBase64();
  147. } catch (e) {
  148. print(e);
  149. }
  150. controller.saveCachedRecord();
  151. print(controller.diagnosisDataValue);
  152. setState(() {});
  153. }
  154. void _onEcgValueUpdate(_, List<int> e) {
  155. try {
  156. EcgViewController ecgViewController = Get.find<EcgViewController>();
  157. ecgPoint.addAll(e);
  158. if (ecgPoint.length > 125 * 3) {
  159. ecgViewController.addData(e);
  160. }
  161. } catch (e) {
  162. print(e);
  163. }
  164. }
  165. void _onConnectFail(sender, e) {
  166. logger.i("设备连接失败:${worker.mac}");
  167. if (errorCount < 3) {
  168. errorCount++;
  169. disconnect();
  170. loadListeners();
  171. } else {
  172. isConnectFail = true;
  173. }
  174. _connectStatus = WorkerStatus.connectionFailed;
  175. setState(() {});
  176. }
  177. void _onDisconnected(sender, e) {
  178. print('设备连接中断');
  179. logger.i("设备连接中断:${worker.mac}");
  180. if (errorCount < 3) {
  181. errorCount++;
  182. disconnect();
  183. loadListeners();
  184. } else {
  185. isConnectFail = true;
  186. }
  187. _connectStatus = WorkerStatus.disconnected;
  188. setState(() {});
  189. }
  190. void _onConnectSuccess(sender, e) {
  191. logger.e('_HeartRateState ${worker.mac}, 设备连接成功');
  192. isConnectFail = false;
  193. errorCount = 0;
  194. _connectStatus = WorkerStatus.connected;
  195. setState(() {});
  196. }
  197. Future<String> createEcgImageBase64(List<int> points) async {
  198. final int width = points.length ~/ 2;
  199. const int height = 240;
  200. final ui.PictureRecorder recorder = ui.PictureRecorder();
  201. final Canvas canvas = Canvas(recorder);
  202. final Path path = Path();
  203. final Paint paint = Paint();
  204. const double strokeWidth = 1.5;
  205. paint.color = Colors.green;
  206. paint.style = PaintingStyle.stroke;
  207. paint.isAntiAlias = true;
  208. paint.strokeWidth = strokeWidth;
  209. path.reset();
  210. int pointCount = points.length;
  211. if (pointCount > 0) {
  212. path.moveTo(0, points[0].toDouble());
  213. int i = 0;
  214. int j = pointCount ~/ 2;
  215. while (i < j) {
  216. path.lineTo(i.toDouble(), points[i].toDouble());
  217. path.lineTo((i + 1).toDouble(), points[i + 1].toDouble());
  218. i += 2;
  219. }
  220. canvas.drawPath(path, paint);
  221. }
  222. final ui.Picture picture = recorder.endRecording();
  223. final ui.Image image = await picture.toImage(width, height);
  224. final ByteData? byteData =
  225. await image.toByteData(format: ui.ImageByteFormat.png);
  226. final Uint8List pngBytes = byteData!.buffer.asUint8List();
  227. final String base64Image = base64Encode(pngBytes);
  228. return base64Image;
  229. }
  230. /// 需要封装一下
  231. Widget _buildErrorButton() {
  232. return DeviceStatusPosition(
  233. deviceStatus: Row(
  234. children: [
  235. const Text(
  236. '请确认设备是否启动',
  237. style: TextStyle(fontSize: 24, color: Colors.red),
  238. ),
  239. IconButton(
  240. onPressed: () {
  241. worker.connect();
  242. setState(() {
  243. _connectStatus = WorkerStatus.connecting;
  244. isConnectFail = false;
  245. });
  246. },
  247. icon: const Icon(Icons.refresh),
  248. iconSize: 32,
  249. ),
  250. ],
  251. ),
  252. );
  253. }
  254. @override
  255. Widget build(BuildContext context) {
  256. return Stack(
  257. children: [
  258. ExamCard(
  259. titleText: const SizedBox(),
  260. // clickCard: () {},
  261. content: Column(
  262. mainAxisAlignment: MainAxisAlignment.start,
  263. children: [
  264. Row(
  265. mainAxisAlignment: MainAxisAlignment.end,
  266. children: [
  267. const SideBar(
  268. title: '心率',
  269. value: '',
  270. unit: '',
  271. ),
  272. const Expanded(child: SizedBox()),
  273. Text(
  274. _heart.isEmpty ? '--' : _heart,
  275. style: const TextStyle(
  276. fontSize: 60,
  277. color: Colors.black,
  278. ),
  279. ),
  280. const Text(
  281. " bpm",
  282. style: TextStyle(fontSize: 25),
  283. ),
  284. const SizedBox(
  285. width: 15,
  286. )
  287. ],
  288. ),
  289. if (_deviceError.isNotEmpty)
  290. Row(
  291. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  292. crossAxisAlignment: CrossAxisAlignment.start,
  293. children: [
  294. Container(
  295. padding: const EdgeInsets.symmetric(
  296. horizontal: 30,
  297. ),
  298. child: const Text(
  299. '心率测量出错',
  300. style: TextStyle(
  301. fontSize: 25,
  302. ),
  303. ),
  304. ),
  305. Container(
  306. padding: const EdgeInsets.symmetric(
  307. horizontal: 50,
  308. ),
  309. child: Text(
  310. _deviceError,
  311. style: const TextStyle(
  312. fontSize: 24,
  313. ),
  314. ),
  315. ),
  316. ],
  317. ),
  318. if (_assess.isNotEmpty)
  319. Row(
  320. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  321. crossAxisAlignment: CrossAxisAlignment.start,
  322. children: [
  323. Container(
  324. padding: const EdgeInsets.symmetric(
  325. horizontal: 30,
  326. ),
  327. child: const Text(
  328. '心率评估',
  329. style: TextStyle(
  330. fontSize: 25,
  331. ),
  332. ),
  333. ),
  334. Container(
  335. padding: const EdgeInsets.symmetric(
  336. horizontal: 50,
  337. ),
  338. child: Text(
  339. dataConvertManager
  340. .heartRateConversion(int.parse(heartRate)),
  341. style: const TextStyle(
  342. fontSize: 24,
  343. ),
  344. ),
  345. ),
  346. ],
  347. ),
  348. const SizedBox(
  349. height: 10,
  350. ),
  351. SizedBox(
  352. height: 240,
  353. child: LayoutBuilder(builder: (context, constraints) {
  354. List<int> initData = [];
  355. if (controller.diagnosisDataValue['Heart']?['ECG_POINT'] !=
  356. null) {
  357. initData = jsonDecode(controller
  358. .diagnosisDataValue['Heart']?['ECG_POINT'])
  359. .cast<int>();
  360. }
  361. return EcgView(
  362. width: constraints.maxWidth,
  363. height: constraints.maxHeight,
  364. initData: initData, // TODO: 传入初始数据
  365. );
  366. }),
  367. )
  368. ],
  369. )),
  370. if (!isConnectFail)
  371. DeviceStatusPosition(
  372. deviceStatus: DeviceStatus(connectStatus: _connectStatus),
  373. )
  374. else
  375. _buildErrorButton(),
  376. ],
  377. );
  378. }
  379. }