3004 lines
248 KiB
JavaScript
3004 lines
248 KiB
JavaScript
/******/ (function(modules) { // webpackBootstrap
|
||
/******/ // The module cache
|
||
/******/ var installedModules = {};
|
||
/******/
|
||
/******/ // The require function
|
||
/******/ function __webpack_require__(moduleId) {
|
||
/******/
|
||
/******/ // Check if module is in cache
|
||
/******/ if(installedModules[moduleId]) {
|
||
/******/ return installedModules[moduleId].exports;
|
||
/******/ }
|
||
/******/ // Create a new module (and put it into the cache)
|
||
/******/ var module = installedModules[moduleId] = {
|
||
/******/ i: moduleId,
|
||
/******/ l: false,
|
||
/******/ exports: {}
|
||
/******/ };
|
||
/******/
|
||
/******/ // Execute the module function
|
||
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
||
/******/
|
||
/******/ // Flag the module as loaded
|
||
/******/ module.l = true;
|
||
/******/
|
||
/******/ // Return the exports of the module
|
||
/******/ return module.exports;
|
||
/******/ }
|
||
/******/
|
||
/******/
|
||
/******/ // expose the modules object (__webpack_modules__)
|
||
/******/ __webpack_require__.m = modules;
|
||
/******/
|
||
/******/ // expose the module cache
|
||
/******/ __webpack_require__.c = installedModules;
|
||
/******/
|
||
/******/ // define getter function for harmony exports
|
||
/******/ __webpack_require__.d = function(exports, name, getter) {
|
||
/******/ if(!__webpack_require__.o(exports, name)) {
|
||
/******/ Object.defineProperty(exports, name, { enumerable: true, get: getter });
|
||
/******/ }
|
||
/******/ };
|
||
/******/
|
||
/******/ // define __esModule on exports
|
||
/******/ __webpack_require__.r = function(exports) {
|
||
/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
|
||
/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
|
||
/******/ }
|
||
/******/ Object.defineProperty(exports, '__esModule', { value: true });
|
||
/******/ };
|
||
/******/
|
||
/******/ // create a fake namespace object
|
||
/******/ // mode & 1: value is a module id, require it
|
||
/******/ // mode & 2: merge all properties of value into the ns
|
||
/******/ // mode & 4: return value when already ns object
|
||
/******/ // mode & 8|1: behave like require
|
||
/******/ __webpack_require__.t = function(value, mode) {
|
||
/******/ if(mode & 1) value = __webpack_require__(value);
|
||
/******/ if(mode & 8) return value;
|
||
/******/ if((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;
|
||
/******/ var ns = Object.create(null);
|
||
/******/ __webpack_require__.r(ns);
|
||
/******/ Object.defineProperty(ns, 'default', { enumerable: true, value: value });
|
||
/******/ if(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));
|
||
/******/ return ns;
|
||
/******/ };
|
||
/******/
|
||
/******/ // getDefaultExport function for compatibility with non-harmony modules
|
||
/******/ __webpack_require__.n = function(module) {
|
||
/******/ var getter = module && module.__esModule ?
|
||
/******/ function getDefault() { return module['default']; } :
|
||
/******/ function getModuleExports() { return module; };
|
||
/******/ __webpack_require__.d(getter, 'a', getter);
|
||
/******/ return getter;
|
||
/******/ };
|
||
/******/
|
||
/******/ // Object.prototype.hasOwnProperty.call
|
||
/******/ __webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };
|
||
/******/
|
||
/******/ // __webpack_public_path__
|
||
/******/ __webpack_require__.p = "";
|
||
/******/
|
||
/******/
|
||
/******/ // Load entry module and return exports
|
||
/******/ return __webpack_require__(__webpack_require__.s = "./content/index.js");
|
||
/******/ })
|
||
/************************************************************************/
|
||
/******/ ({
|
||
|
||
/***/ "../node_modules/css-loader/index.js!../node_modules/stylus-loader/index.js!./stylus/content.styl":
|
||
/*!**************************************************************************************!*\
|
||
!*** ../node_modules/css-loader!../node_modules/stylus-loader!./stylus/content.styl ***!
|
||
\**************************************************************************************/
|
||
/*! no static exports found */
|
||
/***/ (function(module, exports, __webpack_require__) {
|
||
|
||
exports = module.exports = __webpack_require__(/*! ../../node_modules/css-loader/lib/css-base.js */ "../node_modules/css-loader/lib/css-base.js")(false);
|
||
// imports
|
||
|
||
|
||
// module
|
||
exports.push([module.i, ".chat-color-0 #content #author-name {\n color: #536dfe !important;\n}\n.chat-color-1 #content #author-name {\n color: #2196f3 !important;\n}\n.chat-color-2 #content #author-name {\n color: #03a9f4 !important;\n}\n.chat-color-3 #content #author-name {\n color: #00bcd4 !important;\n}\n.chat-color-4 #content #author-name {\n color: #009688 !important;\n}\n.chat-color-5 #content #author-name {\n color: #4caf50 !important;\n}\n.chat-color-6 #content #author-name {\n color: #8bc34a !important;\n}\n.chat-color-7 #content #author-name {\n color: #cddc39 !important;\n}\n.chat-color-8 #content #author-name {\n color: #ffeb3b !important;\n}\n.chat-color-9 #content #author-name {\n color: #ffc107 !important;\n}\n.chat-color-A #content #author-name {\n color: #f44336 !important;\n}\n.chat-color-B #content #author-name {\n color: #e91e63 !important;\n}\n.chat-color-C #content #author-name {\n color: #9c27b0 !important;\n}\n.chat-color-D #content #author-name {\n color: #673ab7 !important;\n}\n.chat-color-E #content #author-name {\n color: #536dfe !important;\n}\n.chat-color-F #content #author-name {\n color: #2196f3 !important;\n}\n.chat-color-G #content #author-name {\n color: #03a9f4 !important;\n}\n.chat-color-H #content #author-name {\n color: #00bcd4 !important;\n}\n.chat-color-I #content #author-name {\n color: #009688 !important;\n}\n.chat-color-J #content #author-name {\n color: #4caf50 !important;\n}\n.chat-color-K #content #author-name {\n color: #8bc34a !important;\n}\n.chat-color-L #content #author-name {\n color: #cddc39 !important;\n}\n.chat-color-M #content #author-name {\n color: #ffeb3b !important;\n}\n.chat-color-N #content #author-name {\n color: #ffc107 !important;\n}\n.chat-color-O #content #author-name {\n color: #ff9800 !important;\n}\n.chat-color-P #content #author-name {\n color: #ff5722 !important;\n}\n.chat-color-Q #content #author-name {\n color: #f44336 !important;\n}\n.chat-color-R #content #author-name {\n color: #e91e63 !important;\n}\n.chat-color-S #content #author-name {\n color: #9c27b0 !important;\n}\n.chat-color-T #content #author-name {\n color: #673ab7 !important;\n}\n.chat-color-U #content #author-name {\n color: #536dfe !important;\n}\n.chat-color-V #content #author-name {\n color: #2196f3 !important;\n}\n.chat-color-W #content #author-name {\n color: #03a9f4 !important;\n}\n.chat-color-X #content #author-name {\n color: #00bcd4 !important;\n}\n.chat-color-Y #content #author-name {\n color: #009688 !important;\n}\n.chat-color-Z #content #author-name {\n color: #4caf50 !important;\n}\n.chat-color-a #content #author-name {\n color: #8bc34a !important;\n}\n.chat-color-b #content #author-name {\n color: #cddc39 !important;\n}\n.chat-color-c #content #author-name {\n color: #ffeb3b !important;\n}\n.chat-color-d #content #author-name {\n color: #ffc107 !important;\n}\n.chat-color-e #content #author-name {\n color: #ff9800 !important;\n}\n.chat-color-f #content #author-name {\n color: #ff5722 !important;\n}\n.chat-color-g #content #author-name {\n color: #f44336 !important;\n}\n.chat-color-h #content #author-name {\n color: #e91e63 !important;\n}\n.chat-color-i #content #author-name {\n color: #9c27b0 !important;\n}\n.chat-color-j #content #author-name {\n color: #673ab7 !important;\n}\n.chat-color-k #content #author-name {\n color: #536dfe !important;\n}\n.chat-color-l #content #author-name {\n color: #2196f3 !important;\n}\n.chat-color-m #content #author-name {\n color: #03a9f4 !important;\n}\n.chat-color-n #content #author-name {\n color: #00bcd4 !important;\n}\n.chat-color-o #content #author-name {\n color: #009688 !important;\n}\n.chat-color-p #content #author-name {\n color: #4caf50 !important;\n}\n.chat-color-q #content #author-name {\n color: #8bc34a !important;\n}\n.chat-color-r #content #author-name {\n color: #cddc39 !important;\n}\n.chat-color-s #content #author-name {\n color: #ffeb3b !important;\n}\n.chat-color-t #content #author-name {\n color: #ffc107 !important;\n}\n.chat-color-u #content #author-name {\n color: #ff9800 !important;\n}\n.chat-color-v #content #author-name {\n color: #ff5722 !important;\n}\n.chat-color-w #content #author-name {\n color: #f44336 !important;\n}\n.chat-color-x #content #author-name {\n color: #e91e63 !important;\n}\n.chat-color-y #content #author-name {\n color: #9c27b0 !important;\n}\n.chat-color-z #content #author-name {\n color: #673ab7 !important;\n}\n.chat-color-- #content #author-name {\n color: #ff9800 !important;\n}\n.chat-color-_ #content #author-name {\n color: #ff5722 !important;\n}\n.AuthorFix #author-photo.yt-live-chat-text-message-renderer img.yt-img-shadow {\n display: inline-block;\n vertical-align: middle;\n border-radius: 50%;\n height: 1.5em;\n width: 1.5em;\n vertical-align: sub;\n}\n.AuthorFix #content {\n display: inline-block;\n vertical-align: middle;\n}\n.AuthorFix #timestamp.yt-live-chat-text-message-renderer {\n font-size: 1em;\n display: inline-block;\n vertical-align: middle;\n}\n.showTimeStamp #timestamp.yt-live-chat-text-message-renderer {\n display: inline-block;\n}\n.Emote img {\n height: 1.75em;\n align-self: center;\n vertical-align: sub;\n display: inline-block;\n vertical-align: middle;\n}\n.Emote img:hover {\n background-size: 100% 100%;\n background-color: rgba(255,255,255,0.6);\n}\n.hideElement {\n display: none !important;\n}\n.set-background-color-one {\n background-color: #303030 !important;\n}\n.set-background-color-two {\n background-color: transparent !important;\n}\n.setTwitchColors {\n text-shadow: 0 0 1px #000, 0 0 2px #000 !important;\n background: #18181b !important;\n font-family: 'Roboto' !important;\n font-size: 1.3rem !important;\n line-height: 1.5em !important;\n color: #fafafa !important;\n}\n.setTwitchColors #timestamp.yt-live-chat-text-message-renderer {\n display: none;\n}\n.setTwitchColors yt-live-chat-author-chip[is-highlighted] #author-name.owner.yt-live-chat-author-chip,\n.setTwitchColors #author-name.owner.yt-live-chat-author-chip {\n background-color: transparent;\n color: #008000;\n}\n.setTwitchColors yt-live-chat-author-chip[is-highlighted] #author-name.yt-live-chat-author-chip {\n background-color: transparent;\n}\n.emoteDivider {\n width: 60vw;\n border: 2px solid #d3d3d3;\n border-radius: 5px;\n margin-top: 2%;\n margin-bottom: 2%;\n margin-left: auto;\n margin-right: auto;\n}\n.emotePopUpText {\n margin-bottom: 2%;\n}\n.popup {\n background-color: #202020;\n position: absolute;\n top: 15%;\n left: 0;\n right: 0;\n margin-left: auto;\n margin-right: auto;\n height: 70%;\n width: 75%;\n z-index: 999;\n text-align: center;\n border-radius: 5px;\n border: #303030 1px solid;\n font-size: 1em;\n overflow: hidden;\n overflow-y: scroll;\n padding: 1%;\n padding-top: 2%;\n}\n.emoteButton {\n background-color: rgba(255,255,255,0.1);\n background-image: url(\"https://cdn.frankerfacez.com/emoticon/447885/4\");\n background-repeat: no-repeat;\n background-position: center;\n background-size: 80% 80%;\n width: var(--yt-live-chat-32px-icon-button_-_width);\n height: var(--yt-live-chat-32px-icon-button_-_height);\n padding: var(--yt-live-chat-32px-icon-button_-_padding);\n border-radius: 10px;\n border: none;\n cursor: pointer;\n}\n.emoteButton:hover {\n background-color: rgba(255,255,255,0.8);\n}\n.emoteButton:focus {\n outline: 0;\n}\n.bodyFix {\n height: 100%;\n margin: 0;\n overflow: hidden;\n}\n.theater_wrapper_fix {\n height: 100vh;\n width: 100vw;\n}\n.movie_player_fix {\n max-height: none;\n}\n.movie_player_fix video {\n width: calc(100vw - 25vw) !important;\n height: 100vh !important;\n}\n.movie_player_fix .ytp-title {\n color: rgba(255,255,255,0.8) !important;\n}\n.movie_player_fix .ytp-gradient-top {\n max-width: calc(100vw - 25vw);\n}\n.movie_player_fix .ytp-chrome-bottom {\n width: calc(100vw - 25vw) !important;\n left: 0 !important;\n}\n.movie_player_fix .html5-endscreen {\n width: calc(100vw - 25vw) !important;\n}\n.movie_player_fix .ytp-chapter-hover-container {\n width: calc(100vw - 25vw) !important;\n}\n.chat_frame_fix {\n height: 100vh;\n width: calc(100vw - 75vw);\n position: absolute;\n right: 0px;\n top: 0px;\n}\n", ""]);
|
||
|
||
// exports
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "../node_modules/css-loader/lib/css-base.js":
|
||
/*!**************************************************!*\
|
||
!*** ../node_modules/css-loader/lib/css-base.js ***!
|
||
\**************************************************/
|
||
/*! no static exports found */
|
||
/***/ (function(module, exports) {
|
||
|
||
/*
|
||
MIT License http://www.opensource.org/licenses/mit-license.php
|
||
Author Tobias Koppers @sokra
|
||
*/
|
||
// css base code, injected by the css-loader
|
||
module.exports = function(useSourceMap) {
|
||
var list = [];
|
||
|
||
// return the list of modules as css string
|
||
list.toString = function toString() {
|
||
return this.map(function (item) {
|
||
var content = cssWithMappingToString(item, useSourceMap);
|
||
if(item[2]) {
|
||
return "@media " + item[2] + "{" + content + "}";
|
||
} else {
|
||
return content;
|
||
}
|
||
}).join("");
|
||
};
|
||
|
||
// import a list of modules into the list
|
||
list.i = function(modules, mediaQuery) {
|
||
if(typeof modules === "string")
|
||
modules = [[null, modules, ""]];
|
||
var alreadyImportedModules = {};
|
||
for(var i = 0; i < this.length; i++) {
|
||
var id = this[i][0];
|
||
if(typeof id === "number")
|
||
alreadyImportedModules[id] = true;
|
||
}
|
||
for(i = 0; i < modules.length; i++) {
|
||
var item = modules[i];
|
||
// skip already imported module
|
||
// this implementation is not 100% perfect for weird media query combinations
|
||
// when a module is imported multiple times with different media queries.
|
||
// I hope this will never occur (Hey this way we have smaller bundles)
|
||
if(typeof item[0] !== "number" || !alreadyImportedModules[item[0]]) {
|
||
if(mediaQuery && !item[2]) {
|
||
item[2] = mediaQuery;
|
||
} else if(mediaQuery) {
|
||
item[2] = "(" + item[2] + ") and (" + mediaQuery + ")";
|
||
}
|
||
list.push(item);
|
||
}
|
||
}
|
||
};
|
||
return list;
|
||
};
|
||
|
||
function cssWithMappingToString(item, useSourceMap) {
|
||
var content = item[1] || '';
|
||
var cssMapping = item[3];
|
||
if (!cssMapping) {
|
||
return content;
|
||
}
|
||
|
||
if (useSourceMap && typeof btoa === 'function') {
|
||
var sourceMapping = toComment(cssMapping);
|
||
var sourceURLs = cssMapping.sources.map(function (source) {
|
||
return '/*# sourceURL=' + cssMapping.sourceRoot + source + ' */'
|
||
});
|
||
|
||
return [content].concat(sourceURLs).concat([sourceMapping]).join('\n');
|
||
}
|
||
|
||
return [content].join('\n');
|
||
}
|
||
|
||
// Adapted from convert-source-map (MIT)
|
||
function toComment(sourceMap) {
|
||
// eslint-disable-next-line no-undef
|
||
var base64 = btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap))));
|
||
var data = 'sourceMappingURL=data:application/json;charset=utf-8;base64,' + base64;
|
||
|
||
return '/*# ' + data + ' */';
|
||
}
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "../node_modules/events/events.js":
|
||
/*!****************************************!*\
|
||
!*** ../node_modules/events/events.js ***!
|
||
\****************************************/
|
||
/*! no static exports found */
|
||
/***/ (function(module, exports, __webpack_require__) {
|
||
|
||
"use strict";
|
||
// Copyright Joyent, Inc. and other Node contributors.
|
||
//
|
||
// Permission is hereby granted, free of charge, to any person obtaining a
|
||
// copy of this software and associated documentation files (the
|
||
// "Software"), to deal in the Software without restriction, including
|
||
// without limitation the rights to use, copy, modify, merge, publish,
|
||
// distribute, sublicense, and/or sell copies of the Software, and to permit
|
||
// persons to whom the Software is furnished to do so, subject to the
|
||
// following conditions:
|
||
//
|
||
// The above copyright notice and this permission notice shall be included
|
||
// in all copies or substantial portions of the Software.
|
||
//
|
||
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
||
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
|
||
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
|
||
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
||
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
|
||
// USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||
|
||
|
||
|
||
var R = typeof Reflect === 'object' ? Reflect : null
|
||
var ReflectApply = R && typeof R.apply === 'function'
|
||
? R.apply
|
||
: function ReflectApply(target, receiver, args) {
|
||
return Function.prototype.apply.call(target, receiver, args);
|
||
}
|
||
|
||
var ReflectOwnKeys
|
||
if (R && typeof R.ownKeys === 'function') {
|
||
ReflectOwnKeys = R.ownKeys
|
||
} else if (Object.getOwnPropertySymbols) {
|
||
ReflectOwnKeys = function ReflectOwnKeys(target) {
|
||
return Object.getOwnPropertyNames(target)
|
||
.concat(Object.getOwnPropertySymbols(target));
|
||
};
|
||
} else {
|
||
ReflectOwnKeys = function ReflectOwnKeys(target) {
|
||
return Object.getOwnPropertyNames(target);
|
||
};
|
||
}
|
||
|
||
function ProcessEmitWarning(warning) {
|
||
if (console && console.warn) console.warn(warning);
|
||
}
|
||
|
||
var NumberIsNaN = Number.isNaN || function NumberIsNaN(value) {
|
||
return value !== value;
|
||
}
|
||
|
||
function EventEmitter() {
|
||
EventEmitter.init.call(this);
|
||
}
|
||
module.exports = EventEmitter;
|
||
module.exports.once = once;
|
||
|
||
// Backwards-compat with node 0.10.x
|
||
EventEmitter.EventEmitter = EventEmitter;
|
||
|
||
EventEmitter.prototype._events = undefined;
|
||
EventEmitter.prototype._eventsCount = 0;
|
||
EventEmitter.prototype._maxListeners = undefined;
|
||
|
||
// By default EventEmitters will print a warning if more than 10 listeners are
|
||
// added to it. This is a useful default which helps finding memory leaks.
|
||
var defaultMaxListeners = 10;
|
||
|
||
function checkListener(listener) {
|
||
if (typeof listener !== 'function') {
|
||
throw new TypeError('The "listener" argument must be of type Function. Received type ' + typeof listener);
|
||
}
|
||
}
|
||
|
||
Object.defineProperty(EventEmitter, 'defaultMaxListeners', {
|
||
enumerable: true,
|
||
get: function() {
|
||
return defaultMaxListeners;
|
||
},
|
||
set: function(arg) {
|
||
if (typeof arg !== 'number' || arg < 0 || NumberIsNaN(arg)) {
|
||
throw new RangeError('The value of "defaultMaxListeners" is out of range. It must be a non-negative number. Received ' + arg + '.');
|
||
}
|
||
defaultMaxListeners = arg;
|
||
}
|
||
});
|
||
|
||
EventEmitter.init = function() {
|
||
|
||
if (this._events === undefined ||
|
||
this._events === Object.getPrototypeOf(this)._events) {
|
||
this._events = Object.create(null);
|
||
this._eventsCount = 0;
|
||
}
|
||
|
||
this._maxListeners = this._maxListeners || undefined;
|
||
};
|
||
|
||
// Obviously not all Emitters should be limited to 10. This function allows
|
||
// that to be increased. Set to zero for unlimited.
|
||
EventEmitter.prototype.setMaxListeners = function setMaxListeners(n) {
|
||
if (typeof n !== 'number' || n < 0 || NumberIsNaN(n)) {
|
||
throw new RangeError('The value of "n" is out of range. It must be a non-negative number. Received ' + n + '.');
|
||
}
|
||
this._maxListeners = n;
|
||
return this;
|
||
};
|
||
|
||
function _getMaxListeners(that) {
|
||
if (that._maxListeners === undefined)
|
||
return EventEmitter.defaultMaxListeners;
|
||
return that._maxListeners;
|
||
}
|
||
|
||
EventEmitter.prototype.getMaxListeners = function getMaxListeners() {
|
||
return _getMaxListeners(this);
|
||
};
|
||
|
||
EventEmitter.prototype.emit = function emit(type) {
|
||
var args = [];
|
||
for (var i = 1; i < arguments.length; i++) args.push(arguments[i]);
|
||
var doError = (type === 'error');
|
||
|
||
var events = this._events;
|
||
if (events !== undefined)
|
||
doError = (doError && events.error === undefined);
|
||
else if (!doError)
|
||
return false;
|
||
|
||
// If there is no 'error' event listener then throw.
|
||
if (doError) {
|
||
var er;
|
||
if (args.length > 0)
|
||
er = args[0];
|
||
if (er instanceof Error) {
|
||
// Note: The comments on the `throw` lines are intentional, they show
|
||
// up in Node's output if this results in an unhandled exception.
|
||
throw er; // Unhandled 'error' event
|
||
}
|
||
// At least give some kind of context to the user
|
||
var err = new Error('Unhandled error.' + (er ? ' (' + er.message + ')' : ''));
|
||
err.context = er;
|
||
throw err; // Unhandled 'error' event
|
||
}
|
||
|
||
var handler = events[type];
|
||
|
||
if (handler === undefined)
|
||
return false;
|
||
|
||
if (typeof handler === 'function') {
|
||
ReflectApply(handler, this, args);
|
||
} else {
|
||
var len = handler.length;
|
||
var listeners = arrayClone(handler, len);
|
||
for (var i = 0; i < len; ++i)
|
||
ReflectApply(listeners[i], this, args);
|
||
}
|
||
|
||
return true;
|
||
};
|
||
|
||
function _addListener(target, type, listener, prepend) {
|
||
var m;
|
||
var events;
|
||
var existing;
|
||
|
||
checkListener(listener);
|
||
|
||
events = target._events;
|
||
if (events === undefined) {
|
||
events = target._events = Object.create(null);
|
||
target._eventsCount = 0;
|
||
} else {
|
||
// To avoid recursion in the case that type === "newListener"! Before
|
||
// adding it to the listeners, first emit "newListener".
|
||
if (events.newListener !== undefined) {
|
||
target.emit('newListener', type,
|
||
listener.listener ? listener.listener : listener);
|
||
|
||
// Re-assign `events` because a newListener handler could have caused the
|
||
// this._events to be assigned to a new object
|
||
events = target._events;
|
||
}
|
||
existing = events[type];
|
||
}
|
||
|
||
if (existing === undefined) {
|
||
// Optimize the case of one listener. Don't need the extra array object.
|
||
existing = events[type] = listener;
|
||
++target._eventsCount;
|
||
} else {
|
||
if (typeof existing === 'function') {
|
||
// Adding the second element, need to change to array.
|
||
existing = events[type] =
|
||
prepend ? [listener, existing] : [existing, listener];
|
||
// If we've already got an array, just append.
|
||
} else if (prepend) {
|
||
existing.unshift(listener);
|
||
} else {
|
||
existing.push(listener);
|
||
}
|
||
|
||
// Check for listener leak
|
||
m = _getMaxListeners(target);
|
||
if (m > 0 && existing.length > m && !existing.warned) {
|
||
existing.warned = true;
|
||
// No error code for this since it is a Warning
|
||
// eslint-disable-next-line no-restricted-syntax
|
||
var w = new Error('Possible EventEmitter memory leak detected. ' +
|
||
existing.length + ' ' + String(type) + ' listeners ' +
|
||
'added. Use emitter.setMaxListeners() to ' +
|
||
'increase limit');
|
||
w.name = 'MaxListenersExceededWarning';
|
||
w.emitter = target;
|
||
w.type = type;
|
||
w.count = existing.length;
|
||
ProcessEmitWarning(w);
|
||
}
|
||
}
|
||
|
||
return target;
|
||
}
|
||
|
||
EventEmitter.prototype.addListener = function addListener(type, listener) {
|
||
return _addListener(this, type, listener, false);
|
||
};
|
||
|
||
EventEmitter.prototype.on = EventEmitter.prototype.addListener;
|
||
|
||
EventEmitter.prototype.prependListener =
|
||
function prependListener(type, listener) {
|
||
return _addListener(this, type, listener, true);
|
||
};
|
||
|
||
function onceWrapper() {
|
||
if (!this.fired) {
|
||
this.target.removeListener(this.type, this.wrapFn);
|
||
this.fired = true;
|
||
if (arguments.length === 0)
|
||
return this.listener.call(this.target);
|
||
return this.listener.apply(this.target, arguments);
|
||
}
|
||
}
|
||
|
||
function _onceWrap(target, type, listener) {
|
||
var state = { fired: false, wrapFn: undefined, target: target, type: type, listener: listener };
|
||
var wrapped = onceWrapper.bind(state);
|
||
wrapped.listener = listener;
|
||
state.wrapFn = wrapped;
|
||
return wrapped;
|
||
}
|
||
|
||
EventEmitter.prototype.once = function once(type, listener) {
|
||
checkListener(listener);
|
||
this.on(type, _onceWrap(this, type, listener));
|
||
return this;
|
||
};
|
||
|
||
EventEmitter.prototype.prependOnceListener =
|
||
function prependOnceListener(type, listener) {
|
||
checkListener(listener);
|
||
this.prependListener(type, _onceWrap(this, type, listener));
|
||
return this;
|
||
};
|
||
|
||
// Emits a 'removeListener' event if and only if the listener was removed.
|
||
EventEmitter.prototype.removeListener =
|
||
function removeListener(type, listener) {
|
||
var list, events, position, i, originalListener;
|
||
|
||
checkListener(listener);
|
||
|
||
events = this._events;
|
||
if (events === undefined)
|
||
return this;
|
||
|
||
list = events[type];
|
||
if (list === undefined)
|
||
return this;
|
||
|
||
if (list === listener || list.listener === listener) {
|
||
if (--this._eventsCount === 0)
|
||
this._events = Object.create(null);
|
||
else {
|
||
delete events[type];
|
||
if (events.removeListener)
|
||
this.emit('removeListener', type, list.listener || listener);
|
||
}
|
||
} else if (typeof list !== 'function') {
|
||
position = -1;
|
||
|
||
for (i = list.length - 1; i >= 0; i--) {
|
||
if (list[i] === listener || list[i].listener === listener) {
|
||
originalListener = list[i].listener;
|
||
position = i;
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (position < 0)
|
||
return this;
|
||
|
||
if (position === 0)
|
||
list.shift();
|
||
else {
|
||
spliceOne(list, position);
|
||
}
|
||
|
||
if (list.length === 1)
|
||
events[type] = list[0];
|
||
|
||
if (events.removeListener !== undefined)
|
||
this.emit('removeListener', type, originalListener || listener);
|
||
}
|
||
|
||
return this;
|
||
};
|
||
|
||
EventEmitter.prototype.off = EventEmitter.prototype.removeListener;
|
||
|
||
EventEmitter.prototype.removeAllListeners =
|
||
function removeAllListeners(type) {
|
||
var listeners, events, i;
|
||
|
||
events = this._events;
|
||
if (events === undefined)
|
||
return this;
|
||
|
||
// not listening for removeListener, no need to emit
|
||
if (events.removeListener === undefined) {
|
||
if (arguments.length === 0) {
|
||
this._events = Object.create(null);
|
||
this._eventsCount = 0;
|
||
} else if (events[type] !== undefined) {
|
||
if (--this._eventsCount === 0)
|
||
this._events = Object.create(null);
|
||
else
|
||
delete events[type];
|
||
}
|
||
return this;
|
||
}
|
||
|
||
// emit removeListener for all listeners on all events
|
||
if (arguments.length === 0) {
|
||
var keys = Object.keys(events);
|
||
var key;
|
||
for (i = 0; i < keys.length; ++i) {
|
||
key = keys[i];
|
||
if (key === 'removeListener') continue;
|
||
this.removeAllListeners(key);
|
||
}
|
||
this.removeAllListeners('removeListener');
|
||
this._events = Object.create(null);
|
||
this._eventsCount = 0;
|
||
return this;
|
||
}
|
||
|
||
listeners = events[type];
|
||
|
||
if (typeof listeners === 'function') {
|
||
this.removeListener(type, listeners);
|
||
} else if (listeners !== undefined) {
|
||
// LIFO order
|
||
for (i = listeners.length - 1; i >= 0; i--) {
|
||
this.removeListener(type, listeners[i]);
|
||
}
|
||
}
|
||
|
||
return this;
|
||
};
|
||
|
||
function _listeners(target, type, unwrap) {
|
||
var events = target._events;
|
||
|
||
if (events === undefined)
|
||
return [];
|
||
|
||
var evlistener = events[type];
|
||
if (evlistener === undefined)
|
||
return [];
|
||
|
||
if (typeof evlistener === 'function')
|
||
return unwrap ? [evlistener.listener || evlistener] : [evlistener];
|
||
|
||
return unwrap ?
|
||
unwrapListeners(evlistener) : arrayClone(evlistener, evlistener.length);
|
||
}
|
||
|
||
EventEmitter.prototype.listeners = function listeners(type) {
|
||
return _listeners(this, type, true);
|
||
};
|
||
|
||
EventEmitter.prototype.rawListeners = function rawListeners(type) {
|
||
return _listeners(this, type, false);
|
||
};
|
||
|
||
EventEmitter.listenerCount = function(emitter, type) {
|
||
if (typeof emitter.listenerCount === 'function') {
|
||
return emitter.listenerCount(type);
|
||
} else {
|
||
return listenerCount.call(emitter, type);
|
||
}
|
||
};
|
||
|
||
EventEmitter.prototype.listenerCount = listenerCount;
|
||
function listenerCount(type) {
|
||
var events = this._events;
|
||
|
||
if (events !== undefined) {
|
||
var evlistener = events[type];
|
||
|
||
if (typeof evlistener === 'function') {
|
||
return 1;
|
||
} else if (evlistener !== undefined) {
|
||
return evlistener.length;
|
||
}
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
EventEmitter.prototype.eventNames = function eventNames() {
|
||
return this._eventsCount > 0 ? ReflectOwnKeys(this._events) : [];
|
||
};
|
||
|
||
function arrayClone(arr, n) {
|
||
var copy = new Array(n);
|
||
for (var i = 0; i < n; ++i)
|
||
copy[i] = arr[i];
|
||
return copy;
|
||
}
|
||
|
||
function spliceOne(list, index) {
|
||
for (; index + 1 < list.length; index++)
|
||
list[index] = list[index + 1];
|
||
list.pop();
|
||
}
|
||
|
||
function unwrapListeners(arr) {
|
||
var ret = new Array(arr.length);
|
||
for (var i = 0; i < ret.length; ++i) {
|
||
ret[i] = arr[i].listener || arr[i];
|
||
}
|
||
return ret;
|
||
}
|
||
|
||
function once(emitter, name) {
|
||
return new Promise(function (resolve, reject) {
|
||
function eventListener() {
|
||
if (errorListener !== undefined) {
|
||
emitter.removeListener('error', errorListener);
|
||
}
|
||
resolve([].slice.call(arguments));
|
||
};
|
||
var errorListener;
|
||
|
||
// Adding an error listener is not optional because
|
||
// if an error is thrown on an event emitter we cannot
|
||
// guarantee that the actual event we are waiting will
|
||
// be fired. The result could be a silent way to create
|
||
// memory or file descriptor leaks, which is something
|
||
// we should avoid.
|
||
if (name !== 'error') {
|
||
errorListener = function errorListener(err) {
|
||
emitter.removeListener(name, eventListener);
|
||
reject(err);
|
||
};
|
||
|
||
emitter.once('error', errorListener);
|
||
}
|
||
|
||
emitter.once(name, eventListener);
|
||
});
|
||
}
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "../node_modules/style-loader/lib/addStyles.js":
|
||
/*!*****************************************************!*\
|
||
!*** ../node_modules/style-loader/lib/addStyles.js ***!
|
||
\*****************************************************/
|
||
/*! no static exports found */
|
||
/***/ (function(module, exports, __webpack_require__) {
|
||
|
||
/*
|
||
MIT License http://www.opensource.org/licenses/mit-license.php
|
||
Author Tobias Koppers @sokra
|
||
*/
|
||
|
||
var stylesInDom = {};
|
||
|
||
var memoize = function (fn) {
|
||
var memo;
|
||
|
||
return function () {
|
||
if (typeof memo === "undefined") memo = fn.apply(this, arguments);
|
||
return memo;
|
||
};
|
||
};
|
||
|
||
var isOldIE = memoize(function () {
|
||
// Test for IE <= 9 as proposed by Browserhacks
|
||
// @see http://browserhacks.com/#hack-e71d8692f65334173fee715c222cb805
|
||
// Tests for existence of standard globals is to allow style-loader
|
||
// to operate correctly into non-standard environments
|
||
// @see https://github.com/webpack-contrib/style-loader/issues/177
|
||
return window && document && document.all && !window.atob;
|
||
});
|
||
|
||
var getTarget = function (target, parent) {
|
||
if (parent){
|
||
return parent.querySelector(target);
|
||
}
|
||
return document.querySelector(target);
|
||
};
|
||
|
||
var getElement = (function (fn) {
|
||
var memo = {};
|
||
|
||
return function(target, parent) {
|
||
// If passing function in options, then use it for resolve "head" element.
|
||
// Useful for Shadow Root style i.e
|
||
// {
|
||
// insertInto: function () { return document.querySelector("#foo").shadowRoot }
|
||
// }
|
||
if (typeof target === 'function') {
|
||
return target();
|
||
}
|
||
if (typeof memo[target] === "undefined") {
|
||
var styleTarget = getTarget.call(this, target, parent);
|
||
// Special case to return head of iframe instead of iframe itself
|
||
if (window.HTMLIFrameElement && styleTarget instanceof window.HTMLIFrameElement) {
|
||
try {
|
||
// This will throw an exception if access to iframe is blocked
|
||
// due to cross-origin restrictions
|
||
styleTarget = styleTarget.contentDocument.head;
|
||
} catch(e) {
|
||
styleTarget = null;
|
||
}
|
||
}
|
||
memo[target] = styleTarget;
|
||
}
|
||
return memo[target]
|
||
};
|
||
})();
|
||
|
||
var singleton = null;
|
||
var singletonCounter = 0;
|
||
var stylesInsertedAtTop = [];
|
||
|
||
var fixUrls = __webpack_require__(/*! ./urls */ "../node_modules/style-loader/lib/urls.js");
|
||
|
||
module.exports = function(list, options) {
|
||
if (typeof DEBUG !== "undefined" && DEBUG) {
|
||
if (typeof document !== "object") throw new Error("The style-loader cannot be used in a non-browser environment");
|
||
}
|
||
|
||
options = options || {};
|
||
|
||
options.attrs = typeof options.attrs === "object" ? options.attrs : {};
|
||
|
||
// Force single-tag solution on IE6-9, which has a hard limit on the # of <style>
|
||
// tags it will allow on a page
|
||
if (!options.singleton && typeof options.singleton !== "boolean") options.singleton = isOldIE();
|
||
|
||
// By default, add <style> tags to the <head> element
|
||
if (!options.insertInto) options.insertInto = "head";
|
||
|
||
// By default, add <style> tags to the bottom of the target
|
||
if (!options.insertAt) options.insertAt = "bottom";
|
||
|
||
var styles = listToStyles(list, options);
|
||
|
||
addStylesToDom(styles, options);
|
||
|
||
return function update (newList) {
|
||
var mayRemove = [];
|
||
|
||
for (var i = 0; i < styles.length; i++) {
|
||
var item = styles[i];
|
||
var domStyle = stylesInDom[item.id];
|
||
|
||
domStyle.refs--;
|
||
mayRemove.push(domStyle);
|
||
}
|
||
|
||
if(newList) {
|
||
var newStyles = listToStyles(newList, options);
|
||
addStylesToDom(newStyles, options);
|
||
}
|
||
|
||
for (var i = 0; i < mayRemove.length; i++) {
|
||
var domStyle = mayRemove[i];
|
||
|
||
if(domStyle.refs === 0) {
|
||
for (var j = 0; j < domStyle.parts.length; j++) domStyle.parts[j]();
|
||
|
||
delete stylesInDom[domStyle.id];
|
||
}
|
||
}
|
||
};
|
||
};
|
||
|
||
function addStylesToDom (styles, options) {
|
||
for (var i = 0; i < styles.length; i++) {
|
||
var item = styles[i];
|
||
var domStyle = stylesInDom[item.id];
|
||
|
||
if(domStyle) {
|
||
domStyle.refs++;
|
||
|
||
for(var j = 0; j < domStyle.parts.length; j++) {
|
||
domStyle.parts[j](item.parts[j]);
|
||
}
|
||
|
||
for(; j < item.parts.length; j++) {
|
||
domStyle.parts.push(addStyle(item.parts[j], options));
|
||
}
|
||
} else {
|
||
var parts = [];
|
||
|
||
for(var j = 0; j < item.parts.length; j++) {
|
||
parts.push(addStyle(item.parts[j], options));
|
||
}
|
||
|
||
stylesInDom[item.id] = {id: item.id, refs: 1, parts: parts};
|
||
}
|
||
}
|
||
}
|
||
|
||
function listToStyles (list, options) {
|
||
var styles = [];
|
||
var newStyles = {};
|
||
|
||
for (var i = 0; i < list.length; i++) {
|
||
var item = list[i];
|
||
var id = options.base ? item[0] + options.base : item[0];
|
||
var css = item[1];
|
||
var media = item[2];
|
||
var sourceMap = item[3];
|
||
var part = {css: css, media: media, sourceMap: sourceMap};
|
||
|
||
if(!newStyles[id]) styles.push(newStyles[id] = {id: id, parts: [part]});
|
||
else newStyles[id].parts.push(part);
|
||
}
|
||
|
||
return styles;
|
||
}
|
||
|
||
function insertStyleElement (options, style) {
|
||
var target = getElement(options.insertInto)
|
||
|
||
if (!target) {
|
||
throw new Error("Couldn't find a style target. This probably means that the value for the 'insertInto' parameter is invalid.");
|
||
}
|
||
|
||
var lastStyleElementInsertedAtTop = stylesInsertedAtTop[stylesInsertedAtTop.length - 1];
|
||
|
||
if (options.insertAt === "top") {
|
||
if (!lastStyleElementInsertedAtTop) {
|
||
target.insertBefore(style, target.firstChild);
|
||
} else if (lastStyleElementInsertedAtTop.nextSibling) {
|
||
target.insertBefore(style, lastStyleElementInsertedAtTop.nextSibling);
|
||
} else {
|
||
target.appendChild(style);
|
||
}
|
||
stylesInsertedAtTop.push(style);
|
||
} else if (options.insertAt === "bottom") {
|
||
target.appendChild(style);
|
||
} else if (typeof options.insertAt === "object" && options.insertAt.before) {
|
||
var nextSibling = getElement(options.insertAt.before, target);
|
||
target.insertBefore(style, nextSibling);
|
||
} else {
|
||
throw new Error("[Style Loader]\n\n Invalid value for parameter 'insertAt' ('options.insertAt') found.\n Must be 'top', 'bottom', or Object.\n (https://github.com/webpack-contrib/style-loader#insertat)\n");
|
||
}
|
||
}
|
||
|
||
function removeStyleElement (style) {
|
||
if (style.parentNode === null) return false;
|
||
style.parentNode.removeChild(style);
|
||
|
||
var idx = stylesInsertedAtTop.indexOf(style);
|
||
if(idx >= 0) {
|
||
stylesInsertedAtTop.splice(idx, 1);
|
||
}
|
||
}
|
||
|
||
function createStyleElement (options) {
|
||
var style = document.createElement("style");
|
||
|
||
if(options.attrs.type === undefined) {
|
||
options.attrs.type = "text/css";
|
||
}
|
||
|
||
if(options.attrs.nonce === undefined) {
|
||
var nonce = getNonce();
|
||
if (nonce) {
|
||
options.attrs.nonce = nonce;
|
||
}
|
||
}
|
||
|
||
addAttrs(style, options.attrs);
|
||
insertStyleElement(options, style);
|
||
|
||
return style;
|
||
}
|
||
|
||
function createLinkElement (options) {
|
||
var link = document.createElement("link");
|
||
|
||
if(options.attrs.type === undefined) {
|
||
options.attrs.type = "text/css";
|
||
}
|
||
options.attrs.rel = "stylesheet";
|
||
|
||
addAttrs(link, options.attrs);
|
||
insertStyleElement(options, link);
|
||
|
||
return link;
|
||
}
|
||
|
||
function addAttrs (el, attrs) {
|
||
Object.keys(attrs).forEach(function (key) {
|
||
el.setAttribute(key, attrs[key]);
|
||
});
|
||
}
|
||
|
||
function getNonce() {
|
||
if (false) {}
|
||
|
||
return __webpack_require__.nc;
|
||
}
|
||
|
||
function addStyle (obj, options) {
|
||
var style, update, remove, result;
|
||
|
||
// If a transform function was defined, run it on the css
|
||
if (options.transform && obj.css) {
|
||
result = options.transform(obj.css);
|
||
|
||
if (result) {
|
||
// If transform returns a value, use that instead of the original css.
|
||
// This allows running runtime transformations on the css.
|
||
obj.css = result;
|
||
} else {
|
||
// If the transform function returns a falsy value, don't add this css.
|
||
// This allows conditional loading of css
|
||
return function() {
|
||
// noop
|
||
};
|
||
}
|
||
}
|
||
|
||
if (options.singleton) {
|
||
var styleIndex = singletonCounter++;
|
||
|
||
style = singleton || (singleton = createStyleElement(options));
|
||
|
||
update = applyToSingletonTag.bind(null, style, styleIndex, false);
|
||
remove = applyToSingletonTag.bind(null, style, styleIndex, true);
|
||
|
||
} else if (
|
||
obj.sourceMap &&
|
||
typeof URL === "function" &&
|
||
typeof URL.createObjectURL === "function" &&
|
||
typeof URL.revokeObjectURL === "function" &&
|
||
typeof Blob === "function" &&
|
||
typeof btoa === "function"
|
||
) {
|
||
style = createLinkElement(options);
|
||
update = updateLink.bind(null, style, options);
|
||
remove = function () {
|
||
removeStyleElement(style);
|
||
|
||
if(style.href) URL.revokeObjectURL(style.href);
|
||
};
|
||
} else {
|
||
style = createStyleElement(options);
|
||
update = applyToTag.bind(null, style);
|
||
remove = function () {
|
||
removeStyleElement(style);
|
||
};
|
||
}
|
||
|
||
update(obj);
|
||
|
||
return function updateStyle (newObj) {
|
||
if (newObj) {
|
||
if (
|
||
newObj.css === obj.css &&
|
||
newObj.media === obj.media &&
|
||
newObj.sourceMap === obj.sourceMap
|
||
) {
|
||
return;
|
||
}
|
||
|
||
update(obj = newObj);
|
||
} else {
|
||
remove();
|
||
}
|
||
};
|
||
}
|
||
|
||
var replaceText = (function () {
|
||
var textStore = [];
|
||
|
||
return function (index, replacement) {
|
||
textStore[index] = replacement;
|
||
|
||
return textStore.filter(Boolean).join('\n');
|
||
};
|
||
})();
|
||
|
||
function applyToSingletonTag (style, index, remove, obj) {
|
||
var css = remove ? "" : obj.css;
|
||
|
||
if (style.styleSheet) {
|
||
style.styleSheet.cssText = replaceText(index, css);
|
||
} else {
|
||
var cssNode = document.createTextNode(css);
|
||
var childNodes = style.childNodes;
|
||
|
||
if (childNodes[index]) style.removeChild(childNodes[index]);
|
||
|
||
if (childNodes.length) {
|
||
style.insertBefore(cssNode, childNodes[index]);
|
||
} else {
|
||
style.appendChild(cssNode);
|
||
}
|
||
}
|
||
}
|
||
|
||
function applyToTag (style, obj) {
|
||
var css = obj.css;
|
||
var media = obj.media;
|
||
|
||
if(media) {
|
||
style.setAttribute("media", media)
|
||
}
|
||
|
||
if(style.styleSheet) {
|
||
style.styleSheet.cssText = css;
|
||
} else {
|
||
while(style.firstChild) {
|
||
style.removeChild(style.firstChild);
|
||
}
|
||
|
||
style.appendChild(document.createTextNode(css));
|
||
}
|
||
}
|
||
|
||
function updateLink (link, options, obj) {
|
||
var css = obj.css;
|
||
var sourceMap = obj.sourceMap;
|
||
|
||
/*
|
||
If convertToAbsoluteUrls isn't defined, but sourcemaps are enabled
|
||
and there is no publicPath defined then lets turn convertToAbsoluteUrls
|
||
on by default. Otherwise default to the convertToAbsoluteUrls option
|
||
directly
|
||
*/
|
||
var autoFixUrls = options.convertToAbsoluteUrls === undefined && sourceMap;
|
||
|
||
if (options.convertToAbsoluteUrls || autoFixUrls) {
|
||
css = fixUrls(css);
|
||
}
|
||
|
||
if (sourceMap) {
|
||
// http://stackoverflow.com/a/26603875
|
||
css += "\n/*# sourceMappingURL=data:application/json;base64," + btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap)))) + " */";
|
||
}
|
||
|
||
var blob = new Blob([css], { type: "text/css" });
|
||
|
||
var oldSrc = link.href;
|
||
|
||
link.href = URL.createObjectURL(blob);
|
||
|
||
if(oldSrc) URL.revokeObjectURL(oldSrc);
|
||
}
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "../node_modules/style-loader/lib/urls.js":
|
||
/*!************************************************!*\
|
||
!*** ../node_modules/style-loader/lib/urls.js ***!
|
||
\************************************************/
|
||
/*! no static exports found */
|
||
/***/ (function(module, exports) {
|
||
|
||
|
||
/**
|
||
* When source maps are enabled, `style-loader` uses a link element with a data-uri to
|
||
* embed the css on the page. This breaks all relative urls because now they are relative to a
|
||
* bundle instead of the current page.
|
||
*
|
||
* One solution is to only use full urls, but that may be impossible.
|
||
*
|
||
* Instead, this function "fixes" the relative urls to be absolute according to the current page location.
|
||
*
|
||
* A rudimentary test suite is located at `test/fixUrls.js` and can be run via the `npm test` command.
|
||
*
|
||
*/
|
||
|
||
module.exports = function (css) {
|
||
// get current location
|
||
var location = typeof window !== "undefined" && window.location;
|
||
|
||
if (!location) {
|
||
throw new Error("fixUrls requires window.location");
|
||
}
|
||
|
||
// blank or null?
|
||
if (!css || typeof css !== "string") {
|
||
return css;
|
||
}
|
||
|
||
var baseUrl = location.protocol + "//" + location.host;
|
||
var currentDir = baseUrl + location.pathname.replace(/\/[^\/]*$/, "/");
|
||
|
||
// convert each url(...)
|
||
/*
|
||
This regular expression is just a way to recursively match brackets within
|
||
a string.
|
||
|
||
/url\s*\( = Match on the word "url" with any whitespace after it and then a parens
|
||
( = Start a capturing group
|
||
(?: = Start a non-capturing group
|
||
[^)(] = Match anything that isn't a parentheses
|
||
| = OR
|
||
\( = Match a start parentheses
|
||
(?: = Start another non-capturing groups
|
||
[^)(]+ = Match anything that isn't a parentheses
|
||
| = OR
|
||
\( = Match a start parentheses
|
||
[^)(]* = Match anything that isn't a parentheses
|
||
\) = Match a end parentheses
|
||
) = End Group
|
||
*\) = Match anything and then a close parens
|
||
) = Close non-capturing group
|
||
* = Match anything
|
||
) = Close capturing group
|
||
\) = Match a close parens
|
||
|
||
/gi = Get all matches, not the first. Be case insensitive.
|
||
*/
|
||
var fixedCss = css.replace(/url\s*\(((?:[^)(]|\((?:[^)(]+|\([^)(]*\))*\))*)\)/gi, function(fullMatch, origUrl) {
|
||
// strip quotes (if they exist)
|
||
var unquotedOrigUrl = origUrl
|
||
.trim()
|
||
.replace(/^"(.*)"$/, function(o, $1){ return $1; })
|
||
.replace(/^'(.*)'$/, function(o, $1){ return $1; });
|
||
|
||
// already a full url? no change
|
||
if (/^(#|data:|http:\/\/|https:\/\/|file:\/\/\/|\s*$)/i.test(unquotedOrigUrl)) {
|
||
return fullMatch;
|
||
}
|
||
|
||
// convert the url to a full url
|
||
var newUrl;
|
||
|
||
if (unquotedOrigUrl.indexOf("//") === 0) {
|
||
//TODO: should we add protocol?
|
||
newUrl = unquotedOrigUrl;
|
||
} else if (unquotedOrigUrl.indexOf("/") === 0) {
|
||
// path should be relative to the base url
|
||
newUrl = baseUrl + unquotedOrigUrl; // already starts with '/'
|
||
} else {
|
||
// path should be relative to current directory
|
||
newUrl = currentDir + unquotedOrigUrl.replace(/^\.\//, ""); // Strip leading './'
|
||
}
|
||
|
||
// send back the fixed url(...)
|
||
return "url(" + JSON.stringify(newUrl) + ")";
|
||
});
|
||
|
||
// send back the fixed css
|
||
return fixedCss;
|
||
};
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./content/ChatScroller.js":
|
||
/*!*********************************!*\
|
||
!*** ./content/ChatScroller.js ***!
|
||
\*********************************/
|
||
/*! exports provided: default */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
class ChatScroller {
|
||
constructor() {
|
||
this.scroll = this.scroll.bind(this);
|
||
this.start = this.start.bind(this);
|
||
this.stop = this.stop.bind(this);
|
||
|
||
this.scroller = null;
|
||
this.interval = null;
|
||
}
|
||
|
||
init() {
|
||
this.getScroller()
|
||
.then(() => {
|
||
this.scroller.addEventListener('mouseleave', this.start);
|
||
this.scroller.addEventListener('mouseenter', this.stop);
|
||
this.start();
|
||
});
|
||
}
|
||
|
||
start() {
|
||
this.interval = setInterval(
|
||
this.scroll,
|
||
250
|
||
);
|
||
}
|
||
|
||
stop() {
|
||
clearInterval(this.interval);
|
||
}
|
||
|
||
scroll() {
|
||
this.scroller.scrollTop = 9999;
|
||
}
|
||
|
||
getScroller() {
|
||
const checkForScroller = (res, rej) => {
|
||
this.scroller = document.getElementById('item-scroller');
|
||
if(this.scroller !== null) {
|
||
res();
|
||
} else {
|
||
setTimeout(checkForScroller.bind(this, res, rej), 250);
|
||
}
|
||
};
|
||
return new Promise(checkForScroller);
|
||
}
|
||
}
|
||
|
||
/* harmony default export */ __webpack_exports__["default"] = (ChatScroller);
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./content/ChatWatcher.js":
|
||
/*!********************************!*\
|
||
!*** ./content/ChatWatcher.js ***!
|
||
\********************************/
|
||
/*! exports provided: default */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/* harmony import */ var _Emotes__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Emotes */ "./content/Emotes/index.js");
|
||
/* harmony import */ var _Message__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./Message */ "./content/Message.js");
|
||
/* harmony import */ var src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! src/helpers/PersistentSyncStorage */ "./helpers/PersistentSyncStorage.js");
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
class ChatWatcher {
|
||
|
||
|
||
|
||
constructor() {
|
||
this.watchChat = this.watchChat.bind(this);
|
||
this._chatContainer = null;
|
||
this._observer = null;
|
||
this.messages = new Map();
|
||
}
|
||
|
||
init() {
|
||
|
||
return new Promise((res, rej) => {
|
||
this.getChatContainer().then(_Emotes__WEBPACK_IMPORTED_MODULE_0__["default"].init).then(() => {
|
||
this.addEmotePopup();
|
||
this.watchChat();
|
||
this.parsePreloadedMessages();
|
||
});
|
||
});
|
||
}
|
||
|
||
|
||
getChatContainer() {
|
||
// Parent of actual chat (children are messages)
|
||
const checkForContainer = (res, rej) => {
|
||
this._chatContainer = document.querySelector('#items.style-scope.yt-live-chat-item-list-renderer');
|
||
if(this._chatContainer !== null) {
|
||
res();
|
||
} else {
|
||
setTimeout(checkForContainer.bind(this, res, rej), 250);
|
||
}
|
||
};
|
||
return new Promise(checkForContainer);
|
||
}
|
||
|
||
parsePreloadedMessages() {
|
||
const messages = this._chatContainer.children;
|
||
|
||
for(let i = messages.length-1; i >= 0; i--) {
|
||
const node = messages[i];
|
||
if(this.isMessageNode(node)) {
|
||
const message = new _Message__WEBPACK_IMPORTED_MODULE_1__["default"](node);
|
||
}
|
||
}
|
||
}
|
||
|
||
watchChat() {
|
||
console.log('Chat observer started');
|
||
this._observer = new MutationObserver(mutations => {
|
||
|
||
mutations.forEach(mutation => {
|
||
|
||
|
||
|
||
const { addedNodes, removedNodes } = mutation;
|
||
|
||
// Added nodes
|
||
if(typeof addedNodes !== 'undefined' && addedNodes.length > 0) {
|
||
for(let i = 0, length = addedNodes.length-1; i <= length; i++) {
|
||
const node = addedNodes[i];
|
||
if(this.isMessageNode(node)) {
|
||
this.onNewMessage(node);
|
||
}
|
||
}
|
||
}
|
||
|
||
// Removed nodes
|
||
if(typeof removedNodes !== 'undefined' && removedNodes.length > 0) {
|
||
for(let i = 0, length = removedNodes.length-1; i <= length; i++) {
|
||
const node = removedNodes[i];
|
||
if(this.isMessageNode(node) && this.isObservedMessage(node)) {
|
||
this.onObservedMessageRemoved(node);
|
||
}
|
||
}
|
||
}
|
||
});
|
||
});
|
||
|
||
this._observer.observe(this._chatContainer, {
|
||
childList: true,
|
||
attributes: false,
|
||
characterData: false,
|
||
subtree: false
|
||
});
|
||
}
|
||
|
||
onNewMessage(node) {
|
||
const message = new _Message__WEBPACK_IMPORTED_MODULE_1__["default"](node);
|
||
|
||
// Don't store message if has 0 emotes
|
||
if(message.hasEmotes) {
|
||
this.messages.set(message.id, message);
|
||
}
|
||
}
|
||
|
||
onObservedMessageRemoved(node) {
|
||
const messageId = node.getAttribute('message-id');
|
||
const message = this.messages.get(messageId);
|
||
if(message != undefined){
|
||
message.destroy();
|
||
}
|
||
|
||
this.messages.delete(messageId);
|
||
}
|
||
|
||
isMessageNode(node) {
|
||
return node.tagName === 'YT-LIVE-CHAT-TEXT-MESSAGE-RENDERER';
|
||
}
|
||
|
||
isObservedMessage(node) {
|
||
return node.getAttribute('message-id') !== null;
|
||
}
|
||
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
addEmotePopup(){
|
||
|
||
// create emote button
|
||
const emoteButton = document.createElement('button');
|
||
emoteButton.classList.add('emoteButton');
|
||
emoteButton.textContent = '';
|
||
|
||
// append button to action-buttons list
|
||
const chatButtonSelectionList = document.getElementById('action-buttons');
|
||
chatButtonSelectionList.parentNode.insertBefore(emoteButton, chatButtonSelectionList);
|
||
|
||
// create popupDiv
|
||
const popUpDiv = document.createElement('div');
|
||
popUpDiv.classList.add('popup');
|
||
popUpDiv.classList.add('hideElement');
|
||
|
||
|
||
|
||
// creates array of keys from emote dictionary and adds to div
|
||
// let keysITer = Array.from(Emotes.dictionary.keys());
|
||
// for (let index = 0; index < keysITer.length; index++) {
|
||
// const element = keysITer[index];
|
||
// var emote_div = document.createElement('emote_div');
|
||
// emote_div.innerHTML = (Emotes.get(element).html);
|
||
// popUpDiv.appendChild(emote_div);
|
||
// }
|
||
|
||
|
||
|
||
|
||
|
||
function emoteAppend(keysITer){
|
||
|
||
// create divider
|
||
var hr = document.createElement('hr');
|
||
hr.classList.add('emoteDivider');
|
||
|
||
for (let index = 0; index < keysITer.length; index++) {
|
||
const element = keysITer[index];
|
||
var emote_div = document.createElement('emote_div');
|
||
emote_div.innerHTML = (_Emotes__WEBPACK_IMPORTED_MODULE_0__["default"].get(element).html);
|
||
popUpDiv.appendChild(emote_div);
|
||
}
|
||
popUpDiv.appendChild(hr);
|
||
}
|
||
|
||
// create text
|
||
var bttv_text = document.createElement('h2');
|
||
bttv_text.textContent = "BTTV";
|
||
bttv_text.classList.add('emotePopUpText');
|
||
var franker_text = document.createElement('h2');
|
||
franker_text.textContent = "FrankerFacez";
|
||
franker_text.classList.add('emotePopUpText');
|
||
var twitch_text = document.createElement('h2');
|
||
twitch_text.textContent = "Twitch";
|
||
twitch_text.classList.add('emotePopUpText');
|
||
|
||
|
||
|
||
// need ittr to search each dict and append to dom
|
||
let keysITer = null;
|
||
|
||
keysITer = Array.from(_Emotes__WEBPACK_IMPORTED_MODULE_0__["default"].specialEmotesDictionary.keys());
|
||
emoteAppend(keysITer);
|
||
|
||
if(src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_2__["default"].data.options.enableBTTVEmotes){
|
||
popUpDiv.appendChild(bttv_text);
|
||
keysITer = Array.from(_Emotes__WEBPACK_IMPORTED_MODULE_0__["default"].bttv_Dictionary.keys());
|
||
emoteAppend(keysITer);
|
||
}
|
||
|
||
if(src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_2__["default"].data.options.enableFrankerEmotes){
|
||
popUpDiv.appendChild(franker_text);
|
||
keysITer = Array.from(_Emotes__WEBPACK_IMPORTED_MODULE_0__["default"].franker_Dictionary.keys());
|
||
emoteAppend(keysITer);
|
||
}
|
||
|
||
if(src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_2__["default"].data.options.enableTwitchEmotes){
|
||
popUpDiv.appendChild(twitch_text);
|
||
keysITer = Array.from(_Emotes__WEBPACK_IMPORTED_MODULE_0__["default"].twitch_Dictionary.keys());
|
||
emoteAppend(keysITer);
|
||
}
|
||
|
||
|
||
|
||
// add div to doc
|
||
chatButtonSelectionList.appendChild(popUpDiv);
|
||
|
||
// listen for popup button
|
||
emoteButton.addEventListener('click', function(){
|
||
popUpDiv.classList.toggle('hideElement');
|
||
console.log('emote popup button clicked');
|
||
});
|
||
|
||
// get input area
|
||
var inputArea = document.querySelector('#input.yt-live-chat-text-input-field-renderer');
|
||
var inputAreaLabel = document.querySelector('#label.yt-live-chat-text-input-field-renderer');
|
||
|
||
// add alt tag to chat
|
||
function emoteToTextArea(){
|
||
inputArea.textContent += this.alt + " ";
|
||
inputArea.focus();
|
||
inputAreaLabel.textContent = "";
|
||
popUpDiv.classList.toggle('hideElement');
|
||
console.log(this.alt + " emote button selected");
|
||
}
|
||
|
||
// listener button for emotes
|
||
var EMOTICONS = document.getElementsByTagName('img');
|
||
for (let index = 0; index < EMOTICONS.length; index++) {
|
||
const element = EMOTICONS[index];
|
||
element.addEventListener('click', emoteToTextArea, false);
|
||
}
|
||
|
||
console.log((keysITer.length+1) + " Emotes Added");
|
||
}// end addEmotePopup
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
}// end chat watcher
|
||
|
||
/* harmony default export */ __webpack_exports__["default"] = (ChatWatcher);
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./content/Emotes/Emote.js":
|
||
/*!*********************************!*\
|
||
!*** ./content/Emotes/Emote.js ***!
|
||
\*********************************/
|
||
/*! exports provided: default */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
class Emote {
|
||
constructor({ code, url }) {
|
||
this.code = code;
|
||
this.url = url;
|
||
}
|
||
|
||
get html() {
|
||
return (`
|
||
<span class="Emote">
|
||
<img src="${this.url}" alt="${this.code}">
|
||
</span>
|
||
`).trim();
|
||
}
|
||
}
|
||
|
||
/* harmony default export */ __webpack_exports__["default"] = (Emote);
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./content/Emotes/index.js":
|
||
/*!*********************************!*\
|
||
!*** ./content/Emotes/index.js ***!
|
||
\*********************************/
|
||
/*! exports provided: default */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/* harmony import */ var src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! src/helpers/PersistentSyncStorage */ "./helpers/PersistentSyncStorage.js");
|
||
/* harmony import */ var _Emote__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./Emote */ "./content/Emotes/Emote.js");
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
|
||
|
||
|
||
class Emotes {
|
||
constructor() {
|
||
this.dictionary = new Map();
|
||
|
||
// identification for popup
|
||
this.twitch_Dictionary = new Map();
|
||
this.bttv_Dictionary = new Map();
|
||
this.franker_Dictionary = new Map();
|
||
this.specialEmotesDictionary = new Map();
|
||
|
||
|
||
this.init = this.init.bind(this);
|
||
}
|
||
|
||
init() {
|
||
return Promise.all([
|
||
(src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_0__["default"].data.options.enableBTTVEmotes && this.loadBTTVEmote()),
|
||
(src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_0__["default"].data.options.enableFrankerEmotes && this.loadFrankerEmotes()),
|
||
(src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_0__["default"].data.options.enableTwitchEmotes && this.loadTwitchEmotes()),
|
||
(this.specialEmotes())
|
||
]);
|
||
}
|
||
|
||
/////////////////////////////////////////////////////////////////////////////////
|
||
|
||
get(key) {
|
||
return this.dictionary.get(key);
|
||
}
|
||
|
||
set(key, value) {
|
||
return this.dictionary.set(key, new _Emote__WEBPACK_IMPORTED_MODULE_1__["default"](value));
|
||
}
|
||
|
||
has(key) {
|
||
return this.dictionary.has(key);
|
||
}
|
||
|
||
|
||
//////////////////////////////////////////////////////////////////
|
||
|
||
bbtv_ToDict(json){
|
||
for (let index = 0; index < json.length; index++) {
|
||
|
||
const { emote, total } = json[index];
|
||
|
||
const url = `https://cdn.betterttv.net/emote/${emote.id}/3x`;
|
||
|
||
this.dictionary.set(emote.code, new _Emote__WEBPACK_IMPORTED_MODULE_1__["default"]({ code: emote.code, url }));
|
||
this.bttv_Dictionary.set(emote.code, new _Emote__WEBPACK_IMPORTED_MODULE_1__["default"]({ code: emote.code, url }));
|
||
}
|
||
}
|
||
|
||
bbtv_cached_ToDict(json){
|
||
for (let index = 0; index < json.length; index++) {
|
||
|
||
const { id, code } = json[index];
|
||
|
||
const url = `https://cdn.betterttv.net/emote/${id}/3x`;
|
||
|
||
this.dictionary.set(code, new _Emote__WEBPACK_IMPORTED_MODULE_1__["default"]({ code: code, url }));
|
||
this.bttv_Dictionary.set(code, new _Emote__WEBPACK_IMPORTED_MODULE_1__["default"]({ code: code, url }));
|
||
}
|
||
}
|
||
|
||
// loadEmote is where we collect an object array of emotes from bttv api
|
||
async loadBTTVEmote(){
|
||
|
||
// top 100 emotes query = ?limit=100&offset=100
|
||
const bttv_top_api_url = "https://api.betterttv.net/3/emotes/shared/top?limit=100";
|
||
const bttv_top_api_response = await fetch(bttv_top_api_url);
|
||
var top_Json = await bttv_top_api_response.json();
|
||
|
||
// tredning emotes
|
||
const bttv_trending_api_url = "https://api.betterttv.net/3/emotes/shared/trending?limit=100";
|
||
const bttv_trending_api_response = await fetch(bttv_trending_api_url);
|
||
var trending_Json = await bttv_trending_api_response.json();
|
||
|
||
// global emotes are weird, stored in seperate cache and do not give all the normal attributes
|
||
const bttv_global_api_url = "https://api.betterttv.net/3/cached/emotes/global";
|
||
const bttv_global_api_response = await fetch(bttv_global_api_url);
|
||
var global_Json = await bttv_global_api_response.json();
|
||
|
||
this.bbtv_ToDict(top_Json);
|
||
this.bbtv_ToDict(trending_Json);
|
||
this.bbtv_cached_ToDict(global_Json);
|
||
}
|
||
|
||
////////////////////////////////////////////////////////////////
|
||
|
||
frankerToDict(json){
|
||
for (let index = 0; index < json.emoticons.length; index++) {
|
||
|
||
const { name, urls } = json.emoticons[index];
|
||
|
||
var url = "";
|
||
if(urls[4] != undefined){
|
||
url = urls[4];
|
||
}else if(urls[2] != undefined){
|
||
url = urls[2];
|
||
}else{
|
||
url = urls[1];
|
||
}
|
||
|
||
this.dictionary.set(name, new _Emote__WEBPACK_IMPORTED_MODULE_1__["default"]({ code: name, url }));
|
||
this.franker_Dictionary.set(name, new _Emote__WEBPACK_IMPORTED_MODULE_1__["default"]({ code: name, url }));
|
||
}
|
||
}
|
||
|
||
// loadFrankerEmotes is where we collect an object array of emotes from franker api
|
||
async loadFrankerEmotes(){
|
||
|
||
const franker_top_api_url = "https://api.frankerfacez.com/v1/emoticons?sort=count-desc";
|
||
|
||
const first50Response = await fetch(franker_top_api_url);
|
||
var first50json = await first50Response.json();
|
||
var next50Link = first50json._links.next;
|
||
const second50Response = await fetch(next50Link);
|
||
var second50json = await second50Response.json();
|
||
|
||
// Top 100
|
||
this.frankerToDict(first50json);
|
||
this.frankerToDict(second50json);
|
||
}
|
||
|
||
////////////////////////////////////////////////////////////////
|
||
|
||
twitchToDict(json){
|
||
for (let index = 0; index < json.emotes.length; index++) {
|
||
|
||
const { code, id } = json.emotes[index];
|
||
|
||
const url = `https://static-cdn.jtvnw.net/emoticons/v1/${id}/3.0`;
|
||
|
||
this.dictionary.set(code, new _Emote__WEBPACK_IMPORTED_MODULE_1__["default"]({ code: code, url }));
|
||
this.twitch_Dictionary.set(code, new _Emote__WEBPACK_IMPORTED_MODULE_1__["default"]({ code: code, url }));
|
||
}
|
||
}
|
||
|
||
// loadTwitchEmotes is where we collect an object array of emotes from twitch api
|
||
async loadTwitchEmotes(){
|
||
|
||
// https://api.twitchemotes.com/api/v4/channels/0 - twitch globals - 232 items
|
||
// https://static-cdn.jtvnw.net/emoticons/v1/25/1.0 - cdn
|
||
|
||
// Global
|
||
const twitch_global_api_url = "https://api.twitchemotes.com/api/v4/channels/0";
|
||
const twitch_global_api_response = await fetch(twitch_global_api_url);
|
||
var twitch_global_Json = await twitch_global_api_response.json();
|
||
|
||
this.twitchToDict(twitch_global_Json);
|
||
}
|
||
|
||
// ♥
|
||
specialEmotes(){
|
||
|
||
var emoteObj = {
|
||
"emotes": [
|
||
{
|
||
"code": "wompWTF",
|
||
"url": "https://static-cdn.jtvnw.net/emoticons/v1/301653066/3.0"
|
||
},
|
||
{
|
||
"code": "wompISeeYou",
|
||
"url": "https://static-cdn.jtvnw.net/emoticons/v1/301506153/3.0"
|
||
},
|
||
{
|
||
"code": "wompCry",
|
||
"url": "https://static-cdn.jtvnw.net/emoticons/v1/301506193/3.0"
|
||
},
|
||
{
|
||
"code": "BabyCorona",
|
||
"url": "https://static-cdn.jtvnw.net/emoticons/v1/301629296/3.0"
|
||
},
|
||
{
|
||
"code": "LEL",
|
||
"url": "https://static-cdn.jtvnw.net/emoticons/v1/431249/3.0"
|
||
}
|
||
]
|
||
};
|
||
|
||
for (let index = 0; index < emoteObj.emotes.length; index++) {
|
||
const element = emoteObj[index];
|
||
const { code, url } = emoteObj.emotes[index];
|
||
this.dictionary.set(code, new _Emote__WEBPACK_IMPORTED_MODULE_1__["default"]({ code: code, url}));
|
||
this.specialEmotesDictionary.set(code, new _Emote__WEBPACK_IMPORTED_MODULE_1__["default"]({ code: code, url}));
|
||
}
|
||
}
|
||
}// End Emotes
|
||
|
||
/* harmony default export */ __webpack_exports__["default"] = (new Emotes);
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./content/Message.js":
|
||
/*!****************************!*\
|
||
!*** ./content/Message.js ***!
|
||
\****************************/
|
||
/*! exports provided: default */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/* harmony import */ var _Emotes__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Emotes */ "./content/Emotes/index.js");
|
||
/* harmony import */ var src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! src/helpers/PersistentSyncStorage */ "./helpers/PersistentSyncStorage.js");
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
|
||
|
||
|
||
const idRegexp = /\/-([A-Za-z-_\d])/;
|
||
var colorNumberIndex = 0;
|
||
|
||
class Message {
|
||
constructor(messageNode) {
|
||
this.node = messageNode;
|
||
this.id = this.node.id; // this.id should not be used to reference the node, dom id changes due to optimisitc updates
|
||
this.hasEmotes = null;
|
||
this.observer = null;
|
||
this.parsedText = ''; // This should be fine since you can't edit/change messages
|
||
|
||
this.parseText();
|
||
|
||
|
||
|
||
// Set Defaults
|
||
this.setDefaultSelections();
|
||
|
||
|
||
if(this.hasEmotes) {
|
||
this.node.setAttribute('message-id', this.id);
|
||
this.setHtml();
|
||
this.watch();
|
||
}
|
||
}
|
||
|
||
get textNode() {
|
||
const node = this.node.querySelector('#message');
|
||
return {
|
||
node,
|
||
text: node.innerText
|
||
};
|
||
}
|
||
|
||
parseText() {
|
||
const rawWords = this.textNode.text.split(' ');
|
||
|
||
for(let i = 0, length = rawWords.length; i < length; i++) {
|
||
const word = this.parseIllegalCharcters(rawWords[i]);
|
||
const emote = _Emotes__WEBPACK_IMPORTED_MODULE_0__["default"].get(word);
|
||
|
||
//console.log(Emotes.get(word));
|
||
|
||
|
||
if(typeof emote === 'undefined') {
|
||
this.parsedText += word + ' ';
|
||
} else {
|
||
this.hasEmotes = true;
|
||
this.parsedText += emote.html + ' ';
|
||
}
|
||
}
|
||
}
|
||
|
||
watch() {
|
||
this.observer = new MutationObserver(mutations => {
|
||
let emoteRemoved = false;
|
||
|
||
mutations.forEach(mutation => {
|
||
if(typeof mutation.removedNodes === 'undefined') return;
|
||
if(mutation.removedNodes.length <= 0) return; // This must be after undefined check
|
||
|
||
for(let i = 0, length = mutation.removedNodes.length; i < length; i++) {
|
||
const removedNode = mutation.removedNodes[i];
|
||
if(typeof removedNode.className === 'string' && // check if className exists, is 'SVGAnimatedString' when window resized and removed
|
||
~removedNode.className.indexOf('Emote') !== 0) {
|
||
emoteRemoved = true;
|
||
}
|
||
}
|
||
|
||
});
|
||
|
||
if(emoteRemoved && document.body.contains(this.node)) {
|
||
this.setHtml();
|
||
}
|
||
});
|
||
|
||
this.observer.observe(this.node, {
|
||
childList: true,
|
||
attributes: false,
|
||
characterData: false,
|
||
subtree: true
|
||
});
|
||
}
|
||
|
||
setHtml() {
|
||
this.textNode.node.innerHTML = this.parsedText;
|
||
}
|
||
|
||
parseIllegalCharcters(word) {
|
||
// === 'ZERO WIDTH NO-BREAK SPACE'
|
||
return word.replace('', '').trim();
|
||
}
|
||
|
||
destroy() {
|
||
if(this.observer !== null) {
|
||
this.observer.disconnect();
|
||
this.observer = null;
|
||
}
|
||
}
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
// Setting Options for Each Message
|
||
setDefaultSelections(){
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
// Checks for kappa and replaces emoji element with kappa
|
||
if(src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_1__["default"].data.options.kappaFix) {
|
||
|
||
var stupidKappa = document.querySelectorAll('#message.yt-live-chat-text-message-renderer .emoji.yt-live-chat-text-message-renderer');
|
||
|
||
for (let index = 0; index < stupidKappa.length; index++) {
|
||
const stupidElement = stupidKappa[index];
|
||
var stupidToolTip = stupidElement.getAttribute('shared-tooltip-text');
|
||
|
||
if(stupidToolTip == ':full_moon_face:'){
|
||
|
||
const newSpan = document.createElement('span');
|
||
newSpan.classList.add('Emote');
|
||
newSpan.innerHTML = '<img src="https://static-cdn.jtvnw.net/emoticons/v1/25/3.0" alt="kappa">';
|
||
|
||
stupidElement.parentNode.replaceChild(newSpan, stupidElement);
|
||
}
|
||
}
|
||
}
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
//Set Author Colors
|
||
if(src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_1__["default"].data.options.setAuthorColor && this.node.getAttribute('author-type') !== 'owner') {
|
||
this.setAuthorColor();
|
||
}
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
// Author Icons
|
||
var author_photo = this.node.querySelector('#author-photo');
|
||
|
||
// Set Hide Author Icons
|
||
if (src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_1__["default"].data.options.hideAuthorIcons) {
|
||
author_photo.classList.add("hideElement");
|
||
}
|
||
|
||
// TimeStamp
|
||
var timestamp = this.node.querySelector('#timestamp');
|
||
|
||
// Set Show TimeStamp
|
||
if (src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_1__["default"].data.options.showTimeStamp) {
|
||
this.node.classList.add("showTimeStamp");
|
||
}
|
||
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
// Author Name @ auto paste in text area
|
||
this.node.querySelector('#author-name').addEventListener("click", function(){
|
||
var inputArea = document.querySelector('#input.yt-live-chat-text-input-field-renderer');
|
||
var inputAreaLabel = document.querySelector('#label.yt-live-chat-text-input-field-renderer');
|
||
inputArea.innerText = "@" + this.innerText;
|
||
const textLength = inputArea.innerText.length;
|
||
inputArea.focus();
|
||
inputAreaLabel.innerText = "";
|
||
});
|
||
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
// Set Font Size
|
||
var textSizeSlider = src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_1__["default"].data.options.textSizeSlider;
|
||
|
||
if (src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_1__["default"].data.options.allowTextSlider) {
|
||
this.node.setAttribute('style', 'font-size:' + textSizeSlider + 'px' + '!important');
|
||
this.node.classList.add("AuthorFix");
|
||
}
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
// Set Twitch Styling
|
||
if (src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_1__["default"].data.options.setTwitchColors) {
|
||
this.node.classList.add("setTwitchColors");
|
||
author_photo.classList.add("hideElement");
|
||
}
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
// Set Alternate message Colors
|
||
if (src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_1__["default"].data.options.alternateLineColor) {
|
||
this.alternateLineColor();
|
||
}
|
||
|
||
}// end setDefaultSelections
|
||
|
||
setAuthorColor() {
|
||
|
||
let imageSrc = this.node.querySelector('#img').src;
|
||
|
||
// src for client user input is data instead of http/s
|
||
if(imageSrc[0] !== 'h') {
|
||
imageSrc = document.querySelector('yt-live-chat-message-input-renderer #avatar #img').src;
|
||
}
|
||
|
||
const regexParse = idRegexp.exec(imageSrc);
|
||
const colorId = regexParse.length > 1 ? regexParse[1] : null;
|
||
|
||
if(colorId !== null) {
|
||
this.node.classList.add(`chat-color-${colorId}`);
|
||
} else {
|
||
// log error, couldn't get colorID from `x` url
|
||
}
|
||
|
||
}
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
// changes color every line
|
||
alternateLineColor(){
|
||
|
||
if(colorNumberIndex % 2 == 0){
|
||
this.node.classList.add("set-background-color-one");
|
||
}else{
|
||
this.node.classList.add("set-background-color-two");
|
||
}
|
||
colorNumberIndex++;
|
||
}
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
// removes color attr
|
||
removelternateLineColor(){
|
||
this.node.classList.remove("set-background-color-one");
|
||
this.node.classList.remove("set-background-color-two");
|
||
}
|
||
|
||
}// end Message
|
||
|
||
|
||
|
||
/* harmony default export */ __webpack_exports__["default"] = (Message);
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./content/RouteWatcher.js":
|
||
/*!*********************************!*\
|
||
!*** ./content/RouteWatcher.js ***!
|
||
\*********************************/
|
||
/*! exports provided: default */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/* harmony import */ var events__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! events */ "../node_modules/events/events.js");
|
||
/* harmony import */ var events__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(events__WEBPACK_IMPORTED_MODULE_0__);
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
|
||
|
||
class RouteWatcher extends events__WEBPACK_IMPORTED_MODULE_0___default.a {
|
||
constructor() {
|
||
super();
|
||
|
||
this.target = document.querySelector('head > title');
|
||
this.observer = null;
|
||
|
||
this.init();
|
||
}
|
||
|
||
init() {
|
||
this.observer = new MutationObserver(mutations => {
|
||
mutations.forEach((m) => {
|
||
/**
|
||
* Title is set to 'YouTube Gaming' on main routes
|
||
* and between routes.
|
||
*/
|
||
if(m.target.innerText === 'YouTube Gaming') {
|
||
this.emit('main');
|
||
} else {
|
||
this.emit('change');
|
||
}
|
||
});
|
||
});
|
||
|
||
if(this.target !== null) { // Popout chat does not have title tag
|
||
this.observer.observe(this.target, {
|
||
childList: true,
|
||
attributes: false,
|
||
characterData: true,
|
||
subtree: true
|
||
});
|
||
}
|
||
}
|
||
}
|
||
|
||
/* harmony default export */ __webpack_exports__["default"] = (RouteWatcher);
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./content/index.js":
|
||
/*!**************************!*\
|
||
!*** ./content/index.js ***!
|
||
\**************************/
|
||
/*! no exports provided */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/* harmony import */ var src_stylus_content_styl__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! src/stylus/content.styl */ "./stylus/content.styl");
|
||
/* harmony import */ var src_stylus_content_styl__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(src_stylus_content_styl__WEBPACK_IMPORTED_MODULE_0__);
|
||
/* harmony import */ var _ChatScroller__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./ChatScroller */ "./content/ChatScroller.js");
|
||
/* harmony import */ var _ChatWatcher__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./ChatWatcher */ "./content/ChatWatcher.js");
|
||
/* harmony import */ var _RouteWatcher__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./RouteWatcher */ "./content/RouteWatcher.js");
|
||
/* harmony import */ var src_helpers_Identification__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! src/helpers/Identification */ "./helpers/Identification.js");
|
||
/* harmony import */ var src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! src/helpers/PersistentSyncStorage */ "./helpers/PersistentSyncStorage.js");
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
|
||
let MAIN = null;
|
||
var inTheaterMode = false;
|
||
var bodyBeforeChanges = null;
|
||
|
||
// ---
|
||
|
||
class Main {
|
||
constructor() {
|
||
this.chatWatcher = null;
|
||
this.chatScroller = null;
|
||
this.routeWatcher = null;
|
||
this.onRouteChange = this.onRouteChange.bind(this);
|
||
this.load();
|
||
|
||
|
||
// button class - ytp-size-button ytp-button
|
||
// right player controls - ytp-right-controls
|
||
// player div id - ytd-player
|
||
// chatframe id - chatframe
|
||
// movieframe id - movie_player_fix
|
||
// dono ticker id - ticker
|
||
|
||
// player-theater-container
|
||
}
|
||
|
||
load() {
|
||
this.routeWatcher = new _RouteWatcher__WEBPACK_IMPORTED_MODULE_3__["default"]();
|
||
this.routeWatcher.on("change", this.onRouteChange);
|
||
this.onRouteChange();
|
||
}
|
||
|
||
onRouteChange() {
|
||
if(Object(src_helpers_Identification__WEBPACK_IMPORTED_MODULE_4__["isLivestream"])() && ((Object(src_helpers_Identification__WEBPACK_IMPORTED_MODULE_4__["isYoutubeGaming"])()) || (Object(src_helpers_Identification__WEBPACK_IMPORTED_MODULE_4__["isYoutubeVanilla"])()) || (Object(src_helpers_Identification__WEBPACK_IMPORTED_MODULE_4__["isYoutubeEmbed"])()) || Object(src_helpers_Identification__WEBPACK_IMPORTED_MODULE_4__["isPopOut"])())) {
|
||
this.init();
|
||
}
|
||
|
||
if(Object(src_helpers_Identification__WEBPACK_IMPORTED_MODULE_4__["isLivestream"])()) {
|
||
if (src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_5__["default"].data.options.theaterModeFix) {
|
||
if(document.getElementById('movie_player') != null){
|
||
|
||
var theaterButton = document.getElementsByClassName('ytp-size-button')[0];
|
||
|
||
var handler = function(event) {
|
||
removeEventListener('click', handler, false);
|
||
theaterMode();
|
||
};
|
||
|
||
theaterButton.addEventListener('click', handler);
|
||
}
|
||
}
|
||
}
|
||
}// end onRouteChange
|
||
|
||
init() {
|
||
|
||
this.chatWatcher = new _ChatWatcher__WEBPACK_IMPORTED_MODULE_2__["default"]();
|
||
this.chatWatcher.init();
|
||
this.chatScroller = new _ChatScroller__WEBPACK_IMPORTED_MODULE_1__["default"]();
|
||
this.chatScroller.init();
|
||
|
||
setDefaults();
|
||
console.log("INIT");
|
||
|
||
}// end init
|
||
|
||
|
||
}// end main
|
||
|
||
function setDefaults() {
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
//Live Chat Default Option
|
||
if (src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_5__["default"].data.options.setLiveChat) {
|
||
document.getElementsByClassName("yt-simple-endpoint style-scope yt-dropdown-menu").item(1).click();
|
||
} else {
|
||
// do nothing, let user pick option if not set as default in options menu
|
||
}
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
// Welcome Banner
|
||
var welcomBanner = document.querySelector("yt-live-chat-viewer-engagement-message-renderer");
|
||
|
||
// Set Hide Welcome Banner
|
||
if (src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_5__["default"].data.options.hideWelcomBanner) {
|
||
welcomBanner.classList.add("hideElement");
|
||
} else {
|
||
welcomBanner.classList.remove("hideElement");
|
||
}
|
||
|
||
///////////////////////////////////////////////////////////////////
|
||
|
||
}
|
||
|
||
// --- Every Frame Loaded
|
||
src_helpers_PersistentSyncStorage__WEBPACK_IMPORTED_MODULE_5__["default"].on("ready", () => {
|
||
MAIN = new Main();
|
||
});
|
||
|
||
|
||
|
||
function theaterMode() {
|
||
|
||
console.log('theaterButton clicked');
|
||
inTheaterMode = !inTheaterMode;
|
||
|
||
const movie_player = document.getElementById('movie_player');
|
||
const chat_frame = document.getElementById('chatframe');
|
||
var theater_wrapper = document.createElement('div');
|
||
|
||
|
||
if(inTheaterMode){
|
||
|
||
theater_wrapper.classList = ('theater_wrapper_fix');
|
||
movie_player.classList = ('movie_player_fix');
|
||
chat_frame.classList = ('chat_frame_fix');
|
||
|
||
for (let index = 0; index < document.body.childNodes.length; index++) {
|
||
document.body.childNodes[index].remove();
|
||
}
|
||
|
||
document.removeChild(movie_player);
|
||
document.removeChild(chat_frame);
|
||
|
||
document.body.classList.add('bodyFix');
|
||
|
||
theater_wrapper.appendChild(movie_player);
|
||
theater_wrapper.appendChild(chat_frame);
|
||
document.body.appendChild(theater_wrapper);
|
||
}
|
||
|
||
if(!inTheaterMode){
|
||
|
||
// for (let index = 0; index < document.body.childNodes.length; index++) {
|
||
// document.body.childNodes[index].classList.remove('hidden');
|
||
// }
|
||
|
||
theater_wrapper.classList.remove('theater_wrapper_fix');
|
||
movie_player.classList.remove('movie_player_fix');
|
||
chat_frame.classList.remove('chat_frame_fix');
|
||
}
|
||
|
||
|
||
}// end enterTheaterMode
|
||
|
||
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./helpers/Identification.js":
|
||
/*!***********************************!*\
|
||
!*** ./helpers/Identification.js ***!
|
||
\***********************************/
|
||
/*! exports provided: isLivestream, isYoutubeGaming, isYoutubeVanilla, isYoutubeEmbed, isPopOut */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "isLivestream", function() { return isLivestream; });
|
||
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "isYoutubeGaming", function() { return isYoutubeGaming; });
|
||
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "isYoutubeVanilla", function() { return isYoutubeVanilla; });
|
||
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "isYoutubeEmbed", function() { return isYoutubeEmbed; });
|
||
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "isPopOut", function() { return isPopOut; });
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
const isLivestream = () => {
|
||
const timeDisplay = document.querySelector('.ytp-time-display');
|
||
const chatApp = document.querySelector('yt-live-chat-app');
|
||
const chatHeader = document.querySelector('.yt-live-chat-renderer-0');
|
||
const timeDisplayCheck = timeDisplay && timeDisplay.classList.contains('ytp-live');
|
||
const chatCheck = (document.body.contains(chatApp) || document.body.contains(chatHeader));
|
||
|
||
return (timeDisplayCheck || chatCheck);
|
||
};
|
||
|
||
// isYoutubeGaming checks for the presence of ytg-app, the top level element for YT Gaming
|
||
const isYoutubeGaming = () => {
|
||
return !!document.querySelector('ytg-app');
|
||
};
|
||
|
||
// isYoutubeEmbed checks that this is an iframe, and it is being used on youtube.com
|
||
const isYoutubeVanilla = () => {
|
||
// window.frameElement is only available from youtube.com sites from within iframe per CORS
|
||
return !!window.frameElement;
|
||
};
|
||
|
||
// isYoutubeEmbed checks that this is an iframe, and it is **not** loaded from youtube.com (main site uses embed too)
|
||
const isYoutubeEmbed = () => {
|
||
|
||
// If the frameElement is available, then CORS means that we must be on youtube.com.
|
||
if (window.frameElement) {
|
||
return false;
|
||
}
|
||
|
||
// If the window location isn't the parent location, then we are in an iframe.
|
||
return (window.location != window.parent.location);
|
||
};
|
||
|
||
// isPopOut fix for popout page
|
||
const isPopOut = () => {
|
||
|
||
// If the frameElement is available, then CORS means that we must be on youtube.com.
|
||
if (window.frameElement) {
|
||
return false;
|
||
}
|
||
|
||
// Checks href for page
|
||
if(window.location.href.includes('is_popout=1')){
|
||
return !!window.location.href.includes('popout=1');
|
||
}
|
||
|
||
return false;
|
||
};
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./helpers/PersistentSyncStorage.js":
|
||
/*!******************************************!*\
|
||
!*** ./helpers/PersistentSyncStorage.js ***!
|
||
\******************************************/
|
||
/*! exports provided: default */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/* harmony import */ var _utils_chrome__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../utils/chrome */ "./utils/chrome/index.js");
|
||
/* harmony import */ var events__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! events */ "../node_modules/events/events.js");
|
||
/* harmony import */ var events__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(events__WEBPACK_IMPORTED_MODULE_1__);
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
|
||
|
||
|
||
class PersistentSyncStorage extends events__WEBPACK_IMPORTED_MODULE_1__["EventEmitter"] {
|
||
constructor() {
|
||
super();
|
||
|
||
this._data = null;
|
||
this.state = 'initiating';
|
||
|
||
this._init();
|
||
}
|
||
|
||
async _init() {
|
||
const fetchedData = await _utils_chrome__WEBPACK_IMPORTED_MODULE_0__["SyncStorage"].get();
|
||
this._initListener();
|
||
this._data = fetchedData;
|
||
this.state = 'ready';
|
||
this.emit(this.state);
|
||
}
|
||
|
||
_initListener() {
|
||
_utils_chrome__WEBPACK_IMPORTED_MODULE_0__["SyncStorage"].listen((changes) => {
|
||
|
||
Object.keys(changes).forEach((changeKey) => {
|
||
if(changes[changeKey].hasOwnProperty('newValue')) {
|
||
this._data[changeKey] = changes[changeKey].newValue;
|
||
} else {
|
||
console.error('No newValue in sync storge change');
|
||
}
|
||
});
|
||
|
||
this.emit('change', this.data, changes);
|
||
});
|
||
}
|
||
|
||
set(items) {
|
||
return _utils_chrome__WEBPACK_IMPORTED_MODULE_0__["SyncStorage"].set(items);
|
||
}
|
||
|
||
get data() {
|
||
return this._data;
|
||
}
|
||
|
||
|
||
has(item) {
|
||
return this.data.hasOwnProperty(item);
|
||
}
|
||
}
|
||
|
||
/* harmony default export */ __webpack_exports__["default"] = (new PersistentSyncStorage());
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./stylus/content.styl":
|
||
/*!*****************************!*\
|
||
!*** ./stylus/content.styl ***!
|
||
\*****************************/
|
||
/*! no static exports found */
|
||
/***/ (function(module, exports, __webpack_require__) {
|
||
|
||
|
||
var content = __webpack_require__(/*! !../../node_modules/css-loader!../../node_modules/stylus-loader!./content.styl */ "../node_modules/css-loader/index.js!../node_modules/stylus-loader/index.js!./stylus/content.styl");
|
||
|
||
if(typeof content === 'string') content = [[module.i, content, '']];
|
||
|
||
var transform;
|
||
var insertInto;
|
||
|
||
|
||
|
||
var options = {"hmr":true}
|
||
|
||
options.transform = transform
|
||
options.insertInto = undefined;
|
||
|
||
var update = __webpack_require__(/*! ../../node_modules/style-loader/lib/addStyles.js */ "../node_modules/style-loader/lib/addStyles.js")(content, options);
|
||
|
||
if(content.locals) module.exports = content.locals;
|
||
|
||
if(false) {}
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./utils/chrome/LocalStorage.js":
|
||
/*!**************************************!*\
|
||
!*** ./utils/chrome/LocalStorage.js ***!
|
||
\**************************************/
|
||
/*! exports provided: default */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/* harmony import */ var _Storage__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Storage */ "./utils/chrome/Storage.js");
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
|
||
|
||
class LocalStorage extends _Storage__WEBPACK_IMPORTED_MODULE_0__["default"] {
|
||
constructor() {
|
||
super();
|
||
this.store = 'local';
|
||
}
|
||
}
|
||
|
||
/* harmony default export */ __webpack_exports__["default"] = (LocalStorage);
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./utils/chrome/Notifications.js":
|
||
/*!***************************************!*\
|
||
!*** ./utils/chrome/Notifications.js ***!
|
||
\***************************************/
|
||
/*! exports provided: default */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
class Notifications {
|
||
|
||
create(notificationId = null, options) {
|
||
// notificationId is optional
|
||
if(typeof notificationId === 'object') {
|
||
options = notificationId;
|
||
notificationId = null;
|
||
}
|
||
|
||
return new Promise((res, rej) => {
|
||
// resolve args = notificationId:string
|
||
chrome.notifications.create(notificationId, options, res);
|
||
});
|
||
}
|
||
|
||
update(notificationId, options) {
|
||
return new Promise((res, rej) => {
|
||
// resolve args = wasUpdated:boolean
|
||
chrome.notifications.update(notificationId, options, res);
|
||
});
|
||
}
|
||
|
||
clear(notificationId) {
|
||
return new Promise((res, rej) => {
|
||
// resolve args = wasCleared:boolean
|
||
chrome.notifications.clear(notificationId, res);
|
||
});
|
||
}
|
||
|
||
getAll() {
|
||
return new Promise((res, rej) => {
|
||
// resolve args = notifications:object
|
||
chrome.notifications.getAll(res);
|
||
});
|
||
}
|
||
|
||
getPermissionLevel() {
|
||
return new Promise((res, rej) => {
|
||
// resolve args = level:PermissionLevel (https://developer.chrome.com/apps/notifications#type-PermissionLevel)
|
||
chrome.notifications.getPermissionLevel(res);
|
||
});
|
||
}
|
||
|
||
listen(event, notificationId = null, callback) {
|
||
// event = 'onClosed' | 'onClicked' | 'onButtonClicked' | 'onPermissionLevelChanged' | 'onShowSettings'
|
||
// notificationId is optional
|
||
if(typeof notificationId === 'function') {
|
||
callback = notificationId;
|
||
notificationId = null;
|
||
}
|
||
|
||
if(event === 'onPermissionLevelChanged' || event === 'onShowSettings') {
|
||
return this._nonNotificationIdListen(event, callback);
|
||
}
|
||
|
||
/**
|
||
* https://developer.chrome.com/apps/notifications#events
|
||
*
|
||
* Resolve args (by event):
|
||
* onClosed = notificationId:string, byUser:boolean
|
||
* onClicked = notificationId:string
|
||
* onButtonClicked = notificationId:string, buttonIndex:integer
|
||
*
|
||
* onPermissionLevelChanged = level:PermissionLevel (https://developer.chrome.com/apps/notifications#type-PermissionLevel)
|
||
* onShowSettings = (none)
|
||
*/
|
||
|
||
// This callback relates only to those events that have notificationId arg
|
||
const ListenerCallback = (() => {
|
||
if(notificationId !== null) {
|
||
return (passedNotificationId, ...args) => {
|
||
if(notificationId === passedNotificationId) {
|
||
callback(passedNotificationId, ...args);
|
||
}
|
||
};
|
||
} else {
|
||
return callback;
|
||
}
|
||
})();
|
||
|
||
|
||
chrome.notifications[event].addListener(ListenerCallback);
|
||
}
|
||
|
||
_nonNotificationIdListen(event, callback) {
|
||
chrome.notifications[event].addListener(callback);
|
||
}
|
||
}
|
||
|
||
/* harmony default export */ __webpack_exports__["default"] = (Notifications);
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./utils/chrome/Storage.js":
|
||
/*!*********************************!*\
|
||
!*** ./utils/chrome/Storage.js ***!
|
||
\*********************************/
|
||
/*! exports provided: default */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
class Storage {
|
||
|
||
get(keys = null) {
|
||
return new Promise((res, rej) => {
|
||
const returnSingle = typeof keys === 'string' || typeof keys === 'number';
|
||
// resolve args = items:object
|
||
chrome.storage[this.store].get(keys, (items) => {
|
||
if(returnSingle) {
|
||
res(items[keys]);
|
||
} else {
|
||
res(items);
|
||
}
|
||
});
|
||
});
|
||
}
|
||
|
||
getBytesInUse(keys = null) {
|
||
return new Promise((res, rej) => {
|
||
// resolve args = bytesInUse:integer
|
||
chrome.storage[this.store].getBytesInUse(keys, res);
|
||
});
|
||
}
|
||
|
||
set(items) {
|
||
return new Promise((res, rej) => {
|
||
// resolve args = (none)
|
||
chrome.storage[this.store].set(items, res);
|
||
});
|
||
}
|
||
|
||
remove(keys) {
|
||
// resolve args = (none)
|
||
return new Promise((res, rej) => {
|
||
chrome.storage[this.store].remove(keys, res);
|
||
});
|
||
}
|
||
|
||
clear() {
|
||
// resolve args = (none)
|
||
return new Promise((res, rej) => {
|
||
chrome.storage[this.store].clear(res);
|
||
});
|
||
}
|
||
|
||
listen(item, onChange) {
|
||
if(typeof item === 'function') {
|
||
onChange = item;
|
||
item = null;
|
||
}
|
||
|
||
chrome.storage.onChanged.addListener((changes, areaName) => {
|
||
if(areaName === this.store) {
|
||
if(item !== null) {
|
||
if(changes.hasOwnProperty(item)) {
|
||
const oldValue = changes[item].oldValue || null;
|
||
const newValue = changes[item].newValue || null;
|
||
onChange(oldValue, newValue);
|
||
}
|
||
} else {
|
||
onChange(changes);
|
||
}
|
||
}
|
||
});
|
||
}
|
||
}
|
||
|
||
/* harmony default export */ __webpack_exports__["default"] = (Storage);
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./utils/chrome/SyncStorage.js":
|
||
/*!*************************************!*\
|
||
!*** ./utils/chrome/SyncStorage.js ***!
|
||
\*************************************/
|
||
/*! exports provided: default */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/* harmony import */ var _Storage__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Storage */ "./utils/chrome/Storage.js");
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
|
||
|
||
class SyncStorage extends _Storage__WEBPACK_IMPORTED_MODULE_0__["default"] {
|
||
constructor() {
|
||
super();
|
||
this.store = 'sync';
|
||
}
|
||
}
|
||
|
||
/* harmony default export */ __webpack_exports__["default"] = (SyncStorage);
|
||
|
||
|
||
/***/ }),
|
||
|
||
/***/ "./utils/chrome/index.js":
|
||
/*!*******************************!*\
|
||
!*** ./utils/chrome/index.js ***!
|
||
\*******************************/
|
||
/*! exports provided: LocalStorage, SyncStorage, Notifications */
|
||
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
||
|
||
"use strict";
|
||
__webpack_require__.r(__webpack_exports__);
|
||
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "LocalStorage", function() { return LocalStorage; });
|
||
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "SyncStorage", function() { return SyncStorage; });
|
||
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "Notifications", function() { return Notifications; });
|
||
/* harmony import */ var _LocalStorage__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./LocalStorage */ "./utils/chrome/LocalStorage.js");
|
||
/* harmony import */ var _SyncStorage__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./SyncStorage */ "./utils/chrome/SyncStorage.js");
|
||
/* harmony import */ var _Notifications__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./Notifications */ "./utils/chrome/Notifications.js");
|
||
/**
|
||
* MIT License
|
||
*
|
||
* Copyright (c) 2020 wompmacho
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in all
|
||
* copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
* SOFTWARE.
|
||
*/
|
||
|
||
|
||
|
||
|
||
|
||
// export default {
|
||
// LocalStorage: new _LocalStorage,
|
||
// SyncStorage: new _SyncStorage,
|
||
// Notifications: new _Notifications
|
||
// }
|
||
|
||
const LocalStorage = new _LocalStorage__WEBPACK_IMPORTED_MODULE_0__["default"];
|
||
const SyncStorage = new _SyncStorage__WEBPACK_IMPORTED_MODULE_1__["default"];
|
||
const Notifications = new _Notifications__WEBPACK_IMPORTED_MODULE_2__["default"];
|
||
|
||
/***/ })
|
||
|
||
/******/ });
|
||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./content.js","sources":["webpack:///webpack/bootstrap","webpack:///./stylus/content.styl","webpack:///../node_modules/css-loader/lib/css-base.js","webpack:///../node_modules/events/events.js","webpack:///../node_modules/style-loader/lib/addStyles.js","webpack:///../node_modules/style-loader/lib/urls.js","webpack:///./content/ChatScroller.js","webpack:///./content/ChatWatcher.js","webpack:///./content/Emotes/Emote.js","webpack:///./content/Emotes/index.js","webpack:///./content/Message.js","webpack:///./content/RouteWatcher.js","webpack:///./content/index.js","webpack:///./helpers/Identification.js","webpack:///./helpers/PersistentSyncStorage.js","webpack:///./stylus/content.styl?3a0c","webpack:///./utils/chrome/LocalStorage.js","webpack:///./utils/chrome/Notifications.js","webpack:///./utils/chrome/Storage.js","webpack:///./utils/chrome/SyncStorage.js","webpack:///./utils/chrome/index.js"],"sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = \"./content/index.js\");\n","exports = module.exports = require(\"../../node_modules/css-loader/lib/css-base.js\")(false);\n// imports\n\n\n// module\nexports.push([module.id, \".chat-color-0 #content #author-name {\\n  color: #536dfe !important;\\n}\\n.chat-color-1 #content #author-name {\\n  color: #2196f3 !important;\\n}\\n.chat-color-2 #content #author-name {\\n  color: #03a9f4 !important;\\n}\\n.chat-color-3 #content #author-name {\\n  color: #00bcd4 !important;\\n}\\n.chat-color-4 #content #author-name {\\n  color: #009688 !important;\\n}\\n.chat-color-5 #content #author-name {\\n  color: #4caf50 !important;\\n}\\n.chat-color-6 #content #author-name {\\n  color: #8bc34a !important;\\n}\\n.chat-color-7 #content #author-name {\\n  color: #cddc39 !important;\\n}\\n.chat-color-8 #content #author-name {\\n  color: #ffeb3b !important;\\n}\\n.chat-color-9 #content #author-name {\\n  color: #ffc107 !important;\\n}\\n.chat-color-A #content #author-name {\\n  color: #f44336 !important;\\n}\\n.chat-color-B #content #author-name {\\n  color: #e91e63 !important;\\n}\\n.chat-color-C #content #author-name {\\n  color: #9c27b0 !important;\\n}\\n.chat-color-D #content #author-name {\\n  color: #673ab7 !important;\\n}\\n.chat-color-E #content #author-name {\\n  color: #536dfe !important;\\n}\\n.chat-color-F #content #author-name {\\n  color: #2196f3 !important;\\n}\\n.chat-color-G #content #author-name {\\n  color: #03a9f4 !important;\\n}\\n.chat-color-H #content #author-name {\\n  color: #00bcd4 !important;\\n}\\n.chat-color-I #content #author-name {\\n  color: #009688 !important;\\n}\\n.chat-color-J #content #author-name {\\n  color: #4caf50 !important;\\n}\\n.chat-color-K #content #author-name {\\n  color: #8bc34a !important;\\n}\\n.chat-color-L #content #author-name {\\n  color: #cddc39 !important;\\n}\\n.chat-color-M #content #author-name {\\n  color: #ffeb3b !important;\\n}\\n.chat-color-N #content #author-name {\\n  color: #ffc107 !important;\\n}\\n.chat-color-O #content #author-name {\\n  color: #ff9800 !important;\\n}\\n.chat-color-P #content #author-name {\\n  color: #ff5722 !important;\\n}\\n.chat-color-Q #content #author-name {\\n  color: #f44336 !important;\\n}\\n.chat-color-R #content #author-name {\\n  color: #e91e63 !important;\\n}\\n.chat-color-S #content #author-name {\\n  color: #9c27b0 !important;\\n}\\n.chat-color-T #content #author-name {\\n  color: #673ab7 !important;\\n}\\n.chat-color-U #content #author-name {\\n  color: #536dfe !important;\\n}\\n.chat-color-V #content #author-name {\\n  color: #2196f3 !important;\\n}\\n.chat-color-W #content #author-name {\\n  color: #03a9f4 !important;\\n}\\n.chat-color-X #content #author-name {\\n  color: #00bcd4 !important;\\n}\\n.chat-color-Y #content #author-name {\\n  color: #009688 !important;\\n}\\n.chat-color-Z #content #author-name {\\n  color: #4caf50 !important;\\n}\\n.chat-color-a #content #author-name {\\n  color: #8bc34a !important;\\n}\\n.chat-color-b #content #author-name {\\n  color: #cddc39 !important;\\n}\\n.chat-color-c #content #author-name {\\n  color: #ffeb3b !important;\\n}\\n.chat-color-d #content #author-name {\\n  color: #ffc107 !important;\\n}\\n.chat-color-e #content #author-name {\\n  color: #ff9800 !important;\\n}\\n.chat-color-f #content #author-name {\\n  color: #ff5722 !important;\\n}\\n.chat-color-g #content #author-name {\\n  color: #f44336 !important;\\n}\\n.chat-color-h #content #author-name {\\n  color: #e91e63 !important;\\n}\\n.chat-color-i #content #author-name {\\n  color: #9c27b0 !important;\\n}\\n.chat-color-j #content #author-name {\\n  color: #673ab7 !important;\\n}\\n.chat-color-k #content #author-name {\\n  color: #536dfe !important;\\n}\\n.chat-color-l #content #author-name {\\n  color: #2196f3 !important;\\n}\\n.chat-color-m #content #author-name {\\n  color: #03a9f4 !important;\\n}\\n.chat-color-n #content #author-name {\\n  color: #00bcd4 !important;\\n}\\n.chat-color-o #content #author-name {\\n  color: #009688 !important;\\n}\\n.chat-color-p #content #author-name {\\n  color: #4caf50 !important;\\n}\\n.chat-color-q #content #author-name {\\n  color: #8bc34a !important;\\n}\\n.chat-color-r #content #author-name {\\n  color: #cddc39 !important;\\n}\\n.chat-color-s #content #author-name {\\n  color: #ffeb3b !important;\\n}\\n.chat-color-t #content #author-name {\\n  color: #ffc107 !important;\\n}\\n.chat-color-u #content #author-name {\\n  color: #ff9800 !important;\\n}\\n.chat-color-v #content #author-name {\\n  color: #ff5722 !important;\\n}\\n.chat-color-w #content #author-name {\\n  color: #f44336 !important;\\n}\\n.chat-color-x #content #author-name {\\n  color: #e91e63 !important;\\n}\\n.chat-color-y #content #author-name {\\n  color: #9c27b0 !important;\\n}\\n.chat-color-z #content #author-name {\\n  color: #673ab7 !important;\\n}\\n.chat-color-- #content #author-name {\\n  color: #ff9800 !important;\\n}\\n.chat-color-_ #content #author-name {\\n  color: #ff5722 !important;\\n}\\n.AuthorFix #author-photo.yt-live-chat-text-message-renderer img.yt-img-shadow {\\n  display: inline-block;\\n  vertical-align: middle;\\n  border-radius: 50%;\\n  height: 1.5em;\\n  width: 1.5em;\\n  vertical-align: sub;\\n}\\n.AuthorFix #content {\\n  display: inline-block;\\n  vertical-align: middle;\\n}\\n.AuthorFix #timestamp.yt-live-chat-text-message-renderer {\\n  font-size: 1em;\\n  display: inline-block;\\n  vertical-align: middle;\\n}\\n.showTimeStamp #timestamp.yt-live-chat-text-message-renderer {\\n  display: inline-block;\\n}\\n.Emote img {\\n  height: 1.75em;\\n  align-self: center;\\n  vertical-align: sub;\\n  display: inline-block;\\n  vertical-align: middle;\\n}\\n.Emote img:hover {\\n  background-size: 100% 100%;\\n  background-color: rgba(255,255,255,0.6);\\n}\\n.hideElement {\\n  display: none !important;\\n}\\n.set-background-color-one {\\n  background-color: #303030 !important;\\n}\\n.set-background-color-two {\\n  background-color: transparent !important;\\n}\\n.setTwitchColors {\\n  text-shadow: 0 0 1px #000, 0 0 2px #000 !important;\\n  background: #18181b !important;\\n  font-family: 'Roboto' !important;\\n  font-size: 1.3rem !important;\\n  line-height: 1.5em !important;\\n  color: #fafafa !important;\\n}\\n.setTwitchColors #timestamp.yt-live-chat-text-message-renderer {\\n  display: none;\\n}\\n.setTwitchColors yt-live-chat-author-chip[is-highlighted] #author-name.owner.yt-live-chat-author-chip,\\n.setTwitchColors #author-name.owner.yt-live-chat-author-chip {\\n  background-color: transparent;\\n  color: #008000;\\n}\\n.setTwitchColors yt-live-chat-author-chip[is-highlighted] #author-name.yt-live-chat-author-chip {\\n  background-color: transparent;\\n}\\n.emoteDivider {\\n  width: 60vw;\\n  border: 2px solid #d3d3d3;\\n  border-radius: 5px;\\n  margin-top: 2%;\\n  margin-bottom: 2%;\\n  margin-left: auto;\\n  margin-right: auto;\\n}\\n.emotePopUpText {\\n  margin-bottom: 2%;\\n}\\n.popup {\\n  background-color: #202020;\\n  position: absolute;\\n  top: 15%;\\n  left: 0;\\n  right: 0;\\n  margin-left: auto;\\n  margin-right: auto;\\n  height: 70%;\\n  width: 75%;\\n  z-index: 999;\\n  text-align: center;\\n  border-radius: 5px;\\n  border: #303030 1px solid;\\n  font-size: 1em;\\n  overflow: hidden;\\n  overflow-y: scroll;\\n  padding: 1%;\\n  padding-top: 2%;\\n}\\n.emoteButton {\\n  background-color: rgba(255,255,255,0.1);\\n  background-image: url(\\\"https://cdn.frankerfacez.com/emoticon/447885/4\\\");\\n  background-repeat: no-repeat;\\n  background-position: center;\\n  background-size: 80% 80%;\\n  width: var(--yt-live-chat-32px-icon-button_-_width);\\n  height: var(--yt-live-chat-32px-icon-button_-_height);\\n  padding: var(--yt-live-chat-32px-icon-button_-_padding);\\n  border-radius: 10px;\\n  border: none;\\n  cursor: pointer;\\n}\\n.emoteButton:hover {\\n  background-color: rgba(255,255,255,0.8);\\n}\\n.emoteButton:focus {\\n  outline: 0;\\n}\\n.bodyFix {\\n  height: 100%;\\n  margin: 0;\\n  overflow: hidden;\\n}\\n.theater_wrapper_fix {\\n  height: 100vh;\\n  width: 100vw;\\n}\\n.movie_player_fix {\\n  max-height: none;\\n}\\n.movie_player_fix video {\\n  width: calc(100vw - 25vw) !important;\\n  height: 100vh !important;\\n}\\n.movie_player_fix .ytp-title {\\n  color: rgba(255,255,255,0.8) !important;\\n}\\n.movie_player_fix .ytp-gradient-top {\\n  max-width: calc(100vw - 25vw);\\n}\\n.movie_player_fix .ytp-chrome-bottom {\\n  width: calc(100vw - 25vw) !important;\\n  left: 0 !important;\\n}\\n.movie_player_fix .html5-endscreen {\\n  width: calc(100vw - 25vw) !important;\\n}\\n.movie_player_fix .ytp-chapter-hover-container {\\n  width: calc(100vw - 25vw) !important;\\n}\\n.chat_frame_fix {\\n  height: 100vh;\\n  width: calc(100vw - 75vw);\\n  position: absolute;\\n  right: 0px;\\n  top: 0px;\\n}\\n\", \"\"]);\n\n// exports\n","/*\n\tMIT License http://www.opensource.org/licenses/mit-license.php\n\tAuthor Tobias Koppers @sokra\n*/\n// css base code, injected by the css-loader\nmodule.exports = function(useSourceMap) {\n\tvar list = [];\n\n\t// return the list of modules as css string\n\tlist.toString = function toString() {\n\t\treturn this.map(function (item) {\n\t\t\tvar content = cssWithMappingToString(item, useSourceMap);\n\t\t\tif(item[2]) {\n\t\t\t\treturn \"@media \" + item[2] + \"{\" + content + \"}\";\n\t\t\t} else {\n\t\t\t\treturn content;\n\t\t\t}\n\t\t}).join(\"\");\n\t};\n\n\t// import a list of modules into the list\n\tlist.i = function(modules, mediaQuery) {\n\t\tif(typeof modules === \"string\")\n\t\t\tmodules = [[null, modules, \"\"]];\n\t\tvar alreadyImportedModules = {};\n\t\tfor(var i = 0; i < this.length; i++) {\n\t\t\tvar id = this[i][0];\n\t\t\tif(typeof id === \"number\")\n\t\t\t\talreadyImportedModules[id] = true;\n\t\t}\n\t\tfor(i = 0; i < modules.length; i++) {\n\t\t\tvar item = modules[i];\n\t\t\t// skip already imported module\n\t\t\t// this implementation is not 100% perfect for weird media query combinations\n\t\t\t//  when a module is imported multiple times with different media queries.\n\t\t\t//  I hope this will never occur (Hey this way we have smaller bundles)\n\t\t\tif(typeof item[0] !== \"number\" || !alreadyImportedModules[item[0]]) {\n\t\t\t\tif(mediaQuery && !item[2]) {\n\t\t\t\t\titem[2] = mediaQuery;\n\t\t\t\t} else if(mediaQuery) {\n\t\t\t\t\titem[2] = \"(\" + item[2] + \") and (\" + mediaQuery + \")\";\n\t\t\t\t}\n\t\t\t\tlist.push(item);\n\t\t\t}\n\t\t}\n\t};\n\treturn list;\n};\n\nfunction cssWithMappingToString(item, useSourceMap) {\n\tvar content = item[1] || '';\n\tvar cssMapping = item[3];\n\tif (!cssMapping) {\n\t\treturn content;\n\t}\n\n\tif (useSourceMap && typeof btoa === 'function') {\n\t\tvar sourceMapping = toComment(cssMapping);\n\t\tvar sourceURLs = cssMapping.sources.map(function (source) {\n\t\t\treturn '/*# sourceURL=' + cssMapping.sourceRoot + source + ' */'\n\t\t});\n\n\t\treturn [content].concat(sourceURLs).concat([sourceMapping]).join('\\n');\n\t}\n\n\treturn [content].join('\\n');\n}\n\n// Adapted from convert-source-map (MIT)\nfunction toComment(sourceMap) {\n\t// eslint-disable-next-line no-undef\n\tvar base64 = btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap))));\n\tvar data = 'sourceMappingURL=data:application/json;charset=utf-8;base64,' + base64;\n\n\treturn '/*# ' + data + ' */';\n}\n","// Copyright Joyent, Inc. and other Node contributors.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a\n// copy of this software and associated documentation files (the\n// \"Software\"), to deal in the Software without restriction, including\n// without limitation the rights to use, copy, modify, merge, publish,\n// distribute, sublicense, and/or sell copies of the Software, and to permit\n// persons to whom the Software is furnished to do so, subject to the\n// following conditions:\n//\n// The above copyright notice and this permission notice shall be included\n// in all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS\n// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\n// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN\n// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,\n// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR\n// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE\n// USE OR OTHER DEALINGS IN THE SOFTWARE.\n\n'use strict';\n\nvar R = typeof Reflect === 'object' ? Reflect : null\nvar ReflectApply = R && typeof R.apply === 'function'\n  ? R.apply\n  : function ReflectApply(target, receiver, args) {\n    return Function.prototype.apply.call(target, receiver, args);\n  }\n\nvar ReflectOwnKeys\nif (R && typeof R.ownKeys === 'function') {\n  ReflectOwnKeys = R.ownKeys\n} else if (Object.getOwnPropertySymbols) {\n  ReflectOwnKeys = function ReflectOwnKeys(target) {\n    return Object.getOwnPropertyNames(target)\n      .concat(Object.getOwnPropertySymbols(target));\n  };\n} else {\n  ReflectOwnKeys = function ReflectOwnKeys(target) {\n    return Object.getOwnPropertyNames(target);\n  };\n}\n\nfunction ProcessEmitWarning(warning) {\n  if (console && console.warn) console.warn(warning);\n}\n\nvar NumberIsNaN = Number.isNaN || function NumberIsNaN(value) {\n  return value !== value;\n}\n\nfunction EventEmitter() {\n  EventEmitter.init.call(this);\n}\nmodule.exports = EventEmitter;\nmodule.exports.once = once;\n\n// Backwards-compat with node 0.10.x\nEventEmitter.EventEmitter = EventEmitter;\n\nEventEmitter.prototype._events = undefined;\nEventEmitter.prototype._eventsCount = 0;\nEventEmitter.prototype._maxListeners = undefined;\n\n// By default EventEmitters will print a warning if more than 10 listeners are\n// added to it. This is a useful default which helps finding memory leaks.\nvar defaultMaxListeners = 10;\n\nfunction checkListener(listener) {\n  if (typeof listener !== 'function') {\n    throw new TypeError('The \"listener\" argument must be of type Function. Received type ' + typeof listener);\n  }\n}\n\nObject.defineProperty(EventEmitter, 'defaultMaxListeners', {\n  enumerable: true,\n  get: function() {\n    return defaultMaxListeners;\n  },\n  set: function(arg) {\n    if (typeof arg !== 'number' || arg < 0 || NumberIsNaN(arg)) {\n      throw new RangeError('The value of \"defaultMaxListeners\" is out of range. It must be a non-negative number. Received ' + arg + '.');\n    }\n    defaultMaxListeners = arg;\n  }\n});\n\nEventEmitter.init = function() {\n\n  if (this._events === undefined ||\n      this._events === Object.getPrototypeOf(this)._events) {\n    this._events = Object.create(null);\n    this._eventsCount = 0;\n  }\n\n  this._maxListeners = this._maxListeners || undefined;\n};\n\n// Obviously not all Emitters should be limited to 10. This function allows\n// that to be increased. Set to zero for unlimited.\nEventEmitter.prototype.setMaxListeners = function setMaxListeners(n) {\n  if (typeof n !== 'number' || n < 0 || NumberIsNaN(n)) {\n    throw new RangeError('The value of \"n\" is out of range. It must be a non-negative number. Received ' + n + '.');\n  }\n  this._maxListeners = n;\n  return this;\n};\n\nfunction _getMaxListeners(that) {\n  if (that._maxListeners === undefined)\n    return EventEmitter.defaultMaxListeners;\n  return that._maxListeners;\n}\n\nEventEmitter.prototype.getMaxListeners = function getMaxListeners() {\n  return _getMaxListeners(this);\n};\n\nEventEmitter.prototype.emit = function emit(type) {\n  var args = [];\n  for (var i = 1; i < arguments.length; i++) args.push(arguments[i]);\n  var doError = (type === 'error');\n\n  var events = this._events;\n  if (events !== undefined)\n    doError = (doError && events.error === undefined);\n  else if (!doError)\n    return false;\n\n  // If there is no 'error' event listener then throw.\n  if (doError) {\n    var er;\n    if (args.length > 0)\n      er = args[0];\n    if (er instanceof Error) {\n      // Note: The comments on the `throw` lines are intentional, they show\n      // up in Node's output if this results in an unhandled exception.\n      throw er; // Unhandled 'error' event\n    }\n    // At least give some kind of context to the user\n    var err = new Error('Unhandled error.' + (er ? ' (' + er.message + ')' : ''));\n    err.context = er;\n    throw err; // Unhandled 'error' event\n  }\n\n  var handler = events[type];\n\n  if (handler === undefined)\n    return false;\n\n  if (typeof handler === 'function') {\n    ReflectApply(handler, this, args);\n  } else {\n    var len = handler.length;\n    var listeners = arrayClone(handler, len);\n    for (var i = 0; i < len; ++i)\n      ReflectApply(listeners[i], this, args);\n  }\n\n  return true;\n};\n\nfunction _addListener(target, type, listener, prepend) {\n  var m;\n  var events;\n  var existing;\n\n  checkListener(listener);\n\n  events = target._events;\n  if (events === undefined) {\n    events = target._events = Object.create(null);\n    target._eventsCount = 0;\n  } else {\n    // To avoid recursion in the case that type === \"newListener\"! Before\n    // adding it to the listeners, first emit \"newListener\".\n    if (events.newListener !== undefined) {\n      target.emit('newListener', type,\n                  listener.listener ? listener.listener : listener);\n\n      // Re-assign `events` because a newListener handler could have caused the\n      // this._events to be assigned to a new object\n      events = target._events;\n    }\n    existing = events[type];\n  }\n\n  if (existing === undefined) {\n    // Optimize the case of one listener. Don't need the extra array object.\n    existing = events[type] = listener;\n    ++target._eventsCount;\n  } else {\n    if (typeof existing === 'function') {\n      // Adding the second element, need to change to array.\n      existing = events[type] =\n        prepend ? [listener, existing] : [existing, listener];\n      // If we've already got an array, just append.\n    } else if (prepend) {\n      existing.unshift(listener);\n    } else {\n      existing.push(listener);\n    }\n\n    // Check for listener leak\n    m = _getMaxListeners(target);\n    if (m > 0 && existing.length > m && !existing.warned) {\n      existing.warned = true;\n      // No error code for this since it is a Warning\n      // eslint-disable-next-line no-restricted-syntax\n      var w = new Error('Possible EventEmitter memory leak detected. ' +\n                          existing.length + ' ' + String(type) + ' listeners ' +\n                          'added. Use emitter.setMaxListeners() to ' +\n                          'increase limit');\n      w.name = 'MaxListenersExceededWarning';\n      w.emitter = target;\n      w.type = type;\n      w.count = existing.length;\n      ProcessEmitWarning(w);\n    }\n  }\n\n  return target;\n}\n\nEventEmitter.prototype.addListener = function addListener(type, listener) {\n  return _addListener(this, type, listener, false);\n};\n\nEventEmitter.prototype.on = EventEmitter.prototype.addListener;\n\nEventEmitter.prototype.prependListener =\n    function prependListener(type, listener) {\n      return _addListener(this, type, listener, true);\n    };\n\nfunction onceWrapper() {\n  if (!this.fired) {\n    this.target.removeListener(this.type, this.wrapFn);\n    this.fired = true;\n    if (arguments.length === 0)\n      return this.listener.call(this.target);\n    return this.listener.apply(this.target, arguments);\n  }\n}\n\nfunction _onceWrap(target, type, listener) {\n  var state = { fired: false, wrapFn: undefined, target: target, type: type, listener: listener };\n  var wrapped = onceWrapper.bind(state);\n  wrapped.listener = listener;\n  state.wrapFn = wrapped;\n  return wrapped;\n}\n\nEventEmitter.prototype.once = function once(type, listener) {\n  checkListener(listener);\n  this.on(type, _onceWrap(this, type, listener));\n  return this;\n};\n\nEventEmitter.prototype.prependOnceListener =\n    function prependOnceListener(type, listener) {\n      checkListener(listener);\n      this.prependListener(type, _onceWrap(this, type, listener));\n      return this;\n    };\n\n// Emits a 'removeListener' event if and only if the listener was removed.\nEventEmitter.prototype.removeListener =\n    function removeListener(type, listener) {\n      var list, events, position, i, originalListener;\n\n      checkListener(listener);\n\n      events = this._events;\n      if (events === undefined)\n        return this;\n\n      list = events[type];\n      if (list === undefined)\n        return this;\n\n      if (list === listener || list.listener === listener) {\n        if (--this._eventsCount === 0)\n          this._events = Object.create(null);\n        else {\n          delete events[type];\n          if (events.removeListener)\n            this.emit('removeListener', type, list.listener || listener);\n        }\n      } else if (typeof list !== 'function') {\n        position = -1;\n\n        for (i = list.length - 1; i >= 0; i--) {\n          if (list[i] === listener || list[i].listener === listener) {\n            originalListener = list[i].listener;\n            position = i;\n            break;\n          }\n        }\n\n        if (position < 0)\n          return this;\n\n        if (position === 0)\n          list.shift();\n        else {\n          spliceOne(list, position);\n        }\n\n        if (list.length === 1)\n          events[type] = list[0];\n\n        if (events.removeListener !== undefined)\n          this.emit('removeListener', type, originalListener || listener);\n      }\n\n      return this;\n    };\n\nEventEmitter.prototype.off = EventEmitter.prototype.removeListener;\n\nEventEmitter.prototype.removeAllListeners =\n    function removeAllListeners(type) {\n      var listeners, events, i;\n\n      events = this._events;\n      if (events === undefined)\n        return this;\n\n      // not listening for removeListener, no need to emit\n      if (events.removeListener === undefined) {\n        if (arguments.length === 0) {\n          this._events = Object.create(null);\n          this._eventsCount = 0;\n        } else if (events[type] !== undefined) {\n          if (--this._eventsCount === 0)\n            this._events = Object.create(null);\n          else\n            delete events[type];\n        }\n        return this;\n      }\n\n      // emit removeListener for all listeners on all events\n      if (arguments.length === 0) {\n        var keys = Object.keys(events);\n        var key;\n        for (i = 0; i < keys.length; ++i) {\n          key = keys[i];\n          if (key === 'removeListener') continue;\n          this.removeAllListeners(key);\n        }\n        this.removeAllListeners('removeListener');\n        this._events = Object.create(null);\n        this._eventsCount = 0;\n        return this;\n      }\n\n      listeners = events[type];\n\n      if (typeof listeners === 'function') {\n        this.removeListener(type, listeners);\n      } else if (listeners !== undefined) {\n        // LIFO order\n        for (i = listeners.length - 1; i >= 0; i--) {\n          this.removeListener(type, listeners[i]);\n        }\n      }\n\n      return this;\n    };\n\nfunction _listeners(target, type, unwrap) {\n  var events = target._events;\n\n  if (events === undefined)\n    return [];\n\n  var evlistener = events[type];\n  if (evlistener === undefined)\n    return [];\n\n  if (typeof evlistener === 'function')\n    return unwrap ? [evlistener.listener || evlistener] : [evlistener];\n\n  return unwrap ?\n    unwrapListeners(evlistener) : arrayClone(evlistener, evlistener.length);\n}\n\nEventEmitter.prototype.listeners = function listeners(type) {\n  return _listeners(this, type, true);\n};\n\nEventEmitter.prototype.rawListeners = function rawListeners(type) {\n  return _listeners(this, type, false);\n};\n\nEventEmitter.listenerCount = function(emitter, type) {\n  if (typeof emitter.listenerCount === 'function') {\n    return emitter.listenerCount(type);\n  } else {\n    return listenerCount.call(emitter, type);\n  }\n};\n\nEventEmitter.prototype.listenerCount = listenerCount;\nfunction listenerCount(type) {\n  var events = this._events;\n\n  if (events !== undefined) {\n    var evlistener = events[type];\n\n    if (typeof evlistener === 'function') {\n      return 1;\n    } else if (evlistener !== undefined) {\n      return evlistener.length;\n    }\n  }\n\n  return 0;\n}\n\nEventEmitter.prototype.eventNames = function eventNames() {\n  return this._eventsCount > 0 ? ReflectOwnKeys(this._events) : [];\n};\n\nfunction arrayClone(arr, n) {\n  var copy = new Array(n);\n  for (var i = 0; i < n; ++i)\n    copy[i] = arr[i];\n  return copy;\n}\n\nfunction spliceOne(list, index) {\n  for (; index + 1 < list.length; index++)\n    list[index] = list[index + 1];\n  list.pop();\n}\n\nfunction unwrapListeners(arr) {\n  var ret = new Array(arr.length);\n  for (var i = 0; i < ret.length; ++i) {\n    ret[i] = arr[i].listener || arr[i];\n  }\n  return ret;\n}\n\nfunction once(emitter, name) {\n  return new Promise(function (resolve, reject) {\n    function eventListener() {\n      if (errorListener !== undefined) {\n        emitter.removeListener('error', errorListener);\n      }\n      resolve([].slice.call(arguments));\n    };\n    var errorListener;\n\n    // Adding an error listener is not optional because\n    // if an error is thrown on an event emitter we cannot\n    // guarantee that the actual event we are waiting will\n    // be fired. The result could be a silent way to create\n    // memory or file descriptor leaks, which is something\n    // we should avoid.\n    if (name !== 'error') {\n      errorListener = function errorListener(err) {\n        emitter.removeListener(name, eventListener);\n        reject(err);\n      };\n\n      emitter.once('error', errorListener);\n    }\n\n    emitter.once(name, eventListener);\n  });\n}\n","/*\n\tMIT License http://www.opensource.org/licenses/mit-license.php\n\tAuthor Tobias Koppers @sokra\n*/\n\nvar stylesInDom = {};\n\nvar\tmemoize = function (fn) {\n\tvar memo;\n\n\treturn function () {\n\t\tif (typeof memo === \"undefined\") memo = fn.apply(this, arguments);\n\t\treturn memo;\n\t};\n};\n\nvar isOldIE = memoize(function () {\n\t// Test for IE <= 9 as proposed by Browserhacks\n\t// @see http://browserhacks.com/#hack-e71d8692f65334173fee715c222cb805\n\t// Tests for existence of standard globals is to allow style-loader\n\t// to operate correctly into non-standard environments\n\t// @see https://github.com/webpack-contrib/style-loader/issues/177\n\treturn window && document && document.all && !window.atob;\n});\n\nvar getTarget = function (target, parent) {\n  if (parent){\n    return parent.querySelector(target);\n  }\n  return document.querySelector(target);\n};\n\nvar getElement = (function (fn) {\n\tvar memo = {};\n\n\treturn function(target, parent) {\n                // If passing function in options, then use it for resolve \"head\" element.\n                // Useful for Shadow Root style i.e\n                // {\n                //   insertInto: function () { return document.querySelector(\"#foo\").shadowRoot }\n                // }\n                if (typeof target === 'function') {\n                        return target();\n                }\n                if (typeof memo[target] === \"undefined\") {\n\t\t\tvar styleTarget = getTarget.call(this, target, parent);\n\t\t\t// Special case to return head of iframe instead of iframe itself\n\t\t\tif (window.HTMLIFrameElement && styleTarget instanceof window.HTMLIFrameElement) {\n\t\t\t\ttry {\n\t\t\t\t\t// This will throw an exception if access to iframe is blocked\n\t\t\t\t\t// due to cross-origin restrictions\n\t\t\t\t\tstyleTarget = styleTarget.contentDocument.head;\n\t\t\t\t} catch(e) {\n\t\t\t\t\tstyleTarget = null;\n\t\t\t\t}\n\t\t\t}\n\t\t\tmemo[target] = styleTarget;\n\t\t}\n\t\treturn memo[target]\n\t};\n})();\n\nvar singleton = null;\nvar\tsingletonCounter = 0;\nvar\tstylesInsertedAtTop = [];\n\nvar\tfixUrls = require(\"./urls\");\n\nmodule.exports = function(list, options) {\n\tif (typeof DEBUG !== \"undefined\" && DEBUG) {\n\t\tif (typeof document !== \"object\") throw new Error(\"The style-loader cannot be used in a non-browser environment\");\n\t}\n\n\toptions = options || {};\n\n\toptions.attrs = typeof options.attrs === \"object\" ? options.attrs : {};\n\n\t// Force single-tag solution on IE6-9, which has a hard limit on the # of <style>\n\t// tags it will allow on a page\n\tif (!options.singleton && typeof options.singleton !== \"boolean\") options.singleton = isOldIE();\n\n\t// By default, add <style> tags to the <head> element\n        if (!options.insertInto) options.insertInto = \"head\";\n\n\t// By default, add <style> tags to the bottom of the target\n\tif (!options.insertAt) options.insertAt = \"bottom\";\n\n\tvar styles = listToStyles(list, options);\n\n\taddStylesToDom(styles, options);\n\n\treturn function update (newList) {\n\t\tvar mayRemove = [];\n\n\t\tfor (var i = 0; i < styles.length; i++) {\n\t\t\tvar item = styles[i];\n\t\t\tvar domStyle = stylesInDom[item.id];\n\n\t\t\tdomStyle.refs--;\n\t\t\tmayRemove.push(domStyle);\n\t\t}\n\n\t\tif(newList) {\n\t\t\tvar newStyles = listToStyles(newList, options);\n\t\t\taddStylesToDom(newStyles, options);\n\t\t}\n\n\t\tfor (var i = 0; i < mayRemove.length; i++) {\n\t\t\tvar domStyle = mayRemove[i];\n\n\t\t\tif(domStyle.refs === 0) {\n\t\t\t\tfor (var j = 0; j < domStyle.parts.length; j++) domStyle.parts[j]();\n\n\t\t\t\tdelete stylesInDom[domStyle.id];\n\t\t\t}\n\t\t}\n\t};\n};\n\nfunction addStylesToDom (styles, options) {\n\tfor (var i = 0; i < styles.length; i++) {\n\t\tvar item = styles[i];\n\t\tvar domStyle = stylesInDom[item.id];\n\n\t\tif(domStyle) {\n\t\t\tdomStyle.refs++;\n\n\t\t\tfor(var j = 0; j < domStyle.parts.length; j++) {\n\t\t\t\tdomStyle.parts[j](item.parts[j]);\n\t\t\t}\n\n\t\t\tfor(; j < item.parts.length; j++) {\n\t\t\t\tdomStyle.parts.push(addStyle(item.parts[j], options));\n\t\t\t}\n\t\t} else {\n\t\t\tvar parts = [];\n\n\t\t\tfor(var j = 0; j < item.parts.length; j++) {\n\t\t\t\tparts.push(addStyle(item.parts[j], options));\n\t\t\t}\n\n\t\t\tstylesInDom[item.id] = {id: item.id, refs: 1, parts: parts};\n\t\t}\n\t}\n}\n\nfunction listToStyles (list, options) {\n\tvar styles = [];\n\tvar newStyles = {};\n\n\tfor (var i = 0; i < list.length; i++) {\n\t\tvar item = list[i];\n\t\tvar id = options.base ? item[0] + options.base : item[0];\n\t\tvar css = item[1];\n\t\tvar media = item[2];\n\t\tvar sourceMap = item[3];\n\t\tvar part = {css: css, media: media, sourceMap: sourceMap};\n\n\t\tif(!newStyles[id]) styles.push(newStyles[id] = {id: id, parts: [part]});\n\t\telse newStyles[id].parts.push(part);\n\t}\n\n\treturn styles;\n}\n\nfunction insertStyleElement (options, style) {\n\tvar target = getElement(options.insertInto)\n\n\tif (!target) {\n\t\tthrow new Error(\"Couldn't find a style target. This probably means that the value for the 'insertInto' parameter is invalid.\");\n\t}\n\n\tvar lastStyleElementInsertedAtTop = stylesInsertedAtTop[stylesInsertedAtTop.length - 1];\n\n\tif (options.insertAt === \"top\") {\n\t\tif (!lastStyleElementInsertedAtTop) {\n\t\t\ttarget.insertBefore(style, target.firstChild);\n\t\t} else if (lastStyleElementInsertedAtTop.nextSibling) {\n\t\t\ttarget.insertBefore(style, lastStyleElementInsertedAtTop.nextSibling);\n\t\t} else {\n\t\t\ttarget.appendChild(style);\n\t\t}\n\t\tstylesInsertedAtTop.push(style);\n\t} else if (options.insertAt === \"bottom\") {\n\t\ttarget.appendChild(style);\n\t} else if (typeof options.insertAt === \"object\" && options.insertAt.before) {\n\t\tvar nextSibling = getElement(options.insertAt.before, target);\n\t\ttarget.insertBefore(style, nextSibling);\n\t} else {\n\t\tthrow new Error(\"[Style Loader]\\n\\n Invalid value for parameter 'insertAt' ('options.insertAt') found.\\n Must be 'top', 'bottom', or Object.\\n (https://github.com/webpack-contrib/style-loader#insertat)\\n\");\n\t}\n}\n\nfunction removeStyleElement (style) {\n\tif (style.parentNode === null) return false;\n\tstyle.parentNode.removeChild(style);\n\n\tvar idx = stylesInsertedAtTop.indexOf(style);\n\tif(idx >= 0) {\n\t\tstylesInsertedAtTop.splice(idx, 1);\n\t}\n}\n\nfunction createStyleElement (options) {\n\tvar style = document.createElement(\"style\");\n\n\tif(options.attrs.type === undefined) {\n\t\toptions.attrs.type = \"text/css\";\n\t}\n\n\tif(options.attrs.nonce === undefined) {\n\t\tvar nonce = getNonce();\n\t\tif (nonce) {\n\t\t\toptions.attrs.nonce = nonce;\n\t\t}\n\t}\n\n\taddAttrs(style, options.attrs);\n\tinsertStyleElement(options, style);\n\n\treturn style;\n}\n\nfunction createLinkElement (options) {\n\tvar link = document.createElement(\"link\");\n\n\tif(options.attrs.type === undefined) {\n\t\toptions.attrs.type = \"text/css\";\n\t}\n\toptions.attrs.rel = \"stylesheet\";\n\n\taddAttrs(link, options.attrs);\n\tinsertStyleElement(options, link);\n\n\treturn link;\n}\n\nfunction addAttrs (el, attrs) {\n\tObject.keys(attrs).forEach(function (key) {\n\t\tel.setAttribute(key, attrs[key]);\n\t});\n}\n\nfunction getNonce() {\n\tif (typeof __webpack_nonce__ === 'undefined') {\n\t\treturn null;\n\t}\n\n\treturn __webpack_nonce__;\n}\n\nfunction addStyle (obj, options) {\n\tvar style, update, remove, result;\n\n\t// If a transform function was defined, run it on the css\n\tif (options.transform && obj.css) {\n\t    result = options.transform(obj.css);\n\n\t    if (result) {\n\t    \t// If transform returns a value, use that instead of the original css.\n\t    \t// This allows running runtime transformations on the css.\n\t    \tobj.css = result;\n\t    } else {\n\t    \t// If the transform function returns a falsy value, don't add this css.\n\t    \t// This allows conditional loading of css\n\t    \treturn function() {\n\t    \t\t// noop\n\t    \t};\n\t    }\n\t}\n\n\tif (options.singleton) {\n\t\tvar styleIndex = singletonCounter++;\n\n\t\tstyle = singleton || (singleton = createStyleElement(options));\n\n\t\tupdate = applyToSingletonTag.bind(null, style, styleIndex, false);\n\t\tremove = applyToSingletonTag.bind(null, style, styleIndex, true);\n\n\t} else if (\n\t\tobj.sourceMap &&\n\t\ttypeof URL === \"function\" &&\n\t\ttypeof URL.createObjectURL === \"function\" &&\n\t\ttypeof URL.revokeObjectURL === \"function\" &&\n\t\ttypeof Blob === \"function\" &&\n\t\ttypeof btoa === \"function\"\n\t) {\n\t\tstyle = createLinkElement(options);\n\t\tupdate = updateLink.bind(null, style, options);\n\t\tremove = function () {\n\t\t\tremoveStyleElement(style);\n\n\t\t\tif(style.href) URL.revokeObjectURL(style.href);\n\t\t};\n\t} else {\n\t\tstyle = createStyleElement(options);\n\t\tupdate = applyToTag.bind(null, style);\n\t\tremove = function () {\n\t\t\tremoveStyleElement(style);\n\t\t};\n\t}\n\n\tupdate(obj);\n\n\treturn function updateStyle (newObj) {\n\t\tif (newObj) {\n\t\t\tif (\n\t\t\t\tnewObj.css === obj.css &&\n\t\t\t\tnewObj.media === obj.media &&\n\t\t\t\tnewObj.sourceMap === obj.sourceMap\n\t\t\t) {\n\t\t\t\treturn;\n\t\t\t}\n\n\t\t\tupdate(obj = newObj);\n\t\t} else {\n\t\t\tremove();\n\t\t}\n\t};\n}\n\nvar replaceText = (function () {\n\tvar textStore = [];\n\n\treturn function (index, replacement) {\n\t\ttextStore[index] = replacement;\n\n\t\treturn textStore.filter(Boolean).join('\\n');\n\t};\n})();\n\nfunction applyToSingletonTag (style, index, remove, obj) {\n\tvar css = remove ? \"\" : obj.css;\n\n\tif (style.styleSheet) {\n\t\tstyle.styleSheet.cssText = replaceText(index, css);\n\t} else {\n\t\tvar cssNode = document.createTextNode(css);\n\t\tvar childNodes = style.childNodes;\n\n\t\tif (childNodes[index]) style.removeChild(childNodes[index]);\n\n\t\tif (childNodes.length) {\n\t\t\tstyle.insertBefore(cssNode, childNodes[index]);\n\t\t} else {\n\t\t\tstyle.appendChild(cssNode);\n\t\t}\n\t}\n}\n\nfunction applyToTag (style, obj) {\n\tvar css = obj.css;\n\tvar media = obj.media;\n\n\tif(media) {\n\t\tstyle.setAttribute(\"media\", media)\n\t}\n\n\tif(style.styleSheet) {\n\t\tstyle.styleSheet.cssText = css;\n\t} else {\n\t\twhile(style.firstChild) {\n\t\t\tstyle.removeChild(style.firstChild);\n\t\t}\n\n\t\tstyle.appendChild(document.createTextNode(css));\n\t}\n}\n\nfunction updateLink (link, options, obj) {\n\tvar css = obj.css;\n\tvar sourceMap = obj.sourceMap;\n\n\t/*\n\t\tIf convertToAbsoluteUrls isn't defined, but sourcemaps are enabled\n\t\tand there is no publicPath defined then lets turn convertToAbsoluteUrls\n\t\ton by default.  Otherwise default to the convertToAbsoluteUrls option\n\t\tdirectly\n\t*/\n\tvar autoFixUrls = options.convertToAbsoluteUrls === undefined && sourceMap;\n\n\tif (options.convertToAbsoluteUrls || autoFixUrls) {\n\t\tcss = fixUrls(css);\n\t}\n\n\tif (sourceMap) {\n\t\t// http://stackoverflow.com/a/26603875\n\t\tcss += \"\\n/*# sourceMappingURL=data:application/json;base64,\" + btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap)))) + \" */\";\n\t}\n\n\tvar blob = new Blob([css], { type: \"text/css\" });\n\n\tvar oldSrc = link.href;\n\n\tlink.href = URL.createObjectURL(blob);\n\n\tif(oldSrc) URL.revokeObjectURL(oldSrc);\n}\n","\n/**\n * When source maps are enabled, `style-loader` uses a link element with a data-uri to\n * embed the css on the page. This breaks all relative urls because now they are relative to a\n * bundle instead of the current page.\n *\n * One solution is to only use full urls, but that may be impossible.\n *\n * Instead, this function \"fixes\" the relative urls to be absolute according to the current page location.\n *\n * A rudimentary test suite is located at `test/fixUrls.js` and can be run via the `npm test` command.\n *\n */\n\nmodule.exports = function (css) {\n  // get current location\n  var location = typeof window !== \"undefined\" && window.location;\n\n  if (!location) {\n    throw new Error(\"fixUrls requires window.location\");\n  }\n\n\t// blank or null?\n\tif (!css || typeof css !== \"string\") {\n\t  return css;\n  }\n\n  var baseUrl = location.protocol + \"//\" + location.host;\n  var currentDir = baseUrl + location.pathname.replace(/\\/[^\\/]*$/, \"/\");\n\n\t// convert each url(...)\n\t/*\n\tThis regular expression is just a way to recursively match brackets within\n\ta string.\n\n\t /url\\s*\\(  = Match on the word \"url\" with any whitespace after it and then a parens\n\t   (  = Start a capturing group\n\t     (?:  = Start a non-capturing group\n\t         [^)(]  = Match anything that isn't a parentheses\n\t         |  = OR\n\t         \\(  = Match a start parentheses\n\t             (?:  = Start another non-capturing groups\n\t                 [^)(]+  = Match anything that isn't a parentheses\n\t                 |  = OR\n\t                 \\(  = Match a start parentheses\n\t                     [^)(]*  = Match anything that isn't a parentheses\n\t                 \\)  = Match a end parentheses\n\t             )  = End Group\n              *\\) = Match anything and then a close parens\n          )  = Close non-capturing group\n          *  = Match anything\n       )  = Close capturing group\n\t \\)  = Match a close parens\n\n\t /gi  = Get all matches, not the first.  Be case insensitive.\n\t */\n\tvar fixedCss = css.replace(/url\\s*\\(((?:[^)(]|\\((?:[^)(]+|\\([^)(]*\\))*\\))*)\\)/gi, function(fullMatch, origUrl) {\n\t\t// strip quotes (if they exist)\n\t\tvar unquotedOrigUrl = origUrl\n\t\t\t.trim()\n\t\t\t.replace(/^\"(.*)\"$/, function(o, $1){ return $1; })\n\t\t\t.replace(/^'(.*)'$/, function(o, $1){ return $1; });\n\n\t\t// already a full url? no change\n\t\tif (/^(#|data:|http:\\/\\/|https:\\/\\/|file:\\/\\/\\/|\\s*$)/i.test(unquotedOrigUrl)) {\n\t\t  return fullMatch;\n\t\t}\n\n\t\t// convert the url to a full url\n\t\tvar newUrl;\n\n\t\tif (unquotedOrigUrl.indexOf(\"//\") === 0) {\n\t\t  \t//TODO: should we add protocol?\n\t\t\tnewUrl = unquotedOrigUrl;\n\t\t} else if (unquotedOrigUrl.indexOf(\"/\") === 0) {\n\t\t\t// path should be relative to the base url\n\t\t\tnewUrl = baseUrl + unquotedOrigUrl; // already starts with '/'\n\t\t} else {\n\t\t\t// path should be relative to current directory\n\t\t\tnewUrl = currentDir + unquotedOrigUrl.replace(/^\\.\\//, \"\"); // Strip leading './'\n\t\t}\n\n\t\t// send back the fixed url(...)\n\t\treturn \"url(\" + JSON.stringify(newUrl) + \")\";\n\t});\n\n\t// send back the fixed css\n\treturn fixedCss;\n};\n","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\n\nclass ChatScroller {\n  constructor() {\n    this.scroll = this.scroll.bind(this);\n    this.start = this.start.bind(this);\n    this.stop = this.stop.bind(this); \n    \n    this.scroller = null;\n    this.interval = null;\n  }\n\n  init() {\n    this.getScroller()\n      .then(() => {\n        this.scroller.addEventListener('mouseleave', this.start);\n        this.scroller.addEventListener('mouseenter', this.stop);\n        this.start();\n      });\n  }\n\n  start() {\n    this.interval = setInterval(\n      this.scroll,\n      250\n    );\n  }\n\n  stop() {\n    clearInterval(this.interval);\n  }\n  \n  scroll() {\n    this.scroller.scrollTop = 9999;\n  }\n\n  getScroller() {\n    const checkForScroller = (res, rej) => {\n      this.scroller = document.getElementById('item-scroller');\n      if(this.scroller !== null) {\n        res();\n      } else {\n        setTimeout(checkForScroller.bind(this, res, rej), 250);\n      }\n    };\n    return new Promise(checkForScroller);\n  }\n}\n\nexport default ChatScroller;","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\n\nimport Emotes from './Emotes';\nimport Message from './Message';\nimport PersistentSyncStorage from 'src/helpers/PersistentSyncStorage';\n\n\n\nclass ChatWatcher {\n\n  \n\n  constructor() {\n    this.watchChat = this.watchChat.bind(this);\n    this._chatContainer = null;\n    this._observer = null;\n    this.messages = new Map();\n  }\n\n  init() {\n\n    return new Promise((res, rej) => {\n      this.getChatContainer().then(Emotes.init).then(() => {\n        this.addEmotePopup();\n        this.watchChat();\n        this.parsePreloadedMessages();\n      });\n    });\n  }\n\n  \n  getChatContainer() {\n    // Parent of actual chat (children are messages)\n    const checkForContainer = (res, rej) => {\n      this._chatContainer = document.querySelector('#items.style-scope.yt-live-chat-item-list-renderer');\n      if(this._chatContainer !== null) {\n        res();\n      } else {\n        setTimeout(checkForContainer.bind(this, res, rej), 250);\n      }\n    };\n    return new Promise(checkForContainer);\n  }\n\n  parsePreloadedMessages() {\n    const messages = this._chatContainer.children;\n\n    for(let i = messages.length-1; i >= 0; i--) {\n      const node = messages[i];\n      if(this.isMessageNode(node)) {\n        const message = new Message(node);\n      }\n    }\n  }\n\n  watchChat() {\n    console.log('Chat observer started');\n    this._observer = new MutationObserver(mutations => {\n\n      mutations.forEach(mutation => {\n\n        \n\n        const { addedNodes, removedNodes } = mutation;\n        \n        // Added nodes\n        if(typeof addedNodes !== 'undefined' && addedNodes.length > 0) {\n          for(let i = 0, length = addedNodes.length-1; i <= length; i++) {\n            const node = addedNodes[i];\n            if(this.isMessageNode(node)) {\n              this.onNewMessage(node);\n            }\n          }\n        }\n\n        // Removed nodes\n        if(typeof removedNodes !== 'undefined' && removedNodes.length > 0) {\n          for(let i = 0, length = removedNodes.length-1; i <= length; i++) {\n            const node = removedNodes[i];\n            if(this.isMessageNode(node) && this.isObservedMessage(node)) {\n              this.onObservedMessageRemoved(node);\n            }\n          }\n        }\n      });\n    });\n\n    this._observer.observe(this._chatContainer, {\n      childList: true,\n      attributes: false,\n      characterData: false,\n      subtree: false\n    });\n  }\n  \n  onNewMessage(node) {\n    const message = new Message(node);\n\n    // Don't store message if has 0 emotes\n    if(message.hasEmotes) {\n      this.messages.set(message.id, message);\n    }\n  }\n\n  onObservedMessageRemoved(node) {\n    const messageId = node.getAttribute('message-id');\n    const message = this.messages.get(messageId);\n    if(message != undefined){\n      message.destroy();\n    }\n\n    this.messages.delete(messageId);\n  }\n\n  isMessageNode(node) {\n    return node.tagName === 'YT-LIVE-CHAT-TEXT-MESSAGE-RENDERER';\n  }\n\n  isObservedMessage(node) {\n    return node.getAttribute('message-id') !== null;\n  }\n\n\n  ///////////////////////////////////////////////////////////////////\n\n  addEmotePopup(){\n\n    //  create emote button\n    const emoteButton = document.createElement('button');\n    emoteButton.classList.add('emoteButton');\n    emoteButton.textContent = '';\n\n    //  append button to action-buttons list\n    const chatButtonSelectionList = document.getElementById('action-buttons');\n    chatButtonSelectionList.parentNode.insertBefore(emoteButton, chatButtonSelectionList);\n\n    // create popupDiv\n    const popUpDiv = document.createElement('div');\n    popUpDiv.classList.add('popup');\n    popUpDiv.classList.add('hideElement');\n\n\n\n    //  creates array of keys from emote dictionary and adds to div\n    // let keysITer = Array.from(Emotes.dictionary.keys());\n    // for (let index = 0; index < keysITer.length; index++) {\n    //   const element = keysITer[index];\n    //   var emote_div = document.createElement('emote_div'); \n    //   emote_div.innerHTML = (Emotes.get(element).html);\n    //   popUpDiv.appendChild(emote_div);\n    // }\n\n    \n\n\n\n    function emoteAppend(keysITer){\n\n      //  create divider\n      var hr = document.createElement('hr');\n      hr.classList.add('emoteDivider');\n      \n      for (let index = 0; index < keysITer.length; index++) {\n        const element = keysITer[index];\n        var emote_div = document.createElement('emote_div'); \n        emote_div.innerHTML = (Emotes.get(element).html);\n        popUpDiv.appendChild(emote_div);\n      }\n      popUpDiv.appendChild(hr);\n    }\n\n    //  create text\n    var bttv_text = document.createElement('h2');\n    bttv_text.textContent = \"BTTV\";\n    bttv_text.classList.add('emotePopUpText');\n    var franker_text = document.createElement('h2');\n    franker_text.textContent = \"FrankerFacez\";\n    franker_text.classList.add('emotePopUpText');\n    var twitch_text = document.createElement('h2');\n    twitch_text.textContent = \"Twitch\";\n    twitch_text.classList.add('emotePopUpText');\n    \n    \n\n    //  need ittr to search each dict and append to dom\n    let keysITer = null;\n\n    keysITer = Array.from(Emotes.specialEmotesDictionary.keys());\n    emoteAppend(keysITer);\n\n    if(PersistentSyncStorage.data.options.enableBTTVEmotes){\n      popUpDiv.appendChild(bttv_text);\n      keysITer = Array.from(Emotes.bttv_Dictionary.keys());\n      emoteAppend(keysITer);\n    }\n\n    if(PersistentSyncStorage.data.options.enableFrankerEmotes){\n      popUpDiv.appendChild(franker_text);\n      keysITer = Array.from(Emotes.franker_Dictionary.keys());\n      emoteAppend(keysITer);\n    }\n\n    if(PersistentSyncStorage.data.options.enableTwitchEmotes){\n      popUpDiv.appendChild(twitch_text);\n      keysITer = Array.from(Emotes.twitch_Dictionary.keys());\n      emoteAppend(keysITer);\n    }\n\n \n\n    //  add div to doc\n    chatButtonSelectionList.appendChild(popUpDiv);\n\n    // listen for popup button\n    emoteButton.addEventListener('click', function(){\n      popUpDiv.classList.toggle('hideElement');\n      console.log('emote popup button clicked');\n    });\n\n    //  get input area\n    var inputArea = document.querySelector('#input.yt-live-chat-text-input-field-renderer');\n    var inputAreaLabel = document.querySelector('#label.yt-live-chat-text-input-field-renderer');\n\n    //  add alt tag to chat\n    function emoteToTextArea(){\n      inputArea.textContent += this.alt + \" \";\n      inputArea.focus();\n      inputAreaLabel.textContent = \"\";\n      popUpDiv.classList.toggle('hideElement');\n      console.log(this.alt + \" emote button selected\");\n    }\n\n    //  listener button for emotes\n    var EMOTICONS = document.getElementsByTagName('img');\n    for (let index = 0; index < EMOTICONS.length; index++) {\n      const element = EMOTICONS[index];\n      element.addEventListener('click', emoteToTextArea, false); \n    }\n\n    console.log((keysITer.length+1) + \" Emotes Added\");\n  }// end addEmotePopup\n\n  ///////////////////////////////////////////////////////////////////\n\n}// end chat watcher\n\nexport default ChatWatcher;\n","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\n\nclass Emote {\n  constructor({ code, url }) {\n    this.code = code;\n    this.url = url;\n  }\n\n  get html() {\n    return (`\n      <span class=\"Emote\">\n        <img src=\"${this.url}\" alt=\"${this.code}\">\n      </span>\n    `).trim();\n  }\n}\n\nexport default Emote;","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\nimport PersistentSyncStorage from 'src/helpers/PersistentSyncStorage';\nimport Emote from './Emote';\n\n\nclass Emotes {\n  constructor() {\n    this.dictionary = new Map();\n\n    //  identification for popup\n    this.twitch_Dictionary = new Map();\n    this.bttv_Dictionary = new Map();\n    this.franker_Dictionary = new Map();\n    this.specialEmotesDictionary = new Map();\n\n\n    this.init = this.init.bind(this);\n  }\n\n  init() {\n    return Promise.all([\n      (PersistentSyncStorage.data.options.enableBTTVEmotes && this.loadBTTVEmote()),\n      (PersistentSyncStorage.data.options.enableFrankerEmotes && this.loadFrankerEmotes()),\n      (PersistentSyncStorage.data.options.enableTwitchEmotes && this.loadTwitchEmotes()),\n      (this.specialEmotes())\n    ]);\n  }\n  \n  /////////////////////////////////////////////////////////////////////////////////\n\n  get(key) {\n    return this.dictionary.get(key);\n  }\n\n  set(key, value) {\n    return this.dictionary.set(key, new Emote(value));\n  }\n\n  has(key) {\n    return this.dictionary.has(key);\n  }\n\n\n  //////////////////////////////////////////////////////////////////\n\n  bbtv_ToDict(json){\n    for (let index = 0; index < json.length; index++) {\n      \n      const { emote, total } = json[index];\n\n      const url = `https://cdn.betterttv.net/emote/${emote.id}/3x`;\n\n      this.dictionary.set(emote.code, new Emote({ code: emote.code, url }));\n      this.bttv_Dictionary.set(emote.code, new Emote({ code: emote.code, url }));\n    }\n  }\n\n  bbtv_cached_ToDict(json){\n    for (let index = 0; index < json.length; index++) {\n      \n      const { id, code } = json[index];\n\n      const url = `https://cdn.betterttv.net/emote/${id}/3x`;\n\n      this.dictionary.set(code, new Emote({ code: code, url }));\n      this.bttv_Dictionary.set(code, new Emote({ code: code, url }));\n    } \n  }\n\n  // loadEmote is where we collect an object array of emotes from bttv api\n  async loadBTTVEmote(){\n\n    // top 100 emotes query = ?limit=100&offset=100\n    const bttv_top_api_url = \"https://api.betterttv.net/3/emotes/shared/top?limit=100\";\n    const bttv_top_api_response = await fetch(bttv_top_api_url);\n    var top_Json = await bttv_top_api_response.json();\n\n    // tredning emotes\n    const bttv_trending_api_url = \"https://api.betterttv.net/3/emotes/shared/trending?limit=100\";\n    const bttv_trending_api_response = await fetch(bttv_trending_api_url);\n    var trending_Json = await bttv_trending_api_response.json();\n\n    // global emotes are weird, stored in seperate cache and do not give all the normal attributes\n    const bttv_global_api_url = \"https://api.betterttv.net/3/cached/emotes/global\";\n    const bttv_global_api_response = await fetch(bttv_global_api_url);\n    var global_Json = await bttv_global_api_response.json();\n\n    this.bbtv_ToDict(top_Json);\n    this.bbtv_ToDict(trending_Json);\n    this.bbtv_cached_ToDict(global_Json);\n  }\n\n  ////////////////////////////////////////////////////////////////\n\n  frankerToDict(json){\n    for (let index = 0; index < json.emoticons.length; index++) {\n      \n      const { name, urls } = json.emoticons[index];\n\n      var url = \"\";\n      if(urls[4] != undefined){\n        url = urls[4];\n      }else if(urls[2] != undefined){\n        url = urls[2];\n      }else{\n        url = urls[1];\n      }\n\n      this.dictionary.set(name, new Emote({ code: name, url }));\n      this.franker_Dictionary.set(name, new Emote({ code: name, url }));\n    }\n  }\n\n  // loadFrankerEmotes is where we collect an object array of emotes from franker api\n  async loadFrankerEmotes(){\n\n    const franker_top_api_url = \"https://api.frankerfacez.com/v1/emoticons?sort=count-desc\";\n    \n    const first50Response = await fetch(franker_top_api_url);\n    var first50json = await first50Response.json();\n    var next50Link = first50json._links.next;\n    const second50Response = await fetch(next50Link);\n    var second50json = await second50Response.json();\n\n    // Top 100\n    this.frankerToDict(first50json);\n    this.frankerToDict(second50json);\n  }\n  \n  ////////////////////////////////////////////////////////////////\n\n  twitchToDict(json){\n    for (let index = 0; index < json.emotes.length; index++) {\n      \n      const { code, id } = json.emotes[index];\n\n      const url = `https://static-cdn.jtvnw.net/emoticons/v1/${id}/3.0`;\n\n      this.dictionary.set(code, new Emote({ code: code, url }));\n      this.twitch_Dictionary.set(code, new Emote({ code: code, url }));\n    }\n  }\n\n  // loadTwitchEmotes is where we collect an object array of emotes from twitch api\n  async loadTwitchEmotes(){\n\n    //  https://api.twitchemotes.com/api/v4/channels/0 - twitch globals - 232 items\n    //  https://static-cdn.jtvnw.net/emoticons/v1/25/1.0 - cdn\n\n    // Global\n    const twitch_global_api_url = \"https://api.twitchemotes.com/api/v4/channels/0\";\n    const twitch_global_api_response = await fetch(twitch_global_api_url);\n    var twitch_global_Json = await twitch_global_api_response.json();\n\n    this.twitchToDict(twitch_global_Json);\n  }\n\n  // ♥\n  specialEmotes(){\n\n    var emoteObj = {\n      \"emotes\": [\n        {\n          \"code\": \"wompWTF\", \n          \"url\": \"https://static-cdn.jtvnw.net/emoticons/v1/301653066/3.0\"\n        },\n        {\n          \"code\": \"wompISeeYou\", \n          \"url\": \"https://static-cdn.jtvnw.net/emoticons/v1/301506153/3.0\"\n        },\n        {\n          \"code\": \"wompCry\", \n          \"url\": \"https://static-cdn.jtvnw.net/emoticons/v1/301506193/3.0\"\n        },\n        {\n          \"code\": \"BabyCorona\", \n          \"url\": \"https://static-cdn.jtvnw.net/emoticons/v1/301629296/3.0\"\n        },\n        {\n          \"code\": \"LEL\", \n          \"url\": \"https://static-cdn.jtvnw.net/emoticons/v1/431249/3.0\"\n        }\n      ]\n    };\n\n    for (let index = 0; index < emoteObj.emotes.length; index++) {\n      const element = emoteObj[index];\n      const { code, url } = emoteObj.emotes[index];\n      this.dictionary.set(code, new Emote({ code: code, url}));\n      this.specialEmotesDictionary.set(code, new Emote({ code: code, url}));\n    }    \n  }\n}// End Emotes\n\nexport default new Emotes;","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\n\nimport Emotes from './Emotes';\nimport PersistentSyncStorage from 'src/helpers/PersistentSyncStorage';\n\nconst idRegexp = /\\/-([A-Za-z-_\\d])/;\nvar colorNumberIndex = 0;\n\nclass Message {\n  constructor(messageNode) {\n    this.node = messageNode;\n    this.id = this.node.id; // this.id should not be used to reference the node, dom id changes due to optimisitc updates\n    this.hasEmotes = null;\n    this.observer = null;\n    this.parsedText = ''; // This should be fine since you can't edit/change messages\n    \n    this.parseText();\n\n    \n\n    //  Set Defaults\n    this.setDefaultSelections();\n    \n\n    if(this.hasEmotes) {\n      this.node.setAttribute('message-id', this.id);\n      this.setHtml();\n      this.watch();\n    }\n  }\n\n  get textNode() {\n    const node = this.node.querySelector('#message');\n    return {\n      node,\n      text: node.innerText\n    };\n  }\n\n  parseText() {\n    const rawWords = this.textNode.text.split(' ');\n\n    for(let i = 0, length = rawWords.length; i < length; i++) {\n      const word = this.parseIllegalCharcters(rawWords[i]);\n      const emote = Emotes.get(word);\n\n      //console.log(Emotes.get(word));\n\n      \n      if(typeof emote === 'undefined') {\n        this.parsedText += word + ' ';\n      } else {\n        this.hasEmotes = true;\n        this.parsedText += emote.html + ' ';\n      }\n    }\n  }\n\n  watch() {\n    this.observer = new MutationObserver(mutations => {\n      let emoteRemoved = false;\n\n      mutations.forEach(mutation => {\n        if(typeof mutation.removedNodes === 'undefined') return;\n        if(mutation.removedNodes.length <= 0) return; // This must be after undefined check\n\n        for(let i = 0, length = mutation.removedNodes.length; i < length; i++) {\n          const removedNode = mutation.removedNodes[i];\n          if(typeof removedNode.className === 'string' && // check if className exists, is 'SVGAnimatedString' when window resized and removed \n              ~removedNode.className.indexOf('Emote') !== 0) {\n            emoteRemoved = true;\n          }\n        }\n\n      });\n\n      if(emoteRemoved && document.body.contains(this.node)) {\n        this.setHtml();\n      }\n    });\n\n    this.observer.observe(this.node, {\n      childList: true,\n      attributes: false,\n      characterData: false,\n      subtree: true\n    });\n  }\n\n  setHtml() {\n    this.textNode.node.innerHTML = this.parsedText;\n  }\n\n  parseIllegalCharcters(word) {\n    // ﻿ === 'ZERO WIDTH NO-BREAK SPACE'\n    return word.replace('﻿', '').trim();\n  }\n  \n  destroy() {\n    if(this.observer !== null) {\n      this.observer.disconnect();\n      this.observer = null;\n    }\n  }\n\n  ///////////////////////////////////////////////////////////////////\n\n  // Setting Options for Each Message\n  setDefaultSelections(){\n\n    ///////////////////////////////////////////////////////////////////\n\n    //  Checks for kappa and replaces emoji element with kappa\n    if(PersistentSyncStorage.data.options.kappaFix) {\n\n      var stupidKappa = document.querySelectorAll('#message.yt-live-chat-text-message-renderer .emoji.yt-live-chat-text-message-renderer');\n      \n      for (let index = 0; index < stupidKappa.length; index++) {\n        const stupidElement = stupidKappa[index];\n        var stupidToolTip = stupidElement.getAttribute('shared-tooltip-text');\n\n        if(stupidToolTip == ':full_moon_face:'){\n\n          const newSpan = document.createElement('span');\n          newSpan.classList.add('Emote');\n          newSpan.innerHTML = '<img src=\"https://static-cdn.jtvnw.net/emoticons/v1/25/3.0\" alt=\"kappa\">';\n\n          stupidElement.parentNode.replaceChild(newSpan, stupidElement);\n        }\n      }   \n    }\n    ///////////////////////////////////////////////////////////////////\n\n    //Set Author Colors\n    if(PersistentSyncStorage.data.options.setAuthorColor && this.node.getAttribute('author-type') !== 'owner') {\n      this.setAuthorColor();\n    }\n\n    ///////////////////////////////////////////////////////////////////\n\n    // Author Icons\n    var author_photo = this.node.querySelector('#author-photo');\n\n    // Set Hide Author Icons\n    if (PersistentSyncStorage.data.options.hideAuthorIcons) {\n      author_photo.classList.add(\"hideElement\");\n    }\n\n    // TimeStamp\n    var timestamp = this.node.querySelector('#timestamp');\n\n    // Set Show TimeStamp\n    if (PersistentSyncStorage.data.options.showTimeStamp) {\n      this.node.classList.add(\"showTimeStamp\");\n    }\n\n    \n    ///////////////////////////////////////////////////////////////////\n    \n    // Author Name @ auto paste in text area\n    this.node.querySelector('#author-name').addEventListener(\"click\", function(){\n      var inputArea = document.querySelector('#input.yt-live-chat-text-input-field-renderer');\n      var inputAreaLabel = document.querySelector('#label.yt-live-chat-text-input-field-renderer');\n      inputArea.innerText = \"@\" + this.innerText;\n      const textLength = inputArea.innerText.length;\n      inputArea.focus();\n      inputAreaLabel.innerText = \"\";\n    });\n  \n  \n    ///////////////////////////////////////////////////////////////////\n  \n    // Set Font Size\n    var textSizeSlider = PersistentSyncStorage.data.options.textSizeSlider;  \n\n    if (PersistentSyncStorage.data.options.allowTextSlider) {\n      this.node.setAttribute('style', 'font-size:' + textSizeSlider + 'px' + '!important');\n      this.node.classList.add(\"AuthorFix\");\n    }\n  \n    ///////////////////////////////////////////////////////////////////\n  \n    //  Set Twitch Styling\n    if (PersistentSyncStorage.data.options.setTwitchColors) {\n      this.node.classList.add(\"setTwitchColors\");\n      author_photo.classList.add(\"hideElement\");\n    }\n      \n    ///////////////////////////////////////////////////////////////////\n  \n    //  Set Alternate message Colors\n    if (PersistentSyncStorage.data.options.alternateLineColor) {\n      this.alternateLineColor();\n    }   \n\n  }// end setDefaultSelections\n\n  setAuthorColor() {\n    \n    let imageSrc = this.node.querySelector('#img').src;\n\n    // src for client user input is data instead of http/s\n    if(imageSrc[0] !== 'h') { \n      imageSrc = document.querySelector('yt-live-chat-message-input-renderer #avatar #img').src;\n    }\n    \n    const regexParse = idRegexp.exec(imageSrc);\n    const colorId = regexParse.length > 1 ? regexParse[1] : null;\n    \n    if(colorId !== null) {\n      this.node.classList.add(`chat-color-${colorId}`);\n    } else {\n      // log error, couldn't get colorID from `x` url\n    }\n\n  }\n\n  ///////////////////////////////////////////////////////////////////\n  \n  //  changes color every line\n  alternateLineColor(){\n\n      if(colorNumberIndex % 2 == 0){\n        this.node.classList.add(\"set-background-color-one\");\n      }else{\n        this.node.classList.add(\"set-background-color-two\");\n      }\n      colorNumberIndex++;\n  }\n  \n  ///////////////////////////////////////////////////////////////////\n\n  //  removes color attr\n  removelternateLineColor(){\n    this.node.classList.remove(\"set-background-color-one\");\n    this.node.classList.remove(\"set-background-color-two\");\n  }\n\n}// end Message\n\n\n\nexport default Message;","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\n\nimport EventEmitter from 'events';\n\nclass RouteWatcher extends EventEmitter {\n  constructor() {\n    super();\n\n    this.target = document.querySelector('head > title');\n    this.observer = null;\n\n    this.init();\n  }\n\n  init() {\n    this.observer = new MutationObserver(mutations => {\n      mutations.forEach((m) => {\n        /**\n         * Title is set to 'YouTube Gaming' on main routes\n         * and between routes.\n         */\n        if(m.target.innerText === 'YouTube Gaming') {\n          this.emit('main');\n        } else {\n          this.emit('change');\n        }\n      });\n    });\n\n    if(this.target !== null) { // Popout chat does not have title tag\n      this.observer.observe(this.target, {\n        childList: true,\n        attributes: false,\n        characterData: true,\n        subtree: true\n      });\n    }\n  }\n}\n\nexport default RouteWatcher;\n","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\nimport \"src/stylus/content.styl\";\nimport ChatScroller from \"./ChatScroller\";\nimport ChatWatcher from \"./ChatWatcher\";\nimport RouteWatcher from \"./RouteWatcher\";\n\n\nimport {\n  isLivestream, isYoutubeGaming,\n  isYoutubeEmbed, isYoutubeVanilla,\n  isPopOut\n} from \"src/helpers/Identification\";\n\nimport PersistentSyncStorage from \"src/helpers/PersistentSyncStorage\";\n\nlet MAIN = null;\nvar inTheaterMode = false;\nvar bodyBeforeChanges = null;\n\n// ---\n\nclass Main {\n  constructor() {\n    this.chatWatcher = null;\n    this.chatScroller = null;\n    this.routeWatcher = null;\n    this.onRouteChange = this.onRouteChange.bind(this);\n    this.load();\n\n    \n      //  button class - ytp-size-button ytp-button\n      //  right player controls - ytp-right-controls\n      //  player div id - ytd-player\n      //  chatframe id - chatframe\n      //  movieframe id - movie_player_fix\n      //  dono ticker id - ticker\n\n      //  player-theater-container\n  }\n\n  load() {\n    this.routeWatcher = new RouteWatcher();\n    this.routeWatcher.on(\"change\", this.onRouteChange);\n    this.onRouteChange();\n  }\n\n  onRouteChange() {\n    if(isLivestream() && ((isYoutubeGaming()) || (isYoutubeVanilla()) || (isYoutubeEmbed()) || isPopOut())) {\n      this.init();\n    }\n\n    if(isLivestream()) {\n      if (PersistentSyncStorage.data.options.theaterModeFix) {\n        if(document.getElementById('movie_player') != null){\n\n          var theaterButton = document.getElementsByClassName('ytp-size-button')[0];\n\n          var handler = function(event) {\n            removeEventListener('click', handler, false);\n            theaterMode();\n          };\n\n          theaterButton.addEventListener('click', handler);\n        }\n      }\n    }\n  }//  end onRouteChange\n\n  init() {\n\n    this.chatWatcher = new ChatWatcher();\n    this.chatWatcher.init();\n    this.chatScroller = new ChatScroller();\n    this.chatScroller.init();\n\n    setDefaults();\n    console.log(\"INIT\");\n\n  }// end init\n\n\n}// end main\n\n  function setDefaults() {\n\n    ///////////////////////////////////////////////////////////////////\n\n    //Live Chat Default Option\n    if (PersistentSyncStorage.data.options.setLiveChat) {\n      document.getElementsByClassName(\"yt-simple-endpoint style-scope yt-dropdown-menu\").item(1).click();\n    } else {\n      // do nothing, let user pick option if not set as default in options menu\n    }\n\n    ///////////////////////////////////////////////////////////////////\n\n    // Welcome Banner\n    var welcomBanner = document.querySelector(\"yt-live-chat-viewer-engagement-message-renderer\");\n  \n    // Set Hide Welcome Banner\n    if (PersistentSyncStorage.data.options.hideWelcomBanner) {\n      welcomBanner.classList.add(\"hideElement\");\n    } else {\n      welcomBanner.classList.remove(\"hideElement\");\n    }\n\n    ///////////////////////////////////////////////////////////////////\n\n  }\n\n// --- Every Frame Loaded\nPersistentSyncStorage.on(\"ready\", () => {\n  MAIN = new Main();\n});\n\n\n\nfunction theaterMode() {\n\n  console.log('theaterButton clicked');\n  inTheaterMode = !inTheaterMode;\n\n  const movie_player = document.getElementById('movie_player');\n  const chat_frame = document.getElementById('chatframe');\n  var theater_wrapper = document.createElement('div');\n  \n\n  if(inTheaterMode){\n    \n    theater_wrapper.classList = ('theater_wrapper_fix');\n    movie_player.classList = ('movie_player_fix');\n    chat_frame.classList = ('chat_frame_fix');\n  \n    for (let index = 0; index < document.body.childNodes.length; index++) {\n      document.body.childNodes[index].remove();\n    }\n\n    document.removeChild(movie_player);\n    document.removeChild(chat_frame);\n  \n    document.body.classList.add('bodyFix');\n  \n    theater_wrapper.appendChild(movie_player);\n    theater_wrapper.appendChild(chat_frame);\n    document.body.appendChild(theater_wrapper);\n  }\n\n  if(!inTheaterMode){\n\n    // for (let index = 0; index < document.body.childNodes.length; index++) {\n    //   document.body.childNodes[index].classList.remove('hidden');\n    // }\n\n    theater_wrapper.classList.remove('theater_wrapper_fix');\n    movie_player.classList.remove('movie_player_fix');\n    chat_frame.classList.remove('chat_frame_fix');\n  }\n\n\n}// end enterTheaterMode\n\n\n","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\n\nexport const isLivestream = () => {\n    const timeDisplay = document.querySelector('.ytp-time-display');\n    const chatApp = document.querySelector('yt-live-chat-app');\n    const chatHeader = document.querySelector('.yt-live-chat-renderer-0');\n    const timeDisplayCheck = timeDisplay && timeDisplay.classList.contains('ytp-live');\n    const chatCheck = (document.body.contains(chatApp) || document.body.contains(chatHeader));\n\n    return (timeDisplayCheck || chatCheck);\n};\n\n// isYoutubeGaming checks for the presence of ytg-app, the top level element for YT Gaming\nexport const isYoutubeGaming = () => {\n    return !!document.querySelector('ytg-app');\n};\n\n// isYoutubeEmbed checks that this is an iframe, and it is being used on youtube.com\nexport const isYoutubeVanilla = () => {\n    // window.frameElement is only available from youtube.com sites from within iframe per CORS\n    return !!window.frameElement;\n};\n\n// isYoutubeEmbed checks that this is an iframe, and it is **not** loaded from youtube.com (main site uses embed too)\nexport const isYoutubeEmbed = () => {\n\n    // If the frameElement is available, then CORS means that we must be on youtube.com.\n    if (window.frameElement) {\n        return false;\n    }\n\n    // If the window location isn't the parent location, then we are in an iframe.\n    return (window.location != window.parent.location);\n};\n\n// isPopOut fix for popout page\nexport const isPopOut = () => {\n    \n    // If the frameElement is available, then CORS means that we must be on youtube.com.\n    if (window.frameElement) {\n        return false;\n    }\n\n    // Checks href for page \n    if(window.location.href.includes('is_popout=1')){\n        return !!window.location.href.includes('popout=1');\n    }\n\n    return false;\n};","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\n\nimport { SyncStorage } from '../utils/chrome';\nimport { EventEmitter } from 'events';\n\nclass PersistentSyncStorage extends EventEmitter {\n  constructor() {\n    super();\n\n    this._data = null;\n    this.state = 'initiating';\n\n    this._init();\n  }\n\n  async _init() {\n    const fetchedData = await SyncStorage.get();\n    this._initListener();\n    this._data = fetchedData;\n    this.state = 'ready';\n    this.emit(this.state);\n  }\n\n  _initListener() {\n    SyncStorage.listen((changes) => {\n\n      Object.keys(changes).forEach((changeKey) => {\n        if(changes[changeKey].hasOwnProperty('newValue')) {\n          this._data[changeKey] = changes[changeKey].newValue;\n        } else {\n          console.error('No newValue in sync storge change');\n        }\n      });\n      \n      this.emit('change', this.data, changes);\n    });\n  }\n\n  set(items) {\n    return SyncStorage.set(items);\n  }\n  \n  get data() {\n    return this._data;\n  }\n\n\n  has(item) {\n    return this.data.hasOwnProperty(item);\n  }\n}\n\nexport default new PersistentSyncStorage();","\nvar content = require(\"!!../../node_modules/css-loader/index.js!../../node_modules/stylus-loader/index.js!./content.styl\");\n\nif(typeof content === 'string') content = [[module.id, content, '']];\n\nvar transform;\nvar insertInto;\n\n\n\nvar options = {\"hmr\":true}\n\noptions.transform = transform\noptions.insertInto = undefined;\n\nvar update = require(\"!../../node_modules/style-loader/lib/addStyles.js\")(content, options);\n\nif(content.locals) module.exports = content.locals;\n\nif(module.hot) {\n\tmodule.hot.accept(\"!!../../node_modules/css-loader/index.js!../../node_modules/stylus-loader/index.js!./content.styl\", function() {\n\t\tvar newContent = require(\"!!../../node_modules/css-loader/index.js!../../node_modules/stylus-loader/index.js!./content.styl\");\n\n\t\tif(typeof newContent === 'string') newContent = [[module.id, newContent, '']];\n\n\t\tvar locals = (function(a, b) {\n\t\t\tvar key, idx = 0;\n\n\t\t\tfor(key in a) {\n\t\t\t\tif(!b || a[key] !== b[key]) return false;\n\t\t\t\tidx++;\n\t\t\t}\n\n\t\t\tfor(key in b) idx--;\n\n\t\t\treturn idx === 0;\n\t\t}(content.locals, newContent.locals));\n\n\t\tif(!locals) throw new Error('Aborting CSS HMR due to changed css-modules locals.');\n\n\t\tupdate(newContent);\n\t});\n\n\tmodule.hot.dispose(function() { update(); });\n}","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\n\nimport Storage from './Storage';\n\nclass LocalStorage extends Storage {\n  constructor() {\n    super();\n    this.store = 'local';\n  }\n}\n\nexport default LocalStorage;\n","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\n\nclass Notifications {\n\n  create(notificationId = null, options) {\n    // notificationId is optional\n    if(typeof notificationId === 'object') {\n      options = notificationId;\n      notificationId = null;\n    }\n\n    return new Promise((res, rej) => {\n      // resolve args = notificationId:string \n      chrome.notifications.create(notificationId, options, res);\n    });\n  }\n\n  update(notificationId, options) {\n    return new Promise((res, rej) => {\n      // resolve args = wasUpdated:boolean\n      chrome.notifications.update(notificationId, options, res);\n    });\n  }\n\n  clear(notificationId) {\n    return new Promise((res, rej) => {\n      // resolve args = wasCleared:boolean\n      chrome.notifications.clear(notificationId, res);\n    });\n  }\n\n  getAll() {\n    return new Promise((res, rej) => {\n      // resolve args = notifications:object\n      chrome.notifications.getAll(res);\n    });\n  }\n\n  getPermissionLevel() {\n    return new Promise((res, rej) => {\n      // resolve args = level:PermissionLevel (https://developer.chrome.com/apps/notifications#type-PermissionLevel)\n      chrome.notifications.getPermissionLevel(res);\n    });\n  }\n\n  listen(event, notificationId = null, callback) {\n    // event = 'onClosed' | 'onClicked' | 'onButtonClicked' | 'onPermissionLevelChanged' | 'onShowSettings'\n    // notificationId is optional\n    if(typeof notificationId === 'function') {\n      callback = notificationId;\n      notificationId = null;\n    }\n\n    if(event === 'onPermissionLevelChanged' || event === 'onShowSettings') {\n      return this._nonNotificationIdListen(event, callback);\n    }\n\n    /**\n     * https://developer.chrome.com/apps/notifications#events\n     * \n     * Resolve args (by event):\n     * onClosed = notificationId:string, byUser:boolean\n     * onClicked = notificationId:string\n     * onButtonClicked = notificationId:string, buttonIndex:integer\n     * \n     * onPermissionLevelChanged = level:PermissionLevel (https://developer.chrome.com/apps/notifications#type-PermissionLevel)\n     * onShowSettings = (none)\n     */\n\n    // This callback relates only to those events that have notificationId arg\n    const ListenerCallback = (() => {\n      if(notificationId !== null) {\n        return (passedNotificationId, ...args) => {\n          if(notificationId === passedNotificationId) {\n            callback(passedNotificationId, ...args);\n          }\n        };\n      } else {\n        return callback;\n      }\n    })();\n\n\n    chrome.notifications[event].addListener(ListenerCallback);\n  }\n\n  _nonNotificationIdListen(event, callback) {\n    chrome.notifications[event].addListener(callback);\n  }\n}\n\nexport default Notifications;\n","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\n\nclass Storage {\n\n  get(keys = null) {\n    return new Promise((res, rej) => {\n      const returnSingle = typeof keys === 'string' || typeof keys === 'number';\n      // resolve args = items:object\n      chrome.storage[this.store].get(keys, (items) => {\n        if(returnSingle) {\n          res(items[keys]);\n        } else {\n          res(items);\n        }\n      });\n    });\n  }\n\n  getBytesInUse(keys = null) {\n    return new Promise((res, rej) => {\n      // resolve args = bytesInUse:integer\n      chrome.storage[this.store].getBytesInUse(keys, res);\n    });\n  }\n\n  set(items) {\n    return new Promise((res, rej) => {\n      // resolve args = (none)\n      chrome.storage[this.store].set(items, res);\n    });\n  }\n\n  remove(keys) {\n    // resolve args = (none)\n    return new Promise((res, rej) => {\n      chrome.storage[this.store].remove(keys, res);\n    });\n  }\n\n  clear() {\n    // resolve args = (none)\n    return new Promise((res, rej) => {\n      chrome.storage[this.store].clear(res);\n    });\n  }\n\n  listen(item, onChange) {\n    if(typeof item === 'function') {\n      onChange = item;\n      item = null;\n    }\n\n    chrome.storage.onChanged.addListener((changes, areaName) => {\n      if(areaName === this.store) {\n        if(item !== null) {\n          if(changes.hasOwnProperty(item)) {\n            const oldValue = changes[item].oldValue || null;\n            const newValue = changes[item].newValue || null;\n            onChange(oldValue, newValue);\n          }\n        } else {\n          onChange(changes);\n        }\n      }\n    });\n  }\n}\n\nexport default Storage;\n","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\n\nimport Storage from './Storage';\n\nclass SyncStorage extends Storage {\n  constructor() {\n    super();\n    this.store = 'sync';\n  }\n}\n\nexport default SyncStorage;\n","/**\n* MIT License\n*\n* Copyright (c) 2020 wompmacho\n* \n* Permission is hereby granted, free of charge, to any person obtaining a copy\n* of this software and associated documentation files (the \"Software\"), to deal\n* in the Software without restriction, including without limitation the rights\n* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n* copies of the Software, and to permit persons to whom the Software is\n* furnished to do so, subject to the following conditions:\n* \n* The above copyright notice and this permission notice shall be included in all\n* copies or substantial portions of the Software.\n* \n* THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n* SOFTWARE.\n*/\n\nimport _LocalStorage from './LocalStorage';\nimport _SyncStorage from './SyncStorage';\nimport _Notifications from './Notifications';\n\n// export default {\n//   LocalStorage: new _LocalStorage,\n//   SyncStorage: new _SyncStorage,\n//   Notifications: new _Notifications\n// }\n\nexport const LocalStorage = new _LocalStorage;\nexport const SyncStorage = new _SyncStorage;\nexport const Notifications = new _Notifications;"],"mappings":";AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AClFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC3EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC3dA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;AC7YA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACxFA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACxEA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC7QA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACxCA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACzNA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACzQA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC/DA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;ACxLA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACxEA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;AC3EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACnBA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AClCA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AClHA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AC3FA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;AClCA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;A","sourceRoot":""}
|