blood_sugar.dart 7.7 KB

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