Ver Fonte

添加视频播放

lex há 1 ano atrás
pai
commit
18d47563f6

+ 90 - 3
src/components/card-preview/video-modal/index.module.less

@@ -58,6 +58,7 @@
 
     .actionWrap {
       display: flex;
+      align-items: center;
     }
 
     .actionBtn {
@@ -75,9 +76,36 @@
     }
 
 
+    .actionBtnSpeed {
+      width: 40px;
+      height: 40px;
+      background-color: transparent;
+      cursor: pointer;
+
+      &>img {
+        width: 100%;
+        height: 100%;
+      }
+    }
+
+
     .iconReplay {
-      width: 31px;
-      height: 29px;
+      width: 40px;
+      height: 40px;
+      background-color: transparent;
+      cursor: pointer;
+      margin: 0 22px;
+
+      &>img {
+        width: 100%;
+        height: 100%;
+      }
+    }
+
+    .iconDownload {
+      width: 40px;
+      height: 40px;
+      margin-left: 5px;
       background-color: transparent;
       cursor: pointer;
 
@@ -90,7 +118,7 @@
 
   .slider {
     width: 100%;
-    padding: 0 20px 0 12px;
+    padding: 0 0 0 12px;
 
     :global {
 
@@ -110,6 +138,65 @@
 
 }
 
+.sliderPopup {
+  position: absolute;
+  z-index: 9999;
+  left: 144px;
+  bottom: 82px;
+  display: flex;
+  align-items: center;
+  flex-direction: column;
+  height: 252px;
+  width: 59Px;
+  padding: 12Px 0 15Px;
+  background: url('') no-repeat top center;
+  background-size: contain;
+
+  .iconAdd,
+  .iconCut {
+    display: inline-block;
+    width: 24Px;
+    height: 24Px;
+    background: url('') no-repeat center;
+    background-size: contain;
+    flex-shrink: 0;
+    cursor: pointer;
+
+    &.disabled {
+      opacity: 0.7;
+    }
+  }
+
+  .iconCut {
+    background: url('') no-repeat center;
+    background-size: contain;
+  }
+
+  .sliderPoint {
+    background: #FFFFFF;
+    box-shadow: 0px 2px 4px 0px rgba(102, 102, 102, 0.77);
+    border-radius: 14px;
+    font-size: 14Px;
+    font-weight: 500;
+    height: 22Px;
+    color: #198CFE;
+    min-width: 40Px;
+    text-align: center;
+    vertical-align: text-bottom;
+
+    span {
+      font-size: 12Px;
+    }
+  }
+
+  :global {
+    .n-slider {
+      margin: 7px 0;
+      padding: 0;
+    }
+  }
+}
+
 // .videoWrap {
 //   width: 100%;
 //   height: 100%;

+ 103 - 24
src/components/card-preview/video-modal/index.tsx

@@ -9,6 +9,7 @@ import iconplay from '@views/attend-class/image/icon-pause.png';
 import iconpause from '@views/attend-class/image/icon-play.png';
 import iconReplay from '@views/attend-class/image/icon-replay.png';
 import iconPreviewDownload from '@views/attend-class/image/icon-preivew-download.png';
+import iconSpeed from '@views/attend-class/image/icon-speed.png';
 import { NSlider, useMessage } from 'naive-ui';
 import { saveAs } from 'file-saver';
 
@@ -46,7 +47,12 @@ export default defineComponent({
       currentTime: '00:00',
       durationNum: 0,
       duration: '00:00',
-      showBar: true
+      showBar: true,
+      speedControl: false,
+      speedStyle: {
+        left: '1px'
+      },
+      defaultSpeed: 1 // 默认速度
     });
     const videoRef = ref();
     const videoItem = ref();
@@ -66,15 +72,18 @@ export default defineComponent({
     //
     const toggleHideControl = (isShow: false) => {
       videoFroms.showBar = isShow;
+      videoFroms.speedControl = false;
     };
 
     const onReplay = () => {
+      videoFroms.speedControl = false;
       if (!videoItem.value) return;
       videoItem.value.currentTime(0);
     };
 
     // 切换音频播放
     const onToggleVideo = (e?: MouseEvent) => {
+      videoFroms.speedControl = false;
       e?.stopPropagation();
       if (videoFroms.paused) {
         videoItem.value.play();
@@ -104,11 +113,7 @@ export default defineComponent({
         });
     };
 
-    onMounted(() => {
-      videoItem.value = TCPlayer(videoID, {
-        appID: '',
-        controls: false
-      }); // player-container-id 为播放器容器 ID,必须与 html 中一致
+    const __init = () => {
       if (videoItem.value) {
         videoItem.value.poster(poster.value); // 封面
         videoItem.value.src(isEmtry.value ? '' : src.value); // url 播放地址
@@ -140,6 +145,14 @@ export default defineComponent({
           emit('ended');
         });
       }
+    };
+
+    onMounted(() => {
+      videoItem.value = TCPlayer(videoID, {
+        appID: '',
+        controls: false
+      }); // player-container-id 为播放器容器 ID,必须与 html 中一致
+      __init();
     });
     expose({
       // changePlayBtn,
@@ -175,18 +188,17 @@ export default defineComponent({
                   <img class={styles.playIcon} src={iconpause} />
                 )}
               </button>
-            </div>
-            <div class={styles.time}>
-              <div
-                class="plyr__time plyr__time--current"
-                aria-label="Current time">
-                {videoFroms.currentTime}
-              </div>
-              <span class={styles.line}>/</span>
+
+              <button class={styles.iconReplay} onClick={onReplay}>
+                <img src={iconReplay} />
+              </button>
+
               <div
-                class="plyr__time plyr__time--duration"
-                aria-label="Duration">
-                {videoFroms.duration}
+                class={styles.actionBtnSpeed}
+                onClick={() => {
+                  videoFroms.speedControl = !videoFroms.speedControl;
+                }}>
+                <img src={iconSpeed} />
               </div>
             </div>
           </div>
@@ -198,6 +210,7 @@ export default defineComponent({
               max={videoFroms.durationNum}
               tooltip={false}
               onUpdate:value={(val: number) => {
+                videoFroms.speedControl = false;
                 videoItem.value.currentTime(val);
                 videoFroms.currentTimeNum = val;
                 videoFroms.currentTime = timeFormat(Math.round(val || 0));
@@ -206,21 +219,87 @@ export default defineComponent({
           </div>
 
           <div class={styles.actions}>
+            <div class={styles.time}>
+              <div
+                class="plyr__time plyr__time--current"
+                aria-label="Current time">
+                {videoFroms.currentTime}
+              </div>
+              <span class={styles.line}>/</span>
+              <div
+                class="plyr__time plyr__time--duration"
+                aria-label="Duration">
+                {videoFroms.duration}
+              </div>
+            </div>
             <div class={styles.actionWrap}>
-              <button class={styles.iconReplay} onClick={onReplay}>
-                <img src={iconReplay} />
-              </button>
               {props.isDownload && (
-                <button
-                  class={styles.iconReplay}
-                  onClick={onDownload}
-                  style={{ marginLeft: '15px' }}>
+                <button class={styles.iconDownload} onClick={onDownload}>
                   <img src={iconPreviewDownload} />
                 </button>
               )}
             </div>
           </div>
         </div>
+
+        <div
+          style={{
+            display: videoFroms.speedControl ? 'block' : 'none'
+          }}>
+          <div
+            class={styles.sliderPopup}
+            onClick={(e: Event) => {
+              e.stopPropagation();
+            }}>
+            <i
+              class={styles.iconAdd}
+              onClick={() => {
+                if (videoFroms.defaultSpeed >= 1.5) {
+                  return;
+                }
+
+                if (videoItem.value) {
+                  videoFroms.defaultSpeed =
+                    (videoFroms.defaultSpeed * 10 + 1) / 10;
+                  videoItem.value.playbackRate(videoFroms.defaultSpeed);
+                }
+              }}></i>
+            <NSlider
+              value={videoFroms.defaultSpeed}
+              step={0.1}
+              max={1.5}
+              min={0.6}
+              vertical
+              tooltip={false}
+              onUpdate:value={(val: number) => {
+                videoFroms.defaultSpeed = val;
+                if (videoItem.value) {
+                  videoItem.value.playbackRate(videoFroms.defaultSpeed);
+                }
+              }}>
+              {{
+                thumb: () => (
+                  <div class={styles.sliderPoint}>
+                    {videoFroms.defaultSpeed}
+                    <span>x</span>
+                  </div>
+                )
+              }}
+            </NSlider>
+            <i
+              class={[styles.iconCut]}
+              onClick={() => {
+                if (videoFroms.defaultSpeed <= 0.6) {
+                  return;
+                }
+                if (videoItem.value) {
+                  videoFroms.defaultSpeed =
+                    (videoFroms.defaultSpeed * 10 - 1) / 10;
+                  videoItem.value.playbackRate(videoFroms.defaultSpeed);
+                }
+              }}></i>
+          </div>
+        </div>
       </div>
     );
   }

+ 1 - 0
src/views/attend-class/component/video-play.tsx

@@ -93,6 +93,7 @@ export default defineComponent({
     //
     const toggleHideControl = (isShow: false) => {
       videoFroms.showBar = isShow;
+      videoFroms.speedControl = false;
     };
 
     const onReplay = () => {

+ 5 - 0
src/views/attend-class/component/video.module.less

@@ -98,6 +98,11 @@
       height: 40px;
       background-color: transparent;
       cursor: pointer;
+
+      &>img {
+        width: 100%;
+        height: 100%;
+      }
     }