controller.dart 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636
  1. import 'dart:async';
  2. import 'dart:developer';
  3. import 'dart:io';
  4. import 'package:connectivity_plus/connectivity_plus.dart';
  5. import 'package:electronic_assistant/base/base_controller.dart';
  6. import 'package:electronic_assistant/data/consts/event_report_id.dart';
  7. import 'package:electronic_assistant/data/repositories/account_repository.dart';
  8. import 'package:electronic_assistant/data/repositories/task_repository.dart';
  9. import 'package:electronic_assistant/handler/event_handler.dart';
  10. import 'package:electronic_assistant/module/chat/view.dart';
  11. import 'package:electronic_assistant/module/login/view.dart';
  12. import 'package:electronic_assistant/module/record/record_handler.dart';
  13. import 'package:electronic_assistant/module/store/view.dart';
  14. import 'package:electronic_assistant/module/talk/summary/view.dart';
  15. import 'package:electronic_assistant/module/talk/todo/controller.dart';
  16. import 'package:electronic_assistant/module/talk/todo/view.dart';
  17. import 'package:electronic_assistant/module/talk/view.dart';
  18. import 'package:electronic_assistant/resource/assets.gen.dart';
  19. import 'package:electronic_assistant/resource/colors.gen.dart';
  20. import 'package:electronic_assistant/resource/string.gen.dart';
  21. import 'package:electronic_assistant/router/app_pages.dart';
  22. import 'package:electronic_assistant/utils/audio_picker_utils.dart';
  23. import 'package:electronic_assistant/utils/error_handler.dart';
  24. import 'package:electronic_assistant/utils/expand.dart';
  25. import 'package:electronic_assistant/utils/file_upload_check_helper.dart';
  26. import 'package:electronic_assistant/utils/mmkv_util.dart';
  27. import 'package:flutter/cupertino.dart';
  28. import 'package:flutter/material.dart';
  29. import 'package:flutter_screenutil/flutter_screenutil.dart';
  30. import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
  31. import 'package:get/get.dart';
  32. import 'package:just_audio/just_audio.dart';
  33. import 'package:share_plus/share_plus.dart';
  34. import 'package:wakelock_plus/wakelock_plus.dart';
  35. import '../../data/api/request/agenda_update_bean.dart';
  36. import '../../data/bean/agenda.dart';
  37. import '../../data/bean/agenda_list_all_bean.dart';
  38. import '../../data/bean/talks.dart';
  39. import '../../data/repositories/agenda_repository.dart';
  40. import '../../data/repositories/talk_repository.dart';
  41. import '../../dialog/add_agenda_dialog.dart';
  42. import '../../dialog/alert_dialog.dart';
  43. import '../../dialog/talk_share_dialog.dart';
  44. import '../../utils/common_utils.dart';
  45. import '../../utils/event_bus.dart';
  46. import '../../utils/system_share_util.dart';
  47. import '../../utils/toast_util.dart';
  48. import 'original/view.dart';
  49. class TalkController extends BaseController {
  50. static const String argumentItem = 'argument_item';
  51. static const String argumentTalkId = 'argument_talk_id';
  52. static const String argumentEventTag = 'argument_event_tag';
  53. final String uploadNoPrompts = "UPLOAD_NO_PROMPTS";
  54. final Rxn<TalkBean> talkBean = Rxn();
  55. StreamSubscription? _talkUploadListener;
  56. final RxDouble uploadProgress = RxDouble(0);
  57. final isShowElectricLow = false.obs;
  58. bool isAudioLoading = false;
  59. final double sliderMax = 1;
  60. bool? audioFileIsExist;
  61. bool? isUploadedFile;
  62. Rxn<bool> isUploading = Rxn();
  63. final isAudioPlaying = false.obs;
  64. final audioProgressValue = 0.0.obs;
  65. final audioDuration = Duration.zero.obs;
  66. final agendaOriginalAllList = <AgendaListAllBean>[];
  67. final agendaAllList = <AgendaListAllBean>[].obs;
  68. final _isEditModel = false.obs;
  69. final TextEditingController editTalkNameController = TextEditingController();
  70. final tabIndex = 0.obs;
  71. final List<String> tabBeans = [
  72. StringName.talkTabSummary.tr,
  73. StringName.talkTabMyTask.tr,
  74. StringName.talkTabOriginal.tr
  75. ];
  76. bool get isEditModel => _isEditModel.value;
  77. RxBool get isEditModelRx => _isEditModel;
  78. final _audioPlayer = AudioPlayer();
  79. StreamSubscription? _talkBeanListener;
  80. TextEditingController? _agendaContentController;
  81. TextEditingController? _agendaNameController;
  82. TextEditingController get agendaContentController {
  83. _agendaContentController ??= TextEditingController();
  84. return _agendaContentController!;
  85. }
  86. TextEditingController get agendaNameController {
  87. _agendaNameController ??= TextEditingController();
  88. return _agendaNameController!;
  89. }
  90. String? paramId;
  91. String? eventTag;
  92. bool isLocalFileHas = false;
  93. final Rxn<Duration> playingDuration = Rxn();
  94. @override
  95. void onReady() {
  96. super.onReady();
  97. _initAudioPlayer();
  98. _getArguments();
  99. eventReport(EventId.event_101001, params: {EventId.id: eventTag});
  100. }
  101. void eventReport(String eventId, {Map<String, dynamic>? params}) {
  102. if (talkBean.value == null || talkBean.value?.isExample == true) {
  103. return;
  104. }
  105. EventHandler.report(eventId, params: params);
  106. }
  107. void _initAudioPlayer() {
  108. _audioPlayer.playerStateStream.listen((playerState) {
  109. if (playerState.processingState == ProcessingState.loading ||
  110. playerState.processingState == ProcessingState.buffering) {
  111. isAudioLoading = true;
  112. debugPrint('音频load = true');
  113. } else {
  114. debugPrint('音频load = false');
  115. isAudioLoading = false;
  116. if (playerState.processingState == ProcessingState.completed) {
  117. _audioPlayer.stop();
  118. _audioPlayer.seek(Duration.zero);
  119. isAudioPlaying.value = false;
  120. debugPrint('音频 播放结束了');
  121. }
  122. }
  123. isAudioPlaying.value = playerState.playing;
  124. }, onError: (Object e, StackTrace stackTrace) {
  125. debugPrint('音频加载异常 == $e');
  126. });
  127. _audioPlayer.durationStream.listen((duration) {
  128. if (duration != null) {
  129. debugPrint('音频总播放时长 == ${duration.inMilliseconds}');
  130. audioDuration.value = duration;
  131. }
  132. });
  133. _audioPlayer.positionStream.listen((duration) {
  134. debugPrint('音频播放时长 == ${duration.inMilliseconds}');
  135. playingDuration.value = duration;
  136. if (audioDuration.value.inMilliseconds > 0) {
  137. audioProgressValue.value =
  138. (duration.inMilliseconds / audioDuration.value.inMilliseconds)
  139. .clamp(0.0, sliderMax);
  140. }
  141. });
  142. }
  143. void _dealTalk(TalkBean? bean) async {
  144. debugPrint('talkBean == $bean');
  145. String? id = bean?.id;
  146. if (id == null) {
  147. return;
  148. }
  149. _loadAudioFile(bean);
  150. if (bean?.status.value == TalkStatus.notAnalysis) {
  151. setUploadingProgress(id);
  152. }
  153. if (bean?.status.value == TalkStatus.notAnalysis &&
  154. talkRepository.isUploadingTalk(id)) {
  155. isUploading.value = true;
  156. } else {
  157. isUploading.value = false;
  158. }
  159. }
  160. void setUploadingProgress(String id) {
  161. talkRepository.getUploadProgress(id).listen((progress) {
  162. uploadProgress.value = (progress * 20).toFormattedDouble(1);
  163. });
  164. }
  165. Future<void> _loadAudioFile(TalkBean? bean, {bool? loadPlay}) async {
  166. try {
  167. Uri? uri;
  168. if (bean?.isExample == true && bean?.audioUrl != null) {
  169. uri = Uri.parse(bean!.audioUrl!);
  170. } else {
  171. File? file = await getFileByTalk(talkBean.value);
  172. if (file?.existsSync() == true) {
  173. uri = file?.uri;
  174. }
  175. }
  176. if (uri == null) {
  177. throw '音频文件不存在';
  178. }
  179. await _audioPlayer.setAudioSource(AudioSource.uri(uri));
  180. if (loadPlay == true) {
  181. clickPlayAudio();
  182. }
  183. audioFileIsExist = true;
  184. } catch (e) {
  185. audioFileIsExist = false;
  186. debugPrint('音频设置异常 == $e');
  187. }
  188. }
  189. void _getArguments() {
  190. TalkBean? bean = parameters?[argumentItem];
  191. if (bean != null) {
  192. talkBean.value = bean;
  193. _dealTalk(bean);
  194. } else {
  195. paramId = parameters?[argumentTalkId];
  196. if (paramId != null) {
  197. talkRepository.talkInfo(paramId!).then((data) {
  198. talkBean.value = data.talkInfo;
  199. _dealTalk(data.talkInfo);
  200. }).catchError((error) {
  201. ErrorHandler.toastError(error);
  202. });
  203. }
  204. }
  205. eventTag = parameters?[argumentEventTag];
  206. }
  207. void updateProgress(double value) {
  208. final newPosition = Duration(
  209. milliseconds: (value * audioDuration.value.inMilliseconds).toInt());
  210. _audioPlayer.seek(newPosition);
  211. }
  212. void clickPlayAudio() async {
  213. if (audioFileIsExist != true && isLocalFileHas == false) {
  214. ToastUtil.showToast(StringName.talkFileNotFind.tr);
  215. return;
  216. }
  217. if (isLocalFileHas == true &&
  218. audioFileIsExist == false &&
  219. !await AudioPickerUtils.hasPermission()) {
  220. bool has = await AudioPickerUtils.requestPermissionExtend();
  221. if (has == false) {
  222. ToastUtil.showToast(StringName.authorizationFailed.tr);
  223. return;
  224. }
  225. //重新加载
  226. _loadAudioFile(talkBean.value, loadPlay: true);
  227. return;
  228. }
  229. if (isAudioLoading) {
  230. ToastUtil.showToast(StringName.talkAudioLoading.tr);
  231. return;
  232. }
  233. if (_audioPlayer.playing) {
  234. _audioPlayer.pause();
  235. isAudioPlaying.value = false;
  236. } else {
  237. _audioPlayer.play();
  238. isAudioPlaying.value = true;
  239. }
  240. }
  241. void _checkFileSizeAndNet() async {
  242. String? id = talkBean.value?.id;
  243. if (id == null) {
  244. return;
  245. }
  246. File? file = await getFileByTalk(talkBean.value);
  247. if (isLocalFileHas == true &&
  248. audioFileIsExist == false &&
  249. !await AudioPickerUtils.hasPermission()) {
  250. bool has = await AudioPickerUtils.requestPermissionExtend();
  251. if (has == false) {
  252. ToastUtil.showToast(StringName.authorizationFailed.tr);
  253. return;
  254. }
  255. //重新上传
  256. _checkFileSizeAndNet();
  257. return;
  258. }
  259. if (file == null || !file.existsSync()) {
  260. ToastUtil.showToast(StringName.talkUploadFileNotExist.tr);
  261. return;
  262. }
  263. bool isCheckRemind = KVUtil.getBool(uploadNoPrompts, false);
  264. if (isCheckRemind) {
  265. _requestAnalyze(file);
  266. return;
  267. }
  268. //如果文件大小低于250MB 不弹窗提醒
  269. if (file.lengthSync() < 250 * 1024 * 1024) {
  270. _requestAnalyze(file);
  271. return;
  272. }
  273. final List<ConnectivityResult> connectivityResult =
  274. await (Connectivity().checkConnectivity());
  275. if (connectivityResult.contains(ConnectivityResult.wifi)) {
  276. _requestAnalyze(file);
  277. } else {
  278. _showTrafficRemindDialog(file.lengthSync().toReadableSize(),
  279. confirmOnTap: (isCheckRemind) {
  280. if (isCheckRemind) {
  281. KVUtil.putBool(uploadNoPrompts, true);
  282. }
  283. _requestAnalyze(file);
  284. });
  285. }
  286. }
  287. void _showTrafficRemindDialog(String holderTxt,
  288. {void Function(bool isCheckRemind)? confirmOnTap}) {
  289. final remindTrafficConsume = false.obs;
  290. Widget getSelectIcon() {
  291. return Obx(() {
  292. return remindTrafficConsume.value
  293. ? Assets.images.iconSelectTrue.image()
  294. : Assets.images.iconSelectFalse.image();
  295. });
  296. }
  297. Assets.images.iconSelectTrue.image();
  298. EAAlertDialog.show(
  299. contentWidget: Column(
  300. children: [
  301. Text(
  302. StringName.talkTrafficRemindTitle.tr
  303. .replacePlaceholders([holderTxt]),
  304. style:
  305. TextStyle(fontSize: 15.sp, color: ColorName.primaryTextColor),
  306. ),
  307. SizedBox(height: 8.h),
  308. GestureDetector(
  309. onTap: () {
  310. remindTrafficConsume.value = !remindTrafficConsume.value;
  311. },
  312. child: IntrinsicWidth(
  313. child: Row(
  314. children: [
  315. SizedBox(width: 20.w, height: 20.w, child: getSelectIcon()),
  316. SizedBox(width: 5.w),
  317. Text(
  318. StringName.talkTrafficRemindTips.tr,
  319. style: TextStyle(
  320. fontSize: 15.sp, color: ColorName.tertiaryTextColor),
  321. )
  322. ],
  323. ),
  324. ),
  325. )
  326. ],
  327. ),
  328. cancelText: StringName.cancel.tr,
  329. confirmText: StringName.sure.tr,
  330. confirmOnTap: () {
  331. confirmOnTap?.call(remindTrafficConsume.value);
  332. });
  333. }
  334. void checkCanAnalyze() {
  335. String? id = talkBean.value?.id;
  336. double? duration = talkBean.value?.duration;
  337. if (id == null || duration == null) {
  338. return;
  339. }
  340. eventReport(EventId.event_101002);
  341. talkRepository.checkElectric(duration).then((data) {
  342. if (data.enough) {
  343. //检查网络以及文件大小
  344. _checkFileSizeAndNet();
  345. } else {
  346. ToastUtil.showToast(StringName.talkAnalyseLowToast.tr);
  347. isShowElectricLow.value = true;
  348. }
  349. }).catchError((error) {
  350. ErrorHandler.toastError(error);
  351. });
  352. }
  353. void _requestAnalyze(File file) {
  354. String? talkId = talkBean.value?.id;
  355. double? duration = talkBean.value?.duration;
  356. if (talkId == null || duration == null || isUploadedFile == true) {
  357. return;
  358. }
  359. isUploading.value = true;
  360. WakelockPlus.enable();
  361. talkRepository.uploadTalkFile(talkId, duration, file).then((taskId) {
  362. isUploadedFile = true;
  363. isUploading.value = false;
  364. talkBean.value?.progressContent.value = '录音上传中,请勿关闭小听';
  365. talkBean.value?.progress.value = 20;
  366. talkBean.value?.status.value = TalkStatus.analysing;
  367. taskRepository.addTask(taskId);
  368. }).catchError((error) {
  369. isUploading.value = false;
  370. ErrorHandler.toastError(error);
  371. }).whenComplete(() => WakelockPlus.disable());
  372. }
  373. void refreshAgendaAllData({bool isForceRefresh = false}) {
  374. String? id = talkBean.value?.id;
  375. if (id == null || (!isForceRefresh && agendaAllList.isNotEmpty)) {
  376. return;
  377. }
  378. agendaRepository.agendaListAll(id).then((agenda) {
  379. agendaAllList.clear();
  380. agendaOriginalAllList.clear();
  381. if (agenda.list != null) {
  382. agendaOriginalAllList.addAll(
  383. agenda.list!.map((item) => AgendaListAllBean.from(item)).toList());
  384. agendaAllList.addAll(agenda.list!);
  385. }
  386. });
  387. }
  388. void clickAIAnalysis() async {
  389. if (!await checkLogin()) {
  390. return;
  391. }
  392. if (talkBean.value != null) {
  393. eventReport(EventId.event_101003);
  394. ChatPage.startByTalk(
  395. talkBean.value!.isExample == true
  396. ? ChatFromType.fromTalkExample
  397. : ChatFromType.fromTalkDetail,
  398. talkBean.value!);
  399. }
  400. }
  401. void onGoElectricStore() {
  402. StorePage.start(fromType: StoreFromType.analyse);
  403. Future.delayed(const Duration(milliseconds: 250), () {
  404. isShowElectricLow.value = false;
  405. });
  406. }
  407. void onEditModelClick() async {
  408. if (!await checkLogin()) {
  409. return;
  410. }
  411. _isEditModel.value = true;
  412. if (_audioPlayer.playing) {
  413. _audioPlayer.pause();
  414. isAudioPlaying.value = false;
  415. }
  416. editTalkNameController.text = talkBean.value?.title.value ?? '';
  417. }
  418. void updateTabIndex(int index) {
  419. tabIndex.value = index;
  420. }
  421. void onEditCancel() {
  422. _isEditModel.value = false;
  423. agendaAllList.assignAll(agendaOriginalAllList
  424. .map((item) => AgendaListAllBean.from(item))
  425. .toList());
  426. }
  427. void onEditDoneClick() {
  428. if (talkBean.value == null) {
  429. return;
  430. }
  431. List<AgendaUpdateBean> list = [];
  432. for (AgendaListAllBean item in agendaAllList) {
  433. if (item.list != null) {
  434. for (Agenda agenda in item.list!) {
  435. list.add(AgendaUpdateBean(agenda.id, agenda.name, agenda.content));
  436. }
  437. }
  438. }
  439. agendaRepository.agendaUpdate(talkBean.value!.id, list).then((data) {
  440. refreshAgendaAllData(isForceRefresh: true);
  441. eventBus.emit(TodoController.refreshTalkMineTask);
  442. isEditModelRx.value = false;
  443. }).catchError((error) {
  444. ErrorHandler.toastError(error);
  445. });
  446. if (tabIndex.value == 0) {
  447. String updateName = editTalkNameController.text;
  448. talkRepository.talkRename(talkBean.value!.id, updateName).then((data) {
  449. talkBean.value?.title.value = updateName;
  450. }).catchError((error) {
  451. ErrorHandler.toastError(error);
  452. });
  453. }
  454. }
  455. void removeTalkAgenda(List<Agenda>? list, Agenda agenda) {
  456. list?.remove(agenda);
  457. agendaAllList.refresh();
  458. }
  459. void showSingleAddAgendaDialog(BuildContext context) {
  460. showAddAgendaDialog(context, agendaContentController, agendaNameController,
  461. list: agendaAllList.map((e) => e.name ?? "").toList(), callback: () {
  462. if (agendaContentController.text.isEmpty) {
  463. ToastUtil.showToast(StringName.talkAddAgendaContentHint.tr);
  464. return;
  465. }
  466. if (agendaNameController.text.isEmpty) {
  467. ToastUtil.showToast(StringName.talkAddAgendaNameHint.tr);
  468. return;
  469. }
  470. Get.back();
  471. _dealAddProcedureList();
  472. });
  473. }
  474. void _dealAddProcedureList() {
  475. String name = agendaNameController.text;
  476. final addItem = Agenda(
  477. id: "",
  478. talkId: "",
  479. name: name,
  480. );
  481. addItem.content = agendaContentController.text;
  482. for (AgendaListAllBean item in agendaAllList) {
  483. if (item.name == name) {
  484. List<Agenda> list = item.list ?? [];
  485. list.add(addItem);
  486. item.list = list;
  487. agendaAllList.refresh();
  488. agendaContentController.clear();
  489. agendaNameController.clear();
  490. return;
  491. }
  492. }
  493. agendaAllList.add(AgendaListAllBean(name: name, list: [addItem]));
  494. agendaContentController.clear();
  495. agendaNameController.clear();
  496. }
  497. Future<File?> getFileByTalk(TalkBean? bean) async {
  498. isLocalFileHas = false;
  499. if (bean == null) {
  500. return null;
  501. }
  502. if (bean.uploadType == TalkUploadType.localUpload) {
  503. String? audioId =
  504. FileUploadCheckHelper.getLocalAudioId(bean.localAudioUrl);
  505. if (audioId != null && audioId.isNotEmpty) {
  506. isLocalFileHas = true;
  507. return await AudioPickerUtils.getAssetFile(audioId);
  508. } else {
  509. return await FileUploadCheckHelper.getChoiceUploadFile(bean.id);
  510. }
  511. } else {
  512. return await RecordHandler.getRecordFile(bean.id);
  513. }
  514. }
  515. Future<bool> checkLogin() async {
  516. if (!accountRepository.isLogin.value) {
  517. bool isLogin = await LoginPage.start(fromType: LoginFromType.talkDetail);
  518. if (isLogin) {
  519. backToSpecificPage(RoutePath.mainTab);
  520. }
  521. return false;
  522. }
  523. return true;
  524. }
  525. void onShareClick() async {
  526. if (!await checkLogin()) {
  527. return;
  528. }
  529. if (talkBean.value?.status.value != TalkStatus.analysisSuccess) {
  530. return;
  531. }
  532. eventReport(EventId.event_101004);
  533. showTalkShareDialog(talkBean.value?.title.value,
  534. (type, shareTo, fileName, tag) {
  535. talkRepository
  536. .talkExport(talkBean.value!.id, fileName, type)
  537. .then((file) async {
  538. if (shareTo == ShareTo.ios) {
  539. await Share.shareXFiles([XFile(file.path)], subject: fileName);
  540. } else if (shareTo == ShareTo.wechat) {
  541. await SystemShareUtil.shareWechatFile(file.path);
  542. } else {
  543. await SystemShareUtil.shareQQFile(file.path);
  544. }
  545. SmartDialog.dismiss(tag: tag);
  546. }).catchError((error) {
  547. if (error is SystemShareException) {
  548. ToastUtil.showToast(error.message);
  549. } else {
  550. ErrorHandler.toastError(error);
  551. }
  552. });
  553. });
  554. }
  555. @override
  556. void onClose() {
  557. super.onClose();
  558. _talkUploadListener?.cancel();
  559. _talkBeanListener?.cancel();
  560. _audioPlayer.dispose();
  561. _agendaContentController?.dispose();
  562. _agendaNameController?.dispose();
  563. }
  564. void seekTo(int? startMs) {
  565. if (startMs == null) {
  566. return;
  567. }
  568. _audioPlayer.seek(Duration(milliseconds: startMs));
  569. }
  570. }