mediaUpload.vue 14 KB

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