UNPKG

@tarojs/components

Version:
501 lines (496 loc) 34 kB
import { proxyCustomElement, HTMLElement, createEvent, h, Host } from '@stencil/core/internal/client'; import Taro from '@tarojs/taro'; import { c as classnames } from './index2.js'; import { t as throttle } from './index3.js'; import { n as normalizeNumber, i as isHls, s as screenFn, a as scene, f as formatTime, d as defineCustomElement$3, c as calcDist } from './video-control.js'; import { d as defineCustomElement$2 } from './video-danmu.js'; const indexCss = ".taro-video{width:100%;height:225px;line-height:0;display:inline-block;position:relative;overflow:hidden}.taro-video[hidden]{display:none}.taro-video-container{-o-object-position:inherit;object-position:inherit;background-color:#000;width:100%;height:100%;display:inline-block;position:absolute;top:0;left:0}.taro-video-container.taro-video-type-fullscreen{z-index:999;position:fixed;inset:0}.taro-video-container.taro-video-type-fullscreen.taro-video-type-rotate-left{-webkit-transform:translate(-50%,-50%)rotate(-90deg);transform:translate(-50%,-50%)rotate(-90deg)}.taro-video-container.taro-video-type-fullscreen.taro-video-type-rotate-right{-webkit-transform:translate(-50%,-50%)rotate(90deg);transform:translate(-50%,-50%)rotate(90deg)}.taro-video-video{-o-object-position:inherit;object-position:inherit;width:100%;height:100%;display:block}.taro-video-cover{z-index:1;background-color:rgba(1,1,1,.5);-webkit-box-orient:vertical;-webkit-box-direction:normal;-ms-flex-direction:column;flex-direction:column;-webkit-box-align:center;-webkit-box-pack:center;-ms-flex-pack:center;justify-content:center;-ms-flex-align:center;align-items:center;width:100%;display:-ms-flexbox;display:flex;position:absolute;top:0;bottom:0;left:0}.taro-video-cover-play-button{background-position:50%;background-repeat:no-repeat;background-size:50%;width:40px;height:40px}.taro-video-cover-duration{color:#fff;margin-top:10px;font-size:16px;line-height:1}.taro-video-bar{visibility:hidden;z-index:1;background-color:rgba(0,0,0,.5);-webkit-box-align:center;-ms-flex-align:center;align-items:center;height:44px;padding:0 10px;display:-ms-flexbox;display:flex;position:absolute;bottom:0;right:0;overflow:hidden}.taro-video-bar.taro-video-bar-full{left:0}.taro-video-controls{-webkit-box-flex:1;-ms-flex-positive:1;flex-grow:1;margin:0 8.5px;display:-ms-flexbox;display:flex}.taro-video-control-button{-webkit-box-sizing:content-box;box-sizing:content-box;width:13px;height:15px;margin-left:-8.5px;padding:14.5px 12.5px}.taro-video-control-button:after{content:\"\";background-position:50%;background-repeat:no-repeat;background-size:100%;width:100%;height:100%;display:block}.taro-video-control-button.taro-video-control-button-play:after,.taro-video-cover-play-button{background-image:url()}.taro-video-control-button.taro-video-control-button-pause:after{background-image:url()}.taro-video-current-time,.taro-video-duration{color:#cbcbcb;height:14.5px;margin-top:15px;margin-bottom:14.5px;font-size:12px;line-height:14.5px}.taro-video-progress-container{-webkit-box-flex:2;-ms-flex-positive:2;flex-grow:2;position:relative}.taro-video-progress{background-color:rgba(255,255,255,.4);height:2px;margin:21px 12px;position:relative}.taro-video-progress-buffered{background-color:rgba(255,255,255,.8);width:0;height:100%;-webkit-transition:width .1s;transition:width .1s;position:absolute;top:0;left:0}.taro-video-ball{-webkit-box-sizing:content-box;box-sizing:content-box;width:16px;height:16px;margin-left:-22px;padding:14px;position:absolute;top:-21px;left:0}.taro-video-inner{background-color:#fff;border-radius:50%;width:100%;height:100%}.taro-video-danmu-button{color:#fff;white-space:nowrap;border:1px solid #fff;border-radius:5px;margin:0 8.5px;padding:2px 10px;font-size:13px;line-height:1}.taro-video-danmu-button.taro-video-danmu-button-active{color:#48c23d;border-color:#48c23d}.taro-video-fullscreen,.taro-video-mute{-webkit-box-sizing:content-box;box-sizing:content-box;background-position:50%;background-repeat:no-repeat;background-size:50%;width:17px;height:17px;padding:8.5px}.taro-video-fullscreen{background-image:url()}.taro-video-fullscreen.taro-video-type-fullscreen{background-image:url()}.taro-video-mute{background-image:url()}.taro-video-mute.taro-video-type-mute{background-image:url()}.taro-video-danmu{width:100%;margin-top:14px;margin-bottom:44px;font-size:14px;line-height:14px;position:absolute;top:0;bottom:0;left:0;overflow:visible}.taro-video-danmu-item{color:#fff;white-space:nowrap;line-height:1;-webkit-transition-property:left,-webkit-transform;transition-property:left,-webkit-transform;transition-property:left,transform;transition-property:left,transform,-webkit-transform;-webkit-transition-duration:3s;transition-duration:3s;-webkit-transition-timing-function:linear;transition-timing-function:linear;position:absolute;left:100%;-webkit-transform:translate(0);transform:translate(0)}.taro-video-toast{visibility:hidden;pointer-events:none;color:#000;background-color:rgba(255,255,255,.8);border-radius:5px;display:block;position:absolute;top:50%;left:50%;-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%)}.taro-video-toast.taro-video-toast-volume{width:100px;height:100px;display:block}.taro-video-toast-volume .taro-video-toast-title{text-align:center;width:100%;margin-top:10px;font-size:12px;line-height:16px;display:block}.taro-video-toast-volume .taro-video-toast-icon{fill:#000;background-image:url();background-position:50%;background-repeat:no-repeat;background-size:50%;width:50%;height:50%;margin-left:25%;display:block}.taro-video-toast-volume .taro-video-toast-value{width:80px;height:5px;margin-top:5px;margin-left:10px}.taro-video-toast-volume .taro-video-toast-value>.taro-video-toast-value-content{overflow:hidden}.taro-video-toast-volume-grids{width:80px;height:5px}.taro-video-toast-volume-grids-item{float:left;background-color:#000;width:7.1px;height:5px}.taro-video-toast-volume-grids-item:not(:first-child){margin-left:1px}.taro-video-toast.taro-video-toast-progress{color:#fff;background-color:rgba(0,0,0,.8);padding:6px;font-size:14px;line-height:18px}"; const Video = /*@__PURE__*/ proxyCustomElement(class extends HTMLElement { constructor() { super(); this.__registerHost(); this.onPlay = createEvent(this, "play", 7); this.onPause = createEvent(this, "pause", 7); this.onEnded = createEvent(this, "ended", 7); this.onTimeUpdate = createEvent(this, "timeupdate", 7); this.onError = createEvent(this, "error", 7); this.onFullScreenChange = createEvent(this, "fullscreenchange", 7); this.onProgress = createEvent(this, "progress", 7); this.onLoadedMetaData = createEvent(this, "loadedmetadata", 7); this.currentTime = 0; this.isDraggingProgress = false; this.gestureType = 'none'; this.analyzeGesture = (e) => { var _a; const obj = { type: 'none' }; const nowX = e.touches[0].screenX; const nowY = e.touches[0].screenY; const distX = nowX - this.lastTouchScreenX; const distY = nowY - this.lastTouchScreenY; const enableVslideGesture = this.isFullScreen ? this.vslideGestureInFullscreen : this.vslideGesture; if (this.gestureType === 'none') { // 两点间距离 const dist = calcDist(distX, distY); // 没有移动 if (dist < 10) return obj; if (Math.abs(distY) >= Math.abs(distX)) { // 垂直方向移动:调整音量 if (enableVslideGesture) { this.gestureType = 'adjustVolume'; this.lastVolume = this.videoRef.volume; } else { return obj; } } else if (Math.abs(distY) < Math.abs(distX)) { // 水平方向移动:调整进度 if (this.enableProgressGesture) { this.gestureType = 'adjustProgress'; this.lastPercentage = this.currentTime / ((_a = this.duration) !== null && _a !== void 0 ? _a : this._duration); } else { return obj; } } } obj.type = this.gestureType; obj.dataX = normalizeNumber(distX / 200); obj.dataY = normalizeNumber(distY / 200); return obj; }; this.loadNativePlayer = () => { var _a, _b; if (this.videoRef) { this.videoRef.src = this.src; (_b = (_a = this.videoRef).load) === null || _b === void 0 ? void 0 : _b.call(_a); } }; this.init = () => { const { src, videoRef } = this; if (isHls(src)) { import( /* webpackExports: ["default"] */ 'hls.js').then(e => { const Hls = e.default; this.HLS = Hls; if (Hls.isSupported()) { if (this.hls) { this.hls.destroy(); } this.hls = new Hls(); this.hls.loadSource(src); this.hls.attachMedia(videoRef); this.hls.on(Hls.Events.MANIFEST_PARSED, () => { this.autoplay && this.play(); }); this.hls.on(Hls.Events.ERROR, (_, data) => { this.handleError(data); }); } else if (videoRef.canPlayType('application/vnd.apple.mpegurl')) { this.loadNativePlayer(); } else { console.error('该浏览器不支持 HLS 播放'); } }); } else { this.loadNativePlayer(); } }; this.handlePlay = () => { this.isPlaying = true; this.isFirst = false; this.controlsRef.toggleVisibility(true); this.onPlay.emit(); }; this.handlePause = () => { this.isPlaying = false; this.controlsRef.toggleVisibility(true); this.onPause.emit(); }; this.handleEnded = () => { this.isFirst = true; this.pause(); this.controlsRef.toggleVisibility(); this.onEnded.emit(); }; this.handleTimeUpdate = throttle(async (e) => { var _a, _b; this.currentTime = this.videoRef.currentTime; const duration = this.duration || this._duration; const isControlDragging = await this.controlsRef.getIsDraggingProgressBall(); if (this.controls && this.showProgress) { if (!isControlDragging && !this.isDraggingProgress) { this.controlsRef.setProgressBall(this.currentTime / duration); this.controlsRef.setCurrentTime(this.currentTime); } } this.danmuRef.tick(this.currentTime); this.onTimeUpdate.emit({ duration: (_a = e.target) === null || _a === void 0 ? void 0 : _a.duration, currentTime: (_b = e.target) === null || _b === void 0 ? void 0 : _b.currentTime }); if (this.duration) { if (this.currentTime >= this.duration) { this.seek(0); this.handleEnded(); } } }, 250); this.handleError = e => { var _a, _b; if (this.hls) { switch (e.type) { case this.HLS.ErrorTypes.NETWORK_ERROR: // try to recover network error this.onError.emit({ errMsg: e.response }); this.hls.startLoad(); break; case this.HLS.ErrorTypes.MEDIA_ERROR: this.onError.emit({ errMsg: e.reason || '媒体错误,请重试' }); this.hls.recoverMediaError(); break; } } else { this.onError.emit({ errMsg: (_b = (_a = e.target) === null || _a === void 0 ? void 0 : _a.error) === null || _b === void 0 ? void 0 : _b.message, }); } }; this.handleDurationChange = () => { this._duration = this.videoRef.duration; }; this.handleProgress = () => { this.onProgress.emit(); }; this.handleLoadedMetaData = (e) => { const target = e.target; this.onLoadedMetaData.emit({ width: target.videoWidth, height: target.videoHeight, duration: target.duration }); }; this._play = () => this.videoRef.play(); this._pause = () => this.videoRef.pause(); this._stop = () => { this.videoRef.pause(); this._seek(0); }; this._seek = (position) => { this.videoRef.currentTime = position; }; this.onTouchStartContainer = (e) => { this.lastTouchScreenX = e.touches[0].screenX; this.lastTouchScreenY = e.touches[0].screenY; }; this.onClickContainer = () => { if (this.enablePlayGesture) { const now = Date.now(); if (now - this.lastClickedTime < 300) { // 双击 this.isPlaying ? this.pause() : this.play(); } this.lastClickedTime = now; } this.controlsRef.toggleVisibility(); }; this.onClickFullScreenBtn = (e) => { e.stopPropagation(); this.toggleFullScreen(); }; this.handleFullScreenChange = e => { // 全屏后,"退出"走的是浏览器事件,在此同步状态 const timestamp = new Date().getTime(); if (!e.detail && this.isFullScreen && !document[screenFn.fullscreenElement] && timestamp - this.fullScreenTimestamp > 100) { this.toggleFullScreen(false, true); } }; this.toggleFullScreen = (isFullScreen = !this.isFullScreen, fromBrowser = false) => { this.isFullScreen = isFullScreen; // this.videoRef?.['webkitDisplayingFullscreen'] this.controlsRef.toggleVisibility(true); this.fullScreenTimestamp = new Date().getTime(); this.onFullScreenChange.emit({ fullScreen: this.isFullScreen, direction: 'vertical' }); if (this.isFullScreen && !document[screenFn.fullscreenElement]) { setTimeout(() => { this.videoRef[screenFn.requestFullscreen]({ navigationUI: 'auto' }); Taro.eventCenter.trigger('__taroEnterFullScreen', {}); }, 0); } else { if (!fromBrowser) { // Note: 全屏后,"退出全屏"是浏览器按钮是浏览器内部按钮,非html按钮,点击"退出全屏"按钮是浏览器内部实现功能。此时再次调用exitFullscreen反而会报错,因此不再调用 document[screenFn.exitFullscreen](); } Taro.eventCenter.trigger('__taroExitFullScreen', {}); } }; this.toggleMute = (e) => { e.stopPropagation(); this.videoRef.muted = !this.isMute; this.controlsRef.toggleVisibility(true); this.isMute = !this.isMute; }; this.toggleDanmu = (e) => { e.stopPropagation(); this.controlsRef.toggleVisibility(true); this._enableDanmu = !this._enableDanmu; }; this.src = undefined; this.duration = undefined; this.controls = true; this.autoplay = false; this.loop = false; this.muted = false; this.initialTime = 0; this.poster = undefined; this.objectFit = 'contain'; this.showProgress = true; this.showFullscreenBtn = true; this.showPlayBtn = true; this.showCenterPlayBtn = true; this.showMuteBtn = false; this.danmuList = undefined; this.danmuBtn = false; this.enableDanmu = false; this.enablePlayGesture = false; this.enableProgressGesture = true; this.vslideGesture = false; this.vslideGestureInFullscreen = true; this.nativeProps = {}; this._duration = undefined; this._enableDanmu = false; this.isPlaying = false; this.isFirst = true; this.isFullScreen = false; this.fullScreenTimestamp = new Date().getTime(); this.isMute = false; } componentWillLoad() { this._enableDanmu = this.enableDanmu; this.isMute = this.muted; } componentDidLoad() { var _a, _b; this.init(); if (this.initialTime) { this.videoRef.currentTime = this.initialTime; } // 目前只支持 danmuList 初始化弹幕列表,还未支持更新弹幕列表 (_b = (_a = this.danmuRef).sendDanmu) === null || _b === void 0 ? void 0 : _b.call(_a, this.danmuList); if (document.addEventListener) { document.addEventListener(screenFn.fullscreenchange, this.handleFullScreenChange); } if (this.videoRef && scene === 'iOS') { // NOTE: iOS 场景下 fullscreenchange 并不会在退出全屏状态下触发,仅 webkitpresentationmodechanged 与 webkitendfullscreen 可替代 this.videoRef.addEventListener('webkitendfullscreen', this.handleFullScreenChange); } } componentDidRender() { } disconnectedCallback() { if (document.removeEventListener) { document.removeEventListener(screenFn.fullscreenchange, this.handleFullScreenChange); } if (this.videoRef && scene === 'iOS') { this.videoRef.removeEventListener('webkitendfullscreen', this.handleFullScreenChange); } } watchEnableDanmu(newVal) { this._enableDanmu = newVal; } watchSrc() { this.init(); } async onDocumentTouchMove(e) { if (this.lastTouchScreenX === undefined || this.lastTouchScreenY === undefined) return; if (await this.controlsRef.getIsDraggingProgressBall()) return; const gestureObj = this.analyzeGesture(e); if (gestureObj.type === 'adjustVolume') { this.toastVolumeRef.style.visibility = 'visible'; const nextVolume = Math.max(Math.min(this.lastVolume - gestureObj.dataY, 1), 0); this.videoRef.volume = nextVolume; this.toastVolumeBarRef.style.width = `${nextVolume * 100}%`; } else if (gestureObj.type === 'adjustProgress') { this.isDraggingProgress = true; this.nextPercentage = Math.max(Math.min(this.lastPercentage + (gestureObj.dataX || 0), 1), 0); if (this.controls && this.showProgress) { this.controlsRef.setProgressBall(this.nextPercentage); this.controlsRef.toggleVisibility(true); } const duration = this.duration || this._duration; this.toastProgressTitleRef.innerHTML = `${formatTime(this.nextPercentage * duration)} / ${formatTime(duration)}`; this.toastProgressRef.style.visibility = 'visible'; } } onDocumentTouchEnd() { var _a; if (this.gestureType === 'adjustVolume') { this.toastVolumeRef.style.visibility = 'hidden'; } else if (this.gestureType === 'adjustProgress') { this.toastProgressRef.style.visibility = 'hidden'; } if (this.isDraggingProgress) { this.isDraggingProgress = false; this.seek(this.nextPercentage * ((_a = this.duration) !== null && _a !== void 0 ? _a : this._duration)); } this.gestureType = 'none'; this.lastTouchScreenX = undefined; this.lastTouchScreenY = undefined; } async getHlsObject() { // Note: H5 端专属方法,获取 HLS 实例 fix #11894 return this.hls; } /** 播放视频 */ async play() { this._play(); } /** 暂停视频 */ async pause() { this._pause(); } /** 停止视频 */ async stop() { this._stop(); } /** 跳转到指定位置 */ async seek(position) { this._seek(position); } /** 进入全屏。若有自定义内容需在全屏时展示,需将内容节点放置到 video 节点内。 */ async requestFullScreen() { this.toggleFullScreen(true); } /** 退出全屏 */ async exitFullScreen() { this.toggleFullScreen(false); } render() { const { controls, autoplay, loop, muted, poster, objectFit, isFirst, isMute, isFullScreen, showCenterPlayBtn, isPlaying, _enableDanmu, showMuteBtn, danmuBtn, showFullscreenBtn, nativeProps } = this; const duration = this.duration || this._duration; const durationTime = formatTime(duration); return (h(Host, { class: classnames('taro-video-container', { 'taro-video-type-fullscreen': isFullScreen }), onTouchStart: this.onTouchStartContainer, onClick: this.onClickContainer }, h("video", Object.assign({ class: 'taro-video-video', style: { 'object-fit': objectFit }, ref: dom => { if (dom) { this.videoRef = dom; } }, autoplay: autoplay, loop: loop, muted: muted, poster: controls ? poster : undefined, playsinline: true, "webkit-playsinline": true, onPlay: this.handlePlay, onPause: this.handlePause, onEnded: this.handleEnded, onTimeUpdate: this.handleTimeUpdate, onError: this.handleError, onDurationChange: this.handleDurationChange, onProgress: this.handleProgress, onLoadedMetaData: this.handleLoadedMetaData }, nativeProps), "\u6682\u65F6\u4E0D\u652F\u6301\u64AD\u653E\u8BE5\u89C6\u9891"), h("taro-video-danmu", { ref: dom => { if (dom) { this.danmuRef = dom; } }, enable: _enableDanmu }), isFirst && showCenterPlayBtn && !isPlaying && (h("div", { class: 'taro-video-cover' }, h("div", { class: 'taro-video-cover-play-button', onClick: () => this.play() }), h("p", { class: 'taro-video-cover-duration' }, durationTime))), h("taro-video-control", { ref: dom => { if (dom) { this.controlsRef = dom; } }, controls: controls, currentTime: this.currentTime, duration: duration, isPlaying: this.isPlaying, pauseFunc: this._pause, playFunc: this._play, seekFunc: this._seek, showPlayBtn: this.showPlayBtn, showProgress: this.showProgress }, showMuteBtn && (h("div", { class: classnames('taro-video-mute', { 'taro-video-type-mute': isMute }), onClick: this.toggleMute })), danmuBtn && (h("div", { class: classnames('taro-video-danmu-button', { 'taro-video-danmu-button-active': _enableDanmu }), onClick: this.toggleDanmu }, "\u5F39\u5E55")), showFullscreenBtn && (h("div", { class: classnames('taro-video-fullscreen', { 'taro-video-type-fullscreen': isFullScreen }), onClick: this.onClickFullScreenBtn }))), h("div", { class: 'taro-video-toast taro-video-toast-volume', ref: dom => { if (dom) { this.toastVolumeRef = dom; } } }, h("div", { class: 'taro-video-toast-title' }, "\u97F3\u91CF"), h("div", { class: 'taro-video-toast-icon' }), h("div", { class: 'taro-video-toast-value' }, h("div", { class: 'taro-video-toast-value-content', ref: dom => { if (dom) { this.toastVolumeBarRef = dom; } } }, h("div", { class: 'taro-video-toast-volume-grids' }, Array(10).fill(1).map(() => (h("div", { class: 'taro-video-toast-volume-grids-item' }))))))), h("div", { class: 'taro-video-toast taro-video-toast-progress', ref: dom => { if (dom) { this.toastProgressRef = dom; } } }, h("div", { class: 'taro-video-toast-title', ref: dom => { if (dom) { this.toastProgressTitleRef = dom; } } })))); } get el() { return this; } static get watchers() { return { "enableDanmu": ["watchEnableDanmu"], "src": ["watchSrc"] }; } static get style() { return indexCss; } }, [0, "taro-video-core", { "src": [1], "duration": [2], "controls": [4], "autoplay": [4], "loop": [4], "muted": [4], "initialTime": [2, "initial-time"], "poster": [1], "objectFit": [1, "object-fit"], "showProgress": [4, "show-progress"], "showFullscreenBtn": [4, "show-fullscreen-btn"], "showPlayBtn": [4, "show-play-btn"], "showCenterPlayBtn": [4, "show-center-play-btn"], "showMuteBtn": [4, "show-mute-btn"], "danmuList": [16], "danmuBtn": [4, "danmu-btn"], "enableDanmu": [4, "enable-danmu"], "enablePlayGesture": [4, "enable-play-gesture"], "enableProgressGesture": [4, "enable-progress-gesture"], "vslideGesture": [4, "vslide-gesture"], "vslideGestureInFullscreen": [4, "vslide-gesture-in-fullscreen"], "nativeProps": [16], "_duration": [32], "_enableDanmu": [32], "isPlaying": [32], "isFirst": [32], "isFullScreen": [32], "fullScreenTimestamp": [32], "isMute": [32], "getHlsObject": [64], "play": [64], "pause": [64], "stop": [64], "seek": [64], "requestFullScreen": [64], "exitFullScreen": [64] }, [[5, "touchmove", "onDocumentTouchMove"], [5, "touchend", "onDocumentTouchEnd"], [5, "touchcancel", "onDocumentTouchEnd"]]]); function defineCustomElement$1() { if (typeof customElements === "undefined") { return; } const components = ["taro-video-core", "taro-video-control", "taro-video-danmu"]; components.forEach(tagName => { switch (tagName) { case "taro-video-core": if (!customElements.get(tagName)) { customElements.define(tagName, Video); } break; case "taro-video-control": if (!customElements.get(tagName)) { defineCustomElement$3(); } break; case "taro-video-danmu": if (!customElements.get(tagName)) { defineCustomElement$2(); } break; } }); } const TaroVideoCore = Video; const defineCustomElement = defineCustomElement$1; export { TaroVideoCore, defineCustomElement };