language_view.dart 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. import 'package:flutter/material.dart';
  2. import 'package:flutter_screenutil/flutter_screenutil.dart';
  3. import 'package:get/get.dart';
  4. import 'package:nomo/app/base/base_view.dart';
  5. import 'package:nomo/app/widgets/click_opacity.dart';
  6. import 'package:nomo/app/widgets/ix_app_bar.dart';
  7. import 'package:nomo/config/theme/theme_extensions/theme_extension.dart';
  8. import '../controllers/language_controller.dart';
  9. class LanguageView extends BaseView<LanguageController> {
  10. const LanguageView({super.key});
  11. @override
  12. PreferredSizeWidget? get appBar => IXAppBar(title: 'Language');
  13. @override
  14. Widget buildContent(BuildContext context) {
  15. return SingleChildScrollView(
  16. child: Padding(
  17. padding: EdgeInsets.symmetric(horizontal: 14.w, vertical: 10.w),
  18. child: Container(
  19. decoration: BoxDecoration(
  20. color: Get.reactiveTheme.highlightColor,
  21. borderRadius: BorderRadius.circular(12.r),
  22. ),
  23. child: Obx(
  24. () => Column(
  25. children: controller.languages.asMap().entries.map((entry) {
  26. final index = entry.key;
  27. final language = entry.value;
  28. final isSelected =
  29. controller.selectedLanguage.value == language.code;
  30. return Column(
  31. children: [
  32. _buildLanguageOption(
  33. language: language,
  34. isSelected: isSelected,
  35. onTap: () => controller.selectLanguage(language.code),
  36. ),
  37. if (index < controller.languages.length - 1)
  38. _buildDivider(),
  39. ],
  40. );
  41. }).toList(),
  42. ),
  43. ),
  44. ),
  45. ),
  46. );
  47. }
  48. /// 构建语言选项
  49. Widget _buildLanguageOption({
  50. required LanguageInfo language,
  51. required bool isSelected,
  52. required VoidCallback onTap,
  53. }) {
  54. return ClickOpacity(
  55. onTap: onTap,
  56. child: Padding(
  57. padding: EdgeInsets.all(14.w),
  58. child: Row(
  59. children: [
  60. // 语言信息
  61. Expanded(
  62. child: Column(
  63. crossAxisAlignment: CrossAxisAlignment.start,
  64. children: [
  65. Text(
  66. language.name,
  67. style: TextStyle(
  68. fontSize: 14.sp,
  69. height: 1.4,
  70. fontWeight: FontWeight.w600,
  71. color: isSelected
  72. ? Get.reactiveTheme.shadowColor
  73. : Get.reactiveTheme.textTheme.bodyLarge!.color,
  74. ),
  75. ),
  76. Text(
  77. language.nativeName,
  78. style: TextStyle(
  79. fontSize: 12.sp,
  80. height: 1.6,
  81. color: Get.reactiveTheme.hintColor,
  82. ),
  83. ),
  84. ],
  85. ),
  86. ),
  87. // 选中指示器
  88. Container(
  89. width: 20.w,
  90. height: 20.w,
  91. decoration: BoxDecoration(
  92. shape: BoxShape.circle,
  93. color: isSelected
  94. ? Get.reactiveTheme.shadowColor
  95. : Colors.transparent,
  96. border: Border.all(
  97. color: isSelected
  98. ? Get.reactiveTheme.shadowColor
  99. : Colors.grey[400]!,
  100. width: 1.5.w,
  101. ),
  102. ),
  103. child: isSelected
  104. ? Icon(Icons.check, color: Colors.white, size: 16.w)
  105. : null,
  106. ),
  107. ],
  108. ),
  109. ),
  110. );
  111. }
  112. /// 构建分割线
  113. Widget _buildDivider() {
  114. return Divider(color: Get.reactiveTheme.dividerColor, height: 1.w);
  115. }
  116. }