欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

vue+flask實現(xiàn)視頻合成功能(拖拽上傳)

 更新時間:2021年03月04日 17:10:21   作者:代碼哈士奇  
這篇文章主要介紹了vue+flask實現(xiàn)視頻合成功能(拖拽上傳),本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下

vue+flask實現(xiàn)視頻合成
效果如下

在這里插入圖片描述

拖拽上傳我們之前一個文章有寫過

//www.dbjr.com.cn/article/206543.htm

原理就是 監(jiān)聽drop事件 來獲取拖拽的文件列表

在這里插入圖片描述
在這里插入圖片描述

上傳文件

通過axios 上傳文件

this,.fileList就是我們的文件列表

let files = this.fileList;
let formd = new FormData();
let i = 1;

//添加上傳列表
files.forEach(item => {
	formd.append(i + "", item, item.name)
	i++;
})
formd.append("type", i)
let config = {
	headers: {
		"Content-Type": "multipart/form-data"
	}
}

//上傳文件請求
axios.post("/qwe", formd, config).then(res => {
	console.log(res.data)
})

flask處理文件

完整代碼見最底部

邏輯如下
接收文件
為每次合成請求隨機生成一個文件夾 臨時保存文件
拼接視頻
返回文件路徑

@app.route("/file",methods=['POST'])
def test():

 #獲取文件
 files = request.files
 #合成隊列
 videoL = []
 #隨機字符串
 dirs = sjs()
 #生成文件夾
 os.mkdir(dirs)
 #保存文件并添加至合成隊列
 for file in files.values():
  print(file)
  dst = dirs + "/" + file.name + ".mp4"
  file.save(dst)
  video = VideoFileClip(dirs + "/" + file.name + ".mp4")
  videoL.append(video)
 
 #拼接視頻
 final = concatenate_videoclips(videoL)
 #文件路徑
 fileName = dirs + "/" +"{}.mp4".format(sjs())
 #生成視頻
 final.to_videofile(fileName)
 
 #銷毀文件夾
 def sc():
  shutil.rmtree(dirs)
 
 #30秒后銷毀文件夾
 timer = threading.Timer(30, sc)
 timer.start()

 # 返回文件路徑
 return fileName

拼接獲取文件路徑

首先我們看flask

邏輯如下
通過文件名 獲取文件 返回文件

app.route("/getvoi",methods=['GET'])
def getImg():
 #獲取文件名
 ss = request.args['name']
 #文件加至返回響應(yīng)
 response = make_response(
  send_file(ss))

 #刪除文件
 def sc():
  os.remove(ss)
 
 #30秒后刪除文件
 timer = threading.Timer(30, sc)
 timer.start()
 
 return response

前端獲取

通過a標簽下載

<a s :href="herfs" rel="external nofollow" rel="external nofollow" :download="fileName">下載</a>

herfs如下

在這里插入圖片描述

我們上傳文件后 通過falsk處理返回文件路徑 拼接后獲取文件地址

a標簽添加download屬性可以給下載的文件命名

如果你對/qwe /voi有疑惑 請看下面的配置代理說明

配置代理說明

配置代理是為了解決跨域問題 開發(fā)環(huán)境可在vue.config.js配置即可使用
生產(chǎn)環(huán)境需要額外配置nginx

在這里插入圖片描述

/qwe實際上就是 http://127.0.0.1:8087/file
/voi實際上就是 http://127.0.0.1:8087/getvoi
對應(yīng)我們flask中的

在這里插入圖片描述

額外說明(如果你使用uni-app)

如果你使用uni-app 可參照文檔使用api
上傳文件api https://uniapp.dcloud.io/api/request/network-file?id=uploadfile
下載文件api https://uniapp.dcloud.io/api/request/network-file?id=downloadfile
或者直接使用別人封裝好的 插件畢竟比較方便

完整代碼

如果你不想一個一個復(fù)制可以去下載
下載途徑1: https://download.csdn.net/download/qq_42027681/15561897
下載途徑2:https://github.com/dmhsq/vue-flask-videoSynthesis

flask代碼

md5random.py 用于隨機字符串生成

import random
import hashlib
def sjs():
 a = random.randint(0, 100)
 a = "a" + str(a);
 b = random.randint(100, 10000);
 b = "b" + str(b);
 c = hashlib.md5(a.encode(encoding='UTF-8')).hexdigest() + hashlib.md5(b.encode(encoding='UTF-8')).hexdigest();
 c = "c" + str(c);
 d = random.randint(10, 100);
 d = "d" + str(d);
 e = hashlib.md5(c.encode(encoding='UTF-8')).hexdigest() + hashlib.md5(d.encode(encoding='UTF-8')).hexdigest();
 e = hashlib.md5(e.encode(encoding='UTF-8')).hexdigest()
 return e;

