|  | @@ -0,0 +1,471 @@
 | 
	
		
			
				|  |  | +<template>
 | 
	
		
			
				|  |  | +  <div>
 | 
	
		
			
				|  |  | +    <h2 class="tc">
 | 
	
		
			
				|  |  | +      <span>{{curTimes|useTime}}</span>
 | 
	
		
			
				|  |  | +      <strong>/</strong>
 | 
	
		
			
				|  |  | +      <span>{{media.duration|useTime}}</span>
 | 
	
		
			
				|  |  | +    </h2>
 | 
	
		
			
				|  |  | +    <el-row style="width: 1100px;">
 | 
	
		
			
				|  |  | +      <el-col :span="18" style="width:680px;">
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        <div style="height: 540px;">
 | 
	
		
			
				|  |  | +        <video-player id="myVideo" class="video-player-box" ref="videoPlayer" :playsinline="true"
 | 
	
		
			
				|  |  | +          @pause="onPlayerPause($event)" @play="onPlayerStart($event)" @ready="playerReadied"
 | 
	
		
			
				|  |  | +          @timeupdate="onPlayerTimeupdate($event)" @ended="onPlayerEnded($event)" :globalOptions="{controls:true}"
 | 
	
		
			
				|  |  | +          :options="options">
 | 
	
		
			
				|  |  | +        </video-player>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        </div>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        <div class="tc">
 | 
	
		
			
				|  |  | +             <p  v-if="errMsg" style="font-size: 30px;color: red;"> {{errMsg}}</p>
 | 
	
		
			
				|  |  | +        </div>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      </el-col>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      <el-col :span="6" style="width: 420px;float:right;">
 | 
	
		
			
				|  |  | +        <div class="account-tit2">
 | 
	
		
			
				|  |  | +          <a :class="{'current':required===1}" @click="required=1" style="width: 100px;">必修课程</a>
 | 
	
		
			
				|  |  | +          <a :class="{'current':required===0}" @click="required=0" style="width: 100px;">选修课程</a>
 | 
	
		
			
				|  |  | +          <a :class="{'current':required===2}" @click="required=2" style="width: 100px;">讨论区</a>
 | 
	
		
			
				|  |  | +        </div>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        <el-menu v-if="required==1" style="width: 400px;font-size: 10px;">
 | 
	
		
			
				|  |  | +          <ul class="m-chapter-list">
 | 
	
		
			
				|  |  | +            <li v-for="(item, index) in chapter.required" :key="index" :class="{'current':item.name==activeChapter}">
 | 
	
		
			
				|  |  | +              <a href="javascript:void(0)" style="text-decoration: none" @click="goState(item)" class="ng-binding">
 | 
	
		
			
				|  |  | +                {{parseInt(item.xs/10)}}学时 {{item.name}}</a>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +              <div class="sub-list" v-if="item.name==activeChapter">
 | 
	
		
			
				|  |  | +                <a v-for="(subItem,index)  in list" :key="subItem.id" v-if="subItem.chapterName == activeChapter"
 | 
	
		
			
				|  |  | +                  @click="goSubState(subItem, index)" :class="{'current':subItem.name==activeName}">
 | 
	
		
			
				|  |  | +                  <span style="width: 8px;height: 8px;">
 | 
	
		
			
				|  |  | +                    <el-progress :percentage="subItem.percent" type="circle" :width="16" :height="16"
 | 
	
		
			
				|  |  | +                      :format="()=>{return ''}" v-if="subItem.percent>=100" color="green"></el-progress>
 | 
	
		
			
				|  |  | +                    <el-progress :percentage="subItem.percent" type="circle" :width="16" :height="16"
 | 
	
		
			
				|  |  | +                      :format="()=>{return ''}" v-else-if="subItem.percent>=50" color="cyan"></el-progress>
 | 
	
		
			
				|  |  | +                    <el-progress :percentage="subItem.percent" type="circle" :width="16" :height="16"
 | 
	
		
			
				|  |  | +                      :format="()=>{return ''}" v-else></el-progress>
 | 
	
		
			
				|  |  | +                  </span>
 | 
	
		
			
				|  |  | +                  <span style="margin-left: 16px;"> {{subItem.name}} </span>
 | 
	
		
			
				|  |  | +                </a>
 | 
	
		
			
				|  |  | +              </div>
 | 
	
		
			
				|  |  | +            </li>
 | 
	
		
			
				|  |  | +          </ul>
 | 
	
		
			
				|  |  | +        </el-menu>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        <el-menu v-if="required==0">
 | 
	
		
			
				|  |  | +          <ul class="m-chapter-list" style="width: 400px;font-size: 10px;">
 | 
	
		
			
				|  |  | +            <li v-for="(item, index) in chapter.normal" :key="index" :class="{'current':item.name==activeChapter}">
 | 
	
		
			
				|  |  | +              <a href="javascript:void(0)" style="text-decoration: none" @click="goState(item)"
 | 
	
		
			
				|  |  | +                class="ng-binding">{{parseInt(item.xs/10)}}学时 {{item.name}}</a>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +              <div class="sub-list pt10" v-if="item.name==activeChapter">
 | 
	
		
			
				|  |  | +                <a v-for="(subItem,index) in list" :key="subItem.id" v-if="subItem.chapterName == activeChapter"
 | 
	
		
			
				|  |  | +                  @click="goSubState(subItem, index)" :class="{'current':subItem.name==activeName}">
 | 
	
		
			
				|  |  | +                  <span style="width: 8px;height: 8px;">
 | 
	
		
			
				|  |  | +                    <el-progress :percentage="subItem.percent" type="circle" :width="16" :height="16"
 | 
	
		
			
				|  |  | +                      :format="()=>{return ''}" v-if="subItem.percent>=100" color="green"></el-progress>
 | 
	
		
			
				|  |  | +                    <el-progress :percentage="subItem.percent" type="circle" :width="16" :height="16"
 | 
	
		
			
				|  |  | +                      :format="()=>{return ''}" v-else-if="subItem.percent>=50" color="cyan"></el-progress>
 | 
	
		
			
				|  |  | +                    <el-progress :percentage="subItem.percent" type="circle" :width="16" :height="16"
 | 
	
		
			
				|  |  | +                      :format="()=>{return ''}" v-else></el-progress>
 | 
	
		
			
				|  |  | +                  </span>
 | 
	
		
			
				|  |  | +                  <span style="margin-left: 16px;"> {{subItem.name}} </span>
 | 
	
		
			
				|  |  | +                </a>
 | 
	
		
			
				|  |  | +                </a>
 | 
	
		
			
				|  |  | +              </div>
 | 
	
		
			
				|  |  | +            </li>
 | 
	
		
			
				|  |  | +          </ul>
 | 
	
		
			
				|  |  | +          <!-- <el-menu-item v-for="item in chapter.normal" :key="item">{{item}}</el-menu-item> -->
 | 
	
		
			
				|  |  | +        </el-menu>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        <div v-if="required==2">
 | 
	
		
			
				|  |  | +          <i-message :media-id="media.mediaId"></i-message>
 | 
	
		
			
				|  |  | +        </div>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      </el-col>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    </el-row>
 | 
	
		
			
				|  |  | +    <div class="left-float" v-if="!closeFace" v-drag>
 | 
	
		
			
				|  |  | +     <i-collect></i-collect>
 | 
	
		
			
				|  |  | +    </div>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  </div>
 | 
	
		
			
				|  |  | +</template>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +<script>
 | 
	
		
			
				|  |  | +  // import tracking from '@/assets/tracking/build/tracking-min.js';
 | 
	
		
			
				|  |  | +  // import '@/assets/tracking/build/data/face-min.js';
 | 
	
		
			
				|  |  | +  import {
 | 
	
		
			
				|  |  | +    httpServer
 | 
	
		
			
				|  |  | +  } from "@/components/httpServer/httpServer.js";
 | 
	
		
			
				|  |  | +  import ICollect from "@/components/face/icollect.vue";
 | 
	
		
			
				|  |  | +  import md5 from 'js-md5';
 | 
	
		
			
				|  |  | +  import {
 | 
	
		
			
				|  |  | +    videoPlayer
 | 
	
		
			
				|  |  | +  } from 'vue-video-player';
 | 
	
		
			
				|  |  | +  import IMessage from './iMessage.vue'
 | 
	
		
			
				|  |  | +  import 'video.js/dist/video-js.css'
 | 
	
		
			
				|  |  | +  // import html2canvas from "html2canvas";
 | 
	
		
			
				|  |  | +  import {
 | 
	
		
			
				|  |  | +    MessageBox
 | 
	
		
			
				|  |  | +  } from "element-ui";
 | 
	
		
			
				|  |  | +  export default {
 | 
	
		
			
				|  |  | +    name: "Index",
 | 
	
		
			
				|  |  | +    data() {
 | 
	
		
			
				|  |  | +      return {
 | 
	
		
			
				|  |  | +        timer: false,
 | 
	
		
			
				|  |  | +        required: 1,
 | 
	
		
			
				|  |  | +        tickNum: 0,
 | 
	
		
			
				|  |  | +        prevTime: 0,
 | 
	
		
			
				|  |  | +        isReady: false,
 | 
	
		
			
				|  |  | +        isnotbtn: false,
 | 
	
		
			
				|  |  | +        ontakebtn: false,
 | 
	
		
			
				|  |  | +        activeChapter: '',
 | 
	
		
			
				|  |  | +        activeName: '',
 | 
	
		
			
				|  |  | +        curTimes: 0,
 | 
	
		
			
				|  |  | +        errMsg: '',
 | 
	
		
			
				|  |  | +        errCount: 0,
 | 
	
		
			
				|  |  | +        onPlay: false
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +    },
 | 
	
		
			
				|  |  | +    components: {
 | 
	
		
			
				|  |  | +      videoPlayer,
 | 
	
		
			
				|  |  | +      IMessage,
 | 
	
		
			
				|  |  | +      ICollect
 | 
	
		
			
				|  |  | +    },
 | 
	
		
			
				|  |  | +    props: ['media', 'options', 'heartbeat', 'maxErrorCount', 'collectBeat', 'closeFace', 'chapter', 'list'],
 | 
	
		
			
				|  |  | +    filters: {
 | 
	
		
			
				|  |  | +      useTime(val) {
 | 
	
		
			
				|  |  | +        let timestr = ""
 | 
	
		
			
				|  |  | +        let hour = parseInt(val / 3600);
 | 
	
		
			
				|  |  | +        let min = parseInt(val / 60 % 60);
 | 
	
		
			
				|  |  | +        let sec = parseInt(val % 60);
 | 
	
		
			
				|  |  | +        if (hour < 10) hour = "0" + hour;
 | 
	
		
			
				|  |  | +        if (min < 10) min = "0" + min;
 | 
	
		
			
				|  |  | +        if (sec < 10) sec = "0" + sec;
 | 
	
		
			
				|  |  | +        return hour + ":" + min + ":" + sec
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +    },
 | 
	
		
			
				|  |  | +    beforeDestroy() {
 | 
	
		
			
				|  |  | +      this.stopTick()
 | 
	
		
			
				|  |  | +      this.reportErr("play", 'destroy');
 | 
	
		
			
				|  |  | +    },
 | 
	
		
			
				|  |  | +    directives: {
 | 
	
		
			
				|  |  | +      drag(el, bindings) {
 | 
	
		
			
				|  |  | +        el.onmousedown = function(e) {
 | 
	
		
			
				|  |  | +          var disx = e.pageX - el.offsetLeft
 | 
	
		
			
				|  |  | +          var disy = e.pageY - el.offsetTop
 | 
	
		
			
				|  |  | +          document.onmousemove = function(e) {
 | 
	
		
			
				|  |  | +            el.style.left = e.pageX - disx + 'px'
 | 
	
		
			
				|  |  | +            el.style.top = e.pageY - disy + 'px'
 | 
	
		
			
				|  |  | +          }
 | 
	
		
			
				|  |  | +          document.onmouseup = function() {
 | 
	
		
			
				|  |  | +            document.onmousemove = document.onmouseup = null
 | 
	
		
			
				|  |  | +          }
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +    },
 | 
	
		
			
				|  |  | +    computed: {
 | 
	
		
			
				|  |  | +      player() {
 | 
	
		
			
				|  |  | +        return this.$refs.videoPlayer.player
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +    },
 | 
	
		
			
				|  |  | +    beforeDestroy() {
 | 
	
		
			
				|  |  | +      this.destroyTimer()
 | 
	
		
			
				|  |  | +    },
 | 
	
		
			
				|  |  | +    created() {
 | 
	
		
			
				|  |  | +      this.startTick()
 | 
	
		
			
				|  |  | +      this.startMonitor();
 | 
	
		
			
				|  |  | +      this.activeChapter = this.media.chapterName;
 | 
	
		
			
				|  |  | +      this.activeName = this.media.name
 | 
	
		
			
				|  |  | +    },
 | 
	
		
			
				|  |  | +    watch: {
 | 
	
		
			
				|  |  | +      "media.name"() {
 | 
	
		
			
				|  |  | +        this.activeChapter = this.media.chapterName;
 | 
	
		
			
				|  |  | +        this.activeName = this.media.name
 | 
	
		
			
				|  |  | +        this.tickNum = 0
 | 
	
		
			
				|  |  | +        this.errMsg = ''
 | 
	
		
			
				|  |  | +        this.errCount = 0
 | 
	
		
			
				|  |  | +        this.setposition( this.media.position )
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +    },
 | 
	
		
			
				|  |  | +    methods: {
 | 
	
		
			
				|  |  | +      destroyTimer() {
 | 
	
		
			
				|  |  | +        if (this.timer) clearInterval(this.timer);
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      goState(item) {
 | 
	
		
			
				|  |  | +        if (item.name == this.activeChapter) {
 | 
	
		
			
				|  |  | +          this.activeChapter = ""
 | 
	
		
			
				|  |  | +        } else {
 | 
	
		
			
				|  |  | +          this.activeChapter = item.name;
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      goSubState(item, index) {
 | 
	
		
			
				|  |  | +        this.$emit('loadMedia', item, index)
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      startTick() {
 | 
	
		
			
				|  |  | +        let tick = this.tryTick;
 | 
	
		
			
				|  |  | +        this.destroyTimer();
 | 
	
		
			
				|  |  | +        this.tickNum = 0;
 | 
	
		
			
				|  |  | +        this.timer = setTimeout(tick, 1 * 1000);
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      stopTick() {
 | 
	
		
			
				|  |  | +        if (this.timer) clearInterval(this.timer);
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      tryTick() {
 | 
	
		
			
				|  |  | +        let that = this;
 | 
	
		
			
				|  |  | +        try {
 | 
	
		
			
				|  |  | +          that.tick()
 | 
	
		
			
				|  |  | +        } catch (err) {
 | 
	
		
			
				|  |  | +          that.reportErr("play", '' + err.message)
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        this.destroyTimer()
 | 
	
		
			
				|  |  | +        this.timer = setTimeout(this.tryTick, 1 * 1000);
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      playerReadied(audio) {
 | 
	
		
			
				|  |  | +        let that = this;
 | 
	
		
			
				|  |  | +        let {
 | 
	
		
			
				|  |  | +          position,
 | 
	
		
			
				|  |  | +          duration
 | 
	
		
			
				|  |  | +        } = this.media
 | 
	
		
			
				|  |  | +        if (position > 5 && position < duration) {
 | 
	
		
			
				|  |  | +          setTimeout(() => {
 | 
	
		
			
				|  |  | +            this.setposition(position)
 | 
	
		
			
				|  |  | +          }, 2000)
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        this.isReady = true
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      onPlayerTimeupdate(player) {
 | 
	
		
			
				|  |  | +        let curTimes = player.cache_.currentTime;
 | 
	
		
			
				|  |  | +        if (curTimes > 30 && curTimes > this.curTimes + 2) {
 | 
	
		
			
				|  |  | +          console.log("return", curTimes, this.media.position)
 | 
	
		
			
				|  |  | +          player.currentTime(this.curTimes);
 | 
	
		
			
				|  |  | +          return;
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        this.curTimes = curTimes
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      setposition(position) {
 | 
	
		
			
				|  |  | +        if (position > this.media.duration) position = this.media.duration;
 | 
	
		
			
				|  |  | +        let player = this.$refs.videoPlayer.player;
 | 
	
		
			
				|  |  | +        let res = player.currentTime(position);
 | 
	
		
			
				|  |  | +        console.log("setposition", position)
 | 
	
		
			
				|  |  | +        // player.play()
 | 
	
		
			
				|  |  | +        this.curTimes = position;
 | 
	
		
			
				|  |  | +        if (this.media.isFinish) return;
 | 
	
		
			
				|  |  | +        if (this.media.position >= this.media.duration - 2*this.heartbeat && !this.media.isFinish) {
 | 
	
		
			
				|  |  | +          this.tick(true)
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      onPlayerPause(event) {
 | 
	
		
			
				|  |  | +        this.reportErr("play", 'pause');
 | 
	
		
			
				|  |  | +        this.stopTick()
 | 
	
		
			
				|  |  | +        this.onPlay = false
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      onPlayerEnded(event) {
 | 
	
		
			
				|  |  | +        this.reportErr("play", 'end');
 | 
	
		
			
				|  |  | +        this.tick(true)
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      onClose() {
 | 
	
		
			
				|  |  | +        this.reportErr("play", 'close')
 | 
	
		
			
				|  |  | +        this.doPause()
 | 
	
		
			
				|  |  | +        this.$emit("close")
 | 
	
		
			
				|  |  | +        this.closeCamera()
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      doPause() {
 | 
	
		
			
				|  |  | +        this.stopTick()
 | 
	
		
			
				|  |  | +        this.onPlay = false
 | 
	
		
			
				|  |  | +        let myPlayer = this.$refs.videoPlayer.player;
 | 
	
		
			
				|  |  | +        myPlayer && myPlayer.pause()
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      doPlay() {
 | 
	
		
			
				|  |  | +        this.onPlay = true
 | 
	
		
			
				|  |  | +        this.startTick();
 | 
	
		
			
				|  |  | +        if (!this.$refs.videoPlayer || !this.$refs.videoPlayer.player) return;
 | 
	
		
			
				|  |  | +        // if (!this.dialog) return this.doPause();
 | 
	
		
			
				|  |  | +        let myPlayer = this.$refs.videoPlayer.player;
 | 
	
		
			
				|  |  | +        myPlayer && myPlayer.play()
 | 
	
		
			
				|  |  | +        this.tickNum = 0
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      onPlayerStart() {
 | 
	
		
			
				|  |  | +        console.log("onPlayerStart")
 | 
	
		
			
				|  |  | +        this.reportErr("play", 'start');
 | 
	
		
			
				|  |  | +        this.startTick();
 | 
	
		
			
				|  |  | +        this.onPlay = true
 | 
	
		
			
				|  |  | +        this.tickNum = 0
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      reportErr(action, msg) {
 | 
	
		
			
				|  |  | +        httpServer("course.report", {
 | 
	
		
			
				|  |  | +          action,
 | 
	
		
			
				|  |  | +          msg
 | 
	
		
			
				|  |  | +        })
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      startMonitor() {
 | 
	
		
			
				|  |  | +        let that = this
 | 
	
		
			
				|  |  | +        document.addEventListener("visibilitychange", function() {
 | 
	
		
			
				|  |  | +          // || document.hidden
 | 
	
		
			
				|  |  | +          if (document.visibilityState == "hidden") {
 | 
	
		
			
				|  |  | +            // that.doPause( )
 | 
	
		
			
				|  |  | +            that.reportErr("play", 'hidden');
 | 
	
		
			
				|  |  | +          } else {
 | 
	
		
			
				|  |  | +            that.reportErr("play", 'show');
 | 
	
		
			
				|  |  | +            // that.doPlay()
 | 
	
		
			
				|  |  | +          }
 | 
	
		
			
				|  |  | +        });
 | 
	
		
			
				|  |  | +      },
 | 
	
		
			
				|  |  | +      tick(force = false) {
 | 
	
		
			
				|  |  | +        let media = this.media;
 | 
	
		
			
				|  |  | +        this.tickNum++
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        // 已经完成
 | 
	
		
			
				|  |  | +        if (this.media.isFinish) {
 | 
	
		
			
				|  |  | +          console.log("finish")
 | 
	
		
			
				|  |  | +          return;
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        // 每5秒一次心跳
 | 
	
		
			
				|  |  | +        if (this.tickNum % this.heartbeat != 0) {
 | 
	
		
			
				|  |  | +          return;
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        if (!this.isnotbtn && !this.media.isFinish && this.onPlay && !this.closeFace) {
 | 
	
		
			
				|  |  | +          console.log(this.isnotbtn , this.media.isFinish , this.onPlay, this.closeFace)
 | 
	
		
			
				|  |  | +          this.$message.errorMsg("需要安装摄像头才能学习", 2);
 | 
	
		
			
				|  |  | +          this.doPause()
 | 
	
		
			
				|  |  | +          return;
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        if (this.errCount >= this.maxErrorCount) {
 | 
	
		
			
				|  |  | +          this.$message.errorMsg("人脸不在摄像头上", 5);
 | 
	
		
			
				|  |  | +          this.destroyTimer()
 | 
	
		
			
				|  |  | +          this.$emit("close");
 | 
	
		
			
				|  |  | +          return;
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        // 主动暂停
 | 
	
		
			
				|  |  | +        let myPlayer = this.$refs.videoPlayer.player;
 | 
	
		
			
				|  |  | +        let curTimes = parseInt(myPlayer.currentTime());
 | 
	
		
			
				|  |  | +        // 后退无心跳
 | 
	
		
			
				|  |  | +        if( !force ){
 | 
	
		
			
				|  |  | +          if (curTimes < this.media.position+this.heartbeat) {
 | 
	
		
			
				|  |  | +            return;
 | 
	
		
			
				|  |  | +          }
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        let isFinish = force ? 1 : 0
 | 
	
		
			
				|  |  | +        if (curTimes >= media.duration) isFinish = 1;
 | 
	
		
			
				|  |  | +        //  拉到后面
 | 
	
		
			
				|  |  | +        if (!isFinish) {
 | 
	
		
			
				|  |  | +          if (!this.onPlay) return;
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +        // 强制完成
 | 
	
		
			
				|  |  | +        let param = {
 | 
	
		
			
				|  |  | +          id: media.id,
 | 
	
		
			
				|  |  | +          position: curTimes,
 | 
	
		
			
				|  |  | +          isFinish
 | 
	
		
			
				|  |  | +        };
 | 
	
		
			
				|  |  | +        httpServer("course.tick", param, true).then(res => {
 | 
	
		
			
				|  |  | +          if (res.code == 200) {
 | 
	
		
			
				|  |  | +            let {
 | 
	
		
			
				|  |  | +              skip,
 | 
	
		
			
				|  |  | +              position,
 | 
	
		
			
				|  |  | +              pause,
 | 
	
		
			
				|  |  | +              closed
 | 
	
		
			
				|  |  | +            } = res.data
 | 
	
		
			
				|  |  | +            if (pause || closed) {
 | 
	
		
			
				|  |  | +              this.doPause();
 | 
	
		
			
				|  |  | +              this.$emit("close")
 | 
	
		
			
				|  |  | +              if (closed) {
 | 
	
		
			
				|  |  | +                this.$message.errorMsg("课程关闭学习", 5);
 | 
	
		
			
				|  |  | +              } else if (pause) {
 | 
	
		
			
				|  |  | +                this.$message.errorMsg("多处同时播放视频", 5);
 | 
	
		
			
				|  |  | +              }
 | 
	
		
			
				|  |  | +              return
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            if (!skip) {
 | 
	
		
			
				|  |  | +              setTimeout(() => {
 | 
	
		
			
				|  |  | +                this.setposition(position)
 | 
	
		
			
				|  |  | +              }, 2000);
 | 
	
		
			
				|  |  | +            };
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            Object.assign(param, res.data)
 | 
	
		
			
				|  |  | +            this.$emit("update", param)
 | 
	
		
			
				|  |  | +          }
 | 
	
		
			
				|  |  | +        })
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +</script>
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +<style>
 | 
	
		
			
				|  |  | +  @import url("./imedia.css");
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  .video-js {
 | 
	
		
			
				|  |  | +    .vjs-control-bar {
 | 
	
		
			
				|  |  | +      .vjs-icon-custombutton {
 | 
	
		
			
				|  |  | +        font-family: VideoJS;
 | 
	
		
			
				|  |  | +        font-weight: normal;
 | 
	
		
			
				|  |  | +        font-style: normal;
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +      .vjs-icon-custombutton:before {
 | 
	
		
			
				|  |  | +        content: "\f108";
 | 
	
		
			
				|  |  | +        font-size: 1.8em;
 | 
	
		
			
				|  |  | +        line-height: 1.67;
 | 
	
		
			
				|  |  | +      }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  .left-float {
 | 
	
		
			
				|  |  | +    width: 240px;
 | 
	
		
			
				|  |  | +    height: 180px;
 | 
	
		
			
				|  |  | +    background-color: #8bbdf5;
 | 
	
		
			
				|  |  | +    position: fixed;
 | 
	
		
			
				|  |  | +    transition: bottom ease .9s;
 | 
	
		
			
				|  |  | +    z-index: 0;
 | 
	
		
			
				|  |  | +    left: 60px;
 | 
	
		
			
				|  |  | +    top: 120px;
 | 
	
		
			
				|  |  | +    text-align: center;
 | 
	
		
			
				|  |  | +    border-radius: 5px;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  .p-process {
 | 
	
		
			
				|  |  | +    width: 100%;
 | 
	
		
			
				|  |  | +    margin: 20px auto;
 | 
	
		
			
				|  |  | +    height: 30px;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  .media-footer {
 | 
	
		
			
				|  |  | +    padding: 0px 30px;
 | 
	
		
			
				|  |  | +    text-align: left;
 | 
	
		
			
				|  |  | +    line-height: 40px !important;
 | 
	
		
			
				|  |  | +    bottom: -10px;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  .media-center {
 | 
	
		
			
				|  |  | +    text-align: center;
 | 
	
		
			
				|  |  | +    padding: 0px;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  .media-time {
 | 
	
		
			
				|  |  | +    font-size: 18px;
 | 
	
		
			
				|  |  | +    vertical-align: center;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  .media-select {
 | 
	
		
			
				|  |  | +    white-space: nowrap;
 | 
	
		
			
				|  |  | +    text-align: right;
 | 
	
		
			
				|  |  | +    line-height: 40px !important;
 | 
	
		
			
				|  |  | +    float: right;
 | 
	
		
			
				|  |  | +    margin: 0px !important;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  .bicon {
 | 
	
		
			
				|  |  | +    font-size: 28px !important;
 | 
	
		
			
				|  |  | +    padding: 4px !important;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +  .media-el-select {
 | 
	
		
			
				|  |  | +    font-size: 28px !important;
 | 
	
		
			
				|  |  | +    width: 80px;
 | 
	
		
			
				|  |  | +    padding: -4px auto !important;
 | 
	
		
			
				|  |  | +  }
 | 
	
		
			
				|  |  | +</style>
 |