sample_detail_page.dart 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237
  1. import 'package:flutter/material.dart';
  2. import 'package:chicken_farm/components/vb_app_bar.dart';
  3. import 'package:chicken_farm/modes/experiment/sample/sample.dart';
  4. import 'package:chicken_farm/modes/experiment/sample/sample_flow_log.dart';
  5. import 'package:chicken_farm/apis/experiment/_sample.dart';
  6. import 'package:chicken_farm/core/utils/toast.dart';
  7. import 'package:chicken_farm/core/utils/logger.dart';
  8. import 'package:chicken_farm/core/utils/datetime_util.dart';
  9. class SampleDetailPage extends StatefulWidget {
  10. final String id;
  11. const SampleDetailPage({super.key, required this.id});
  12. @override
  13. State<SampleDetailPage> createState() => _SampleDetailPageState();
  14. }
  15. class _SampleDetailPageState extends State<SampleDetailPage> {
  16. SampleModel? _sample;
  17. List<SampleFlowLogModel> _flowLogs = [];
  18. bool _isLoading = true;
  19. @override
  20. void initState() {
  21. super.initState();
  22. // 将 _loadSampleData 调用推迟到下一帧,避免在构建过程中修改状态
  23. WidgetsBinding.instance.addPostFrameCallback((_) {
  24. _loadSampleData();
  25. });
  26. }
  27. Future<void> _loadSampleData() async {
  28. try {
  29. final sampleFuture = SampleApi().querySample(widget.id);
  30. final flowLogsFuture = SampleApi().queryFlowLogs(widget.id);
  31. final results = await Future.wait([sampleFuture, flowLogsFuture]);
  32. setState(() {
  33. _sample = results[0] as SampleModel;
  34. _flowLogs = results[1] as List<SampleFlowLogModel>;
  35. _isLoading = false;
  36. });
  37. } catch (e, stackTrace) {
  38. logger.e('加载样品详情失败: $e, error: e, stackTrace: $stackTrace');
  39. setState(() {
  40. _isLoading = false;
  41. });
  42. ToastUtil.error('加载样品详情失败: $e');
  43. }
  44. }
  45. @override
  46. Widget build(BuildContext context) {
  47. return Scaffold(
  48. appBar: const VberAppBar(title: '样品详情', showLeftButton: true),
  49. body: _buildBody(),
  50. );
  51. }
  52. Widget _buildBody() {
  53. if (_isLoading) {
  54. return const Center(child: CircularProgressIndicator());
  55. }
  56. if (_sample == null) {
  57. return const Center(
  58. child: Text(
  59. '无法加载样品信息',
  60. style: TextStyle(fontSize: 16, color: Colors.grey),
  61. ),
  62. );
  63. }
  64. return SingleChildScrollView(
  65. child: Column(
  66. crossAxisAlignment: CrossAxisAlignment.start,
  67. children: [
  68. _buildSampleInfo(),
  69. const Divider(),
  70. _buildFlowLogsTitle(),
  71. _buildFlowLogsList(),
  72. ],
  73. ),
  74. );
  75. }
  76. Widget _buildSampleInfo() {
  77. return Container(
  78. width: double.infinity,
  79. padding: const EdgeInsets.all(16.0),
  80. child: Card(
  81. child: Padding(
  82. padding: const EdgeInsets.all(16.0),
  83. child: Column(
  84. crossAxisAlignment: CrossAxisAlignment.start,
  85. children: [
  86. Row(
  87. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  88. children: [
  89. Text(
  90. _sample!.sampleName,
  91. style: Theme.of(context).textTheme.titleLarge,
  92. ),
  93. Chip(
  94. label: Text(
  95. _getSampleStatusText(_sample!.sampleStatus),
  96. style: TextStyle(
  97. color: _getSampleStatusColor(_sample!.sampleStatus),
  98. ),
  99. ),
  100. ),
  101. ],
  102. ),
  103. const SizedBox(height: 16),
  104. _buildInfoRow('样品编号', _sample!.id.toString()),
  105. _buildInfoRow('样品类型', _getSampleTypeText(_sample!.sampleType)),
  106. _buildInfoRow('取样批次', _sample!.batchNum),
  107. _buildInfoRow('取样翅号', _sample!.wingTagNum),
  108. _buildInfoRow(
  109. '取样时间',
  110. DateTimeUtil.toStandardString(_sample!.sampleTime),
  111. ),
  112. _buildInfoRow('取样人', _sample!.createByName),
  113. const SizedBox(height: 8),
  114. Text('样品描述: ${_sample!.description}'),
  115. ],
  116. ),
  117. ),
  118. ),
  119. );
  120. }
  121. Widget _buildInfoRow(String label, String value) {
  122. return Padding(
  123. padding: const EdgeInsets.symmetric(vertical: 4.0),
  124. child: Row(
  125. children: [
  126. SizedBox(
  127. width: 80,
  128. child: Text(
  129. '$label:',
  130. style: const TextStyle(fontWeight: FontWeight.bold),
  131. ),
  132. ),
  133. Expanded(child: Text(value)),
  134. ],
  135. ),
  136. );
  137. }
  138. Widget _buildFlowLogsTitle() {
  139. return Padding(
  140. padding: const EdgeInsets.all(16.0),
  141. child: Text('流转记录', style: Theme.of(context).textTheme.titleLarge),
  142. );
  143. }
  144. Widget _buildFlowLogsList() {
  145. if (_flowLogs.isEmpty) {
  146. return const Center(
  147. child: Padding(
  148. padding: EdgeInsets.all(32.0),
  149. child: Text(
  150. '暂无流转记录',
  151. style: TextStyle(fontSize: 16, color: Colors.grey),
  152. ),
  153. ),
  154. );
  155. }
  156. return ListView.builder(
  157. shrinkWrap: true,
  158. physics: const NeverScrollableScrollPhysics(),
  159. itemCount: _flowLogs.length,
  160. itemBuilder: (context, index) {
  161. final log = _flowLogs[index];
  162. return _buildFlowLogItem(log);
  163. },
  164. );
  165. }
  166. Widget _buildFlowLogItem(SampleFlowLogModel log) {
  167. return Card(
  168. margin: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
  169. child: ListTile(
  170. title: Text(log.handlerName),
  171. subtitle: Column(
  172. crossAxisAlignment: CrossAxisAlignment.start,
  173. children: [
  174. Text('查询时间: ${DateTimeUtil.toStandardString(log.handleTime)}'),
  175. ],
  176. ),
  177. ),
  178. );
  179. }
  180. String _getSampleStatusText(int status) {
  181. switch (status) {
  182. case 0:
  183. return '待处理';
  184. case 1:
  185. return '处理中';
  186. case 2:
  187. return '已完成';
  188. default:
  189. return '未知状态';
  190. }
  191. }
  192. Color _getSampleStatusColor(int status) {
  193. switch (status) {
  194. case 0:
  195. return Colors.orange;
  196. case 1:
  197. return Colors.blue;
  198. case 2:
  199. return Colors.green;
  200. default:
  201. return Colors.grey;
  202. }
  203. }
  204. String _getSampleTypeText(int type) {
  205. switch (type) {
  206. case 1:
  207. return '血液样品';
  208. case 2:
  209. return '组织样品';
  210. case 3:
  211. return '粪便样品';
  212. default:
  213. return '未知类型';
  214. }
  215. }
  216. }