@ezuikit/player-theme
Version:
player theme
996 lines (937 loc) • 708 kB
JavaScript
/*
* @ezuikit/player-theme v1.0.1-alpha.2
* Copyright (c) 2025-11-14 Ezviz-OpenBiz
* Released under the MIT License.
*/
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
typeof define === 'function' && define.amd ? define(factory) :
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.Theme = factory());
})(this, (function () { 'use strict';
function getDefaultExportFromCjs (x) {
return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x;
}
var eventemitter3 = {exports: {}};
var hasRequiredEventemitter3;
function requireEventemitter3 () {
if (hasRequiredEventemitter3) return eventemitter3.exports;
hasRequiredEventemitter3 = 1;
(function (module) {
var has = Object.prototype.hasOwnProperty
, prefix = '~';
/**
* Constructor to create a storage for our `EE` objects.
* An `Events` instance is a plain object whose properties are event names.
*
* @constructor
* @private
*/
function Events() {}
//
// We try to not inherit from `Object.prototype`. In some engines creating an
// instance in this way is faster than calling `Object.create(null)` directly.
// If `Object.create(null)` is not supported we prefix the event names with a
// character to make sure that the built-in object properties are not
// overridden or used as an attack vector.
//
if (Object.create) {
Events.prototype = Object.create(null);
//
// This hack is needed because the `__proto__` property is still inherited in
// some old browsers like Android 4, iPhone 5.1, Opera 11 and Safari 5.
//
if (!new Events().__proto__) prefix = false;
}
/**
* Representation of a single event listener.
*
* @param {Function} fn The listener function.
* @param {*} context The context to invoke the listener with.
* @param {Boolean} [once=false] Specify if the listener is a one-time listener.
* @constructor
* @private
*/
function EE(fn, context, once) {
this.fn = fn;
this.context = context;
this.once = once || false;
}
/**
* Add a listener for a given event.
*
* @param {EventEmitter} emitter Reference to the `EventEmitter` instance.
* @param {(String|Symbol)} event The event name.
* @param {Function} fn The listener function.
* @param {*} context The context to invoke the listener with.
* @param {Boolean} once Specify if the listener is a one-time listener.
* @returns {EventEmitter}
* @private
*/
function addListener(emitter, event, fn, context, once) {
if (typeof fn !== 'function') {
throw new TypeError('The listener must be a function');
}
var listener = new EE(fn, context || emitter, once)
, evt = prefix ? prefix + event : event;
if (!emitter._events[evt]) emitter._events[evt] = listener, emitter._eventsCount++;
else if (!emitter._events[evt].fn) emitter._events[evt].push(listener);
else emitter._events[evt] = [emitter._events[evt], listener];
return emitter;
}
/**
* Clear event by name.
*
* @param {EventEmitter} emitter Reference to the `EventEmitter` instance.
* @param {(String|Symbol)} evt The Event name.
* @private
*/
function clearEvent(emitter, evt) {
if (--emitter._eventsCount === 0) emitter._events = new Events();
else delete emitter._events[evt];
}
/**
* Minimal `EventEmitter` interface that is molded against the Node.js
* `EventEmitter` interface.
*
* @constructor
* @public
*/
function EventEmitter() {
this._events = new Events();
this._eventsCount = 0;
}
/**
* Return an array listing the events for which the emitter has registered
* listeners.
*
* @returns {Array}
* @public
*/
EventEmitter.prototype.eventNames = function eventNames() {
var names = []
, events
, name;
if (this._eventsCount === 0) return names;
for (name in (events = this._events)) {
if (has.call(events, name)) names.push(prefix ? name.slice(1) : name);
}
if (Object.getOwnPropertySymbols) {
return names.concat(Object.getOwnPropertySymbols(events));
}
return names;
};
/**
* Return the listeners registered for a given event.
*
* @param {(String|Symbol)} event The event name.
* @returns {Array} The registered listeners.
* @public
*/
EventEmitter.prototype.listeners = function listeners(event) {
var evt = prefix ? prefix + event : event
, handlers = this._events[evt];
if (!handlers) return [];
if (handlers.fn) return [handlers.fn];
for (var i = 0, l = handlers.length, ee = new Array(l); i < l; i++) {
ee[i] = handlers[i].fn;
}
return ee;
};
/**
* Return the number of listeners listening to a given event.
*
* @param {(String|Symbol)} event The event name.
* @returns {Number} The number of listeners.
* @public
*/
EventEmitter.prototype.listenerCount = function listenerCount(event) {
var evt = prefix ? prefix + event : event
, listeners = this._events[evt];
if (!listeners) return 0;
if (listeners.fn) return 1;
return listeners.length;
};
/**
* Calls each of the listeners registered for a given event.
*
* @param {(String|Symbol)} event The event name.
* @returns {Boolean} `true` if the event had listeners, else `false`.
* @public
*/
EventEmitter.prototype.emit = function emit(event, a1, a2, a3, a4, a5) {
var evt = prefix ? prefix + event : event;
if (!this._events[evt]) return false;
var listeners = this._events[evt]
, len = arguments.length
, args
, i;
if (listeners.fn) {
if (listeners.once) this.removeListener(event, listeners.fn, undefined, true);
switch (len) {
case 1: return listeners.fn.call(listeners.context), true;
case 2: return listeners.fn.call(listeners.context, a1), true;
case 3: return listeners.fn.call(listeners.context, a1, a2), true;
case 4: return listeners.fn.call(listeners.context, a1, a2, a3), true;
case 5: return listeners.fn.call(listeners.context, a1, a2, a3, a4), true;
case 6: return listeners.fn.call(listeners.context, a1, a2, a3, a4, a5), true;
}
for (i = 1, args = new Array(len -1); i < len; i++) {
args[i - 1] = arguments[i];
}
listeners.fn.apply(listeners.context, args);
} else {
var length = listeners.length
, j;
for (i = 0; i < length; i++) {
if (listeners[i].once) this.removeListener(event, listeners[i].fn, undefined, true);
switch (len) {
case 1: listeners[i].fn.call(listeners[i].context); break;
case 2: listeners[i].fn.call(listeners[i].context, a1); break;
case 3: listeners[i].fn.call(listeners[i].context, a1, a2); break;
case 4: listeners[i].fn.call(listeners[i].context, a1, a2, a3); break;
default:
if (!args) for (j = 1, args = new Array(len -1); j < len; j++) {
args[j - 1] = arguments[j];
}
listeners[i].fn.apply(listeners[i].context, args);
}
}
}
return true;
};
/**
* Add a listener for a given event.
*
* @param {(String|Symbol)} event The event name.
* @param {Function} fn The listener function.
* @param {*} [context=this] The context to invoke the listener with.
* @returns {EventEmitter} `this`.
* @public
*/
EventEmitter.prototype.on = function on(event, fn, context) {
return addListener(this, event, fn, context, false);
};
/**
* Add a one-time listener for a given event.
*
* @param {(String|Symbol)} event The event name.
* @param {Function} fn The listener function.
* @param {*} [context=this] The context to invoke the listener with.
* @returns {EventEmitter} `this`.
* @public
*/
EventEmitter.prototype.once = function once(event, fn, context) {
return addListener(this, event, fn, context, true);
};
/**
* Remove the listeners of a given event.
*
* @param {(String|Symbol)} event The event name.
* @param {Function} fn Only remove the listeners that match this function.
* @param {*} context Only remove the listeners that have this context.
* @param {Boolean} once Only remove one-time listeners.
* @returns {EventEmitter} `this`.
* @public
*/
EventEmitter.prototype.removeListener = function removeListener(event, fn, context, once) {
var evt = prefix ? prefix + event : event;
if (!this._events[evt]) return this;
if (!fn) {
clearEvent(this, evt);
return this;
}
var listeners = this._events[evt];
if (listeners.fn) {
if (
listeners.fn === fn &&
(!once || listeners.once) &&
(!context || listeners.context === context)
) {
clearEvent(this, evt);
}
} else {
for (var i = 0, events = [], length = listeners.length; i < length; i++) {
if (
listeners[i].fn !== fn ||
(once && !listeners[i].once) ||
(context && listeners[i].context !== context)
) {
events.push(listeners[i]);
}
}
//
// Reset the array, or remove it completely if we have no more listeners.
//
if (events.length) this._events[evt] = events.length === 1 ? events[0] : events;
else clearEvent(this, evt);
}
return this;
};
/**
* Remove all listeners, or those of the specified event.
*
* @param {(String|Symbol)} [event] The event name.
* @returns {EventEmitter} `this`.
* @public
*/
EventEmitter.prototype.removeAllListeners = function removeAllListeners(event) {
var evt;
if (event) {
evt = prefix ? prefix + event : event;
if (this._events[evt]) clearEvent(this, evt);
} else {
this._events = new Events();
this._eventsCount = 0;
}
return this;
};
//
// Alias methods names because people roll like that.
//
EventEmitter.prototype.off = EventEmitter.prototype.removeListener;
EventEmitter.prototype.addListener = EventEmitter.prototype.on;
//
// Expose the prefix.
//
EventEmitter.prefixed = prefix;
//
// Allow `EventEmitter` to be imported as module namespace.
//
EventEmitter.EventEmitter = EventEmitter;
//
// Expose the module.
//
{
module.exports = EventEmitter;
}
} (eventemitter3));
return eventemitter3.exports;
}
var eventemitter3Exports = requireEventemitter3();
var EventEmitter = /*@__PURE__*/getDefaultExportFromCjs(eventemitter3Exports);
/**
* 播放器的类名前缀
*/ var PREFIX_CLASS = 'ezplayer';
/** */ var DATE_PICKER_ICON_WIDTH = 36;
/** 填充模式 */ var THEME_SCALE_MODE_TYPE = {
/** 画面完全填充canvas区域,画面会被拉伸 */ full: 0,
/** 画面做等比缩放后,高或宽对齐canvas区域,画面不被拉伸,但有黑边 */ auto: 1,
/** 视频画面做等比缩放后,完全填充canvas区域,画面不被拉伸,没有黑边,但画面显示不全 */ fullAuto: 2
};
// 切换主题时把当前的状态传给控件, 而不是传 theme 对象, 为了解耦
// 不要添加方法,只添加状态
var THEME_PROPS = [
'width',
'height',
'playing',
'volume',
'muted',
'loading',
'recType',
'isCurrentFullscreen',
'orientationAngle',
'zooming',
'zoom',
'recording',
'recordList',
'speed',
'urlInfo',
'videoLevelList',
'videoLevel'
];
/**
* 当移动端 picker 控件打开时需要关闭动画(定时器)
*/ var CLEAR_TIMER_HEADER_FOOTER_ANIMATION = 'CLEAR_TIMER_HEADER_FOOTER_ANIMATION';
/**
* 'rec', 'cloudRec', 'cloudRecord' 是一个组,并且要在头部(移动端特殊处理)
*
* | rec | cloudRec | cloudRecord |
* |:---------------------:|:------------:|:---------------:|
* | 本地回放(SDK存储回放) | 云存储回放 | 云录制回放 |
*/ var REC_GROUP = [
'rec',
'cloudRec',
'cloudRecord'
];
/**
* 设备信息
*
* | deviceID | deviceName |
* |:-------------------:|:-------------:|
* | 设备ID(唯一标识) | 设备名称 |
*/ var DEVICE_INFO_GROUP = [
'deviceID',
'deviceName'
];
/**
* 暂停时能禁用的控件按钮
*/ var PAUSE_DISABLED_BTN = [
'capturePicture',
'volume',
'ptz',
'record',
'talk',
'definition',
'speed',
'globalFullscreen'
];
/**
* 移动端可以扩展的控件
* ['ptz', 'talk', 'record', 'capturePicture', 'timeLine', 'rec', 'date'];
*/ var MOBILE_EXTENDS = [
'ptz',
'timeLine',
'rec',
'date'
];
/**
*
* 事件名
*
* --------------- ------------------
* - - theme.emit(EVENTS.xxxxx) - -
* - - -------------------------------------> - -
* - - - -
* - Theme - - Controls -
* - - control.emit(EVENTS.control.xxx) - -
* - - <-------------------------------------- - -
* --------------- ------------------
*
* @example
* ```ts
* theme.on(EVENTS.loading)
* theme.on(EVENTS.loading)
* ```
*
*/ // 事件命名不要超过两层
var EVENTS = {
loading: 'loading',
/** 播放 */ play: 'play',
/** 截图 */ capturePicture: 'capturePicture',
/** 音量变化 */ volumechange: 'volumechange',
/** 缩放变化 */ zoomChange: 'zoomChange',
/** 缩放状态变化, true: 可缩放状态,false: 禁止缩放状态(不能缩放) */ zoomingChange: 'zoomingChange',
/** 缩放平移改变 */ zoomTranslateChange: 'zoomTranslateChange',
/** 音频信息 */ audioInfo: 'audioInfo',
/** 视频信息 */ videoInfo: 'videoInfo',
/** 首帧 */ firstFrameDisplay: 'firstFrameDisplay',
/** 全屏 */ fullscreen: 'fullscreen',
/** 退出全屏 */ exitFullscreen: 'exitFullscreen',
/**
* 全屏变化(移动端也支持触发, 不过移动端是模拟的web全屏 即旋转 180度)
* 当 isCurrentFullscreen 为 true 时,表示当前容器全局全屏状态
*
* 注意:同一个页面中只要有全屏的操作或取消全屏的操作,都会触发该事件 (移动端除外)
*/ fullscreenChange: 'fullscreenChange',
/** 重置容器尺寸 */ resize: 'resize',
/** 屏幕旋转方向变化 */ orientationChange: 'orientationChange',
/** 音频编码不支持 (暂时仅支持标准流 flv) */ audioCodecUnsupported: 'audioCodecUnsupported',
/** 切换主题 themeDate 变化时 */ changeTheme: 'changeTheme',
/** 回放类型切换 */ recTypeChange: 'recTypeChange',
/** 切换清晰度 */ definitionChange: 'definitionChange',
/** 播放速度切换 */ speedChange: 'speedChange',
/** 控件开始录制 */ recordingChange: 'recordingChange',
/** 对讲状态变化 */ talkingChange: 'talkingChange',
/** 麦克风音量变化 */ talkVolumeChange: 'talkVolumeChange',
/** 动态切换日志配置 */ setLoggerOptions: 'setLoggerOptions',
records: 'records',
ptzSpeedChange: 'ptzSpeedChange',
setVideoLevelList: 'setVideoLevelList',
currentVideoLevel: 'currentVideoLevel',
currentVideoLevelAuto: 'currentVideoLevelAuto',
setAllDayRecTimes: 'setAllDayRecTimes',
getOSDTime: 'getOSDTime',
/**
* 控件相关
*/ control: {
/** 点击播放播放/暂停按钮 */ play: 'Control.play',
/** 播放控件销毁 */ playDestroy: 'Control.playDestroy',
/** 截图 */ capturePicture: 'Control.capturePicture',
/** 截图结果 */ capturePictureResult: 'Control.capturePictureResult',
/** 截图控件销毁 */ capturePictureDestroy: 'Control.capturePictureDestroy',
/** 音量变化 */ volumechange: 'Control.volumechange',
/** 音量调节面板 展示隐藏变换 */ volumePanelOpenChange: 'Control.volumePanelOpenChange',
/** 音量控件销毁 */ volumeDestroy: 'Control.volumeDestroy',
/** 控件组件栏(Header/Footer)展示隐藏 */ controlsBarOpenChange: 'Control.controlsBarOpenChange',
/** 当顶部控件栏(header)控件放置不下时, 显示的更多按钮的控件列表变化 */ headerMoreShowControlsChange: 'Control.headerMoreShowControlsChange',
/** 当顶部控件栏(header)控件放置不下时, 更多按钮 展示隐藏变换 */ headerMorePanelOpenChange: 'Control.headerMorePanelOpenChange',
/** 当底部控件栏(footer)控件放置不下时, 显示的更多按钮的控件列表变化 */ footerMoreShowControlsChange: 'Control.footerMoreShowControlsChange',
/** 当底部控件栏(footer)控件放置不下时, 更多按钮 展示隐藏变换 */ footerMorePanelOpenChange: 'Control.footerMorePanelOpenChange',
/** 设备信息控件销毁 */ deviceDestroy: 'Control.deviceDestroy',
/** 回放类型切换控件 */ recTypeChange: 'Control.recTypeChange',
/** 回放类型控件销毁 */ recDestroy: 'Control.recDestroy',
/** 切换清晰度控件 */ definitionChange: 'Control.definitionChange',
definitionList: 'Control.definitionList',
/** 切换清晰度面板 展示隐藏变换, (open, definition, item) */ definitionPanelOpenChange: 'Control.definitionPanelOpenChange',
/** 切换清晰度控件 */ definitionDestroy: 'Control.definitionDestroy',
/** 播放速度切换 */ speedChange: 'Control.speedChange',
/** 播放速度面板 显示隐藏变换 */ speedPanelOpenChange: 'Control.speedPanelOpenChange',
/** 播放速度控件销毁 */ speedDestroy: 'Control.speedDestroy',
/** 云台面板 显示隐藏变换 */ ptzPanelOpenChange: 'Control.ptzPanelOpenChange',
/** 云台速度切换 */ ptzSpeedChange: 'Control.ptzSpeedChange',
ptzError: 'Control.ptzError',
/** 云台控件销毁 */ ptzDestroy: 'Control.ptzDestroy',
/** 录制控件是否在录制中 */ recordingChange: 'Control.recordingChange',
recordDestroy: 'Control.recordDestroy',
/** 开始对讲 */ talkingChange: 'Control.talkingChange',
/** 对讲错误, 一般是麦克风权限被拒绝 */ talkError: 'Control.talkError',
/** 对讲控件销毁 */ talkDestroy: 'Control.talkDestroy',
/** 缩放比例改变 */ zoomChange: 'Control.zoomChange',
/** 音量调节面板 展示隐藏变换 */ zoomPanelOpenChange: 'Control.zoomPanelOpenChange',
/** 缩放控件销毁 */ zoomDestroy: 'Control.zoomDestroy',
/** 全屏控件销毁 */ fullscreenDestroy: 'Control.fullscreenDestroy',
/** 全局全屏控件销毁 */ globalFullscreenDestroy: 'Control.globalFullscreenDestroy',
/** 日期面板展示隐藏变换 */ datePanelOpenChange: 'Control.datePanelOpenChange',
/** 日期改变 */ dateChange: 'Control.dateChange',
/** 日期销毁 */ dateDestroy: 'Control.datePanelDestroy',
/** 时间轴拖动结束 */ timeLineChange: 'Control.timeLineChange',
/** 时间轴图片列表面板 */ timeLinePanelOpenChange: 'Control.timeLinePanelOpenChange',
/** 时间轴控件销毁 */ timeLineDestroy: 'Control.timeLineDestroy',
/** 主题控件挂载前 切换新的主题也会触发,如果想首次获取需要在 onInitializing 回调中进行监听 */ beforeMountControls: 'Control.beforeMountControls',
/** 主题控件挂载完成, 切换新的主题也会触发,如果想首次获取需要在 onInitializing 回调中进行监听 */ mountedControls: 'Control.mountedControls',
/** 主题控件卸载前, 已有控件卸载时才可触发 */ beforeUnmountControls: 'Control.beforeUnmountControls',
/** 主题控件卸载完成, 已有控件卸载时才可触发 */ unmountedControls: 'Control.unmountedControls',
/** 封面控件销毁 */ posterDestroy: 'Control.posterDestroy',
/** 加载控件销毁 */ loadingDestroy: 'Control.loadingDestroy',
/** 消息控件销毁 */ messageDestroy: 'Control.messageDestroy',
/** 播放器内容区域销毁 */ contentDestroy: 'Control.contentDestroy'
},
/**
* 主题控件相关
*/ theme: {
/** 销毁前 */ beforeDestroy: 'Theme.beforeDestroy',
/** 销毁后 */ destroyed: 'Theme.destroyed',
/** 移动端扩展销毁 */ mobileExtendDestroy: 'Theme.mobileExtendDestroy',
/** 回放页底部销毁 */ recFooterDestroy: 'Theme.recFooterDestroy'
},
message: 'message'
};
function getLocale(locales, language) {
if (language === void 0) language = 'zh';
return (locales == null ? void 0 : locales[language]) || (locales == null ? void 0 : locales['zh']) || {};
}
function _defineProperties$9(target, props) {
for(var i = 0; i < props.length; i++){
var descriptor = props[i];
descriptor.enumerable = descriptor.enumerable || false;
descriptor.configurable = true;
if ("value" in descriptor) descriptor.writable = true;
Object.defineProperty(target, descriptor.key, descriptor);
}
}
function _create_class$9(Constructor, protoProps, staticProps) {
if (protoProps) _defineProperties$9(Constructor.prototype, protoProps);
return Constructor;
}
function _inherits$r(subClass, superClass) {
if (typeof superClass !== "function" && superClass !== null) {
throw new TypeError("Super expression must either be null or a function");
}
subClass.prototype = Object.create(superClass && superClass.prototype, {
constructor: {
value: subClass,
writable: true,
configurable: true
}
});
if (superClass) _set_prototype_of$r(subClass, superClass);
}
function _set_prototype_of$r(o, p) {
_set_prototype_of$r = Object.setPrototypeOf || function setPrototypeOf(o, p) {
o.__proto__ = p;
return o;
};
return _set_prototype_of$r(o, p);
}
/**
* 控件基类
* @category Control
* @example
* ```ts
* // 创建控件
* class MyControl extends Control {}
*
* // 使用控件
* const myControl = new MyControl({})
* ```
*/ var Control = /*#__PURE__*/ function(EventEmitter) {
_inherits$r(Control, EventEmitter);
function Control(options) {
var _this;
var _this___options;
_this = EventEmitter.call(this) || this, /** 具体语言对应的值 */ _this.locale = null, _this.__options = {}, _this._disabled = false, _this._active = false;
_this.__options = options;
_this.locale = getLocale(_this.__options.locales, _this.__options.language);
_this.$container = document.createElement(options.tagName || 'span');
_this.$container.classList.add("" + PREFIX_CLASS + "-control"); // 默认隐藏
if ((options == null ? void 0 : options.controlType) === 'text') {
_this.$container.classList.add("" + PREFIX_CLASS + "-control-text");
} else if ((options == null ? void 0 : options.controlType) === 'block') {
_this.$container.classList.add("" + PREFIX_CLASS + "-control-block"); // 块
} else {
_this.$container.classList.add("" + PREFIX_CLASS + "-control-btn"); // 默认隐藏
}
if (options == null ? void 0 : options.classNameSuffix) {
// prettier-ignore
_this.$container.classList.add(PREFIX_CLASS + "-control-" + (options == null ? void 0 : options.classNameSuffix));
}
if (options == null ? void 0 : options.className) {
_this.$container.classList.add(options.className);
}
if (options.getPopupContainer) {
_this.$popupContainer = options.getPopupContainer == null ? void 0 : options.getPopupContainer.call(options);
} else {
_this.$popupContainer = document.body;
}
_this.$popupContainer.appendChild(_this.$container);
_this._onDBlClick = _this._onDBlClick.bind(_this);
_this._onClick();
if (((_this___options = _this.__options) == null ? void 0 : _this___options.controlType) !== 'block') _this.$container.addEventListener('dblclick', _this._onDBlClick);
return _this;
}
var _proto = Control.prototype;
/**
* 重置整个控件
*/ _proto.reset = function reset(hide) {
if (this._camelCaseName) {
this.emit("Control." + this._camelCaseName + "Reset", hide);
}
};
/**
* 重置控件挂载节点
* @param popupContainer 重新挂载的节点
* @param append 添加的方式, 默认 append
* @param element 当 append = before 时 需要 element, 插入 element 前
*
* | prepend | append |
* |:-------------:|:---------------:|
* | 插入第一个位置 | 追加在末尾 |
*/ _proto.resetPopupContainer = function resetPopupContainer(popupContainer, append, element) {
if (popupContainer !== this.$popupContainer) {
if (this.$popupContainer.contains(this.$container)) {
this.$popupContainer.removeChild(this.$container);
}
this.$popupContainer = popupContainer;
if (append === 'prepend') {
// https://caniuse.com/?search=prepend
this.$popupContainer.prepend(this.$container);
} else if (append === 'before') {
if (element) {
this.$popupContainer.insertBefore(this.$container, element);
}
} else {
this.$popupContainer.appendChild(this.$container);
}
if (this._camelCaseName) {
this.emit("Control." + this._camelCaseName + "ResetContainer", this.__options.classNameSuffix, popupContainer);
}
}
};
/**
* 隐藏整个控件
*/ _proto.hide = function hide() {
if (this.$container) {
this.$container.style.display = 'none';
this.$container.classList.add("" + PREFIX_CLASS + "-hide");
}
};
/**
* 销毁控件
*/ _proto.destroy = function destroy() {
var _this___options, _this_$container_remove, _this_$container;
if (this._camelCaseName) {
this.emit("Control." + this._camelCaseName + "Destroy");
}
if (((_this___options = this.__options) == null ? void 0 : _this___options.controlType) !== 'block') this.$container.addEventListener('dblclick', this._onDBlClick);
this._active = false;
this.removeAllListeners();
(_this_$container = this.$container) == null ? void 0 : (_this_$container_remove = _this_$container.remove) == null ? void 0 : _this_$container_remove.call(_this_$container);
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
this.$container = null;
};
_proto._updateDisabledState = function _updateDisabledState(disabled) {
if (disabled) {
var _this_$container;
(_this_$container = this.$container) == null ? void 0 : _this_$container.classList.add("" + PREFIX_CLASS + "-disabled");
} else {
var _this_$container1;
(_this_$container1 = this.$container) == null ? void 0 : _this_$container1.classList.remove("" + PREFIX_CLASS + "-disabled");
}
};
_proto._updateActiveState = function _updateActiveState(active) {
if (active) {
var _this_$container;
(_this_$container = this.$container) == null ? void 0 : _this_$container.classList.add("" + PREFIX_CLASS + "-active");
} else {
var _this_$container1;
(_this_$container1 = this.$container) == null ? void 0 : _this_$container1.classList.remove("" + PREFIX_CLASS + "-active");
}
};
/**
* 当点击 Control 时 触发子类的 _onControlClick
*/ _proto._onClick = function _onClick() {
var _this = this;
this.$container.addEventListener('click', function(e) {
var _this_$container;
// prettier-ignore
if (!((_this_$container = _this.$container) == null ? void 0 : _this_$container.classList.contains("" + PREFIX_CLASS + "-disabled"))) {
// _onControlClick 来源自子类中
_this._onControlClick == null ? void 0 : _this._onControlClick.call(_this, e);
}
});
};
_proto._onDBlClick = function _onDBlClick(e) {
e.stopPropagation();
e.preventDefault();
};
/**
* 点击 Control 控件触发
* @param {Event} e
* @returns {void}
*/ _proto._onControlClick = function _onControlClick(e) {
// 这是一个空函数, 子类可以实现重新改方法
this.__options.onClick == null ? void 0 : this.__options.onClick.call(this.__options, e);
};
_create_class$9(Control, [
{
key: "active",
get: /**
* 是否激活
*/ function get() {
return this._active;
},
set: function set(active) {
if (this._disabled && !this._active) {
// 不允许禁用情况下激活
return;
}
this._active = active;
this._updateActiveState(active);
}
},
{
key: "disabled",
get: /**
* 是否禁用
*/ function get() {
return this._disabled;
},
set: function set(disabled) {
this._disabled = disabled;
this._updateDisabledState(disabled);
}
},
{
key: "_camelCaseName",
get: function get() {
if (this.__options.classNameSuffix) {
return this.__options.classNameSuffix.replace(/[-_\s]+(.)?/g, function(_, c) {
return c ? c.toUpperCase() : '';
}).replace(/^(.)/, function(first) {
return first.toLowerCase();
});
}
return null;
}
}
]);
return Control;
}(EventEmitter);
function _extends$q() {
_extends$q = Object.assign || function(target) {
for(var i = 1; i < arguments.length; i++){
var source = arguments[i];
for(var key in source){
if (Object.prototype.hasOwnProperty.call(source, key)) {
target[key] = source[key];
}
}
}
return target;
};
return _extends$q.apply(this, arguments);
}
function _inherits$q(subClass, superClass) {
if (typeof superClass !== "function" && superClass !== null) {
throw new TypeError("Super expression must either be null or a function");
}
subClass.prototype = Object.create(superClass && superClass.prototype, {
constructor: {
value: subClass,
writable: true,
configurable: true
}
});
if (superClass) _set_prototype_of$q(subClass, superClass);
}
function _set_prototype_of$q(o, p) {
_set_prototype_of$q = Object.setPrototypeOf || function setPrototypeOf(o, p) {
o.__proto__ = p;
return o;
};
return _set_prototype_of$q(o, p);
}
var LOADING_DEFAULT_OPTIONS = {};
/**
* 加载动画控件
* @category Control
*/ var Loading = /*#__PURE__*/ function(Control) {
_inherits$q(Loading, Control);
function Loading(options) {
if (options === void 0) options = {};
var _this;
_this = Control.call(this, Object.assign({}, LOADING_DEFAULT_OPTIONS, _extends$q({}, options, {
tagName: 'div',
controlType: 'block',
classNameSuffix: 'loading'
}))) || this;
_this._options = Object.assign({}, options);
_this.$container.classList.add("" + PREFIX_CLASS + "-loading", "" + PREFIX_CLASS + "-hide"); // 默认隐藏
_this._html();
return _this;
}
var _proto = Loading.prototype;
_proto._html = function _html() {
if (typeof this._options.render === 'function') {
this.$container.innerHTML = this._options.render();
} else {
var _this_locale;
this.$container.innerHTML = '\n <span class="' + PREFIX_CLASS + "-loading-dot " + PREFIX_CLASS + '-loading-dot-load">\n <i class="' + PREFIX_CLASS + '-loading-dot-item"></i>\n <i class="' + PREFIX_CLASS + '-loading-dot-item"></i>\n <i class="' + PREFIX_CLASS + '-loading-dot-item"></i>\n <i class="' + PREFIX_CLASS + '-loading-dot-item"></i>\n </span>\n <div class="' + PREFIX_CLASS + '-loading-text">' + (((_this_locale = this.locale) == null ? void 0 : _this_locale.LOADING) || 'loading...') + "</div>\n ";
}
};
/**
* 动画展示
* @param html 自定义动画内容, 如果不存在则使用默认动画
*/ _proto.show = function show(html) {
if (html) this.$container.innerHTML = html;
// 初始化, 出流再等待中(回放出流结束不包括)
this.$container.style.display = 'flex';
this.$container.classList.remove("" + PREFIX_CLASS + "-hide");
this.$popupContainer.classList.add("" + PREFIX_CLASS + "-has-loading");
};
/**
* 隐藏动画
*/ _proto.hide = function hide() {
Control.prototype.hide.call(this);
this.$popupContainer.classList.remove("" + PREFIX_CLASS + "-has-loading");
};
return Loading;
}(Control);
function _extends$p() {
_extends$p = Object.assign || function(target) {
for(var i = 1; i < arguments.length; i++){
var source = arguments[i];
for(var key in source){
if (Object.prototype.hasOwnProperty.call(source, key)) {
target[key] = source[key];
}
}
}
return target;
};
return _extends$p.apply(this, arguments);
}
function _inherits$p(subClass, superClass) {
if (typeof superClass !== "function" && superClass !== null) {
throw new TypeError("Super expression must either be null or a function");
}
subClass.prototype = Object.create(superClass && superClass.prototype, {
constructor: {
value: subClass,
writable: true,
configurable: true
}
});
if (superClass) _set_prototype_of$p(subClass, superClass);
}
function _set_prototype_of$p(o, p) {
_set_prototype_of$p = Object.setPrototypeOf || function setPrototypeOf(o, p) {
o.__proto__ = p;
return o;
};
return _set_prototype_of$p(o, p);
}
// 不放在服务器上 是因为有可能http加载失败
// prettier-ignore
var DEFAULT_POSTER = "";
var POSTER_OPTIONS = {
poster: DEFAULT_POSTER
};
/**
* 封面控件
* @category Control
*/ var Poster = /*#__PURE__*/ function(Control) {
_inherits$p(Poster, Control);
function Poster(options) {
if (options === void 0) options = {};
var _this;
_this = Control.call(this, Object.assign({}, POSTER_OPTIONS, _extends$p({}, options, {
tagName: 'div',
controlType: 'block',
classNameSuffix: 'poster'
}))) || this;
_this._options = Object.assign({}, POSTER_OPTIONS, options);
_this.$container.classList.add("" + PREFIX_CLASS + "-poster", "" + PREFIX_CLASS + "-hide");
_this._imgLoadErrorEvent = _this._imgLoadErrorEvent.bind(_this);
if (_this._options.poster) {
_this.setPoster(_this._options.poster);
}
return _this;
}
var _proto = Poster.prototype;
/**
* 封面图片加载失败
* @param error
*/ _proto._imgLoadErrorEvent = function _imgLoadErrorEvent(error) {
var _error_target;
// 封面加载失败, 一般只有自定义封面会出现这种情况
// 这里不做img src重置处理 防止无限循环
this._options.onLoadImgError == null ? void 0 : this._options.onLoadImgError.call(this._options, ((_error_target = error.target) == null ? void 0 : _error_target.getAttribute('src')) || '');
};
/**
* 设置封面 这里不对 poster 进行缓存,如果有值优先使用,如果没有值优先使用 初始化传入的值
* @param {string} poster 封面地址或 base64 数据
*/ _proto.setPoster = function setPoster(poster) {
var // eslint-disable-next-line @typescript-eslint/unbound-method
// prettier-ignore
_this_$container_querySelector;
poster = poster != null ? poster : this._options.poster;
if (poster === '') {
this.$container.innerHTML = '';
return;
}
this.$container.innerHTML = '<img class="' + PREFIX_CLASS + '-poster-img" src="' + poster + '" />';
(_this_$container_querySelector = this.$container.querySelector("." + PREFIX_CLASS + "-poster-img")) == null ? void 0 : _this_$container_querySelector.addEventListener("error", this._imgLoadErrorEvent);
};
/**
* 展示封面 这里不对 poster 进行缓存, 如果有值优先使用, 如果没有值优先使用 初始化传入的值
*/ _proto.show = function show() {
if (this.$container) {
this.$container.style.display = 'flex';
this.$container.classList.remove("" + PREFIX_CLASS + "-hide");
}
};
/**
* 隐藏封面
*/ _proto.hide = function hide() {
var // eslint-disable-next-line @typescript-eslint/unbound-method
_this_$container_querySelector, _this_$container;
Control.prototype.hide.call(this);
(_this_$container = this.$container) == null ? void 0 : (_this_$container_querySelector = _this_$container.querySelector("." + PREFIX_CLASS + "-poster-img")) == null ? void 0 : _this_$container_querySelector.removeEventListener('error', this._imgLoadErrorEvent);
};
/**
* 销毁
*/ _proto.destroy = function destroy() {
this.hide();
Control.prototype.destroy.call(this);
};
return Poster;
}(Control);
var Icons = {
/** 播放 */ play: '<svg width="1em" height="1em" viewBox="0 0 24 24" fill="currentColor" focusable="false" aria-hidden="true" data-icon="play">\n <rect x="6.5" y="5.5" rx="1.25" width="2.5" height="13"/>\n <rect x="15" y="5.5" rx="1.25" width="2.5" height="13"/>\n </svg>',
/** 暂停 */ pause: '<svg width="1em" height="1em" viewBox="0 0 24 24" fill="currentColor" focusable="false" aria-hidden="true" data-icon="pause"> <path d="M17.5 13.66L9.1 19.26C7.78 20.14 6 19.19 6 17.59L6 6.4C6 4.8 7.78 3.85 9.1 4.73L17.5 10.33C18.69 11.12 18.69 12.87 17.5 13.66Z" /></svg>',
/** 音量 */ volume: function(prefix) {
return '<svg width="1em" height="1em" viewBox="0 0 24 24" stroke="currentColor" fill="none" focusable="false" aria-hidden="true" data-icon="volume">\n <path class="' + prefix + '-icon-volume-muted" d="M20.57 9.69L16.07 14.19" stroke-width="1.500000" stroke-linejoin="round" stroke-linecap="round"/>\n <path class="' + prefix + '-icon-volume-muted" d="M20.57 14.19L16.07 9.69" stroke-width="1.500000" stroke-linejoin="round" stroke-linecap="round"/>\n <!-- 音低 -->\n <path class="' + prefix + '-icon-volume-low" d="M15.53 15.97C16.69 15.25 17.49 13.75 17.49 12C17.49 10.25 16.69 8.75 15.53 8.02" stroke-width="1.500000" stroke-linejoin="round" stroke-linecap="round"/>\n <!-- 音高 -->\n <path class="' + prefix + '-icon-volume-high" d="M18.5 19.06C20.31 17.5 21.49 14.93 21.49 12C21.49 9.07 20.31 6.48 18.49 4.93" stroke-width="1.500000" stroke-linejoin="round" stroke-linecap="round"/>\n <path d="M5.87 8.62L9.85 5.25C10.5 4.7 11.49 5.16 11.49 6.01L11.49 17.98C11.49 18.83 10.5 19.29 9.85 18.74L5.87 15.37" stroke-width="1.500000" stroke-linejoin="round" stroke-linecap="round"/>\n <path d="M5.87 15.37L3.49 15.37C2.94 15.37 2.49 14.92 2.49 14.37L2.49 9.62C2.49 9.07 2.94 8.62 3.49 8.62L5.87 8.62" stroke-width="1.500000" stroke-linejoin="round" stroke-linecap="round"/>\n </svg>';
},
/** 全屏 */ mobileFullscreen: '<svg width="1em" height="1em" viewBox="0 0 24 24" fill="currentColor" focusable="false" aria-hidden="true" data-icon="mobile-fullscreen">\n <path d="M4 2.9082L11 2.9082C12.5188 2.9082 13.75 4.1394 13.75 5.6582L13.75 17.6582C13.75 19.177 12.5188 20.4082 11 20.4082L4 20.4082C2.4812 20.4082 1.25 19.177 1.25 17.6582L1.25 5.6582C1.25 4.1394 2.4812 2.9082 4 2.9082ZM4 4.4082L11 4.4082C11.6903 4.4082 12.25 4.9679 12.25 5.6582L12.25 17.6582C12.25 18.3485 11.6903 18.9082 11 18.9082L4 18.9082C3.30969 18.9082 2.75 18.3485 2.75 17.6582L2.75 5.6582C2.75 4.9679 3.30969 4.4082 4 4.4082ZM22.1511 18.3113C22.1511 19.3595 21.2467 20.1652 20.1509 20.2362L19.993 20.2413L15.2939 20.2413C14.8798 20.2413 14.5439 19.9055 14.5439 19.4913C14.5439 19.1116 14.8262 18.7979 15.1921 18.7482L15.2939 18.7413L19.993 18.7413C20.344 18.7413 20.5962 18.5592 20.6432 18.3732L20.6511 18.3113L20.6511 12.4895C20.6511 12.3048 20.4338 12.1042 20.1062 12.066L19.993 12.0594L15.2939 12.0594C14.8798 12.0594 14.5439 11.7238 14.5439 11.3094C14.5439 10.9298 14.8262 10.616 15.1921 10.5663L15.2939 10.5594L19.993 10.5594C21.1055 10.5594 22.0605 11.3175 22.145 12.3416L22.1511 12.4895L22.1511 18.3113ZM10.3225 16.1035C10.3225 15.6893 9.98669 15.3535 9.57251 15.3535L5.84644 15.3535L5.74463 15.3604C5.37854 15.41 5.09644 15.7239 5.09644 16.1035C5.09644 16.5177 5.43225 16.8535 5.84644 16.8535L9.57251 16.8535L9.67432 16.8467C10.0404 16.797 10.3225 16.4832 10.3225 16.1035Z" clip-rule="evenodd" fill-rule="evenodd"/>\n </svg>',
fullscreen: '<svg width="1em" height="1em" viewBox="0 0 24 24" fill="currentColor" focusable="false" aria-hidden="true" data-icon="webFullscreen">\n <path d="M18.4009 2.40125L5.59972 2.40125C4.74849 2.39276 3.9297 2.72729 3.32788 3.32935C2.72607 3.9314 2.39188 4.75031 2.40062 5.6015L2.40062 18.4015C2.3922 19.2527 2.7267 20.0715 3.32874 20.6733C3.93079 21.2751 4.74969 21.6093 5.60091 21.6006L18.4009 21.6006C19.2519 21.609 20.0705 21.2747 20.6723 20.6729C21.2741 20.0711 21.6084 19.2525 21.6 18.4015L21.6 5.60034C21.6084 4.74933 21.2741 3.93073 20.6723 3.32892C20