sample_detail_page.dart 6.3 KB

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