app_service.py 服務(wù)代碼

from flask import Flask,request,send_file,make_response
import os,json,threading,shutil
from moviepy.editor import *
from md5random import sjs

app = Flask(__name__)

@app.route("/file",methods=['POST'])
def test():

 #獲取文件
 files = request.files
 #合成隊列
 videoL = []
 #隨機字符串
 dirs = sjs()
 #生成文件夾
 os.mkdir(dirs)
 #保存文件并添加至合成隊列
 for file in files.values():
  print(file)
  dst = dirs + "/" + file.name + ".mp4"
  file.save(dst)
  video = VideoFileClip(dirs + "/" + file.name + ".mp4")
  videoL.append(video)

 #拼接視頻
 final = concatenate_videoclips(videoL)
 #文件路徑
 fileName = dirs + "/" +"{}.mp4".format(sjs())
 #生成視頻
 final.to_videofile(fileName)

 #銷毀文件夾
 def sc():
  shutil.rmtree(dirs)

 #30秒后銷毀文件夾
 timer = threading.Timer(30, sc)
 timer.start()

 # 返回文件路徑
 return fileName


@app.route("/getvoi",methods=['GET'])
def getImg():
 #獲取文件名
 ss = request.args['name']
 #文件加至返回響應(yīng)
 response = make_response(
  send_file(ss))

 #刪除文件
 def sc():
  os.remove(ss)

 #30秒后刪除文件
 timer = threading.Timer(30, sc)
 timer.start()

 return response

if __name__ == '__main__':
 app.run(host='0.0.0.0',port=8087)

vue代碼

演示文件代碼

<template>
 <div>
 <div
  v-on:dragover="tts"
  v-on:drop="ttrs"
  style="width: 800px;height: 200px;border: 1px solid black;font-size: 40px;line-height: 200px"
 >
  {{ dt }}
 </div>
 <div
  v-for="(item, index) in fileList"
  :key="index"
  style="width: 800px;height: 200px;border: 1px solid black;font-size: 40px;position: relative;top:10px"
 >
  <p
  style="font-size: 20px;float: left;position: relative;left: 20pxword-wrap:break-word;word-break:normal;"
  >
  {{ item.name }}
  </p>
  <h5 style="float:right;position: absolute;top: 80px;right: 20px">
  {{ item.type }}
  </h5>
  <h6 style="position: absolute;top: 80px;float: left;left: 20px">
  {{ item.size | sizeType }}
  </h6>
  <button style="float: right" @click="del(index)">刪除</button>
 </div>
 <!-- 此處為展示最后一個上傳的文件 -->
<!-- <div style="position:relative;top: 100px">-->
<!--  <img v-if="isImage" :src="srcs" style="width: 800px" />-->
<!--  <video v-if="isVideo" controls :src="srcs" style="width: 800px"></video>-->
<!--  <audio v-if="isAudio" controls :src="srcs" style="width: 800px"></audio>-->
<!-- </div>-->

 <el-button style="position: relative;top: 50px" type="success" @click="ups()" :disabled="!isCan">合成</el-button>
 <el-button style="position: relative;top: 50px" v-loading="loading" type="success" >。。。</el-button>
 <a style="position: relative;top: 50px;left: 15px;" type="success" :href="herfs" rel="external nofollow" rel="external nofollow" :download="fileName"><el-button :disabled="isCans"><span style="color: black">下載</span></el-button></a>
 <div style="position: relative;top: 100px">文件下載有效時間{{times}}s</div>
 </div>
</template>

<script>
import axios from "axios";

