modifyVideoPage.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502
  1. <template>
  2. <!-- 新增编辑视频 -->
  3. <div class="modify-video-page-wrap">
  4. <t-form :data="form" :rules="rules" ref="formRules">
  5. <t-form-item label="所属分类" name="ClassifyId">
  6. <t-cascader placeholder="选择所属分类"
  7. v-model="form.ClassifyId"
  8. :options="classifyList"
  9. style="width: 300px;"
  10. clearable
  11. :keys="{
  12. emitPath:false,
  13. label:'ClassifyName',
  14. value:'ClassifyId',
  15. children:'Children'}">
  16. </t-cascader>
  17. </t-form-item>
  18. <t-form-item label="视频名称" name="Title">
  19. <t-input style="width: 300px;" v-model="form.Title" placeholder="请输入视频名称"></t-input>
  20. </t-form-item>
  21. <t-form-item label="视频简介" name="Introduce">
  22. <t-textarea v-model="form.Introduce"
  23. placeholder="请输入视频简介"
  24. maxlength="50" show-word-limit :autosize="{ minRows: 5, maxRows: 5 }"></t-textarea >
  25. </t-form-item>
  26. <t-form-item label="上传封面" name="CoverImg">
  27. <div style="display: block;">
  28. <t-upload accept="image/*"
  29. :requestMethod="handleUploadImg" theme="custom" :show-image-file-name="false" :disabled="isImageUploading">
  30. <t-button theme="primary" :loading="isImageUploading">点击上传</t-button>
  31. <span style="color:#999999;margin-left: 5px;" @click.stop>建议尺寸比例3:2,支持png、jpg、gif、jpeg格式</span>
  32. </t-upload>
  33. <div class="img-box">
  34. <img :src="form.CoverImg" v-if="form.CoverImg">
  35. <span v-else style="color:#999999;line-height: 100px;">请上传封面图</span>
  36. </div>
  37. </div>
  38. </t-form-item>
  39. <t-form-item label="上传视频" name="VideoUrl">
  40. <div style="display: block;">
  41. <t-upload accept=".mp4"
  42. :request-method="handleUploadVideo" theme="custom" :show-file-list="false" :disabled="isVideoUploading">
  43. <t-button theme="primary" :loading="isVideoUploading">点击上传</t-button>
  44. <span style="color:#999999;margin-left: 5px;" @click.stop>仅支持mp4格式</span>
  45. </t-upload>
  46. <div class="img-box">
  47. <t-progress type="circle" :percentage="percentage" width="40" v-if="isVideoUploading"></t-progress>
  48. <span v-if="form.VideoUrl&&!form.VideoId" class="duration">{{timeDuration}}</span>
  49. <img :src="form.CoverImg" v-if="form.VideoUrl" @click="handlePreviewVideo">
  50. <span v-else style="color:#999999;line-height: 100px;">请上传视频</span>
  51. </div>
  52. </div>
  53. </t-form-item>
  54. <t-form-item label="视频标签" name="TagIds">
  55. <t-button variant="text" theme="primary" @click="isModifyDialogShow = true">选择标签</t-button>
  56. <div class="tag-list" :key="tagIdKey">
  57. <span class="tag-item" v-for="tag in form.TagIds" :key="tag.TagId">
  58. <span>{{tag.TagName}}</span>
  59. <span @click.stop="removeTag(tag)"><i class="t-icon-close"></i></span>
  60. </span>
  61. </div>
  62. </t-form-item>
  63. </t-form>
  64. <div class="btn-box">
  65. <t-button variant="outline" plain @click="changeRoute">取消</t-button>
  66. <t-button theme="primary" @click="modifyVideo" :loading="isImageUploading||isVideoUploading">保存</t-button>
  67. <t-button theme="primary" @click="publishVideo" :loading="isImageUploading||isVideoUploading">发布</t-button>
  68. </div>
  69. <!-- 选择标签弹窗 -->
  70. <AddTags
  71. :isModifyDialogShow="isModifyDialogShow"
  72. :Tags="form.TagIds"
  73. :tagList="tagList"
  74. :getTagList="getTagList"
  75. @modify="modifyTags"
  76. @close="isModifyDialogShow=false"
  77. />
  78. <!-- 预览视频弹窗 -->
  79. <t-dialog
  80. :visible.sync="previewPop"
  81. width="60vw"
  82. :header="previewPopTitle"
  83. :footer="false"
  84. @close="endingPreview"
  85. >
  86. <video style="width: 100%;height: 100%;max-height: 70vh;outline: none;"
  87. controls :src="previewVideoUrl" autoplay ref="previewVideo">
  88. 您的浏览器暂不支持,请更换浏览器
  89. </video>
  90. </t-dialog>
  91. </div>
  92. </template>
  93. <script setup>
  94. import { ref, onMounted } from 'vue';
  95. import MD5 from 'js-md5';
  96. import { VideoInterface, TagInterface, ClassifyInterface } from '@/api/modules/trainingApi';
  97. import AddTags from './components/addTags.vue';
  98. import _ from 'lodash';
  99. import { useRoute, useRouter } from 'vue-router'
  100. const router=useRouter()
  101. const route=useRoute()
  102. const form = ref({
  103. Title: '',
  104. Introduce: '',
  105. ClassifyId: '',
  106. TagIds: [],
  107. CoverImg: '',
  108. VideoUrl: ''
  109. });
  110. const rules = {
  111. ClassifyId: [{ required: true, message: '请选择所属分类' }],
  112. Title: [{ required: true, message: '请输入视频名称' }],
  113. CoverImg: [{ required: true, message: '请选择视频封面' }],
  114. VideoUrl: [{ required: true, message: '请上传视频' }],
  115. TagIds: [{
  116. required: true,
  117. validator: (rule, value, callback) => {
  118. if (!value.length) {
  119. return callback(new Error("请至少选择一个标签"));
  120. } else {
  121. return callback();
  122. }
  123. }
  124. }]
  125. }
  126. const tagIdKey = ref(0);
  127. const isModifyDialogShow = ref(false);
  128. const isImageUploading = ref(false);
  129. const isVideoUploading = ref(false);
  130. const percentage = ref(0);
  131. const timeDuration = ref('');
  132. const previewVideoUrl = ref('');
  133. const previewPop = ref(false);
  134. const previewPopTitle = ref('');
  135. const tagList = ref([]);
  136. const classifyList = ref([]);
  137. const previewVideo = ref(null);
  138. const formRules = ref(null);
  139. // 定义全局变量
  140. let ALOSSINS = null;
  141. let ALOSSAbortCheckpoint = null;
  142. // 获取分类列表
  143. const getClassifyList = async (type = '') => {
  144. try {
  145. const res = await ClassifyInterface.getClassifyList({});
  146. if (res.Ret !== 200) return;
  147. classifyList.value = (res.Data && res.Data.List) || [];
  148. filterNodes(classifyList.value);
  149. classifyList.value = classifyList.value.map(item => {
  150. if (!item.Children) {
  151. item.disabled = true;
  152. }
  153. return item;
  154. });
  155. } catch (error) {
  156. console.error('获取分类列表失败:', error);
  157. }
  158. };
  159. // 过滤节点
  160. const filterNodes = (arr) => {
  161. if (arr.length) {
  162. arr.forEach(item => {
  163. if (item.Children && item.Children.length) {
  164. filterNodes(item.Children);
  165. }
  166. if (item.Children && !item.Children.length) {
  167. delete item.Children;
  168. }
  169. });
  170. }
  171. };
  172. // 获取标签列表
  173. const getTagList = async (keyword = '') => {
  174. try {
  175. const res = await TagInterface.getTagList({
  176. Keyword: keyword,
  177. PageSize: 1000,
  178. CurrentIndex: 1
  179. });
  180. if (res.Ret !== 200) return;
  181. tagList.value = (res.Data && res.Data.List) || [];
  182. } catch (error) {
  183. console.error('获取标签列表失败:', error);
  184. }
  185. };
  186. // 检查图片是否合法
  187. const handleUploadImg = async (file) => {
  188. isImageUploading.value = true;
  189. const { type } = file.raw;
  190. if (!['image/png', 'image/jpeg'].includes(type)) {
  191. MessagePlugin.warning('仅支持png、jpg格式的图片');
  192. isImageUploading.value = false;
  193. return;
  194. }
  195. const res = await uploadImg(file);
  196. return {
  197. status: res ? 'success' : 'fail',
  198. response: {
  199. url: form.value.CoverImg,
  200. },
  201. error: '上传失败',
  202. };
  203. };
  204. // 上传图片
  205. const uploadImg = async (file) => {
  206. const formData = new FormData();
  207. formData.append('file', file.raw);
  208. const res = await VideoInterface.bannerupload(formData)
  209. isImageUploading.value = false;
  210. if (res.Ret !== 200) return false;
  211. form.value.CoverImg = res.Data.ResourceUrl;
  212. return true;
  213. };
  214. // 检查视频是否合法,并获取视频时长
  215. const handleUploadVideo = async (file) => {
  216. if (file.type !== 'video/mp4') {
  217. MessagePlugin.warning('上传失败,上传视频格式不正确');
  218. return;
  219. }
  220. const duration = await handleGetDuration(file);
  221. timeDuration.value = `${String(parseInt(duration / 60)).padStart(2, '0')}:${String(parseInt(duration % 60)).padStart(2, '0')}`;
  222. uploadVideo(file);
  223. isVideoUploading.value = true;
  224. return {
  225. status: 'success',
  226. response: {},
  227. };
  228. };
  229. // 获取视频时长的Promise
  230. const handleGetDuration = (file) => {
  231. return new Promise((resolve, reject) => {
  232. const fileUrl = URL.createObjectURL(file.raw);
  233. const audioEl = new Audio(fileUrl);
  234. audioEl.addEventListener('loadedmetadata', (e) => {
  235. const t = e.composedPath()[0].duration;
  236. resolve(t);
  237. });
  238. });
  239. };
  240. // 上传视频
  241. const uploadVideo = async (file) => {
  242. const res = await VideoInterface.getOSSSign();
  243. if (res.Ret === 200) {
  244. handleUploadToOSS(file.raw, res.Data);
  245. }
  246. };
  247. // 上传到阿里云
  248. const handleUploadToOSS = async (file, { AccessKeyId, AccessKeySecret, SecurityToken }) => {
  249. ALOSSINS = new OSS({
  250. region: 'oss-cn-shanghai',
  251. accessKeyId: AccessKeyId,
  252. accessKeySecret: AccessKeySecret,
  253. stsToken: SecurityToken,
  254. bucket: 'hzchart',
  255. endpoint: 'hzstatic.hzinsights.com',
  256. cname: true,
  257. timeout: 600000000000
  258. });
  259. const t = new Date().getTime().toString();
  260. const temName = `static/yb/video/${MD5(t)}.${file.type.split('/')[1]}`;
  261. const options = {
  262. progress: (p, cpt, res) => {
  263. ALOSSAbortCheckpoint = cpt;
  264. percentage.value = parseInt(p * 100);
  265. },
  266. parallel: 10,
  267. partSize: 1024 * 1024 * 10
  268. };
  269. try {
  270. console.log(temName, file, { ...options });
  271. const res = await ALOSSINS.multipartUpload(temName, file, { ...options });
  272. console.log('上传结果', res);
  273. if (res.res.status === 200) {
  274. form.value.VideoUrl = `https://hzstatic.hzinsights.com/${res.name}`;
  275. percentage.value = 0;
  276. ALOSSAbortCheckpoint = null;
  277. isVideoUploading.value = false;
  278. }
  279. } catch (error) {
  280. if (error.name !== 'cancel') {
  281. MessagePlugin.warning('上传失败,请刷新重试');
  282. }
  283. percentage.value = 0;
  284. ALOSSAbortCheckpoint = null;
  285. isVideoUploading.value = false;
  286. }
  287. };
  288. // 删除所选标签
  289. const removeTag = (tag) => {
  290. const index = form.value.TagIds.findIndex(i => i.TagId === tag.TagId);
  291. if (index !== -1) {
  292. form.value.TagIds.splice(index, 1);
  293. }
  294. tagIdKey.value++;
  295. };
  296. // 改变所选标签
  297. const modifyTags = (tags) => {
  298. form.value.TagIds = _.cloneDeep(tags);
  299. isModifyDialogShow.value = false;
  300. };
  301. // 获取视频信息
  302. const getVideoDetail = () => {
  303. const { VideoId } = route.query;
  304. if (!Number(VideoId)) return;
  305. VideoInterface.getVideoDetail({ VideoId: Number(VideoId) }).then(res => {
  306. if (res.Ret !== 200) return;
  307. form.value = res.Data || {};
  308. if (form.value.Classify) {
  309. const classifyArr = getDataClassify(form.value.Classify);
  310. form.value.ClassifyId = classifyArr[classifyArr.length - 1];
  311. delete form.value.Classify;
  312. }
  313. if (form.value.Tags) {
  314. form.value.TagIds = form.value.Tags;
  315. delete form.value.Tags;
  316. }
  317. });
  318. };
  319. // 获取视频分类路径
  320. const getDataClassify = (classify, classifyArr = []) => {
  321. classifyArr.push(classify.ClassifyId);
  322. if (classify.Children && classify.Children.length) {
  323. return getDataClassify(classify.Children[0], classifyArr);
  324. }
  325. return classifyArr;
  326. };
  327. // 添加/编辑视频
  328. const modifyVideo = async (type = 'modify') => {
  329. const valid = await formRules.value.validate()
  330. if (valid !== true) return
  331. let res = null;
  332. const params = { ...form.value, TagIds: form.value.TagIds.map(t => t.TagId) };
  333. if (!form.value.VideoId) {
  334. res = await VideoInterface.addVideo(params);
  335. } else {
  336. res = await VideoInterface.editVideo(params);
  337. }
  338. if (res.Ret !== 200) return;
  339. if (type !== 'publish') {
  340. MessagePlugin.success(`${form.value.VideoId ? '编辑' : '添加'}成功`);
  341. changeRoute();
  342. }
  343. if (!form.value.VideoId) {
  344. form.value.VideoId = res.Data || '';
  345. }
  346. };
  347. // 发布视频
  348. const publishVideo = async () => {
  349. let res = {};
  350. await modifyVideo('publish');
  351. if (form.value.VideoId) {
  352. res = await VideoInterface.publishVideo({ VideoId: Number(form.value.VideoId), PublishState: 1 });
  353. if (res.Ret !== 200) return;
  354. MessagePlugin.success('发布成功');
  355. changeRoute();
  356. }
  357. };
  358. // 改变路由
  359. const changeRoute = () => {
  360. if (ALOSSAbortCheckpoint) {
  361. console.log('终止上传');
  362. ALOSSINS.abortMultipartUpload(ALOSSAbortCheckpoint.name, ALOSSAbortCheckpoint.uploadId);
  363. }
  364. router.push('/training/trainingVideo');
  365. };
  366. // 预览视频
  367. const handlePreviewVideo = () => {
  368. if (isVideoUploading.value || !form.value.VideoUrl) return;
  369. previewPopTitle.value = form.value.Title || '暂无标题';
  370. previewVideoUrl.value = form.value.VideoUrl;
  371. previewPop.value = true;
  372. previewVideo.value.play();
  373. };
  374. // 结束预览弹窗关闭回调 -- 暂停视频
  375. const endingPreview = () => {
  376. if (previewVideo.value) {
  377. previewPop.value = false;
  378. previewVideo.value.pause();
  379. }
  380. };
  381. // 生命周期钩子 - 组件挂载时执行
  382. onMounted(() => {
  383. getClassifyList('leaf');
  384. getTagList();
  385. getVideoDetail();
  386. });
  387. </script>
  388. <style lang="scss">
  389. .modify-video-page-wrap{
  390. .t-textarea__inner{
  391. resize: none;
  392. }
  393. .t-form-item{
  394. .t-form-item__content{
  395. display: flex;
  396. flex-direction: column;
  397. align-items:flex-start;
  398. }
  399. }
  400. }
  401. </style>
  402. <style scoped lang="scss">
  403. .modify-video-page-wrap{
  404. box-sizing: border-box;
  405. padding:40px;
  406. background-color: #fff;
  407. border-radius: 4px;
  408. .t-form{
  409. .t-input,.t-select,.t-textarea{
  410. width:500px;
  411. }
  412. .img-box{
  413. /* background-color: #D9D9D9; */
  414. border:1px dashed #d9d9d9;
  415. width:150px;
  416. height:100px;
  417. text-align: center;
  418. margin-top: 10px;
  419. position: relative;
  420. line-height: normal;
  421. img{
  422. width:100%;
  423. height:100%;
  424. }
  425. .duration{
  426. position:absolute;
  427. right:0;
  428. bottom:0;
  429. background-color: black;
  430. color:#fff;
  431. padding:4px;
  432. }
  433. .t-progress{
  434. position: absolute;
  435. left:50%;
  436. top:50%;
  437. transform: translate(-50%,-50%);
  438. background-color: white;
  439. border-radius: 50%;
  440. }
  441. }
  442. .tag-list{
  443. display: flex;
  444. gap:10px;
  445. line-height:0;
  446. .tag-item{
  447. cursor: pointer;
  448. text-align: center;
  449. box-sizing: border-box;
  450. padding:12px;
  451. min-width:78px;
  452. color: #409EFF;
  453. background-color: #EAF3FE;
  454. border:1px solid #409EFF;
  455. border-radius: 2px;
  456. }
  457. }
  458. }
  459. .btn-box{
  460. margin-top: 20px;
  461. text-align: center;
  462. .t-button{
  463. margin-right: 50px;
  464. width:120px;
  465. text-align: center;
  466. }
  467. }
  468. }
  469. </style>