sample_detail_page.dart 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. import 'package:chicken_farm/components/vb_dict_label.dart';
  2. import 'package:flutter/material.dart';
  3. import 'package:chicken_farm/components/vb_app_bar.dart';
  4. import 'package:chicken_farm/modes/experiment/sample/sample.dart';
  5. import 'package:chicken_farm/modes/experiment/sample/sample_flow_log.dart';
  6. import 'package:chicken_farm/apis/experiment/_sample.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. }
  43. }
  44. @override
  45. Widget build(BuildContext context) {
  46. return Scaffold(
  47. appBar: const VberAppBar(title: '样品详情', showLeftButton: true),
  48. body: _buildBody(),
  49. );
  50. }
  51. Widget _buildBody() {
  52. if (_isLoading) {
  53. return const Center(child: CircularProgressIndicator());
  54. }
  55. if (_sample == null) {
  56. return const Center(
  57. child: Text(
  58. '无法加载样品信息',
  59. style: TextStyle(fontSize: 16, color: Colors.grey),
  60. ),
  61. );
  62. }
  63. return SingleChildScrollView(
  64. child: Column(
  65. crossAxisAlignment: CrossAxisAlignment.start,
  66. children: [
  67. _buildSampleInfo(),
  68. const Divider(),
  69. _buildFlowLogsTitle(),
  70. _buildFlowLogsList(),
  71. ],
  72. ),
  73. );
  74. }
  75. Widget _buildSampleInfo() {
  76. return Container(
  77. width: double.infinity,
  78. padding: const EdgeInsets.all(16.0),
  79. child: Card(
  80. child: Padding(
  81. padding: const EdgeInsets.all(16.0),
  82. child: Column(
  83. crossAxisAlignment: CrossAxisAlignment.start,
  84. children: [
  85. Row(
  86. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  87. children: [
  88. Text(
  89. _sample!.sampleName,
  90. style: Theme.of(context).textTheme.titleLarge,
  91. ),
  92. VberDictLabel(
  93. dictType: "experiment_sample_status",
  94. value: _sample!.sampleStatus.toString(),
  95. ),
  96. ],
  97. ),
  98. const SizedBox(height: 16),
  99. _buildInfoRow('样品编号', _sample!.id.toString()),
  100. _buildInfoRow(
  101. '样品类型',
  102. VberDictLabel(
  103. dictType: "experiment_sample_type",
  104. value: _sample!.sampleType.toString(),
  105. ),
  106. ),
  107. _buildInfoRow('取样批次', _sample!.batchNum),
  108. _buildInfoRow('取样翅号', _sample!.wingTagNum),
  109. _buildInfoRow(
  110. '取样时间',
  111. DateTimeUtil.toStandardString(_sample!.sampleTime),
  112. ),
  113. _buildInfoRow('取样人', _sample!.createByName),
  114. const SizedBox(height: 8),
  115. Text('样品描述: ${_sample!.description}'),
  116. ],
  117. ),
  118. ),
  119. ),
  120. );
  121. }
  122. Widget _buildInfoRow(String label, dynamic value) {
  123. Widget valueWidget;
  124. if (value is String) {
  125. valueWidget = Text(value);
  126. } else if (value is Widget) {
  127. valueWidget = value;
  128. } else {
  129. valueWidget = Text(value.toString());
  130. }
  131. return Padding(
  132. padding: const EdgeInsets.symmetric(vertical: 4.0),
  133. child: Row(
  134. children: [
  135. SizedBox(
  136. width: 80,
  137. child: Text(
  138. '$label:',
  139. style: const TextStyle(fontWeight: FontWeight.bold),
  140. ),
  141. ),
  142. Expanded(child: valueWidget),
  143. ],
  144. ),
  145. );
  146. }
  147. Widget _buildFlowLogsTitle() {
  148. return Padding(
  149. padding: const EdgeInsets.all(16.0),
  150. child: Text('流转记录', style: Theme.of(context).textTheme.titleLarge),
  151. );
  152. }
  153. Widget _buildFlowLogsList() {
  154. if (_flowLogs.isEmpty) {
  155. return const Center(
  156. child: Padding(
  157. padding: EdgeInsets.all(32.0),
  158. child: Text(
  159. '暂无流转记录',
  160. style: TextStyle(fontSize: 16, color: Colors.grey),
  161. ),
  162. ),
  163. );
  164. }
  165. return ListView.builder(
  166. shrinkWrap: true,
  167. physics: const NeverScrollableScrollPhysics(),
  168. itemCount: _flowLogs.length,
  169. itemBuilder: (context, index) {
  170. final log = _flowLogs[index];
  171. return _buildFlowLogItem(log);
  172. },
  173. );
  174. }
  175. Widget _buildFlowLogItem(SampleFlowLogModel log) {
  176. return Card(
  177. margin: const EdgeInsets.symmetric(horizontal: 16, vertical: 8),
  178. child: ListTile(
  179. title: Text(log.handlerName),
  180. subtitle: Column(
  181. crossAxisAlignment: CrossAxisAlignment.start,
  182. children: [
  183. Text('查询时间: ${DateTimeUtil.toStandardString(log.handleTime)}'),
  184. ],
  185. ),
  186. ),
  187. );
  188. }
  189. }