controller.dart 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  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. if (file?.existsSync() == true) {
  114. uri = file?.uri;
  115. }
  116. }
  117. if (uri == null) {
  118. throw '音频文件不存在';
  119. }
  120. await _audioPlayer.setAudioSource(AudioSource.uri(uri));
  121. audioFileIsExist = true;
  122. } catch (e) {
  123. audioFileIsExist = false;
  124. debugPrint('音频设置异常 == $e');
  125. }
  126. }
  127. void _getArguments() {
  128. if (Get.arguments is TalkBean) {
  129. talkBean.value = Get.arguments as TalkBean;
  130. }
  131. }
  132. void updateProgress(double value) {
  133. final newPosition = Duration(
  134. milliseconds: (value * audioDuration.value.inMilliseconds).toInt());
  135. _audioPlayer.seek(newPosition);
  136. }
  137. void clickPlayAudio() {
  138. if (audioFileIsExist != true) {
  139. ToastUtil.showToast(StringName.talkFileNotFind.tr);
  140. return;
  141. }
  142. if (isAudioLoading) {
  143. ToastUtil.showToast(StringName.talkAudioLoading.tr);
  144. return;
  145. }
  146. if (_audioPlayer.playing) {
  147. _audioPlayer.pause();
  148. isAudioPlaying.value = false;
  149. } else {
  150. _audioPlayer.play();
  151. isAudioPlaying.value = true;
  152. }
  153. }
  154. void _checkFileSizeAndNet() async {
  155. String? id = talkBean.value?.id;
  156. if (id == null) {
  157. return;
  158. }
  159. File? file =
  160. await getFileByTalk(talkBean.value?.id, talkBean.value?.uploadType);
  161. if (file == null || !file.existsSync()) {
  162. ToastUtil.showToast(StringName.talkUploadFileNotExist.tr);
  163. return;
  164. }
  165. bool isCheckRemind = KVUtil.getBool(uploadNoPrompts, false);
  166. if (isCheckRemind) {
  167. _requestAnalyze(file);
  168. return;
  169. }
  170. //如果文件大小低于250MB 不弹窗提醒
  171. if (file.lengthSync() < 250 * 1024 * 1024) {
  172. _requestAnalyze(file);
  173. return;
  174. }
  175. final List<ConnectivityResult> connectivityResult =
  176. await (Connectivity().checkConnectivity());
  177. if (connectivityResult.contains(ConnectivityResult.wifi)) {
  178. _requestAnalyze(file);
  179. } else {
  180. _showTrafficRemindDialog(file.lengthSync().toReadableSize(),
  181. confirmOnTap: (isCheckRemind) {
  182. if (isCheckRemind) {
  183. KVUtil.putBool(uploadNoPrompts, true);
  184. }
  185. _requestAnalyze(file);
  186. });
  187. }
  188. }
  189. void _showTrafficRemindDialog(String holderTxt,
  190. {void Function(bool isCheckRemind)? confirmOnTap}) {
  191. final remindTrafficConsume = false.obs;
  192. Widget getSelectIcon() {
  193. return Obx(() {
  194. return remindTrafficConsume.value
  195. ? Assets.images.iconSelectTrue.image()
  196. : Assets.images.iconSelectFalse.image();
  197. });
  198. }
  199. Assets.images.iconSelectTrue.image();
  200. EAAlertDialog.show(
  201. contentWidget: Column(
  202. children: [
  203. Text(
  204. StringName.talkTrafficRemindTitle.tr
  205. .replacePlaceholders([holderTxt]),
  206. style:
  207. TextStyle(fontSize: 15.sp, color: ColorName.primaryTextColor),
  208. ),
  209. SizedBox(height: 8.h),
  210. GestureDetector(
  211. onTap: () {
  212. remindTrafficConsume.value = !remindTrafficConsume.value;
  213. },
  214. child: IntrinsicWidth(
  215. child: Row(
  216. children: [
  217. SizedBox(width: 20.w, height: 20.w, child: getSelectIcon()),
  218. SizedBox(width: 5.w),
  219. Text(
  220. StringName.talkTrafficRemindTips.tr,
  221. style: TextStyle(
  222. fontSize: 15.sp, color: ColorName.tertiaryTextColor),
  223. )
  224. ],
  225. ),
  226. ),
  227. )
  228. ],
  229. ),
  230. cancelText: StringName.cancel.tr,
  231. confirmText: StringName.sure.tr,
  232. confirmOnTap: () {
  233. confirmOnTap?.call(remindTrafficConsume.value);
  234. });
  235. }
  236. void checkCanAnalyze() {
  237. String? id = talkBean.value?.id;
  238. double? duration = talkBean.value?.duration;
  239. if (id == null || duration == null) {
  240. return;
  241. }
  242. talkRepository.checkElectric(duration).then((data) {
  243. if (data.enough) {
  244. //检查网络以及文件大小
  245. _checkFileSizeAndNet();
  246. } else {
  247. ToastUtil.showToast(StringName.talkAnalyseLowToast.tr);
  248. isShowElectricLow.value = true;
  249. }
  250. }).catchError((error) {
  251. ToastUtil.showToast(error);
  252. });
  253. }
  254. void _requestAnalyze(File file) {
  255. String? talkId = talkBean.value?.id;
  256. double? duration = talkBean.value?.duration;
  257. if (talkId == null || duration == null || isUploadedFile == true) {
  258. return;
  259. }
  260. isUploading.value = true;
  261. talkRepository.uploadTalkFile(talkId, duration, file).then((taskId) {
  262. isUploadedFile = true;
  263. taskRepository.addTask(taskId);
  264. }).catchError((error) {
  265. isUploading.value = false;
  266. ErrorHandler.toastError(error);
  267. });
  268. }
  269. void refreshAgendaAllData() {
  270. String? id = talkBean.value?.id;
  271. if (id == null || agendaAllList.isNotEmpty) {
  272. return;
  273. }
  274. agendaRepository.agendaListAll(id).then((agenda) {
  275. if (agenda.list != null) {
  276. agendaAllList.value = agenda.list!;
  277. }
  278. });
  279. }
  280. void clickAIAnalysis() {
  281. if (talkBean.value != null) {
  282. ChatPage.startByTalk(talkBean.value!);
  283. }
  284. }
  285. @override
  286. void onClose() {
  287. super.onClose();
  288. _talkBeanListener?.cancel();
  289. _audioPlayer.dispose();
  290. }
  291. void onGoElectricStore() {
  292. Get.toNamed(RoutePath.store);
  293. Future.delayed(const Duration(milliseconds: 250), () {
  294. isShowElectricLow.value = false;
  295. });
  296. }
  297. }
  298. Future<File?> getFileByTalk(String? talkId, int? uploadType) async {
  299. if (talkId == null) {
  300. return null;
  301. }
  302. if (uploadType == TalkUploadType.localUpload) {
  303. return await getChoiceUploadFile(talkId);
  304. } else {
  305. return await RecordController.getRecordFile(talkId);
  306. }
  307. }