view.dart 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206
  1. import 'package:electronic_assistant/base/base_page.dart';
  2. import 'package:electronic_assistant/data/bean/talks.dart';
  3. import 'package:electronic_assistant/resource/assets.gen.dart';
  4. import 'package:electronic_assistant/resource/colors.gen.dart';
  5. import 'package:electronic_assistant/resource/string.gen.dart';
  6. import 'package:electronic_assistant/utils/expand.dart';
  7. import 'package:flutter/material.dart';
  8. import 'package:flutter_screenutil/flutter_screenutil.dart';
  9. import 'package:get/get.dart';
  10. import 'package:scrollable_positioned_list/scrollable_positioned_list.dart';
  11. import '../../../data/bean/talk_original.dart';
  12. import '../../../widget/high_light_search_text.dart';
  13. import '../common_view.dart';
  14. import 'controller.dart';
  15. class OriginalView extends BasePage<OriginalController> {
  16. final String? talkId;
  17. OriginalView(this.talkId, {super.key}) {
  18. Get.lazyPut(() => OriginalController(talkId), tag: talkId);
  19. }
  20. @override
  21. get controller => Get.find<OriginalController>(tag: talkId);
  22. @override
  23. Widget buildBody(BuildContext context) {
  24. return buildOriginalContentView();
  25. }
  26. @override
  27. bool immersive() {
  28. return true;
  29. }
  30. Widget buildOriginalContentView() {
  31. return Obx(() {
  32. if (controller.originalList.isEmpty &&
  33. controller.talkController.talkBean.value?.status.value ==
  34. TalkStatus.analysisFail) {
  35. return getTalkFailView();
  36. } else if (controller.originalList.isEmpty) {
  37. return getTalkAnalyseView(
  38. controller.talkController.isUploading.value == true &&
  39. controller.talkController.talkBean.value?.status.value ==
  40. TalkStatus.notAnalysis
  41. ? StringName.talkUploadingFileTip.tr
  42. : controller
  43. .talkController.talkBean.value?.progressContent.value ??
  44. '',
  45. controller.talkController.isUploading.value == true &&
  46. controller.talkController.talkBean.value?.status.value ==
  47. TalkStatus.notAnalysis
  48. ? controller.talkController.uploadProgress.value
  49. : controller.talkController.talkBean.value?.progress.value
  50. ?.toDouble() ??
  51. 0.0);
  52. } else {
  53. return ScrollablePositionedList.builder(
  54. itemScrollController: controller.itemScrollController,
  55. padding: EdgeInsets.only(bottom: 150.h),
  56. itemBuilder: _buildOriginalItem,
  57. itemCount: controller.originalList.length,
  58. );
  59. }
  60. });
  61. }
  62. Widget _buildOriginalItem(BuildContext context, int index) {
  63. TalkOriginal item = controller.originalList[index];
  64. return GestureDetector(
  65. onTap: () {
  66. controller.talkController.seekTo(item.startMs);
  67. },
  68. child: Container(
  69. color: ColorName.transparent,
  70. padding:
  71. EdgeInsets.only(left: 12.w, right: 12.w, top: 11.h, bottom: 13.h),
  72. child: Column(
  73. crossAxisAlignment: CrossAxisAlignment.start,
  74. children: [
  75. SizedBox(
  76. height: 24.w,
  77. child: Row(
  78. children: [
  79. Container(
  80. decoration: const BoxDecoration(
  81. color: ColorName.colorPrimary,
  82. shape: BoxShape.circle,
  83. ),
  84. width: 20.w,
  85. height: 20.w,
  86. child: Center(
  87. child: Text(
  88. item.speakerId.toString(),
  89. style: TextStyle(fontSize: 12.sp, color: Colors.white),
  90. ),
  91. ),
  92. ),
  93. SizedBox(width: 6.w),
  94. Text(item.speaker.toString(),
  95. style: TextStyle(
  96. fontSize: 14.sp,
  97. color: ColorName.secondaryTextColor)),
  98. SizedBox(width: 4.w),
  99. Text(formatMilliseconds(item.startMs),
  100. style: TextStyle(
  101. fontSize: 12.sp, color: ColorName.tertiaryTextColor)),
  102. const Spacer(),
  103. Obx(() {
  104. return Visibility(
  105. visible: item.isSelected(),
  106. child: GestureDetector(
  107. onTap: () {
  108. controller.talkTranslateClick(item);
  109. },
  110. child: Container(
  111. margin: EdgeInsets.only(right: 12.w),
  112. child: Assets.images.iconTalkTranslate
  113. .image(width: 24.w, height: 24.w)),
  114. ),
  115. );
  116. })
  117. ],
  118. ),
  119. ),
  120. SizedBox(height: 12.h),
  121. Obx(() {
  122. Color txtColor = item.isSelected()
  123. ? ColorName.colorPrimary
  124. : ColorName.primaryTextColor;
  125. return HighlightSearchText(
  126. defaultHighlightIndex: item.checkIndex.value,
  127. normalTextStyle: TextStyle(fontSize: 14.sp, color: txtColor),
  128. highlightTextStyle: TextStyle(
  129. fontSize: 14.sp,
  130. color: txtColor,
  131. backgroundColor: '#ACE6FF'.color),
  132. activeHighlightTextStyle: TextStyle(
  133. fontSize: 14.sp,
  134. color: txtColor,
  135. backgroundColor: '#FFE078'.color),
  136. text: item.sentence.toString(),
  137. searchKeyword: controller.searchPrint.value);
  138. }),
  139. Obx(() {
  140. return Visibility(
  141. visible:
  142. item.getTranslateStatus() == TalkTranslate.translating ||
  143. item.getTranslateStatus() == TalkTranslate.translated,
  144. child: Container(
  145. width: double.infinity,
  146. margin: EdgeInsets.only(top: 8.w),
  147. decoration: BoxDecoration(
  148. color: '#FAF9FB'.toColor(),
  149. border: Border.all(color: '#F2EFF5'.toColor(), width: 1.w),
  150. borderRadius: BorderRadius.circular(6.w),
  151. ),
  152. padding:
  153. EdgeInsets.symmetric(vertical: 11.w, horizontal: 10.w),
  154. child: RichText(
  155. text: TextSpan(
  156. children: [
  157. WidgetSpan(
  158. alignment: PlaceholderAlignment.middle,
  159. child: Container(
  160. margin: EdgeInsets.only(right: 6.w),
  161. child: Assets.images.iconTalkTranslateSmall
  162. .image(width: 22.w, height: 22.w),
  163. ),
  164. ),
  165. TextSpan(
  166. text: item.getTranslatedSentence() ??
  167. StringName.translating.tr,
  168. style: TextStyle(
  169. fontSize: 14.sp,
  170. color: ColorName.secondaryTextColor),
  171. ),
  172. ],
  173. ),
  174. ),
  175. ),
  176. );
  177. })
  178. ],
  179. ),
  180. ),
  181. );
  182. }
  183. String formatMilliseconds(int? totalMilliseconds) {
  184. if (totalMilliseconds == null) {
  185. return '';
  186. }
  187. int totalSeconds = (totalMilliseconds / 1000).round();
  188. int hours = totalSeconds ~/ 3600;
  189. int minutes = (totalSeconds % 3600) ~/ 60;
  190. int seconds = totalSeconds % 60;
  191. if (hours > 0) {
  192. return '${hours.toString().padLeft(2, '0')}:${minutes.toString().padLeft(2, '0')}:${seconds.toString().padLeft(2, '0')}';
  193. } else {
  194. return '${minutes.toString().padLeft(2, '0')}:${seconds.toString().padLeft(2, '0')}';
  195. }
  196. }
  197. }