mediaUpload.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444
  1. <template>
  2. <div>
  3. <el-dialog
  4. :visible.sync="diaShow"
  5. :close-on-click-modal="false"
  6. :modal-append-to-body='false'
  7. :title="$t('SemanticsManage.ASRpage.upload_title')"
  8. @close="closeHandle"
  9. center top="10vh"
  10. width="800px">
  11. <div class="dialog-main">
  12. <el-form
  13. ref="diaForm"
  14. label-position="left"
  15. hide-required-asterisk
  16. :model="formData">
  17. <!-- 所属目录 -->
  18. <el-form-item :label="$t('SemanticsManage.ASRpage.belonging_directory')" :rules="{required:true,message:$t('SemanticsManage.ASRpage.belonging_directory_required'),trigger:'change'}"
  19. prop="catalogue" label-width="76px">
  20. <el-cascader v-model="formData.catalogue" :placeholder="$t('SemanticsManage.ASRpage.belonging_directory_placeholder')" clearable id="media-upload-cascader"
  21. :options="articleCatalogue" :props="{value:'MenuId',label:'MenuName',children:'Children',checkStrictly:true,emitPath:false}"
  22. style="width: 604px;"></el-cascader>
  23. </el-form-item>
  24. </el-form>
  25. <div class="file-upload-zone">
  26. <div class="file-upload-title">
  27. <!-- 文件数量 -->
  28. <span>{{$t('SemanticsManage.ASRpage.file_count')}} :{{ fileList.length }} / {{ this.fileLimit }}</span>
  29. <span v-if="fileList.length>0" class="file-readdition"
  30. :class="{'disable-readdition':!( (this.fileList.length+this.uploadingFileNames.length) <fileLimit)}" @click="uploadClick">{{ $t('SemanticsManage.ASRpage.continue_adding') }}</span>
  31. </div>
  32. <div class="file-upload-main" id="drop-upload-zone" @dragover="uploadDragover" @drop="uploadDrop">
  33. <el-table :data="fileList" id="file-list-table">
  34. <el-table-column :label="$t('SemanticsManage.ASRpage.file_name')" prop="name" show-overflow-tooltip >
  35. <template slot-scope="{row}">
  36. <span class="table-span">{{ row.name }}</span>
  37. </template>
  38. </el-table-column>
  39. <el-table-column :label="$t('SemanticsManage.ASRpage.file_size')" prop="sizeText" width="120">
  40. <template slot-scope="{row}" >
  41. <span class="table-span">{{ row.sizeText }}</span>
  42. </template>
  43. </el-table-column>
  44. <el-table-column :label="$t('SemanticsManage.ASRpage.audio_video')" prop="type" width="100">
  45. <template slot-scope="{row}">
  46. <span class="table-span">{{ row.type }}</span>
  47. </template>
  48. </el-table-column>
  49. <el-table-column :label="$t('Table.column_operations')" prop="operation" width="60">
  50. <template slot-scope="{row,$index}">
  51. <span class="table-button" @click="deleteFile($index)">{{ $t('Table.delete_btn') }}</span>
  52. </template>
  53. </el-table-column>
  54. <div slot="empty" class="no-file-show" @click="uploadClick">
  55. <div class="upload-text"><span style="color:#0052D9 ;">{{ $t('SemanticsManage.ASRpage.start_upload_hint') }}</span>{{ $t('SemanticsManage.ASRpage.start_upload_hint2') }}</div>
  56. <div class="upload-message-box">
  57. <div class="upload-message-row">{{$t('SemanticsManage.ASRpage.supported_audio_formats')}}:mp3、wav、m4a、amr、wma、aac、ogg-opus、flac</div>
  58. <div class="upload-message-row">{{$t('SemanticsManage.ASRpage.supported_video_formats')}}:mp4、flv、3gp</div>
  59. <div class="upload-message-row">{{$t('SemanticsManage.ASRpage.upload_description')}}</div>
  60. </div>
  61. </div>
  62. </el-table>
  63. </div>
  64. </div>
  65. <div class="dia-bot">
  66. <el-button type="primary" plain style="margin-right:20px;min-width:120px ;" @click="cancelHandle">{{ $t('Dialog.cancel_btn') }}</el-button>
  67. <!-- 开始转写 -->
  68. <el-button type="primary" @click="saveHandle" style="min-width:120px;" :disabled="fileList.length==0">{{ $t('SemanticsManage.ASRpage.start_trans') }}</el-button>
  69. </div>
  70. <!-- :http-request="handleUpload"
  71. :before-upload="handleBeforeUpload" -->
  72. <el-upload
  73. ref="uploadRef" id="upload-media" style="display: none;"
  74. accept=".mp3,.wav,.m4a,.amr,.wma,.aac,.opus,.ogg,.flac,.mp4,.flv,.3gp"
  75. action="#" multiple :limit="fileLimit"
  76. :on-change="uploadMedia"
  77. :show-file-list="false"
  78. :auto-upload="false"
  79. :disabled="!((this.fileList.length+this.uploadingFileNames.length)<fileLimit)">
  80. </el-upload>
  81. </div>
  82. </el-dialog>
  83. </div>
  84. </template>
  85. <script>
  86. import {uploadFileDirect} from "@/utils/common.js"
  87. import {asrInterface} from '@/api/modules/semanticsApi.js';
  88. import MD5 from "js-md5";
  89. export default {
  90. name:"editTag",
  91. props:{
  92. diaShow:{
  93. type:Boolean,
  94. default:false
  95. },
  96. articleCatalogue:{
  97. type:Array,
  98. required:true
  99. }
  100. },
  101. watch:{
  102. uploadingFileNames(value){
  103. if(!(value && value.length>0)){
  104. if(this.uploadHint){
  105. this.uploadHint.close()
  106. this.uploadHint=null
  107. }
  108. if(this.closeDia) return
  109. this.$message.success(/* "上传完成,请检查转写列表" */this.$t('SemanticsManage.ASRpage.success_upload_hint'))
  110. }
  111. },
  112. diaShow(value){
  113. if(value){
  114. this.current++
  115. }
  116. }
  117. },
  118. data() {
  119. return {
  120. current:0,
  121. diaTitle:'上传本地音视频文件',
  122. formData:{
  123. catalogue:''
  124. },
  125. fileLimit:50,
  126. fileExp:new RegExp(/\.(mp3|wav|m4a|amr|wma|aac|opus|ogg|flac|mp4|flv|3gp)$/,'i'),
  127. videoExp:new RegExp(/\.(mp4|flv|3gp)$/,'i'),
  128. fileList:[],
  129. uploadingFileNames:[],
  130. uploadHint:null,
  131. closeDia:false
  132. }
  133. },
  134. methods: {
  135. uploadClick(){
  136. if(!( (this.fileList.length+this.uploadingFileNames.length) < this.fileLimit)) return
  137. $('#upload-media input').trigger('click')
  138. },
  139. uploadDragover(event){
  140. event.preventDefault(); //阻止默认行为,允许放置
  141. },
  142. uploadDrop(event){
  143. event.preventDefault(); //阻止浏览器默认行为
  144. // 获取文件的数据
  145. const DataTransferItemList = event.dataTransfer.files
  146. DataTransferItemList.forEach(file =>{
  147. file.raw=file
  148. setTimeout(()=>{
  149. this.uploadMedia(file)
  150. })
  151. })
  152. },
  153. async uploadMedia(file){
  154. // console.log(file,this.uploadingFileNames,'this.uploadingFileNames');
  155. let fileFront=file.name.split('.')[0]
  156. if(this.uploadingFileNames.some(item=> item == fileFront)){
  157. return this.$message.error(file.name+/* '同名文件正在上传,请稍后重试' */this.$t('SemanticsManage.ASRpage.same_file_hint'))
  158. }
  159. if(!(this.fileExp.test(file.name))){
  160. return this.$message.error(file.name+/* '格式不受支持' */ this.$t('SemanticsManage.ASRpage.format_hint'))
  161. }
  162. if(!(file.size/1024/1024 < 1024.1)){
  163. return this.$message.error(file.name+/* '大小已超限' */this.$t('SemanticsManage.ASRpage.size_hint'))
  164. }
  165. let res = await this.checkDuration(file)
  166. if(!res.flag){
  167. return this.$message.error(res.msg);
  168. }
  169. let audioDuration = res.duration
  170. if(!( (this.fileList.length+this.uploadingFileNames.length) < this.fileLimit)){
  171. return this.$message.error(
  172. /* `上传文件已超过${this.fileLimit}个,${file.name}上传失败` */
  173. this.$t('SemanticsManage.ASRpage.limit_hint',{num:this.fileLimit,name:file.name})
  174. )
  175. }
  176. // 判断于 在库的文件名是否重名
  177. let flag=true
  178. let checkRes = await asrInterface.speechFlieRepetitionCheck({FileName:fileFront})
  179. if(checkRes.Ret == 200) flag=false
  180. if(flag) return
  181. if(!this.uploadHint){
  182. this.uploadHint = this.$message({
  183. type:"info",
  184. message:this.$t('SemanticsManage.ASRpage.uploading_hint'),
  185. duration:0,
  186. iconClass:'el-icon-loading'
  187. })
  188. }
  189. // 上传
  190. this.uploadingFileNames.push(fileFront)
  191. const t = new Date().getTime().toString();
  192. const temName = `asr/media/${process.env.NODE_ENV}/${MD5(t)}.${
  193. file.raw.type.split("/")[1]
  194. }`;
  195. let clientType = this.$setting.dynamicOutLinks.ObjectStorageClient ||
  196. this.$store.state.dynamicOutLinks.ObjectStorageClient ||
  197. JSON.parse(localStorage.getItem('dynamicOutLinks')).ObjectStorageClient
  198. let options={}
  199. if((file.size/1024/1024 > 100)){
  200. // 大于100MB,分片上传
  201. console.log("切片上传");
  202. // 上传的配置
  203. options={
  204. // 阿里云
  205. OSS:{
  206. // 设置并发上传的分片数量。
  207. parallel: 10,
  208. // 设置分片大小。默认值为1 MB,最小值为100 KB。
  209. partSize: 1024 * 1024 * 10, // 10MB
  210. },
  211. /**
  212. * minIO的上传方法自动将较大的文件进行分片,不需要特殊配置
  213. * 官网文档翻译文:单个对象的最大大小限制为5TB。putObject透明地将大于64MiB的对象分成多个部分上传。使用MD5SUM签名仔细验证上传的数据。
  214. */
  215. //S3
  216. S3:{
  217. partSize: 10 * 1024 * 1024, // 设置每个分片大小为10MB,默认为8MB。
  218. queueSize: 10, // 并发数,默认为4。
  219. }
  220. }
  221. }
  222. let windowNum=this.current
  223. uploadFileDirect(clientType,file.raw,temName,options).then(res=>{
  224. if(!this.diaShow || (this.current!=windowNum)) return
  225. let item = {
  226. name:fileFront,
  227. fileUrl:res,
  228. sizeText:this.sizeFormat(file.size),
  229. size:file.size,
  230. type:this.videoExp.test(res)?this.$t('SemanticsManage.ASRpage.video_text'):this.$t('SemanticsManage.ASRpage.audio_text'),
  231. duration:audioDuration || 0
  232. }
  233. this.fileList.push(item)
  234. }).finally(()=>{
  235. this.deleteUploadItem(fileFront)
  236. })
  237. },
  238. sizeFormat(s){
  239. let size = Number(s)
  240. if(size<1024){
  241. return size+'B'
  242. }else if(size<(1024*1024)){
  243. return Math.floor((size/1024)*100)/100 +'KB'
  244. }else{
  245. return Math.floor((size/1024/1024)*100)/100 +'MB'
  246. }
  247. },
  248. deleteUploadItem(name){
  249. let index = this.uploadingFileNames.findIndex(fileName=> fileName==name)
  250. if(index!=-1){
  251. this.uploadingFileNames.splice(index,1)
  252. }
  253. },
  254. checkDuration(file){
  255. return new Promise((resolve,reject)=>{
  256. try {
  257. let url = URL.createObjectURL(file.raw);
  258. let audioElement = new Audio(url);
  259. audioElement.addEventListener('loadeddata', () => {
  260. if(audioElement.duration>(60*60*5+1)){
  261. resolve({flag:false,msg:file.name+/* '时长已超限' */this.$t('SemanticsManage.ASRpage.time_hint'),duration:audioElement.duration})
  262. }else{
  263. resolve({flag:true,msg:'',duration:audioElement.duration})
  264. }
  265. URL.revokeObjectURL(url);
  266. });
  267. setTimeout(()=>{
  268. resolve({flag:false,msg:file.name+/* "获取不到时长,请重试" */this.$t('SemanticsManage.ASRpage.time_fail')})
  269. },5000)
  270. } catch (error) {
  271. this.$message.error(error.msg)
  272. console.error(error.msg);
  273. reject(error)
  274. }
  275. })
  276. },
  277. deleteFile(index){
  278. this.fileList.splice(index,1)
  279. },
  280. cancelHandle(){
  281. this.$emit("update:diaShow",false)
  282. },
  283. saveHandle(){
  284. if(this.uploadingFileNames && this.uploadingFileNames.length>0){
  285. return this.$message.error(/* "还有音/视频未上传完成,请等待上传成功后再转写!" */this.$t('SemanticsManage.ASRpage.trans_hint'))
  286. }
  287. let params={
  288. MenuId:this.formData.catalogue,
  289. Files:this.fileList.map(f =>{
  290. return {
  291. FileName:f.name,
  292. ResourceUrl:f.fileUrl,
  293. FileSecond:Math.round(f.duration),
  294. FileSize:f.size
  295. }
  296. })
  297. }
  298. this.$refs.diaForm.validate(valid=>{
  299. if(valid){
  300. asrInterface.speechTransfer(params).then(res=>{
  301. if(res.Ret == 200){
  302. this.$message.success(/* "提交转写成功" */ this.$t('SemanticsManage.ASRpage.trans_success_hint'))
  303. this.$emit("uploadSuccess")
  304. this.$emit("update:diaShow",false)
  305. }
  306. })
  307. }
  308. })
  309. },
  310. closeHandle(){
  311. this.formData={}
  312. this.closeDia=true
  313. this.uploadingFileNames=[]
  314. this.fileList=[]
  315. this.$emit("update:diaShow",false)
  316. this.$nextTick(()=>{
  317. this.$refs.diaForm.clearValidate()
  318. this.closeDia=false
  319. })
  320. }
  321. },
  322. }
  323. </script>
  324. <style lang="scss" scoped>
  325. @mixin font-type {
  326. font-size: 14px;
  327. line-height: 22px;
  328. font-weight: 400;
  329. color: #333333;
  330. }
  331. .dialog-main{
  332. padding: 5px 35px 35px;
  333. .file-upload-zone{
  334. .file-upload-title{
  335. display: flex;
  336. align-items: center;
  337. justify-content: space-between;
  338. span{
  339. @include font-type
  340. }
  341. .file-readdition{
  342. color: #0052D9;
  343. cursor: pointer;
  344. }
  345. .disable-readdition{
  346. color: #999999;
  347. cursor:not-allowed;
  348. }
  349. }
  350. .file-upload-main{
  351. width: 100%;
  352. height: 380px;
  353. box-sizing: border-box;
  354. // border: 1px solid #C8CDD9;
  355. background-color: #ECF2FE;
  356. border-radius: 4px;
  357. margin-top: 10px;
  358. .table-span{
  359. @include font-type;
  360. }
  361. .table-button{
  362. @include font-type;
  363. color: #C54322;
  364. cursor: pointer;
  365. }
  366. .no-file-show{
  367. height: 332px;
  368. width: 100%;
  369. box-sizing: border-box;
  370. padding: 130px 0 30px;
  371. cursor: pointer;
  372. .upload-text{
  373. margin-bottom:80px;
  374. @include font-type
  375. }
  376. .upload-message-box{
  377. text-align: left;
  378. padding-left: 30px;
  379. .upload-message-row{
  380. @include font-type;
  381. color: #666666;
  382. }
  383. }
  384. }
  385. }
  386. }
  387. .dia-bot{
  388. text-align: center;
  389. margin-top: 60px;
  390. }
  391. }
  392. </style>
  393. <style lang="scss">
  394. @mixin font-type {
  395. font-size: 14px;
  396. line-height: 22px;
  397. font-weight: 400;
  398. color: #333333;
  399. }
  400. .el-cascader-node{
  401. max-width: 200px;
  402. }
  403. #media-upload-cascader{
  404. .el-input{
  405. width: 100%;
  406. }
  407. }
  408. #file-list-table{
  409. background-color: #ECF2FE;
  410. border: 1px solid #C8CDD9;
  411. border-radius: 4px;
  412. th{
  413. background-color: #ECF2FE!important;
  414. border-color: #C8CDD9!important;
  415. @include font-type;
  416. color: #999999;
  417. }
  418. td{
  419. border: none;
  420. background-color: #ECF2FE!important;
  421. padding: 10px 0;
  422. }
  423. .el-table__body-wrapper{
  424. height: 332px;
  425. overflow-y: auto;
  426. }
  427. .el-table__empty-block{
  428. .el-table__empty-text{
  429. width: 100%;
  430. color: unset;
  431. line-height: unset;
  432. }
  433. }
  434. }
  435. </style>