blood_sugar.dart 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. import 'package:flutter/material.dart';
  2. import 'package:get/get.dart';
  3. import 'package:vitalapp/pages/medical/widgets/device_status_position.dart';
  4. import 'package:vnote_device_plugin/consts/types.dart';
  5. import 'package:vnote_device_plugin/devices/sugar.dart';
  6. import 'package:vnote_device_plugin/models/exams/error.dart';
  7. import 'package:vnote_device_plugin/models/exams/sugar.dart';
  8. import 'package:vitalapp/managers/interfaces/models/device.dart';
  9. import 'package:vitalapp/pages/check/widgets/exam_configurable/exam_card.dart';
  10. import 'package:vitalapp/components/dialog_number.dart';
  11. import 'package:vitalapp/pages/medical/controller.dart';
  12. import 'package:vitalapp/pages/medical/controllers/sugar.dart';
  13. import 'package:vitalapp/pages/medical/models/worker.dart';
  14. import 'package:vitalapp/pages/medical/widgets/device_status.dart';
  15. import 'package:fis_common/logger/logger.dart';
  16. // ignore: must_be_immutable
  17. class BloodSugar extends StatefulWidget {
  18. const BloodSugar({
  19. super.key,
  20. });
  21. @override
  22. State<BloodSugar> createState() => _ExamBloodSugarState();
  23. }
  24. class _ExamBloodSugarState extends State<BloodSugar> {
  25. var controller = Get.find<MedicalController>();
  26. late SugarDeviceController sugar;
  27. SugarDeviceWorker? worker;
  28. bool isConnectFail = false;
  29. WorkerStatus _connectStatus = WorkerStatus.connecting;
  30. int errorCount = 0;
  31. String _bloodStatus = '';
  32. late String _value = controller.diagnosisDataValue['GLU']?['sugar'] ?? '';
  33. @override
  34. void initState() {
  35. WidgetsBinding.instance.addPostFrameCallback((timeStamp) {
  36. initSugar();
  37. });
  38. super.initState();
  39. }
  40. void loadListeners() {
  41. worker!.successEvent.addListener(_onSuccess);
  42. worker!.connectErrorEvent.addListener(_onConnectFail);
  43. worker!.connectedEvent.addListener(_onConnectSuccess);
  44. worker!.disconnectedEvent.addListener(_onDisconnected);
  45. worker!.warnEvent.addListener(_onWarn);
  46. worker!.statusEvent.addListener(_onstatus);
  47. worker!.connect();
  48. }
  49. Future<void> currentDevice() async {
  50. DeviceModel? device = await controller.getDevice(DeviceTypes.SUGAR);
  51. if (device == null) {
  52. _connectStatus = WorkerStatus.unboundDevice;
  53. worker = null;
  54. setState(() {});
  55. return;
  56. }
  57. sugar = SugarDeviceController(device.model, device.mac);
  58. worker = sugar.worker;
  59. _connectStatus = sugar.connectStatus;
  60. loadListeners();
  61. }
  62. Future<void> initSugar() async {
  63. currentDevice();
  64. await initData();
  65. }
  66. Future<void> initData() async {
  67. await controller.readCachedRecord();
  68. _value = controller.diagnosisDataValue['GLU']?['sugar'] ?? '';
  69. setState(() {});
  70. }
  71. Future<void> disconnect() async {
  72. if (worker != null) {
  73. await worker!.disconnect();
  74. releaseListeners();
  75. }
  76. }
  77. void releaseListeners() {
  78. worker!.connectErrorEvent.removeListener(_onConnectFail);
  79. worker!.connectedEvent.removeListener(_onConnectSuccess);
  80. worker!.successEvent.removeListener(_onSuccess);
  81. worker!.disconnectedEvent.removeListener(_onDisconnected);
  82. worker!.warnEvent.removeListener(_onWarn);
  83. worker!.statusEvent.removeListener(_onstatus);
  84. }
  85. @override
  86. void dispose() {
  87. disconnect();
  88. worker?.dispose();
  89. super.dispose();
  90. }
  91. void _onWarn(_, ExamError e) {
  92. _bloodStatus = e.message ?? '';
  93. setState(() {});
  94. }
  95. void _onstatus(_, ExamError e) {
  96. _bloodStatus = e.message ?? '';
  97. setState(() {});
  98. }
  99. void _onSuccess(_, SugarExamData e) {
  100. setState(() {
  101. _value = e.sugar.toString();
  102. controller.diagnosisDataValue['GLU'] = {'sugar': _value};
  103. controller.saveCachedRecord();
  104. _connectStatus = WorkerStatus.connected;
  105. });
  106. }
  107. void _onConnectFail(sender, e) {
  108. print('连接设备失败');
  109. logger.i("连接设备失败:${worker!.mac}");
  110. if (errorCount < 3) {
  111. errorCount++;
  112. disconnect();
  113. loadListeners();
  114. } else {
  115. isConnectFail = true;
  116. }
  117. _bloodStatus = '';
  118. _connectStatus = WorkerStatus.connectionFailed;
  119. setState(() {});
  120. }
  121. void _onConnectSuccess(sender, e) {
  122. logger.i("设备连接成功:${worker!.mac}");
  123. _connectStatus = WorkerStatus.connected;
  124. errorCount = 0;
  125. _bloodStatus = '';
  126. isConnectFail = false;
  127. setState(() {});
  128. }
  129. void _onDisconnected(sender, e) {
  130. print('设备连接中断');
  131. logger.i("设备连接中断:${worker!.mac}");
  132. if (errorCount < 3) {
  133. errorCount++;
  134. disconnect();
  135. loadListeners();
  136. } else {
  137. isConnectFail = true;
  138. }
  139. _bloodStatus = '';
  140. _connectStatus = WorkerStatus.disconnected;
  141. setState(() {});
  142. }
  143. // void _onConnectFail(sender, e) {
  144. // print('连接设备失败');
  145. // isConnectFail = true;
  146. // _bloodStatus = '';
  147. // _connectStatus = WorkerStatus.connectionFailed;
  148. // setState(() {});
  149. // }
  150. // void _onDisconnected(sender, e) {
  151. // print('设备连接中断');
  152. // isConnectFail = true;
  153. // _bloodStatus = '';
  154. // _connectStatus = WorkerStatus.disconnected;
  155. // setState(() {});
  156. // }
  157. // void _onConnectSuccess(sender, e) {
  158. // _connectStatus = WorkerStatus.connected;
  159. // _bloodStatus = '';
  160. // isConnectFail = false;
  161. // setState(() {});
  162. // }
  163. @override
  164. Widget build(BuildContext context) {
  165. return Stack(
  166. children: [
  167. _buildSugar(),
  168. if (!isConnectFail)
  169. DeviceStatusPosition(
  170. deviceStatus: DeviceStatus(connectStatus: _connectStatus),
  171. )
  172. else
  173. _buildErrorButton(),
  174. Positioned(
  175. right: 10,
  176. top: 60,
  177. child: Text(
  178. _bloodStatus,
  179. style: const TextStyle(
  180. fontSize: 18,
  181. ),
  182. ),
  183. ),
  184. ],
  185. );
  186. }
  187. /// 需要封装一下
  188. Widget _buildErrorButton() {
  189. return DeviceStatusPosition(
  190. deviceStatus: Row(
  191. children: [
  192. const Text(
  193. '请确认设备是否启动',
  194. style: TextStyle(fontSize: 24, color: Colors.red),
  195. ),
  196. IconButton(
  197. onPressed: () {
  198. worker?.connect();
  199. setState(() {
  200. _connectStatus = WorkerStatus.connecting;
  201. isConnectFail = false;
  202. });
  203. },
  204. icon: const Icon(Icons.refresh),
  205. iconSize: 32,
  206. ),
  207. ],
  208. ),
  209. );
  210. }
  211. Widget _buildSugar() {
  212. return ExamCard(
  213. title: '',
  214. clickCard: () {
  215. _inputSugar();
  216. },
  217. content: Row(
  218. children: [
  219. const SizedBox(
  220. width: 25,
  221. ),
  222. const Text(
  223. '血糖',
  224. style: TextStyle(fontSize: 25),
  225. ),
  226. const Expanded(
  227. child: SizedBox(),
  228. ),
  229. SizedBox(
  230. height: 150,
  231. child: FittedBox(
  232. child: Row(
  233. mainAxisAlignment: MainAxisAlignment.end,
  234. crossAxisAlignment: CrossAxisAlignment.center,
  235. children: [
  236. Text(
  237. _value.isEmpty ? '--' : _value,
  238. style: const TextStyle(
  239. fontSize: 80,
  240. color: Colors.black,
  241. ),
  242. ),
  243. const Text(
  244. ' mmol/L',
  245. style: TextStyle(fontSize: 25),
  246. ),
  247. const SizedBox(
  248. width: 15,
  249. )
  250. ],
  251. ),
  252. ),
  253. ),
  254. ],
  255. ),
  256. );
  257. }
  258. Future<void> _inputSugar() async {
  259. String? result = await VDialogNumber(
  260. title: '血糖',
  261. initialValue: _value,
  262. ).show();
  263. if (result?.isNotEmpty ?? false) {
  264. _value = result ?? '';
  265. controller.diagnosisDataValue['GLU'] = {'sugar': _value};
  266. controller.saveCachedRecord();
  267. }
  268. setState(() {});
  269. }
  270. }