custom_check_box.dart 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. import 'package:cs_resources/constants/color_constants.dart';
  2. import 'package:cs_resources/generated/assets.dart';
  3. import 'package:flutter/material.dart';
  4. import 'package:shared/utils/log_utils.dart';
  5. import 'package:widgets/ext/ex_widget.dart';
  6. import 'package:widgets/my_load_image.dart';
  7. import 'package:widgets/my_text_view.dart';
  8. class CustomCheckBox extends StatefulWidget {
  9. final List<String> options;
  10. final Function(List<int> selectedIndexes) onOptionsSelected; // 选中项的索引回调
  11. final List<String> selectedOptions; // 已选中的选项列表
  12. CustomCheckBox({
  13. required this.options,
  14. required this.onOptionsSelected,
  15. required this.selectedOptions,
  16. });
  17. @override
  18. _CustomCheckBoxState createState() => _CustomCheckBoxState();
  19. }
  20. class _CustomCheckBoxState extends State<CustomCheckBox> {
  21. late List<String> _selectedOptions;
  22. late List<int> _selectedIndexes;
  23. @override
  24. void initState() {
  25. super.initState();
  26. _initializeSelectedOptions();
  27. }
  28. void _initializeSelectedOptions() {
  29. _selectedOptions = List.from(widget.selectedOptions);
  30. _selectedIndexes =
  31. widget.options.asMap().entries.where((entry) => _selectedOptions.contains(entry.value)).map((entry) => entry.key).toList(); // 根据选项匹配初始化索引
  32. }
  33. @override
  34. void didUpdateWidget(CustomCheckBox oldWidget) {
  35. super.didUpdateWidget(oldWidget);
  36. Log.d("oldWidget - selectedOptions :${oldWidget.selectedOptions} newWidget - selectedOptions:${widget.selectedOptions} ");
  37. // 使用ListEquality来比较两个列表的内容
  38. if (!_listEquals(oldWidget.selectedOptions, widget.selectedOptions)) {
  39. _initializeSelectedOptions();
  40. }
  41. }
  42. bool _listEquals(List<String> list1, List<String> list2) {
  43. if (list1.length != list2.length) {
  44. return false;
  45. }
  46. for (int i = 0; i < list1.length; i++) {
  47. if (list1[i] != list2[i]) {
  48. return false;
  49. }
  50. }
  51. return true;
  52. }
  53. @override
  54. Widget build(BuildContext context) {
  55. return Wrap(
  56. spacing: 15.0,
  57. runSpacing: 15.0,
  58. children: widget.options.asMap().entries.map((entry) {
  59. int index = entry.key;
  60. String option = entry.value;
  61. return _buildCheckBoxWithIconAndText(
  62. path: _selectedIndexes.contains(index) ? Assets.baseServiceCheckBoxChecked : Assets.baseServiceCheckBoxUncheck,
  63. text: option,
  64. value: _selectedIndexes.contains(index),
  65. onChanged: (isChecked) {
  66. setState(() {
  67. if (isChecked) {
  68. if (!_selectedOptions.contains(option)) {
  69. _selectedOptions.add(option);
  70. _selectedIndexes.add(index);
  71. }
  72. } else {
  73. _selectedOptions.remove(option);
  74. _selectedIndexes.remove(index);
  75. }
  76. widget.onOptionsSelected(_selectedIndexes);
  77. });
  78. },
  79. );
  80. }).toList(),
  81. );
  82. }
  83. Widget _buildCheckBoxWithIconAndText({
  84. required String path,
  85. required String text,
  86. required bool value,
  87. required Function(bool) onChanged,
  88. }) {
  89. return Row(
  90. mainAxisSize: MainAxisSize.min,
  91. children: <Widget>[
  92. MyAssetImage(path, width: 20.5, height: 20.5),
  93. const SizedBox(width: 10),
  94. MyTextView(
  95. text,
  96. textColor: ColorConstants.white,
  97. fontSize: 14,
  98. isFontRegular: true,
  99. ),
  100. ],
  101. ).onTap(() {
  102. onChanged(!value); // 点击时切换选中状态
  103. });
  104. }
  105. }