file_upload_check_helper.dart 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. import 'dart:io';
  2. import 'package:electronic_assistant/utils/pair.dart';
  3. import 'package:electronic_assistant/utils/toast_util.dart';
  4. import 'package:get/get.dart';
  5. import 'package:just_audio/just_audio.dart';
  6. import 'package:uuid/uuid.dart';
  7. import '../data/bean/talks.dart';
  8. import '../data/repositories/talk_repository.dart';
  9. import '../dialog/loading_dialog.dart';
  10. import '../module/talk/view.dart';
  11. import '../resource/string.gen.dart';
  12. import 'audio_picker_utils.dart';
  13. import 'package:path_provider/path_provider.dart';
  14. import 'common_utils.dart';
  15. import 'error_handler.dart';
  16. class FileUploadCheckHelper {
  17. FileUploadCheckHelper._();
  18. static Future<void> choicePlatformLocalFileAndCreateOrder() async {
  19. TalkBean? bean;
  20. try {
  21. Pair<File, Duration>? pair =
  22. await FileUploadCheckHelper.pickerPlatformFile();
  23. if (pair == null) {
  24. //未选择文件
  25. return;
  26. }
  27. LoadingDialog.show(StringName.fileImporting.tr);
  28. bean = await FileUploadCheckHelper.createTalkFromLocalFile(
  29. pair.first, pair.second);
  30. TalkPage.start(bean);
  31. } catch (e) {
  32. if (e is PickerException) {
  33. ToastUtil.showToast(e.message);
  34. } else {
  35. ErrorHandler.toastError(e, message: StringName.fileImportFail.tr);
  36. }
  37. } finally {
  38. LoadingDialog.hide();
  39. }
  40. if (bean != null) {
  41. TalkPage.start(bean);
  42. }
  43. }
  44. static Future<Duration> checkCanUpload(File file) async {
  45. //文件不能超过500M
  46. if (file.lengthSync() > 500 * 1024 * 1024) {
  47. throw PickerException(StringName.fileChoiceSizeLimit.tr);
  48. }
  49. AudioPlayer? player;
  50. try {
  51. player = AudioPlayer();
  52. player.setAudioSource(AudioSource.uri(file.uri));
  53. Duration? duration = await player.durationStream
  54. .firstWhere((duration) => duration != null);
  55. if (duration == null) {
  56. throw PickerException(StringName.fileAudioDurationCannotObtained.tr);
  57. }
  58. //录音时长不能超过5小时
  59. if (duration.inHours > 5) {
  60. throw PickerException(StringName.fileAudioDurationLimit.tr);
  61. }
  62. return duration;
  63. } finally {
  64. player?.dispose();
  65. }
  66. }
  67. static Future<Pair<File, Duration>?> pickerPlatformFile() async {
  68. File? file = await AudioPickerUtils.pickSingleFileByPlatform();
  69. if (file == null) {
  70. return null;
  71. }
  72. Duration duration = await checkCanUpload(file);
  73. return Pair(file, duration);
  74. }
  75. static Future<TalkBean> createTalkFromLocalFile(
  76. File file, Duration duration) async {
  77. TalkBean bean = await talkRepository.talkCreate(
  78. const Uuid().v4(), duration.inSeconds,
  79. uploadType: FileUploadType.local);
  80. String childDirName = bean.id;
  81. Directory dir = await getChoiceUploadDir(childDirName);
  82. await moveFileToDirectory(file, dir);
  83. return bean;
  84. }
  85. static Future<Directory> getChoiceUploadDir(String talkId) async {
  86. Directory documentDir = await getApplicationDocumentsDirectory();
  87. return Directory("${documentDir.path}/.atmob/choice/$talkId");
  88. }
  89. static Future<File?> getChoiceUploadFile(String talkId) async {
  90. Directory dir = await getChoiceUploadDir(talkId);
  91. if (!dir.existsSync()) {
  92. return null;
  93. }
  94. List<FileSystemEntity> list = dir.listSync();
  95. if (list.isEmpty) {
  96. return null;
  97. }
  98. return list.first as File;
  99. }
  100. static Future<void> moveFileToDirectory(File file, Directory dir) async {
  101. if (!dir.existsSync()) {
  102. dir.createSync(recursive: true);
  103. }
  104. String newFilePath = '${dir.path}/${file.uri.pathSegments.last}';
  105. file.renameSync(newFilePath);
  106. }
  107. static const String _uploadServerAudioTag = '_&&@@##_';
  108. //拼接设备ID用于标识
  109. static String joinUploadServerAudioTag(String id) {
  110. return '${getDeviceId()}$_uploadServerAudioTag$id';
  111. }
  112. static String? getLocalAudioId(String? tag) {
  113. if (tag == null || tag.isEmpty) {
  114. return null;
  115. }
  116. if (!tag.contains(_uploadServerAudioTag)) {
  117. return null;
  118. }
  119. List<String> audioTag = tag.split(_uploadServerAudioTag);
  120. if (audioTag.first != getDeviceId()) {
  121. return null;
  122. }
  123. return tag.split(_uploadServerAudioTag).last;
  124. }
  125. }
  126. class PickerException implements Exception {
  127. final String message;
  128. PickerException(this.message);
  129. @override
  130. String toString() {
  131. return 'PickerException: $message';
  132. }
  133. }