controller.dart 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345
  1. import 'dart:async';
  2. import 'dart:io';
  3. import 'package:electronic_assistant/base/base_controller.dart';
  4. import 'package:electronic_assistant/data/repositories/task_repository.dart';
  5. import 'package:electronic_assistant/module/chat/view.dart';
  6. import 'package:electronic_assistant/module/home/controller.dart';
  7. import 'package:electronic_assistant/module/talk/summary/view.dart';
  8. import 'package:electronic_assistant/module/talk/todo/view.dart';
  9. import 'package:electronic_assistant/resource/assets.gen.dart';
  10. import 'package:electronic_assistant/resource/colors.gen.dart';
  11. import 'package:electronic_assistant/resource/string.gen.dart';
  12. import 'package:electronic_assistant/utils/error_handler.dart';
  13. import 'package:electronic_assistant/utils/expand.dart';
  14. import 'package:electronic_assistant/utils/mmkv_util.dart';
  15. import 'package:flutter/cupertino.dart';
  16. import 'package:flutter_screenutil/flutter_screenutil.dart';
  17. import 'package:get/get.dart';
  18. import 'package:connectivity_plus/connectivity_plus.dart';
  19. import '../../data/bean/agenda_list_all_bean.dart';
  20. import '../../data/bean/talks.dart';
  21. import '../../data/repositories/agenda_repository.dart';
  22. import '../../data/repositories/talk_repository.dart';
  23. import '../../dialog/alert_dialog.dart';
  24. import '../../router/app_pages.dart';
  25. import '../../utils/toast_util.dart';
  26. import '../record/controller.dart';
  27. import 'original/view.dart';
  28. import 'package:just_audio/just_audio.dart';
  29. class TalkController extends BaseController {
  30. final String uploadNoPrompts = "UPLOAD_NO_PROMPTS";
  31. final Rxn<TalkBean> talkBean = Rxn();
  32. final isShowElectricLow = false.obs;
  33. bool isAudioLoading = false;
  34. final double sliderMax = 1;
  35. bool? audioFileIsExist;
  36. bool? isUploadedFile;
  37. Rxn<bool> isUploading = Rxn();
  38. final isAudioPlaying = false.obs;
  39. final audioProgressValue = 0.0.obs;
  40. final audioDuration = Duration.zero.obs;
  41. final agendaAllList = <AgendaListAllBean>[].obs;
  42. final List<String> tabBeans = [
  43. StringName.talkTabSummary.tr,
  44. StringName.talkTabMyTask.tr,
  45. StringName.talkTabOriginal.tr
  46. ];
  47. final _audioPlayer = AudioPlayer();
  48. StreamSubscription? _talkBeanListener;
  49. final pages = [const SummaryView(), const TodoView(), const OriginalView()];
  50. @override
  51. void onReady() {
  52. super.onReady();
  53. _initAudioPlayer();
  54. _initListener();
  55. _getArguments();
  56. }
  57. void _initAudioPlayer() {
  58. _audioPlayer.playerStateStream.listen((playerState) {
  59. if (playerState.processingState == ProcessingState.loading ||
  60. playerState.processingState == ProcessingState.buffering) {
  61. isAudioLoading = true;
  62. debugPrint('音频load = true');
  63. } else {
  64. debugPrint('音频load = false');
  65. isAudioLoading = false;
  66. if (playerState.processingState == ProcessingState.completed) {
  67. _audioPlayer.stop();
  68. _audioPlayer.seek(Duration.zero);
  69. isAudioPlaying.value = false;
  70. debugPrint('音频 播放结束了');
  71. }
  72. }
  73. isAudioPlaying.value = playerState.playing;
  74. }, onError: (Object e, StackTrace stackTrace) {
  75. debugPrint('音频加载异常 == $e');
  76. });
  77. _audioPlayer.durationStream.listen((duration) {
  78. if (duration != null) {
  79. debugPrint('音频总播放时长 == ${duration.inMilliseconds}');
  80. audioDuration.value = duration;
  81. }
  82. });
  83. _audioPlayer.positionStream.listen((position) {
  84. if (audioDuration.value.inMilliseconds > 0) {
  85. audioProgressValue.value =
  86. (position.inMilliseconds / audioDuration.value.inMilliseconds)
  87. .clamp(0.0, sliderMax);
  88. }
  89. });
  90. }
  91. void _initListener() {
  92. _talkBeanListener = talkBean.listen((bean) {
  93. _dealTalkUpdate(bean);
  94. });
  95. }
  96. void _dealTalkUpdate(TalkBean? bean) async {
  97. String? id = talkBean.value?.id;
  98. if (id == null) {
  99. return;
  100. }
  101. if (talkRepository.isUploadingTalk(id) &&
  102. talkBean.value?.status.value == TalkStatus.notAnalysis) {
  103. isUploading.value = true;
  104. } else {
  105. isUploading.value = false;
  106. }
  107. try {
  108. Uri? uri;
  109. if (bean?.isExample == true && bean?.audioUrl != null) {
  110. uri = Uri.parse(bean!.audioUrl!);
  111. } else {
  112. File? file = await getFileByTalk(id, bean?.uploadType);
  113. uri = file?.uri;
  114. }
  115. if (uri == null) {
  116. throw '音频文件不存在';
  117. }
  118. await _audioPlayer.setAudioSource(AudioSource.uri(uri));
  119. audioFileIsExist = true;
  120. } catch (e) {
  121. audioFileIsExist = false;
  122. debugPrint('音频设置异常 == $e');
  123. }
  124. }
  125. void _getArguments() {
  126. if (Get.arguments is TalkBean) {
  127. talkBean.value = Get.arguments as TalkBean;
  128. }
  129. }
  130. void updateProgress(double value) {
  131. final newPosition = Duration(
  132. milliseconds: (value * audioDuration.value.inMilliseconds).toInt());
  133. _audioPlayer.seek(newPosition);
  134. }
  135. void clickPlayAudio() {
  136. if (audioFileIsExist != true) {
  137. ToastUtil.showToast(StringName.talkFileNotFind.tr);
  138. return;
  139. }
  140. if (isAudioLoading) {
  141. ToastUtil.showToast(StringName.talkAudioLoading.tr);
  142. return;
  143. }
  144. if (_audioPlayer.playing) {
  145. _audioPlayer.pause();
  146. isAudioPlaying.value = false;
  147. } else {
  148. _audioPlayer.play();
  149. isAudioPlaying.value = true;
  150. }
  151. }
  152. void _checkFileSizeAndNet() async {
  153. String? id = talkBean.value?.id;
  154. if (id == null) {
  155. return;
  156. }
  157. File? file =
  158. await getFileByTalk(talkBean.value?.id, talkBean.value?.uploadType);
  159. if (file == null || !file.existsSync()) {
  160. ToastUtil.showToast(StringName.talkUploadFileNotExist.tr);
  161. return;
  162. }
  163. bool isCheckRemind = KVUtil.getBool(uploadNoPrompts, false);
  164. if (isCheckRemind) {
  165. _requestAnalyze(file);
  166. return;
  167. }
  168. //如果文件大小低于250MB 不弹窗提醒
  169. if (file.lengthSync() < 250 * 1024 * 1024) {
  170. _requestAnalyze(file);
  171. return;
  172. }
  173. final List<ConnectivityResult> connectivityResult =
  174. await (Connectivity().checkConnectivity());
  175. if (connectivityResult.contains(ConnectivityResult.wifi)) {
  176. _requestAnalyze(file);
  177. } else {
  178. _showTrafficRemindDialog(file.lengthSync().toReadableSize(),
  179. confirmOnTap: (isCheckRemind) {
  180. if (isCheckRemind) {
  181. KVUtil.putBool(uploadNoPrompts, true);
  182. }
  183. _requestAnalyze(file);
  184. });
  185. }
  186. }
  187. void _showTrafficRemindDialog(String holderTxt,
  188. {void Function(bool isCheckRemind)? confirmOnTap}) {
  189. final remindTrafficConsume = false.obs;
  190. Widget getSelectIcon() {
  191. return Obx(() {
  192. return remindTrafficConsume.value
  193. ? Assets.images.iconSelectTrue.image()
  194. : Assets.images.iconSelectFalse.image();
  195. });
  196. }
  197. Assets.images.iconSelectTrue.image();
  198. EAAlertDialog.show(
  199. contentWidget: Column(
  200. children: [
  201. Text(
  202. StringName.talkTrafficRemindTitle.tr
  203. .replacePlaceholders([holderTxt]),
  204. style:
  205. TextStyle(fontSize: 15.sp, color: ColorName.primaryTextColor),
  206. ),
  207. SizedBox(height: 8.h),
  208. GestureDetector(
  209. onTap: () {
  210. remindTrafficConsume.value = !remindTrafficConsume.value;
  211. },
  212. child: IntrinsicWidth(
  213. child: Row(
  214. children: [
  215. SizedBox(width: 20.w, height: 20.w, child: getSelectIcon()),
  216. SizedBox(width: 5.w),
  217. Text(
  218. StringName.talkTrafficRemindTips.tr,
  219. style: TextStyle(
  220. fontSize: 15.sp, color: ColorName.tertiaryTextColor),
  221. )
  222. ],
  223. ),
  224. ),
  225. )
  226. ],
  227. ),
  228. cancelText: StringName.cancel.tr,
  229. confirmText: StringName.sure.tr,
  230. confirmOnTap: () {
  231. confirmOnTap?.call(remindTrafficConsume.value);
  232. });
  233. }
  234. void checkCanAnalyze() {
  235. String? id = talkBean.value?.id;
  236. double? duration = talkBean.value?.duration;
  237. if (id == null || duration == null) {
  238. return;
  239. }
  240. talkRepository.checkElectric(duration).then((data) {
  241. if (data.enough) {
  242. //检查网络以及文件大小
  243. _checkFileSizeAndNet();
  244. } else {
  245. ToastUtil.showToast(StringName.talkAnalyseLowToast.tr);
  246. isShowElectricLow.value = true;
  247. }
  248. }).catchError((error) {
  249. ToastUtil.showToast(error);
  250. });
  251. }
  252. void _requestAnalyze(File file) {
  253. String? talkId = talkBean.value?.id;
  254. double? duration = talkBean.value?.duration;
  255. if (talkId == null || duration == null || isUploadedFile == true) {
  256. return;
  257. }
  258. isUploading.value = true;
  259. talkRepository.uploadTalkFile(talkId, duration, file).then((taskId) {
  260. isUploadedFile = true;
  261. taskRepository.addTask(taskId);
  262. }).catchError((error) {
  263. isUploading.value = false;
  264. ErrorHandler.toastError(error);
  265. });
  266. }
  267. void refreshAgendaAllData() {
  268. String? id = talkBean.value?.id;
  269. if (id == null || agendaAllList.isNotEmpty) {
  270. return;
  271. }
  272. agendaRepository.agendaListAll(id).then((agenda) {
  273. if (agenda.list != null) {
  274. agendaAllList.value = agenda.list!;
  275. }
  276. });
  277. }
  278. void clickAIAnalysis() {
  279. if (talkBean.value != null) {
  280. ChatPage.startByTalk(talkBean.value!);
  281. }
  282. }
  283. @override
  284. void onClose() {
  285. super.onClose();
  286. _talkBeanListener?.cancel();
  287. _audioPlayer.dispose();
  288. }
  289. void onGoElectricStore() {
  290. Get.toNamed(RoutePath.store);
  291. Future.delayed(const Duration(milliseconds: 250), () {
  292. isShowElectricLow.value = false;
  293. });
  294. }
  295. }
  296. Future<File?> getFileByTalk(String? talkId, int? uploadType) async {
  297. if (talkId == null) {
  298. return null;
  299. }
  300. if (uploadType == TalkUploadType.localUpload) {
  301. return await getChoiceUploadFile(talkId);
  302. } else {
  303. return await RecordController.getRecordFile(talkId);
  304. }
  305. }