controller.dart 17 KB

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