video.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507
  1. <template>
  2. <div
  3. ref="custom-video_container"
  4. class="custom-video_container"
  5. @mouseover="handleControls($event, 'start')"
  6. @mouseleave="handleControls($event, 'end')"
  7. >
  8. <video
  9. ref="custom-video"
  10. :poster="videoOption.poster"
  11. class="custom-video_video"
  12. >
  13. <source :src="videoSrc" type="video/mp4">
  14. <p>设备不支持</p>
  15. </video>
  16. <span
  17. v-if="videoState.play"
  18. ref="videoPlay"
  19. class="custom-video_play custom-video_play-pause iconfont icon-zanting"
  20. @click="pause('btn')"
  21. />
  22. <span
  23. v-else
  24. class="custom-video_play custom-video_play-play iconfont icon-bofang"
  25. @click="play('btn')"
  26. />
  27. <!-- 控制区域背景 -->
  28. <transition
  29. name="fade"
  30. >
  31. <div
  32. v-show="!videoState.hideControl || !videoState.play"
  33. class="custom-video_control"
  34. >
  35. <!-- 进度条 -->
  36. <div
  37. class="custom-video_control-bg"
  38. @mousedown="handlePrograssDown"
  39. @mousemove="handlePrograssMove"
  40. @mouseup="handlePrograssUp"
  41. >
  42. <div
  43. ref="custom-video_control-bg-outside"
  44. class="custom-video_control-bg-outside"
  45. >
  46. <span
  47. ref="custom-video_control-bg-inside"
  48. class="custom-video_control-bg-inside"
  49. />
  50. <span
  51. ref="custom-video_control-bg-inside-point"
  52. class="custom-video_control-bg-inside-point"
  53. />
  54. </div>
  55. </div>
  56. <!-- 声音 -->
  57. <div
  58. class="custom-video_control-voice"
  59. >
  60. <span
  61. class="custom-video_control-voice-play iconfont icon-shengyin"
  62. />
  63. <div
  64. ref="custom-video_control-voice-bg"
  65. class="custom-video_control-voice-bg"
  66. @mousedown="handleVolPrograssDown"
  67. @mousemove="handleVolPrograssMove"
  68. @mouseup="handleVolPrograssUp"
  69. >
  70. <div
  71. ref="custom-video_control-voice-bg-outside"
  72. class="custom-video_control-voice-bg-outside"
  73. >
  74. <span
  75. ref="custom-video_control-voice-bg-inside"
  76. class="custom-video_control-voice-bg-inside"
  77. />
  78. <span
  79. ref="custom-video_control-voice-bg-point"
  80. class="custom-video_control-voice-bg-point"
  81. />
  82. </div>
  83. </div>
  84. </div>
  85. <!-- 时间 -->
  86. <div
  87. class="custom-video_control-time"
  88. >
  89. <span>{{ currentTime ? currentTime : "00:00" }}</span>
  90. /
  91. <span>{{ duration ? duration : "00:00" }}</span>
  92. </div>
  93. <!-- 全屏缩放 -->
  94. <span
  95. class="custom-video_control-full iconfont icon-quanping"
  96. @click="handleScreen"
  97. />
  98. </div>
  99. </transition>
  100. </div>
  101. </template>
  102. <script>
  103. export default {
  104. name: 'Video',
  105. props: {
  106. videoSrc: {
  107. type: '',
  108. default: ''
  109. },
  110. videoName: {
  111. type: String,
  112. default: ''
  113. }
  114. },
  115. data() {
  116. return {
  117. videoOption: {
  118. poster: '', // 初始化占位图片
  119. volume: 20 // 初始化声音
  120. },
  121. videoState: {
  122. play: false, // 播放状态
  123. hideControl: false, // 控制栏状态
  124. distance: 0, // 移动的距离
  125. downState: false, // 鼠标点击进度条
  126. playState: false,
  127. leftInit: 0, // 当前进度初始偏移量
  128. screenState: false
  129. },
  130. voiceState: { // 同上
  131. distance: 0,
  132. downState: false,
  133. topInit: 0
  134. },
  135. videoDom: null, // video
  136. videoProOut: null, // 视频总进度条
  137. videoPro: null, // 视频进度条
  138. videoPoi: null, // 视频进度点
  139. duration: 0, // 视频总时长
  140. currentTime: 0, // 视频当前播放时长
  141. processWidth: 0, // 视频进度条总长度
  142. voiceProOut: null, // 音频总进度条
  143. voicePro: null, // 音频进度条
  144. voicePoi: null, // 音频进度点
  145. volProcessHeight: 0
  146. }
  147. },
  148. mounted() {
  149. // 初始化相关元数据
  150. this.videoDom = this.$refs['custom-video']
  151. this.videoProOut = this.$refs['custom-video_control-bg-outside']
  152. this.videoPro = this.$refs['custom-video_control-bg-inside']
  153. this.videoPoi = this.$refs['custom-video_control-bg-inside-point']
  154. this.voiceProOut = this.$refs['custom-video_control-voice-bg-outside']
  155. this.voicePro = this.$refs['custom-video_control-voice-bg-inside']
  156. this.voicePoi = this.$refs['custom-video_control-voice-bg-point']
  157. this.processWidth = this.videoProOut.clientWidth
  158. this.videoState.leftInit = this.getOffset(this.videoProOut).left
  159. this.videoDom.volume = this.videoOption.volume / 100 // 设置初始化声音
  160. this.initMedaData()
  161. this.play('btn')
  162. },
  163. methods: {
  164. initMedaData() { // 初始化video相关事件
  165. this.videoDom.addEventListener('loadedmetadata', () => { // 获取视频总时长
  166. this.duration = this.timeTranslate(this.videoDom.duration)
  167. })
  168. this.videoDom.addEventListener('click', () => { // 点击视频区域可以进行播放或者暂停
  169. this.$emit('myFull', this.videoName)
  170. // 点击视频区域改为可以放大缩小
  171. // if (this.videoDom.paused || this.videoDom.ended) {
  172. // if (this.videoDom.ended) {
  173. // this.videoDom.currentTime = 0
  174. // }
  175. // this.play('btn')
  176. // } else {
  177. // this.pause('btn')
  178. // }
  179. })
  180. this.videoDom.addEventListener('timeupdate', () => { // 监听视频播放过程中的时间
  181. const percentage = 100 * this.videoDom.currentTime / this.videoDom.duration
  182. this.videoPro.style.width = percentage + '%'
  183. this.videoPoi.style.left = percentage - 1 + '%'
  184. this.currentTime = this.timeTranslate(this.videoDom.currentTime)
  185. })
  186. this.videoDom.addEventListener('ended', () => { // 监听结束播放事件
  187. this.videoPro.style.width = 0
  188. this.videoPoi.style.left = 0
  189. this.currentTime = 0
  190. this.videoState.play = false
  191. this.videoState.hideControl = false
  192. })
  193. this.videoDom.addEventListener('volumechange', () => {
  194. const percentage = this.videoDom.volume * 100
  195. this.voicePro.style.height = percentage + '%'
  196. this.voicePoi.style.bottom = percentage + '%'
  197. })
  198. },
  199. play(flag) { // 播放按钮事件
  200. if (flag) this.videoState.playState = true
  201. this.videoState.play = true
  202. console.log(this.videoDom)
  203. this.videoDom.play()
  204. },
  205. pause(flag) { // 暂停按钮事件
  206. if (flag) this.videoState.playState = false
  207. this.videoDom.pause()
  208. this.videoState.play = false
  209. },
  210. handlePrograssDown(ev) { // 监听点击进度条事件,方便获取初始点击的位置
  211. // 视频暂停
  212. this.videoState.downState = true // 按下鼠标标志
  213. this.pause()
  214. this.videoState.distance = ev.clientX - this.videoState.leftInit
  215. },
  216. handlePrograssMove(ev) { // 监听移动进度条事件,同步播放相关事件
  217. if (!this.videoState.downState) return
  218. let disX = ev.clientX - this.videoState.leftInit
  219. if (disX > this.processWidth) {
  220. disX = this.processWidth
  221. }
  222. if (disX < 0) {
  223. disX = 0
  224. }
  225. this.videoState.distance = disX
  226. this.videoDom.currentTime = this.videoState.distance / this.processWidth * this.videoDom.duration
  227. },
  228. handlePrograssUp() { // 松开鼠标,播放当前进度条视频
  229. this.videoState.downState = false
  230. // 视频播放
  231. this.videoDom.currentTime = this.videoState.distance / this.processWidth * this.videoDom.duration
  232. this.currentTime = this.timeTranslate(this.videoDom.currentTime)
  233. if (this.videoState.playState) {
  234. this.play()
  235. }
  236. },
  237. handleVolPrograssDown(ev) { // 监听声音点击事件
  238. this.voiceState.topInit = this.getOffset(this.voiceProOut).top
  239. this.volProcessHeight = this.voiceProOut.clientHeight
  240. this.voiceState.downState = true // 按下鼠标标志
  241. this.voiceState.distance = ev.clientY - this.voiceState.topInit
  242. },
  243. handleVolPrograssMove(ev) { // 监听声音进度条移动事件
  244. if (!this.voiceState.downState) return
  245. let disY = this.voiceState.topInit + this.volProcessHeight - ev.clientY
  246. if (disY > this.volProcessHeight - 2) {
  247. disY = this.volProcessHeight - 2
  248. }
  249. if (disY < 0) {
  250. disY = 0
  251. }
  252. this.voiceState.distance = disY
  253. this.videoDom.volume = this.voiceState.distance / this.volProcessHeight
  254. this.videoOption.volume = Math.round(this.videoDom.volume * 100)
  255. },
  256. handleVolPrograssUp() { // 监听声音鼠标离开事件
  257. this.voiceState.downState = false // 按下鼠标标志
  258. this.videoDom.volume = this.voiceState.distance / this.volProcessHeight
  259. this.videoOption.volume = Math.round(this.videoDom.volume * 100)
  260. },
  261. handleControls(ev, flag) { // 监听离开或者进入视频区域隐藏或者展示控制栏
  262. // let yanshi = null
  263. switch (flag) {
  264. case 'start':
  265. // clearTimeout(yanshi)
  266. this.videoState.hideControl = false
  267. break
  268. case 'end':
  269. setTimeout(() => {
  270. this.videoState.hideControl = true
  271. }, 3000)
  272. break
  273. default:
  274. break
  275. }
  276. },
  277. handleScreen() { // 全屏操作
  278. this.$emit('myFull', this.videoName)
  279. // 全屏事件暂时注释
  280. // this.videoState.screenState = !this.videoState.screenState
  281. // if (this.videoState.screenState) {
  282. // this.fullScreen()
  283. // } else {
  284. // this.exitFullscreen()
  285. // }
  286. },
  287. timeTranslate(t) { // 时间转化
  288. let m = Math.floor(t / 60)
  289. m < 10 && (m = '0' + m)
  290. return m + ':' + (t % 60 / 100).toFixed(2).slice(-2)
  291. },
  292. getOffset(node, offset) { // 获取当前屏幕下进度条的左偏移量和又偏移量
  293. if (!offset) {
  294. offset = {}
  295. offset.left = 0
  296. offset.top = 0
  297. }
  298. if (node === document.body || node === null) {
  299. return offset
  300. }
  301. offset.top += node.offsetTop
  302. offset.left += node.offsetLeft
  303. return this.getOffset(node.offsetParent, offset)
  304. },
  305. fullScreen() {
  306. const ele = document.documentElement
  307. if (ele.requestFullscreen) {
  308. ele.requestFullscreen()
  309. } else if (ele.mozRequestFullScreen) {
  310. ele.mozRequestFullScreen()
  311. } else if (ele.webkitRequestFullScreen) {
  312. ele.webkitRequestFullScreen()
  313. }
  314. this.$refs['custom-video_container'].style.width = '100%'
  315. this.$refs['custom-video_container'].style.height = '100%'
  316. },
  317. exitFullscreen() {
  318. const de = document
  319. if (de.exitFullscreen) {
  320. de.exitFullscreen()
  321. } else if (de.mozCancelFullScreen) {
  322. de.mozCancelFullScreen()
  323. } else if (de.webkitCancelFullScreen) {
  324. de.webkitCancelFullScreen()
  325. }
  326. this.$refs['custom-video_container'].style.width = '100%'
  327. this.$refs['custom-video_container'].style.height = '100%'
  328. }
  329. }
  330. }
  331. </script>
  332. <style scoped>
  333. /* 总容器 */
  334. .custom-video_container{
  335. width: 100%;
  336. height: 100%;
  337. position: relative;
  338. overflow: hidden;
  339. background: #000;
  340. }
  341. /* 视频标签 */
  342. .custom-video_video{
  343. width: 100%;
  344. height: 100%;
  345. }
  346. /* 暂停 或者 播放 */
  347. .custom-video_play{
  348. display: inline-block;
  349. position: absolute;
  350. right: 50px;
  351. bottom: 50px;
  352. width: 30px;
  353. height: 30px;
  354. border-radius: 50%;
  355. font-size: 30px;
  356. color: cornflowerblue;
  357. }
  358. /* 暂停隐藏 */
  359. .custom-video_play-pause{
  360. display: none;
  361. }
  362. /* hover 显示 */
  363. .custom-video_container:hover > .custom-video_play-pause{
  364. display: inline-block;
  365. }
  366. /* hover 播放按钮动画 */
  367. .custom-video_play:hover{
  368. box-shadow: 0 0 10px #5A4180;
  369. transition: all 0.4s;
  370. }
  371. /* 控制栏 */
  372. .custom-video_control{
  373. position: absolute;
  374. width: 100%;
  375. height: 50px;
  376. left: 0;
  377. bottom: 0;
  378. background-color: rgba(0, 0, 0, .55);
  379. display: flex;
  380. flex-direction: row;
  381. align-items: center;
  382. }
  383. /* 控制栏进度条 */
  384. .custom-video_control-bg{
  385. flex: 1;
  386. height: 100%;
  387. display: flex;
  388. justify-content: center;
  389. align-items: center;
  390. margin: 0 10px;
  391. }
  392. /* 控制栏进度条 —— 总长度 */
  393. .custom-video_control-bg-outside{
  394. width: 100%;
  395. height: 5px;
  396. border-radius: 2.5px;
  397. background-color: #aaa;
  398. position: relative;
  399. cursor: pointer;
  400. }
  401. /* 控制栏进度条 —— 播放长度 */
  402. .custom-video_control-bg-inside{
  403. position: absolute;
  404. display: inline-block;
  405. width: 0;
  406. height: 100%;
  407. border-radius: 2.5px;
  408. left: 0;
  409. top: 0;
  410. background-color: #fff;
  411. transition: all 0.2s;
  412. }
  413. /* 控制栏进度条 —— 播放点 */
  414. .custom-video_control-bg-inside-point{
  415. display: inline-block;
  416. width: 10px;
  417. height: 10px;
  418. background-color: #fff;
  419. border-radius: 50%;
  420. position: absolute;
  421. top: -2.5px;
  422. left: -1%;
  423. transition: all 0.2s;
  424. }
  425. /* 控制栏 —— 声音、时间、全屏缩放 */
  426. .custom-video_control-voice,
  427. .custom-video_control-time,
  428. .custom-video_control-full{
  429. display: flex;
  430. flex-direction: row;
  431. justify-content: center;
  432. align-items: center;
  433. color: #fff;
  434. position: relative;
  435. }
  436. .custom-video_control-voice:hover > .custom-video_control-voice-bg{
  437. display: block;
  438. }
  439. .custom-video_control-voice-play{
  440. z-index: 10;
  441. }
  442. .custom-video_control-voice-bg{
  443. display: none;
  444. position: absolute;
  445. width: 30px;
  446. height: 100px;
  447. background-color: rgba(0, 0, 0, .55);
  448. left: 0;
  449. bottom: 0px;
  450. border-radius: 15px;
  451. }
  452. .custom-video_control-voice-bg-outside{
  453. width: 5px;
  454. height: 70px;
  455. border-radius: 2.5px;
  456. background-color: #aaa;
  457. position: absolute;
  458. left: 50%;
  459. transform: translate3d(-50%, 10%, 0);
  460. cursor: pointer;
  461. }
  462. .custom-video_control-voice-bg-inside{
  463. display: inline-block;
  464. position: absolute;
  465. width: 100%;
  466. bottom: 0;
  467. left: 0;
  468. border-radius: 2.5px;
  469. background-color: #fff;
  470. height: 0;
  471. }
  472. .custom-video_control-voice-bg-point{
  473. display: inline-block;
  474. width: 10px;
  475. height: 10px;
  476. background-color: #fff;
  477. border-radius: 50%;
  478. position: absolute;
  479. left: -2.5px;
  480. bottom: -1px;
  481. }
  482. .custom-video_control-time{
  483. font-size: 12px;
  484. }
  485. .custom-video_control-full{
  486. font-size: 14px;
  487. }
  488. .custom-video_control-voice,
  489. .custom-video_control-full{
  490. width: 30px;
  491. height: 30px;
  492. cursor: pointer;
  493. }
  494. /* 控制栏隐藏动画 */
  495. .fade-enter-active {
  496. transition: all .3s ease;
  497. }
  498. .fade-leave-active {
  499. transition: all .8s cubic-bezier(1.0, 0.5, 0.8, 1.0);
  500. }
  501. .fade-enter, .fade-leave-to {
  502. transform: translateY(50px);
  503. opacity: 0;
  504. }
  505. </style>