// ==UserScript== // @name agefans Enhance // @namespace https://github.com/IronKinoko/agefans-enhance // @icon https://www.agemys.com/favicon.ico // @version 1.43.9 // @description 增强播放功能,实现自动换集、无缝换集、画中画、历史记录、断点续播、弹幕等功能。适配agefans、NT动漫、bimiacg、mutefun、次元城、稀饭动漫 // @author IronKinoko // @include https://www.age.tv/* // @include https://www.agefans.* // @include https://www.agemys.* // @include https://www.agedm.* // @include https://m.agedm.* // @include http*://www.ntdm9.* // @include http*://www.bimiacg*.net* // @include https://pro.ascepan.top/* // @include https://danmu.yhdmjx.com/* // @include https://*.sp-flv.com* // @include https://*43.240.74.134* // @include https://*43.240.156.118* // @include https://www.mutedm.com/* // @include https://www.mutean.com/* // @include https://www.mute01.com/* // @include https://www.cycanime.com/* // @include https://www.cyc-anime.net/* // @include https://www.cycani.org/* // @include https://player.cycanime.com/* // @include https://dick.xfani.com/* // @include https://player.moedot.net/* // @run-at document-end // @require https://unpkg.com/jquery@3.6.0/dist/jquery.min.js // @require https://unpkg.com/plyr@3.6.4/dist/plyr.min.js // @require https://unpkg.com/hls.js@1.0.9/dist/hls.min.js // @require https://unpkg.com/@ironkinoko/danmaku@1.4.1/dist/danmaku.umd.js // @grant GM_getValue // @grant GM_setValue // @grant GM_xmlhttpRequest // @connect dandanplay.net // @license MIT // @downloadURL none // ==/UserScript== /** * 权限声明: * 1. GM_xmlhttpRequest * 脚本会请求有限的网络权限。仅用于访问弹幕查询功能需要链接到的 dandanplay.net 第三方域名 * 你可以从 脚本编辑/设置/XHR安全 中管理网络权限 * * 2. GM_getValue, GM_setValue * 脚本会使用本地存储功能,用于在不同页面间保存“播放器配置”与“agefans 历史浏览记录”。 * * 3. @include * 脚本还匹配了 agefans 以外的一些链接,用于提供相同视频资源搜索功能 */ (function (Hls, Plyr, Danmaku) { 'use strict'; var e=[],t=[];function n(n,r){if(n&&"undefined"!=typeof document){var a,s=!0===r.prepend?"prepend":"append",d=!0===r.singleTag,i="string"==typeof r.container?document.querySelector(r.container):document.getElementsByTagName("head")[0];if(d){var u=e.indexOf(i);-1===u&&(u=e.push(i)-1,t[u]={}),a=t[u]&&t[u][s]?t[u][s]:t[u][s]=c();}else a=c();65279===n.charCodeAt(0)&&(n=n.substring(1)),a.styleSheet?a.styleSheet.cssText+=n:a.appendChild(document.createTextNode(n));}function c(){var e=document.createElement("style");if(e.setAttribute("type","text/css"),r.attributes)for(var t=Object.keys(r.attributes),n=0;n true * * _.isObjectLike([1, 2, 3]); * // => true * * _.isObjectLike(_.noop); * // => false * * _.isObjectLike(null); * // => false */ function isObjectLike(value) { return value != null && typeof value == 'object'; } /** `Object#toString` result references. */ var symbolTag = '[object Symbol]'; /** * Checks if `value` is classified as a `Symbol` primitive or object. * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a symbol, else `false`. * @example * * _.isSymbol(Symbol.iterator); * // => true * * _.isSymbol('abc'); * // => false */ function isSymbol(value) { return typeof value == 'symbol' || (isObjectLike(value) && baseGetTag(value) == symbolTag); } /** Used to match a single whitespace character. */ var reWhitespace = /\s/; /** * Used by `_.trim` and `_.trimEnd` to get the index of the last non-whitespace * character of `string`. * * @private * @param {string} string The string to inspect. * @returns {number} Returns the index of the last non-whitespace character. */ function trimmedEndIndex(string) { var index = string.length; while (index-- && reWhitespace.test(string.charAt(index))) {} return index; } /** Used to match leading whitespace. */ var reTrimStart = /^\s+/; /** * The base implementation of `_.trim`. * * @private * @param {string} string The string to trim. * @returns {string} Returns the trimmed string. */ function baseTrim(string) { return string ? string.slice(0, trimmedEndIndex(string) + 1).replace(reTrimStart, '') : string; } /** * Checks if `value` is the * [language type](http://www.ecma-international.org/ecma-262/7.0/#sec-ecmascript-language-types) * of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`) * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is an object, else `false`. * @example * * _.isObject({}); * // => true * * _.isObject([1, 2, 3]); * // => true * * _.isObject(_.noop); * // => true * * _.isObject(null); * // => false */ function isObject(value) { var type = typeof value; return value != null && (type == 'object' || type == 'function'); } /** Used as references for various `Number` constants. */ var NAN = 0 / 0; /** Used to detect bad signed hexadecimal string values. */ var reIsBadHex = /^[-+]0x[0-9a-f]+$/i; /** Used to detect binary string values. */ var reIsBinary = /^0b[01]+$/i; /** Used to detect octal string values. */ var reIsOctal = /^0o[0-7]+$/i; /** Built-in method references without a dependency on `root`. */ var freeParseInt = parseInt; /** * Converts `value` to a number. * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to process. * @returns {number} Returns the number. * @example * * _.toNumber(3.2); * // => 3.2 * * _.toNumber(Number.MIN_VALUE); * // => 5e-324 * * _.toNumber(Infinity); * // => Infinity * * _.toNumber('3.2'); * // => 3.2 */ function toNumber(value) { if (typeof value == 'number') { return value; } if (isSymbol(value)) { return NAN; } if (isObject(value)) { var other = typeof value.valueOf == 'function' ? value.valueOf() : value; value = isObject(other) ? (other + '') : other; } if (typeof value != 'string') { return value === 0 ? value : +value; } value = baseTrim(value); var isBinary = reIsBinary.test(value); return (isBinary || reIsOctal.test(value)) ? freeParseInt(value.slice(2), isBinary ? 2 : 8) : (reIsBadHex.test(value) ? NAN : +value); } /** `Object#toString` result references. */ var asyncTag = '[object AsyncFunction]', funcTag = '[object Function]', genTag = '[object GeneratorFunction]', proxyTag = '[object Proxy]'; /** * Checks if `value` is classified as a `Function` object. * * @static * @memberOf _ * @since 0.1.0 * @category Lang * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a function, else `false`. * @example * * _.isFunction(_); * // => true * * _.isFunction(/abc/); * // => false */ function isFunction(value) { if (!isObject(value)) { return false; } // The use of `Object#toString` avoids issues with the `typeof` operator // in Safari 9 which returns 'object' for typed arrays and other constructors. var tag = baseGetTag(value); return tag == funcTag || tag == genTag || tag == asyncTag || tag == proxyTag; } /** Used to detect overreaching core-js shims. */ var coreJsData = root['__core-js_shared__']; /** Used to detect methods masquerading as native. */ var maskSrcKey = (function() { var uid = /[^.]+$/.exec(coreJsData && coreJsData.keys && coreJsData.keys.IE_PROTO || ''); return uid ? ('Symbol(src)_1.' + uid) : ''; }()); /** * Checks if `func` has its source masked. * * @private * @param {Function} func The function to check. * @returns {boolean} Returns `true` if `func` is masked, else `false`. */ function isMasked(func) { return !!maskSrcKey && (maskSrcKey in func); } /** Used for built-in method references. */ var funcProto$1 = Function.prototype; /** Used to resolve the decompiled source of functions. */ var funcToString$1 = funcProto$1.toString; /** * Converts `func` to its source code. * * @private * @param {Function} func The function to convert. * @returns {string} Returns the source code. */ function toSource(func) { if (func != null) { try { return funcToString$1.call(func); } catch (e) {} try { return (func + ''); } catch (e) {} } return ''; } /** * Used to match `RegExp` * [syntax characters](http://ecma-international.org/ecma-262/7.0/#sec-patterns). */ var reRegExpChar = /[\\^$.*+?()[\]{}|]/g; /** Used to detect host constructors (Safari). */ var reIsHostCtor = /^\[object .+?Constructor\]$/; /** Used for built-in method references. */ var funcProto = Function.prototype, objectProto$2 = Object.prototype; /** Used to resolve the decompiled source of functions. */ var funcToString = funcProto.toString; /** Used to check objects for own properties. */ var hasOwnProperty$2 = objectProto$2.hasOwnProperty; /** Used to detect if a method is native. */ var reIsNative = RegExp('^' + funcToString.call(hasOwnProperty$2).replace(reRegExpChar, '\\$&') .replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g, '$1.*?') + '$' ); /** * The base implementation of `_.isNative` without bad shim checks. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is a native function, * else `false`. */ function baseIsNative(value) { if (!isObject(value) || isMasked(value)) { return false; } var pattern = isFunction(value) ? reIsNative : reIsHostCtor; return pattern.test(toSource(value)); } /** * Gets the value at `key` of `object`. * * @private * @param {Object} [object] The object to query. * @param {string} key The key of the property to get. * @returns {*} Returns the property value. */ function getValue(object, key) { return object == null ? undefined : object[key]; } /** * Gets the native function at `key` of `object`. * * @private * @param {Object} object The object to query. * @param {string} key The key of the method to get. * @returns {*} Returns the function if it's native, else `undefined`. */ function getNative(object, key) { var value = getValue(object, key); return baseIsNative(value) ? value : undefined; } /** * Performs a * [`SameValueZero`](http://ecma-international.org/ecma-262/7.0/#sec-samevaluezero) * comparison between two values to determine if they are equivalent. * * @static * @memberOf _ * @since 4.0.0 * @category Lang * @param {*} value The value to compare. * @param {*} other The other value to compare. * @returns {boolean} Returns `true` if the values are equivalent, else `false`. * @example * * var object = { 'a': 1 }; * var other = { 'a': 1 }; * * _.eq(object, object); * // => true * * _.eq(object, other); * // => false * * _.eq('a', 'a'); * // => true * * _.eq('a', Object('a')); * // => false * * _.eq(NaN, NaN); * // => true */ function eq(value, other) { return value === other || (value !== value && other !== other); } /* Built-in method references that are verified to be native. */ var nativeCreate = getNative(Object, 'create'); /** * Removes all key-value entries from the hash. * * @private * @name clear * @memberOf Hash */ function hashClear() { this.__data__ = nativeCreate ? nativeCreate(null) : {}; this.size = 0; } /** * Removes `key` and its value from the hash. * * @private * @name delete * @memberOf Hash * @param {Object} hash The hash to modify. * @param {string} key The key of the value to remove. * @returns {boolean} Returns `true` if the entry was removed, else `false`. */ function hashDelete(key) { var result = this.has(key) && delete this.__data__[key]; this.size -= result ? 1 : 0; return result; } /** Used to stand-in for `undefined` hash values. */ var HASH_UNDEFINED$1 = '__lodash_hash_undefined__'; /** Used for built-in method references. */ var objectProto$1 = Object.prototype; /** Used to check objects for own properties. */ var hasOwnProperty$1 = objectProto$1.hasOwnProperty; /** * Gets the hash value for `key`. * * @private * @name get * @memberOf Hash * @param {string} key The key of the value to get. * @returns {*} Returns the entry value. */ function hashGet(key) { var data = this.__data__; if (nativeCreate) { var result = data[key]; return result === HASH_UNDEFINED$1 ? undefined : result; } return hasOwnProperty$1.call(data, key) ? data[key] : undefined; } /** Used for built-in method references. */ var objectProto = Object.prototype; /** Used to check objects for own properties. */ var hasOwnProperty = objectProto.hasOwnProperty; /** * Checks if a hash value for `key` exists. * * @private * @name has * @memberOf Hash * @param {string} key The key of the entry to check. * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. */ function hashHas(key) { var data = this.__data__; return nativeCreate ? (data[key] !== undefined) : hasOwnProperty.call(data, key); } /** Used to stand-in for `undefined` hash values. */ var HASH_UNDEFINED = '__lodash_hash_undefined__'; /** * Sets the hash `key` to `value`. * * @private * @name set * @memberOf Hash * @param {string} key The key of the value to set. * @param {*} value The value to set. * @returns {Object} Returns the hash instance. */ function hashSet(key, value) { var data = this.__data__; this.size += this.has(key) ? 0 : 1; data[key] = (nativeCreate && value === undefined) ? HASH_UNDEFINED : value; return this; } /** * Creates a hash object. * * @private * @constructor * @param {Array} [entries] The key-value pairs to cache. */ function Hash(entries) { var index = -1, length = entries == null ? 0 : entries.length; this.clear(); while (++index < length) { var entry = entries[index]; this.set(entry[0], entry[1]); } } // Add methods to `Hash`. Hash.prototype.clear = hashClear; Hash.prototype['delete'] = hashDelete; Hash.prototype.get = hashGet; Hash.prototype.has = hashHas; Hash.prototype.set = hashSet; /** * Removes all key-value entries from the list cache. * * @private * @name clear * @memberOf ListCache */ function listCacheClear() { this.__data__ = []; this.size = 0; } /** * Gets the index at which the `key` is found in `array` of key-value pairs. * * @private * @param {Array} array The array to inspect. * @param {*} key The key to search for. * @returns {number} Returns the index of the matched value, else `-1`. */ function assocIndexOf(array, key) { var length = array.length; while (length--) { if (eq(array[length][0], key)) { return length; } } return -1; } /** Used for built-in method references. */ var arrayProto = Array.prototype; /** Built-in value references. */ var splice = arrayProto.splice; /** * Removes `key` and its value from the list cache. * * @private * @name delete * @memberOf ListCache * @param {string} key The key of the value to remove. * @returns {boolean} Returns `true` if the entry was removed, else `false`. */ function listCacheDelete(key) { var data = this.__data__, index = assocIndexOf(data, key); if (index < 0) { return false; } var lastIndex = data.length - 1; if (index == lastIndex) { data.pop(); } else { splice.call(data, index, 1); } --this.size; return true; } /** * Gets the list cache value for `key`. * * @private * @name get * @memberOf ListCache * @param {string} key The key of the value to get. * @returns {*} Returns the entry value. */ function listCacheGet(key) { var data = this.__data__, index = assocIndexOf(data, key); return index < 0 ? undefined : data[index][1]; } /** * Checks if a list cache value for `key` exists. * * @private * @name has * @memberOf ListCache * @param {string} key The key of the entry to check. * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. */ function listCacheHas(key) { return assocIndexOf(this.__data__, key) > -1; } /** * Sets the list cache `key` to `value`. * * @private * @name set * @memberOf ListCache * @param {string} key The key of the value to set. * @param {*} value The value to set. * @returns {Object} Returns the list cache instance. */ function listCacheSet(key, value) { var data = this.__data__, index = assocIndexOf(data, key); if (index < 0) { ++this.size; data.push([key, value]); } else { data[index][1] = value; } return this; } /** * Creates an list cache object. * * @private * @constructor * @param {Array} [entries] The key-value pairs to cache. */ function ListCache(entries) { var index = -1, length = entries == null ? 0 : entries.length; this.clear(); while (++index < length) { var entry = entries[index]; this.set(entry[0], entry[1]); } } // Add methods to `ListCache`. ListCache.prototype.clear = listCacheClear; ListCache.prototype['delete'] = listCacheDelete; ListCache.prototype.get = listCacheGet; ListCache.prototype.has = listCacheHas; ListCache.prototype.set = listCacheSet; /* Built-in method references that are verified to be native. */ var Map = getNative(root, 'Map'); /** * Removes all key-value entries from the map. * * @private * @name clear * @memberOf MapCache */ function mapCacheClear() { this.size = 0; this.__data__ = { 'hash': new Hash, 'map': new (Map || ListCache), 'string': new Hash }; } /** * Checks if `value` is suitable for use as unique object key. * * @private * @param {*} value The value to check. * @returns {boolean} Returns `true` if `value` is suitable, else `false`. */ function isKeyable(value) { var type = typeof value; return (type == 'string' || type == 'number' || type == 'symbol' || type == 'boolean') ? (value !== '__proto__') : (value === null); } /** * Gets the data for `map`. * * @private * @param {Object} map The map to query. * @param {string} key The reference key. * @returns {*} Returns the map data. */ function getMapData(map, key) { var data = map.__data__; return isKeyable(key) ? data[typeof key == 'string' ? 'string' : 'hash'] : data.map; } /** * Removes `key` and its value from the map. * * @private * @name delete * @memberOf MapCache * @param {string} key The key of the value to remove. * @returns {boolean} Returns `true` if the entry was removed, else `false`. */ function mapCacheDelete(key) { var result = getMapData(this, key)['delete'](key); this.size -= result ? 1 : 0; return result; } /** * Gets the map value for `key`. * * @private * @name get * @memberOf MapCache * @param {string} key The key of the value to get. * @returns {*} Returns the entry value. */ function mapCacheGet(key) { return getMapData(this, key).get(key); } /** * Checks if a map value for `key` exists. * * @private * @name has * @memberOf MapCache * @param {string} key The key of the entry to check. * @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. */ function mapCacheHas(key) { return getMapData(this, key).has(key); } /** * Sets the map `key` to `value`. * * @private * @name set * @memberOf MapCache * @param {string} key The key of the value to set. * @param {*} value The value to set. * @returns {Object} Returns the map cache instance. */ function mapCacheSet(key, value) { var data = getMapData(this, key), size = data.size; data.set(key, value); this.size += data.size == size ? 0 : 1; return this; } /** * Creates a map cache object to store key-value pairs. * * @private * @constructor * @param {Array} [entries] The key-value pairs to cache. */ function MapCache(entries) { var index = -1, length = entries == null ? 0 : entries.length; this.clear(); while (++index < length) { var entry = entries[index]; this.set(entry[0], entry[1]); } } // Add methods to `MapCache`. MapCache.prototype.clear = mapCacheClear; MapCache.prototype['delete'] = mapCacheDelete; MapCache.prototype.get = mapCacheGet; MapCache.prototype.has = mapCacheHas; MapCache.prototype.set = mapCacheSet; /** Error message constants. */ var FUNC_ERROR_TEXT$2 = 'Expected a function'; /** * Creates a function that memoizes the result of `func`. If `resolver` is * provided, it determines the cache key for storing the result based on the * arguments provided to the memoized function. By default, the first argument * provided to the memoized function is used as the map cache key. The `func` * is invoked with the `this` binding of the memoized function. * * **Note:** The cache is exposed as the `cache` property on the memoized * function. Its creation may be customized by replacing the `_.memoize.Cache` * constructor with one whose instances implement the * [`Map`](http://ecma-international.org/ecma-262/7.0/#sec-properties-of-the-map-prototype-object) * method interface of `clear`, `delete`, `get`, `has`, and `set`. * * @static * @memberOf _ * @since 0.1.0 * @category Function * @param {Function} func The function to have its output memoized. * @param {Function} [resolver] The function to resolve the cache key. * @returns {Function} Returns the new memoized function. * @example * * var object = { 'a': 1, 'b': 2 }; * var other = { 'c': 3, 'd': 4 }; * * var values = _.memoize(_.values); * values(object); * // => [1, 2] * * values(other); * // => [3, 4] * * object.a = 2; * values(object); * // => [1, 2] * * // Modify the result cache. * values.cache.set(object, ['a', 'b']); * values(object); * // => ['a', 'b'] * * // Replace `_.memoize.Cache`. * _.memoize.Cache = WeakMap; */ function memoize(func, resolver) { if (typeof func != 'function' || (resolver != null && typeof resolver != 'function')) { throw new TypeError(FUNC_ERROR_TEXT$2); } var memoized = function() { var args = arguments, key = resolver ? resolver.apply(this, args) : args[0], cache = memoized.cache; if (cache.has(key)) { return cache.get(key); } var result = func.apply(this, args); memoized.cache = cache.set(key, result) || cache; return result; }; memoized.cache = new (memoize.Cache || MapCache); return memoized; } // Expose `MapCache`. memoize.Cache = MapCache; /** * The base implementation of `_.clamp` which doesn't coerce arguments. * * @private * @param {number} number The number to clamp. * @param {number} [lower] The lower bound. * @param {number} upper The upper bound. * @returns {number} Returns the clamped number. */ function baseClamp(number, lower, upper) { if (number === number) { if (upper !== undefined) { number = number <= upper ? number : upper; } if (lower !== undefined) { number = number >= lower ? number : lower; } } return number; } /** * Clamps `number` within the inclusive `lower` and `upper` bounds. * * @static * @memberOf _ * @since 4.0.0 * @category Number * @param {number} number The number to clamp. * @param {number} [lower] The lower bound. * @param {number} upper The upper bound. * @returns {number} Returns the clamped number. * @example * * _.clamp(-10, -5, 5); * // => -5 * * _.clamp(10, -5, 5); * // => 5 */ function clamp(number, lower, upper) { if (upper === undefined) { upper = lower; lower = undefined; } if (upper !== undefined) { upper = toNumber(upper); upper = upper === upper ? upper : 0; } if (lower !== undefined) { lower = toNumber(lower); lower = lower === lower ? lower : 0; } return baseClamp(toNumber(number), lower, upper); } /** * Gets the timestamp of the number of milliseconds that have elapsed since * the Unix epoch (1 January 1970 00:00:00 UTC). * * @static * @memberOf _ * @since 2.4.0 * @category Date * @returns {number} Returns the timestamp. * @example * * _.defer(function(stamp) { * console.log(_.now() - stamp); * }, _.now()); * // => Logs the number of milliseconds it took for the deferred invocation. */ var now = function() { return root.Date.now(); }; /** Error message constants. */ var FUNC_ERROR_TEXT$1 = 'Expected a function'; /* Built-in method references for those with the same name as other `lodash` methods. */ var nativeMax = Math.max, nativeMin = Math.min; /** * Creates a debounced function that delays invoking `func` until after `wait` * milliseconds have elapsed since the last time the debounced function was * invoked. The debounced function comes with a `cancel` method to cancel * delayed `func` invocations and a `flush` method to immediately invoke them. * Provide `options` to indicate whether `func` should be invoked on the * leading and/or trailing edge of the `wait` timeout. The `func` is invoked * with the last arguments provided to the debounced function. Subsequent * calls to the debounced function return the result of the last `func` * invocation. * * **Note:** If `leading` and `trailing` options are `true`, `func` is * invoked on the trailing edge of the timeout only if the debounced function * is invoked more than once during the `wait` timeout. * * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred * until to the next tick, similar to `setTimeout` with a timeout of `0`. * * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/) * for details over the differences between `_.debounce` and `_.throttle`. * * @static * @memberOf _ * @since 0.1.0 * @category Function * @param {Function} func The function to debounce. * @param {number} [wait=0] The number of milliseconds to delay. * @param {Object} [options={}] The options object. * @param {boolean} [options.leading=false] * Specify invoking on the leading edge of the timeout. * @param {number} [options.maxWait] * The maximum time `func` is allowed to be delayed before it's invoked. * @param {boolean} [options.trailing=true] * Specify invoking on the trailing edge of the timeout. * @returns {Function} Returns the new debounced function. * @example * * // Avoid costly calculations while the window size is in flux. * jQuery(window).on('resize', _.debounce(calculateLayout, 150)); * * // Invoke `sendMail` when clicked, debouncing subsequent calls. * jQuery(element).on('click', _.debounce(sendMail, 300, { * 'leading': true, * 'trailing': false * })); * * // Ensure `batchLog` is invoked once after 1 second of debounced calls. * var debounced = _.debounce(batchLog, 250, { 'maxWait': 1000 }); * var source = new EventSource('/stream'); * jQuery(source).on('message', debounced); * * // Cancel the trailing debounced invocation. * jQuery(window).on('popstate', debounced.cancel); */ function debounce(func, wait, options) { var lastArgs, lastThis, maxWait, result, timerId, lastCallTime, lastInvokeTime = 0, leading = false, maxing = false, trailing = true; if (typeof func != 'function') { throw new TypeError(FUNC_ERROR_TEXT$1); } wait = toNumber(wait) || 0; if (isObject(options)) { leading = !!options.leading; maxing = 'maxWait' in options; maxWait = maxing ? nativeMax(toNumber(options.maxWait) || 0, wait) : maxWait; trailing = 'trailing' in options ? !!options.trailing : trailing; } function invokeFunc(time) { var args = lastArgs, thisArg = lastThis; lastArgs = lastThis = undefined; lastInvokeTime = time; result = func.apply(thisArg, args); return result; } function leadingEdge(time) { // Reset any `maxWait` timer. lastInvokeTime = time; // Start the timer for the trailing edge. timerId = setTimeout(timerExpired, wait); // Invoke the leading edge. return leading ? invokeFunc(time) : result; } function remainingWait(time) { var timeSinceLastCall = time - lastCallTime, timeSinceLastInvoke = time - lastInvokeTime, timeWaiting = wait - timeSinceLastCall; return maxing ? nativeMin(timeWaiting, maxWait - timeSinceLastInvoke) : timeWaiting; } function shouldInvoke(time) { var timeSinceLastCall = time - lastCallTime, timeSinceLastInvoke = time - lastInvokeTime; // Either this is the first call, activity has stopped and we're at the // trailing edge, the system time has gone backwards and we're treating // it as the trailing edge, or we've hit the `maxWait` limit. return (lastCallTime === undefined || (timeSinceLastCall >= wait) || (timeSinceLastCall < 0) || (maxing && timeSinceLastInvoke >= maxWait)); } function timerExpired() { var time = now(); if (shouldInvoke(time)) { return trailingEdge(time); } // Restart the timer. timerId = setTimeout(timerExpired, remainingWait(time)); } function trailingEdge(time) { timerId = undefined; // Only invoke if we have `lastArgs` which means `func` has been // debounced at least once. if (trailing && lastArgs) { return invokeFunc(time); } lastArgs = lastThis = undefined; return result; } function cancel() { if (timerId !== undefined) { clearTimeout(timerId); } lastInvokeTime = 0; lastArgs = lastCallTime = lastThis = timerId = undefined; } function flush() { return timerId === undefined ? result : trailingEdge(now()); } function debounced() { var time = now(), isInvoking = shouldInvoke(time); lastArgs = arguments; lastThis = this; lastCallTime = time; if (isInvoking) { if (timerId === undefined) { return leadingEdge(lastCallTime); } if (maxing) { // Handle invocations in a tight loop. clearTimeout(timerId); timerId = setTimeout(timerExpired, wait); return invokeFunc(lastCallTime); } } if (timerId === undefined) { timerId = setTimeout(timerExpired, wait); } return result; } debounced.cancel = cancel; debounced.flush = flush; return debounced; } /** Error message constants. */ var FUNC_ERROR_TEXT = 'Expected a function'; /** * Creates a throttled function that only invokes `func` at most once per * every `wait` milliseconds. The throttled function comes with a `cancel` * method to cancel delayed `func` invocations and a `flush` method to * immediately invoke them. Provide `options` to indicate whether `func` * should be invoked on the leading and/or trailing edge of the `wait` * timeout. The `func` is invoked with the last arguments provided to the * throttled function. Subsequent calls to the throttled function return the * result of the last `func` invocation. * * **Note:** If `leading` and `trailing` options are `true`, `func` is * invoked on the trailing edge of the timeout only if the throttled function * is invoked more than once during the `wait` timeout. * * If `wait` is `0` and `leading` is `false`, `func` invocation is deferred * until to the next tick, similar to `setTimeout` with a timeout of `0`. * * See [David Corbacho's article](https://css-tricks.com/debouncing-throttling-explained-examples/) * for details over the differences between `_.throttle` and `_.debounce`. * * @static * @memberOf _ * @since 0.1.0 * @category Function * @param {Function} func The function to throttle. * @param {number} [wait=0] The number of milliseconds to throttle invocations to. * @param {Object} [options={}] The options object. * @param {boolean} [options.leading=true] * Specify invoking on the leading edge of the timeout. * @param {boolean} [options.trailing=true] * Specify invoking on the trailing edge of the timeout. * @returns {Function} Returns the new throttled function. * @example * * // Avoid excessively updating the position while scrolling. * jQuery(window).on('scroll', _.throttle(updatePosition, 100)); * * // Invoke `renewToken` when the click event is fired, but not more than once every 5 minutes. * var throttled = _.throttle(renewToken, 300000, { 'trailing': false }); * jQuery(element).on('click', throttled); * * // Cancel the trailing throttled invocation. * jQuery(window).on('popstate', throttled.cancel); */ function throttle(func, wait, options) { var leading = true, trailing = true; if (typeof func != 'function') { throw new TypeError(FUNC_ERROR_TEXT); } if (isObject(options)) { leading = 'leading' in options ? !!options.leading : leading; trailing = 'trailing' in options ? !!options.trailing : trailing; } return debounce(func, wait, { 'leading': leading, 'maxWait': wait, 'trailing': trailing }); } function createTest(target) { return (test) => typeof test === "function" ? test() : typeof test === "string" ? target.includes(test) || test === "*" : test.test(target); } class Runtime { constructor() { this.getTopLocationHref = memoize( async () => { if (parent === self) return window.location.href; return new Promise((resolve) => { window.addEventListener("message", function once(e) { var _a; if (((_a = e.data) == null ? void 0 : _a.key) === "getLocationHref") { window.removeEventListener("message", once); resolve(e.data.url); } }); parent.postMessage({ key: "getLocationHref" }, "*"); }); }, () => window.location.href ); this.list = [ { domains: [], opts: [], search: { name: "[BT]\u871C\u67D1\u8BA1\u5212", search: (name) => `https://mikanani.me/Home/Search?searchstr=${name}` } } ]; if (parent === self) { window.addEventListener("message", (e) => { var _a, _b; if (((_a = e.data) == null ? void 0 : _a.key) === "getLocationHref") { (_b = e.source) == null ? void 0 : _b.postMessage( { key: "getLocationHref", url: location.href }, { targetOrigin: "*" } ); } }); } } register(item) { this.list.push(item); } async getSearchActions() { const isInIframe = parent !== self; const searchs = this.list.map((o) => o.search).filter(Boolean).filter((o) => !(isInIframe && o.disabledInIframe)); const register = this.getActiveRegister(); const info = await this.getCurrentVideoNameAndEpisode(); if (!(info == null ? void 0 : info.name)) return []; let name = info.name; return searchs.filter((search) => search !== register.search && search.search).map((search) => ({ name: search.name, search: () => { const url = search.search(encodeURIComponent(name)); if (!url) return; if (isInIframe) parent.postMessage({ key: "openLink", url }, "*"); else window.open(url); } })); } async getCurrentVideoNameAndEpisode() { var _a, _b, _c, _d; const register = this.getActiveRegister(); if (!((_a = register.search) == null ? void 0 : _a.getSearchName)) return; let rawName = await register.search.getSearchName() || ""; let episode = await ((_c = (_b = register.search).getEpisode) == null ? void 0 : _c.call(_b)) || ""; let name = rawName.replace(/第.季/, "").replace(/[<>《》''‘’""“”\[\]]/g, "").trim(); episode = ((_d = episode.match(/([0-9.]+)[集话]/)) == null ? void 0 : _d[1].replace(/^0+/, "")) || episode.replace(/[第集话()()]/g, "") || episode; return { name, rawName, episode }; } getActiveRegister() { const registers = this.list.filter( ({ domains }) => domains.some(createTest(location.origin)) ); if (registers.length !== 1) { console.log(window.location, registers); throw new Error(`\u6FC0\u6D3B\u7684\u57DF\u540D\u5E94\u8BE5\u5C31\u4E00\u4E2A`); } console.log("\u6FC0\u6D3B\u7684Register", registers[0]); return registers[0]; } getActiveOpts() { const register = this.getActiveRegister(); return register.opts.filter(({ test }) => { const testArr = Array.isArray(test) ? test : [test]; return testArr.some(createTest(location.pathname + location.search)); }); } run() { let setupList = []; let runList = []; const opts = this.getActiveOpts(); opts.forEach((opt) => { const { run, setup, runInIframe } = opt; let needRun = runInIframe ? parent !== self : parent === self; if (needRun) { console.log("\u6FC0\u6D3B\u7684opt", opt); setup && setupList.push(setup); runList.push(run); } }); const init = () => { setupList.forEach((setup) => setup()); runList.forEach((run) => run()); }; if (document.readyState !== "loading") { init(); } else { window.addEventListener("DOMContentLoaded", init); } } } const runtime = new Runtime(); var css$e = "#k-player-message {\n z-index: 999;\n position: absolute;\n left: 20px;\n bottom: 60px;\n}\n#k-player-message .k-player-message-item {\n display: block;\n width: max-content;\n padding: 8px 16px;\n background: var(--k-player-background);\n border-radius: 4px;\n color: white;\n font-size: 14px;\n white-space: nowrap;\n overflow: hidden;\n box-sizing: border-box;\n margin-top: 4px;\n}\n#k-player-message .k-player-message-item:hover {\n background: var(--k-player-background-highlight);\n transition: all 0.3s;\n}"; n(css$e,{}); class Message { constructor(selector) { this.MaxLength = 5; this.$message = $('
'); this.$message.appendTo($(selector)); } info(message, ms = 1500) { if (this.$message.children().length > this.MaxLength) { this.$message.children().first().remove(); } return new Promise((resolve) => { $(`
`).append(message).hide().appendTo(this.$message).show(150).delay(ms).hide(150, function() { $(this).remove(); resolve(); }); }); } destroy() { this.$message.empty(); } } function parseTime(time = 0) { time = Math.round(time); return `${Math.floor(time / 60).toString().padStart(2, "0")}:${(time % 60).toString().padStart(2, "0")}`; } function createStorage$1(storage) { function getItem(key, defaultValue) { try { const value = storage.getItem(key); if (value) return JSON.parse(value); return defaultValue; } catch (error) { return defaultValue; } } return { getItem, setItem(key, value) { storage.setItem(key, JSON.stringify(value)); }, removeItem: storage.removeItem.bind(storage), clear: storage.clear.bind(storage) }; } const session = createStorage$1(window.sessionStorage); const local = createStorage$1(window.localStorage); let gm; try { gm = { getItem: GM_getValue, setItem: GM_setValue }; } catch (error) { gm = local; } var css$d = ".k-popover {\n position: relative;\n}\n.k-popover-overlay {\n position: absolute;\n display: none;\n bottom: 100%;\n left: 50%;\n transform: translateX(-50%);\n z-index: 100;\n padding-bottom: 20px;\n}\n.k-popover-content {\n background: var(--k-player-background);\n border-radius: 4px;\n overflow-x: hidden;\n overflow-y: auto;\n cursor: initial;\n max-height: var(--k-player-popover-max-height, 70vh);\n}\n.k-popover-content::-webkit-scrollbar {\n display: none;\n}"; n(css$d,{}); function popover(target, overlay, trigger = "hover") { const $target = $(target); const $content = $( `
` ); $content.on("click", (e) => e.stopPropagation()); $content.find(".k-popover-content").append(overlay); $target.addClass("k-popover"); $target.append($content); if (trigger === "click") { $target.on("click", () => { $content.fadeIn("fast"); $target.addClass("k-popover-active"); }); window.addEventListener("click", (e) => { if (!$target[0].contains(e.target)) { $content.fadeOut("fast"); $target.removeClass("k-popover-active"); } }); } else { let timeID; $target.on("mouseenter", () => { clearTimeout(timeID); timeID = window.setTimeout(() => { $content.fadeIn("fast"); $target.addClass("k-popover-active"); }, 100); }); $target.on("mouseleave", () => { clearTimeout(timeID); timeID = window.setTimeout(() => { $content.fadeOut("fast"); $target.removeClass("k-popover-active"); }, 100); }); } return $target; } const isMac$1 = /macintosh|mac os x/i.test(navigator.userAgent); const KeyMap = { ArrowUp: "\u2191", ArrowDown: "\u2193", ArrowLeft: "\u2190", ArrowRight: "\u2192", ctrl: "Ctrl", alt: "Alt", shift: "Shift" }; const MacKeyMap = { ctrl: "\u2303", meta: "\u2318", alt: "\u2325", shift: "\u21E7" }; if (isMac$1) { Object.assign(KeyMap, MacKeyMap); } function renderKey(key) { Object.entries(KeyMap).forEach(([k, v]) => { key = key.replace(new RegExp(k, "i"), v); }); return key; } var Commands$1 = /* @__PURE__ */ ((Commands2) => { Commands2["forward5"] = "forward5"; Commands2["backward5"] = "backward5"; Commands2["forward30"] = "forward30"; Commands2["backward30"] = "backward30"; Commands2["forward60"] = "forward60"; Commands2["backward60"] = "backward60"; Commands2["forward90"] = "forward90"; Commands2["backward90"] = "backward90"; Commands2["togglePlay"] = "togglePlay"; Commands2["next"] = "next"; Commands2["prev"] = "prev"; Commands2["toggleWidescreen"] = "toggleWidescreen"; Commands2["Escape"] = "Escape"; Commands2["restoreSpeed"] = "restoreSpeed"; Commands2["increaseSpeed"] = "increaseSpeed"; Commands2["decreaseSpeed"] = "decreaseSpeed"; Commands2["temporaryIncreaseSpeed"] = "temporaryIncreaseSpeed"; Commands2["togglePIP"] = "togglePIP"; Commands2["internal"] = "internal"; Commands2["help"] = "help"; Commands2["prevFrame"] = "prevFrame"; Commands2["nextFrame"] = "nextFrame"; Commands2["toggleFullscreen"] = "toggleFullscreen"; Commands2["decreaseVolume"] = "decreaseVolume"; Commands2["increaseVolume"] = "increaseVolume"; Commands2["toggleMute"] = "toggleMute"; Commands2["forwardCustom"] = "forwardCustom"; Commands2["backwardCustom"] = "backwardCustom"; Commands2["recordCustomSeekTime"] = "recordCustomSeekTime"; return Commands2; })(Commands$1 || {}); var __defProp$3 = Object.defineProperty; var __defProps$2 = Object.defineProperties; var __getOwnPropDescs$2 = Object.getOwnPropertyDescriptors; var __getOwnPropSymbols$3 = Object.getOwnPropertySymbols; var __hasOwnProp$3 = Object.prototype.hasOwnProperty; var __propIsEnum$3 = Object.prototype.propertyIsEnumerable; var __defNormalProp$3 = (obj, key, value) => key in obj ? __defProp$3(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value; var __spreadValues$3 = (a, b) => { for (var prop in b || (b = {})) if (__hasOwnProp$3.call(b, prop)) __defNormalProp$3(a, prop, b[prop]); if (__getOwnPropSymbols$3) for (var prop of __getOwnPropSymbols$3(b)) { if (__propIsEnum$3.call(b, prop)) __defNormalProp$3(a, prop, b[prop]); } return a; }; var __spreadProps$2 = (a, b) => __defProps$2(a, __getOwnPropDescs$2(b)); const isMac = /macintosh|mac os x/i.test(navigator.userAgent); const DefaultKeyBindings = [ { command: Commands$1.togglePlay, key: "Space", description: "\u64AD\u653E/\u6682\u505C" }, { command: Commands$1.backward5, key: "ArrowLeft", description: "\u6B65\u90005s" }, { command: Commands$1.forward5, key: "ArrowRight", description: "\u6B65\u8FDB5s" }, { command: Commands$1.backward30, key: "shift ArrowLeft", description: "\u6B65\u900030s" }, { command: Commands$1.forward30, key: "shift ArrowRight", description: "\u6B65\u8FDB30s" }, { command: Commands$1.backward60, key: "alt ArrowLeft", description: "\u6B65\u900060s" }, { command: Commands$1.forward60, key: "alt ArrowRight", description: "\u6B65\u8FDB60s" }, { command: Commands$1.backward90, key: "ctrl ArrowLeft", mac: "meta ArrowLeft", description: "\u6B65\u900090s" }, { command: Commands$1.forward90, key: "ctrl ArrowRight", mac: "meta ArrowRight", description: "\u6B65\u8FDB90s" }, { command: Commands$1.backwardCustom, key: "shift J", description: "\u6B65\u9000[\u8BB0\u5FC6\u65F6\u95F4]" }, { command: Commands$1.forwardCustom, key: "J", description: "\u6B65\u8FDB[\u8BB0\u5FC6\u65F6\u95F4]" }, { command: Commands$1.recordCustomSeekTime, key: "K", description: "\u8BB0\u5F55\u6B65\u8FDB\u65F6\u95F4" }, { command: Commands$1.prevFrame, key: "", description: "\u4E0A\u4E00\u5E27" }, { command: Commands$1.nextFrame, key: "", description: "\u4E0B\u4E00\u5E27" }, { command: Commands$1.prev, key: "P", description: "\u4E0A\u4E00\u96C6" }, { command: Commands$1.next, key: "N", description: "\u4E0B\u4E00\u96C6" }, { command: Commands$1.toggleWidescreen, key: "W", description: "\u5BBD\u5C4F" }, { command: Commands$1.toggleFullscreen, key: "F", description: "\u5168\u5C4F" }, { command: Commands$1.Escape, key: "Escape", editable: false, description: "\u9000\u51FA\u5168\u5C4F/\u5BBD\u5C4F" }, { command: Commands$1.restoreSpeed, key: "Z", description: "\u539F\u901F\u64AD\u653E" }, { command: Commands$1.decreaseSpeed, key: "X", description: "\u51CF\u901F\u64AD\u653E" }, { command: Commands$1.increaseSpeed, key: "C", description: "\u52A0\u901F\u64AD\u653E" }, { command: Commands$1.temporaryIncreaseSpeed, key: "V", description: "\u957F\u6309\u52A0\u901F" }, { command: Commands$1.togglePIP, key: "I", description: "\u753B\u4E2D\u753B" }, { command: Commands$1.increaseVolume, key: "ArrowUp", description: "\u589E\u5927\u97F3\u91CF" }, { command: Commands$1.decreaseVolume, key: "ArrowDown", description: "\u51CF\u5C0F\u97F3\u91CF" }, { command: Commands$1.toggleMute, key: "M", description: "\u5207\u6362\u9759\u97F3" }, { command: Commands$1.internal, key: "?", editable: false, description: "\u663E\u793A\u5E2E\u52A9" } ]; class KeyBindings { constructor() { this.storageKey = "user-custom-keybindings"; this.listener = []; } getCustomKeyBindings() { return gm.getItem(this.storageKey, []); } setCustomKeyBindings(keyBindings) { gm.setItem(this.storageKey, keyBindings); } registerKeyBinding(keyBinding) { DefaultKeyBindings.push(keyBinding); this.notify(); } setKeyBinding(command, key) { let customKeyBindings = this.getCustomKeyBindings(); customKeyBindings = customKeyBindings.filter((o) => o.command !== command); if (key) { customKeyBindings.push({ command, key }); } this.setCustomKeyBindings(customKeyBindings); this.notify(); } getKeyBindings() { const customKeyBindings = this.getCustomKeyBindings(); return DefaultKeyBindings.map((keyBinding) => { const customKeyBinding = customKeyBindings.find( (o) => o.command === keyBinding.command ); const nextKeyBinding = __spreadProps$2(__spreadValues$3({}, keyBinding), { originKey: "", customKey: "" }); if (isMac && nextKeyBinding.mac) { nextKeyBinding.key = nextKeyBinding.mac; } nextKeyBinding.originKey = nextKeyBinding.key; if (customKeyBinding) { nextKeyBinding.key = customKeyBinding.key; nextKeyBinding.customKey = customKeyBinding.key; } return nextKeyBinding; }); } getKeyBinding(command) { const keyBindings = this.getKeyBindings(); return keyBindings.find((o) => o.command === command); } getCommand(key) { var _a; const keyBindings = this.getKeyBindings(); return (_a = keyBindings.find((o) => o.key === key)) == null ? void 0 : _a.command; } subscribe(cb) { this.listener.push(cb); return () => { this.listener = this.listener.filter((fn) => fn !== cb); }; } notify() { this.listener.forEach((fn) => fn()); } } function normalizeKeyEvent(e) { const SPECIAL_KEY_EN = "`-=[]\\;',./~!@#$%^&*()_+{}|:\"<>?".split(""); const SPECIAL_KEY_ZH = "\xB7-=\u3010\u3011\u3001\uFF1B\u2018\uFF0C\u3002/\uFF5E\uFF01@#\xA5%\u2026&*\uFF08\uFF09\u2014+\u300C\u300D\uFF5C\uFF1A\u201C\u300A\u300B\uFF1F".split(""); let key = e.key; if (e.code === "Space") { key = "Space"; } if (/^[a-z]$/.test(key)) { key = key.toUpperCase(); } else if (SPECIAL_KEY_ZH.includes(key)) { key = SPECIAL_KEY_EN[SPECIAL_KEY_ZH.indexOf(key)]; } let keyArr = []; e.ctrlKey && keyArr.push("ctrl"); e.metaKey && keyArr.push("meta"); e.shiftKey && !SPECIAL_KEY_EN.includes(key) && keyArr.push("shift"); e.altKey && keyArr.push("alt"); if (!/Control|Meta|Shift|Alt/i.test(key)) keyArr.push(key); keyArr = [...new Set(keyArr)]; return keyArr.join(" "); } const _Shortcuts = class { constructor(player) { this.player = player; this.handleKeyEvent = (e) => { var _a; if (/input|textarea|select/i.test((_a = document.activeElement) == null ? void 0 : _a.tagName)) return; const key = normalizeKeyEvent(e); const command = _Shortcuts.keyBindings.getCommand(key); if (command) { e.preventDefault(); this.invoke(command, e); } }; window.addEventListener("keydown", this.handleKeyEvent); window.addEventListener("keyup", this.handleKeyEvent); } static registerCommand(command, keydown, keyup) { this.commands.push({ command, keydown, keyup }); } invoke(command, e) { var _a; const cmd = _Shortcuts.commands.find((cmd2) => cmd2.command === command); if (cmd) { const type = e.type === "keydown" ? "keydown" : "keyup"; (_a = cmd[type]) == null ? void 0 : _a.call(this.player, e); } } }; let Shortcuts = _Shortcuts; Shortcuts.Commands = Commands$1; Shortcuts.keyBindings = new KeyBindings(); Shortcuts.commands = []; customElements.define( "k-shortcuts-tip", class extends HTMLElement { constructor() { super(); this.node = document.createElement("span"); const shadowRoot = this.attachShadow({ mode: "open" }); shadowRoot.appendChild(this.node); this.unsubscribe = Shortcuts.keyBindings.subscribe(() => { this.renderKey(); }); this.renderKey(); } renderKey() { const command = this.getAttribute("command"); const kb = Shortcuts.keyBindings.getKeyBinding(command); if (kb) { this.node.textContent = renderKey(kb.key); } } disconnectedCallback() { this.unsubscribe(); } } ); function setup$1(player) { new Shortcuts(player); } const SHIFT_KEY = '~!@#$%^&*()_+{}|:"<>?\uFF5E\uFF01@#\xA5%\u2026&*\uFF08\uFF09\u2014\u2014+\u300C\u300D\uFF5C\uFF1A\u201C\u300A\u300B\uFF1F'; function keybind(keys, cb) { const isMac = /macintosh|mac os x/i.test(navigator.userAgent); keys = keys.filter((key) => !key.includes(isMac ? "ctrl" : "meta")); $(window).on("keydown", (e) => { var _a; if (((_a = document.activeElement) == null ? void 0 : _a.tagName) === "INPUT") return; let keyArr = []; e.ctrlKey && keyArr.push("ctrl"); e.metaKey && keyArr.push("meta"); e.shiftKey && !SHIFT_KEY.includes(e.key) && keyArr.push("shift"); e.altKey && keyArr.push("alt"); if (!["Control", "Meta", "Shift", "Alt"].includes(e.key)) { keyArr.push(e.key); } keyArr = [...new Set(keyArr)]; const key = keyArr.join("+"); if (keys.includes(key)) { cb(e.originalEvent, key); } }); } var css$c = ".k-modal {\n position: fixed;\n left: 0;\n right: 0;\n top: 0;\n bottom: 0;\n display: flex;\n align-items: center;\n justify-content: center;\n z-index: 2147483647;\n text-align: left;\n animation: fadeIn 0.3s ease forwards;\n color: rgba(0, 0, 0, 0.85);\n font-family: system-ui, -apple-system, BlinkMacSystemFont, \"Segoe UI\", Roboto, Oxygen, Ubuntu, Cantarell, \"Open Sans\", \"Helvetica Neue\", sans-serif;\n}\n@keyframes fadeIn {\n from {\n opacity: 0;\n }\n to {\n opacity: 1;\n }\n}\n.k-modal * {\n color: inherit;\n}\n.k-modal-mask {\n position: fixed;\n left: 0;\n right: 0;\n bottom: 0;\n top: 0;\n background: rgba(0, 0, 0, 0.45);\n cursor: pointer;\n}\n.k-modal-wrap {\n position: fixed;\n left: 0;\n right: 0;\n bottom: 0;\n top: 0;\n overflow: auto;\n text-align: center;\n user-select: none;\n}\n.k-modal-wrap::before {\n content: \"\";\n display: inline-block;\n width: 0;\n height: 100%;\n vertical-align: middle;\n}\n.k-modal-container {\n margin: 20px 0;\n display: inline-block;\n vertical-align: middle;\n text-align: left;\n position: relative;\n width: 520px;\n min-height: 100px;\n background: white;\n border-radius: 2px;\n user-select: text;\n}\n.k-modal-header {\n font-size: 16px;\n border-bottom: 1px solid #f1f1f1;\n display: flex;\n justify-content: space-between;\n align-items: center;\n}\n.k-modal-header-title {\n padding: 16px;\n font-weight: 500;\n}\n.k-modal-close {\n cursor: pointer;\n height: 55px;\n width: 55px;\n position: absolute;\n right: 0;\n top: 0;\n display: flex;\n justify-content: center;\n align-items: center;\n user-select: none;\n}\n.k-modal-close * {\n color: rgba(0, 0, 0, 0.45);\n transition: color 0.15s ease;\n}\n.k-modal-close:hover * {\n color: rgba(0, 0, 0, 0.85);\n}\n.k-modal-body {\n padding: 16px;\n font-size: 14px;\n}\n.k-modal-footer {\n padding: 10px 16px;\n font-size: 14px;\n border-top: 1px solid #f1f1f1;\n display: flex;\n justify-content: flex-end;\n}\n.k-modal-btn {\n user-select: none;\n display: flex;\n align-items: center;\n justify-content: center;\n height: 32px;\n line-height: 32px;\n border-radius: 2px;\n border: 1px solid #1890ff;\n background: #1890ff;\n color: white;\n min-width: 64px;\n cursor: pointer;\n padding: 0 8px;\n}"; n(css$c,{}); function modal(opts) { const { title, content, onClose, onOk, afterClose, okText = "\u786E \u5B9A" } = opts; const store = { width: document.body.style.width, overflow: document.body.style.overflow }; const ID = Math.random().toString(16).slice(2); $(` `).appendTo("body"); $("body").css({ width: `calc(100% - ${window.innerWidth - document.body.clientWidth}px)`, overflow: "hidden" }); if (title) { $(`#${ID} .k-modal-header-title`).append(title); } else $(`#${ID} .k-modal-header-title`).remove(); $(`#${ID} .k-modal-body`).append(content); $(`#${ID} .k-modal-close`).on("click", () => { handleClose(); }); $(`#${ID} .k-modal-container`).on("click", (e) => { e.stopPropagation(); }); $(`#${ID} .k-modal-wrap`).on("click", () => { handleClose(); }); function reset() { $(`#${ID}`).remove(); $("body").css(store); window.removeEventListener("keydown", fn, { capture: true }); afterClose == null ? void 0 : afterClose(); } function handleClose() { onClose == null ? void 0 : onClose(); reset(); } function handleOk() { onOk == null ? void 0 : onOk(); reset(); } function fn(e) { if (["Escape"].includes(e.key)) { e.stopPropagation(); handleClose(); } } window.addEventListener("keydown", fn, { capture: true }); if (onOk) { $(`#${ID} .k-modal-container`).append(` `); $(`#${ID} .k-modal-ok`).on("click", () => { handleOk(); }); } } var css$b = ".k-alert {\n margin-bottom: 16px;\n box-sizing: border-box;\n color: black;\n font-size: 14px;\n font-variant: tabular-nums;\n line-height: 1.5715;\n list-style: none;\n font-feature-settings: \"tnum\";\n position: relative;\n display: flex;\n align-items: center;\n padding: 8px 15px;\n word-wrap: break-word;\n border-radius: 2px;\n}\n.k-alert-icon {\n margin-right: 8px;\n display: block;\n color: var(--k-player-primary-color);\n}\n.k-alert-content {\n flex: 1;\n min-width: 0;\n}\n.k-alert-info {\n background-color: var(--k-player-primary-color-highlight);\n border: 1px solid var(--k-player-primary-color);\n}"; n(css$b,{}); function alert(html) { return `
${html}
`; } var css$a = ".k-tab {\n flex: 1;\n white-space: nowrap;\n cursor: pointer;\n text-align: center;\n padding: 8px 0;\n}\n.k-tabs {\n display: flex;\n position: relative;\n border-bottom: 1px solid rgba(255, 255, 255, 0.2);\n}\n.k-tabs-wrapper {\n text-align: left;\n overflow: hidden;\n}\n.k-tabs-wrapper * {\n box-sizing: border-box;\n}\n.k-tab-indicator {\n position: absolute;\n width: 0;\n height: 1px;\n left: 0;\n bottom: -1px;\n background-color: var(--k-player-primary-color);\n transition: all 0.3s;\n}\n.k-tabs-panes {\n display: flex;\n flex-wrap: nowrap;\n transition: all 0.3s;\n}\n.k-tab-pane {\n flex: 0 0 100%;\n width: 100%;\n padding: 8px;\n position: relative;\n}"; n(css$a,{}); function tabs(opts) { const tabsHTML = []; const tabsContentHTML = []; opts.forEach((tab, idx) => { const tabHTML = `
${tab.name}
`; const $contentHTML = $( `
` ); $contentHTML.append( typeof tab.content === "function" ? tab.content() : tab.content ); tabsHTML.push(tabHTML); tabsContentHTML.push($contentHTML); }); const $root = $(`
${tabsHTML.join("")}
`); $root.find(".k-tabs-panes").append(...tabsContentHTML); const $indicator = $root.find(".k-tab-indicator"); $root.find(".k-tab").on("click", (e) => { $root.find(".k-tab").removeClass("active"); const $tab = $(e.target).addClass("active"); const idx = parseInt($tab.attr("data-idx")); $root.find(".k-tabs-panes").css("transform", `translateX(-${idx * 100}%)`); function updateIndictor() { const width = $tab.outerWidth(); if (width) $indicator.css({ width, left: idx * width }); else requestAnimationFrame(updateIndictor); } updateIndictor(); }); $root.find(".k-tab:first").trigger("click"); return $root; } var css$9 = ".script-info .k-modal-body {\n padding: 0;\n}\n.script-info .k-modal-body * {\n box-sizing: border-box;\n font-size: 14px;\n line-height: normal;\n font-family: -apple-system, BlinkMacSystemFont, \"Segoe UI\", Roboto, Oxygen, Ubuntu, Cantarell, \"Open Sans\", \"Helvetica Neue\", sans-serif;\n}\n.script-info .k-modal-body table {\n width: 100%;\n border-spacing: 0;\n border-collapse: separate;\n}\n.script-info .k-modal-body tbody tr td:first-child {\n white-space: nowrap;\n width: 85px;\n}\n.script-info .k-modal-body th,\n.script-info .k-modal-body td {\n padding: 8px;\n border-bottom: 1px solid #f1f1f1;\n word-wrap: break-word;\n word-break: break-all;\n}\n.script-info .k-modal-body .info-title {\n font-weight: 600;\n padding-top: 24px;\n}\n.script-info .k-modal-body a {\n color: var(--k-player-primary-color);\n margin: -4px 0 -4px -8px;\n padding: 4px 8px;\n border-radius: 4px;\n text-decoration: none;\n cursor: pointer;\n display: inline-block;\n white-space: nowrap;\n}\n.script-info .k-modal-body a:hover {\n color: var(--k-player-primary-color);\n text-decoration: underline;\n background-color: var(--k-player-primary-color-highlight);\n}\n.script-info .k-modal-body .k-tabs {\n border-bottom: 1px solid #f1f1f1;\n}\n.script-info .k-modal-body .shortcuts {\n padding: 8px;\n}\n.script-info .k-modal-body .shortcuts-wrapper {\n height: 400px;\n padding: 0;\n overflow-y: scroll;\n position: relative;\n}\n.script-info .k-modal-body .shortcuts-wrapper::-webkit-scrollbar {\n width: 8px;\n}\n.script-info .k-modal-body .shortcuts-wrapper::-webkit-scrollbar-thumb {\n background: rgba(0, 0, 0, 0.15);\n border-radius: 4px;\n}\n.script-info .k-modal-body .shortcuts-wrapper::-webkit-scrollbar-thumb:hover {\n background-color: rgba(0, 0, 0, 0.45);\n}\n.script-info .k-modal-body .shortcuts th {\n position: sticky;\n background-color: white;\n top: 0;\n z-index: 1;\n}\n.script-info .k-modal-body .shortcuts .shortcuts-input-wrapper {\n display: flex;\n align-items: center;\n}\n.script-info .k-modal-body .shortcuts .k-input {\n flex: 1;\n padding: 4px 8px;\n border-radius: 4px;\n}\n.script-info .k-modal-body .shortcuts a {\n margin-left: 8px;\n}\n.script-info .k-modal-body .shortcuts .k-font-kbd {\n font-family: consolas, monospace;\n}\n.script-info .k-modal-body .feature {\n margin-bottom: 8px;\n}\n.script-info .k-modal-body .feature-title {\n font-weight: 500;\n}\n.script-info .k-modal-body .feature-description {\n color: #666;\n}"; n(css$9,{}); function genIssueURL({ title, body }) { const url = new URL( `https://github.com/IronKinoko/agefans-enhance/issues/new` ); url.searchParams.set("title", title); url.searchParams.set("body", body); return url.toString(); } const scriptInfo = (video) => { const githubIssueURL = genIssueURL({ title: "\u{1F41B}[Bug]", body: issueBody(video == null ? void 0 : video.src) }); return tabs([ { name: "\u811A\u672C\u4FE1\u606F", content: ` ${video ? `` : ""}
\u811A\u672C\u7248\u672C${"1.43.9"}
\u811A\u672C\u4F5C\u8005 IronKinoko
\u811A\u672C\u6E90\u7801 GitHub \u66F4\u65B0\u8BB0\u5F55
\u62A5\u9519/\u610F\u89C1 GitHub Issues Greasy Fork \u53CD\u9988
\u7279\u522B\u9E23\u8C22 \u5F39\u5F39play\u63D0\u4F9B\u5F39\u5E55\u670D\u52A1
\u89C6\u9891\u4FE1\u606F
\u89C6\u9891\u94FE\u63A5${video.src}
\u89C6\u9891\u4FE1\u606F${video.videoWidth} x ${video.videoHeight}
` }, { name: "\u5FEB\u6377\u952E", className: "shortcuts-wrapper", content: () => { const $root = $(`
${alert("\u81EA\u5B9A\u4E49\u6309\u952E\u7ACB\u5373\u751F\u6548\uFF0C\u8BF7\u4F7F\u7528\u82F1\u6587\u8F93\u5165\u6CD5")}
\u52A8\u4F5C \u9ED8\u8BA4\u6309\u952E \u81EA\u5B9A\u4E49
`); const keyBindings = Shortcuts.keyBindings.getKeyBindings(); keyBindings.forEach((kb) => { const $tr = $(` ${kb.description} ${renderKey(kb.originKey)} `); if (kb.editable !== false) { $tr.find("input").val(renderKey(kb.customKey)).on("keydown", function(e) { e.stopPropagation(); e.preventDefault(); const key = normalizeKeyEvent(e.originalEvent); this.value = renderKey(key); Shortcuts.keyBindings.setKeyBinding(kb.command, key); }); $tr.find("a").on("click", function(e) { $tr.find("input").val(""); Shortcuts.keyBindings.setKeyBinding(kb.command, ""); }); } else { $tr.find("td").eq(2).html("\u4E0D\u652F\u6301\u81EA\u5B9A\u4E49"); } $root.find("tbody").append($tr); }); return $root; } }, { name: "\u5B9E\u9A8C\u6027\u529F\u80FD", className: "feature-wrapper", content: () => `
${alert("\u5B9E\u9A8C\u6027\u529F\u80FD\u53EF\u80FD\u5B58\u5728\u95EE\u9898\uFF0C\u4EC5\u4F9B\u5C1D\u8BD5")}
  • \u64AD\u653E\u672C\u5730\u6587\u4EF6
    \u5C06\u672C\u5730\u6587\u4EF6\u62D6\u5165\u5230\u89C6\u9891\u533A\u57DF\uFF0C\u53EF\u4EE5\u64AD\u653E\u672C\u5730\u6587\u4EF6\uFF0C\u5E38\u7528\u4E8E\u64AD\u653E\u672C\u5730\u66F4\u9AD8\u6E05\u7684\u89C6\u9891
