batch_create_page.dart 10.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  1. import 'package:chicken_farm/core/utils/logger.dart';
  2. import 'package:chicken_farm/apis/index.dart';
  3. import 'package:chicken_farm/components/vb_rfid_field.dart';
  4. import 'package:chicken_farm/components/vb_search_select.dart';
  5. import 'package:chicken_farm/components/vb_select.dart';
  6. import 'package:chicken_farm/modes/breeding/batch.dart';
  7. import 'package:chicken_farm/modes/breeding/family.dart';
  8. import 'package:chicken_farm/modes/rfid/rfid_model.dart';
  9. import 'package:flutter/material.dart';
  10. import 'package:chicken_farm/components/vb_app_bar.dart';
  11. import 'package:chicken_farm/core/utils/toast.dart';
  12. class BatchCreatePage extends StatefulWidget {
  13. const BatchCreatePage({super.key});
  14. @override
  15. State<BatchCreatePage> createState() => _BatchCreatePageState();
  16. }
  17. class _BatchCreatePageState extends State<BatchCreatePage> {
  18. final List<String> _rfids = [];
  19. String? _selectedBatchNum;
  20. String? _selectedFamilyId;
  21. @override
  22. Widget build(BuildContext context) {
  23. return Scaffold(
  24. appBar: const VberAppBar(title: '个体绑定', showLeftButton: true),
  25. body: Padding(
  26. padding: const EdgeInsets.all(16.0),
  27. child: Column(
  28. crossAxisAlignment: CrossAxisAlignment.start,
  29. children: [
  30. // 批次选择
  31. _buildBatchSearchSelect(),
  32. const SizedBox(height: 10),
  33. // 家系号选择
  34. _buildFamilySearchSelect(),
  35. const SizedBox(height: 20),
  36. // 电子编号区域
  37. _buildRfidSection(),
  38. const SizedBox(height: 20),
  39. // 提交按钮
  40. SizedBox(
  41. width: double.infinity,
  42. child: ElevatedButton(
  43. onPressed:
  44. _rfids.isNotEmpty &&
  45. _selectedBatchNum != null &&
  46. _selectedFamilyId != null
  47. ? _handleSubmit
  48. : null,
  49. style: ElevatedButton.styleFrom(
  50. backgroundColor:
  51. _rfids.isNotEmpty &&
  52. _selectedBatchNum != null &&
  53. _selectedFamilyId != null
  54. ? Colors.blue
  55. : Colors.grey,
  56. foregroundColor: Colors.white,
  57. ),
  58. child: const Text('提交'),
  59. ),
  60. ),
  61. const SizedBox(height: 20),
  62. // 已识别的电子编号列表
  63. if (_rfids.isNotEmpty) ...[
  64. Row(
  65. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  66. children: [
  67. const Text(
  68. '已识别的电子编号',
  69. style: TextStyle(fontWeight: FontWeight.bold),
  70. ),
  71. IconButton(
  72. icon: const Icon(Icons.clear, size: 18),
  73. onPressed: _clearRfids,
  74. tooltip: '清空编号',
  75. ),
  76. ],
  77. ),
  78. const SizedBox(height: 10),
  79. Expanded(
  80. child: Container(
  81. padding: const EdgeInsets.all(10),
  82. decoration: BoxDecoration(
  83. border: Border.all(color: Colors.grey),
  84. borderRadius: BorderRadius.circular(8),
  85. ),
  86. child: ListView.builder(
  87. padding: EdgeInsets.zero,
  88. itemCount: _rfids.length,
  89. itemBuilder: (context, index) {
  90. return ListTile(
  91. visualDensity: VisualDensity.compact,
  92. contentPadding: const EdgeInsets.symmetric(
  93. horizontal: 2,
  94. vertical: 0,
  95. ),
  96. title: Text(_rfids[index]),
  97. trailing: IconButton(
  98. icon: const Icon(
  99. Icons.delete,
  100. size: 18,
  101. color: Colors.red,
  102. ),
  103. onPressed: () => _removeRfid(index),
  104. ),
  105. );
  106. },
  107. ),
  108. ),
  109. ),
  110. ],
  111. const SizedBox(height: 20),
  112. ],
  113. ),
  114. ),
  115. );
  116. }
  117. Widget _buildBatchSearchSelect() {
  118. return Column(
  119. crossAxisAlignment: CrossAxisAlignment.start,
  120. children: [
  121. const Text('选择批次', style: TextStyle(fontWeight: FontWeight.bold)),
  122. const SizedBox(height: 10),
  123. Container(
  124. padding: const EdgeInsets.fromLTRB(16, 2, 16, 2),
  125. decoration: BoxDecoration(
  126. border: Border.all(color: Colors.grey),
  127. borderRadius: BorderRadius.circular(8),
  128. ),
  129. child: VberSearchSelect<BatchModel>(
  130. searchApi: ({required dynamic queryParams}) async {
  131. final result = await apis.breeding.queryApi.queryPageBatchs(
  132. queryParams,
  133. );
  134. return {'rows': result.rows, 'total': result.total};
  135. },
  136. converter: (BatchModel data) {
  137. return SelectOption(
  138. label: '${data.batchNum}(${data.batchName})',
  139. value: data.batchNum,
  140. extra: data,
  141. );
  142. },
  143. value: _selectedBatchNum,
  144. hint: '批次号',
  145. onChanged: (String? value) {
  146. setState(() {
  147. _selectedBatchNum = value;
  148. // 当切换批次时,重置后续选项和数据
  149. _selectedFamilyId = null;
  150. });
  151. // // 获取翅号数据
  152. // if (value != null) {
  153. // // 使用Future.microtask确保在下一个微任务中加载数据,避免在构建过程中修改状态
  154. // Future.microtask(() => _loadWingTags(value));
  155. // }
  156. },
  157. hideUnderline: true,
  158. ),
  159. ),
  160. ],
  161. );
  162. }
  163. Widget _buildFamilySearchSelect() {
  164. return Column(
  165. crossAxisAlignment: CrossAxisAlignment.start,
  166. children: [
  167. const Text('选择家系号', style: TextStyle(fontWeight: FontWeight.bold)),
  168. const SizedBox(height: 10),
  169. Container(
  170. padding: const EdgeInsets.fromLTRB(16, 2, 16, 2),
  171. decoration: BoxDecoration(
  172. border: Border.all(color: Colors.grey),
  173. borderRadius: BorderRadius.circular(8),
  174. ),
  175. child: VberSearchSelect<FamilyModel>(
  176. searchApi: ({required dynamic queryParams}) async {
  177. final result = await apis.breeding.queryApi.queryPageFamilys(
  178. queryParams,
  179. );
  180. return {'rows': result.rows, 'total': result.total};
  181. },
  182. converter: (FamilyModel data) {
  183. return SelectOption(
  184. label: data.familyNum,
  185. value: data.id.toString(),
  186. extra: data,
  187. );
  188. },
  189. value: _selectedFamilyId,
  190. hint: '家系号',
  191. onChanged: (String? value) {
  192. setState(() {
  193. _selectedFamilyId = value;
  194. });
  195. },
  196. hideUnderline: true,
  197. ),
  198. ),
  199. ],
  200. );
  201. }
  202. Widget _buildRfidSection() {
  203. return VberRfidField(
  204. rfids: _rfids,
  205. onRfidsScanned: (List<RfidModel> scannedRfids) {
  206. // 过滤出未存在的RFID
  207. final newRfids = scannedRfids
  208. .where((rfid) => !_rfids.contains(rfid.uid))
  209. .toList();
  210. if (newRfids.isNotEmpty) {
  211. setState(() {
  212. // 将新的RFID添加到列表中
  213. for (var rfid in newRfids) {
  214. _rfids.insert(0, rfid.uid);
  215. }
  216. });
  217. ToastUtil.success("新增 ${newRfids.length} 枚电子编号");
  218. // if (newRfids.length == scannedRfids.length) {
  219. // // 全都是新添加的
  220. // ToastUtil.success("成功添加 ${newRfids.length} 枚新的电子编号");
  221. // } else {
  222. // // 部分是重复的
  223. // ToastUtil.info("新增 ${newRfids.length} 枚电子编号");
  224. // // ,${scannedRfids.length - newRfids.length} 枚已存在
  225. // }
  226. } else {
  227. // 所有RFID都已存在
  228. ToastUtil.info("电子编号已存在");
  229. }
  230. },
  231. multiple: true,
  232. label: '电子编号',
  233. multiplePlaceholder: '未识别',
  234. multipleFormat: '已识别 %d 枚电子编号',
  235. );
  236. }
  237. // 移除指定索引的电子编号
  238. void _removeRfid(int index) {
  239. setState(() {
  240. _rfids.removeAt(index);
  241. });
  242. }
  243. // 清空所有已识别的电子编号
  244. void _clearRfids() {
  245. setState(() {
  246. _rfids.clear();
  247. });
  248. ToastUtil.info('已清空所有电子编号');
  249. }
  250. // 提交数据
  251. void _handleSubmit() {
  252. final data = _rfids.map((rfid) {
  253. return {
  254. 'rfid': rfid,
  255. 'batchNum': _selectedBatchNum,
  256. 'familyId': _selectedFamilyId,
  257. };
  258. }).toList();
  259. apis.breeding.submitApi
  260. .create(data)
  261. .then((_) {
  262. if (mounted) {
  263. ScaffoldMessenger.of(context).showSnackBar(
  264. const SnackBar(
  265. content: Text('批量绑定个体成功'),
  266. backgroundColor: Colors.green,
  267. ),
  268. );
  269. // 提交后重置表单
  270. setState(() {
  271. _rfids.clear();
  272. });
  273. }
  274. })
  275. .catchError((err) {
  276. ToastUtil.error('批量绑定个体失败');
  277. if (mounted && err != null) {
  278. String errorMessage = err.toString();
  279. if (err is Exception) {
  280. errorMessage = err.toString();
  281. }
  282. logger.e(errorMessage);
  283. ScaffoldMessenger.of(context).showSnackBar(
  284. SnackBar(
  285. content: Text(errorMessage),
  286. backgroundColor: Colors.red,
  287. ),
  288. );
  289. }
  290. });
  291. }
  292. }