Jelajahi Sumber

Merge branch 'master' into ybht5.1

jwyu 2 tahun lalu
induk
melakukan
16a48d6c17

+ 1 - 1
README.md

@@ -2,4 +2,4 @@
  
 1. 该项目为弘则研报英文版,包含移动端适配
 2. 测试服前端地址:http://8.136.199.33:8301/report/index
-
+3. 正式服前端地址:https://ybpcen.hzinsights.com/report/index

+ 1 - 0
package.json

@@ -14,6 +14,7 @@
     "axios": "^1.2.0",
     "element-plus": "^2.2.25",
     "normalize.css": "^8.0.1",
+    "vconsole": "^3.15.0",
     "vue": "^3.2.41",
     "vue-router": "^4.1.6"
   },

+ 3 - 2
src/api/report.js

@@ -15,9 +15,10 @@ export const apiReportDetail=params=>{
 
 /**
  * 研报分类
+ * @param classify_type 0是英文研报,1是线上路演
  */
-export const apiReportClassifyList=()=>{
-    return get('/english_report/classify',{})
+export const apiReportClassifyList=(params)=>{
+    return get('/english_report/classify',params)
 }
 
 /**

+ 34 - 0
src/api/roadShow.js

@@ -0,0 +1,34 @@
+/**
+ * 路演视频模块
+ */
+ import {get,post} from './http'
+
+/**
+ * 路演视频列表
+ * @param page_size  
+ * @param current
+ * @param classify_id_first
+ * @param classify_id_second
+ */
+export const apiRoadShowVideoList=params=>{
+    return get('/english_report/video/list',params)
+}
+
+/**
+ * 路演视频详情
+ * @param video_code  
+ * @param share_email
+ */
+export const apiRoadShowVideoDetail=params=>{
+    return get('/english_report/video/detail',params)
+}
+
+/**
+ * 视频播放日志
+ * @param id 日志ID 为0时 新增,大于0时更新
+ * @param video_id 路演视频ID, 新增时必传
+ * @param stop_seconds 停止时的秒数
+ */
+export const apiVideoPlayLog=params=>{
+    return post('/english_report/video/play_log',params)
+}

TEMPAT SAMPAH
src/assets/icon_play.png


+ 482 - 0
src/components/VideoPlayBox.vue

@@ -0,0 +1,482 @@
+<script setup>
+import { nextTick, reactive,ref,onMounted, computed } from "vue-demi"
+import {debounce} from 'lodash'
+/**
+ * 格式化时间
+ * @param {number} e 秒
+ * @returns string eg:03:27
+ */
+function formatTime(e){
+    const h=parseInt(e/3600)
+    const m=parseInt(e/60%60)
+    const s=parseInt(e%60)
+    return `${h>0?h>9?h+':':'0'+h+':':''}${m>9?m:'0'+m}:${s>9?s:'0'+s}`
+}
+
+/**
+ * 是否全屏
+ * @returns boolean
+ */
+function isFullScreen(){
+    return document.webkitIsFullScreen || document.fullscreen
+}
+/**
+ * 设置全屏
+ */
+function setFullScreen(el){
+    if (el.requestFullscreen) {
+        el.requestFullscreen()
+        return true
+    } else if (el.msRequestFullscreen) {
+        el.msRequestFullscreen()
+        return true
+    } else if (el.mozRequestFullScreen) {
+        el.mozRequestFullScreen()
+        return true
+    } else if (el.webkitRequestFullScreen) {
+        el.webkitRequestFullScreen()
+        return true
+    }
+    return false
+}
+/**
+ * 关闭全屏
+ */
+function closeFullScreen(){
+    document.exitFullscreen() || document.webkitExitFullScreen() || document.msExitFullscreen() || document.mozCancelFullScreen()
+}
+//监听页面全屏变化
+function listenFullScreen(){
+    document.addEventListener('fullscreenchange',()=>{
+        videoState.isFullScreen=isFullScreen()
+    })
+}
+
+
+const props=defineProps({
+    videoUrl:{
+        type:String,
+        require:true
+    },
+    speedOpt:{
+        type:Array,
+        default:['2.0','1.5','1.25','1.0','0.8','0.5']
+    }
+})
+
+const videoIns=ref(null)//视频实例
+let videoWrap=ref(null)//视频盒子
+let preloadSlider=computed(()=>{//缓冲的进度
+    return (videoState.cacheTime/videoState.duration)*100+'%'
+})
+let activeSlider=computed(()=>{//当前播放的进度
+    return (videoState.time/videoState.duration)*100+'%'
+})
+//视频状态
+let videoState=reactive({
+    play:false,
+    isPageFullScreen:false,//网页全屏
+    isFullScreen:false,//是否全屏
+    duration:0,//视频时长
+    time:0,//当前播放时长
+    cacheTime:0,//当前缓冲到的时间
+    speed:'1.0',
+    speedOpt:props.speedOpt,
+    volume:100,
+    isHover:false
+})
+const emit=defineEmits(['play','pause','timeupdate'])
+//视频事件
+function videoPlay(e){
+    videoState.play=true
+    emit('play',e)
+}
+function videoCanPlay(e){
+    const target=e.target
+    videoState.duration=target.duration
+}
+function videoTimeUpdate(e){
+    const target=e.target
+    videoState.time=target.currentTime
+    videoState.duration=target.duration
+    emit('timeupdate')
+}
+function videoPause(e){
+    videoState.play=false
+    emit('pause',e)
+}
+function videoProgress(e){
+    const target=e.target
+    videoState.cacheTime=target.buffered.length && target.buffered.end(target.buffered.length - 1)
+}
+
+// 点击暂停\播放视频
+function handleChangeVideoPlay(){
+    if(videoState.play){
+        videoIns.value.pause()
+    }else{
+        videoIns.value.play()
+    }
+}
+
+//切换倍速
+function handleChangeVideoSpeed(item){
+    if(videoState.speed==item) return
+    videoState.speed=item
+    videoIns.value.playbackRate=Number(item)
+}
+
+// 音量变化
+function handleChangeVideoVolume(){
+    videoIns.value.volume=videoState.volume/100
+}
+//一键静音
+function handleVideoMute(){
+    videoState.volume=0
+    videoIns.value.volume=0
+}
+
+//拖动进度条
+function handleDragVideoSlider(val){
+    console.log(val);
+    videoIns.value.currentTime=val
+}
+
+//点击切换全屏
+function handleChangeVideoScreen(){
+    // 如果网页全屏为true 
+    if(videoState.isPageFullScreen){
+        videoState.isPageFullScreen=false
+        return
+    }
+
+    if(!isFullScreen()){
+        setFullScreen(videoWrap.value)
+        setTimeout(() => {
+            // 无法设置全屏 则设置为网页全屏
+            if(!isFullScreen()){
+                videoState.isPageFullScreen=true
+            }
+        }, 100);
+    }else{
+        closeFullScreen()
+    }
+}
+
+// 鼠标进入视频区域
+function mouseMoveVideoWrap(){
+    videoState.isHover=true 
+    hideVideoControlBox()
+}
+//隐藏控制器
+const hideVideoControlBox=debounce(()=>{
+    videoState.isHover=false
+},3000)
+
+onMounted(() => {
+    listenFullScreen()
+})
+
+
+</script>
+
+<template>
+    <div 
+        ref="videoWrap" 
+        :class="[
+            'video-wrap',
+            videoState.isPageFullScreen?'page-fullscreen':''
+        ]"
+        @contextmenu="e=>e.preventDefault()"
+        @mousemove="mouseMoveVideoWrap"
+    >
+        <video 
+            class="video" 
+            ref="videoIns"
+            :src="videoUrl"
+            autoplay
+            @canplay="videoCanPlay"
+            @play="videoPlay" 
+            @pause="videoPause" 
+            @timeupdate="videoTimeUpdate" 
+            @progress="videoProgress"
+            @click="handleChangeVideoPlay"
+            :controls="false"
+            x5-playsinline
+            raw-controls
+            controls360=no
+            playsinline
+            webkit-playsinline
+            x-webkit-airplay="allow"
+            x5-video-player-type="h5"
+            x5-video-player-fullscreen
+            x5-video-orientation="portraint"
+        >
+            <span>您的浏览器不支持 video 标签。</span>
+        </video>
+
+        <!-- 视频控制器模块 -->
+        <div class="video-control-box" v-show="videoState.isHover||!videoState.play">
+            <!-- 进度条 -->
+            <div class="video-progress-slider-box" >
+                <!-- 缓冲进度条 -->
+                <div class="preload-slider" :style="{width:preloadSlider}"></div>
+                <!-- 实时进度 -->
+                <el-slider 
+                    class="active-slider" 
+                    size="small" 
+                    v-model="videoState.time" 
+                    :max="videoState.duration" 
+                    :show-tooltip="false" 
+                    @input="handleDragVideoSlider"
+                />
+            </div>
+            <div class="top-btn-box">
+                <div style="display:flex;align-items: center;">
+                    <div class="play-btn hover-btn" @click.stop="handleChangeVideoPlay">
+                        <svg v-if="!videoState.play" t="1668758429272" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2674" xmlns:xlink="http://www.w3.org/1999/xlink" width="200" height="200">
+                            <path d="M870.2 466.333333l-618.666667-373.28a53.333333 53.333333 0 0 0-80.866666 45.666667v746.56a53.206667 53.206667 0 0 0 80.886666 45.666667l618.666667-373.28a53.333333 53.333333 0 0 0 0-91.333334z" fill="#ffffff" p-id="2675" data-spm-anchor-id="a313x.7781069.0.i0" class="selected"></path>
+                        </svg>
+                        <svg v-else t="1668760616711" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3627" xmlns:xlink="http://www.w3.org/1999/xlink" width="200" height="200">
+                            <path d="M428.539658 833.494155c0 15.954367-13.053294 29.007661-29.007661 29.007661L285.613458 862.501816c-15.954367 0-29.007661-13.053294-29.007661-29.007661l0-639.423111c0-15.954367 13.053294-29.007661 29.007661-29.007661l113.918539 0c15.954367 0 29.007661 13.053294 29.007661 29.007661L428.539658 833.494155z" p-id="3628" data-spm-anchor-id="a313x.7781069.0.i3" class="selected" fill="#ffffff"></path><path d="M760.124635 833.494155c0 15.954367-13.053294 29.007661-29.007661 29.007661l-113.918539 0c-15.954367 0-29.007661-13.053294-29.007661-29.007661l0-639.423111c0-15.954367 13.053294-29.007661 29.007661-29.007661l113.918539 0c15.954367 0 29.007661 13.053294 29.007661 29.007661L760.124635 833.494155z" p-id="3629" data-spm-anchor-id="a313x.7781069.0.i4" class="selected" fill="#ffffff"></path>
+                        </svg>
+                    </div>
+                    <span class="video-time">{{formatTime(videoState.time)}}/{{formatTime(videoState.duration)}}</span>
+                </div>
+                <div style="display:flex;align-items: center;">
+                    <!-- 声音大小 -->
+                    <div class="volume-btn">
+                        <svg v-if="videoState.volume==0" t="1669010015626" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3708" xmlns:xlink="http://www.w3.org/1999/xlink" width="200" height="200">
+                            <path d="M128 420.576v200.864h149.12l175.456 140.064V284.288l-169.792 136.288H128z m132.256-64l204.288-163.968a32 32 0 0 1 52.032 24.96v610.432a32 32 0 0 1-51.968 24.992l-209.92-167.552H96a32 32 0 0 1-32-32v-264.864a32 32 0 0 1 32-32h164.256zM752 458.656L870.4 300.8a32 32 0 1 1 51.2 38.4L792 512l129.6 172.8a32 32 0 0 1-51.2 38.4l-118.4-157.856-118.4 157.856a32 32 0 0 1-51.2-38.4l129.6-172.8-129.6-172.8a32 32 0 0 1 51.2-38.4l118.4 157.856z" p-id="3709" data-spm-anchor-id="a313x.7781069.0.i3" class="selected" fill="#ffffff"></path>
+                        </svg>
+                        <svg v-else @click.stop="handleVideoMute" t="1669010104115" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="6297" xmlns:xlink="http://www.w3.org/1999/xlink" width="200" height="200">
+                            <path d="M257.493333 322.4l215.573334-133.056c24.981333-15.413333 57.877333-7.914667 73.493333 16.746667 5.301333 8.373333 8.106667 18.048 8.106667 27.914666v555.989334C554.666667 819.093333 530.784 842.666667 501.333333 842.666667c-9.994667 0-19.786667-2.773333-28.266666-8L257.493333 701.6H160c-41.237333 0-74.666667-33.013333-74.666667-73.738667V396.138667c0-40.725333 33.429333-73.738667 74.666667-73.738667h97.493333z m26.133334 58.4a32.298667 32.298667 0 0 1-16.96 4.8H160c-5.888 0-10.666667 4.714667-10.666667 10.538667v231.733333c0 5.813333 4.778667 10.538667 10.666667 10.538667h106.666667c5.994667 0 11.872 1.664 16.96 4.8L490.666667 770.986667V253.013333L283.626667 380.8zM800.906667 829.653333a32.288 32.288 0 0 1-45.248-0.757333 31.317333 31.317333 0 0 1 0.768-44.693333c157.653333-150.464 157.653333-393.962667 0-544.426667a31.317333 31.317333 0 0 1-0.768-44.682667 32.288 32.288 0 0 1 45.248-0.757333c183.68 175.306667 183.68 460.010667 0 635.317333z m-106.901334-126.186666a32.288 32.288 0 0 1-45.248-1.216 31.328 31.328 0 0 1 1.237334-44.672c86.229333-80.608 86.229333-210.56 0-291.178667a31.328 31.328 0 0 1-1.237334-44.672 32.288 32.288 0 0 1 45.248-1.216c112.885333 105.546667 112.885333 277.418667 0 382.965333z" p-id="6298" data-spm-anchor-id="a313x.7781069.0.i5" class="selected" fill="#ffffff"></path>
+                        </svg>
+                        <div class="volume-slider-box">
+                            <el-slider v-model="videoState.volume" vertical height="100px" @input="handleChangeVideoVolume" />
+                        </div>
+                    </div>
+                    <!-- 倍速 -->
+                    <div class="speed-btn">
+                        <span class="num">{{videoState.speed}}X</span>
+                        <div class="speed-opt-box">
+                            <span 
+                                :class="['item',videoState.speed==item&&'active']" 
+                                v-for="item in videoState.speedOpt" 
+                                :key="item"
+                                @click.stop="handleChangeVideoSpeed(item)"
+                            >{{item}}x</span>
+                        </div>
+                    </div>
+                    <!-- 全屏按钮 -->
+                    <div class="fullpage-screen-btn hover-btn" @click.stop="handleChangeVideoScreen">
+                        <svg v-if="videoState.isPageFullScreen||videoState.isFullScreen" t="1668999840019" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="1951" xmlns:xlink="http://www.w3.org/1999/xlink" width="200" height="200">
+                            <path d="M298.666667 0c-25.6 0-42.666667 17.066667-42.666667 42.666667v192c0 12.8-8.533333 21.333333-21.333333 21.333333H42.666667c-25.6 0-42.666667 17.066667-42.666667 42.666667s17.066667 42.666667 42.666667 42.666666h192C294.4 341.333333 341.333333 294.4 341.333333 234.666667V42.666667c0-25.6-17.066667-42.666667-42.666666-42.666667zM789.333333 341.333333H981.333333c25.6 0 42.666667-17.066667 42.666667-42.666666s-17.066667-42.666667-42.666667-42.666667h-192c-12.8 0-21.333333-8.533333-21.333333-21.333333V42.666667c0-25.6-17.066667-42.666667-42.666667-42.666667s-42.666667 17.066667-42.666666 42.666667v192C682.666667 294.4 729.6 341.333333 789.333333 341.333333zM234.666667 682.666667H42.666667c-25.6 0-42.666667 17.066667-42.666667 42.666666s17.066667 42.666667 42.666667 42.666667h192c12.8 0 21.333333 8.533333 21.333333 21.333333V981.333333c0 25.6 17.066667 42.666667 42.666667 42.666667s42.666667-17.066667 42.666666-42.666667v-192C341.333333 729.6 294.4 682.666667 234.666667 682.666667zM981.333333 682.666667h-192c-59.733333 0-106.666667 46.933333-106.666666 106.666666V981.333333c0 25.6 17.066667 42.666667 42.666666 42.666667s42.666667-17.066667 42.666667-42.666667v-192c0-12.8 8.533333-21.333333 21.333333-21.333333H981.333333c25.6 0 42.666667-17.066667 42.666667-42.666667s-17.066667-42.666667-42.666667-42.666666z" p-id="1952" data-spm-anchor-id="a313x.7781069.0.i0" class="selected" fill="#ffffff"></path>
+                        </svg>
+                        <svg v-else t="1668762444009" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="2682" xmlns:xlink="http://www.w3.org/1999/xlink" width="200" height="200">
+                            <path d="M170.666667 170.666667v213.333333H85.333333V85.333333h298.666667v85.333334H170.666667z m682.666666 213.333333V170.666667h-213.333333V85.333333h298.666667v298.666667h-85.333334zM170.666667 640v213.333333h213.333333v85.333334H85.333333v-298.666667h85.333334z m682.666666 0h85.333334v298.666667h-298.666667v-85.333334h213.333333v-213.333333z" p-id="2683" data-spm-anchor-id="a313x.7781069.0.i0" class="selected" fill="#ffffff"></path>
+                        </svg>
+                    </div>
+                </div>
+            </div>
+        </div>
+        <!-- 弹幕画布 -->
+        <canvas class="barrage-box"></canvas>
+    </div>
+</template>
+
+<style lang="scss" scoped>
+// 清除浮动
+.clear-float::after {
+  content: "";
+  height: 0;
+  line-height: 0;
+  display: block;
+  visibility: hidden;
+  clear: both;
+}
+.video-wrap{
+    position: relative;
+    width: 100%;
+    height: 100%;
+    overflow: hidden;
+    background-color: rgba(0,0,0,1);
+    .video{
+        width: 100%;
+        height: 100%;
+        object-fit: contain;
+        display: block;
+        box-sizing: border-box;
+    }
+    .video-control-box{
+        position: absolute;
+        z-index: 5;
+        left: 0;
+        right: 0;
+        bottom: 0;
+        .top-btn-box{
+            padding: 5px;
+            background-color: rgba(0,0,0,0.7);
+            display: flex;
+            align-items: center;
+            justify-content: space-between;
+            position: relative;
+            z-index: 10;
+            .hover-btn{
+                cursor: pointer;
+                &:hover{
+                    background-color: #303031;
+                }
+            }
+            .play-btn{
+                width: 40px;
+                height: 40px;
+                padding-left: 10px;
+                padding-top: 10px;
+                border-radius: 50%;
+                .icon{
+                    width: 20px;
+                    height: 20px;
+                }
+            }
+            .video-time{
+                color: #fff;
+                font-size: 14px;
+                display: inline-block;
+                margin-left: 10px;
+            }
+            .fullpage-screen-btn{
+                width: 40px;
+                height: 40px;
+                padding-left: 10px;
+                padding-top: 10px;
+                border-radius: 50%;
+                .icon{
+                    width: 20px;
+                    height: 20px;
+                }
+            }
+            .speed-btn{
+                color: #fff;
+                font-size: 13px;
+                margin-right: 10px;
+                cursor: pointer;
+                display: inline-block;
+                padding: 3px 8px;
+                border-radius: 20px;
+                position: relative;
+                &:hover{
+                    background-color: #303031;
+                    .speed-opt-box{
+                        display: block;
+                    }
+                }
+                .speed-opt-box{
+                    display: none;
+                    position: absolute;
+                    width: 80px;
+                    bottom: 105%;
+                    left: 50%;
+                    transform: translateX(-50%);
+                    background-color: #303031;
+                    color: #fff;
+                    font-size: 14px;
+                    border-radius: 2px;
+                    padding: 5px 0;
+                    .item{
+                        display: block;
+                        text-align: center;
+                        padding: 5px 0;
+                        &.active{
+                            color: #F3A52F;
+                        }
+                    }
+                }
+            }
+            .volume-btn{
+                width: 40px;
+                height: 40px;
+                padding-left: 10px;
+                padding-top: 10px;
+                border-radius: 50%;
+                cursor: pointer;
+                margin-right: 10px;
+                position: relative;
+                &:hover{
+                    .volume-slider-box{
+                        display: block;
+                    }
+                }
+                .icon{
+                    width: 20px;
+                    height: 20px;
+                }
+                .volume-slider-box{
+                    display: none;
+                    position: absolute;
+                    bottom: 100%;
+                    left: 50%;
+                    transform: translateX(-50%);
+                }
+            }
+        }
+
+        .video-progress-slider-box{
+            position: relative;
+            width: 100%;
+            height: 3px;
+            background-color: #303031;
+            cursor: pointer;
+            .preload-slider{
+                position: absolute;
+                height: 100%;
+                background-color: #717171;
+                left: 0;
+                z-index: 1;
+            }
+            .active-slider{
+                position: absolute;
+                height: 100%;
+                left: 0;
+                z-index: 15;
+                --el-slider-height:3px;
+                --el-slider-button-wrapper-size:18px;
+                --el-slider-button-size:16px;
+                --el-slider-button-wrapper-offset:-12px;
+                :deep(.el-slider__button-wrapper){
+                    display: none;
+                }
+                :deep(.el-slider__runway){
+                    background-color: transparent;
+                    &:hover{
+                        .el-slider__button-wrapper{
+                            display: block;
+                        }
+                    }
+                }
+                
+            }
+        }
+    }
+    .barrage-box{
+        width: 100%;
+        height: 100%;
+        position: absolute;
+        left: 0;
+        top: 0;
+        right: 0;
+        bottom: 0;
+        pointer-events: none;
+        z-index: 10;
+    }
+}
+.page-fullscreen{
+    position: fixed;
+    left: 0;
+    top: 0;
+    right: 0;
+    bottom: 0;
+}
+
+
+</style>

+ 4 - 0
src/main.js

@@ -5,7 +5,11 @@ import ElementPlus from 'element-plus'
 import 'element-plus/dist/index.css'
 import 'normalize.css'
 import "@/style/global.scss";//全局样式
+import VConsole from 'vconsole';
 
+if(import.meta.env.MODE==='test'){
+    const vConsole = new VConsole();
+}
 
 const app = createApp(App)
 

+ 5 - 0
src/router/index.js

@@ -25,6 +25,11 @@ const routes=[
     name: "ReportDetail",
     component: ()=>import("@/views/report/Detail.vue"),
   },
+  {
+    path: "/roadshow/detail",
+    name: "RoadshowDetail",
+    component: ()=>import("@/views/roadShow/Detail.vue"),
+  },
   
   {
     path: '/:pathMatch(.*)',

+ 101 - 0
src/views/report/Detail.vue

@@ -62,6 +62,54 @@ function handlePreviewImg(e){
     showPreViewImg.value=true
 }
 
+//分类
+let navList=ref([])
+let showFilter=ref(false)
+async function getReportNav(){
+    const res=await apiReportClassifyList({classify_type:0})
+    const videores=await apiReportClassifyList({classify_type:1})
+    if(res.code===200){
+        const arr=res.data.list||[]
+        let videoResList=[]
+        if(videores.code===200){
+            videoResList=videores.data.list||[]
+        }
+        navList.value=[
+            ...arr,
+            {
+                id: -1,
+                classify_name: "ONLINE VEDIO",
+                child: videoResList
+            },
+            {
+                id: 0,
+                classify_name: "ABOUT US",
+                child: []
+            },
+        ]
+    }
+}
+getReportNav()
+
+function handleChangeFirstClassify(item){
+    if(item.id==0){
+        router.replace({
+            path:'/'
+        })
+    }    
+}
+
+function handleChangeSecClassify(item,pitem){
+        router.replace({
+            path:'/report/index',
+            query:{
+                firstclassifyid:pitem.id,
+                secclassifyid:item.id
+            }
+        })
+
+}
+
 onMounted(()=>{
     wwidth.value=window.innerWidth
 })
@@ -77,6 +125,7 @@ onMounted(()=>{
             <div style="color:#00459F;cursor: pointer;" @click="$router.replace('/')">ABOUT US</div>
         </div>
         <div class="mobile-header-wrap">
+            <img class="menu-icon" @click="showFilter=true;filterSize='100%'" src="@/assets/menu.svg" alt="">
             <span @click="$router.replace('/')">HORIZON INSIGHTS</span>
         </div>
         <div class="no-select-text detail-wrap">
@@ -111,8 +160,31 @@ onMounted(()=>{
         @close="showPreViewImg=false"
         :url-list="preViewImgs"
     />
+    <!-- 筛选 -->
+    <el-drawer
+        v-model="showFilter"
+        direction="ltr"
+        size="100%"
+        class="self-drawer-box"
+    >
+        <div class="filter-wrap">
+            <div class="item" v-for="item in navList" :key="item.id">
+                <span  class="title" @click="handleChangeFirstClassify(item)">{{item.classify_name}}</span>
+                <div class="text" v-for="_item in item.child" :key="_item.id" @click="handleChangeSecClassify(_item,item)">{{_item.classify_name}}</div>
+            </div>
+        </div>
+    </el-drawer>
 </template>
 
+<style lang="scss">
+.self-drawer-box{
+
+    .el-drawer__header{
+        margin-bottom: 0;
+    }
+}
+</style>
+
 <style lang="scss" scoped>
 .report-detail-page{
     padding: 0 20px;
@@ -151,6 +223,14 @@ onMounted(()=>{
     background-color: #fff;
     z-index: 99;
     display: none;
+    position: relative;
+    .menu-icon{
+        position: absolute;
+        left: 20px;
+        top: 50%;
+        transform: translateY(-50%);
+        width: 17px;
+    }
 }
 @media (max-width: 768px){
     .header-wrap{
@@ -259,4 +339,25 @@ onMounted(()=>{
     }
 }
 
+.filter-wrap{
+    .item{
+        margin-bottom: 50px;
+        .title{
+            font-size: 28px;
+            font-weight: bold;
+            margin-bottom: 30px;
+            display: block;
+            cursor: pointer;
+        }
+        .text{
+            cursor: pointer;
+            font-size: 18px;
+            margin-bottom: 20px;
+            &:hover{
+                color: var(--el-color-primary);
+            }
+        }
+    }
+}
+
 </style>

+ 40 - 10
src/views/report/Index.vue

@@ -3,6 +3,8 @@ import {ref,reactive, onMounted} from 'vue'
 import { useRoute, useRouter } from "vue-router";
 import {apiReportClassifyList,apiReportList} from '@/api/report'
 import { useWindowSize } from '@vueuse/core'
+import videoPlayBox from '@/components/VideoPlayBox.vue'
+import roadShowList from '@/views/roadShow/List.vue'
 
 const { width, height } = useWindowSize()
 
@@ -12,20 +14,43 @@ const route=useRoute()
 //分类
 let navList=ref([])
 async function getReportClassify(){
-    const res=await apiReportClassifyList()
+    const res=await apiReportClassifyList({classify_type:0})
+    const videores=await apiReportClassifyList({classify_type:1})
+
     if(res.code===200){
         const arr=res.data.list||[]
+        let videoResList=[]
+        if(videores.code===200){
+            videoResList=videores.data.list||[]
+        }
         navList.value=[
+            ...arr,
+            {
+                id: -1,
+                classify_name: "Online Video",
+                child: videoResList
+            },
             {
                 id: 0,
                 classify_name: "ABOUT US",
                 child: []
-            },
-            ...arr
+            }
         ]
 
+        if(route.query.secclassifyid){
+            navList.value.forEach(item => {
+                if(item.id==route.query.firstclassifyid){
+                    item.child.forEach(citem=>{
+                        if(citem.id==route.query.secclassifyid){
+                            handleChangeSecClassify(citem,item)
+                        }
+                    })
+                }
+            });
+            return
+        }
         if(route.query.firstclassifyid){
-            arr.forEach(item => {
+            navList.value.forEach(item => {
                 if(item.id==route.query.firstclassifyid){
                     handleChangeFirstClassify(item)
                 }
@@ -98,14 +123,13 @@ async function getList(){
         if(arr.length<listState.pageSize){
             listState.finished=true
         }
-        
     }
 }
 
 // 监听页面滚动
 function listenScroll(){
     window.onscroll=(e)=>{
-        if(listState.firstClassifyId===0) return
+        if([0,-1].includes(listState.firstClassifyId)) return
         if(listState.loading||listState.finished) return
         const scrollTop = document.documentElement.scrollTop||document.body.scrollTop;
         const windowHeight = document.documentElement.clientHeight || document.body.clientHeight;
@@ -156,7 +180,7 @@ let filterSize=ref('30%')
                 @click="handleChangeFirstClassify(item)"
             >{{item.classify_name}}</span>
         </div>
-        <div class="sub-nav-wrap">
+        <div class="sub-nav-wrap" v-if="listState.firstClassifyId!=-1">
             <h2 class="label">{{listState.firstClassifyName}}</h2>
             <div class="sub-nav-list">
                 <span 
@@ -168,7 +192,7 @@ let filterSize=ref('30%')
             </div>
         </div>
         <!-- 固定展示的报告区域 -->
-        <div class="fix-report-wrap" v-show="listState.firstClassifyId!=0&&listState.secClassifyId===0">
+        <div class="fix-report-wrap" v-show="![-1,0].includes(listState.firstClassifyId)&&listState.secClassifyId===0">
             <div class="left-wrap">
                 <div class="report-item-normal" v-for="item in listState.list.slice(0,3)" :key="item.id" @click="goReportDetail(item)">
                     <div class="title">{{item.title}}</div>
@@ -203,7 +227,7 @@ let filterSize=ref('30%')
             </div>
         </div>
         <!-- 报告列表 -->
-        <div class="report-list-wrap" v-show="listState.secClassifyId!=0">
+        <div class="report-list-wrap" v-show="listState.secClassifyId!=0&&listState.firstClassifyId!=-1">
             <div class="item" v-for="item in listState.list" :key="item.id" @click="goReportDetail(item)">
                 <div class="title">
                     <span class="label">【NO.{{item.stage}}|FICC】</span>
@@ -219,7 +243,10 @@ let filterSize=ref('30%')
         <div class="about-us-box" v-show="listState.firstClassifyId==0||(listState.firstClassifyId!=0&&listState.secClassifyId==0&&width<768)">
             <div class="flex box top-box">
                 <div class="left">
-                    <video controls autoplay loop src="https://hongze.oss-cn-shanghai.aliyuncs.com/static/video_03.mp4"></video>
+                    <!-- <video controls loop src="https://hongze.oss-cn-shanghai.aliyuncs.com/static/video_03.mp4"></video> -->
+                    <videoPlayBox
+                        videoUrl="https://hongze.oss-cn-shanghai.aliyuncs.com/static/video_03.mp4"
+                    />
                 </div>
                 <div class="right">
                     <div class="con">
@@ -247,6 +274,9 @@ let filterSize=ref('30%')
                 </div>
             </div>
         </div>
+
+        <!-- 路演视频模块 -->
+        <roadShowList v-if="listState.firstClassifyId==-1" :defaultClassifyId="listState.secClassifyId"></roadShowList>
         </div>
     </div>
     <!-- 筛选 -->

+ 59 - 7
src/views/report/Search.vue

@@ -52,6 +52,16 @@ function goReportDetail(item){
     })
     window.open(url.href,'_blank')
 }
+//跳转详情
+function goVideoDetail(item){
+    const url=router.resolve({
+        path:'/roadshow/detail',
+        query:{
+            code:item.report_code
+        }
+    })
+    window.open(url.href,'_blank')
+}
 
 </script>
 
@@ -71,14 +81,23 @@ function goReportDetail(item){
         </div>
         <div v-if="listState.list.length===0" class="empty">no results</div>
         <div class="report-list-wrap" v-else>
-            <div class="item" v-for="item in listState.list" :key="item.id" @click="goReportDetail(item)">
-                <div class="title">
-                    <span class="label">【NO.{{item.stage}}|FICC】</span>
-                    <span v-html="item.title"></span>
-                    <span v-if="item.create_time">({{item.create_time.substring(5,7)}}{{item.create_time.substring(8,10)}})</span>
+            <div class="item" v-for="item in listState.list" :key="item.id" >
+                <div @click="goReportDetail(item)" v-if="item.report_type===0">
+                    <div class="title">
+                        <span class="label">【NO.{{item.stage}}|FICC】</span>
+                        <span v-html="item.title"></span>
+                        <span v-if="item.create_time">({{item.create_time.substring(5,7)}}{{item.create_time.substring(8,10)}})</span>
+                    </div>
+                    <div class="intro" v-html="item.content_sub"></div>
+                    <div class="time">{{item.publish_time}}</div>
+                </div>
+                <div @click="goVideoDetail(item)" v-if="item.report_type===1">
+                    <div class="title" v-html="item.title"></div>
+                    <div class="video-img">
+                        <img :src="item.video_cover_url" alt="">
+                    </div>
+                    <div class="time">{{item.publish_time}}</div>
                 </div>
-                <div class="intro" v-html="item.content_sub"></div>
-                <div class="time">{{item.publish_time}}</div>
             </div>
         </div>
     </div>
@@ -170,6 +189,31 @@ function goReportDetail(item){
             line-height: 17px;
             margin: 10px 0;
         }
+        .video-img{
+            width: 216px;
+            height: 132px;
+            margin: 10px 0;
+            img{
+                width: 100%;
+                height: 100%;
+                object-fit: cover;
+                display: block;
+            }
+            position: relative;
+                &::after{
+                    content: '';
+                    display: block;
+                    width: 32px;
+                    height: 32px;
+                    background-image: url('@/assets/icon_play.png');
+                    background-size: cover;
+                    position: absolute;
+                    left: 50%;
+                    top: 50%;
+                    transform: translate(-50%,-50%);
+                    cursor: pointer;
+                }
+        }
         .time{
             font-size: 14px;
             color: #999;
@@ -196,6 +240,14 @@ function goReportDetail(item){
             .time{
                 font-size: 12px;
             }
+            .video-img{
+                width: 100%;
+                height: 200px;
+                &::after{
+                    width: 50px;
+                    height: 50px;
+                }
+            }
         }
     }
 }

+ 140 - 0
src/views/roadShow/Detail.vue

@@ -0,0 +1,140 @@
+<script setup>
+import {ref} from 'vue'
+import {apiRoadShowVideoDetail,apiVideoPlayLog} from '@/api/roadShow'
+import videoPlayBox from '@/components/VideoPlayBox.vue'
+import { useRoute } from 'vue-router'
+
+const route=useRoute()
+
+const code=route.query.code
+const email=route.query.ShareEmail||0
+let info=ref(null)
+// 视频详情
+async function getDetail(){
+    const res=await apiRoadShowVideoDetail({
+        video_code:code,
+        share_email:email
+    })
+    if(res.code===200){
+        info.value=res.data.Video
+    }
+}
+getDetail()
+
+let videoPlayRecordId=0
+//视频播放
+function handleVideoPlay(){
+    if(videoPlayRecordId) return
+    apiVideoPlayLog({
+        id:0,
+        video_id:info.value.id,
+        stop_seconds:0
+    }).then(res=>{
+        if(res.code==200){
+            videoPlayRecordId=res.data.id
+        }
+    })
+}
+
+function handleVideoPause(e){
+    const target=e.target||e.path[0]
+    const t=target.currentTime||0
+    apiVideoPlayLog({
+        id:videoPlayRecordId,
+        video_id:info.value.id,
+        stop_seconds:parseInt(t)
+    }).then(res=>{
+        if(res.code==200){
+            console.log('记录视频播放时长成功');
+        }
+    })
+}
+
+
+
+</script>
+
+<template>
+    <div class="roadshow-video-detail-page">
+        <div class="header-wrap">
+            <span>HORIZON INSIGHTS</span>
+            <div class="search-box" @click="$router.push('/report/search')">
+                <img src="@/assets/search.svg" alt="">
+                <span style="margin-left:10px">search for</span>
+            </div>
+        </div>
+        <div class="detail-wrap" v-if="info">
+            <div class="title">{{info.title}}</div>
+            <videoPlayBox
+                :videoUrl="info.video_url"
+                @play="handleVideoPlay"
+                @pause="handleVideoPause"
+            />
+        </div>
+    </div>
+</template>
+
+<style lang="scss" scoped>
+.header-wrap{
+    text-align: center;
+    font-size: 40px;
+    font-weight: bold;
+    color: var(--el-color-primary);
+    padding: 40px 0;
+    border-bottom: 1px solid#E6E6E6;
+    position: sticky;
+    top: 0;
+    background-color: #fff;
+    z-index: 99;
+    .search-box{
+        position: absolute;
+        top: 50%;
+        transform: translateY(-50%);
+        right: 20px;
+        font-size: 16px;
+        color: #333;
+        font-weight: 500;
+        cursor: pointer;
+        span,img{
+            vertical-align: middle;
+        }
+    }
+    .menu-icon{
+        position: absolute;
+        left: 20px;
+        top: 50%;
+        transform: translateY(-50%);
+        display: none;
+    }
+}
+@media (max-width: 768px){
+    .header-wrap{
+        border: none;
+        font-size: 17px;
+        box-shadow: 0px 4px 20px rgba(180, 180, 180, 0.16);
+        padding: 15px 0;
+        .search-box{
+            span{
+                display: none;
+            }
+        }
+        .menu-icon{
+            display: block;
+            width: 17px;
+        }
+    }
+}
+.detail-wrap{
+    margin-top: 20px;
+    .title{
+        margin-bottom: 20px;
+    }
+}
+@media (max-width: 768px){
+    .detail-wrap{
+        .title{
+            padding: 0 16px;
+        }
+    }
+}
+</style>

+ 363 - 0
src/views/roadShow/List.vue

@@ -0,0 +1,363 @@
+<script setup>
+import {ref,reactive, onMounted, watch} from 'vue'
+import {apiReportClassifyList} from '@/api/report'
+import {apiRoadShowVideoList} from '@/api/roadShow'
+import { useRouter } from 'vue-router'
+
+const props=defineProps({
+    defaultClassifyId:{
+        type:Number,
+        default:0
+    }
+})
+
+const router=useRouter()
+
+watch(
+    ()=>props.defaultClassifyId,
+    (n)=>{
+        if(n&&navList.value.length){
+             navList.value.forEach(item=>{
+                if(item.id==props.defaultClassifyId){
+                    handleChangeFirstClassify(item)
+                }
+            })
+        }
+    }
+)
+
+let navList=ref([])
+async function getReportClassify(){
+    const res=await apiReportClassifyList({classify_type:1})
+    if(res.code===200){
+        const arr=res.data.list||[]
+        navList.value=[
+            {
+                id: 0,
+                classify_name: "ALL",
+                child: []
+            },
+            ...arr,
+        ]
+        if(props.defaultClassifyId){
+            navList.value.forEach(item=>{
+                if(item.id==props.defaultClassifyId){
+                    handleChangeFirstClassify(item)
+                }
+            })
+        }else{
+            initList()
+        }
+    }
+}
+getReportClassify()
+
+let listState=reactive({
+    firstClassifyId:0,
+    secClassifyId:0,
+    secClassifyOpt:[],
+
+    page:1,
+    pageSize:20,
+    list:[],
+    loading:false,
+    finished:false
+})
+
+function handleChangeFirstClassify(item){
+    // if(listState.firstClassifyId===item.id) return
+    listState.firstClassifyId=item.id
+    listState.secClassifyId=0
+    listState.secClassifyOpt=item.child||[]
+    initList()
+}
+
+function handleChangeSecClassify(item){
+    if(listState.secClassifyId===item.id) return
+    listState.secClassifyId=item.id
+    initList()
+}
+
+//初始化列表
+function initList(){
+    listState.page=1
+    listState.finished=false
+    listState.list=[]
+    getVideoList()
+}
+
+
+async function getVideoList(){
+    listState.loading=true
+    const res=await apiRoadShowVideoList({
+        page_size:listState.pageSize,
+        current:listState.page,
+        classify_id_first:listState.firstClassifyId,
+        classify_id_second:listState.secClassifyId
+    })
+    setTimeout(() => {
+        listState.loading=false
+    }, 100);
+    if(res.code===200){
+        const arr=res.data.list||[]
+        listState.list=[...listState.list,...arr]
+        if(arr.length<listState.pageSize){
+            listState.finished=true
+        }
+    }
+}
+
+// 监听页面滚动
+function listenScroll(){
+    window.onscroll=(e)=>{
+        if(listState.loading||listState.finished) return
+        const scrollTop = document.documentElement.scrollTop||document.body.scrollTop;
+        const windowHeight = document.documentElement.clientHeight || document.body.clientHeight;
+        const scrollHeight = document.documentElement.scrollHeight||document.body.scrollHeight;
+        if(scrollTop+windowHeight>=scrollHeight){   //考虑到滚动的位置一般可能会大于一点可滚动的高度,所以这里不能用等于
+            console.log("距顶部"+scrollTop+"可视区高度"+windowHeight+"滚动条总高度"+scrollHeight);
+            listState.page++
+            getVideoList()
+        }   
+    }
+}
+listenScroll()
+
+function goDetail(item){
+    const url=router.resolve({
+        path:'/roadshow/detail',
+        query:{
+            code:item.video_code
+        }
+    })
+    window.open(url.href,'_blank')
+}
+
+// 格式化视频时长
+function getVideoTime(e){
+    const minus = parseInt(e / 60);
+    const sec = parseInt(e % 60);
+    return `${minus > 9 ? minus : "0" + minus}:${sec > 9 ? sec : "0" + sec}`;
+}
+
+
+</script>
+
+<template>
+    <div class="roadshow-list-page">
+        <div class="nav-warp">
+            <div class="first-nav-box">
+                <span 
+                    v-for="item in navList" 
+                    :key="item.id"
+                    :class="item.id===listState.firstClassifyId?'active':''"
+                    @click="handleChangeFirstClassify(item)"
+                >{{item.classify_name}}</span>
+            </div>
+            <div class="sec-nav-box">
+                <span 
+                    v-for="item in listState.secClassifyOpt" 
+                    :key="item.id"
+                    :class="item.id===listState.secClassifyId?'active':''"
+                    @click="handleChangeSecClassify(item)"
+                >{{item.classify_name}}</span>
+            </div>
+        </div>
+        <div class="mobile-nav-wrap">
+            <span :class="!listState.secClassifyId?'active':''" @click="handleChangeSecClassify({id:0})">All</span>
+            <span 
+                v-for="item in listState.secClassifyOpt" 
+                :key="item.id"
+                :class="item.id===listState.secClassifyId?'active':''"
+                @click="handleChangeSecClassify(item)"
+            >{{item.classify_name}}</span>
+        </div>
+        <div class="empty-box" v-if="listState.list.length==0&&listState.finished">no results</div>
+        <div class="video-list-wrap" v-else>
+            <div class="item" v-for="item in listState.list" :key="item.id" @click="goDetail(item)">
+                <div class="multi-ellipsis-l2 title">{{item.title}}</div>
+                <div class="img">
+                    <img :src="item.video_cover_url" alt="">
+                    <span class="time">{{getVideoTime(item.video_seconds)}}</span>
+                </div>
+                <div class="time">{{item.publish_time}}</div>
+            </div>
+            <div class="last-item"></div>
+            <div class="last-item"></div>
+        </div>
+    </div>
+</template>
+
+<style lang="scss" scoped>
+.roadshow-list-page{
+    .first-nav-box{
+        padding: 20px 0 28px 0;
+        span{
+            display: inline-block;
+            margin-right: 40px;
+            cursor: pointer;
+            position: relative;
+            &.active::after{
+                display: block;
+                content: '';
+                position: absolute;
+                bottom: -10px;
+                width: 100%;
+                height: 4px;
+                background-color: var(--el-color-primary);
+            }
+            &:hover{
+                color: var(--el-color-primary);
+            }
+        }
+    }
+    .sec-nav-box{
+        span{
+            display: inline-block;
+            padding: 5px 20px;
+            cursor: pointer;
+            background: #F3F3F3;
+            border-radius: 4px;
+            color: #666;
+            margin-right: 20px;
+            &.active{
+                color: var(--el-color-primary);
+            }
+        }
+    }
+    .mobile-nav-wrap{
+        display: none;
+    }
+    .empty-box{
+        text-align: center;
+        color: #666;
+        font-size: 16px;
+        padding-top: 100px;
+    }
+    .video-list-wrap{
+        margin-top: 30px;
+        display: flex;
+        flex-wrap: wrap;
+        justify-content: center;
+        .item{
+            width: 384px;
+            padding: 37px 12px 30px 12px;
+            border: 1px solid #E6E6E6;
+            .title{
+                min-height: 36px;
+                margin-bottom: 10px;
+            }
+            .time{
+                text-align: right;
+                margin-top: 10px ;
+                color: #999;
+            }
+            .img{
+                width: 358px;
+                height: 220px;
+                img{
+                    object-fit: cover;
+                    display: block;
+                    cursor: pointer;
+                    width: 100%;
+                    height: 100%;
+                }
+                
+                position: relative;
+                .time{
+                    position: absolute;
+                    bottom: 10px;
+                    right: 10px;
+                    background-color: rgba(0, 0, 0, 0.3);
+                    color: #fff;
+                    font-size: 12px;
+                    padding: 4px 8px;
+                }
+                &::after{
+                    content: '';
+                    display: block;
+                    width: 55px;
+                    height: 55px;
+                    background-image: url('@/assets/icon_play.png');
+                    background-size: cover;
+                    position: absolute;
+                    left: 50%;
+                    top: 50%;
+                    transform: translate(-50%,-50%);
+                    cursor: pointer;
+                }
+            }
+        }
+        .last-item{
+            width: 384px;
+            height: 0;
+        }
+    }
+}
+@media (max-width: 768px){
+    .roadshow-list-page{
+        .nav-warp{
+            display: none;
+        }
+        .mobile-nav-wrap{
+            display: flex;
+            padding: 15px 20px 10px 20px;
+            border-bottom: 1px solid #E6E6E6;
+            width: 100vw;
+            position: relative;
+            left: -20px;
+            overflow-x: auto;
+            margin-bottom: 10px;
+            span{
+                margin-right: 25px;
+                position: relative;
+                &.active::after{
+                    display: block;
+                    content: '';
+                    position: absolute;
+                    bottom: -10px;
+                    width: 100%;
+                    height: 4px;
+                    background-color: var(--el-color-primary);
+                }
+            }
+        }
+        // .first-nav-box{
+        //     padding: 15px 20px 10px 20px;
+        //     border-bottom: 1px solid #E6E6E6;
+        //     width: 100vw;
+        //     position: relative;
+        //     left: -20px;
+        //     display: flex;
+        //     overflow-x: auto;
+        //     margin-bottom: 10px;
+        //     span{
+        //         margin-right: 25px;
+        //         &.active::after{
+        //             bottom: -10px;
+        //             height: 2px;
+        //         }
+        //     }
+        // }
+        .video-list-wrap{
+            display: block;
+            margin-top: 0;
+            .item{
+                width: 100%;
+                border-top: none;
+                border-left: none;
+                border-right: none;
+                padding: 15px 0;
+                .title{
+                    min-height: 0;
+                    margin-bottom: 5px;
+                }
+                .img{
+                    width: 100%;
+                    height: 200px;
+                }
+            }
+        }
+    }
+}
+</style>