controller.dart 21 KB

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