export default {
 name: "trs",
 data() {
 return {
  dt: "",//上傳提醒 "拖動到此處上傳文件“或者"上傳完成,可繼續(xù)上傳"
  fileList: [],//文件列表
  loading:false,
  srcs: "",//圖片/視頻/音頻 base64
  isImage: false,//是否是圖片
  isAudio: false,//是否是音頻
  isVideo: false,//是否是視頻
  isCan: true,//是否能合成
  isCans:true,//是否能下載
  herfs: "",//下載地址
  fileName: "",//文件名
  times: 25//下載有效時間
 };
 },
 filters: {
 //格式化文件大小
 sizeType(val) {
  let kbs = val / 1024;
  let mbs = 0;
  let gbs = 0;
  if (kbs >= 1024) {
  mbs = kbs / 1024;
  }
  if (mbs >= 1024) {
  gbs = mbs / 1024;
  return gbs.toFixed(2) + "GB";
  } else if (mbs >= 1) {
  return mbs.toFixed(2) + "MB";
  } else {
  return kbs.toFixed(2) + "KB";
  }
 }
 },
 mounted() {
 let vm = this;
 window.addEventListener("dragdrop", this.testfunc, false);

 //全局監(jiān)聽 當頁面內(nèi)有文件拖動 提醒拖動到此處
 document.addEventListener("dragover", function() {
  console.log(111);
  vm.dt = "拖動到此處上傳文件";
  console.log(vm.dt);
 });
 },
 methods: {
 //展示文件 主要為三個類型 圖片/視頻/音頻
 readFile(file) {
  let vm = this;
  let reader = new FileReader();
  reader.readAsDataURL(file);
  reader.onload = function() {
  let type = file.type.substr(0, 5);
  if (type == "image") {
   vm.isImage = true;
   vm.isAudio = false;
   vm.isVideo = false;
  } else if (type == "audio") {
   vm.isImage = false;
   vm.isAudio = true;
   vm.isVideo = false;
  } else if (type == "video") {
   vm.isImage = false;
   vm.isAudio = false;
   vm.isVideo = true;
  } else {
   alert("不是圖片/視頻/音頻");
  }
  vm.srcs = reader.result;
  // this.$nextTick(()=>{
  //
  // })
  };
 },
 //全局監(jiān)聽drop的觸發(fā)事件 取消drop彈窗顯示資源
 testfunc(event) {
  alert("dragdrop!");

  //取消drop彈窗顯示資源
  event.stopPropagation();
  event.preventDefault();
 },
 del(index) {
  this.fileList.splice(index, 1);
  if (this.fileList.length === 0) {
  this.dt = "";
  }
 },
 //監(jiān)聽div上傳框 當有文件拖動時 顯示"拖動到此處上傳文件"
 tts(e) {
  console.log(e);
  this.dt = "拖動到此處上傳文件";
 },
 //監(jiān)聽div上傳框 drop事件觸發(fā)
 ttrs(e) {
  console.log(e);
  console.log(e.dataTransfer.files);

  //獲取文件
  let datas = e.dataTransfer.files;

  //取消drop彈窗顯示資源
  e.stopPropagation();
  e.preventDefault();
  datas.forEach(item => {
  if(item.type=="video/mp4"){
   this.fileList.push(item);
  }
  });

  //讀取文件 如果不想展示圖片/視頻/音頻可忽略
  this.readFile(this.fileList[this.fileList.length - 1]);



  this.dt = "上傳完成,可繼續(xù)上傳";
 },

 //上傳文件到服務(wù)器
 ups(){
  if(this.fileList.length==0){
  this.$message('文件列表為空');
  return ;
  }
  this.loading = true;
  this.isCan = false;
  this.isCans = true;
  let files = this.fileList;
  let formd = new FormData();
  let i = 1;

  //添加上傳列表
  files.forEach(item=>{
  formd.append(i+"",item,item.name)
  i++;
  })
  formd.append("type",i)
  let config={
  headers:{"Content-Type":"multipart/form-data"}
  }

  //上傳文件請求
  axios.post("/qwe",formd,config).then(res=>{
  console.log(res.data)
  this.loading = false
  //合成下載路徑
  this.herfs = "/voi?name="+res.data

  this.fileName = res.data.split('/')[1]
  //禁止合成
  this.isCan = false

  this.isCans = false

  //設(shè)置下載有效時間 時間到后無法下載但可以繼續(xù)合成
  let timer = setInterval(()=>{
   this.times--;
  },1000)
  this.setCans(timer)
  })
 },
 setCans(timer){
  setTimeout(()=>{
  this.isCans = true
  this.isCan = true
  this.fileName =""
  clearInterval(timer)
  this.times = 25
  },25000)
 }
 }
};
</script>

<style scoped></style>

vue.config.js

module.exports = {
 devServer: {
 // assetsSubDirectory: 'static',
 // assetsPublicPath: '/',
 proxy: {
  "/qwe": {
  target: "http://127.0.0.1:8087/file",
  changeOrigin: true,
  pathRewrite: {
   "^/qwe": ""
  }
  },
  "/voi": {
  target: "http://127.0.0.1:8087/getvoi",
  changeOrigin: true,
  pathRewrite: {
   "^/voi": ""
  }
  }
 }
 }
};

