上传视频的核心代码

  /*** 上传学习视频信息*/@Log(title = "上传学习视频信息", businessType = BusinessType.INSERT)@PostMapping("/uploadVideo")public AjaxResult add(HttpServletRequest request) {return toAjax(videoInfoService.insertVideoInfo(request));}/*** 上传学习视频信息** @param request 学习视频信息* @return 结果*/@Overridepublic int insertVideoInfo(HttpServletRequest request) {// 获取视频文件MultipartFile videoFile = ((MultipartHttpServletRequest) request).getFile("file");VideoInfo sysVideoInfo = new VideoInfo();if ((videoFile != null) && (!videoFile.isEmpty())) {// 获取项目路径String projectPath = RuoYiConfig.getUploadPath();// 获得文件名String oriFileName = videoFile.getOriginalFilename();//不带点后缀String suffix = FilenameUtils.getExtension(oriFileName);// 获取文件大小long size = videoFile.getSize() / 1024;// 设置默认deviceNumberString deviceNumber = "learning";sysVideoInfo.setDeviceNumber(deviceNumber);// 获取文件夹路径String folderPath = projectPath + File.separator + "videoInfo/info" + File.separator + deviceNumber + File.separator;// 获取文件路径String filePath = folderPath + oriFileName;//判断文件夹是否存在File file1 = new File(folderPath);if (!file1.exists()) {file1.mkdirs();//创建目录}File dest = new File(filePath);if (dest.exists()) {//文件已存在return 2;}try {videoFile.transferTo(dest);sysVideoInfo.setFileMd5(Md5Util.getFileMD5(dest));} catch (IOException e) {e.printStackTrace();}// 设置IDsysVideoInfo.setId(GUIDCreater.getGUID());// 设置视频类型//sysVideoInfo.setVideoType(videoType);// 设置文件名sysVideoInfo.setFileName(oriFileName);// 设置存储路径sysVideoInfo.setFilePath(filePath);// 设置下载路径String downloadPath = "videoInfo/info/download/" + deviceNumber + "/" + oriFileName.substring(0, oriFileName.lastIndexOf(".")) + "/" + suffix;sysVideoInfo.setDownloadPath(downloadPath);// 设置文件大小sysVideoInfo.setFileSize(String.valueOf(size));} else {return 0;}return videoInfoMapper.insertVideoInfo(sysVideoInfo);}/*** 下载视频** @param fileName   文件名* @param fileSuffix 文件后缀* @param request* @param response* @throws IOException* @throws ServletException*/@GetMapping("/download/{device_number}/{file_name}/{file_suffix}")public void downloadProgramFile(@PathVariable("device_number") String deviceNumber, @PathVariable("file_name") String fileName, @PathVariable("file_suffix") String fileSuffix, HttpServletRequest request, HttpServletResponse response) throws IOException, ServletException {fileName += "." + fileSuffix;String localPath = RuoYiConfig.getProfile();String fileRealPath = localPath + File.separator + "upload" + File.separator + "/videoInfo/info" + File.separator + deviceNumber + File.separator + fileName;//设置文件路径File file = new File(fileRealPath);if (!file.exists()) {response.sendError(404, "视频文件不存在,下载失败");return;}Path filePath = Paths.get(fileRealPath);String mimeType = Files.probeContentType(filePath);response.setContentType(mimeType);request.setAttribute(NonStaticResourceHttpRequestHandler.ATTR_FILE, filePath);nonStaticResourceHttpRequestHandler.handleRequest(request, response);}//新建一个文件
package com.ruoyi.image;
import org.springframework.core.io.FileSystemResource;
import org.springframework.core.io.Resource;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.resource.ResourceHttpRequestHandler;
import javax.servlet.http.HttpServletRequest;
import java.nio.file.Path;@Component
public class NonStaticResourceHttpRequestHandler extends ResourceHttpRequestHandler {public final static String ATTR_FILE = "NON-STATIC-FILE";@Overrideprotected Resource getResource(HttpServletRequest request) {final Path filePath = (Path) request.getAttribute(ATTR_FILE);return new FileSystemResource(filePath.toString());}}-------------------------------------------------------------------------------------------------------------
前端记录
<template><div class="app-container"><el-form :model="queryParams" ref="queryForm" :inline="true" v-show="showSearch" label-width="68px"><el-form-item label="视频名称" prop="fileName"><el-inputv-model="queryParams.fileName"placeholder="请输入视频名称"clearablesize="small"@keyup.enter.native="handleQuery"/></el-form-item><el-form-item><el-button type="primary" icon="el-icon-search" size="mini" @click="handleQuery">搜索</el-button><el-button icon="el-icon-refresh" size="mini" @click="resetQuery">重置</el-button></el-form-item></el-form><el-row :gutter="10" class="mb8"><el-col :span="1.5"><el-buttontype="primary"plainicon="el-icon-plus"size="mini"@click="handleImport">上传</el-button></el-col><el-col :span="1.5"><el-buttontype="danger"plainicon="el-icon-delete"size="mini":disabled="multiple"@click="handleDelete"v-hasPermi="['videoInfo:info:remove']">删除</el-button></el-col><right-toolbar :showSearch.sync="showSearch" @queryTable="getList"></right-toolbar></el-row><div ref="tableDom"><el-table v-loading="loading" :data="infoList" @selection-change="handleSelectionChange"><el-table-column type="selection" width="55" align="center" /><el-table-column label="视频文件名称" align="center" prop="fileName" /><el-table-column label="视频文件存储路径" align="center" prop="filePath" /><el-table-column label="视频文件大小" align="center" prop="fileSize" /><el-table-column label="视频文件上传时间" align="center" prop="uploadDate" width="180"><template slot-scope="scope"><span>{{ parseTime(scope.row.uploadDate, '{y}-{m}-{d}') }}</span></template></el-table-column><el-table-column label="下载路径" align="center" prop="downloadPath" /><el-table-column label="视频预览" align="center"><template slot-scope="scope"><el-buttontype="primary"size="small"@click="handlePlay(scope.row)">预览</el-button></template></el-table-column><el-table-column label="操作" align="center" class-name="small-padding fixed-width"><template slot-scope="scope"><el-buttonsize="mini"type="text"icon="el-icon-delete"@click="handleDelete(scope.row)"v-hasPermi="['videoInfo:info:remove']">删除</el-button></template></el-table-column></el-table></div><paginationv-show="total>0":total="total":page.sync="queryParams.pageNum":limit.sync="queryParams.pageSize"@pagination="getList"/><el-dialog title="视频预览" :visible.sync="open"><video-playerclass="video-player vjs-custom-skin"ref="videoPlayer":playsinline="true":options="playerOptions"></video-player></el-dialog><uploadVideoDialog@handleSaveSuccess="getList"ref="uploadVideoDialogRef"></uploadVideoDialog></div>
</template><script>
import { listInfo, getInfo, delInfo, addInfo, updateInfo } from "@/api/videoInfo/info";
import uploadVideoDialog from "./uploadVideoDialog";
// import { videoPlayer } from 'vue-video-player'
// import 'video.js/dist/video-js.css'export default {components: { uploadVideoDialog},name: "Info",data() {return {// 遮罩层loading: true,// 选中数组ids: [],// 非单个禁用single: true,// 非多个禁用multiple: true,// 显示搜索条件showSearch: true,// 总条数total: 0,// 学习视频信息表格数据infoList: [],// 弹出层标题title: "",// 是否显示弹出层open: false,// 查询参数queryParams: {pageNum: 1,pageSize: 10,fileName: null,},// 视频预览窗口open: false,// 表单参数form: {},// 播放参数playerOptions: {},removeParams: {filePaths: [],ids: [],},// 选中视频idvideoId: "",tableHeight: "",// 表单参数form: {},// 表单校验rules: {}};},created() {this.getList();},mounted() {var _this = this;this.resizeTable();window.onresize = function () {_this.resizeTable();};},//记得清空  如果在keepalive中 则使用activated  deactivateddestroyed() {window.onresize = null;},methods: {/** 查询学习视频信息列表 */getList() {this.loading = true;listInfo(this.queryParams).then(response => {this.infoList = response.rows;this.total = response.total;this.loading = false;});},// 取消按钮cancel() {this.open = false;this.reset();},// 表单重置reset() {this.form = {id: null,videoType: null,fileName: null,filePath: null,fileSize: null,uploadDate: null,delFlag: null,fileMd5: null,deviceNumber: null,nickName: null,jobNumber: null,userId: null,downloadPath: null};this.resetForm("form");},/** 搜索按钮操作 */handleQuery() {this.queryParams.pageNum = 1;this.getList();},/** 重置按钮操作 */resetQuery() {this.resetForm("queryForm");this.handleQuery();},// 多选框选中数据handleSelectionChange(selection) {this.ids = selection.map(item => item.id)this.single = selection.length!==1this.multiple = !selection.length},/** 预览按钮 */handlePlay(row) {console.log("row");console.log(row);const name = row.fileName.substring(0, row.fileName.lastIndexOf("."));const downloadPath = row.downloadPath;console.log(downloadPath);this.open = true;console.log(process.env.VUE_APP_BASE_API + "/" + downloadPath);this.playerOptions = {playbackRates: [0.75, 1.0, 1.25, 1.5, 2.0], //播放速度autoplay: false, // 是否自动播放。muted: false, // 是否静音播放,默认情况下将会消除任何音频。loop: false, // 是否循环播放。preload: "auto", // 建议浏览器在<video>加载元素后是否应该开始下载视频数据。auto浏览器选择最佳行为,立即开始加载视频(如果浏览器支持)language: "zh-CN",aspectRatio: "16:9", // 将播放器置于流畅模式,并在计算播放器的动态大小时使用该值。值应该代表一个比例 - 用冒号分隔的两个数字(例如"16:9"或"4:3")fluid: true, // 是否流体从而按比例缩放以适应其容器。flash: { hls: { withCreadentials: false } }, //可以播放rtmp视频html5: { hls: { withCreadentials: false } }, //可以播放m3u8视频sources: [{type: "video/mp4", // 播放的类型,在这里项目中需要的是rtmp,也可以修改为:‘video/mp4’src:process.env.VUE_APP_BASE_API + "/" + downloadPath,//src:"http://127.0.0.1:8088/videoInfo/info/download/learning/测试视频/mp4/",//src :"https://qiniu-web-assets.dcloud.net.cn/unidoc/zh/2minute-demo.mp4"},],width: "100%",notSupportedMessage: "此视频暂无法播放...", // 当无法播放时允许覆盖Video.js,显示的默认信息。controlBar: {timeDivider: true,durationDisplay: true,remainingTimeDisplay: false,fullscreenToggle: true,},};},// 打开上传文件弹出框handleImport() {this.$refs.uploadVideoDialogRef.open();},//table自动高度returnTableHeight(ref, bottomOffset) {let height = null;height =window.innerHeight - ref.getBoundingClientRect().top - bottomOffset;return height;},resizeTable() {this.$nextTick(() => {this.tableHeight = this.returnTableHeight(this.$refs.tableDom, 100);}, 0);},/** 提交按钮 */submitForm() {this.$refs["form"].validate(valid => {if (valid) {if (this.form.id != null) {updateInfo(this.form).then(response => {this.$modal.msgSuccess("修改成功");this.open = false;this.getList();});} else {addInfo(this.form).then(response => {this.$modal.msgSuccess("新增成功");this.open = false;this.getList();});}}});},/** 删除按钮操作 */handleDelete(row) {const ids = row.id || this.ids;this.$modal.confirm('是否确认删除学习视频信息编号为"' + ids + '"的数据项?').then(function() {return delInfo(ids);}).then(() => {this.getList();this.$modal.msgSuccess("删除成功");}).catch(() => {});},}
};
</script>上传文件的弹窗
<template><div class="app-container"><el-dialog:title="setLightTitle":visible.sync="visible":close-on-click-modal="false"width="25%"><el-formref="appForm"style="margin-bottom: -30px":model="appData":rules="rules"size="small"label-width="100px"><div style="margin: 0 auto; text-align: center"><el-uploadaction="#"ref="upload":show-file-list="true":auto-upload="false":multiple="false":on-change="uploadFile":limit="1"dragaccept=".mp4"><i class="el-icon-upload"></i><div class="el-upload__text">将文件拖到此处,或<em>点击上传</em></div><div class="el-upload__tip text-center" slot="tip"><span>仅允许mp4文件。</span></div></el-upload></div></el-form><span slot="footer" class="dialog-footer"><el-button @click="visible = false" type="primary" size="small">取 消</el-button><el-buttontype="primary"size="small"@click="handleUpload":loading="loading">确 定</el-button></span></el-dialog></div>
</template><script>
import { addInfo } from "@/api/videoInfo/info";export default {name: "upload-file-dialog",data() {return {file: null,formData: null,updateVersion: null, //上传版本updateSupport: null, //是否全部更新appData: {handwareid: null,updateVersion: null,updateSupport: null,},loading: false,visible: false,setLightTitle: "上传视频",acceptFile: ".mp4",rules: {updateVersion: [{ required: true, message: "请xx", trigger: "blur" },],},};},methods: {dialogClose() {this.visible = false;},open() {this.file = null;this.formData = null;this.visible = true;this.updateVersion = null;this.updateSupport = false;this.$refs.upload?.clearFiles();},uploadFile(item) {this.file = item.raw; // 通过DOM取文件数据this.formData = new FormData(); //new一个formData事件this.formData.append("file", this.file); //将file属性添加到formData里},handleUpload() {if (this.file == null) {this.$message("请先选择文件!");return;}let vm = this;this.$refs.appForm.validate((valid) => {if (valid) {vm.loading = true;addInfo(vm.formData).then((response) => {vm.loading = false;this.$refs.upload.clearFiles();if (response.code === 200) {vm.visible = false; //关闭窗口vm.$emit("handleSaveSuccess"); //成功通知父窗口vm.$message("上传成功!");}}).catch(function (error) {vm.loading = false;setTimeout(() => {vm.$message("上传失败!" + error);}, 150);});}});},},
};
</script><style scoped>
/deep/ .el-upload {width: 70%;height: 100%;
}
/deep/ .el-upload .el-upload-dragger {width: 100%;height: 100%;
}
</style>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/796850.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PHP实现网站微信扫码关注公众号后自动注册登陆实现方法及代码【关注收藏】

在网站注册登陆这环节&#xff0c;增加微信扫码注册登陆&#xff0c;普通的方法需要开通微信开发者平台&#xff0c;生成二维码扫码后才能获取用户的uinonid或openid&#xff0c;实现注册登陆&#xff0c;但这样比较麻烦还要企业认证交费开发者平台&#xff0c;而且没有和公众号…

如何控制Docker容器退出后的自动重启行为?

在Docker中&#xff0c;可以通过以下两种方式来控制容器退出后的自动重启行为&#xff1a; 使用docker run命令时&#xff0c;通过设置--restart参数来指定容器退出后的重启策略。可以使用以下值之一&#xff1a; no: 默认值&#xff0c;容器退出后不会自动重启。always: 容器…

什么是EL表达式?怎么使用?

文章目录 一、什么是EL表达式1、命令格式&#xff1a;${作用域对象别名.共享数据} 二、EL表达式与作用域对象别名1、JSP文件可以使用的作用域对象2、EL表达式提供作用域对象别名3、EL表达式将引用对象属性写入到响应体4、EL表达式简化版 三、EL表达式与运算表达式四、EL表达式提…

【SQL】1890. 2020年最后一次登录(简单写法;窗口函数写法)

前述 sql 中 between 的边界问题 ---- between 边界&#xff1a;闭区间&#xff0c;not between 边界&#xff1a;开区间 在 sql 中&#xff0c; between 边界&#xff1a;闭区间not between 边界&#xff1a;开区间 题目描述 leetcode题目&#xff1a;1890. 2020年最后一…

【leetcode面试经典150题】16.接雨水(C++)

【leetcode面试经典150题】专栏系列将为准备暑期实习生以及秋招的同学们提高在面试时的经典面试算法题的思路和想法。本专栏将以一题多解和精简算法思路为主&#xff0c;题解使用C语言。&#xff08;若有使用其他语言的同学也可了解题解思路&#xff0c;本质上语法内容一致&…

aardio教程五) 写Python风格的aardio代码(字符串篇)

前言 熟悉一个新的语言最麻烦的就是需要了解一些库的使用&#xff0c;特别是基础库的使用。 所以我想给aardio封装一个Python风格的库&#xff0c;Python里的基础库是什么方法名&#xff0c;aardio里也封装同样的方法名。 这样就不需要单独去了解aardio里一些方法的使用细节…

Lanelets_ 高效的自动驾驶地图表达方式

Lanelets: 高效的自动驾驶地图表达方式 附赠自动驾驶学习资料和量产经验&#xff1a;链接 LaneLets是自动驾驶领域高精度地图的一种高效表达方式&#xff0c;它以彼此相互连接的LaneLets来描述自动驾驶可行驶区域&#xff0c;不仅可以表达车道几何&#xff0c;也可以完整表述车…

.NET9 PreView2+.AOT ILC 的重大变化

RyuJIT 增强功能 1. 环路优化 (循环优化) 这种优化实际上是一种 for 循环叠加态的优化&#xff0c;for 循环叠加计算的过程中&#xff0c;会对其中部分变量进行感应。比如循环中放置 0 扩展 (第一个索引为 0)&#xff0c;这种优化灵感来源于 LLVM 标量演化。下面看例子&#…

每天一个数据分析题(二百五十四)

在大数据时代背景下&#xff0c;我们使用的数据主要包含两种类别&#xff0c;一种称为结构化数据&#xff0c;另一种称为非结构化数据。请问以下哪个选项属于非结构化数据&#xff1f; A. 利润表 B. 短视频 C. 产品库存表 D. 产品进货表 题目来源于CDA模拟题库 点击此处获…

LeetCode 每日一题 2024/4/1-2024/4/7

记录了初步解题思路 以及本地实现代码&#xff1b;并不一定为最优 也希望大家能一起探讨 一起进步 目录 4/1 2810. 故障键盘4/2 894. 所有可能的真二叉树4/3 1379. 找出克隆二叉树中的相同节点4/4 2192. 有向无环图中一个节点的所有祖先4/5 1026. 节点与其祖先之间的最大差值4/…

8种专坑运维的 SQL 写法,性能降低100倍,您不来看看?

1、LIMIT 语句 分页查询是最常用的场景之一&#xff0c;但也通常也是最容易出问题的地方。比如对于下面简单的语句&#xff0c;一般 DBA 想到的办法是在 type&#xff0c;name&#xff0c; create_time 字段上加组合索引。这样条件排序都能有效的利用到索引&#xff0c;性能迅…

AIGC实战——ProGAN(Progressive Growing Generative Adversarial Network)

AIGC实战——ProGAN 0. 前言1. ProGAN2. 渐进式训练3. 其他技术3.1 小批标准差3.2 均等学习率3.3 逐像素归一化 4. 图像生成小结系列链接 0. 前言 我们已经学习了使用生成对抗网络 (Generative Adversarial Network, GAN) 解决各种图像生成任务。GAN 的模型架构和训练过程具有…

真实的招生办对话邮件及美国高校官网更新的反 AI 政策

这两年 ChatGPT 的热度水涨船高&#xff0c;其编写功能强大&#xff0c;且具备强大的信息整合效果&#xff0c;所以呈现的内容在一定程度上具备可读性。 那么&#xff0c;美国留学文书可以用 ChatGPT 写吗&#xff1f;使用是否有风险&#xff1f;外网博主 Kushi Uppu 在这个申…

C++20 semaphore(信号量) 详解

头文件在C20中是并发库技术规范&#xff08;Technical Specification, TS&#xff09;的一部分。信号量是同步原语&#xff0c;帮助控制多线程程序中对共享资源的访问。头文件提供了标准C方式来使用信号量。 使用环境 Windows&#xff1a;VS中打开项目属性&#xff0c;修改C语…

基于卷积神经网络的天气识别系统(pytorch框架)【python源码+UI界面+前端界面+功能源码详解】

功能演示&#xff1a; 天气识别系统&#xff0c;vgg16&#xff0c;mobilenet卷积神经网络&#xff08;pytorch框架&#xff09;_哔哩哔哩_bilibili &#xff08;一&#xff09;简介 基于卷积神经网络的天气识别系统是在pytorch框架下实现的&#xff0c;系统中有两个模型可选…

vue+elementUI实现表格组件的封装

效果图&#xff1a; 在父组件使用表格组件 <table-listref"table":stripe"true":loading"loading":set-table-h"slotProps.setMainCardBodyH":table-data"tableData":columns"columns.tableList || []":ra…

快速创建Python库文档:pdoc的简便之道

快速创建Python库文档:pdoc的简便之道 什么是pdoc&#xff1f; pdoc 是一个用于自动生成 Python 模块文档的库,它可以根据代码中的 docstrings 自动生成漂亮的文档网页. 为什么选择使用pdoc库&#xff1f; 简单易用: pdoc不需要复杂的配置,只需运行一个命令即可生成文档。 …

项目管理工具对比:甘特图与看板

许多项目管理工具都能帮助你规划、管理和跟踪项目&#xff0c;比如甘特图和看板。如果比较一下甘特图和看板&#xff0c;会发现两者在不同方面都对项目有价值。 让我们来看看看板与甘特图的区别&#xff0c;了解它们是如何工作的&#xff0c;以及各自的优缺点&#xff0c;看看…

基于Springboot的Java学习平台

采用技术 基于Springbootjava学习平台的设计与实现~ 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;SpringBootMyBatis 工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 页面展示效果 系统功能模块 后台管理 用户注册 课程信息 作业信息 资料信息…

电工技术学习笔记——正弦交流电路

一、正弦交流电路 1. 正弦量的向量表示法 向量表示方法&#xff1a;正弦交流电路中&#xff0c;相量表示法是一种常用的方法&#xff0c;用于描述电压、电流及其相位关系。相量表示法将正弦交流信号表示为复数&#xff0c;通过复数的运算来描述电路中各种参数的相互关系 …