` } ]); }; const issueBody = (src = "") => `# \u6587\u5B57\u63CF\u8FF0 # \u7F51\u5740\u94FE\u63A5 ${window.location.href} # \u89C6\u9891\u94FE\u63A5 ${src} # \u73AF\u5883 userAgent: ${navigator.userAgent} \u811A\u672C\u7248\u672C: ${"1.43.9"} `; const GlobalKey = "show-help-info"; function help() { if (!document.fullscreenElement) { const video = $("#k-player")[0]; if (parent !== self) { parent.postMessage( { key: GlobalKey, video: video ? { src: video.currentSrc, videoWidth: video.videoWidth, videoHeight: video.videoHeight } : null }, "*" ); return; } showHelp(video); } } function showHelp(video) { if ($(".script-info").length) return; modal({ className: "script-info", title: "agefans Enhance", content: scriptInfo(video) }); } keybind(["?", "\uFF1F"], help); window.addEventListener("message", (e) => { var _a; if (((_a = e.data) == null ? void 0 : _a.key) !== GlobalKey) return; showHelp(e.data.video); }); function seekTime(duration) { return function() { this.currentTime = clamp(this.currentTime + duration, 0, this.plyr.duration); this.message.info(`\u6B65${duration < 0 ? "\u9000" : "\u8FDB"}${Math.abs(duration)}s`); }; } Shortcuts.registerCommand(Commands$1.forward5, seekTime(5)); Shortcuts.registerCommand(Commands$1.backward5, seekTime(-5)); Shortcuts.registerCommand(Commands$1.forward30, seekTime(30)); Shortcuts.registerCommand(Commands$1.backward30, seekTime(-30)); Shortcuts.registerCommand(Commands$1.forward60, seekTime(60)); Shortcuts.registerCommand(Commands$1.backward60, seekTime(-60)); Shortcuts.registerCommand(Commands$1.forward90, seekTime(90)); Shortcuts.registerCommand(Commands$1.backward90, seekTime(-90)); Shortcuts.registerCommand(Commands$1.forwardCustom, function(e) { seekTime(+this.localConfig.customSeekTime).call(this, e); }); Shortcuts.registerCommand(Commands$1.backwardCustom, function(e) { seekTime(-this.localConfig.customSeekTime).call(this, e); }); Shortcuts.registerCommand( Commands$1.recordCustomSeekTime, (() => { let start = null; return function() { if (start === null) { start = this.currentTime; this.message.info("\u5F00\u59CB\u8BB0\u5F55\u81EA\u5B9A\u4E49\u8DF3\u8F6C\u65F6\u95F4"); } else { this.configSaveToLocal( "customSeekTime", Math.abs(Math.round(this.currentTime - start)) ); this.message.info( `\u8BB0\u5F55\u6210\u529F\uFF0C\u81EA\u5B9A\u4E49\u8DF3\u8F6C\u65F6\u95F4\u4E3A${this.localConfig.customSeekTime}s` ); start = null; } }; })() ); Shortcuts.registerCommand(Commands$1.prev, function() { this.trigger("prev"); }); Shortcuts.registerCommand(Commands$1.next, function() { this.trigger("next"); }); Shortcuts.registerCommand(Commands$1.toggleWidescreen, function() { if (this.plyr.fullscreen.active) return; this.toggleWidescreen(); }); Shortcuts.registerCommand(Commands$1.togglePlay, function() { this.plyr.togglePlay(); }); Shortcuts.registerCommand(Commands$1.Escape, function() { if (this.plyr.fullscreen.active || !this.isWideScreen) return; this.toggleWidescreen(false); }); Shortcuts.registerCommand( Commands$1.restoreSpeed, (() => { let prevSpeed = 1; return function() { if (this.speed !== 1) { prevSpeed = this.speed; this.speed = 1; } else { if (this.speed !== prevSpeed) { this.speed = prevSpeed; } } }; })() ); function changeSpeed(diff) { return function() { let idx = this.speedList.indexOf(this.speed); const newIdx = clamp(idx + diff, 0, this.speedList.length - 1); if (newIdx === idx) return; const speed = this.speedList[newIdx]; this.speed = speed; }; } Shortcuts.registerCommand(Commands$1.increaseSpeed, changeSpeed(1)); Shortcuts.registerCommand(Commands$1.decreaseSpeed, changeSpeed(-1)); function createTemporaryIncreaseSpeed() { let prevSpeed = 1; let isIncreasingSpeed = false; return [ function keydown(e) { if (!e.repeat || isIncreasingSpeed) return; isIncreasingSpeed = true; prevSpeed = this.speed; this.plyr.speed = 3; this.message.info("\u500D\u901F\u64AD\u653E\u4E2D", 500); }, function keyup(e) { if (!isIncreasingSpeed) return; isIncreasingSpeed = false; this.plyr.speed = prevSpeed; } ]; } Shortcuts.registerCommand( Commands$1.temporaryIncreaseSpeed, ...createTemporaryIncreaseSpeed() ); Shortcuts.registerCommand(Commands$1.togglePIP, function() { this.plyr.pip = !this.plyr.pip; }); Shortcuts.registerCommand(Commands$1.internal, function() { }); function changeFrame(diff) { let fps = 30; let isSuspend = false; return function() { this.plyr.pause(); this.currentTime = clamp( this.currentTime + diff / fps, 0, this.plyr.duration ); if (this.localConfig.autoplay) { if (!isSuspend) { this.plyr.play = ((play) => { isSuspend = true; return () => { isSuspend = false; this.plyr.play = play; }; })(this.plyr.play); } } this.message.destroy(); this.message.info(`${diff > 0 ? "\u4E0B" : "\u4E0A"}\u4E00\u5E27`); }; } Shortcuts.registerCommand(Commands$1.prevFrame, changeFrame(-1)); Shortcuts.registerCommand(Commands$1.nextFrame, changeFrame(1)); Shortcuts.registerCommand(Commands$1.toggleFullscreen, function() { this.plyr.fullscreen.toggle(); }); Shortcuts.registerCommand(Commands$1.increaseVolume, function() { this.plyr.increaseVolume(0.05); this.message.info(`\u97F3\u91CF${Math.round(this.plyr.volume * 100)}%`); }); Shortcuts.registerCommand(Commands$1.decreaseVolume, function() { this.plyr.decreaseVolume(0.05); this.message.info(`\u97F3\u91CF${Math.round(this.plyr.volume * 100)}%`); }); Shortcuts.registerCommand(Commands$1.toggleMute, function() { this.plyr.muted = !this.plyr.muted; this.message.info(this.plyr.muted ? "\u9759\u97F3" : "\u53D6\u6D88\u9759\u97F3"); }); const icons = ` `; $("body").append(icons); const loadingHTML = ` `; const errorHTML = ` `; const pipHTML = ` `; const speedList = [0.5, 0.75, 1, 1.25, 1.5, 2, 2.5, 3, 3.5, 4]; const speedHTML = popover( `
\u500D\u901F
`, `` ); const settingsHTML = popover( ` `, `
` ); const searchActionsHTML = popover( `
\u753B\u8D28
`, `` ); const progressHTML = `
`; const i18n = { restart: "\u91CD\u64AD", rewind: "\u5FEB\u9000 {seektime}s", play: "\u64AD\u653E(\u7A7A\u683C\u952E)", pause: "\u6682\u505C(\u7A7A\u683C\u952E)", fastForward: "\u5FEB\u8FDB {seektime}s", seek: "Seek", seekLabel: "{currentTime} / {duration}", played: "\u5DF2\u64AD\u653E", buffered: "\u5DF2\u7F13\u51B2", currentTime: "\u5F53\u524D\u65F6\u95F4", duration: "\u7247\u957F", volume: "\u97F3\u91CF", mute: "\u9759\u97F3(M)", unmute: "\u53D6\u6D88\u9759\u97F3(M)", enableCaptions: "\u663E\u793A\u5B57\u5E55", disableCaptions: "\u9690\u85CF\u5B57\u5E55", download: "\u4E0B\u8F7D", enterFullscreen: "\u8FDB\u5165\u5168\u5C4F(F)", exitFullscreen: "\u9000\u51FA\u5168\u5C4F(F)", frameTitle: "\u6807\u9898\u540D\u79F0\uFF1A {title}", captions: "\u5B57\u5E55", settings: "\u8BBE\u7F6E", menuBack: "\u8FD4\u56DE\u4E0A\u7EA7", speed: "\u500D\u901F", normal: "1.0x", quality: "\u5206\u8FA8\u7387", loop: "\u5FAA\u73AF", start: "\u5F00\u59CB", end: "\u7ED3\u675F", all: "\u5168\u90E8", reset: "\u91CD\u7F6E", disabled: "\u7981\u7528", enabled: "\u542F\u7528", advertisement: "\u5E7F\u544A", qualityBadge: { 2160: "4K", 1440: "HD", 1080: "HD", 720: "HD", 576: "SD", 480: "SD" } }; var css$8 = "#k-player-wrapper {\n position: relative;\n width: 100%;\n height: 100%;\n background: #000;\n overflow: hidden;\n font-size: 14px;\n --k-player-error-background: url();\n --k-player-tsuma-length: 2;\n --plyr-line-height: 1;\n --plyr-tooltip-background: var(--k-player-background);\n --plyr-tooltip-color: var(--k-player-color);\n --plyr-range-thumb-background: url() no-repeat\n center/contain;\n --plyr-range-thumb-width: 18px;\n --plyr-range-thumb-height: 18px;\n --plyr-color-main: var(--k-player-primary-color);\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range] {\n cursor: pointer;\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range]::-webkit-slider-thumb {\n transform: scale(0);\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range]:hover::-webkit-slider-thumb {\n transform: scale(1);\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range]:active::-webkit-slider-thumb {\n transition: all 0.1s linear;\n box-shadow: none;\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range].shake-0:active::-webkit-slider-thumb {\n transform: scale(1.3) rotate(15deg);\n}\n#k-player-wrapper .plyr--full-ui.plyr--video input[type=range].shake-1:active::-webkit-slider-thumb {\n transform: scale(1.3) rotate(-15deg);\n}\n#k-player-wrapper.k-player-widescreen {\n position: fixed;\n left: 0;\n top: 0;\n z-index: 10000;\n}\n#k-player-wrapper .k-player-contianer {\n width: 100%;\n height: 100%;\n}\n#k-player-wrapper .k-player-controls-spacer {\n flex: 1;\n}\n#k-player-wrapper #k-player-loading,\n#k-player-wrapper #k-player-error {\n position: absolute;\n left: 0;\n top: 0;\n right: 0;\n bottom: 0;\n z-index: 10;\n font-size: 66px;\n color: white;\n pointer-events: none;\n background: black;\n}\n#k-player-wrapper .k-player-error-img {\n background: var(--k-player-error-background) no-repeat center/contain;\n width: 200px;\n height: 200px;\n opacity: 0.4;\n}\n#k-player-wrapper .k-player-error-info {\n text-align: center;\n padding: 24px;\n font-size: 18px;\n}\n#k-player-wrapper #k-player-pip {\n position: absolute;\n left: 0;\n top: 0;\n right: 0;\n bottom: 0;\n z-index: 10;\n pointer-events: none;\n}\n#k-player-wrapper .k-player-tsuma {\n width: 200px;\n height: 200px;\n position: absolute;\n bottom: 0;\n right: 0;\n background: no-repeat center/contain;\n opacity: 0.1;\n z-index: -1;\n pointer-events: none;\n}\n#k-player-wrapper .k-player-tsuma[data-bg-idx=\"0\"] {\n background-image: url();\n}\n#k-player-wrapper .k-player-tsuma[data-bg-idx=\"1\"] {\n background-image: url();\n}\n#k-player-wrapper .k-player-center {\n width: 100%;\n height: 100%;\n display: flex;\n flex-direction: column;\n align-items: center;\n justify-content: center;\n}\n#k-player-wrapper #k-player-header {\n transform: translateY(0);\n transition: transform 0.3s;\n position: absolute;\n top: 0;\n left: 0;\n width: 100%;\n padding: 8px;\n text-align: right;\n}\n#k-player-wrapper #k-player-header .k-player-question-icon {\n font-size: 24px;\n width: 1em;\n height: 1em;\n color: white;\n cursor: pointer;\n}\n#k-player-wrapper .plyr--hide-controls #k-player-header {\n transform: translateY(-100%);\n}\n#k-player-wrapper .plyr {\n width: 100%;\n height: 100%;\n}\n#k-player-wrapper .plyr__control svg {\n font-size: 18px;\n}\n#k-player-wrapper video {\n display: block;\n}\n#k-player-wrapper .plyr__next svg {\n transform: scale(1.7);\n}\n#k-player-wrapper .plyr__widescreen svg {\n transform: scale(1.3);\n}\n#k-player-wrapper .plyr--hide-cursor {\n cursor: none;\n}\n#k-player-wrapper .plyr__control span:not(.plyr__tooltip) {\n color: inherit;\n}\n#k-player-wrapper .plyr--hide-controls .k-player-progress {\n opacity: 1;\n transition: opacity 0.3s ease-in 0.2s;\n}\n#k-player-wrapper .k-player-fullscreen .k-player-progress,\n#k-player-wrapper .k-player-fullscreen [data-plyr=widescreen] {\n display: none;\n}\n#k-player-wrapper .k-player-progress {\n opacity: 0;\n transition: opacity 0.2s ease-out;\n height: 2px;\n width: 100%;\n position: absolute;\n bottom: 0;\n}\n#k-player-wrapper .k-player-progress .k-player-progress-current {\n position: absolute;\n left: 0;\n top: 0;\n height: 100%;\n z-index: 2;\n background-color: var(--k-player-primary-color);\n}\n#k-player-wrapper .k-player-progress .k-player-progress-buffer {\n position: absolute;\n left: 0;\n top: 0;\n z-index: 1;\n height: 100%;\n background-color: var(--plyr-video-progress-buffered-background, rgba(255, 255, 255, 0.25));\n}\n#k-player-wrapper .plyr__controls {\n z-index: 20;\n}\n#k-player-wrapper .plyr__controls .plyr__controls__item:first-child {\n margin-right: 0;\n}\n#k-player-wrapper .plyr__controls .plyr__controls__item.plyr__progress__container {\n position: absolute;\n top: 15px;\n left: 10px;\n right: 10px;\n --plyr-range-track-height: 2px;\n}\n#k-player-wrapper .plyr__controls .plyr__controls__item.plyr__progress__container:hover {\n --plyr-range-track-height: 4px;\n}\n#k-player-wrapper .plyr__controls .k-text-btn {\n display: inline-block;\n padding: 0 8px;\n text-align: center;\n}\n#k-player-wrapper .plyr__controls .k-text-btn-text {\n line-height: 32px;\n user-select: none;\n}\n@media (max-width: 576px) {\n #k-player-wrapper .plyr__controls {\n padding-top: 30px;\n }\n #k-player-wrapper [data-plyr=pip],\n #k-player-wrapper [data-plyr=widescreen],\n #k-player-wrapper .plyr__volume {\n display: none;\n }\n}\n\n.lds-spinner {\n color: official;\n display: inline-block;\n position: relative;\n width: 80px;\n height: 80px;\n}\n.lds-spinner div {\n transform-origin: 40px 40px;\n animation: lds-spinner 1.2s linear infinite;\n}\n.lds-spinner div::after {\n content: \" \";\n display: block;\n position: absolute;\n top: 3px;\n left: 37px;\n width: 6px;\n height: 18px;\n border-radius: 20%;\n background: #fff;\n}\n.lds-spinner div:nth-child(1) {\n transform: rotate(0deg);\n animation-delay: -1.1s;\n}\n.lds-spinner div:nth-child(2) {\n transform: rotate(30deg);\n animation-delay: -1s;\n}\n.lds-spinner div:nth-child(3) {\n transform: rotate(60deg);\n animation-delay: -0.9s;\n}\n.lds-spinner div:nth-child(4) {\n transform: rotate(90deg);\n animation-delay: -0.8s;\n}\n.lds-spinner div:nth-child(5) {\n transform: rotate(120deg);\n animation-delay: -0.7s;\n}\n.lds-spinner div:nth-child(6) {\n transform: rotate(150deg);\n animation-delay: -0.6s;\n}\n.lds-spinner div:nth-child(7) {\n transform: rotate(180deg);\n animation-delay: -0.5s;\n}\n.lds-spinner div:nth-child(8) {\n transform: rotate(210deg);\n animation-delay: -0.4s;\n}\n.lds-spinner div:nth-child(9) {\n transform: rotate(240deg);\n animation-delay: -0.3s;\n}\n.lds-spinner div:nth-child(10) {\n transform: rotate(270deg);\n animation-delay: -0.2s;\n}\n.lds-spinner div:nth-child(11) {\n transform: rotate(300deg);\n animation-delay: -0.1s;\n}\n.lds-spinner div:nth-child(12) {\n transform: rotate(330deg);\n animation-delay: 0s;\n}\n\n@keyframes lds-spinner {\n 0% {\n opacity: 1;\n }\n 100% {\n opacity: 0;\n }\n}"; n(css$8,{}); var __defProp$2 = Object.defineProperty; var __getOwnPropSymbols$2 = Object.getOwnPropertySymbols; var __hasOwnProp$2 = Object.prototype.hasOwnProperty; var __propIsEnum$2 = Object.prototype.propertyIsEnumerable; var __defNormalProp$2 = (obj, key, value) => key in obj ? __defProp$2(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value; var __spreadValues$2 = (a, b) => { for (var prop in b || (b = {})) if (__hasOwnProp$2.call(b, prop)) __defNormalProp$2(a, prop, b[prop]); if (__getOwnPropSymbols$2) for (var prop of __getOwnPropSymbols$2(b)) { if (__propIsEnum$2.call(b, prop)) __defNormalProp$2(a, prop, b[prop]); } return a; }; const MediaErrorMessage = { 1: "\u4F60\u4E2D\u6B62\u4E86\u5A92\u4F53\u64AD\u653E", 2: "\u7F51\u7EDC\u9519\u8BEF", 3: "\u6587\u4EF6\u635F\u574F", 4: "\u8D44\u6E90\u6709\u95EE\u9898\u770B\u4E0D\u4E86", 5: "\u8D44\u6E90\u88AB\u52A0\u5BC6\u4E86" }; const defaultConfig = { customSeekTime: 89, speed: 1, continuePlay: true, autoNext: true, showProgress: true, volume: 1, showSearchActions: true, autoplay: true, showPlayLarge: false }; const _KPlayer = class { constructor(selector, opts = {}) { this.isHoverControls = false; this.speedList = speedList; this.setCurrentTimeLogThrottled = throttle(() => { if (this.currentTime > 3) this.setCurrentTimeLog(); }, 1e3); this.hideControlsDebounced = debounce(() => { const dom = document.querySelector(".plyr"); if (!this.isHoverControls) dom == null ? void 0 : dom.classList.add("plyr--hide-controls"); }, 1e3); this.hideCursorDebounced = debounce(() => { const dom = document.querySelector(".plyr"); dom == null ? void 0 : dom.classList.add("plyr--hide-cursor"); }, 1e3); this.isJumped = false; this.opts = opts; this.$wrapper = $('
').replaceAll(selector); this.$loading = $(loadingHTML); this.$error = $(errorHTML); this.$pip = $(pipHTML); this.$video = (opts.video ? $(opts.video) : $("