到此這篇關(guān)于vue+flask實現(xiàn)視頻合成功能(拖拽上傳)的文章就介紹到這了,更多相關(guān)vue視頻合成內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Vue如何實現(xiàn)分頁功能代碼實例

    Vue如何實現(xiàn)分頁功能代碼實例

    這篇文章主要給大家介紹了關(guān)于Vue如何實現(xiàn)分頁功能的相關(guān)資料,Vue分頁功能的實現(xiàn)需要前端和后端共同配合完成,文中通過代碼實例介紹的非常詳細,需要的朋友可以參考下
    2023-09-09
  • Vue中mixins混入的介紹和使用詳解

    Vue中mixins混入的介紹和使用詳解

    mixins(混入)是一種分發(fā)?Vue?組件中可復(fù)用功能的非常靈活的方式,這篇文章主要為大家介紹了mixins混入的介紹和使用,需要的可以參考下
    2023-08-08
  • Vuex中g(shù)etters和actions的使用補充說明

    Vuex中g(shù)etters和actions的使用補充說明

    這篇文章主要介紹了在Vuex中關(guān)于getters和actions使用的補充作了簡要說明,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步
    2021-09-09
  • 關(guān)于VUE點擊父組件按鈕跳轉(zhuǎn)到子組件的問題及解決方案

    關(guān)于VUE點擊父組件按鈕跳轉(zhuǎn)到子組件的問題及解決方案

    本文主要介紹了在Vue框架中,如何通過父組件的點擊事件打開子組件中的彈窗并展示表格內(nèi)容,主要步驟包括在父組件中定義數(shù)據(jù)屬性,創(chuàng)建并定義子組件的彈窗和表格內(nèi)容,通過props屬性和自定義事件實現(xiàn)父子組件間的數(shù)據(jù)傳遞和方法調(diào)用
    2024-10-10
  • vue3使用vue-i18n的方法詳解(ts中使用$t,?vue3不用this)

    vue3使用vue-i18n的方法詳解(ts中使用$t,?vue3不用this)

    所謂的vue-i18n國際化方案就是根據(jù)它的規(guī)則自己建立一套語言字典,對于每一個字(message)都有一個統(tǒng)一的標識符,下面這篇文章主要給大家介紹了關(guān)于vue3使用vue-i18n(ts中使用$t,?vue3不用this)的相關(guān)資料,需要的朋友可以參考下
    2022-12-12
  • vue.js云存儲實現(xiàn)圖片上傳功能

    vue.js云存儲實現(xiàn)圖片上傳功能

    示對象存儲是騰訊云提供的一種存儲海量文件的分布式存儲服務(wù),本文主要介紹了用vue.js實現(xiàn)圖片上傳功能,感興趣的小伙伴們可以參考一下
    2021-05-05
  • 優(yōu)雅的elementUI table單元格可編輯實現(xiàn)方法詳解

    優(yōu)雅的elementUI table單元格可編輯實現(xiàn)方法詳解

    這篇文章主要介紹了優(yōu)雅的elementUI table單元格可編輯實現(xiàn)方法詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-12-12
  • Vuejs對象常用操作之取對應(yīng)的值、取key和value值、轉(zhuǎn)數(shù)組及合并等

    Vuejs對象常用操作之取對應(yīng)的值、取key和value值、轉(zhuǎn)數(shù)組及合并等

    最近在學Vue和javascript感覺js的好多方法都不太清楚,這里徹底總結(jié)下,這篇文章主要給大家介紹了關(guān)于Vuejs對象常用操作之取對應(yīng)的值、取key和value值、轉(zhuǎn)數(shù)組及合并等的相關(guān)資料,需要的朋友可以參考下
    2024-01-01
  • Vue使用Axios和elementui實現(xiàn)查詢分頁功能

    Vue使用Axios和elementui實現(xiàn)查詢分頁功能

    當今的Web開發(fā)趨勢中,前后端分離已經(jīng)成為一種流行的架構(gòu)模式,它將前端和后端的開發(fā)分離開來,使得前端和后端可以獨立進行開發(fā)和部署,本文給大家介紹了Vue使用Axios和elementui實現(xiàn)查詢分頁功能,需要的朋友可以參考下
    2024-06-06
  • vue自定義權(quán)限標簽詳細代碼示例

    vue自定義權(quán)限標簽詳細代碼示例

    這篇文章主要給大家介紹了關(guān)于vue自定義權(quán)限標簽的相關(guān)資料,在Vue中你可以通過創(chuàng)建自定義組件來實現(xiàn)自定義標簽組件,文中給出了詳細的代碼示例,需要的朋友可以參考下
    2023-09-09

最新評論