urinalysis.dart 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353
  1. import 'dart:async';
  2. import 'package:flutter/foundation.dart';
  3. import 'package:flutter/material.dart';
  4. import 'package:get/get.dart';
  5. import 'package:vitalapp/architecture/app_parameters.dart';
  6. import 'package:vitalapp/architecture/utils/prompt_box.dart';
  7. import 'package:vitalapp/components/button.dart';
  8. import 'package:vitalapp/components/dialog_input.dart';
  9. import 'package:vitalapp/managers/device_controller_manager.dart';
  10. import 'package:vitalapp/pages/mappers/urine.dart';
  11. import 'package:vitalapp/pages/medical/widgets/device_status_position.dart';
  12. import 'package:vnote_device_plugin/consts/types.dart';
  13. import 'package:vitalapp/managers/interfaces/models/device.dart';
  14. import 'package:vitalapp/pages/medical/widgets/exam_card.dart';
  15. import 'package:vitalapp/pages/medical/controller.dart';
  16. import 'package:vitalapp/pages/medical/models/worker.dart';
  17. import 'package:vitalapp/pages/medical/widgets/device_status.dart';
  18. import 'package:vnote_device_plugin/devices/urine.dart';
  19. import 'package:vnote_device_plugin/models/exams/urine.dart';
  20. import 'package:fis_common/logger/logger.dart';
  21. /// 尿常规
  22. class Urinalysis extends StatefulWidget {
  23. const Urinalysis({
  24. super.key,
  25. this.examData,
  26. });
  27. final String? examData;
  28. @override
  29. State<Urinalysis> createState() => _ExamUrinalysisState();
  30. }
  31. class _ExamUrinalysisState extends State<Urinalysis> {
  32. var controller = Get.find<MedicalController>();
  33. bool get isPureSoftwareMode => AppParameters.data.isPureSoftwareMode;
  34. DeviceControllerManager? urinaly;
  35. UrineDeviceWorker? worker;
  36. UrineExamData? urineExamData;
  37. bool isConnectFail = false;
  38. bool isAutoTesting = false;
  39. int errorCount = 0;
  40. String? get deviceType => MedicalController.typeConvertMap[DeviceTypes.URINE];
  41. WorkerStatus connectStatus = WorkerStatus.connecting;
  42. List<Map<String, String>> urinalysis = [
  43. {"name": '尿白细胞', "key": 'LEU'},
  44. {"name": '红细胞/潜血', "key": 'BLD'},
  45. {"name": '尿亚硝酸盐', "key": 'NIT'},
  46. {"name": '酮体', "key": 'KET'},
  47. {"name": '尿胆原', "key": 'UBG'},
  48. {"name": '胆红素', "key": 'BIL'},
  49. {"name": '尿蛋白', "key": 'PRO'},
  50. {"name": '葡萄糖', "key": 'GLU'},
  51. {"name": '酸碱度', "key": 'PH'},
  52. {"name": '尿比重', "key": 'SG'},
  53. {"name": '维生素C', "key": 'VC'},
  54. ];
  55. Map _value = {};
  56. @override
  57. void initState() {
  58. WidgetsBinding.instance.addPostFrameCallback((timeStamp) {
  59. initData();
  60. currentDevice();
  61. controller.changePatient.addListener(_onPatientChange);
  62. });
  63. super.initState();
  64. }
  65. Future<void> currentDevice() async {
  66. DeviceModel? device = await controller.getDevice(DeviceTypes.URINE);
  67. if (device == null) {
  68. connectStatus = WorkerStatus.unboundDevice;
  69. return;
  70. }
  71. urinaly =
  72. DeviceControllerManager(DeviceTypes.URINE, device.model, device.mac);
  73. worker = urinaly!.worker as UrineDeviceWorker;
  74. connectStatus = urinaly!.connectStatus;
  75. loadListeners();
  76. worker!.connect();
  77. }
  78. void loadListeners() {
  79. worker!.successEvent.addListener(_onSuccess);
  80. worker!.connectErrorEvent.addListener(_onConnectFail);
  81. worker!.connectedEvent.addListener(_onConnectSuccess);
  82. worker!.disconnectedEvent.addListener(_onDisconnected);
  83. }
  84. Future<void> initData() async {
  85. if (controller.diagnosisDataValue.isNotEmpty) {
  86. if (controller.diagnosisDataValue.keys.contains("Urine")) {
  87. _value = controller.diagnosisDataValue["Urine"];
  88. } else {
  89. for (var key in controller.diagnosisDataValue.keys) {
  90. if (urinalysis.any((e) => e['key'] == key)) {
  91. _value[key] = controller.diagnosisDataValue[key];
  92. }
  93. }
  94. }
  95. } else {
  96. _value = {};
  97. }
  98. setState(() {});
  99. }
  100. void releaseListeners() {
  101. if (worker != null) {
  102. worker!.connectErrorEvent.removeListener(_onConnectFail);
  103. worker!.connectedEvent.removeListener(_onConnectSuccess);
  104. worker!.successEvent.removeListener(_onSuccess);
  105. worker!.disconnectedEvent.removeListener(_onDisconnected);
  106. // worker!.errorEvent.removeListener(_onError);
  107. }
  108. }
  109. /// 尝试重连
  110. Future<void> tryReconnect() async {
  111. //await worker.disconnect();
  112. //5s后在执行,是为了防止重连与断开事件一直相互触发
  113. Future.delayed(const Duration(seconds: 5), () async {
  114. //如果5秒后仍然是未连接状态,则重试连接
  115. if (connectStatus == WorkerStatus.disconnected ||
  116. connectStatus == WorkerStatus.connectionFailed ||
  117. connectStatus == WorkerStatus.connecting) {
  118. await disconnect();
  119. await connect();
  120. }
  121. });
  122. }
  123. Future<void> disconnect() async {
  124. await worker?.disconnect();
  125. }
  126. Future<void> connect() async {
  127. await worker?.connect();
  128. }
  129. @override
  130. void dispose() async {
  131. controller.changePatient.removeListener(_onPatientChange);
  132. urinaly?.dispose();
  133. releaseListeners();
  134. disconnect();
  135. worker?.dispose();
  136. super.dispose();
  137. }
  138. void _onSuccess(_, UrineExamData e) {
  139. setState(() {
  140. isAutoTesting = false;
  141. urineExamData = e;
  142. controller.diagnosisDataValue[deviceType!] =
  143. UrineExamDataMapper.convertUrineExamDataToMap(urineExamData!);
  144. _value = controller.diagnosisDataValue[deviceType!];
  145. controller.saveCachedRecord();
  146. connectStatus = WorkerStatus.connected;
  147. });
  148. }
  149. void _onConnectFail(sender, e) {
  150. print('连接设备失败');
  151. logger.i("设备连接失败:${worker!.mac}");
  152. if (errorCount < 3) {
  153. errorCount++;
  154. tryReconnect();
  155. } else {
  156. isConnectFail = true;
  157. }
  158. connectStatus = WorkerStatus.connectionFailed;
  159. setState(() {});
  160. }
  161. void _onConnectSuccess(sender, e) {
  162. logger.i("设备连接成功:${worker!.mac}");
  163. isConnectFail = false;
  164. errorCount = 0;
  165. connectStatus = WorkerStatus.connected;
  166. setState(() {});
  167. }
  168. void _onError(sender, e) async {
  169. PromptBox.toast("测试失败,请确定放好试纸并且试纸已浸湿尿液");
  170. isAutoTesting = false;
  171. setState(() {});
  172. }
  173. void _onDisconnected(sender, e) {
  174. print('设备连接中断 errorCount:$errorCount');
  175. logger.i("设备连接中断:${worker!.mac} errorCount:$errorCount");
  176. if (errorCount < 3) {
  177. errorCount++;
  178. tryReconnect();
  179. } else {
  180. isConnectFail = true;
  181. }
  182. connectStatus = WorkerStatus.disconnected;
  183. setState(() {});
  184. }
  185. @override
  186. Widget build(BuildContext context) {
  187. return Stack(
  188. children: [
  189. _buildUrinalysis(),
  190. if (!isPureSoftwareMode) ...[
  191. if (!isConnectFail)
  192. DeviceStatusPosition(
  193. deviceStatus: DeviceStatus(connectStatus: connectStatus),
  194. )
  195. else
  196. _buildErrorButton(),
  197. ],
  198. ],
  199. );
  200. }
  201. Widget _buildErrorButton() {
  202. return DeviceStatusPosition(
  203. deviceStatus: Row(
  204. children: [
  205. const Text(
  206. '请确认设备是否启动',
  207. style: TextStyle(fontSize: 24, color: Colors.red),
  208. ),
  209. IconButton(
  210. onPressed: () {
  211. tryReconnect();
  212. setState(() {
  213. connectStatus = WorkerStatus.connecting;
  214. isConnectFail = false;
  215. });
  216. },
  217. icon: const Icon(Icons.refresh),
  218. iconSize: 32,
  219. ),
  220. ],
  221. ),
  222. );
  223. }
  224. Widget _buildUrinalysis() {
  225. return ExamCard(
  226. title: '尿常规',
  227. content: Container(
  228. alignment: Alignment.center,
  229. padding: const EdgeInsets.only(
  230. bottom: 20,
  231. right: 30,
  232. left: 40,
  233. ),
  234. constraints: const BoxConstraints(minHeight: 50),
  235. child: GridView.count(
  236. shrinkWrap: true,
  237. childAspectRatio: kIsWeb ? 4.5 : 4,
  238. crossAxisCount: 2, // 列数为2,即两列布局
  239. children: urinalysis.map((item) {
  240. return _buildUrineItem(item);
  241. }).toList(),
  242. ),
  243. ),
  244. );
  245. }
  246. Widget _buildUrineItem(Map<String, String> urine) {
  247. String displayValue = "";
  248. var value = _value[urine['key']!];
  249. if (value is String) {
  250. displayValue = value;
  251. }
  252. return InkWell(
  253. onTap: () {
  254. _input(urine['name']!, urine['key']!);
  255. },
  256. child: Row(
  257. mainAxisAlignment: MainAxisAlignment.start,
  258. crossAxisAlignment: CrossAxisAlignment.center,
  259. children: [
  260. Container(
  261. alignment: Alignment.centerLeft,
  262. width: 100,
  263. height: 55,
  264. child: Column(
  265. crossAxisAlignment: CrossAxisAlignment.start,
  266. children: [
  267. Text(
  268. urine['key']!,
  269. style: const TextStyle(
  270. fontSize: 22,
  271. color: Colors.black,
  272. ),
  273. ),
  274. Text(
  275. '(${urine['name']!})',
  276. textAlign: TextAlign.center,
  277. style: const TextStyle(
  278. fontSize: 16,
  279. ),
  280. ),
  281. ],
  282. ),
  283. ),
  284. Container(
  285. height: 55,
  286. width: 140,
  287. alignment: Alignment.centerLeft,
  288. child: Text(
  289. displayValue,
  290. // text: _value.isEmpty ? '--' : _value,
  291. maxLines: 2,
  292. style: const TextStyle(
  293. fontSize: 30,
  294. ),
  295. ),
  296. )
  297. ],
  298. ),
  299. );
  300. }
  301. Future<void> _input(String name, String key) async {
  302. String? result = await VDialogInput(
  303. title: name,
  304. initialValue: _value[key],
  305. ).show();
  306. if (result?.isNotEmpty ?? false) {
  307. _value[key] = result ?? '';
  308. }
  309. controller.diagnosisDataValue[deviceType!] = _value;
  310. controller.saveCachedRecord();
  311. // widget.urinalysis(_value);
  312. setState(() {});
  313. }
  314. void _onPatientChange(Object sender, String e) {
  315. setState(() {
  316. _value = {};
  317. controller.diagnosisDataValue[deviceType!] = {};
  318. });
  319. }
  320. }