Greasy Fork is available in English.

YouTube Me Again!

ytma! automatically converts YouTube(TM), Vimeo, Vine, Soundcloud, WebM, and MP4 links into real embedded videos.

// ==UserScript==
// Do not modify and re-release this script!
// If you would like to add support for other sites, please tell me and I'll put it in the includes.

// @id             youtube-me-again
// @name           YouTube Me Again!
// @namespace      hateradio)))
// @author         hateradio
// @version        8.0.1
// @description    ytma! automatically converts YouTube(TM), Vimeo, Vine, Soundcloud, WebM, and MP4 links into real embedded videos.
// @homepageURL    https://greasyfork.org/en/scripts/1023-youtube-me-again
// @icon           https://www.dropbox.com/s/b85qmq0bsim407s/ytma32.png?dl=1
// @icon64         https://www.dropbox.com/s/5zw3al38yf39wxb/ytma64.png?dl=1
// @screenshot     https://www.dropbox.com/s/syy9916b1prygl9/ytmascreen5.png?dl=1

// @include        https://vine.co/v/*/embed/simple
// @match          https://vine.co/v/*/embed/simple

// @include        https://gfycat.com/iframe/*
// @match          https://gfycat.com/iframe/*

// @include        http*://*.neogaf.com/threads/*
// @include        http*://*.resetera.com/threads/*

// @match          *://*.neogaf.com/threads/*
// @match          *://*.resetera.com/threads/*

// @updated        2020-06-16T08:47:28.845Z

// @grant          GM.xmlhttpRequest
// @grant          GM_xmlhttpRequest

// @run-at         document-end
// ==/UserScript==

/*

## Updates

#### 8.1

* Fix: Checks for supported embeds

#### 8

* Update YTMA to ESNext
* Fix: (XenoForo) YouTube embed

#### 7.9.1

* Fix: (XenoForo) Ignore links in more text areas

#### 7.9

* Fix: (XenoForo) Ignore links in text areas

#### 7.8

* Fix iFrame selectors
* Adds a monitoring interface to check for updated pages

#### 7.7

* New: YTMA will now try to find new links on AJAX-loaded posts on ResetEra
* Reverts Gfycat iFrame support; removes video tag for wider support
* Removes an outdated Chrome blacklist

#### 7.6

* Fix: overrides resetera support of embeded videos

#### 7.5

* New: Support for ResetEra
* Fix: Parses hours from YouTube URLs

#### 7.2.2

* Updates YouTube iFrame to hide related video feature when pausing

#### 7.2.1

* New: Extension info
* Updates JSHint options
* Removes outdated @include links

#### 7.2

* New: CSS rule to make videos fit within smaller windows
* New: GitHub repository and update links
* New: Streamable favicon
* Fix: Vimeo favicon

#### 7.1

* HTTPS links for Vimeo and Gfycat
* Fix: Safari bug

#### 7

* New: NeoGAF HTTPS Support
* New: Streamable.com added
* New: Soundcloud playlist support
* Improved time parser
* Upon scrolling, cached descriptions are shown
* Code reorganization makes adding new media sites easier

### 6

* New: Imgur GIFV (WEBM/MP4) support
* New: Button to remove cache (descriptions/thumbnail links/etc)
* New: SoundCloud playlist support
* Default video quality is now 720p/HD
* Soundcloud now uses HTML5 player
* Players that open on scroll will no longer trigger the opening of players higher on the page
* Adds HTML5, Gfycat, Imgur icons on links
* Improved Soundcloud and GfyCat URL matchers
* Restructured code base to simplify creation of media controls
* Restructured CSS
* Patched back Gfycat iFrame setting for Safari (it is incompatible with new settings)
* Updates YouTube data API
* Removes:
	* Object tag for YouTube for Flash (Deprecated)
	* "Batch" loading of descriptions (Only manual and scroll methods are supported)

// #Updates

Whitelist these sites on NoScript/NotScript/etc.
------------------------------------------------

* neogaf.com
* youtube.com
* youtube-nocookie.com
* googlevideo.com (HTML5 player sends videos from this domain)
* googleapis.com (YT video descriptions)
* vimeo.com
* vimeocdn.com
* soundcloud.com
* sndcdn.com
* vineco.com
* vine.com
* vine.co
* gfycat.com
* github.io


Whitelist these on Ghostery
---------------------------

* SoundCloud (Widgets, Audio / Music Player)

*/


/******/ (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 = "./src/index.js");
/******/ })
/************************************************************************/
/******/ ({

/***/ "./src/index.js":
/*!**********************!*\
  !*** ./src/index.js ***!
  \**********************/
/*! no exports provided */
/***/ (function(module, __webpack_exports__, __webpack_require__) {

"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony import */ var _modules_Helpers__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./modules/Helpers */ "./src/modules/Helpers.js");
/* harmony import */ var _modules___WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./modules/_ */ "./src/modules/_.js");
/* harmony import */ var _modules_strg__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./modules/strg */ "./src/modules/strg.js");
/* harmony import */ var _modules_update__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./modules/update */ "./src/modules/update.js");





(() => {

	/** Y T M A CLASS
	 * @private
	 * Base YTMA class, filled through constructor() or reactivate() though sub-classes
	 * Y's only concerned about the anchor and the data props
	 *
	 * @param {object} props Properties
	 * @param {String|Number} props.id Unique ID
	 * @param {String} props.site Website name eg: youtube, vimeo
	 * @param {HTMLAnchorElement} props.anchor Anchor element
	 */
	class Y {

		constructor({ id, site, anchor }) {
			const uid = Y.escapeId(`${id}_${Y.num += 1}`);

			this.state = {
				id,
				uid: Y.escapeId(uid), // unique id
				sid: Y.escapeId(id), // shared id
				site,
				uri: anchor.href
			};

			if (anchor && !anchor.dataset.ytmscroll) { anchor.dataset.ytmscroll = true; }

			this.anchor = anchor;
		}

		/**
		 * Recreates a YTMA object from a trigger element
		 * @param {HTMLElement} element the element's dataset for the resurection!
		 */
		reactivate({ dataset }) {
			const id = dataset.ytmid;
			const anchor = document.querySelector(`a[data-ytmuid="${dataset.ytmuid}"]`);

			this.state = {
				id,
				uid: dataset.ytmuid,
				sid: dataset.ytmsid,
				site: dataset.ytmsite,
				uri: anchor.href
			};

			this.anchor = anchor;

			return this;
		}

		disableOpenOnScroll() {
			this.anchor.dataset.ytmscroll = false;
		}

		canScroll() {
			return this.anchor.dataset.ytmscroll === 'true';
		}

		isBelow(link) {
			return Scroll.compare(this.anchor, link) < 1;
		}

		canShowUnder(link) {
			this.canScroll() && this.isBelow(link);
		}

		updateAnchor() {
			if (this.anchor.getElementsByTagName('img').length === 0) {
				this.anchor.classList.add('ytm_link', `ytm_link_${this.state.site}`);
			}
			this.anchor.dataset.ytmid = this.state.id;
			this.anchor.dataset.ytmuid = this.state.uid;
			this.anchor.dataset.ytmsid = this.state.sid;
			this.anchor.title = 'Visit the video page.';
		}

	}

	/** C O N T A I N E R CLASS
	 * The container, as the name implies, contains all the interactive elements
	 * Thumbnail, Player, Controls, etc.
	 */
	class Container extends Y {

		createInterface() {
			const { state } = this;
			this.site = Y.DB.sites[state.site];
			const { ajax, slim } = this.site;

			this.updateAnchor();

			this.body = _modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('div', {
				id: `w${state.uid}`,
				className: `ytm_spacer ytm_block ytm_site_${state.site}`,
				innerHTML: this.createThumbnailTemplate()
			});

			this.thumbnail = this.body.firstElementChild;

			if (ajax) { this.createAjaxLink(); }
			if (slim) { this.body.classList.add('ytm_site_slim'); }

			this.anchor.insertAdjacentElement('afterend', this.body);

			try {
				Container.decorators[state.site].gui(this);
			} catch (e) {
				// meh
			}
		}

		updateAnchor() {
			const { scroll, https } = this;
			if (scroll) { this.anchor.classList.add('ytm_scroll'); }
			if (https) { this.anchor.href = this.anchor.href.replace('http:', 'https:'); }

			super.updateAnchor();
		}

		createThumbnailTemplate() {
			const { title, thumb = '' } = this.site;
			const { id, uid, sid, site } = this.state;

			const bg = thumb ? `background-image: ${thumb.replace('%key', id)}` : '';

			const template = `
				<span class="ytm_trigger ytm_block ytm_normalize ytm_sans"
					title="${title}"
					data-ytmid="${id}"
					data-ytmsite="${id}"
					style="${bg}">
					<span class="ytm_init ytm_label ytm_sans ytm_box">${title}</span>
						<var class="ytm_label ytm_box"
							data-ytmid="${id}"
							data-ytmuid="${uid}"
							data-ytmsid="${sid}"
							data-ytmsite="${site}">\u25B6</var>
					</span>
				</span>`;
			return template;
		}

		createAjaxLink() {
			const { sid, id, site, uri } = this.state;
			const template = `
				<span class="ytm_bd ytm_normalize ytm_manual _${sid}">
					<a href="#" class="ytm_title" title="Load this video's description."
						data-ytmid="${id}"
						data-ytmsite="${site}"
						data-ytmuri="${uri}"
						data-ytmdescription="true"
					>Load Description</a>
				</span>`;
			this.body.insertAdjacentHTML('beforeend', template);
		}

		createProjector() {
			this.projector = _modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('div', {
				className: 'ytm_projector ytm_none ytm_block ytm_normalize ytm_sans',
				innerHTML: Container.templates.menu
			});
			this.thumbnail.insertAdjacentElement('afterend', this.projector);
		}

		showPlayer() {
			this.thumbnail.classList.add('ytm_none');
			this.projector.classList.remove('ytm_none');
		}

		hidePlayer() {
			this.thumbnail.classList.remove('ytm_none');
			this.projector.classList.add('ytm_none');
		}

	}

	Container.templates = {
		menu: `
			<ul class="ytm_options ytm_sans">
				<li>
					<ul class="ytm_ratios">
						<li data-type="ratio" data-value="1" title="SD">4:3</li>
						<li data-type="ratio" data-value="2" title="Landscape">16:9</li>
						<li data-type="ratio" data-value="3" title="Portrait">9:16</li>
					</ul>
				</li>
				<li>
					<ul class="ytm_sizes">
						<li data-type="size" data-value="0" title="Hide the video.">\u00D8</li>
						<li data-type="size" data-value="240" title="240p">S</li>
						<li data-type="size" data-value="360" title="360p">M</li>
						<li data-type="size" data-value="480" title="480p">L</li>
						<li data-type="size" data-value="720" title="720p">X</li>
					</ul>
				</li>
				<li>
					<ul class="ytm_options">
						${_modules_strg__WEBPACK_IMPORTED_MODULE_2__["default"].on ? '<li data-type="settings" data-value="" title="YTMA Settings">!</li>' : ''}
						<li data-type="close" data-value="" title="Close the video.">\u00D7</li>
					</ul>
				</li>
			</ul>`
	};

	Container.decorators = { // modify interface according to site
		youtube: {
			gui: function (control) {
				control.anchor.href = this.anchor.href.replace('youtu.be/', 'youtube.com/watch?v=');
			},
			thumbEvent: function (e) {
				let time = +this.dataset.time || 1;
				if (this.classList.contains('ytm_trigger') && e.type === 'mouseenter' && time < 50) {
					this.dataset.thumb = ((this.dataset.thumb || 0) + 1) % 3;
					this.style.backgroundImage = `url(https://i3.ytimg.com/vi/${this.dataset.ytmid}/${(+this.dataset.thumb) + 1}.jpg)`;
					window.clearTimeout(this.dataset.timeout);
					console.log('mouseenter -- clear before setting new ', this.dataset);
					this.dataset.timeout = window.setTimeout(Container.decorators.youtube.thumbEvent.bind(this, e), 800);
					console.log('mouseenter -- new timeout', this.dataset);
					this.dataset.time = time += 1;
				} else {
					window.clearTimeout(this.dataset.timeout);
					this.dataset.time = 0;
					console.log('mouseleave -- ', this.dataset);
				}
			}
		}
	};

	Container.events = {
		setup: () => {
			Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["on"])(document.body, 'click', 'var[data-ytmuid]', Container.events.fromTarget);
			Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["on"])(document.body, 'click', 'a[data-ytmdescription]', Container.events.manualLoad);
			Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["on"])(document.body, 'dblclick', 'q[data-full]', Container.events.titleToggle);

			Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["on"])(document.body, 'mouseenter mouseleave', 'div.ytm_site_youtube span.ytm_trigger', Container.decorators.youtube.thumbEvent);
		},
		fromTarget: ({ target }) => { // trigger the ui
			console.info('ytma//click+trig(id)', target.dataset.ytmuid);
			Control.createFromTrigger(target).showPlayer();
		},
		manualLoad: e => {
			e.preventDefault();
			const { target } = e;
			console.info('ytma//click+desc(id)', target.dataset.ytmid);
			if ((target.dataset.tries || 0) <= 4) {
				Y.ajax.loadFromDataset(target.dataset);
			}
		},
		titleToggle: e => {
			const target = e.target;
			target.classList.toggle('ytm_descr_open');
			target.textContent = target.textContent.length < 140 ? target.dataset.full : `${target.dataset.full.substr(0, 130)} . . .`;
			target.removeAttribute('style');
		}
	};

	Y.num = 0;

	Y.addToSet = ytma => Y.set[ytma.state.uid] = ytma;

	Y.create = link => Y.grabIdAndSite(link, (data, err) => {
		if (err) {
			console.warn(link.href, err);
			return {};
		}

		const control = new Control({ ...data, anchor: link });
		Y.addToSet(control);
		control.createInterface();

		return control;
	});

	Y.grabIdAndSite = (link, cb) => {
		let uri = link.href || link.src;
		let id;
		let match;
		try {
			const site = Y.reg.siteByTest[Y.reg.siteExpressions.test(uri) ? RegExp.lastMatch : ''];
			// console.log(site);

			if (site === 'html5') { // || site === 'html5-audio'
				id = uri.slice(-15);
			} else if (site === 'soundcloud') {
				if (!Y.reg.extra.soundcloud.playlist.test(uri)) {
					link.href = uri = Y.reg.fix.soundcloud(uri);
				}

				match = Y.DB.sites.soundcloud.matcher.exec(uri);
				id = Y.escapeId(match[1]);

				if (match && Y.reg.extra.soundcloud.tracks.test(uri)) {
					id = id.slice(-50);
				}
			} else {
				id = uri.match(Y.DB.sites[site].matcher)[1];
			}

			console.info('ytma//id+site', id, site, match);
			if (id && Y.DB.sites[site]) {
				return cb({ id, site }, null);
			}
			throw TypeError(`Invalid ID/Site: ${id} @ ${site}`);
		} catch (e) {
			return cb(null, e);
		}
	};

	Y.escapeId = id => `${id}`.replace(/(?:\W)/g, '_');

	Y.set = {};

	Y.collect = id => {
		const a = Object.values(Y.set).filter(ytma => ytma && ytma.data.id === id);
		return a;
	};

	Y.route = {
		host: document.location.host.replace('www.', ''),
		control: {
			$: {
				checkStorage: function () {
					if (_modules_strg__WEBPACK_IMPORTED_MODULE_2__["default"].full() === true) {
						console.log('YTMA ERROR: Storage is full!');
						try {
							localStorage.removeItem(Y.external.version);
							_modules_strg__WEBPACK_IMPORTED_MODULE_2__["default"].on = _modules_strg__WEBPACK_IMPORTED_MODULE_2__["default"].test();
						} catch (e) {
							console.error(e);
						}
					}
				},
				runOnce: function (loop) {
					if (!document.body.dataset.ytmaenabled) {
						document.body.dataset.ytmaenabled = true;

						this.checkStorage();

						if (!Y.DB.extension) { _modules_update__WEBPACK_IMPORTED_MODULE_3__["default"].check(); }

						Y.css();
						Y.user.init();
						Y.DB.postInit();

						if (loop) {
							document.body.dataset.YTMA_LOOP = window.setInterval(loop, 5000);
							loop();
						}

						Container.events.setup();
					}
				}
			},
			go: function (host) {
				console.info('ytma//host', host);
				if (/(?:googlevideo|youtube-nocookie\.com|youtube\.com\.?)/i.test(host)) {
					this.sites.youtube();
				} else if (this.sites[host]) {
					this.sites[host]();
				} else {
					this.sites.$generic();
				}
			},
			sites: {
				$generic: function () {
					function loop() {
						if (Y.selector.processor() > 0) {
							Y.user.fn.loadPreferences();
						}

						console.info('ytma//again++');
					}

					Y.route.control.$.runOnce(loop);
				},
				'resetera.com': function () {
					_modules___WEBPACK_IMPORTED_MODULE_1__["default"].css('.ytm_options li ul li { height: 24px !important }');
					_modules___WEBPACK_IMPORTED_MODULE_1__["default"].css('.bbCodeQuote .quoteContainer .quote { max-height: initial } .bbCodeQuote .quoteContainer .quoteExpand.quoteCut { display: none }');
					_modules___WEBPACK_IMPORTED_MODULE_1__["default"].css('.bbCodeQuote .ytm_block iframe, .bbCodeQuote .ytm_block [data-s9e-mediaembed], .bbCodeQuote .ytm_block .fb_iframe_widget, .bbCodeQuote .ytm_block object, .bbCodeQuote .ytm_block embed { max-height: initial; max-width: initial }');
					this.$generic();
				},
				'gfycat.com': function () {
					const v = document.querySelector('video');
					v.controls = true;
					_modules___WEBPACK_IMPORTED_MODULE_1__["default"].css('body,html {overflow:hidden;height:100%;width:100%} video {display:table;height:100%;margin:0 auto;}');
					document.body.appendChild(v);
				},
				'vine.co': function () {
					// console.log('vine.co');

					window.addEventListener('resize', () => {
						_modules___WEBPACK_IMPORTED_MODULE_1__["default"].s('[style]', e => {
							e.removeAttribute('style');
						});
					});
				},
				youtube: function () { // lets force some quality parity
				}
			}
		},
		load: function () {
			this.control.go(this.host);
		}
	};

	Y.main = () => {
		Y.reg.siteExpressions = Y.reg.getAllSiteRegExps();
		// console.log(YTMA.reg.siteExpressions);
		Y.route.load();
	};

	Y.reg = {
		siteExpressions: null,
		time: /(?:t=(?:(\d+)h)?(?:(\d+)m)?(\d+))/,
		ios: /(?:\b(?:ipod|iphone|ipad))\b/i,
		extra: {
			soundcloud: {
				playlist: /(?:soundcloud\.com\/.+\/sets\/)/,
				tracks: /(?:soundcloud\.com\/.+\/tracks\/)/
			}
		},
		siteByTest: {
			youtu: 'youtube',
			vimeo: 'vimeo',
			vine: 'vine',
			gfycat: 'gfycat',
			imgur: 'imgur',
			'.webm': 'html5',
			'.mp4': 'html5',
			// '.mp3': 'html5-audio',
			'.gifv': 'html5',
			soundcloud: 'soundcloud',
			'streamable.com': 'streamable'
		},
		getAllSiteRegExps: function () {
			const regs = Object.values(Y.DB.sites)
				.filter(({ reg }) => reg)
				.map(({ reg }) => reg);

			return new RegExp(`\\b${regs.join('|')}`);
		},
		fix: {
			soundcloud: function (uri) {
				const match = Y.DB.sites.soundcloud.matcher.exec(uri);
				if (match) {
					const id = match[1].split('/', 2).join('/');
					uri = Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["removeSearch"])(`https://soundcloud.com/${id}`, true);
				}

				return uri;
			}
		}
	};

	Y.selector = { // to build the selector
		parentBlacklist: ['.smallfont', '.colhead_dark', '.spoiler', 'pre', '.messageUserInfo', '.fr-box'],
		getAllSiteSelectors: function () {
			const sels = Object.values(Y.DB.sites)
				.filter(({ selector }) => selector)
				.map(({ selector }) => selector);

			return sels.join();
		},
		ignore: function () {
			const ignore = [];
			const all = Y.selector.getAllSiteSelectors().split(',');
			const blacklist = this.parentBlacklist;
			for (let i = 0; i < blacklist.length; i++) {
				for (let j = 0; j < all.length; j++) {
					ignore.push(`${blacklist[i]} ${all[j]}`);
				}
			}
			// console.log(ignore.join(','));
			return ignore.join(',');
		},
		iframes: function () { // for resetera, convert iframes back to anchors
			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].s('.message-body iframe', f => {
				if (/vi\/(.+?)\/hqdefault/.test(f.style.backgroundImage)) {
					const src = `https://youtu.be/${RegExp.$1}`;
					const span = f.closest('[data-s9e-mediaembed]');
					span.insertAdjacentHTML('beforebegin', `<a href="${src}">youtube</a>`);
					span.parentElement.removeChild(span);
				}
			});

			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].s('[data-s9e-mediaembed-iframe]', s => {
				if (Y.DB.sites[s.dataset.s9eMediaembed]) {
					const dat = JSON.parse(s.dataset.s9eMediaembedIframe);
					const link = dat[dat.length - 1];
					s.parentElement.parentElement.innerHTML = `<a href="${link}">${link}</a>`;
				}
			});
		},
		links: function () {
			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].s(Y.selector.ignore(), ({ dataset }) => dataset.ytmaignore = true);

			const links = _modules___WEBPACK_IMPORTED_MODULE_1__["default"].qsa(Y.selector.getAllSiteSelectors()).filter(({ dataset }) => {
				const r = !dataset.ytmaprocessed && !dataset.ytmaignore;
				dataset.ytmaprocessed = true;
				return r;
			});

			return links;
		},
		processor: function () {
			this.iframes();
			const links = this.links();

			if (links.length > 0) {
				links.forEach(Y.create);
			}

			return links.length;
		}
	};

	/**
	 * User Preferences
	 * size: Small (240p), Medium (360p), Large (480p), XL (720p)
	 * ratio: 1 4:3, 2 16:9
	 * quality: 240, 360, 480, 720, 1080
	 * focus: 0/1; Will attempt to set the window's focus near the video
	 * autoShow: 0/1; Will automatically display HTML5 videos, which currently lack descriptions and thumbnails
	 * desc: (Descriptions) 0 None; 1 Yes on scroll; 2 Yes all at once
	 * yt_nocookie: 0/1; Will disable/enable youtube-nocookie.com
	 * yt_annotation: 0/1; youtube annotations
	 */
	Y.user = {
		KEY: 'ytmasetts',
		$form: null,
		init: function () {
			this.load();

			if (_modules_strg__WEBPACK_IMPORTED_MODULE_2__["default"].on) {
				this.fn.makeForm();
				this.mark();
			}
		},
		valid: {
			focus: [0, 1],
			desc: [0, 1, 2],
			ratio: [1, 2],
			size: [240, 360, 480, 720],
			quality: [240, 360, 480, 720, 1080],
			autoShow: [0, 1],
			yt_nocookie: [0, 1],
			yt_annotation: [0, 1] // hide | show
		},
		mapping: { // map values to some other values used by an external API, for example
			yt_annotation: [3, 1] // 3 = hide | 1 = show
		},
		validate: function (property, n) {
			n = +n;
			return Y.user.valid[property].includes(n) ? n : Y.user.defaults[property];
		},
		get defaults() {
			return {
				focus: 0,
				desc: 1,
				ratio: 2,
				size: 360,
				quality: 720,
				autoShow: 1,
				yt_nocookie: 0,
				yt_annotation: 1
			};
		},
		load: function () {
			const s = _modules_strg__WEBPACK_IMPORTED_MODULE_2__["default"].grab(Y.user.KEY, {});

			Y.user.preferences = Object.keys(this.defaults).reduce((valid, k) => {
				valid[k] = Y.user.validate(k, s[k]);
				return valid;
			}, {});

			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].o(Y.user.mapping, (key, val) => {
				if (!val.hasOwnProperty('indexOf')) {
					Y.user.preferences[key] = val[Y.user.valid[key].indexOf(Y.user.preferences[key])];
				}
			});

			console.info('ytma//user+loaded(prefs)', Y.user.preferences);
		},
		mark: function () {
			const a = {};
			a.ytma__focus = !!Y.user.preferences.focus;
			a.ytma__autoShow = !!Y.user.preferences.autoShow;
			a.ytma__yt_nocookie = !!Y.user.preferences.yt_nocookie;
			a.ytma__yt_annotation = !!Y.user.preferences.yt_annotation;
			a[`ytma__ratio${Y.user.preferences.ratio}`] = true;
			a[`ytma__size${Y.user.preferences.size}`] = true;
			a[`ytma__desc${Y.user.preferences.desc}`] = true;
			a[`ytma__quality${Y.user.preferences.quality}`] = !!Y.user.preferences.quality;

			// console.log('marking', a);
			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].o(a, (id, val) => {
				try {
					const el = document.getElementById(id);
					el.checked = val;
					el.value = val;
				} catch (e) {
					// console.log(id, e);
				}
			});
		},
		events: {
			save: function () {
				// console.log(YTMA.user.$form.querySelectorAll('[data-key]'));
				// [data-key]:checked
				const settings = Array.from(Y.user.$form.querySelectorAll('[data-key]')).reduce((obj, e) => {
					let key = e.dataset.key;

					if (e.type === 'checkbox') {
						obj[key] = +e.checked;
					} else if (e.type === 'radio') {
						if (e.checked) {
							if (e.dataset.num) {
								obj[key] = +e.dataset.num;
							}
						}
					} else {
						obj[key] = +e.value;
					}

					return obj;
				}, {});

				if (_modules_strg__WEBPACK_IMPORTED_MODULE_2__["default"].save(Y.user.KEY, settings)) {
					Y.user.load();
				} else {
					Y.user.error.classList.remove('ytm_none');
				}

			},
			reset: function () {
				Y.user.preferences = Y.user.defaults;
				Y.user.mark();
				_modules_strg__WEBPACK_IMPORTED_MODULE_2__["default"].wipe(Y.user.KEY);
				Y.user.error.classList.add('ytm_none');
			},
			clear: function () {
				try {
					localStorage.removeItem(Y.external.version);
					Y.user.events.reset();
					console.info('ytma//cache+remove', 'removed all YTMA cache');
				} catch (e) {
					console.error(e);
				}
			},
			formToggle: function (e) {
				if (Y.user.$form && (!e || (e && e.target && !(/(?:INPUT|LABEL)/i).test(e.target.nodeName)))) {
					Y.user.$form.classList.toggle('ytm_none');
				}
			},
			formToggleKeyboard: function (e) {
				// press CTRL+SHIFT+Y (META+SHIFT+Y) to display settings form
				if ((e.ctrlKey || e.metaKey) && e.shiftKey && String.fromCharCode(e.which).toLowerCase() === 'y') {
					e.preventDefault();
					Y.user.events.formToggle();
				}
			}
		},
		fn: {
			$scroller: null,
			$once: false,
			loadPreferences: function () {
				Y.user.fn.onScrollLoadDescriptions(Y.user.preferences.desc === 1);

				this.loadPreferencesOnce();
			},
			loadPreferencesOnce: function () {
				if (this.$once) { return; }

				this.$once = true;

				if (Y.user.preferences.autoShow === 1) {
					Y.user.fn.onScrollViewMedia();
				}
			},
			showMedia: function () {
				// console.info('ytma//user+fn-showMedia');
				return new Scroll('a.ytm_scroll:not([data-ytmscroll="false"])', link => {
					if (Scroll.visibleAll(link, 50)) {
						_modules___WEBPACK_IMPORTED_MODULE_1__["default"].s(`var[data-ytmsid="${link.dataset.ytmsid}"]:not([data-ytmscroll="false"])`, trigger => {
							const ui = Control.createFromTrigger(trigger);
							ui.showOnScroll(link);
						});
					}
				});
			},
			toggleMedia: function () {
				return new Scroll('div.ytm_panel_switcher', div => {
					const v = div.querySelector('video');
					const paused = v && (v.paused || v.ended);
					const ui = Y.set[div.dataset.ytmuid].getControl();

					if (paused && !Scroll.visibleAll(div, 0)) {
						return ui.play.switchStandby();
					}

					if (ui.play.isStandby() && Scroll.visibleAll(div, 200)) {
						return ui.play.switchOn();
					}

					// todo ascertain embedded player properties
					// f = div.querySelector('iframe, object');
					// if (f && !YTMA.Scroll.visibleAll(div, 200)) {
					// 	y.hidePlayer();
					// }
				});
			},
			onScrollViewMedia: function () {
				this.showMedia();
				this.toggleMedia();
			},
			onScrollLoadDescriptions: function (ajax) {
				if (Y.user.fn.$scroller) { Y.user.fn.$scroller.stop(); }

				Y.user.fn.$scroller = new Scroll('span.ytm_manual > a.ytm_title:not(.ytm_error)', a => {
					if (Scroll.visibleAll(a, 200)) {
						if (ajax) {
							Y.ajax.loadFromDataset(a.dataset);
						} else {
							Y.ajax.loadFromCacheDataset(a.dataset);
						}
						// console.log('doc', document.querySelectorAll(YTMA.user.fn.$scroller.selector).length, a.dataset.id);
					}

					if (document.querySelectorAll(Y.user.fn.$scroller.selector).length === 0) {
						Y.user.fn.$scroller.stop();
					}
				});
			},
			makeForm: function () {
				const template = `
					<div id="ytm_settings" class="ytm_sans ytm_block ytm_normalize">
						<form action="" title="Double click to close">
							<div id="ytm_settingst">ytma! Site Settings</div><div class="ytm_field_container">
								<fieldset><legend title="Load descriptions from the content sever.">Load Descriptions</legend><p><span><input id="ytma__desc0" type="radio" data-num="0" name="ytma__desc" data-key="desc"><label for="ytma__desc0" title="Load descriptions on demand">Manually</label></span><span><input id="ytma__desc1" type="radio" data-num="1" name="ytma__desc" data-key="desc"><label for="ytma__desc1" title="Load descriptions as they become visible on the screen.">Automatically, on scrolling</label></span></p></fieldset>
								<fieldset><legend>HTML5 Players</legend><p><input name="ytma__autoShow" data-key="autoShow" id="ytma__autoShow" type="checkbox"><label for="ytma__autoShow">Automatically show WebM, MP4 and Soundcloud players</label></p></fieldset>
								<fieldset><legend>Player Size</legend><p><span><input type="radio" name="ytma__size" data-key="size" data-num="240" id="ytma__size240" /><label for="ytma__size240">S <small>240p</small></label></span><span><input name="ytma__size" data-key="size" type="radio" id="ytma__size360" data-num="360" /><label for="ytma__size360">M <small>360p</small></label></span><span><input type="radio" name="ytma__size" data-key="size" data-num="480" id="ytma__size480" /><label for="ytma__size480">L <small>480p</small></label></span><span><input type="radio" name="ytma__size" data-key="size" data-num="720" id="ytma__size720" /><label for="ytma__size720">X <small>720p</small></label></span></p></fieldset>
								<fieldset><legend>Quality</legend><p><span><input name="ytma__quality" data-key="quality" data-num="240" id="ytma__quality240" type="radio"><label for="ytma__quality240">240p</label></span><span><input name="ytma__quality" data-key="quality" id="ytma__quality360" data-num="360" type="radio"><label for="ytma__quality360">360p</label></span><span><input name="ytma__quality" data-key="quality" data-num="480" id="ytma__quality480" type="radio"><label for="ytma__quality480">480p</label></span><span><input name="ytma__quality" data-key="quality" data-num="720" id="ytma__quality720" type="radio"><label for="ytma__quality720">720p</label></span><span><input name="ytma__quality" data-key="quality" data-num="1080" id="ytma__quality1080" type="radio"><label for="ytma__quality1080">1080p</label></span></p></fieldset>
								<fieldset><legend>Aspect Ratio</legend><p><span><input name="ytma__ratio" data-key="ratio" type="radio" id="ytma__ratio2" data-num="2" /><label for="ytma__ratio2">16:9</label></span><span><input type="radio" name="ytma__ratio" data-key="ratio" data-num="1" id="ytma__ratio1" /><label for="ytma__ratio1">4:3</label></span></p></fieldset>
								<fieldset><legend>YouTube</legend>
									<p><input name="ytma__yt_annotation" data-key="yt_annotation" type="checkbox" id="ytma__yt_annotation" /><label for="ytma__yt_annotation">Enable video annotations</label></p>
									<p><input name="ytma__yt_nocookie" data-key="yt_nocookie" type="checkbox" id="ytma__yt_nocookie" /><label for="ytma__yt_nocookie">Use https://youtube-nocookie.com to load videos</label></p>
								</fieldset>
								<fieldset><legend>Window Focus</legend><p><input name="ytma__focus" data-key="focus" type="checkbox" id="ytma__focus" value="focus" /><label for="ytma__focus">After clicking the thumbnail, set the video at the top of the window.</label></p></fieldset>
							</div>
							<p><small id="ytm_settings_error" class="ytm_error ytm_none ytm_title">Error! Your settings could not be saved.</small></p>
							<p id="ytm_opts">
								<button type="button" id="ytmaclose">Close</button> <button type="button" id="ytmareset">Reset</button> <button type="button" id="ytmaclear" title="Remove all video descriptions that have been cached">Reset & Remove Cache</button>
							</p>
						</form>
					</div>`;

				Y.user.$form = _modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('div', { className: 'ytm_fix_center ytm_none ytm_box', innerHTML: template }, document.body);
				Y.user.error = document.getElementById('ytm_settings_error');

				Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["on"])(Y.user.$form, 'keyup click', 'input, label', Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["debounce"])(Y.user.events.save, 500));
				Y.user.$form.addEventListener('submit', e => e.preventDefault(), false);

				document.getElementById('ytmareset').addEventListener('click', Y.user.events.reset, false);
				document.getElementById('ytmaclear').addEventListener('click', Y.user.events.clear, false);

				// close
				Y.user.$form.addEventListener('dblclick', Y.user.events.formToggle, false);
				document.getElementById('ytmaclose').addEventListener('click', Y.user.events.formToggle, false);
				document.body.addEventListener('keydown', Y.user.events.formToggleKeyboard, false);
			}
		}
	};

	Y.css = () => {
		const playerCss = Player.css.generator();
		const loadingIcon = 'data:image/gif;base64,R0lGODlhDgAKAJEAAP///+BKV////wAAACH/C05FVFNDQVBFMi4wAwEAAAAh+QQFCgACACwAAAAADgAKAAACHFSOeQYI71p6MtAJz41162yBH+do5Ih1kKG0QgEAIfkEBQoAAgAsAAABAA0ACAAAAhSUYGEoerkgdIzKGlu2ET/9ceJmFAAh+QQFCgACACwAAAEADQAIAAACFJRhcbmiglx78SXKYK6za+NxHyYVACH5BAUKAAIALAAAAQANAAgAAAIWVCSAl+hqEGRTLhtbdvTqnlUf9nhTAQAh+QQFCgACACwAAAEADQAIAAACFZRiYCh6uaCRzNXYsKVT+5eBW3gJBQAh+QQJCgACACwAAAAADgAKAAACGpSPaWGwfZhwQtIK8VTUvuxpm9Yp4XlmpiIUADs=';

		// console.log(playerCss);
		_modules___WEBPACK_IMPORTED_MODULE_1__["default"].css(playerCss);

		// images
		// todo update(site, size, padding)
		_modules___WEBPACK_IMPORTED_MODULE_1__["default"].css(`
			.ytm_loading{background:url(${loadingIcon}) 0 3px no-repeat;}
			.ytm_link{position:relative !important;background:url(${Y.DB.sites.youtube.favicon}) 0 center no-repeat !important;margin-left:4px;padding-left:20px!important;}
			.ytm_link.ytm_link_vimeo{background-image:url(${Y.DB.sites.vimeo.favicon}) !important;background-size:12px 12px !important;padding-left:18px!important}
			.ytm_link.ytm_link_vine{background-image:url(${Y.DB.sites.vine.favicon}) !important;background-size:10px 10px!important;padding-left:16px!important}
			.ytm_link.ytm_link_soundcloud{background-image:url(${Y.DB.sites.soundcloud.favicon})!important;padding-left:17px!important}
			.ytm_link.ytm_link_html5{background-image:url(${Y.DB.sites.html5.favicon}) !important;padding-left:16px!important}
			.ytm_link.ytm_link_gfycat{background-image:url(${Y.DB.sites.gfycat.favicon}) !important;background-size:12px 12px !important;padding-left:16px!important;}
			.ytm_link.ytm_link_imgur{background-image:url(${Y.DB.sites.imgur.favicon}) !important;background-size:12px 12px !important;padding-left:16px!important}
			.ytm_link.ytm_link_streamable{background-image:url(${Y.DB.sites.streamable.favicon}) !important; background-size: 12px 12px !important;padding-left: 14px !important;}
		`);

		_modules___WEBPACK_IMPORTED_MODULE_1__["default"].css('.ytm_none,.ytm_link br{display:none!important}.ytm_box{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.ytm_block{display:block;position:relative;clear:both;text-align:left;border:0;margin:0;padding:0;overflow:hidden}.ytm_normalize{font-weight:400!important;font-style:normal!important;line-height:1.2!important}.ytm_sans{font-family:Arial,Helvetica,sans-serif!important}.ytm_spacer{overflow:auto;margin:0 0 6px;padding:4px}.ytm_spacer.ytm_site_slim{display:inline}.ytm_clear:after{content:"";display:table;clear:both}.ytm_center{text-align:center}.ytm_link b,.ytm_link strong{font-weight:400!important}.ytm_link u{text-decoration:none!important}.ytm_link i,.ytm_link em{font-style:normal!important}.ytm_trigger{width:118px;height:66px;background-color:#262626!important;cursor:pointer;background-position:-1px -12px;float:left;box-shadow:2px 2px rgba(0,0,0,.3);background-size:auto 90px!important;color:#fff;text-shadow:#333 0 0 2px;font-size:13px}.ytm_trigger:hover{box-shadow:2px 2px #60656b80;opacity:.95}.ytm_trigger var{z-index:2;height:100%;width:100%;position:absolute;left:0;top:0;text-align:right}.ytm_label{display:block;padding:3px 6px;line-height:1.2;font-style:normal}.ytm_init{height:22px;background:rgba(11,11,11,.62);padding:4px 25px 6px 6px}.ytm_site_vine .ytm_trigger{background-color:#90ee90!important;background-size:120px auto!important}.ytm_site_slim .ytm_trigger{background:#e34c26!important;height:auto;box-shadow:0 0 2px #ffdb9d inset,2px 2px rgba(0,0,0,.3);margin:0 3px 0 0;width:auto;transition:all .3s ease-in-out 0s}.ytm_site_slim .ytm_trigger:hover{opacity:.8}.ytm_site_slim .ytm_label{text-shadow:0 0 1px #f06529}.ytm_site_slim .ytm_init{background:transparent}.ytm_bd{float:left;max-width:450px;margin:2px 10px;font-size:12px}.ytm_title{font-weight:700}.ytm_error{color:#cc2f24;font-style:italic}.ytm_loading{font-style:italic;padding:1px 1.5em}.ytm_descr{word-wrap:break-word;max-height:48px;overflow:auto;padding-right:20px}.ytm_descr[data-full]{cursor:pointer}.ytm_descr_open{resize:both;white-space:pre-line;background:linear-gradient(to bottom, rgba(0,0,0,0) 0%,rgba(0,0,0,0) 50%,rgba(0,0,0,0) 80%,rgba(0,0,0,0.1) 100%)}.ytm_descr_open[style]{max-height:none}.ytm_projector{margin-bottom:4px}ul.ytm_options{overflow:hidden;margin:0!important;padding:3px 0 1px;list-style-position:outside!important}.ytm_options li{display:inline;margin:0!important;padding:0!important}.ytm_options li>ul{display:inline-block;margin:0;padding:0 1px 0 0}.ytm_options li ul li{-webkit-user-select:none;-moz-user-select:none;-o-user-select:none;user-select:none;list-style-type:none;cursor:pointer;float:left;color:#858585;border:1px solid #1d1d1d;border-bottom:1px solid #181818;border-top:1px solid #292929;box-shadow:0 0 1px #555;height:14px;font-size:12px!important;line-height:12px!important;background:#222;background:linear-gradient(#2d2c2c,#222);margin:0!important;padding:5px 9px 3px!important}.ytm_options li ul li:first-child{border-radius:2px 0 0 2px}.ytm_options li ul li:last-child{border-left:0!important;border-radius:0 2px 2px 0;margin:0 2px 0 0!important}.ytm_options li ul li:first-child:last-child,.ytm_li_setting{border-radius:2px}.ytm_options li ul li:hover{color:#ccc;text-shadow:1px 1px 0 #333;background:#181818}.ytm_options li ul li[id]{color:#ddd;text-shadow:0 0 2px #444}.ytm_panel_size{background:#000;max-width:100%;}.ytm_panel_switcher[data-standby="true"]{background:#111}.ytm_panel_switcher[data-standby="true"]:after{cursor:cell;color:#0e0e0e;content:"ytma!";display:block;font-size:85px;font-style:italic;font-weight:700;left:50%;position:absolute;text-shadow:2px 1px #181818,-1px -1px #0a0a0a;top:50%;transform:translate(-50%,-50%)}.ytm_site_soundcloud .ytm_panel_size.ytm_soundcloud-playlist{height:334px!important}.ytm_fix_center{background:rgba(51,51,51,.41);height:100%;left:0;position:fixed;top:0;width:100%;z-index:99998}#ytm_settings{z-index:99999;max-width:500px;max-height:85%;overflow:auto;background:#fbfbfb;border:1px solid #bbb;color:#444;box-shadow:0 0 5px rgba(0,0,0,.2),0 0 3px rgba(239,239,239,.1) inset;margin:4% auto;padding:4px 8px 0}#ytm_settings p{margin:5px 0;padding:0}#ytm_settings fieldset{vertical-align:top;border-radius:3px;border:1px solid #ccc;margin:0 0 5px;padding:3px}#ytm_settings legend{padding:3px}#ytm_settings fieldset span{display:inline-block;min-width:5em}#ytm_settings input{vertical-align:baseline!important;margin:3px 5px!important}#ytm_settingst{font-size:110%;border-bottom:1px solid #d00;margin:3px 0 9px;padding:0 3px 3px}#ytm_settings label{cursor:pointer}#ytm_settings small{font-size:90%}#ytm_opts button{cursor:pointer;margin:10px 5px 8px 2px;padding:3px;border:1px solid #adadad;border-radius:2px;background:#eee;font-size:90%}#ytm_opts button:hover{background:#ddd}');
	};

	Y.ajax = {
		load: function (site, id, uri) {
			console.info('ytma//ajax+load(id)', site, id, uri);
			uri = Y.DB.sites[site].ajax.replace('%key', id).replace('%uri', uri);

			if (Y.DB.sites[site].ajaxExtension) { return this.gmxhr(uri, site, id); }

			console.info('ytma//ajax+load(uri)', Y.DB.sites[site].ajax.replace('%key', id).replace('%uri', uri));
			if (Y.DB.sites[site].ajax) {
				// console.log('preping uri');
				return this.xhr(uri, site, id);
			}

			return null;
		},
		loadFromDataset: function (dataset) {
			if (!this.loadFromCacheDataset(dataset)) {
				return this.load(dataset.ytmsite, dataset.ytmid, dataset.ytmuri);
			}
		},
		loadFromCacheDataset: function ({ ytmsite, ytmid }) {
			const cache = Y.external.dataFromStorage(ytmsite, ytmid);

			console.info('ytma//ajax+cache(id)', ytmsite, ytmid);
			console.info('ytma//ajax+cache(data)', cache);

			if (cache) { Y.external.populate(cache); }

			return cache;
		},
		gmxhr: function (uri, site, id) {
			try {
				// console.log('gmxhr starting!');
				GM.xmlhttpRequest({
					method: 'GET',
					url: uri,
					onload: function ({ responseText }) {
						// console.log(response);
						Y.external.parse(responseText, site, id);
					},
					onerror: function () {
						console.log('GM Cannot XHR');
						Y.ajax.failure.call({ id });
					}
				});

				Y.ajax.preProcess(id);

			} catch (e) {
				if (Y.DB.extension) {
					console.info('ytma//gmxhr-cors');
					this.xhr(uri, site, id);
				} else {
					console.log('No applicable CORS request available.');
					this.failure.call({ id });
				}
			}
		},
		xhr: function (uri, site, id) {
			const x = new XMLHttpRequest();
			console.info('ytma//xhr', uri, id, site);

			Y.ajax.preProcess(id);

			x.onreadystatechange = function () {
				if (this.readyState === this.DONE) {
					// console.log(this.readyState, this.status);
					if (this.status === 200) {
						Y.external.parse(this.responseText, site, id);
					} else if (this.status === 403) {
						Y.external.populate({ site, id, title: 'Error 403', desc: '' });
						Y.external.save({ site, id, title: 'Error 403', desc: '' });
					} else { // if (this.status >= 400 || this.status === 0) {
						Y.ajax.failure.call({ id });
					}
				}
			};

			try {
				// console.info('ytma//xhr+sending');
				x.open('get', uri, true);
				x.send();
			} catch (e) {
				console.error('ytma//xhr+failed(cannot send xhr)', uri);
				Y.ajax.failure.call({ id });
				console.error(e);
			}
		},
		failure: function () {
			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].s(`.ytm_bd._${Y.escapeId(this.id)}`, el => {
				const a = el.querySelector('a');
				a.dataset.tries = a.dataset.tries ? parseFloat(a.dataset.tries) + 1 : 1;
				if (a.dataset.tries >= 5) {
					a.textContent = 'Max attempts reached';
				} else {
					a.textContent = `Error, unable to load data.${a.dataset.tries > 1 ? '' : ' [Retry]'}`;
				}
				a.className = 'ytm_error ytm_title';
			});
		},
		preProcess: function (id) {
			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].s(`.ytm_manual._${Y.escapeId(id)} a`, el => {
				el.classList.add('ytm_loading');
				el.textContent = 'Loading';
				el.title = 'Retry loading data.';
			});
		}
	};

	/** E X T E R N A L Apparatus
	 * Data from external sites
	 */
	Y.external = {
		version: 'ytma.4.1.dat',
		parse: function (response, site, id) {
			if (this.parsers[site]) {
				response = Y.DB.sites[site].rawResponse ? response : JSON.parse(response);
				this.populate(this.helper.cutDescription(this.parsers[site](response, id)));
			}
		},
		parsers: {
			soundcloud: function ({ title, description, thumbnail_url }, id) {
				return {
					site: 'soundcloud',
					id, //unescape(j.html).match(/tracks\/(\d+)/)[1],
					title,
					desc: description,
					th: Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["removeSearch"])(thumbnail_url)
				};
			},
			vimeo: function (j) {
				j = j[0];
				return {
					site: 'vimeo',
					id: j.id,
					title: `${j.title} ${Y.external.time.fromSeconds(j.duration)}`,
					desc: j.description.replace(/<br.?.?>/g, ''),
					th: decodeURI(j.thumbnail_medium)
				};
			},
			youtube: function (j, id) {
				if (j.pageInfo.totalResults < 1 || j.items.length === 0) {
					return { id, error: true };
				}

				j = j.items[0];
				const o = {
					site: 'youtube',
					id,
					title: `${j.snippet.title} ${Y.external.time.fromIso8601(j.contentDetails.duration)}`,
					desc: j.snippet.description
					// aspectRatio: j.contentDetails.aspectRatio
				};

				return o;
			},
			vine: function ({ title, thumbnail_url }, id) {
				return {
					site: 'vine',
					id,
					title,
					th: Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["removeSearch"])(thumbnail_url)
				};
			},
			gfycat: function (html, id) {
				if (html) {
					return {
						site: 'gfycat',
						id: id,
						title: id
					};
				}
			},
			streamable: function ({ title }, id) {
				return {
					site: 'streamable',
					id,
					title: title || 'Untitled'
				};
			}
		},
		set: function (data) {
			if (!this.db[data.site]) {
				this.db[data.site] = {};
			}
			this.db[data.site][data.id] = data;
			return this.save();
		},
		unset: function ({ site, id }) {
			// console.log('unset', data.id);
			if (site) {
				delete this.db[site][id];
				return this.save();
			}
		},
		limitDB: function (max, db) {
			// limits an object's items by half of the max
			// removes the older items at the start of the object
			const keys = Object.keys(db);

			const half = Math.floor(max / 2);
			let start;
			let ndb;
			let i;

			if (keys.length > max) {
				ndb = {};
				start = keys.length - half;

				for (i = start; i < keys.length; i++) {
					ndb[keys[i]] = db[keys[i]];
				}
			}

			return ndb || db;
		},
		save: function () {
			this.db = this.limitDB(1000, this.db);
			return _modules_strg__WEBPACK_IMPORTED_MODULE_2__["default"].save(this.version, this.db);
		},
		helper: {
			cutDescription: function (data) {
				if (data.desc && data.desc.length > 140) {
					data.full = data.desc;
					data.desc = `${data.desc.substr(0, 130)} . . .`;
				}
				return data;
			},
			thumbnail: function ({ id, th }) {
				_modules___WEBPACK_IMPORTED_MODULE_1__["default"].s(`[data-ytmid="${id}"].ytm_trigger`, el => el.setAttribute('style', `background: url(${th})`));
			}
		},
		time: {
			keepMinutesAndSeconds: function (v, i) {
				return i > 1 || v > 0;
			},
			leadingZero: function (v, i) {
				return i > 0 ? (`00${v}`).slice(-2) : v;
			},
			fromArray: function (a) {
				// [days, hours, mins, secs]
				let b;

				let p = '';

				try {
					// Remove empty values, but keep lower indexes (m:s); a[i] > 0 || i > 1
					// Add leading 0's, ignoring the first index
					// a.slice(0, 1).concat(a.slice(1))
					b = a.filter(this.keepMinutesAndSeconds).map(this.leadingZero);
					p = `(${b.join(':')})`;
				} catch (e) {
					console.error('Could not parse this time.');
				}

				console.info('ytma//time+array', { a, b, p });
				return p;
			},
			fromIso8601: function (iso8601) {
				// eg PT3M, T29S
				let a;

				const parseDigits = reg => {
					if (reg.test(iso8601)) {
						return RegExp.lastParen;
					}
					return 0;
				};

				// P#DT#H#M#S || PT#H#M#S
				a = [/(\d+)D/, /(\d+)H/, /(\d+)M/, /(\d+)S/].map(parseDigits);

				return this.fromArray(a);
			},
			fromSeconds: function (seconds) {
				const a = [
					Math.floor(seconds / 86400) % 24,
					Math.floor(seconds / 3600) % 60,
					Math.floor(seconds / 60) % 60,
					seconds % 60
				];
				return this.fromArray(a);
			}
		},
		validate: function (data) {
			if (!data || !data.id || data.error) {
				return Y.ajax.failure.call(data);
			}

			// todo? empty titles and descriptions should be okay
			// if (data.id && !data.title && !data.desc) {
			// 	this.unset(data.id);
			// 	return YTMA.ajax.failure.call(data);
			// }

			return true;
		},
		populate: function (data, ignoreValidation) {
			if (!ignoreValidation && !this.validate(data)) { return; }

			this.set(data);
			const { id, th, full, desc, title } = data;

			if (th) { this.helper.thumbnail(data); }

			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].s(`.ytm_bd._${Y.escapeId(id)}`, el => {
				el.innerHTML = `<span class="ytm_title">${title}</span>`;
				if (desc) {
					const q = _modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('q', { className: 'ytm_descr ytm_block', textContent: desc }, el);
					if (full.length > desc.length) {
						q.dataset.full = full;
						q.title = 'Double click to toggle the description.';
					}
				}
			});
		},
		dataFromStorage: function (site, id) {
			if (this.db && this.db[site]) {
				return this.db[site][id];
			}
		}
	};
	Y.external.db = _modules_strg__WEBPACK_IMPORTED_MODULE_2__["default"].grab(Y.external.version, {});

	/** Database */
	Y.DB = {
		postInit: function () {
			if (Y.user.preferences.yt_nocookie) {
				Y.DB.sites.youtube.home = 'https://www.youtube-nocookie.com/';
				Y.DB.sites.youtube.embed = 'https://www.youtube-nocookie.com/embed/%key';
			}
		},
		extension: window.chrome && window.chrome.extension,
		sites: { // supported sites - to add more also make a parser (if api is available) and add an item to sources (if necessary)
			youtube: {
				title: 'ytma!',
				home: 'https://www.youtube.com/',
				embed: 'https://www.youtube.com/embed/%key',
				ajax: `https://www.googleapis.com/youtube/v3/videos?part=snippet,contentDetails&id=%key${window.atob('JmtleT1BSXphU3lEVG5INkxzRERyVElYaFZTZWRQQjlyRHo1czBSczQzZnM=')}`,
				thumb: 'url(https://i3.ytimg.com/vi/%key/1.jpg)',
				selector: 'a[href*="youtube."], a[href*="youtu.be/"]',
				favicon: 'https://www.youtube.com/favicon.ico',
				key: 'id',
				reg: '(youtu)',
				matcher: /(?:(?:(?:v=|#p\/u\/\d*?\/)|(?:v=|#p\/c\/[a-zA-Z0-9]+\/\d*?\/)|(?:embed\/)|(?:v\/)|(?:\.be\/))([A-Za-z0-9-_]{11}))/i,
				https: true
			},
			vimeo: {
				title: 'vimeo too!',
				home: 'https://vimeo.com/',
				embed: 'https://player.vimeo.com/video/%key?badge=0',
				ajax: 'https://vimeo.com/api/v2/video/%key.json',
				selector: 'a[href*="vimeo.com/"]',
				favicon: 'https://f.vimeocdn.com/images_v6/favicon.ico',
				key: 'id',
				reg: '(vimeo)',
				matcher: /(?:vimeo\.com\/(\d+))/i,
				https: true
			},
			vine: {
				title: 'vine me!',
				home: 'https://vine.co/',
				embed: 'https://vine.co/v/%key/embed/simple?audio=1',
				ajaxExtension: true,
				ajax: 'https://vine.co/oembed.json?url=https%3A%2F%2Fvine.co%2Fv%2F%key',
				selector: 'a[href*="vine.co/"]',
				favicon: 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAABcklEQVQ4jX2SvyvFYRTGP+9NdzAYbpIkysA/ICnKoEwvA4vlUkoyGQzvoOiK4R0YbDLhDhbi9maWIjEqcScWkyRhMNzXcM/3On39OMt5nvOe85wfvSbGyH9mgs8Cq8AUkAMegU1gJVpXqfu3umpbwITiLUBB8HJGuryb4KMJ/izVvUMVnwNjwJPwGYCMkLL4zlT3XoUL0boD4FBNUhO4Fd9ogm9SRVmFn8Una79qgWuV2K3wvcLJdG3iL7TaiUocAI4Fn0vnHDBrgt+L1g2q3NoEl8CH4KHkMVr3CSwI7QeOTPA9WsAk/8AEvw+MSrwrWleuJQVfABZJWbTOZBTfVXg6lbgEHKjQTcL1BFngAWimeuH2aN2LvHUDV1JcjNbl0zdI9l0T2pDsboKvB7Yl/ga4X2+gku+AVqAC9AFzwLik5KN1xT8FRGQEOBL6yfdn2onWTZKyTDoQrSsB60KT4lNSh/1TQETmgQ1ZowQMy41+2BeLRXeRaKuHSAAAAABJRU5ErkJggg==',
				key: 'id',
				reg: '(vine)',
				matcher: /(?:vine\.co\/v\/([A-Za-z0-9-_]{11}))/i
			},
			soundcloud: {
				title: 'sound off!',
				home: 'https://soundcloud.com/',
				embed: 'https://w.soundcloud.com/player/?show_comments=false&url=%key',
				ajax: 'https://soundcloud.com/oembed?format=json&url=%uri',
				favicon: 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAwAAAAMCAYAAABWdVznAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAXZJREFUeNp0UjFOw0AQnD3bCYaEoIgiUIQKikiAKKGgSAkFL0BQ0FEgUfECShpewB+QEE+gokDiBxGRiCCEBBzHd8ucLYGQ4Kw7+9azc7OzJ6oKnFYUxgD+W/DHYNA54GIooifTitYqg1okaPFff6BcTLF5fECYb5IJNJ3kMfUP2dxHPycwcQ0SliARocSGsA46YkKWQW0GS6BDAGlu5Myu84AgSmHKM/DY0NegwwR2+AJrSjBre5BaA1H7GFKpI729QHp9jqiaeTBn5mB7fWirjWB9B9JYBrwEMnop5d0zfD53YO9vYMY8SQcpZGkTZvsIstiivDfoaxfIxt8exYeXCLYOSOwlJZQ6twSpL0DfX4EoZtkCqc7/Njalbxnfdt+oSwhurkBnCWI/1FlgqgKNZ3JnZDSA3FFSpYdQAoF56tKNDmhOYTzJ3OSHXZkflJhYEyawcnH02HpmxZ+DrRJlgmacvrsHRmHln2tRnIiAy5WTLwEGAK4QoBQmtGHkAAAAAElFTkSuQmCC',
				selector: 'a[href*="soundcloud.com/"]',
				key: 'uri',
				reg: '(soundcloud)',
				matcher: /(?:\/\/(?:\bwww|m\.\b)?soundcloud\.com\/(.+?\/.+))/i,
				https: true,
				scroll: true
			},
			gfycat: {
				title: 'gfycat meow!',
				home: 'https://gfycat.com/',
				embed: 'https://gfycat.com/iframe/%key',
				ajaxExtension: true,
				rawResponse: true,
				ajax: 'https://gfycat.com/%key',
				thumb: 'url(https://thumbs.gfycat.com/%key-poster.jpg)',
				selector: 'a[href*="gfycat.com/"]',
				favicon: 'https://gfycat.com/favicon.ico',
				key: 'id',
				reg: '(gfycat)',
				matcher: /(?:gfycat\.com\/(?:(\b(?:[A-Z][a-z]*){3,}\b)))/i,
				https: true,
				scroll: true
			},
			streamable: {
				title: 'streamable!',
				home: 'https://streamable.com/',
				embed: 'https://streamable.com/e/%key',
				ajax: 'https://api.streamable.com/oembed.json?url=%uri',
				thumb: 'url(https://cdn.streamable.com/image/%key.jpg)',
				selector: 'a[href*="streamable.com/"]',
				favicon: 'https://streamable.com/favicon.ico',
				key: 'id',
				reg: '(streamable\\.com)',
				matcher: /(?:streamable\.com\/([A-Za-z0-9-_]+))/i,
				https: true
			},
			imgur: {
				title: 'imgur it!',
				home: 'https://i.imgur.com/',
				embed: 'https://i.imgur.com/%key',
				thumb: 'url(https://i.imgur.com/%keyh.jpg)',
				selector: 'a[href*=".gifv"]',
				favicon: 'https://imgur.com/favicon.ico',
				reg: '(\\.gifv$)|(imgur)',
				matcher: /(?:imgur\.com\/(\w+)\.(?:gifv|mp4|webm))/i,
				https: true,
				scroll: true,
				videoTag: true
			},
			html5: {
				home: true,
				title: 'html5 go!',
				selector: 'a[href*=".webm"], a[href*=".mp4"]',
				favicon: 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAA0AAAANCAYAAABy6+R8AAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAdlJREFUeNpcks9rE0EUx7+b3TRktxvbhUhrNm6SrUWtxxbxJ9iLl0JFKWg8WLD05EWhIuRieogXwYPX4tGb/gPSq/oP6KH+uG2MFaRJJXaTbp7fWbsl+uDDzLx533lv3ozmeyUM2VlSJxeISULygTwhr5Og1JDgOXlPrpLRMAxTIpLlfJa8Ipv/i56Re4lT07SYKIqQSh2eO0/eqYnujI35HF8O19jr9XBneRmt1je0d3ag63qy5ZIAvNObE+WKTJXKktENoVPsrCntdlsWFxbiNXNJyS3KdMUXxv8w6LzY7XZhmiYer9fBYOTzR5HL5XCzWsWxYhFnTs/gxcYGgiCAZVmOyhR5BVcm8nlZvbsi29+3ZdgGg4Gs1+viua4UJiZVpkiJeqo0VaIqxbGPSKfTORRduXQ59peLx5PyItWaj9yDQtmt21XYto1GoxGvbywtxaNhGHE3ab+V6L6aKYczPo7ZuTk8WnuIWq2G64vX4JU8zJw8BXXv5D21gx+xydrn7VEb/f0+tj5twa/4+Pz1C6Y4Oo6DZrOJdDrdYuxkIlKP+ZbZzoV7e8iyk8kD/9rdRXpkBJlMpsmDC/98I97pPF//AQUBl/2/jRvsm5b1kxmeJgJlfwQYAKZQxgzeI6/EAAAAAElFTkSuQmCC',
				reg: '(\\.webm$)|(\\.mp4$)',
				slim: true,
				scroll: true,
				videoTag: true
			},
			'html5-audio': {
				home: true,
				title: 'hey, listen!',
				selector: 'a[href*=".mp3"]',
				reg: '(\\.mp3$)',
				slim: true,
				scroll: true
			}
		}
	};

	class Control extends Container {
		/** U I CLASS
		 * Class for the player controls
		 * This is the control bar once the user clicks on the thumbnail
		 * Contains its own instance of a Player
		 * Acts like a decorator on the YTMA and Container intances by adding events
		 * @param {Y} ytma A YTMA instance
		 */
		constructor(props) {
			super(props);

			this.open = false;
			this.selected = { size: null, ratio: null };
		}

		getControl() {
			if (!this.projector) {
				this.createProjector();
			}
			return this;
		}

		createProjector() {
			super.createProjector();
			this.projector.addEventListener('click', Control.events.videoBar.bind(this), false);
			this.play = new Player(this);

			this.markSelected(`li[data-type="size"][data-value="${this.play.attrs.size}"]`, 'size');
			this.markSelected(`li[data-type="ratio"][data-value="${this.play.attrs.ratio}"]`, 'ratio');
		}

		resetViewSize() {
			this.play.dimmensions();
			this.setControlBarSize(this.play.attrs.size);
		}

		showOnScroll(link) {
			if (!this.open && this.canShowUnder(link)) {
				this.showPlayer();
			}
		}

		showPlayer() {
			this.open = true;

			super.showPlayer();
			this.attachPlayerPanel();
			this.play.switchOn();

			if (Y.user.preferences.focus) {
				document.location.hash = `#${this.body.id}`;
			}
		}

		hidePlayer() {
			this.open = false;

			this.play.switchOff();
			super.hidePlayer();
		}

		attachPlayerPanel() {
			if (!this.play.panel.parentNode) {
				// console.log('attaching display panel');
				this.projector.appendChild(this.play.panel);
			}
		}

		hideAllPlayers() {
			const group = Y.collect(this.state.id);
			console.info('ytma//hide+all(id)', this.state.id, group.length);
			group.forEach(g => {
				g.disableOpenOnScroll();
				g.getControl().hidePlayer();
			});
		}

		setControlBarSize(size) {
			this.markSelected(`li[data-type="size"][data-value="${size}"]`, 'size');
		}

		markSelected(el, type) {
			if (typeof el === 'string') {
				el = this.projector.querySelector(el);
			}
			el.id = type + this.state.uid;
			try {
				this.selected[type].removeAttribute('id');
			} catch (e) { }
			this.selected[type] = el;
		}
	}

	Control.ratios = {
		SD: 1,
		HD: 2,
		PORTRAIT: 3
	};

	Control.sizes = {
		HIDDEN: 0,
		S: 240,
		M: 360,
		L: 480,
		X: 720
	};

	/** Trigger is the VAR element
	 * @param {HTMLElement} t VAR element
	 */
	Control.createFromTrigger = t => {
		// console.info('ytma//trigger');
		if (t && t.dataset.ytmuid && !Y.set[t.dataset.ytmuid]) {
			console.info('ytma//trigger+new');
			Y.addToSet(new Control().reactivate(t));
		}
		console.info('ytma//trigger+control');
		return Y.set[t.dataset.ytmuid].getControl();
	};

	Control.events = {
		$fire: {
			settings: function () {
				Y.user.events.formToggle();
			},
			close: function () {
				if (this.site.scroll) {
					// console.log('events.close-1');
					this.hideAllPlayers();
				} else {
					// console.log('events.close-2');
					this.disableOpenOnScroll();
					this.hidePlayer();
				}
			},
			ratio: function (li) {
				const ratio = parseInt(li.dataset.value, 10);
				this.play.dimmensions({ ratio });
				this.markSelected(li, 'ratio');
			},
			size: function (li) {
				const size = parseInt(li.dataset.value, 10);
				this.play.dimmensions({ size });
				this.markSelected(li, 'size');
			}
		},
		videoBar: function ({ target }) {
			if (target.tagName.toLowerCase() === 'li' && target.dataset && target.dataset.type) {
				const t = target.dataset.type;
				if (Control.events.$fire[t]) {
					Control.events.$fire[t].call(this, target);
				}
			}
		}
	};

	/** P L A Y E R CLASS
	 *  @param {Control} parent Instance
	 */
	class Player {

		constructor(parent) {
			this.parent = parent;

			this.mode = 'off';

			this.attrs = {
				sources: null,
				quality: this.quality,
				size: null,
				ratio: null,
				start: this.time(),
				type: null
			};

			this.attrs.sources = this.sources;

			// todo improve type/media
			this.attrs.type = this.findType();
			this.media = Player.makeMedia[this.attrs.type](this);

			this.channel = _modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('div', { className: 'ytm_panel_channel ytm_block' }, this.media, true);
			this.switcher = _modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('div', { className: `ytm_panel_switcher ytm_panel_size ytm_block ytm_${this.attrs.type}`, _ytmuid: this.parent.state.uid, _standby: true });
			this.panel = _modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('div', { className: 'ytm_panel ytm_block' }, this.switcher, true);

			if (parent.state.site === 'soundcloud' && Y.reg.extra.soundcloud.playlist.test(parent.anchor.href)) {
				this.media.classList.add('ytm_soundcloud-playlist');
				this.switcher.classList.add('ytm_soundcloud-playlist');
			}

			this.dimmensions(Y.user.preferences);
		}

		get sources() {
			try {
				return (Player.sources[this.parent.state.site] || Player.sources.iframe)(this.parent.state, this.attrs);
			} catch (e) {
				console.error(e);
			}
		}

		get quality() {
			return Player.qualities[Y.user.preferences.quality] || Player.qualities[360];
		}

		get className() {
			return `ytm_panel ytm_block ytm_panel-${Player.dimms.ratios[this.attrs.ratio]} ytm_panel-${Player.dimms.sizes[this.attrs.size]}`;
		}

		dimmensions({ ratio, size }) {
			this.attrs.ratio = Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["isNumber"])(ratio) ? ratio : this.attrs.ratio;
			this.attrs.size = Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["isNumber"])(size) ? size : this.attrs.size;
			this.panel.className = this.className;
		}

		time() {
			try {
				// debugger;
				const m = this.parent.state.uri.match(Y.reg.time).slice(1, 4);
				return ((+m[0] || 0) * 60 * 60) + ((+m[1] || 0) * 60) + (+m[2] || 0);
			} catch (e) { return 0; }
		}

		findType() {
			if (this.parent.state.site === 'html5-audio') { return 'audio'; }
			if (this.parent.site.videoTag) { return 'video'; }
			return 'iframe';
		}

		switchOff() {
			// console.log('removed media');

			if (this.media.pause) {
				// console.log('pausing');
				this.media.pause();
			}

			try {
				this.switcher.removeChild(this.channel);
			} catch (e) {
				// console.error(e);
			}
			this.mode = 'off';
		}

		switchOn() {
			if (this.attrs.size === 0) {
				this.attrs.size = Y.user.preferences.size;
				this.parent.resetViewSize();
			}
			// console.log('switch to media');
			this.switcher.appendChild(this.channel);
			this.switcher.dataset.standby = false;
			this.mode = 'on';
		}

		switchStandby() {
			// console.log('switch to standby');
			this.switchOff();
			this.switcher.dataset.standby = true;
			this.mode = 'standby';
		}

		isStandby() {
			return this.mode === 'standby';
		}
	}

	Player.sources = {
		iframe: function (data) {
			const key = Y.DB.sites[data.site].key;

			return [
				{ type: 'text/html', src: Y.DB.sites[data.site].embed.replace('%key', data[key]) }
			];
		},
		'html5-audio': function ({ uri }) {
			return [
				{ type: 'audio/mp3', src: uri }
			];
		},
		html5: function ({ uri }) {
			// attaching the type as either mp4 or webm

			if (/(?:webm)/.test(uri)) {
				return [
					{ type: 'video/webm', src: uri }
				];
			}

			return [
				{ type: 'video/mp4', src: uri },
				{ type: 'video/webm', src: uri },
				{ type: 'video/ogg; codecs="theora, vorbis"', src: uri }
			];
		},
		imgur: function ({ id }) {
			const src = Y.DB.sites.imgur.embed.replace('%key', id);

			return [
				{ type: 'video/webm', src: `${src}.webm` },
				{ type: 'video/mp4', src: `${src}.mp4` }
			];
		},
		youtube: function ({ id }, { quality, start }) {
			const params = `?html5=1&version=3&modestbranding=1&rel=0&showinfo=1&vq=${quality}&iv_load_policy=${Y.user.preferences.yt_annotation}&start=${start}&rel=0`;

			return [
				{ type: 'text/html', src: Y.DB.sites.youtube.embed.replace('%key', id) + params }
			];
		}
	};

	Player.dimms = {
		ratios: {
			1: 'sd',
			2: 'hd',
			3: 'pr'
		},
		sizes: {
			0: 'h',
			240: 's',
			360: 'm',
			480: 'l',
			720: 'xl'
		},
		aspects: {
			1: 4 / 3,
			2: 16 / 9,
			3: 16 / 9
		}
	};

	Player.qualities = {
		240: 'small',
		360: 'medium',
		480: 'large',
		720: 'hd720',
		1080: 'hd1080',
		1081: 'highres'
	};

	Player.css = {
		item: function (key, value) {
			if (Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["isNumber"])(value)) {
				value += 'px';
			}

			return `\t${key}: ${value};\n`;
		},
		iter: function (css, cssEntries) {
			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].o(cssEntries, (key, value) => {
				css.push(Player.css.item(key, value));
			});
			css.push('}');
		},
		generator: function () {
			const css = [];

			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].o(this.sizes, (size, sizes) => {
				_modules___WEBPACK_IMPORTED_MODULE_1__["default"].o(sizes, (dimm, keys) => {
					css.push(`\n.ytm_panel-${size}.ytm_panel-${dimm} .ytm_panel_size {\n`);
					Player.css.iter(css, keys);
				});
			});

			// add site overrides
			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].o(this.sites, (site, data) => {
				_modules___WEBPACK_IMPORTED_MODULE_1__["default"].o(data, (setting, keys) => {
					if (setting === 'all') {
						css.push(`\n.ytm_site_${site} .ytm_panel_size {\n`);
					} else {
						css.push(`\n.ytm_site_${site} .ytm_panel-${setting} .ytm_panel_size {\n`);
					}
					Player.css.iter(css, keys);
				});
			});

			return css.join('');
		},
		sizes: (() => {
			const merge = {};

			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].o(Player.dimms.sizes, (num, size) => {
				if (num >= 0) {
					merge[size] = {};

					_modules___WEBPACK_IMPORTED_MODULE_1__["default"].o(Player.dimms.ratios, (k, ratio) => {
						if (ratio === 'pr') {
							const w = Math.floor(num * 0.95); // smaller than the normal sizes
							merge[size][ratio] = {
								width: w,
								height: Math.floor(w * Player.dimms.aspects[k])
							};
						} else {
							merge[size][ratio] = {
								width: Math.floor(num * Player.dimms.aspects[k]),
								height: num
							};
						}
					});
				}
			});

			return merge;
		})(),
		sites: { // custom sizes per site
			soundcloud: {
				all: {
					height: '118px !important'
				}
			},
			vine: {
				s: {
					width: 240,
					height: 240
				},
				m: {
					width: 360,
					height: 360
				},
				l: {
					width: 480,
					height: 480
				},
				xl: {
					width: 720,
					height: 720
				}
			}
		}
	};

	Player.makeMedia = {
		$css: function (type) {
			return `ytm_panel_media ytm_panel_size ytm_block ytm_${type}`;
		},
		video: function ({ attrs }) {
			const video = _modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('video', {
				controls: true,
				autoplay: false,
				loop: true,
				className: this.$css('video'),
				$allowscriptaccess: true,
				preload: 'metadata'
			});

			const links = [];

			attrs.sources.forEach(({ src, type }) => {
				_modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('source', { src, $type: type }, video);

				links.push(`<a href="${src}">${src}</a>`);
			});

			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('p', { innerHTML: `Could not load source(s): ${links.join('<br />')}` }, video);

			return video;
		},
		iframe: function ({ attrs }) {
			return _modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('iframe', {
				$allowfullscreen: true,
				$referrerpolicy: 'no-referrer',
				// $sandbox: 'allow-same-origin allow-scripts allow-popups',
				$type: attrs.sources[0].type,
				src: attrs.sources[0].src,
				className: this.$css('iframe')
			});
		},
		audio: function ({ attrs }) {
			return _modules___WEBPACK_IMPORTED_MODULE_1__["default"].e('audio', {
				src: attrs.sources[0].src,
				$type: attrs.sources[0].type
			});
		}
	};

	/** S C R O L L CLASS
	 * Window-Scroll Event Helper
	 */
	class Scroll {
		constructor(selector, cb, delay = 500) {
			this.selector = selector;
			this.cb = cb;
			this.monitor = this.monitor.bind(this);

			// console.log('YTMA.Scroll Monitor: ', selector);
			this.bound = Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["debounce"])(this.monitor, delay);

			this.bound();
			window.addEventListener('scroll', this.bound, false);
		}

		stop() {
			// console.log('clear scroll: ', this.selector);
			window.removeEventListener('scroll', this.bound);
		}

		monitor() {
			_modules___WEBPACK_IMPORTED_MODULE_1__["default"].s(this.selector, this.cb);
		}
	}

	Scroll.visible = el => {
		const bound = el.getBoundingClientRect();
		return (bound.top >= 0 && bound.top <= document.documentElement.clientHeight);
	};

	Scroll.visibleAll = (el, offset) => {
		const bound = el.getBoundingClientRect();
		const height = document.documentElement.clientHeight;
		offset = Object(_modules_Helpers__WEBPACK_IMPORTED_MODULE_0__["isNumber"])(offset) ? +offset : 0;
		return ((bound.bottom + offset >= 0)
			&& (bound.top <= height + offset || bound.bottom <= height - offset));
	};

	/** Returns 1, 0, -1 when el1 is above, exactly the same, or below el2 */
	Scroll.compare = (el1, el2) => {
		const a = el1.getBoundingClientRect().y;
		const b = el2.getBoundingClientRect().y;

		if (a < b) { return 1; }
		if (a === b) { return 0; }
		return -1;
	};


	Y.main();
})();


/***/ }),

/***/ "./src/modules/Helpers.js":
/*!********************************!*\
  !*** ./src/modules/Helpers.js ***!
  \********************************/
/*! exports provided: isNumber, removeSearch, on, debounce */
/***/ (function(module, __webpack_exports__, __webpack_require__) {

"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "isNumber", function() { return isNumber; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "removeSearch", function() { return removeSearch; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "on", function() { return on; });
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "debounce", function() { return debounce; });
if (!Element.prototype.matches)
	Element.prototype.matches = Element.prototype.msMatchesSelector || Element.prototype.webkitMatchesSelector;

if (!Element.prototype.closest) {
	Element.prototype.closest = function (s) {
		var el = this;
		if (!document.documentElement.contains(el)) return null;
		do {
			if (el.matches(s)) return el;
			el = el.parentElement || el.parentNode;
		} while (el !== null && el.nodeType === 1);
		return null;
	};
}

const isNumber = n => !isNaN(parseFloat(n)) && isFinite(n);

const removeSearch = (uri, keepHash) => {
	// removes search query from a uri
	const s = uri.indexOf('?');
	const h = uri.indexOf('#');
	let hash = '';
	if (s > -1) {
		if (keepHash && h > -1) {
			hash = uri.substr(h);
		}
		uri = uri.substr(0, s) + hash;
	}
	return uri;
};

/**
 * @param {HTMLElement} element HTML element
 * @param {string} events Space- or coma-separated string of one or more types, eg "click dblclick"
 * @param {string} selector CSS selector for the elements to trigger the event on
 * @param {Function} listener A callback
 * @param {Boolean} cancel Cancel
 */
const on = (element, events, selector, listener, cancel = true) => {
	events = events.split(/(?:\s+|,)/).filter(f => f);

	if (events.length === 0) return;

	const fn = event => {
		const found = event.target.closest(selector);
		if (found) listener.call(found, event);
	};

	events.forEach(type => element.addEventListener(type, fn, cancel));
};

const debounce = (fn, delay = 250) => {
	let timeout;

	return function (...args) {
		const timed = () => {
			timeout = null;
			fn.apply(this, args);
		};

		window.clearTimeout(timeout);
		timeout = window.setTimeout(timed, delay);
	};
};

/***/ }),

/***/ "./src/modules/_.js":
/*!**************************!*\
  !*** ./src/modules/_.js ***!
  \**************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {

"use strict";
__webpack_require__.r(__webpack_exports__);
// H E L P E R Handle
const _ = {
	s: (selector, cb) => {
		const elements = _.qsa(selector);
		elements.some((element, index) => cb(element, index, elements) === false);
	},
	o: function (object, cb) {
		Object.keys(object).some(key => cb(key, object[key], object) === false);
	},
	e: function (t, o, e, p) {
		const c = document.createElement(t);
		_.o(o, (k, v) => {
			const b = k.charAt(0);
			if (b === '_')
				c.dataset[k.substring(1)] = v;
			else if (b === '$')
				c.setAttribute(k.substring(1), v);
			else
				c[k] = v;
		});

		if (e && p) {
			c.appendChild(e);
		} else if (e) {
			e.appendChild(c);
		}
		return c;
	},
	qsa: selector => Array.from(document.querySelectorAll(selector)),
	css: function (text) {
		if (!this.style) {
			this.style = document.createElement('style');
			this.style.type = 'text/css';
			document.body.appendChild(this.style);
		}
		this.style.appendChild(document.createTextNode(`${text}\n`));
	},
	js: t => {
		const j = document.createElement('script');
		j.type = 'text/javascript';
		j[/^https?:\/\//i.test(t) ? 'src' : 'textContent'] = t;
		document.head.appendChild(j);
	}
};

/* harmony default export */ __webpack_exports__["default"] = (_);

/***/ }),

/***/ "./src/modules/strg.js":
/*!*****************************!*\
  !*** ./src/modules/strg.js ***!
  \*****************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {

"use strict";
__webpack_require__.r(__webpack_exports__);
// S T O R A G E HANDLE
const strg = {
	MAX: 5012,
	on: false,
	test: () => {
		try {
			const l = localStorage;
			const c = Math.random().toString(16).substr(2, 8);
			l.setItem(c, c);
			return l.getItem(c) === c ? !l.removeItem(c) : false;
		} catch (e) { return false; }
	},
	read: (key) => {
		try {
			return JSON.parse(localStorage.getItem(key));
		} catch (e) {
			return console.error(`${e.lineNumber}:${e.message}`);
		}
	},
	save: (key, val) => strg.on ? !localStorage.setItem(key, JSON.stringify(val)) : false,
	wipe: key => strg.on ? !localStorage.removeItem(key) : false,
	zero: o => { for (let k in o) { if (o.hasOwnProperty(k)) { return false; } } return true; }, // check if the object is empty
	grab: (key, def) => { const s = strg.read(key); return strg.zero(s) ? def : s; },
	size: () => {
		let length = 0;
		let key;
		try {
			for (key in window.localStorage) {
				if (window.localStorage.hasOwnProperty(key)) {
					length += window.localStorage[key].length;
				}
			}
		} catch (e) {
			// no more space
		}
		return 3 + ((length * 16) / (8 * 1024));
	},
	full: () => {
		try {
			const date = +(new Date());
			localStorage.setItem(date, date);
			localStorage.removeItem(date);
			return false;
		} catch (e) {
			if (e.name === 'QuotaExceededError' ||
					e.name === 'NS_ERROR_DOM_QUOTA_REACHED') {
				return true;
			}
		}
	},
	init: () => { strg.on = strg.test(); }
};
strg.init();

/* harmony default export */ __webpack_exports__["default"] = (strg);

/***/ }),

/***/ "./src/modules/update.js":
/*!*******************************!*\
  !*** ./src/modules/update.js ***!
  \*******************************/
/*! exports provided: default */
/***/ (function(module, __webpack_exports__, __webpack_require__) {

"use strict";
__webpack_require__.r(__webpack_exports__);
/* harmony import */ var _strg__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./strg */ "./src/modules/strg.js");
/* harmony import */ var ___WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./_ */ "./src/modules/_.js");



// U P D A T E HANDLE
const update = {
	name: 'ytma!',
	version: 8010,
	key: 'ujs_YTMA_UPDT_HR',
	callback: 'ytmaupdater',
	page: 'https://greasyfork.org/scripts/1023-youtube-me-again',
	json: 'https://hateradio.github.io/ytma/update.json',
	interval: 7,
	day: (new Date()).getTime(),
	show: false,
	time: () => new Date(update.day + (1000 * 60 * 60 * 24 * update.interval)).getTime(),
	notification: ({ version, page }) => {
		if (update.version < version) {
			_strg__WEBPACK_IMPORTED_MODULE_0__["default"].save(update.key, { date: update.time(), version, page });
			update.link();
		}
	},
	link: () => {
		if (update.show) { return; }
		update.show = true;

		const { page } = _strg__WEBPACK_IMPORTED_MODULE_0__["default"].read(update.key);
		const link = `
				<a href="${page || update.page}" id=updatev3 title=Update target=_blank>
					An update for ${update.name} is available.
				</a>`;

		___WEBPACK_IMPORTED_MODULE_1__["default"].css(update.css);
		document.body.insertAdjacentHTML('beforeend', link);
		___WEBPACK_IMPORTED_MODULE_1__["default"].on(document.body, 'click', '#updatev3', e => e.target.style.display = 'none');
	},
	check: (opt) => {
		if (!_strg__WEBPACK_IMPORTED_MODULE_0__["default"].on) { return; }
		if (window.chrome && window.chrome.extension) { return; }
		const stored = _strg__WEBPACK_IMPORTED_MODULE_0__["default"].read(update.key);
		let page;

		if (opt || !stored || stored.date < update.day) {
			page = (stored && stored.page) || update.page;
			_strg__WEBPACK_IMPORTED_MODULE_0__["default"].save(update.key, { date: update.time(), version: update.version, page });
			fetch(update.json).then(res => res.json()).then(update.notification);
		} else if (update.version < stored.version) {
			update.link();
		}
	},
	css: '#updater3,#updater3:visited{box-shadow:1px 1px 6px #7776;border-bottom:3px solid #e39c2d;cursor:pointer;color:#555;font-family:sans-serif;font-size:12px;font-weight:700;text-align:justify;position:fixed;z-index:999999;right:10px;top:10px;background:#ebebeb url(data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgLTM0IDUxMiA1MTIiPjxwYXRoIGZpbGw9IiNlNWViZjUiIGQ9Ik0yNzAuNzgxIDcuNUgyNDEuMjJMOC42NiA0MTAuMzA1bDE0Ljc4MSAyNS42MDFINDg4LjU2bDE0Ljc4LTI1LjYwMXptMCAwIi8+PHBhdGggZmlsbD0iI2Y4ZTg2OCIgZD0iTTQ4LjcwNyA0MDAuOTM0TDI1My4xMjkgNDYuODY3aDUuNzQybDIwNC40MjIgMzU0LjA2N2MtMS4yNzcgMi4yMS0xLjU5NCAyLjc2NS0yLjg3MSA0Ljk3Nkg1MS41NzhjLTEuMjc3LTIuMjEtMS41OTQtMi43NjUtMi44NzEtNC45NzZ6bTAgMCIvPjxwYXRoIGZpbGw9IiNlMzljMmQiIGQ9Ik0yNzUuOTk2IDc2LjUyN2wtMTcuMTI1LTI5LjY2aC01Ljc0Mkw0OC43MDcgNDAwLjkzNGMxLjI3NyAyLjIxIDEuNTk0IDIuNzY1IDIuODcxIDQuOTc2aDM5Ljk5NmMtMS4yNzctMi4yMS0xLjU5Ny0yLjc2NS0yLjg3LTQuOTc2em0wIDAiLz48cGF0aCBmaWxsPSIjY2FkOGVhIiBkPSJNMjc1Ljk5NiAxNi41MzVMMjcwLjc4MSA3LjVIMjQxLjIyTDguNjYgNDEwLjMwNWM2LjU2NyAxMS4zNzkgOC4yMTEgMTQuMjIyIDE0Ljc4MSAyNS42MDFoMzkuOTk2Yy02LjU3LTExLjM3OS04LjIxNC0xNC4yMjItMTQuNzgtMjUuNjAxem0wIDAiLz48cGF0aCBmaWxsPSIjNzI4NjllIiBkPSJNMjcwLjg1NSAzMDIuNDU3aC0yOS43MWMtMy4wMDQtMy4wMDQtNC42OTItNC42ODctNy42OTYtNy42OTFWMTYzLjg2M2g0NS4xMDJ2MTMwLjkwM2MtMy4wMDggMy4wMDQtNC42OTIgNC42ODctNy42OTYgNy42OTF6bTAgMCIvPjxwYXRoIGZpbGw9IiM1MzYyNzUiIGQ9Ik0yNTMuNDUgMTYzLjg2M2gtMjB2MTMwLjkwM2MzLjAwMyAzLjAwNCA0LjY5IDQuNjg3IDcuNjk1IDcuNjkxaDE5Ljk5NmwtNy42OTItNy42OTF6bTAgMCIvPjxwYXRoIGZpbGw9IiM3Mjg2OWUiIGQ9Ik0yMzMuNDUgMzMwLjgxM2g0NS4xdjQ1LjEwMWgtNDUuMXptMCAwIi8+PHBhdGggZmlsbD0iIzUzNjI3NSIgZD0iTTIzMy40NSAzMzAuODEzaDIwdjQ1LjEwMWgtMjB6bTAgMCIvPjxwYXRoIGZpbGw9IiNmZmYiIGQ9Ik0yNzUuOTk2IDE0MS4zNjdIMjYxdi0xNWgxNC45OTZ6bS0yNC45OTYgMGgtMTV2LTE1aDE1em0wIDAiLz48cGF0aCBkPSJNNDAwLjQzIDIxNy4wNTlsLTEyLjk4OSA3LjVMNDk0LjY4IDQxMC4zMDVsLTEwLjQ1IDE4LjEwMUgyNy43N2wtMTAuNDU0LTE4LjEwMUwyNDUuNTQ2IDE1aDIwLjkwN2wxMDcuMjM4IDE4NS43NDYgMTIuOTg5LTcuNUwyNzUuMTEzIDBoLTM4LjIyMkwwIDQxMC4zMDVsMTkuMTEgMzMuMTAxaDQ3My43NzdMNTEyIDQxMC4zMDV6bTAgMCIvPjxwYXRoIGQ9Ik0xMjUuMTU2IDQxMy40MDZINDY0Ljc1bDcuMjAzLTEyLjQ3Mi0yMDguNzUtMzYxLjU2N2gtMTQuNDA2TDQwLjA0NyA0MDAuOTM0bDcuMiAxMi40NzJoNTcuOTEzVjM5OC40MUg1OC44MjRMMjU2IDU2Ljg5bDE5Ny4xNzYgMzQxLjUyaC0zMjguMDJ6bTAgMCIvPjxwYXRoIGQ9Ik0yODYuMDQ3IDIyOS41NTl2LTczLjE5NmgtNjAuMDk4djE0MS41MDhsMTIuMDkgMTIuMDg2aDM1LjkyMmwxMi4wODYtMTIuMDg2VjI0OS41NkgyNzEuMDV2NDIuMDk3bC0zLjMwMSAzLjMwNWgtMjMuNWwtMy4zLTMuMzA1VjE3MS4zNjNoMzAuMXY1OC4xOTZ6bTAgME0yMjUuOTUgMzgzLjQxaDYwLjA5N3YtNjAuMDk4aC02MC4wOTh6bTE1LTQ1LjA5OGgzMC4xdjMwLjEwMmgtMzAuMXptMCAwIi8+PC9zdmc+) no-repeat 10px center;background-size:40px;padding:0 20px 0 60px;height:55px;line-height:55px}#updater3:hover,#updater3:visited:hover{color:#b33a3a !important;border-color:#ce4b30;text-decoration: none;}' // Icon made by Freepik from www.flaticon.com 
};
update.check();

/* harmony default export */ __webpack_exports__["default"] = (update);

/***/ })

/******/ });
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///./src/index.js","webpack:///./src/modules/Helpers.js","webpack:///./src/modules/_.js","webpack:///./src/modules/strg.js","webpack:///./src/modules/update.js"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;QAAA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;;QAEA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;;;QAGA;QACA;;QAEA;QACA;;QAEA;QACA;QACA;QACA,0CAA0C,gCAAgC;QAC1E;QACA;;QAEA;QACA;QACA;QACA,wDAAwD,kBAAkB;QAC1E;QACA,iDAAiD,cAAc;QAC/D;;QAEA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA,yCAAyC,iCAAiC;QAC1E,gHAAgH,mBAAmB,EAAE;QACrI;QACA;;QAEA;QACA;QACA;QACA,2BAA2B,0BAA0B,EAAE;QACvD,iCAAiC,eAAe;QAChD;QACA;QACA;;QAEA;QACA,sDAAsD,+DAA+D;;QAErH;QACA;;;QAGA;QACA;;;;;;;;;;;;;AClFA;AAAA;AAAA;AAAA;AAAA;AAAyE;AAC7C;AACM;AACI;;AAEtC;;AAEA;AACA;AACA;AACA;AACA;AACA,YAAY,OAAO;AACnB,YAAY,cAAc;AAC1B,YAAY,OAAO;AACnB,YAAY,kBAAkB;AAC9B;AACA;;AAEA,eAAe,mBAAmB;AAClC,6BAA6B,GAAG,GAAG,WAAW;;AAE9C;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,6CAA6C,iCAAiC;;AAE9E;AACA;;AAEA;AACA;AACA,aAAa,YAAY;AACzB;AACA,cAAc,UAAU;AACxB;AACA,2DAA2D,eAAe;;AAE1E;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,sDAAsD,gBAAgB;AACtE;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA,UAAU,QAAQ;AAClB;AACA,UAAU,aAAa;;AAEvB;;AAEA,eAAe,iDAAC;AAChB,YAAY,UAAU;AACtB,gDAAgD,WAAW;AAC3D;AACA,IAAI;;AAEJ;;AAEA,cAAc,uBAAuB;AACrC,cAAc,0CAA0C;;AAExD;;AAEA;AACA;AACA,IAAI;AACJ;AACA;AACA;;AAEA;AACA,UAAU,gBAAgB;AAC1B,gBAAgB,yCAAyC;AACzD,eAAe,gEAAgE;;AAE/E;AACA;;AAEA;AACA,UAAU,oBAAoB;AAC9B,UAAU,qBAAqB;;AAE/B,2CAA2C,0BAA0B;;AAErE;AACA;AACA,cAAc,MAAM;AACpB,mBAAmB,GAAG;AACtB,qBAAqB,GAAG;AACxB,cAAc,GAAG;AACjB,yDAAyD,MAAM;AAC/D;AACA,qBAAqB,GAAG;AACxB,sBAAsB,IAAI;AAC1B,sBAAsB,IAAI;AAC1B,uBAAuB,KAAK;AAC5B;AACA;AACA;AACA;;AAEA;AACA,UAAU,qBAAqB;AAC/B;AACA,oDAAoD,IAAI;AACxD;AACA,oBAAoB,GAAG;AACvB,sBAAsB,KAAK;AAC3B,qBAAqB,IAAI;AACzB;AACA;AACA;AACA;AACA;;AAEA;AACA,oBAAoB,iDAAC;AACrB;AACA;AACA,IAAI;AACJ;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ,qDAAI;AACZ;AACA;AACA;AACA;AACA;;AAEA,yBAAyB;AACzB;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA,iEAAiE,mBAAmB,GAAG,0BAA0B;AACjH;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,GAAG,2DAAE;AACL,GAAG,2DAAE;AACL,GAAG,2DAAE;;AAEL,GAAG,2DAAE;AACL,GAAG;AACH,gBAAgB,SAAS,MAAM;AAC/B;AACA;AACA,GAAG;AACH;AACA;AACA,UAAU,SAAS;AACnB;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,mFAAmF,mCAAmC;AACtH;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA,+BAA+B,wBAAwB;AACvD;AACA;;AAEA;AACA,EAAE;;AAEF;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,0BAA0B;AAC1B;AACA,IAAI;AACJ;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,IAAI;AACJ;AACA;;AAEA;AACA;AACA,eAAe,WAAW;AAC1B;AACA,uCAAuC,GAAG,KAAK,KAAK;AACpD,GAAG;AACH;AACA;AACA;;AAEA,uBAAuB,GAAG;;AAE1B;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,SAAS,qDAAI;AACb;AACA;AACA;AACA,OAAO,qDAAI,MAAM,qDAAI;AACrB,OAAO;AACP;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;;AAEA;;AAEA,4BAA4B,CAAC,uDAAM,SAAS;;AAE5C;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA,KAAK;AACL;AACA,KAAK;AACL;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,KAAK;AACL;AACA,KAAK,iDAAC,6BAA6B,0BAA0B;AAC7D,KAAK,iDAAC,2CAA2C,sBAAsB,qDAAqD,gBAAgB;AAC5I,KAAK,iDAAC,+LAA+L,qBAAqB,qBAAqB;AAC/O;AACA,KAAK;AACL;AACA;AACA;AACA,KAAK,iDAAC,iBAAiB,gBAAgB,YAAY,WAAW,QAAQ,cAAc,YAAY,eAAe;AAC/G;AACA,KAAK;AACL;AACA;;AAEA;AACA,MAAM,iDAAC;AACP;AACA,OAAO;AACP,MAAM;AACN,KAAK;AACL,0BAA0B;AAC1B;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA,cAAc,MAAM;AACpB,WAAW,MAAM;;AAEjB,2BAA2B,eAAe;AAC1C,GAAG;AACH;AACA;AACA;AACA;AACA;AACA,WAAW,qEAAY,2BAA2B,GAAG;AACrD;;AAEA;AACA;AACA;AACA;;AAEA,eAAe;AACf;AACA;AACA;AACA,cAAc,WAAW;AACzB,WAAW,WAAW;;AAEtB;AACA,GAAG;AACH;AACA;AACA;AACA;AACA,kBAAkB,sBAAsB;AACxC,mBAAmB,gBAAgB;AACnC,oBAAoB,aAAa,GAAG,OAAO;AAC3C;AACA;AACA;AACA;AACA,GAAG;AACH,wBAAwB;AACxB,GAAG,iDAAC;AACJ;AACA,qCAAqC,UAAU;AAC/C;AACA,wDAAwD,IAAI;AAC5D;AACA;AACA,IAAI;;AAEJ,GAAG,iDAAC;AACJ;AACA;AACA;AACA,2DAA2D,KAAK,IAAI,KAAK;AACzE;AACA,IAAI;AACJ,GAAG;AACH;AACA,GAAG,iDAAC,0BAA0B,UAAU;;AAExC,iBAAiB,iDAAC,gDAAgD,UAAU;AAC5E;AACA;AACA;AACA,IAAI;;AAEJ;AACA,GAAG;AACH;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf,kBAAkB;AAClB,gCAAgC,iBAAiB;AACjD,qBAAqB;AACrB,uBAAuB;AACvB;AACA;AACA;AACA;AACA;AACA;;AAEA,OAAO,qDAAI;AACX;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH,YAAY;AACZ;AACA,GAAG;AACH;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA,aAAa,qDAAI,oBAAoB;;AAErC;AACA;AACA;AACA,IAAI,IAAI;;AAER,GAAG,iDAAC;AACJ;AACA;AACA;AACA,IAAI;;AAEJ;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA,mBAAmB,yBAAyB;AAC5C,kBAAkB,wBAAwB;AAC1C,kBAAkB,wBAAwB;AAC1C,qBAAqB,2BAA2B;;AAEhD;AACA,GAAG,iDAAC;AACJ;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA,IAAI;AACJ,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;;AAEA;AACA,KAAK,IAAI;;AAET,QAAQ,qDAAI;AACZ;AACA,KAAK;AACL;AACA;;AAEA,IAAI;AACJ;AACA;AACA;AACA,IAAI,qDAAI;AACR;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;;AAEA;AACA,IAAI;AACJ;AACA,qBAAqB,QAAQ;;AAE7B;;AAEA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA,MAAM,iDAAC,uBAAuB,oBAAoB;AAClD;AACA;AACA,OAAO;AACP;AACA,KAAK;AACL,IAAI;AACJ;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL,IAAI;AACJ;AACA;AACA;AACA,IAAI;AACJ;AACA,8BAA8B,4BAA4B;;AAE1D;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,KAAK;AACL,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,mBAAmB,iDAAC,WAAW,oEAAoE;AACnG;;AAEA,IAAI,2DAAE,8CAA8C,iEAAQ;AAC5D;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,sCAAsC;;AAEtC;AACA,EAAE,iDAAC;;AAEH;AACA;AACA,EAAE,iDAAC;AACH,gBAAgB,iBAAiB,YAAY;AAC7C,aAAa,6BAA6B,iBAAiB,2BAA2B,gCAAgC,gBAAgB;AACtI,4BAA4B,uBAAuB,yBAAyB,aAAa,qCAAqC;AAC9H,2BAA2B,uBAAuB,wBAAwB,aAAa,oCAAoC;AAC3H,iCAAiC,uBAAuB,8BAA8B,YAAY;AAClG,4BAA4B,uBAAuB,yBAAyB,aAAa;AACzF,6BAA6B,uBAAuB,0BAA0B,aAAa,qCAAqC;AAChI,4BAA4B,uBAAuB,yBAAyB,aAAa,qCAAqC;AAC9H,iCAAiC,uBAAuB,8BAA8B,aAAa,uCAAuC;AAC1I;;AAEA,EAAE,iDAAC,6BAA6B,uBAAuB,SAAS,8BAA8B,2BAA2B,sBAAsB,WAAW,cAAc,kBAAkB,WAAW,gBAAgB,SAAS,SAAS,UAAU,gBAAgB,eAAe,0BAA0B,4BAA4B,0BAA0B,UAAU,iDAAiD,YAAY,cAAc,eAAe,YAAY,0BAA0B,eAAe,iBAAiB,WAAW,cAAc,WAAW,YAAY,kBAAkB,6BAA6B,0BAA0B,YAAY,+BAA+B,yBAAyB,4BAA4B,aAAa,YAAY,YAAY,mCAAmC,eAAe,+BAA+B,WAAW,kCAAkC,oCAAoC,WAAW,yBAAyB,eAAe,mBAAmB,6BAA6B,YAAY,iBAAiB,UAAU,YAAY,WAAW,kBAAkB,OAAO,MAAM,iBAAiB,WAAW,cAAc,gBAAgB,gBAAgB,kBAAkB,UAAU,YAAY,8BAA8B,yBAAyB,4BAA4B,mCAAmC,qCAAqC,4BAA4B,6BAA6B,YAAY,wDAAwD,iBAAiB,WAAW,kCAAkC,kCAAkC,WAAW,0BAA0B,4BAA4B,yBAAyB,uBAAuB,QAAQ,WAAW,gBAAgB,gBAAgB,eAAe,WAAW,gBAAgB,WAAW,cAAc,kBAAkB,aAAa,kBAAkB,kBAAkB,WAAW,qBAAqB,gBAAgB,cAAc,mBAAmB,sBAAsB,eAAe,gBAAgB,YAAY,qBAAqB,iHAAiH,uBAAuB,gBAAgB,eAAe,kBAAkB,eAAe,gBAAgB,mBAAmB,kBAAkB,sCAAsC,gBAAgB,eAAe,mBAAmB,oBAAoB,mBAAmB,qBAAqB,SAAS,kBAAkB,sBAAsB,yBAAyB,sBAAsB,oBAAoB,iBAAiB,qBAAqB,eAAe,WAAW,cAAc,yBAAyB,gCAAgC,6BAA6B,wBAAwB,YAAY,yBAAyB,2BAA2B,gBAAgB,yCAAyC,mBAAmB,8BAA8B,kCAAkC,0BAA0B,iCAAiC,wBAAwB,0BAA0B,2BAA2B,6DAA6D,kBAAkB,4BAA4B,WAAW,2BAA2B,mBAAmB,0BAA0B,WAAW,yBAAyB,gBAAgB,gBAAgB,gBAAgB,yCAAyC,gBAAgB,+CAA+C,YAAY,cAAc,gBAAgB,cAAc,eAAe,kBAAkB,gBAAgB,SAAS,kBAAkB,8CAA8C,QAAQ,+BAA+B,6DAA6D,uBAAuB,gBAAgB,8BAA8B,YAAY,OAAO,eAAe,MAAM,WAAW,cAAc,cAAc,cAAc,gBAAgB,eAAe,cAAc,mBAAmB,sBAAsB,WAAW,qEAAqE,eAAe,kBAAkB,gBAAgB,aAAa,UAAU,uBAAuB,mBAAmB,kBAAkB,sBAAsB,eAAe,YAAY,qBAAqB,YAAY,4BAA4B,qBAAqB,cAAc,oBAAoB,kCAAkC,yBAAyB,eAAe,eAAe,6BAA6B,iBAAiB,kBAAkB,oBAAoB,eAAe,oBAAoB,cAAc,iBAAiB,eAAe,wBAAwB,YAAY,yBAAyB,kBAAkB,gBAAgB,cAAc,uBAAuB,gBAAgB;AAC7sJ;;AAEA;AACA;AACA;AACA;;AAEA,wCAAwC,kCAAkC;;AAE1E;AACA;AACA;AACA;AACA;;AAEA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA,GAAG;AACH,mCAAmC,iBAAiB;AACpD;;AAEA;AACA;;AAEA,eAAe,4BAA4B;;AAE3C;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,eAAe;AACvC;AACA;AACA,MAAM;AACN;AACA;AACA,2BAA2B,KAAK;AAChC;AACA,KAAK;;AAEL;;AAEA,IAAI;AACJ;AACA;AACA;AACA,KAAK;AACL;AACA,wBAAwB,KAAK;AAC7B;AACA;AACA,GAAG;AACH;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN,2BAA2B,yCAAyC;AACpE,uBAAuB,yCAAyC;AAChE,MAAM,OAAO;AACb,2BAA2B,KAAK;AAChC;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA,yBAAyB,KAAK;AAC9B;AACA;AACA,GAAG;AACH;AACA,GAAG,iDAAC,eAAe,oBAAoB;AACvC;AACA;AACA;AACA;AACA,KAAK;AACL,mDAAmD,sCAAsC;AACzF;AACA;AACA,IAAI;AACJ,GAAG;AACH;AACA,GAAG,iDAAC,mBAAmB,eAAe;AACtC;AACA;AACA;AACA,IAAI;AACJ;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA,0BAA0B,oCAAoC;AAC9D;AACA;AACA;AACA;AACA;AACA,SAAS,qEAAY;AACrB;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA,eAAe,QAAQ,GAAG,wCAAwC;AAClE;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA,aAAa;AACb;;AAEA;AACA;AACA;AACA;AACA,eAAe,gBAAgB,GAAG,uDAAuD;AACzF;AACA;AACA;;AAEA;AACA,IAAI;AACJ,oBAAoB,uBAAuB;AAC3C;AACA;AACA;AACA;AACA,SAAS,qEAAY;AACrB;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ,0BAA0B,QAAQ;AAClC;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH,oBAAoB,WAAW;AAC/B;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,mBAAmB,iBAAiB;AACpC;AACA;AACA;;AAEA;AACA,GAAG;AACH;AACA;AACA,UAAU,qDAAI;AACd,GAAG;AACH;AACA;AACA;AACA;AACA,oBAAoB,yBAAyB;AAC7C;AACA;AACA,IAAI;AACJ,yBAAyB,SAAS;AAClC,IAAI,iDAAC,mBAAmB,GAAG,mEAAmE,GAAG;AACjG;AACA,GAAG;AACH;AACA;AACA;AACA,IAAI;AACJ;AACA,yBAAyB,EAAE;AAC3B,IAAI;AACJ;AACA;AACA;;AAEA;;AAEA;AACA,0DAA0D;AAC1D;AACA;AACA;AACA,aAAa,YAAY;AACzB,KAAK;AACL;AACA;;AAEA,sCAAsC,UAAU;AAChD;AACA,IAAI;AACJ;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA,GAAG;AACH;AACA,mDAAmD,QAAQ;;AAE3D;AACA,UAAU,4BAA4B;;AAEtC,YAAY,6BAA6B;;AAEzC,GAAG,iDAAC,eAAe,eAAe;AAClC,8CAA8C,MAAM;AACpD;AACA,eAAe,iDAAC,SAAS,sDAAsD;AAC/E;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,qDAAI,4BAA4B;;AAEjD;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA,UAAU;AACV;AACA;AACA;AACA;AACA,6FAA6F,4EAA4E;AACzK;AACA;AACA;AACA;AACA;AACA,0HAA0H,GAAG;AAC7H;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B;AAC7B;AACA;AACA,6CAA6C,GAAG;AAChD,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA,6BAA6B;AAC7B;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oDAAoD,GAAG;AACvD;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA,6BAA6B;AAC7B;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa,EAAE;AACf;AACA;AACA;;AAEA;AACA,oBAAoB;AACpB;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,yDAAyD,qBAAqB;AAC9E,0DAA0D,sBAAsB;AAChF;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,iCAAiC,aAAa;AAC9C;AACA;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;;AAEA;AACA,yDAAyD,KAAK;AAC9D;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI,YAAY;AAChB;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,YAAY,YAAY;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA,2BAA2B,QAAQ;AACnC;AACA,IAAI;AACJ;AACA;AACA,2BAA2B,OAAO;AAClC;AACA;AACA,GAAG;AACH,uBAAuB,SAAS;AAChC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,aAAa,QAAQ;AACrB;AACA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA,kBAAkB,iDAAC,WAAW,2CAA2C;AACzE,mBAAmB,iDAAC,WAAW,+DAA+D,gBAAgB,mDAAmD;AACjK,gBAAgB,iDAAC,WAAW,mCAAmC;;AAE/D;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,2CAA2C,sCAAsC,aAAa,oCAAoC;AAClI;;AAEA,eAAe,cAAc;AAC7B,sBAAsB,iEAAQ;AAC9B,qBAAqB,iEAAQ;AAC7B;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,IAAI,YAAY,UAAU;AAC1B;;AAEA;AACA,kDAAkD,gBAAgB;AAClE,mCAAmC,gBAAgB;AACnD;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA,KAAK;AACL;AACA,GAAG;AACH,4BAA4B,MAAM;AAClC;AACA,KAAK;AACL;AACA,GAAG;AACH,oBAAoB,MAAM;AAC1B;;AAEA;AACA;AACA,MAAM;AACN;AACA;;AAEA;AACA,KAAK,8BAA8B;AACnC,KAAK,+BAA+B;AACpC,KAAK,kBAAkB;AACvB;AACA,GAAG;AACH,oBAAoB,KAAK;AACzB;;AAEA;AACA,KAAK,6BAA6B,IAAI,QAAQ;AAC9C,KAAK,4BAA4B,IAAI;AACrC;AACA,GAAG;AACH,sBAAsB,KAAK,GAAG,iBAAiB;AAC/C,6EAA6E,QAAQ,kBAAkB,iCAAiC,SAAS,MAAM;;AAEvJ;AACA,KAAK;AACL;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,OAAO,iEAAQ;AACf;AACA;;AAEA,eAAe,IAAI,IAAI,OAAO;AAC9B,GAAG;AACH;AACA,GAAG,iDAAC;AACJ;AACA,IAAI;AACJ,cAAc;AACd,GAAG;AACH;AACA;;AAEA,GAAG,iDAAC;AACJ,IAAI,iDAAC;AACL,8BAA8B,KAAK,aAAa,KAAK,kBAAkB;AACvE;AACA,KAAK;AACL,IAAI;;AAEJ;AACA,GAAG,iDAAC;AACJ,IAAI,iDAAC;AACL;AACA,8BAA8B,KAAK,kBAAkB;AACrD,MAAM;AACN,8BAA8B,KAAK,cAAc,QAAQ,kBAAkB;AAC3E;AACA;AACA,KAAK;AACL,IAAI;;AAEJ;AACA,GAAG;AACH;AACA;;AAEA,GAAG,iDAAC;AACJ;AACA;;AAEA,KAAK,iDAAC;AACN;AACA,wCAAwC;AACxC;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA,IAAI;;AAEJ;AACA,GAAG;AACH,UAAU;AACV;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,0DAA0D,KAAK;AAC/D,GAAG;AACH,oBAAoB,QAAQ;AAC5B,iBAAiB,iDAAC;AAClB;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;;AAEJ;;AAEA,2BAA2B,YAAY;AACvC,IAAI,iDAAC,cAAc,mBAAmB;;AAEtC,2BAA2B,IAAI,IAAI,IAAI;AACvC,IAAI;;AAEJ,GAAG,iDAAC,SAAS,yCAAyC,qBAAqB,GAAG;;AAE9E;AACA,GAAG;AACH,qBAAqB,QAAQ;AAC7B,UAAU,iDAAC;AACX;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ,GAAG;AACH,oBAAoB,QAAQ;AAC5B,UAAU,iDAAC;AACX;AACA;AACA,IAAI;AACJ;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,gBAAgB,iEAAQ;;AAExB;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,GAAG,iDAAC;AACJ;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,WAAW,iEAAQ;AACnB;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA,cAAc,UAAU;AACxB,gBAAgB,UAAU;AAC1B;AACA;;;AAGA;AACA,CAAC;;;;;;;;;;;;;AClvDD;AAAA;AAAA;AAAA;AAAA;AAAA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;;AAEO;;AAEA;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,WAAW,YAAY;AACvB,WAAW,OAAO;AAClB,WAAW,OAAO;AAClB,WAAW,SAAS;AACpB,WAAW,QAAQ;AACnB;AACO;AACP;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEO;AACP;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,E;;;;;;;;;;;;AC/DA;AAAA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oDAAoD,KAAK;AACzD,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;;AAEe,gEAAC,E;;;;;;;;;;;;AC7ChB;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG,YAAY,cAAc;AAC7B,EAAE;AACF;AACA;AACA;AACA,GAAG;AACH,2BAA2B,aAAa,GAAG,UAAU;AACrD;AACA,EAAE;AACF;AACA;AACA,aAAa,mBAAmB,2BAA2B,cAAc,EAAE,EAAE,aAAa,EAAE;AAC5F,sBAAsB,0BAA0B,+BAA+B,EAAE;AACjF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA,EAAE;AACF,cAAc,uBAAuB;AACrC;AACA;;AAEe,mEAAI,E;;;;;;;;;;;;ACtDnB;AAAA;AAAA;AAA0B;AACN;;AAEpB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,gBAAgB;AACjC;AACA,GAAG,6CAAI,mBAAmB,qCAAqC;AAC/D;AACA;AACA,EAAE;AACF;AACA,oBAAoB,QAAQ;AAC5B;;AAEA,SAAS,OAAO,GAAG,6CAAI;AACvB;AACA,eAAe,oBAAoB;AACnC,qBAAqB,YAAY;AACjC;;AAEA,EAAE,yCAAC;AACH;AACA,EAAE,yCAAC;AACH,EAAE;AACF;AACA,OAAO,6CAAI,MAAM,QAAQ;AACzB,iDAAiD,QAAQ;AACzD,iBAAiB,6CAAI;AACrB;;AAEA;AACA;AACA,GAAG,6CAAI,mBAAmB,qDAAqD;AAC/E;AACA,GAAG;AACH;AACA;AACA,EAAE;AACF,mCAAmC,6BAA6B,gCAAgC,eAAe,WAAW,uBAAuB,eAAe,gBAAgB,mBAAmB,eAAe,eAAe,WAAW,SAAS,0CAA0C,mxEAAmxE,qBAAqB,sBAAsB,YAAY,iBAAiB,wCAAwC,yBAAyB,qBAAqB,uBAAuB;AACvuF;AACA;;AAEe,qEAAM,E","file":"ytma.user.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 = \"./src/index.js\");\n","import { isNumber, removeSearch, on, debounce } from './modules/Helpers';\r\nimport _ from './modules/_';\r\nimport strg from './modules/strg';\r\nimport update from './modules/update';\r\n\r\n(() => {\r\n\r\n\t/** Y T M A CLASS\r\n\t * @private\r\n\t * Base YTMA class, filled through constructor() or reactivate() though sub-classes\r\n\t * Y's only concerned about the anchor and the data props\r\n\t *\r\n\t * @param {object} props Properties\r\n\t * @param {String|Number} props.id Unique ID\r\n\t * @param {String} props.site Website name eg: youtube, vimeo\r\n\t * @param {HTMLAnchorElement} props.anchor Anchor element\r\n\t */\r\n\tclass Y {\r\n\r\n\t\tconstructor({ id, site, anchor }) {\r\n\t\t\tconst uid = Y.escapeId(`${id}_${Y.num += 1}`);\r\n\r\n\t\t\tthis.state = {\r\n\t\t\t\tid,\r\n\t\t\t\tuid: Y.escapeId(uid), // unique id\r\n\t\t\t\tsid: Y.escapeId(id), // shared id\r\n\t\t\t\tsite,\r\n\t\t\t\turi: anchor.href\r\n\t\t\t};\r\n\r\n\t\t\tif (anchor && !anchor.dataset.ytmscroll) { anchor.dataset.ytmscroll = true; }\r\n\r\n\t\t\tthis.anchor = anchor;\r\n\t\t}\r\n\r\n\t\t/**\r\n\t\t * Recreates a YTMA object from a trigger element\r\n\t\t * @param {HTMLElement} element the element's dataset for the resurection!\r\n\t\t */\r\n\t\treactivate({ dataset }) {\r\n\t\t\tconst id = dataset.ytmid;\r\n\t\t\tconst anchor = document.querySelector(`a[data-ytmuid=\"${dataset.ytmuid}\"]`);\r\n\r\n\t\t\tthis.state = {\r\n\t\t\t\tid,\r\n\t\t\t\tuid: dataset.ytmuid,\r\n\t\t\t\tsid: dataset.ytmsid,\r\n\t\t\t\tsite: dataset.ytmsite,\r\n\t\t\t\turi: anchor.href\r\n\t\t\t};\r\n\r\n\t\t\tthis.anchor = anchor;\r\n\r\n\t\t\treturn this;\r\n\t\t}\r\n\r\n\t\tdisableOpenOnScroll() {\r\n\t\t\tthis.anchor.dataset.ytmscroll = false;\r\n\t\t}\r\n\r\n\t\tcanScroll() {\r\n\t\t\treturn this.anchor.dataset.ytmscroll === 'true';\r\n\t\t}\r\n\r\n\t\tisBelow(link) {\r\n\t\t\treturn Scroll.compare(this.anchor, link) < 1;\r\n\t\t}\r\n\r\n\t\tcanShowUnder(link) {\r\n\t\t\tthis.canScroll() && this.isBelow(link);\r\n\t\t}\r\n\r\n\t\tupdateAnchor() {\r\n\t\t\tif (this.anchor.getElementsByTagName('img').length === 0) {\r\n\t\t\t\tthis.anchor.classList.add('ytm_link', `ytm_link_${this.state.site}`);\r\n\t\t\t}\r\n\t\t\tthis.anchor.dataset.ytmid = this.state.id;\r\n\t\t\tthis.anchor.dataset.ytmuid = this.state.uid;\r\n\t\t\tthis.anchor.dataset.ytmsid = this.state.sid;\r\n\t\t\tthis.anchor.title = 'Visit the video page.';\r\n\t\t}\r\n\r\n\t}\r\n\r\n\t/** C O N T A I N E R CLASS\r\n\t * The container, as the name implies, contains all the interactive elements\r\n\t * Thumbnail, Player, Controls, etc.\r\n\t */\r\n\tclass Container extends Y {\r\n\r\n\t\tcreateInterface() {\r\n\t\t\tconst { state } = this;\r\n\t\t\tthis.site = Y.DB.sites[state.site];\r\n\t\t\tconst { ajax, slim } = this.site;\r\n\r\n\t\t\tthis.updateAnchor();\r\n\r\n\t\t\tthis.body = _.e('div', {\r\n\t\t\t\tid: `w${state.uid}`,\r\n\t\t\t\tclassName: `ytm_spacer ytm_block ytm_site_${state.site}`,\r\n\t\t\t\tinnerHTML: this.createThumbnailTemplate()\r\n\t\t\t});\r\n\r\n\t\t\tthis.thumbnail = this.body.firstElementChild;\r\n\r\n\t\t\tif (ajax) { this.createAjaxLink(); }\r\n\t\t\tif (slim) { this.body.classList.add('ytm_site_slim'); }\r\n\r\n\t\t\tthis.anchor.insertAdjacentElement('afterend', this.body);\r\n\r\n\t\t\ttry {\r\n\t\t\t\tContainer.decorators[state.site].gui(this);\r\n\t\t\t} catch (e) {\r\n\t\t\t\t// meh\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\tupdateAnchor() {\r\n\t\t\tconst { scroll, https } = this;\r\n\t\t\tif (scroll) { this.anchor.classList.add('ytm_scroll'); }\r\n\t\t\tif (https) { this.anchor.href = this.anchor.href.replace('http:', 'https:'); }\r\n\r\n\t\t\tsuper.updateAnchor();\r\n\t\t}\r\n\r\n\t\tcreateThumbnailTemplate() {\r\n\t\t\tconst { title, thumb = '' } = this.site;\r\n\t\t\tconst { id, uid, sid, site } = this.state;\r\n\r\n\t\t\tconst bg = thumb ? `background-image: ${thumb.replace('%key', id)}` : '';\r\n\r\n\t\t\tconst template = `\r\n\t\t\t\t<span class=\"ytm_trigger ytm_block ytm_normalize ytm_sans\"\r\n\t\t\t\t\ttitle=\"${title}\"\r\n\t\t\t\t\tdata-ytmid=\"${id}\"\r\n\t\t\t\t\tdata-ytmsite=\"${id}\"\r\n\t\t\t\t\tstyle=\"${bg}\">\r\n\t\t\t\t\t<span class=\"ytm_init ytm_label ytm_sans ytm_box\">${title}</span>\r\n\t\t\t\t\t\t<var class=\"ytm_label ytm_box\"\r\n\t\t\t\t\t\t\tdata-ytmid=\"${id}\"\r\n\t\t\t\t\t\t\tdata-ytmuid=\"${uid}\"\r\n\t\t\t\t\t\t\tdata-ytmsid=\"${sid}\"\r\n\t\t\t\t\t\t\tdata-ytmsite=\"${site}\">\\u25B6</var>\r\n\t\t\t\t\t</span>\r\n\t\t\t\t</span>`;\r\n\t\t\treturn template;\r\n\t\t}\r\n\r\n\t\tcreateAjaxLink() {\r\n\t\t\tconst { sid, id, site, uri } = this.state;\r\n\t\t\tconst template = `\r\n\t\t\t\t<span class=\"ytm_bd ytm_normalize ytm_manual _${sid}\">\r\n\t\t\t\t\t<a href=\"#\" class=\"ytm_title\" title=\"Load this video's description.\"\r\n\t\t\t\t\t\tdata-ytmid=\"${id}\"\r\n\t\t\t\t\t\tdata-ytmsite=\"${site}\"\r\n\t\t\t\t\t\tdata-ytmuri=\"${uri}\"\r\n\t\t\t\t\t\tdata-ytmdescription=\"true\"\r\n\t\t\t\t\t>Load Description</a>\r\n\t\t\t\t</span>`;\r\n\t\t\tthis.body.insertAdjacentHTML('beforeend', template);\r\n\t\t}\r\n\r\n\t\tcreateProjector() {\r\n\t\t\tthis.projector = _.e('div', {\r\n\t\t\t\tclassName: 'ytm_projector ytm_none ytm_block ytm_normalize ytm_sans',\r\n\t\t\t\tinnerHTML: Container.templates.menu\r\n\t\t\t});\r\n\t\t\tthis.thumbnail.insertAdjacentElement('afterend', this.projector);\r\n\t\t}\r\n\r\n\t\tshowPlayer() {\r\n\t\t\tthis.thumbnail.classList.add('ytm_none');\r\n\t\t\tthis.projector.classList.remove('ytm_none');\r\n\t\t}\r\n\r\n\t\thidePlayer() {\r\n\t\t\tthis.thumbnail.classList.remove('ytm_none');\r\n\t\t\tthis.projector.classList.add('ytm_none');\r\n\t\t}\r\n\r\n\t}\r\n\r\n\tContainer.templates = {\r\n\t\tmenu: `\r\n\t\t\t<ul class=\"ytm_options ytm_sans\">\r\n\t\t\t\t<li>\r\n\t\t\t\t\t<ul class=\"ytm_ratios\">\r\n\t\t\t\t\t\t<li data-type=\"ratio\" data-value=\"1\" title=\"SD\">4:3</li>\r\n\t\t\t\t\t\t<li data-type=\"ratio\" data-value=\"2\" title=\"Landscape\">16:9</li>\r\n\t\t\t\t\t\t<li data-type=\"ratio\" data-value=\"3\" title=\"Portrait\">9:16</li>\r\n\t\t\t\t\t</ul>\r\n\t\t\t\t</li>\r\n\t\t\t\t<li>\r\n\t\t\t\t\t<ul class=\"ytm_sizes\">\r\n\t\t\t\t\t\t<li data-type=\"size\" data-value=\"0\" title=\"Hide the video.\">\\u00D8</li>\r\n\t\t\t\t\t\t<li data-type=\"size\" data-value=\"240\" title=\"240p\">S</li>\r\n\t\t\t\t\t\t<li data-type=\"size\" data-value=\"360\" title=\"360p\">M</li>\r\n\t\t\t\t\t\t<li data-type=\"size\" data-value=\"480\" title=\"480p\">L</li>\r\n\t\t\t\t\t\t<li data-type=\"size\" data-value=\"720\" title=\"720p\">X</li>\r\n\t\t\t\t\t</ul>\r\n\t\t\t\t</li>\r\n\t\t\t\t<li>\r\n\t\t\t\t\t<ul class=\"ytm_options\">\r\n\t\t\t\t\t\t${strg.on ? '<li data-type=\"settings\" data-value=\"\" title=\"YTMA Settings\">!</li>' : ''}\r\n\t\t\t\t\t\t<li data-type=\"close\" data-value=\"\" title=\"Close the video.\">\\u00D7</li>\r\n\t\t\t\t\t</ul>\r\n\t\t\t\t</li>\r\n\t\t\t</ul>`\r\n\t};\r\n\r\n\tContainer.decorators = { // modify interface according to site\r\n\t\tyoutube: {\r\n\t\t\tgui: function (control) {\r\n\t\t\t\tcontrol.anchor.href = this.anchor.href.replace('youtu.be/', 'youtube.com/watch?v=');\r\n\t\t\t},\r\n\t\t\tthumbEvent: function (e) {\r\n\t\t\t\tlet time = +this.dataset.time || 1;\r\n\t\t\t\tif (this.classList.contains('ytm_trigger') && e.type === 'mouseenter' && time < 50) {\r\n\t\t\t\t\tthis.dataset.thumb = ((this.dataset.thumb || 0) + 1) % 3;\r\n\t\t\t\t\tthis.style.backgroundImage = `url(https://i3.ytimg.com/vi/${this.dataset.ytmid}/${(+this.dataset.thumb) + 1}.jpg)`;\r\n\t\t\t\t\twindow.clearTimeout(this.dataset.timeout);\r\n\t\t\t\t\tconsole.log('mouseenter -- clear before setting new ', this.dataset);\r\n\t\t\t\t\tthis.dataset.timeout = window.setTimeout(Container.decorators.youtube.thumbEvent.bind(this, e), 800);\r\n\t\t\t\t\tconsole.log('mouseenter -- new timeout', this.dataset);\r\n\t\t\t\t\tthis.dataset.time = time += 1;\r\n\t\t\t\t} else {\r\n\t\t\t\t\twindow.clearTimeout(this.dataset.timeout);\r\n\t\t\t\t\tthis.dataset.time = 0;\r\n\t\t\t\t\tconsole.log('mouseleave -- ', this.dataset);\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\t};\r\n\r\n\tContainer.events = {\r\n\t\tsetup: () => {\r\n\t\t\ton(document.body, 'click', 'var[data-ytmuid]', Container.events.fromTarget);\r\n\t\t\ton(document.body, 'click', 'a[data-ytmdescription]', Container.events.manualLoad);\r\n\t\t\ton(document.body, 'dblclick', 'q[data-full]', Container.events.titleToggle);\r\n\r\n\t\t\ton(document.body, 'mouseenter mouseleave', 'div.ytm_site_youtube span.ytm_trigger', Container.decorators.youtube.thumbEvent);\r\n\t\t},\r\n\t\tfromTarget: ({ target }) => { // trigger the ui\r\n\t\t\tconsole.info('ytma//click+trig(id)', target.dataset.ytmuid);\r\n\t\t\tControl.createFromTrigger(target).showPlayer();\r\n\t\t},\r\n\t\tmanualLoad: e => {\r\n\t\t\te.preventDefault();\r\n\t\t\tconst { target } = e;\r\n\t\t\tconsole.info('ytma//click+desc(id)', target.dataset.ytmid);\r\n\t\t\tif ((target.dataset.tries || 0) <= 4) {\r\n\t\t\t\tY.ajax.loadFromDataset(target.dataset);\r\n\t\t\t}\r\n\t\t},\r\n\t\ttitleToggle: e => {\r\n\t\t\tconst target = e.target;\r\n\t\t\ttarget.classList.toggle('ytm_descr_open');\r\n\t\t\ttarget.textContent = target.textContent.length < 140 ? target.dataset.full : `${target.dataset.full.substr(0, 130)} . . .`;\r\n\t\t\ttarget.removeAttribute('style');\r\n\t\t}\r\n\t};\r\n\r\n\tY.num = 0;\r\n\r\n\tY.addToSet = ytma => Y.set[ytma.state.uid] = ytma;\r\n\r\n\tY.create = link => Y.grabIdAndSite(link, (data, err) => {\r\n\t\tif (err) {\r\n\t\t\tconsole.warn(link.href, err);\r\n\t\t\treturn {};\r\n\t\t}\r\n\r\n\t\tconst control = new Control({ ...data, anchor: link });\r\n\t\tY.addToSet(control);\r\n\t\tcontrol.createInterface();\r\n\r\n\t\treturn control;\r\n\t});\r\n\r\n\tY.grabIdAndSite = (link, cb) => {\r\n\t\tlet uri = link.href || link.src;\r\n\t\tlet id;\r\n\t\tlet match;\r\n\t\ttry {\r\n\t\t\tconst site = Y.reg.siteByTest[Y.reg.siteExpressions.test(uri) ? RegExp.lastMatch : ''];\r\n\t\t\t// console.log(site);\r\n\r\n\t\t\tif (site === 'html5') { // || site === 'html5-audio'\r\n\t\t\t\tid = uri.slice(-15);\r\n\t\t\t} else if (site === 'soundcloud') {\r\n\t\t\t\tif (!Y.reg.extra.soundcloud.playlist.test(uri)) {\r\n\t\t\t\t\tlink.href = uri = Y.reg.fix.soundcloud(uri);\r\n\t\t\t\t}\r\n\r\n\t\t\t\tmatch = Y.DB.sites.soundcloud.matcher.exec(uri);\r\n\t\t\t\tid = Y.escapeId(match[1]);\r\n\r\n\t\t\t\tif (match && Y.reg.extra.soundcloud.tracks.test(uri)) {\r\n\t\t\t\t\tid = id.slice(-50);\r\n\t\t\t\t}\r\n\t\t\t} else {\r\n\t\t\t\tid = uri.match(Y.DB.sites[site].matcher)[1];\r\n\t\t\t}\r\n\r\n\t\t\tconsole.info('ytma//id+site', id, site, match);\r\n\t\t\tif (id && Y.DB.sites[site]) {\r\n\t\t\t\treturn cb({ id, site }, null);\r\n\t\t\t}\r\n\t\t\tthrow TypeError(`Invalid ID/Site: ${id} @ ${site}`);\r\n\t\t} catch (e) {\r\n\t\t\treturn cb(null, e);\r\n\t\t}\r\n\t};\r\n\r\n\tY.escapeId = id => `${id}`.replace(/(?:\\W)/g, '_');\r\n\r\n\tY.set = {};\r\n\r\n\tY.collect = id => {\r\n\t\tconst a = Object.values(Y.set).filter(ytma => ytma && ytma.data.id === id);\r\n\t\treturn a;\r\n\t};\r\n\r\n\tY.route = {\r\n\t\thost: document.location.host.replace('www.', ''),\r\n\t\tcontrol: {\r\n\t\t\t$: {\r\n\t\t\t\tcheckStorage: function () {\r\n\t\t\t\t\tif (strg.full() === true) {\r\n\t\t\t\t\t\tconsole.log('YTMA ERROR: Storage is full!');\r\n\t\t\t\t\t\ttry {\r\n\t\t\t\t\t\t\tlocalStorage.removeItem(Y.external.version);\r\n\t\t\t\t\t\t\tstrg.on = strg.test();\r\n\t\t\t\t\t\t} catch (e) {\r\n\t\t\t\t\t\t\tconsole.error(e);\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}\r\n\t\t\t\t},\r\n\t\t\t\trunOnce: function (loop) {\r\n\t\t\t\t\tif (!document.body.dataset.ytmaenabled) {\r\n\t\t\t\t\t\tdocument.body.dataset.ytmaenabled = true;\r\n\r\n\t\t\t\t\t\tthis.checkStorage();\r\n\r\n\t\t\t\t\t\tif (!Y.DB.extension) { update.check(); }\r\n\r\n\t\t\t\t\t\tY.css();\r\n\t\t\t\t\t\tY.user.init();\r\n\t\t\t\t\t\tY.DB.postInit();\r\n\r\n\t\t\t\t\t\tif (loop) {\r\n\t\t\t\t\t\t\tdocument.body.dataset.YTMA_LOOP = window.setInterval(loop, 5000);\r\n\t\t\t\t\t\t\tloop();\r\n\t\t\t\t\t\t}\r\n\r\n\t\t\t\t\t\tContainer.events.setup();\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tgo: function (host) {\r\n\t\t\t\tconsole.info('ytma//host', host);\r\n\t\t\t\tif (/(?:googlevideo|youtube-nocookie\\.com|youtube\\.com\\.?)/i.test(host)) {\r\n\t\t\t\t\tthis.sites.youtube();\r\n\t\t\t\t} else if (this.sites[host]) {\r\n\t\t\t\t\tthis.sites[host]();\r\n\t\t\t\t} else {\r\n\t\t\t\t\tthis.sites.$generic();\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tsites: {\r\n\t\t\t\t$generic: function () {\r\n\t\t\t\t\tfunction loop() {\r\n\t\t\t\t\t\tif (Y.selector.processor() > 0) {\r\n\t\t\t\t\t\t\tY.user.fn.loadPreferences();\r\n\t\t\t\t\t\t}\r\n\r\n\t\t\t\t\t\tconsole.info('ytma//again++');\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t\tY.route.control.$.runOnce(loop);\r\n\t\t\t\t},\r\n\t\t\t\t'resetera.com': function () {\r\n\t\t\t\t\t_.css('.ytm_options li ul li { height: 24px !important }');\r\n\t\t\t\t\t_.css('.bbCodeQuote .quoteContainer .quote { max-height: initial } .bbCodeQuote .quoteContainer .quoteExpand.quoteCut { display: none }');\r\n\t\t\t\t\t_.css('.bbCodeQuote .ytm_block iframe, .bbCodeQuote .ytm_block [data-s9e-mediaembed], .bbCodeQuote .ytm_block .fb_iframe_widget, .bbCodeQuote .ytm_block object, .bbCodeQuote .ytm_block embed { max-height: initial; max-width: initial }');\r\n\t\t\t\t\tthis.$generic();\r\n\t\t\t\t},\r\n\t\t\t\t'gfycat.com': function () {\r\n\t\t\t\t\tconst v = document.querySelector('video');\r\n\t\t\t\t\tv.controls = true;\r\n\t\t\t\t\t_.css('body,html {overflow:hidden;height:100%;width:100%} video {display:table;height:100%;margin:0 auto;}');\r\n\t\t\t\t\tdocument.body.appendChild(v);\r\n\t\t\t\t},\r\n\t\t\t\t'vine.co': function () {\r\n\t\t\t\t\t// console.log('vine.co');\r\n\r\n\t\t\t\t\twindow.addEventListener('resize', () => {\r\n\t\t\t\t\t\t_.s('[style]', e => {\r\n\t\t\t\t\t\t\te.removeAttribute('style');\r\n\t\t\t\t\t\t});\r\n\t\t\t\t\t});\r\n\t\t\t\t},\r\n\t\t\t\tyoutube: function () { // lets force some quality parity\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t},\r\n\t\tload: function () {\r\n\t\t\tthis.control.go(this.host);\r\n\t\t}\r\n\t};\r\n\r\n\tY.main = () => {\r\n\t\tY.reg.siteExpressions = Y.reg.getAllSiteRegExps();\r\n\t\t// console.log(YTMA.reg.siteExpressions);\r\n\t\tY.route.load();\r\n\t};\r\n\r\n\tY.reg = {\r\n\t\tsiteExpressions: null,\r\n\t\ttime: /(?:t=(?:(\\d+)h)?(?:(\\d+)m)?(\\d+))/,\r\n\t\tios: /(?:\\b(?:ipod|iphone|ipad))\\b/i,\r\n\t\textra: {\r\n\t\t\tsoundcloud: {\r\n\t\t\t\tplaylist: /(?:soundcloud\\.com\\/.+\\/sets\\/)/,\r\n\t\t\t\ttracks: /(?:soundcloud\\.com\\/.+\\/tracks\\/)/\r\n\t\t\t}\r\n\t\t},\r\n\t\tsiteByTest: {\r\n\t\t\tyoutu: 'youtube',\r\n\t\t\tvimeo: 'vimeo',\r\n\t\t\tvine: 'vine',\r\n\t\t\tgfycat: 'gfycat',\r\n\t\t\timgur: 'imgur',\r\n\t\t\t'.webm': 'html5',\r\n\t\t\t'.mp4': 'html5',\r\n\t\t\t// '.mp3': 'html5-audio',\r\n\t\t\t'.gifv': 'html5',\r\n\t\t\tsoundcloud: 'soundcloud',\r\n\t\t\t'streamable.com': 'streamable'\r\n\t\t},\r\n\t\tgetAllSiteRegExps: function () {\r\n\t\t\tconst regs = Object.values(Y.DB.sites)\r\n\t\t\t\t.filter(({ reg }) => reg)\r\n\t\t\t\t.map(({ reg }) => reg);\r\n\r\n\t\t\treturn new RegExp(`\\\\b${regs.join('|')}`);\r\n\t\t},\r\n\t\tfix: {\r\n\t\t\tsoundcloud: function (uri) {\r\n\t\t\t\tconst match = Y.DB.sites.soundcloud.matcher.exec(uri);\r\n\t\t\t\tif (match) {\r\n\t\t\t\t\tconst id = match[1].split('/', 2).join('/');\r\n\t\t\t\t\turi = removeSearch(`https://soundcloud.com/${id}`, true);\r\n\t\t\t\t}\r\n\r\n\t\t\t\treturn uri;\r\n\t\t\t}\r\n\t\t}\r\n\t};\r\n\r\n\tY.selector = { // to build the selector\r\n\t\tparentBlacklist: ['.smallfont', '.colhead_dark', '.spoiler', 'pre', '.messageUserInfo', '.fr-box'],\r\n\t\tgetAllSiteSelectors: function () {\r\n\t\t\tconst sels = Object.values(Y.DB.sites)\r\n\t\t\t\t.filter(({ selector }) => selector)\r\n\t\t\t\t.map(({ selector }) => selector);\r\n\r\n\t\t\treturn sels.join();\r\n\t\t},\r\n\t\tignore: function () {\r\n\t\t\tconst ignore = [];\r\n\t\t\tconst all = Y.selector.getAllSiteSelectors().split(',');\r\n\t\t\tconst blacklist = this.parentBlacklist;\r\n\t\t\tfor (let i = 0; i < blacklist.length; i++) {\r\n\t\t\t\tfor (let j = 0; j < all.length; j++) {\r\n\t\t\t\t\tignore.push(`${blacklist[i]} ${all[j]}`);\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t\t// console.log(ignore.join(','));\r\n\t\t\treturn ignore.join(',');\r\n\t\t},\r\n\t\tiframes: function () { // for resetera, convert iframes back to anchors\r\n\t\t\t_.s('.message-body iframe', f => {\r\n\t\t\t\tif (/vi\\/(.+?)\\/hqdefault/.test(f.style.backgroundImage)) {\r\n\t\t\t\t\tconst src = `https://youtu.be/${RegExp.$1}`;\r\n\t\t\t\t\tconst span = f.closest('[data-s9e-mediaembed]');\r\n\t\t\t\t\tspan.insertAdjacentHTML('beforebegin', `<a href=\"${src}\">youtube</a>`);\r\n\t\t\t\t\tspan.parentElement.removeChild(span);\r\n\t\t\t\t}\r\n\t\t\t});\r\n\r\n\t\t\t_.s('[data-s9e-mediaembed-iframe]', s => {\r\n\t\t\t\tif (Y.DB.sites[s.dataset.s9eMediaembed]) {\r\n\t\t\t\t\tconst dat = JSON.parse(s.dataset.s9eMediaembedIframe);\r\n\t\t\t\t\tconst link = dat[dat.length - 1];\r\n\t\t\t\t\ts.parentElement.parentElement.innerHTML = `<a href=\"${link}\">${link}</a>`;\r\n\t\t\t\t}\r\n\t\t\t});\r\n\t\t},\r\n\t\tlinks: function () {\r\n\t\t\t_.s(Y.selector.ignore(), ({ dataset }) => dataset.ytmaignore = true);\r\n\r\n\t\t\tconst links = _.qsa(Y.selector.getAllSiteSelectors()).filter(({ dataset }) => {\r\n\t\t\t\tconst r = !dataset.ytmaprocessed && !dataset.ytmaignore;\r\n\t\t\t\tdataset.ytmaprocessed = true;\r\n\t\t\t\treturn r;\r\n\t\t\t});\r\n\r\n\t\t\treturn links;\r\n\t\t},\r\n\t\tprocessor: function () {\r\n\t\t\tthis.iframes();\r\n\t\t\tconst links = this.links();\r\n\r\n\t\t\tif (links.length > 0) {\r\n\t\t\t\tlinks.forEach(Y.create);\r\n\t\t\t}\r\n\r\n\t\t\treturn links.length;\r\n\t\t}\r\n\t};\r\n\r\n\t/**\r\n\t * User Preferences\r\n\t * size: Small (240p), Medium (360p), Large (480p), XL (720p)\r\n\t * ratio: 1 4:3, 2 16:9\r\n\t * quality: 240, 360, 480, 720, 1080\r\n\t * focus: 0/1; Will attempt to set the window's focus near the video\r\n\t * autoShow: 0/1; Will automatically display HTML5 videos, which currently lack descriptions and thumbnails\r\n\t * desc: (Descriptions) 0 None; 1 Yes on scroll; 2 Yes all at once\r\n\t * yt_nocookie: 0/1; Will disable/enable youtube-nocookie.com\r\n\t * yt_annotation: 0/1; youtube annotations\r\n\t */\r\n\tY.user = {\r\n\t\tKEY: 'ytmasetts',\r\n\t\t$form: null,\r\n\t\tinit: function () {\r\n\t\t\tthis.load();\r\n\r\n\t\t\tif (strg.on) {\r\n\t\t\t\tthis.fn.makeForm();\r\n\t\t\t\tthis.mark();\r\n\t\t\t}\r\n\t\t},\r\n\t\tvalid: {\r\n\t\t\tfocus: [0, 1],\r\n\t\t\tdesc: [0, 1, 2],\r\n\t\t\tratio: [1, 2],\r\n\t\t\tsize: [240, 360, 480, 720],\r\n\t\t\tquality: [240, 360, 480, 720, 1080],\r\n\t\t\tautoShow: [0, 1],\r\n\t\t\tyt_nocookie: [0, 1],\r\n\t\t\tyt_annotation: [0, 1] // hide | show\r\n\t\t},\r\n\t\tmapping: { // map values to some other values used by an external API, for example\r\n\t\t\tyt_annotation: [3, 1] // 3 = hide | 1 = show\r\n\t\t},\r\n\t\tvalidate: function (property, n) {\r\n\t\t\tn = +n;\r\n\t\t\treturn Y.user.valid[property].includes(n) ? n : Y.user.defaults[property];\r\n\t\t},\r\n\t\tget defaults() {\r\n\t\t\treturn {\r\n\t\t\t\tfocus: 0,\r\n\t\t\t\tdesc: 1,\r\n\t\t\t\tratio: 2,\r\n\t\t\t\tsize: 360,\r\n\t\t\t\tquality: 720,\r\n\t\t\t\tautoShow: 1,\r\n\t\t\t\tyt_nocookie: 0,\r\n\t\t\t\tyt_annotation: 1\r\n\t\t\t};\r\n\t\t},\r\n\t\tload: function () {\r\n\t\t\tconst s = strg.grab(Y.user.KEY, {});\r\n\r\n\t\t\tY.user.preferences = Object.keys(this.defaults).reduce((valid, k) => {\r\n\t\t\t\tvalid[k] = Y.user.validate(k, s[k]);\r\n\t\t\t\treturn valid;\r\n\t\t\t}, {});\r\n\r\n\t\t\t_.o(Y.user.mapping, (key, val) => {\r\n\t\t\t\tif (!val.hasOwnProperty('indexOf')) {\r\n\t\t\t\t\tY.user.preferences[key] = val[Y.user.valid[key].indexOf(Y.user.preferences[key])];\r\n\t\t\t\t}\r\n\t\t\t});\r\n\r\n\t\t\tconsole.info('ytma//user+loaded(prefs)', Y.user.preferences);\r\n\t\t},\r\n\t\tmark: function () {\r\n\t\t\tconst a = {};\r\n\t\t\ta.ytma__focus = !!Y.user.preferences.focus;\r\n\t\t\ta.ytma__autoShow = !!Y.user.preferences.autoShow;\r\n\t\t\ta.ytma__yt_nocookie = !!Y.user.preferences.yt_nocookie;\r\n\t\t\ta.ytma__yt_annotation = !!Y.user.preferences.yt_annotation;\r\n\t\t\ta[`ytma__ratio${Y.user.preferences.ratio}`] = true;\r\n\t\t\ta[`ytma__size${Y.user.preferences.size}`] = true;\r\n\t\t\ta[`ytma__desc${Y.user.preferences.desc}`] = true;\r\n\t\t\ta[`ytma__quality${Y.user.preferences.quality}`] = !!Y.user.preferences.quality;\r\n\r\n\t\t\t// console.log('marking', a);\r\n\t\t\t_.o(a, (id, val) => {\r\n\t\t\t\ttry {\r\n\t\t\t\t\tconst el = document.getElementById(id);\r\n\t\t\t\t\tel.checked = val;\r\n\t\t\t\t\tel.value = val;\r\n\t\t\t\t} catch (e) {\r\n\t\t\t\t\t// console.log(id, e);\r\n\t\t\t\t}\r\n\t\t\t});\r\n\t\t},\r\n\t\tevents: {\r\n\t\t\tsave: function () {\r\n\t\t\t\t// console.log(YTMA.user.$form.querySelectorAll('[data-key]'));\r\n\t\t\t\t// [data-key]:checked\r\n\t\t\t\tconst settings = Array.from(Y.user.$form.querySelectorAll('[data-key]')).reduce((obj, e) => {\r\n\t\t\t\t\tlet key = e.dataset.key;\r\n\r\n\t\t\t\t\tif (e.type === 'checkbox') {\r\n\t\t\t\t\t\tobj[key] = +e.checked;\r\n\t\t\t\t\t} else if (e.type === 'radio') {\r\n\t\t\t\t\t\tif (e.checked) {\r\n\t\t\t\t\t\t\tif (e.dataset.num) {\r\n\t\t\t\t\t\t\t\tobj[key] = +e.dataset.num;\r\n\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t} else {\r\n\t\t\t\t\t\tobj[key] = +e.value;\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t\treturn obj;\r\n\t\t\t\t}, {});\r\n\r\n\t\t\t\tif (strg.save(Y.user.KEY, settings)) {\r\n\t\t\t\t\tY.user.load();\r\n\t\t\t\t} else {\r\n\t\t\t\t\tY.user.error.classList.remove('ytm_none');\r\n\t\t\t\t}\r\n\r\n\t\t\t},\r\n\t\t\treset: function () {\r\n\t\t\t\tY.user.preferences = Y.user.defaults;\r\n\t\t\t\tY.user.mark();\r\n\t\t\t\tstrg.wipe(Y.user.KEY);\r\n\t\t\t\tY.user.error.classList.add('ytm_none');\r\n\t\t\t},\r\n\t\t\tclear: function () {\r\n\t\t\t\ttry {\r\n\t\t\t\t\tlocalStorage.removeItem(Y.external.version);\r\n\t\t\t\t\tY.user.events.reset();\r\n\t\t\t\t\tconsole.info('ytma//cache+remove', 'removed all YTMA cache');\r\n\t\t\t\t} catch (e) {\r\n\t\t\t\t\tconsole.error(e);\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tformToggle: function (e) {\r\n\t\t\t\tif (Y.user.$form && (!e || (e && e.target && !(/(?:INPUT|LABEL)/i).test(e.target.nodeName)))) {\r\n\t\t\t\t\tY.user.$form.classList.toggle('ytm_none');\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tformToggleKeyboard: function (e) {\r\n\t\t\t\t// press CTRL+SHIFT+Y (META+SHIFT+Y) to display settings form\r\n\t\t\t\tif ((e.ctrlKey || e.metaKey) && e.shiftKey && String.fromCharCode(e.which).toLowerCase() === 'y') {\r\n\t\t\t\t\te.preventDefault();\r\n\t\t\t\t\tY.user.events.formToggle();\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t},\r\n\t\tfn: {\r\n\t\t\t$scroller: null,\r\n\t\t\t$once: false,\r\n\t\t\tloadPreferences: function () {\r\n\t\t\t\tY.user.fn.onScrollLoadDescriptions(Y.user.preferences.desc === 1);\r\n\r\n\t\t\t\tthis.loadPreferencesOnce();\r\n\t\t\t},\r\n\t\t\tloadPreferencesOnce: function () {\r\n\t\t\t\tif (this.$once) { return; }\r\n\r\n\t\t\t\tthis.$once = true;\r\n\r\n\t\t\t\tif (Y.user.preferences.autoShow === 1) {\r\n\t\t\t\t\tY.user.fn.onScrollViewMedia();\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tshowMedia: function () {\r\n\t\t\t\t// console.info('ytma//user+fn-showMedia');\r\n\t\t\t\treturn new Scroll('a.ytm_scroll:not([data-ytmscroll=\"false\"])', link => {\r\n\t\t\t\t\tif (Scroll.visibleAll(link, 50)) {\r\n\t\t\t\t\t\t_.s(`var[data-ytmsid=\"${link.dataset.ytmsid}\"]:not([data-ytmscroll=\"false\"])`, trigger => {\r\n\t\t\t\t\t\t\tconst ui = Control.createFromTrigger(trigger);\r\n\t\t\t\t\t\t\tui.showOnScroll(link);\r\n\t\t\t\t\t\t});\r\n\t\t\t\t\t}\r\n\t\t\t\t});\r\n\t\t\t},\r\n\t\t\ttoggleMedia: function () {\r\n\t\t\t\treturn new Scroll('div.ytm_panel_switcher', div => {\r\n\t\t\t\t\tconst v = div.querySelector('video');\r\n\t\t\t\t\tconst paused = v && (v.paused || v.ended);\r\n\t\t\t\t\tconst ui = Y.set[div.dataset.ytmuid].getControl();\r\n\r\n\t\t\t\t\tif (paused && !Scroll.visibleAll(div, 0)) {\r\n\t\t\t\t\t\treturn ui.play.switchStandby();\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t\tif (ui.play.isStandby() && Scroll.visibleAll(div, 200)) {\r\n\t\t\t\t\t\treturn ui.play.switchOn();\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t\t// todo ascertain embedded player properties\r\n\t\t\t\t\t// f = div.querySelector('iframe, object');\r\n\t\t\t\t\t// if (f && !YTMA.Scroll.visibleAll(div, 200)) {\r\n\t\t\t\t\t// \ty.hidePlayer();\r\n\t\t\t\t\t// }\r\n\t\t\t\t});\r\n\t\t\t},\r\n\t\t\tonScrollViewMedia: function () {\r\n\t\t\t\tthis.showMedia();\r\n\t\t\t\tthis.toggleMedia();\r\n\t\t\t},\r\n\t\t\tonScrollLoadDescriptions: function (ajax) {\r\n\t\t\t\tif (Y.user.fn.$scroller) { Y.user.fn.$scroller.stop(); }\r\n\r\n\t\t\t\tY.user.fn.$scroller = new Scroll('span.ytm_manual > a.ytm_title:not(.ytm_error)', a => {\r\n\t\t\t\t\tif (Scroll.visibleAll(a, 200)) {\r\n\t\t\t\t\t\tif (ajax) {\r\n\t\t\t\t\t\t\tY.ajax.loadFromDataset(a.dataset);\r\n\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\tY.ajax.loadFromCacheDataset(a.dataset);\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t\t// console.log('doc', document.querySelectorAll(YTMA.user.fn.$scroller.selector).length, a.dataset.id);\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t\tif (document.querySelectorAll(Y.user.fn.$scroller.selector).length === 0) {\r\n\t\t\t\t\t\tY.user.fn.$scroller.stop();\r\n\t\t\t\t\t}\r\n\t\t\t\t});\r\n\t\t\t},\r\n\t\t\tmakeForm: function () {\r\n\t\t\t\tconst template = `\r\n\t\t\t\t\t<div id=\"ytm_settings\" class=\"ytm_sans ytm_block ytm_normalize\">\r\n\t\t\t\t\t\t<form action=\"\" title=\"Double click to close\">\r\n\t\t\t\t\t\t\t<div id=\"ytm_settingst\">ytma! Site Settings</div><div class=\"ytm_field_container\">\r\n\t\t\t\t\t\t\t\t<fieldset><legend title=\"Load descriptions from the content sever.\">Load Descriptions</legend><p><span><input id=\"ytma__desc0\" type=\"radio\" data-num=\"0\" name=\"ytma__desc\" data-key=\"desc\"><label for=\"ytma__desc0\" title=\"Load descriptions on demand\">Manually</label></span><span><input id=\"ytma__desc1\" type=\"radio\" data-num=\"1\" name=\"ytma__desc\" data-key=\"desc\"><label for=\"ytma__desc1\" title=\"Load descriptions as they become visible on the screen.\">Automatically, on scrolling</label></span></p></fieldset>\r\n\t\t\t\t\t\t\t\t<fieldset><legend>HTML5 Players</legend><p><input name=\"ytma__autoShow\" data-key=\"autoShow\" id=\"ytma__autoShow\" type=\"checkbox\"><label for=\"ytma__autoShow\">Automatically show WebM, MP4 and Soundcloud players</label></p></fieldset>\r\n\t\t\t\t\t\t\t\t<fieldset><legend>Player Size</legend><p><span><input type=\"radio\" name=\"ytma__size\" data-key=\"size\" data-num=\"240\" id=\"ytma__size240\" /><label for=\"ytma__size240\">S <small>240p</small></label></span><span><input name=\"ytma__size\" data-key=\"size\" type=\"radio\" id=\"ytma__size360\" data-num=\"360\" /><label for=\"ytma__size360\">M <small>360p</small></label></span><span><input type=\"radio\" name=\"ytma__size\" data-key=\"size\" data-num=\"480\" id=\"ytma__size480\" /><label for=\"ytma__size480\">L <small>480p</small></label></span><span><input type=\"radio\" name=\"ytma__size\" data-key=\"size\" data-num=\"720\" id=\"ytma__size720\" /><label for=\"ytma__size720\">X <small>720p</small></label></span></p></fieldset>\r\n\t\t\t\t\t\t\t\t<fieldset><legend>Quality</legend><p><span><input name=\"ytma__quality\" data-key=\"quality\" data-num=\"240\" id=\"ytma__quality240\" type=\"radio\"><label for=\"ytma__quality240\">240p</label></span><span><input name=\"ytma__quality\" data-key=\"quality\" id=\"ytma__quality360\" data-num=\"360\" type=\"radio\"><label for=\"ytma__quality360\">360p</label></span><span><input name=\"ytma__quality\" data-key=\"quality\" data-num=\"480\" id=\"ytma__quality480\" type=\"radio\"><label for=\"ytma__quality480\">480p</label></span><span><input name=\"ytma__quality\" data-key=\"quality\" data-num=\"720\" id=\"ytma__quality720\" type=\"radio\"><label for=\"ytma__quality720\">720p</label></span><span><input name=\"ytma__quality\" data-key=\"quality\" data-num=\"1080\" id=\"ytma__quality1080\" type=\"radio\"><label for=\"ytma__quality1080\">1080p</label></span></p></fieldset>\r\n\t\t\t\t\t\t\t\t<fieldset><legend>Aspect Ratio</legend><p><span><input name=\"ytma__ratio\" data-key=\"ratio\" type=\"radio\" id=\"ytma__ratio2\" data-num=\"2\" /><label for=\"ytma__ratio2\">16:9</label></span><span><input type=\"radio\" name=\"ytma__ratio\" data-key=\"ratio\" data-num=\"1\" id=\"ytma__ratio1\" /><label for=\"ytma__ratio1\">4:3</label></span></p></fieldset>\r\n\t\t\t\t\t\t\t\t<fieldset><legend>YouTube</legend>\r\n\t\t\t\t\t\t\t\t\t<p><input name=\"ytma__yt_annotation\" data-key=\"yt_annotation\" type=\"checkbox\" id=\"ytma__yt_annotation\" /><label for=\"ytma__yt_annotation\">Enable video annotations</label></p>\r\n\t\t\t\t\t\t\t\t\t<p><input name=\"ytma__yt_nocookie\" data-key=\"yt_nocookie\" type=\"checkbox\" id=\"ytma__yt_nocookie\" /><label for=\"ytma__yt_nocookie\">Use https://youtube-nocookie.com to load videos</label></p>\r\n\t\t\t\t\t\t\t\t</fieldset>\r\n\t\t\t\t\t\t\t\t<fieldset><legend>Window Focus</legend><p><input name=\"ytma__focus\" data-key=\"focus\" type=\"checkbox\" id=\"ytma__focus\" value=\"focus\" /><label for=\"ytma__focus\">After clicking the thumbnail, set the video at the top of the window.</label></p></fieldset>\r\n\t\t\t\t\t\t\t</div>\r\n\t\t\t\t\t\t\t<p><small id=\"ytm_settings_error\" class=\"ytm_error ytm_none ytm_title\">Error! Your settings could not be saved.</small></p>\r\n\t\t\t\t\t\t\t<p id=\"ytm_opts\">\r\n\t\t\t\t\t\t\t\t<button type=\"button\" id=\"ytmaclose\">Close</button> <button type=\"button\" id=\"ytmareset\">Reset</button> <button type=\"button\" id=\"ytmaclear\" title=\"Remove all video descriptions that have been cached\">Reset & Remove Cache</button>\r\n\t\t\t\t\t\t\t</p>\r\n\t\t\t\t\t\t</form>\r\n\t\t\t\t\t</div>`;\r\n\r\n\t\t\t\tY.user.$form = _.e('div', { className: 'ytm_fix_center ytm_none ytm_box', innerHTML: template }, document.body);\r\n\t\t\t\tY.user.error = document.getElementById('ytm_settings_error');\r\n\r\n\t\t\t\ton(Y.user.$form, 'keyup click', 'input, label', debounce(Y.user.events.save, 500));\r\n\t\t\t\tY.user.$form.addEventListener('submit', e => e.preventDefault(), false);\r\n\r\n\t\t\t\tdocument.getElementById('ytmareset').addEventListener('click', Y.user.events.reset, false);\r\n\t\t\t\tdocument.getElementById('ytmaclear').addEventListener('click', Y.user.events.clear, false);\r\n\r\n\t\t\t\t// close\r\n\t\t\t\tY.user.$form.addEventListener('dblclick', Y.user.events.formToggle, false);\r\n\t\t\t\tdocument.getElementById('ytmaclose').addEventListener('click', Y.user.events.formToggle, false);\r\n\t\t\t\tdocument.body.addEventListener('keydown', Y.user.events.formToggleKeyboard, false);\r\n\t\t\t}\r\n\t\t}\r\n\t};\r\n\r\n\tY.css = () => {\r\n\t\tconst playerCss = Player.css.generator();\r\n\t\tconst loadingIcon = 'data:image/gif;base64,R0lGODlhDgAKAJEAAP///+BKV////wAAACH/C05FVFNDQVBFMi4wAwEAAAAh+QQFCgACACwAAAAADgAKAAACHFSOeQYI71p6MtAJz41162yBH+do5Ih1kKG0QgEAIfkEBQoAAgAsAAABAA0ACAAAAhSUYGEoerkgdIzKGlu2ET/9ceJmFAAh+QQFCgACACwAAAEADQAIAAACFJRhcbmiglx78SXKYK6za+NxHyYVACH5BAUKAAIALAAAAQANAAgAAAIWVCSAl+hqEGRTLhtbdvTqnlUf9nhTAQAh+QQFCgACACwAAAEADQAIAAACFZRiYCh6uaCRzNXYsKVT+5eBW3gJBQAh+QQJCgACACwAAAAADgAKAAACGpSPaWGwfZhwQtIK8VTUvuxpm9Yp4XlmpiIUADs=';\r\n\r\n\t\t// console.log(playerCss);\r\n\t\t_.css(playerCss);\r\n\r\n\t\t// images\r\n\t\t// todo update(site, size, padding)\r\n\t\t_.css(`\r\n\t\t\t.ytm_loading{background:url(${loadingIcon}) 0 3px no-repeat;}\r\n\t\t\t.ytm_link{position:relative !important;background:url(${Y.DB.sites.youtube.favicon}) 0 center no-repeat !important;margin-left:4px;padding-left:20px!important;}\r\n\t\t\t.ytm_link.ytm_link_vimeo{background-image:url(${Y.DB.sites.vimeo.favicon}) !important;background-size:12px 12px !important;padding-left:18px!important}\r\n\t\t\t.ytm_link.ytm_link_vine{background-image:url(${Y.DB.sites.vine.favicon}) !important;background-size:10px 10px!important;padding-left:16px!important}\r\n\t\t\t.ytm_link.ytm_link_soundcloud{background-image:url(${Y.DB.sites.soundcloud.favicon})!important;padding-left:17px!important}\r\n\t\t\t.ytm_link.ytm_link_html5{background-image:url(${Y.DB.sites.html5.favicon}) !important;padding-left:16px!important}\r\n\t\t\t.ytm_link.ytm_link_gfycat{background-image:url(${Y.DB.sites.gfycat.favicon}) !important;background-size:12px 12px !important;padding-left:16px!important;}\r\n\t\t\t.ytm_link.ytm_link_imgur{background-image:url(${Y.DB.sites.imgur.favicon}) !important;background-size:12px 12px !important;padding-left:16px!important}\r\n\t\t\t.ytm_link.ytm_link_streamable{background-image:url(${Y.DB.sites.streamable.favicon}) !important; background-size: 12px 12px !important;padding-left: 14px !important;}\r\n\t\t`);\r\n\r\n\t\t_.css('.ytm_none,.ytm_link br{display:none!important}.ytm_box{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}.ytm_block{display:block;position:relative;clear:both;text-align:left;border:0;margin:0;padding:0;overflow:hidden}.ytm_normalize{font-weight:400!important;font-style:normal!important;line-height:1.2!important}.ytm_sans{font-family:Arial,Helvetica,sans-serif!important}.ytm_spacer{overflow:auto;margin:0 0 6px;padding:4px}.ytm_spacer.ytm_site_slim{display:inline}.ytm_clear:after{content:\"\";display:table;clear:both}.ytm_center{text-align:center}.ytm_link b,.ytm_link strong{font-weight:400!important}.ytm_link u{text-decoration:none!important}.ytm_link i,.ytm_link em{font-style:normal!important}.ytm_trigger{width:118px;height:66px;background-color:#262626!important;cursor:pointer;background-position:-1px -12px;float:left;box-shadow:2px 2px rgba(0,0,0,.3);background-size:auto 90px!important;color:#fff;text-shadow:#333 0 0 2px;font-size:13px}.ytm_trigger:hover{box-shadow:2px 2px #60656b80;opacity:.95}.ytm_trigger var{z-index:2;height:100%;width:100%;position:absolute;left:0;top:0;text-align:right}.ytm_label{display:block;padding:3px 6px;line-height:1.2;font-style:normal}.ytm_init{height:22px;background:rgba(11,11,11,.62);padding:4px 25px 6px 6px}.ytm_site_vine .ytm_trigger{background-color:#90ee90!important;background-size:120px auto!important}.ytm_site_slim .ytm_trigger{background:#e34c26!important;height:auto;box-shadow:0 0 2px #ffdb9d inset,2px 2px rgba(0,0,0,.3);margin:0 3px 0 0;width:auto;transition:all .3s ease-in-out 0s}.ytm_site_slim .ytm_trigger:hover{opacity:.8}.ytm_site_slim .ytm_label{text-shadow:0 0 1px #f06529}.ytm_site_slim .ytm_init{background:transparent}.ytm_bd{float:left;max-width:450px;margin:2px 10px;font-size:12px}.ytm_title{font-weight:700}.ytm_error{color:#cc2f24;font-style:italic}.ytm_loading{font-style:italic;padding:1px 1.5em}.ytm_descr{word-wrap:break-word;max-height:48px;overflow:auto;padding-right:20px}.ytm_descr[data-full]{cursor:pointer}.ytm_descr_open{resize:both;white-space:pre-line;background:linear-gradient(to bottom, rgba(0,0,0,0) 0%,rgba(0,0,0,0) 50%,rgba(0,0,0,0) 80%,rgba(0,0,0,0.1) 100%)}.ytm_descr_open[style]{max-height:none}.ytm_projector{margin-bottom:4px}ul.ytm_options{overflow:hidden;margin:0!important;padding:3px 0 1px;list-style-position:outside!important}.ytm_options li{display:inline;margin:0!important;padding:0!important}.ytm_options li>ul{display:inline-block;margin:0;padding:0 1px 0 0}.ytm_options li ul li{-webkit-user-select:none;-moz-user-select:none;-o-user-select:none;user-select:none;list-style-type:none;cursor:pointer;float:left;color:#858585;border:1px solid #1d1d1d;border-bottom:1px solid #181818;border-top:1px solid #292929;box-shadow:0 0 1px #555;height:14px;font-size:12px!important;line-height:12px!important;background:#222;background:linear-gradient(#2d2c2c,#222);margin:0!important;padding:5px 9px 3px!important}.ytm_options li ul li:first-child{border-radius:2px 0 0 2px}.ytm_options li ul li:last-child{border-left:0!important;border-radius:0 2px 2px 0;margin:0 2px 0 0!important}.ytm_options li ul li:first-child:last-child,.ytm_li_setting{border-radius:2px}.ytm_options li ul li:hover{color:#ccc;text-shadow:1px 1px 0 #333;background:#181818}.ytm_options li ul li[id]{color:#ddd;text-shadow:0 0 2px #444}.ytm_panel_size{background:#000;max-width:100%;}.ytm_panel_switcher[data-standby=\"true\"]{background:#111}.ytm_panel_switcher[data-standby=\"true\"]:after{cursor:cell;color:#0e0e0e;content:\"ytma!\";display:block;font-size:85px;font-style:italic;font-weight:700;left:50%;position:absolute;text-shadow:2px 1px #181818,-1px -1px #0a0a0a;top:50%;transform:translate(-50%,-50%)}.ytm_site_soundcloud .ytm_panel_size.ytm_soundcloud-playlist{height:334px!important}.ytm_fix_center{background:rgba(51,51,51,.41);height:100%;left:0;position:fixed;top:0;width:100%;z-index:99998}#ytm_settings{z-index:99999;max-width:500px;max-height:85%;overflow:auto;background:#fbfbfb;border:1px solid #bbb;color:#444;box-shadow:0 0 5px rgba(0,0,0,.2),0 0 3px rgba(239,239,239,.1) inset;margin:4% auto;padding:4px 8px 0}#ytm_settings p{margin:5px 0;padding:0}#ytm_settings fieldset{vertical-align:top;border-radius:3px;border:1px solid #ccc;margin:0 0 5px;padding:3px}#ytm_settings legend{padding:3px}#ytm_settings fieldset span{display:inline-block;min-width:5em}#ytm_settings input{vertical-align:baseline!important;margin:3px 5px!important}#ytm_settingst{font-size:110%;border-bottom:1px solid #d00;margin:3px 0 9px;padding:0 3px 3px}#ytm_settings label{cursor:pointer}#ytm_settings small{font-size:90%}#ytm_opts button{cursor:pointer;margin:10px 5px 8px 2px;padding:3px;border:1px solid #adadad;border-radius:2px;background:#eee;font-size:90%}#ytm_opts button:hover{background:#ddd}');\r\n\t};\r\n\r\n\tY.ajax = {\r\n\t\tload: function (site, id, uri) {\r\n\t\t\tconsole.info('ytma//ajax+load(id)', site, id, uri);\r\n\t\t\turi = Y.DB.sites[site].ajax.replace('%key', id).replace('%uri', uri);\r\n\r\n\t\t\tif (Y.DB.sites[site].ajaxExtension) { return this.gmxhr(uri, site, id); }\r\n\r\n\t\t\tconsole.info('ytma//ajax+load(uri)', Y.DB.sites[site].ajax.replace('%key', id).replace('%uri', uri));\r\n\t\t\tif (Y.DB.sites[site].ajax) {\r\n\t\t\t\t// console.log('preping uri');\r\n\t\t\t\treturn this.xhr(uri, site, id);\r\n\t\t\t}\r\n\r\n\t\t\treturn null;\r\n\t\t},\r\n\t\tloadFromDataset: function (dataset) {\r\n\t\t\tif (!this.loadFromCacheDataset(dataset)) {\r\n\t\t\t\treturn this.load(dataset.ytmsite, dataset.ytmid, dataset.ytmuri);\r\n\t\t\t}\r\n\t\t},\r\n\t\tloadFromCacheDataset: function ({ ytmsite, ytmid }) {\r\n\t\t\tconst cache = Y.external.dataFromStorage(ytmsite, ytmid);\r\n\r\n\t\t\tconsole.info('ytma//ajax+cache(id)', ytmsite, ytmid);\r\n\t\t\tconsole.info('ytma//ajax+cache(data)', cache);\r\n\r\n\t\t\tif (cache) { Y.external.populate(cache); }\r\n\r\n\t\t\treturn cache;\r\n\t\t},\r\n\t\tgmxhr: function (uri, site, id) {\r\n\t\t\ttry {\r\n\t\t\t\t// console.log('gmxhr starting!');\r\n\t\t\t\tGM.xmlhttpRequest({\r\n\t\t\t\t\tmethod: 'GET',\r\n\t\t\t\t\turl: uri,\r\n\t\t\t\t\tonload: function ({ responseText }) {\r\n\t\t\t\t\t\t// console.log(response);\r\n\t\t\t\t\t\tY.external.parse(responseText, site, id);\r\n\t\t\t\t\t},\r\n\t\t\t\t\tonerror: function () {\r\n\t\t\t\t\t\tconsole.log('GM Cannot XHR');\r\n\t\t\t\t\t\tY.ajax.failure.call({ id });\r\n\t\t\t\t\t}\r\n\t\t\t\t});\r\n\r\n\t\t\t\tY.ajax.preProcess(id);\r\n\r\n\t\t\t} catch (e) {\r\n\t\t\t\tif (Y.DB.extension) {\r\n\t\t\t\t\tconsole.info('ytma//gmxhr-cors');\r\n\t\t\t\t\tthis.xhr(uri, site, id);\r\n\t\t\t\t} else {\r\n\t\t\t\t\tconsole.log('No applicable CORS request available.');\r\n\t\t\t\t\tthis.failure.call({ id });\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t},\r\n\t\txhr: function (uri, site, id) {\r\n\t\t\tconst x = new XMLHttpRequest();\r\n\t\t\tconsole.info('ytma//xhr', uri, id, site);\r\n\r\n\t\t\tY.ajax.preProcess(id);\r\n\r\n\t\t\tx.onreadystatechange = function () {\r\n\t\t\t\tif (this.readyState === this.DONE) {\r\n\t\t\t\t\t// console.log(this.readyState, this.status);\r\n\t\t\t\t\tif (this.status === 200) {\r\n\t\t\t\t\t\tY.external.parse(this.responseText, site, id);\r\n\t\t\t\t\t} else if (this.status === 403) {\r\n\t\t\t\t\t\tY.external.populate({ site, id, title: 'Error 403', desc: '' });\r\n\t\t\t\t\t\tY.external.save({ site, id, title: 'Error 403', desc: '' });\r\n\t\t\t\t\t} else { // if (this.status >= 400 || this.status === 0) {\r\n\t\t\t\t\t\tY.ajax.failure.call({ id });\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t};\r\n\r\n\t\t\ttry {\r\n\t\t\t\t// console.info('ytma//xhr+sending');\r\n\t\t\t\tx.open('get', uri, true);\r\n\t\t\t\tx.send();\r\n\t\t\t} catch (e) {\r\n\t\t\t\tconsole.error('ytma//xhr+failed(cannot send xhr)', uri);\r\n\t\t\t\tY.ajax.failure.call({ id });\r\n\t\t\t\tconsole.error(e);\r\n\t\t\t}\r\n\t\t},\r\n\t\tfailure: function () {\r\n\t\t\t_.s(`.ytm_bd._${Y.escapeId(this.id)}`, el => {\r\n\t\t\t\tconst a = el.querySelector('a');\r\n\t\t\t\ta.dataset.tries = a.dataset.tries ? parseFloat(a.dataset.tries) + 1 : 1;\r\n\t\t\t\tif (a.dataset.tries >= 5) {\r\n\t\t\t\t\ta.textContent = 'Max attempts reached';\r\n\t\t\t\t} else {\r\n\t\t\t\t\ta.textContent = `Error, unable to load data.${a.dataset.tries > 1 ? '' : ' [Retry]'}`;\r\n\t\t\t\t}\r\n\t\t\t\ta.className = 'ytm_error ytm_title';\r\n\t\t\t});\r\n\t\t},\r\n\t\tpreProcess: function (id) {\r\n\t\t\t_.s(`.ytm_manual._${Y.escapeId(id)} a`, el => {\r\n\t\t\t\tel.classList.add('ytm_loading');\r\n\t\t\t\tel.textContent = 'Loading';\r\n\t\t\t\tel.title = 'Retry loading data.';\r\n\t\t\t});\r\n\t\t}\r\n\t};\r\n\r\n\t/** E X T E R N A L Apparatus\r\n\t * Data from external sites\r\n\t */\r\n\tY.external = {\r\n\t\tversion: 'ytma.4.1.dat',\r\n\t\tparse: function (response, site, id) {\r\n\t\t\tif (this.parsers[site]) {\r\n\t\t\t\tresponse = Y.DB.sites[site].rawResponse ? response : JSON.parse(response);\r\n\t\t\t\tthis.populate(this.helper.cutDescription(this.parsers[site](response, id)));\r\n\t\t\t}\r\n\t\t},\r\n\t\tparsers: {\r\n\t\t\tsoundcloud: function ({ title, description, thumbnail_url }, id) {\r\n\t\t\t\treturn {\r\n\t\t\t\t\tsite: 'soundcloud',\r\n\t\t\t\t\tid, //unescape(j.html).match(/tracks\\/(\\d+)/)[1],\r\n\t\t\t\t\ttitle,\r\n\t\t\t\t\tdesc: description,\r\n\t\t\t\t\tth: removeSearch(thumbnail_url)\r\n\t\t\t\t};\r\n\t\t\t},\r\n\t\t\tvimeo: function (j) {\r\n\t\t\t\tj = j[0];\r\n\t\t\t\treturn {\r\n\t\t\t\t\tsite: 'vimeo',\r\n\t\t\t\t\tid: j.id,\r\n\t\t\t\t\ttitle: `${j.title} ${Y.external.time.fromSeconds(j.duration)}`,\r\n\t\t\t\t\tdesc: j.description.replace(/<br.?.?>/g, ''),\r\n\t\t\t\t\tth: decodeURI(j.thumbnail_medium)\r\n\t\t\t\t};\r\n\t\t\t},\r\n\t\t\tyoutube: function (j, id) {\r\n\t\t\t\tif (j.pageInfo.totalResults < 1 || j.items.length === 0) {\r\n\t\t\t\t\treturn { id, error: true };\r\n\t\t\t\t}\r\n\r\n\t\t\t\tj = j.items[0];\r\n\t\t\t\tconst o = {\r\n\t\t\t\t\tsite: 'youtube',\r\n\t\t\t\t\tid,\r\n\t\t\t\t\ttitle: `${j.snippet.title} ${Y.external.time.fromIso8601(j.contentDetails.duration)}`,\r\n\t\t\t\t\tdesc: j.snippet.description\r\n\t\t\t\t\t// aspectRatio: j.contentDetails.aspectRatio\r\n\t\t\t\t};\r\n\r\n\t\t\t\treturn o;\r\n\t\t\t},\r\n\t\t\tvine: function ({ title, thumbnail_url }, id) {\r\n\t\t\t\treturn {\r\n\t\t\t\t\tsite: 'vine',\r\n\t\t\t\t\tid,\r\n\t\t\t\t\ttitle,\r\n\t\t\t\t\tth: removeSearch(thumbnail_url)\r\n\t\t\t\t};\r\n\t\t\t},\r\n\t\t\tgfycat: function (html, id) {\r\n\t\t\t\tif (html) {\r\n\t\t\t\t\treturn {\r\n\t\t\t\t\t\tsite: 'gfycat',\r\n\t\t\t\t\t\tid: id,\r\n\t\t\t\t\t\ttitle: id\r\n\t\t\t\t\t};\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tstreamable: function ({ title }, id) {\r\n\t\t\t\treturn {\r\n\t\t\t\t\tsite: 'streamable',\r\n\t\t\t\t\tid,\r\n\t\t\t\t\ttitle: title || 'Untitled'\r\n\t\t\t\t};\r\n\t\t\t}\r\n\t\t},\r\n\t\tset: function (data) {\r\n\t\t\tif (!this.db[data.site]) {\r\n\t\t\t\tthis.db[data.site] = {};\r\n\t\t\t}\r\n\t\t\tthis.db[data.site][data.id] = data;\r\n\t\t\treturn this.save();\r\n\t\t},\r\n\t\tunset: function ({ site, id }) {\r\n\t\t\t// console.log('unset', data.id);\r\n\t\t\tif (site) {\r\n\t\t\t\tdelete this.db[site][id];\r\n\t\t\t\treturn this.save();\r\n\t\t\t}\r\n\t\t},\r\n\t\tlimitDB: function (max, db) {\r\n\t\t\t// limits an object's items by half of the max\r\n\t\t\t// removes the older items at the start of the object\r\n\t\t\tconst keys = Object.keys(db);\r\n\r\n\t\t\tconst half = Math.floor(max / 2);\r\n\t\t\tlet start;\r\n\t\t\tlet ndb;\r\n\t\t\tlet i;\r\n\r\n\t\t\tif (keys.length > max) {\r\n\t\t\t\tndb = {};\r\n\t\t\t\tstart = keys.length - half;\r\n\r\n\t\t\t\tfor (i = start; i < keys.length; i++) {\r\n\t\t\t\t\tndb[keys[i]] = db[keys[i]];\r\n\t\t\t\t}\r\n\t\t\t}\r\n\r\n\t\t\treturn ndb || db;\r\n\t\t},\r\n\t\tsave: function () {\r\n\t\t\tthis.db = this.limitDB(1000, this.db);\r\n\t\t\treturn strg.save(this.version, this.db);\r\n\t\t},\r\n\t\thelper: {\r\n\t\t\tcutDescription: function (data) {\r\n\t\t\t\tif (data.desc && data.desc.length > 140) {\r\n\t\t\t\t\tdata.full = data.desc;\r\n\t\t\t\t\tdata.desc = `${data.desc.substr(0, 130)} . . .`;\r\n\t\t\t\t}\r\n\t\t\t\treturn data;\r\n\t\t\t},\r\n\t\t\tthumbnail: function ({ id, th }) {\r\n\t\t\t\t_.s(`[data-ytmid=\"${id}\"].ytm_trigger`, el => el.setAttribute('style', `background: url(${th})`));\r\n\t\t\t}\r\n\t\t},\r\n\t\ttime: {\r\n\t\t\tkeepMinutesAndSeconds: function (v, i) {\r\n\t\t\t\treturn i > 1 || v > 0;\r\n\t\t\t},\r\n\t\t\tleadingZero: function (v, i) {\r\n\t\t\t\treturn i > 0 ? (`00${v}`).slice(-2) : v;\r\n\t\t\t},\r\n\t\t\tfromArray: function (a) {\r\n\t\t\t\t// [days, hours, mins, secs]\r\n\t\t\t\tlet b;\r\n\r\n\t\t\t\tlet p = '';\r\n\r\n\t\t\t\ttry {\r\n\t\t\t\t\t// Remove empty values, but keep lower indexes (m:s); a[i] > 0 || i > 1\r\n\t\t\t\t\t// Add leading 0's, ignoring the first index\r\n\t\t\t\t\t// a.slice(0, 1).concat(a.slice(1))\r\n\t\t\t\t\tb = a.filter(this.keepMinutesAndSeconds).map(this.leadingZero);\r\n\t\t\t\t\tp = `(${b.join(':')})`;\r\n\t\t\t\t} catch (e) {\r\n\t\t\t\t\tconsole.error('Could not parse this time.');\r\n\t\t\t\t}\r\n\r\n\t\t\t\tconsole.info('ytma//time+array', { a, b, p });\r\n\t\t\t\treturn p;\r\n\t\t\t},\r\n\t\t\tfromIso8601: function (iso8601) {\r\n\t\t\t\t// eg PT3M, T29S\r\n\t\t\t\tlet a;\r\n\r\n\t\t\t\tconst parseDigits = reg => {\r\n\t\t\t\t\tif (reg.test(iso8601)) {\r\n\t\t\t\t\t\treturn RegExp.lastParen;\r\n\t\t\t\t\t}\r\n\t\t\t\t\treturn 0;\r\n\t\t\t\t};\r\n\r\n\t\t\t\t// P#DT#H#M#S || PT#H#M#S\r\n\t\t\t\ta = [/(\\d+)D/, /(\\d+)H/, /(\\d+)M/, /(\\d+)S/].map(parseDigits);\r\n\r\n\t\t\t\treturn this.fromArray(a);\r\n\t\t\t},\r\n\t\t\tfromSeconds: function (seconds) {\r\n\t\t\t\tconst a = [\r\n\t\t\t\t\tMath.floor(seconds / 86400) % 24,\r\n\t\t\t\t\tMath.floor(seconds / 3600) % 60,\r\n\t\t\t\t\tMath.floor(seconds / 60) % 60,\r\n\t\t\t\t\tseconds % 60\r\n\t\t\t\t];\r\n\t\t\t\treturn this.fromArray(a);\r\n\t\t\t}\r\n\t\t},\r\n\t\tvalidate: function (data) {\r\n\t\t\tif (!data || !data.id || data.error) {\r\n\t\t\t\treturn Y.ajax.failure.call(data);\r\n\t\t\t}\r\n\r\n\t\t\t// todo? empty titles and descriptions should be okay\r\n\t\t\t// if (data.id && !data.title && !data.desc) {\r\n\t\t\t// \tthis.unset(data.id);\r\n\t\t\t// \treturn YTMA.ajax.failure.call(data);\r\n\t\t\t// }\r\n\r\n\t\t\treturn true;\r\n\t\t},\r\n\t\tpopulate: function (data, ignoreValidation) {\r\n\t\t\tif (!ignoreValidation && !this.validate(data)) { return; }\r\n\r\n\t\t\tthis.set(data);\r\n\t\t\tconst { id, th, full, desc, title } = data;\r\n\r\n\t\t\tif (th) { this.helper.thumbnail(data); }\r\n\r\n\t\t\t_.s(`.ytm_bd._${Y.escapeId(id)}`, el => {\r\n\t\t\t\tel.innerHTML = `<span class=\"ytm_title\">${title}</span>`;\r\n\t\t\t\tif (desc) {\r\n\t\t\t\t\tconst q = _.e('q', { className: 'ytm_descr ytm_block', textContent: desc }, el);\r\n\t\t\t\t\tif (full.length > desc.length) {\r\n\t\t\t\t\t\tq.dataset.full = full;\r\n\t\t\t\t\t\tq.title = 'Double click to toggle the description.';\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t});\r\n\t\t},\r\n\t\tdataFromStorage: function (site, id) {\r\n\t\t\tif (this.db && this.db[site]) {\r\n\t\t\t\treturn this.db[site][id];\r\n\t\t\t}\r\n\t\t}\r\n\t};\r\n\tY.external.db = strg.grab(Y.external.version, {});\r\n\r\n\t/** Database */\r\n\tY.DB = {\r\n\t\tpostInit: function () {\r\n\t\t\tif (Y.user.preferences.yt_nocookie) {\r\n\t\t\t\tY.DB.sites.youtube.home = 'https://www.youtube-nocookie.com/';\r\n\t\t\t\tY.DB.sites.youtube.embed = 'https://www.youtube-nocookie.com/embed/%key';\r\n\t\t\t}\r\n\t\t},\r\n\t\textension: window.chrome && window.chrome.extension,\r\n\t\tsites: { // supported sites - to add more also make a parser (if api is available) and add an item to sources (if necessary)\r\n\t\t\tyoutube: {\r\n\t\t\t\ttitle: 'ytma!',\r\n\t\t\t\thome: 'https://www.youtube.com/',\r\n\t\t\t\tembed: 'https://www.youtube.com/embed/%key',\r\n\t\t\t\tajax: `https://www.googleapis.com/youtube/v3/videos?part=snippet,contentDetails&id=%key${window.atob('JmtleT1BSXphU3lEVG5INkxzRERyVElYaFZTZWRQQjlyRHo1czBSczQzZnM=')}`,\r\n\t\t\t\tthumb: 'url(https://i3.ytimg.com/vi/%key/1.jpg)',\r\n\t\t\t\tselector: 'a[href*=\"youtube.\"], a[href*=\"youtu.be/\"]',\r\n\t\t\t\tfavicon: 'https://www.youtube.com/favicon.ico',\r\n\t\t\t\tkey: 'id',\r\n\t\t\t\treg: '(youtu)',\r\n\t\t\t\tmatcher: /(?:(?:(?:v=|#p\\/u\\/\\d*?\\/)|(?:v=|#p\\/c\\/[a-zA-Z0-9]+\\/\\d*?\\/)|(?:embed\\/)|(?:v\\/)|(?:\\.be\\/))([A-Za-z0-9-_]{11}))/i,\r\n\t\t\t\thttps: true\r\n\t\t\t},\r\n\t\t\tvimeo: {\r\n\t\t\t\ttitle: 'vimeo too!',\r\n\t\t\t\thome: 'https://vimeo.com/',\r\n\t\t\t\tembed: 'https://player.vimeo.com/video/%key?badge=0',\r\n\t\t\t\tajax: 'https://vimeo.com/api/v2/video/%key.json',\r\n\t\t\t\tselector: 'a[href*=\"vimeo.com/\"]',\r\n\t\t\t\tfavicon: 'https://f.vimeocdn.com/images_v6/favicon.ico',\r\n\t\t\t\tkey: 'id',\r\n\t\t\t\treg: '(vimeo)',\r\n\t\t\t\tmatcher: /(?:vimeo\\.com\\/(\\d+))/i,\r\n\t\t\t\thttps: true\r\n\t\t\t},\r\n\t\t\tvine: {\r\n\t\t\t\ttitle: 'vine me!',\r\n\t\t\t\thome: 'https://vine.co/',\r\n\t\t\t\tembed: 'https://vine.co/v/%key/embed/simple?audio=1',\r\n\t\t\t\tajaxExtension: true,\r\n\t\t\t\tajax: 'https://vine.co/oembed.json?url=https%3A%2F%2Fvine.co%2Fv%2F%key',\r\n\t\t\t\tselector: 'a[href*=\"vine.co/\"]',\r\n\t\t\t\tfavicon: 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQCAYAAAAf8/9hAAABcklEQVQ4jX2SvyvFYRTGP+9NdzAYbpIkysA/ICnKoEwvA4vlUkoyGQzvoOiK4R0YbDLhDhbi9maWIjEqcScWkyRhMNzXcM/3On39OMt5nvOe85wfvSbGyH9mgs8Cq8AUkAMegU1gJVpXqfu3umpbwITiLUBB8HJGuryb4KMJ/izVvUMVnwNjwJPwGYCMkLL4zlT3XoUL0boD4FBNUhO4Fd9ogm9SRVmFn8Una79qgWuV2K3wvcLJdG3iL7TaiUocAI4Fn0vnHDBrgt+L1g2q3NoEl8CH4KHkMVr3CSwI7QeOTPA9WsAk/8AEvw+MSrwrWleuJQVfABZJWbTOZBTfVXg6lbgEHKjQTcL1BFngAWimeuH2aN2LvHUDV1JcjNbl0zdI9l0T2pDsboKvB7Yl/ga4X2+gku+AVqAC9AFzwLik5KN1xT8FRGQEOBL6yfdn2onWTZKyTDoQrSsB60KT4lNSh/1TQETmgQ1ZowQMy41+2BeLRXeRaKuHSAAAAABJRU5ErkJggg==',\r\n\t\t\t\tkey: 'id',\r\n\t\t\t\treg: '(vine)',\r\n\t\t\t\tmatcher: /(?:vine\\.co\\/v\\/([A-Za-z0-9-_]{11}))/i\r\n\t\t\t},\r\n\t\t\tsoundcloud: {\r\n\t\t\t\ttitle: 'sound off!',\r\n\t\t\t\thome: 'https://soundcloud.com/',\r\n\t\t\t\tembed: 'https://w.soundcloud.com/player/?show_comments=false&url=%key',\r\n\t\t\t\tajax: 'https://soundcloud.com/oembed?format=json&url=%uri',\r\n\t\t\t\tfavicon: 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAwAAAAMCAYAAABWdVznAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAXZJREFUeNp0UjFOw0AQnD3bCYaEoIgiUIQKikiAKKGgSAkFL0BQ0FEgUfECShpewB+QEE+gokDiBxGRiCCEBBzHd8ucLYGQ4Kw7+9azc7OzJ6oKnFYUxgD+W/DHYNA54GIooifTitYqg1okaPFff6BcTLF5fECYb5IJNJ3kMfUP2dxHPycwcQ0SliARocSGsA46YkKWQW0GS6BDAGlu5Myu84AgSmHKM/DY0NegwwR2+AJrSjBre5BaA1H7GFKpI729QHp9jqiaeTBn5mB7fWirjWB9B9JYBrwEMnop5d0zfD53YO9vYMY8SQcpZGkTZvsIstiivDfoaxfIxt8exYeXCLYOSOwlJZQ6twSpL0DfX4EoZtkCqc7/Njalbxnfdt+oSwhurkBnCWI/1FlgqgKNZ3JnZDSA3FFSpYdQAoF56tKNDmhOYTzJ3OSHXZkflJhYEyawcnH02HpmxZ+DrRJlgmacvrsHRmHln2tRnIiAy5WTLwEGAK4QoBQmtGHkAAAAAElFTkSuQmCC',\r\n\t\t\t\tselector: 'a[href*=\"soundcloud.com/\"]',\r\n\t\t\t\tkey: 'uri',\r\n\t\t\t\treg: '(soundcloud)',\r\n\t\t\t\tmatcher: /(?:\\/\\/(?:\\bwww|m\\.\\b)?soundcloud\\.com\\/(.+?\\/.+))/i,\r\n\t\t\t\thttps: true,\r\n\t\t\t\tscroll: true\r\n\t\t\t},\r\n\t\t\tgfycat: {\r\n\t\t\t\ttitle: 'gfycat meow!',\r\n\t\t\t\thome: 'https://gfycat.com/',\r\n\t\t\t\tembed: 'https://gfycat.com/iframe/%key',\r\n\t\t\t\tajaxExtension: true,\r\n\t\t\t\trawResponse: true,\r\n\t\t\t\tajax: 'https://gfycat.com/%key',\r\n\t\t\t\tthumb: 'url(https://thumbs.gfycat.com/%key-poster.jpg)',\r\n\t\t\t\tselector: 'a[href*=\"gfycat.com/\"]',\r\n\t\t\t\tfavicon: 'https://gfycat.com/favicon.ico',\r\n\t\t\t\tkey: 'id',\r\n\t\t\t\treg: '(gfycat)',\r\n\t\t\t\tmatcher: /(?:gfycat\\.com\\/(?:(\\b(?:[A-Z][a-z]*){3,}\\b)))/i,\r\n\t\t\t\thttps: true,\r\n\t\t\t\tscroll: true\r\n\t\t\t},\r\n\t\t\tstreamable: {\r\n\t\t\t\ttitle: 'streamable!',\r\n\t\t\t\thome: 'https://streamable.com/',\r\n\t\t\t\tembed: 'https://streamable.com/e/%key',\r\n\t\t\t\tajax: 'https://api.streamable.com/oembed.json?url=%uri',\r\n\t\t\t\tthumb: 'url(https://cdn.streamable.com/image/%key.jpg)',\r\n\t\t\t\tselector: 'a[href*=\"streamable.com/\"]',\r\n\t\t\t\tfavicon: 'https://streamable.com/favicon.ico',\r\n\t\t\t\tkey: 'id',\r\n\t\t\t\treg: '(streamable\\\\.com)',\r\n\t\t\t\tmatcher: /(?:streamable\\.com\\/([A-Za-z0-9-_]+))/i,\r\n\t\t\t\thttps: true\r\n\t\t\t},\r\n\t\t\timgur: {\r\n\t\t\t\ttitle: 'imgur it!',\r\n\t\t\t\thome: 'https://i.imgur.com/',\r\n\t\t\t\tembed: 'https://i.imgur.com/%key',\r\n\t\t\t\tthumb: 'url(https://i.imgur.com/%keyh.jpg)',\r\n\t\t\t\tselector: 'a[href*=\".gifv\"]',\r\n\t\t\t\tfavicon: 'https://imgur.com/favicon.ico',\r\n\t\t\t\treg: '(\\\\.gifv$)|(imgur)',\r\n\t\t\t\tmatcher: /(?:imgur\\.com\\/(\\w+)\\.(?:gifv|mp4|webm))/i,\r\n\t\t\t\thttps: true,\r\n\t\t\t\tscroll: true,\r\n\t\t\t\tvideoTag: true\r\n\t\t\t},\r\n\t\t\thtml5: {\r\n\t\t\t\thome: true,\r\n\t\t\t\ttitle: 'html5 go!',\r\n\t\t\t\tselector: 'a[href*=\".webm\"], a[href*=\".mp4\"]',\r\n\t\t\t\tfavicon: 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAA0AAAANCAYAAABy6+R8AAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAdlJREFUeNpcks9rE0EUx7+b3TRktxvbhUhrNm6SrUWtxxbxJ9iLl0JFKWg8WLD05EWhIuRieogXwYPX4tGb/gPSq/oP6KH+uG2MFaRJJXaTbp7fWbsl+uDDzLx533lv3ozmeyUM2VlSJxeISULygTwhr5Og1JDgOXlPrpLRMAxTIpLlfJa8Ipv/i56Re4lT07SYKIqQSh2eO0/eqYnujI35HF8O19jr9XBneRmt1je0d3ag63qy5ZIAvNObE+WKTJXKktENoVPsrCntdlsWFxbiNXNJyS3KdMUXxv8w6LzY7XZhmiYer9fBYOTzR5HL5XCzWsWxYhFnTs/gxcYGgiCAZVmOyhR5BVcm8nlZvbsi29+3ZdgGg4Gs1+viua4UJiZVpkiJeqo0VaIqxbGPSKfTORRduXQ59peLx5PyItWaj9yDQtmt21XYto1GoxGvbywtxaNhGHE3ab+V6L6aKYczPo7ZuTk8WnuIWq2G64vX4JU8zJw8BXXv5D21gx+xydrn7VEb/f0+tj5twa/4+Pz1C6Y4Oo6DZrOJdDrdYuxkIlKP+ZbZzoV7e8iyk8kD/9rdRXpkBJlMpsmDC/98I97pPF//AQUBl/2/jRvsm5b1kxmeJgJlfwQYAKZQxgzeI6/EAAAAAElFTkSuQmCC',\r\n\t\t\t\treg: '(\\\\.webm$)|(\\\\.mp4$)',\r\n\t\t\t\tslim: true,\r\n\t\t\t\tscroll: true,\r\n\t\t\t\tvideoTag: true\r\n\t\t\t},\r\n\t\t\t'html5-audio': {\r\n\t\t\t\thome: true,\r\n\t\t\t\ttitle: 'hey, listen!',\r\n\t\t\t\tselector: 'a[href*=\".mp3\"]',\r\n\t\t\t\treg: '(\\\\.mp3$)',\r\n\t\t\t\tslim: true,\r\n\t\t\t\tscroll: true\r\n\t\t\t}\r\n\t\t}\r\n\t};\r\n\r\n\tclass Control extends Container {\r\n\t\t/** U I CLASS\r\n\t\t * Class for the player controls\r\n\t\t * This is the control bar once the user clicks on the thumbnail\r\n\t\t * Contains its own instance of a Player\r\n\t\t * Acts like a decorator on the YTMA and Container intances by adding events\r\n\t\t * @param {Y} ytma A YTMA instance\r\n\t\t */\r\n\t\tconstructor(props) {\r\n\t\t\tsuper(props);\r\n\r\n\t\t\tthis.open = false;\r\n\t\t\tthis.selected = { size: null, ratio: null };\r\n\t\t}\r\n\r\n\t\tgetControl() {\r\n\t\t\tif (!this.projector) {\r\n\t\t\t\tthis.createProjector();\r\n\t\t\t}\r\n\t\t\treturn this;\r\n\t\t}\r\n\r\n\t\tcreateProjector() {\r\n\t\t\tsuper.createProjector();\r\n\t\t\tthis.projector.addEventListener('click', Control.events.videoBar.bind(this), false);\r\n\t\t\tthis.play = new Player(this);\r\n\r\n\t\t\tthis.markSelected(`li[data-type=\"size\"][data-value=\"${this.play.attrs.size}\"]`, 'size');\r\n\t\t\tthis.markSelected(`li[data-type=\"ratio\"][data-value=\"${this.play.attrs.ratio}\"]`, 'ratio');\r\n\t\t}\r\n\r\n\t\tresetViewSize() {\r\n\t\t\tthis.play.dimmensions();\r\n\t\t\tthis.setControlBarSize(this.play.attrs.size);\r\n\t\t}\r\n\r\n\t\tshowOnScroll(link) {\r\n\t\t\tif (!this.open && this.canShowUnder(link)) {\r\n\t\t\t\tthis.showPlayer();\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\tshowPlayer() {\r\n\t\t\tthis.open = true;\r\n\r\n\t\t\tsuper.showPlayer();\r\n\t\t\tthis.attachPlayerPanel();\r\n\t\t\tthis.play.switchOn();\r\n\r\n\t\t\tif (Y.user.preferences.focus) {\r\n\t\t\t\tdocument.location.hash = `#${this.body.id}`;\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\thidePlayer() {\r\n\t\t\tthis.open = false;\r\n\r\n\t\t\tthis.play.switchOff();\r\n\t\t\tsuper.hidePlayer();\r\n\t\t}\r\n\r\n\t\tattachPlayerPanel() {\r\n\t\t\tif (!this.play.panel.parentNode) {\r\n\t\t\t\t// console.log('attaching display panel');\r\n\t\t\t\tthis.projector.appendChild(this.play.panel);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\thideAllPlayers() {\r\n\t\t\tconst group = Y.collect(this.state.id);\r\n\t\t\tconsole.info('ytma//hide+all(id)', this.state.id, group.length);\r\n\t\t\tgroup.forEach(g => {\r\n\t\t\t\tg.disableOpenOnScroll();\r\n\t\t\t\tg.getControl().hidePlayer();\r\n\t\t\t});\r\n\t\t}\r\n\r\n\t\tsetControlBarSize(size) {\r\n\t\t\tthis.markSelected(`li[data-type=\"size\"][data-value=\"${size}\"]`, 'size');\r\n\t\t}\r\n\r\n\t\tmarkSelected(el, type) {\r\n\t\t\tif (typeof el === 'string') {\r\n\t\t\t\tel = this.projector.querySelector(el);\r\n\t\t\t}\r\n\t\t\tel.id = type + this.state.uid;\r\n\t\t\ttry {\r\n\t\t\t\tthis.selected[type].removeAttribute('id');\r\n\t\t\t} catch (e) { }\r\n\t\t\tthis.selected[type] = el;\r\n\t\t}\r\n\t}\r\n\r\n\tControl.ratios = {\r\n\t\tSD: 1,\r\n\t\tHD: 2,\r\n\t\tPORTRAIT: 3\r\n\t};\r\n\r\n\tControl.sizes = {\r\n\t\tHIDDEN: 0,\r\n\t\tS: 240,\r\n\t\tM: 360,\r\n\t\tL: 480,\r\n\t\tX: 720\r\n\t};\r\n\r\n\t/** Trigger is the VAR element\r\n\t * @param {HTMLElement} t VAR element\r\n\t */\r\n\tControl.createFromTrigger = t => {\r\n\t\t// console.info('ytma//trigger');\r\n\t\tif (t && t.dataset.ytmuid && !Y.set[t.dataset.ytmuid]) {\r\n\t\t\tconsole.info('ytma//trigger+new');\r\n\t\t\tY.addToSet(new Control().reactivate(t));\r\n\t\t}\r\n\t\tconsole.info('ytma//trigger+control');\r\n\t\treturn Y.set[t.dataset.ytmuid].getControl();\r\n\t};\r\n\r\n\tControl.events = {\r\n\t\t$fire: {\r\n\t\t\tsettings: function () {\r\n\t\t\t\tY.user.events.formToggle();\r\n\t\t\t},\r\n\t\t\tclose: function () {\r\n\t\t\t\tif (this.site.scroll) {\r\n\t\t\t\t\t// console.log('events.close-1');\r\n\t\t\t\t\tthis.hideAllPlayers();\r\n\t\t\t\t} else {\r\n\t\t\t\t\t// console.log('events.close-2');\r\n\t\t\t\t\tthis.disableOpenOnScroll();\r\n\t\t\t\t\tthis.hidePlayer();\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tratio: function (li) {\r\n\t\t\t\tconst ratio = parseInt(li.dataset.value, 10);\r\n\t\t\t\tthis.play.dimmensions({ ratio });\r\n\t\t\t\tthis.markSelected(li, 'ratio');\r\n\t\t\t},\r\n\t\t\tsize: function (li) {\r\n\t\t\t\tconst size = parseInt(li.dataset.value, 10);\r\n\t\t\t\tthis.play.dimmensions({ size });\r\n\t\t\t\tthis.markSelected(li, 'size');\r\n\t\t\t}\r\n\t\t},\r\n\t\tvideoBar: function ({ target }) {\r\n\t\t\tif (target.tagName.toLowerCase() === 'li' && target.dataset && target.dataset.type) {\r\n\t\t\t\tconst t = target.dataset.type;\r\n\t\t\t\tif (Control.events.$fire[t]) {\r\n\t\t\t\t\tControl.events.$fire[t].call(this, target);\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\t};\r\n\r\n\t/** P L A Y E R CLASS\r\n\t *  @param {Control} parent Instance\r\n\t */\r\n\tclass Player {\r\n\r\n\t\tconstructor(parent) {\r\n\t\t\tthis.parent = parent;\r\n\r\n\t\t\tthis.mode = 'off';\r\n\r\n\t\t\tthis.attrs = {\r\n\t\t\t\tsources: null,\r\n\t\t\t\tquality: this.quality,\r\n\t\t\t\tsize: null,\r\n\t\t\t\tratio: null,\r\n\t\t\t\tstart: this.time(),\r\n\t\t\t\ttype: null\r\n\t\t\t};\r\n\r\n\t\t\tthis.attrs.sources = this.sources;\r\n\r\n\t\t\t// todo improve type/media\r\n\t\t\tthis.attrs.type = this.findType();\r\n\t\t\tthis.media = Player.makeMedia[this.attrs.type](this);\r\n\r\n\t\t\tthis.channel = _.e('div', { className: 'ytm_panel_channel ytm_block' }, this.media, true);\r\n\t\t\tthis.switcher = _.e('div', { className: `ytm_panel_switcher ytm_panel_size ytm_block ytm_${this.attrs.type}`, _ytmuid: this.parent.state.uid, _standby: true });\r\n\t\t\tthis.panel = _.e('div', { className: 'ytm_panel ytm_block' }, this.switcher, true);\r\n\r\n\t\t\tif (parent.state.site === 'soundcloud' && Y.reg.extra.soundcloud.playlist.test(parent.anchor.href)) {\r\n\t\t\t\tthis.media.classList.add('ytm_soundcloud-playlist');\r\n\t\t\t\tthis.switcher.classList.add('ytm_soundcloud-playlist');\r\n\t\t\t}\r\n\r\n\t\t\tthis.dimmensions(Y.user.preferences);\r\n\t\t}\r\n\r\n\t\tget sources() {\r\n\t\t\ttry {\r\n\t\t\t\treturn (Player.sources[this.parent.state.site] || Player.sources.iframe)(this.parent.state, this.attrs);\r\n\t\t\t} catch (e) {\r\n\t\t\t\tconsole.error(e);\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\tget quality() {\r\n\t\t\treturn Player.qualities[Y.user.preferences.quality] || Player.qualities[360];\r\n\t\t}\r\n\r\n\t\tget className() {\r\n\t\t\treturn `ytm_panel ytm_block ytm_panel-${Player.dimms.ratios[this.attrs.ratio]} ytm_panel-${Player.dimms.sizes[this.attrs.size]}`;\r\n\t\t}\r\n\r\n\t\tdimmensions({ ratio, size }) {\r\n\t\t\tthis.attrs.ratio = isNumber(ratio) ? ratio : this.attrs.ratio;\r\n\t\t\tthis.attrs.size = isNumber(size) ? size : this.attrs.size;\r\n\t\t\tthis.panel.className = this.className;\r\n\t\t}\r\n\r\n\t\ttime() {\r\n\t\t\ttry {\r\n\t\t\t\t// debugger;\r\n\t\t\t\tconst m = this.parent.state.uri.match(Y.reg.time).slice(1, 4);\r\n\t\t\t\treturn ((+m[0] || 0) * 60 * 60) + ((+m[1] || 0) * 60) + (+m[2] || 0);\r\n\t\t\t} catch (e) { return 0; }\r\n\t\t}\r\n\r\n\t\tfindType() {\r\n\t\t\tif (this.parent.state.site === 'html5-audio') { return 'audio'; }\r\n\t\t\tif (this.parent.site.videoTag) { return 'video'; }\r\n\t\t\treturn 'iframe';\r\n\t\t}\r\n\r\n\t\tswitchOff() {\r\n\t\t\t// console.log('removed media');\r\n\r\n\t\t\tif (this.media.pause) {\r\n\t\t\t\t// console.log('pausing');\r\n\t\t\t\tthis.media.pause();\r\n\t\t\t}\r\n\r\n\t\t\ttry {\r\n\t\t\t\tthis.switcher.removeChild(this.channel);\r\n\t\t\t} catch (e) {\r\n\t\t\t\t// console.error(e);\r\n\t\t\t}\r\n\t\t\tthis.mode = 'off';\r\n\t\t}\r\n\r\n\t\tswitchOn() {\r\n\t\t\tif (this.attrs.size === 0) {\r\n\t\t\t\tthis.attrs.size = Y.user.preferences.size;\r\n\t\t\t\tthis.parent.resetViewSize();\r\n\t\t\t}\r\n\t\t\t// console.log('switch to media');\r\n\t\t\tthis.switcher.appendChild(this.channel);\r\n\t\t\tthis.switcher.dataset.standby = false;\r\n\t\t\tthis.mode = 'on';\r\n\t\t}\r\n\r\n\t\tswitchStandby() {\r\n\t\t\t// console.log('switch to standby');\r\n\t\t\tthis.switchOff();\r\n\t\t\tthis.switcher.dataset.standby = true;\r\n\t\t\tthis.mode = 'standby';\r\n\t\t}\r\n\r\n\t\tisStandby() {\r\n\t\t\treturn this.mode === 'standby';\r\n\t\t}\r\n\t}\r\n\r\n\tPlayer.sources = {\r\n\t\tiframe: function (data) {\r\n\t\t\tconst key = Y.DB.sites[data.site].key;\r\n\r\n\t\t\treturn [\r\n\t\t\t\t{ type: 'text/html', src: Y.DB.sites[data.site].embed.replace('%key', data[key]) }\r\n\t\t\t];\r\n\t\t},\r\n\t\t'html5-audio': function ({ uri }) {\r\n\t\t\treturn [\r\n\t\t\t\t{ type: 'audio/mp3', src: uri }\r\n\t\t\t];\r\n\t\t},\r\n\t\thtml5: function ({ uri }) {\r\n\t\t\t// attaching the type as either mp4 or webm\r\n\r\n\t\t\tif (/(?:webm)/.test(uri)) {\r\n\t\t\t\treturn [\r\n\t\t\t\t\t{ type: 'video/webm', src: uri }\r\n\t\t\t\t];\r\n\t\t\t}\r\n\r\n\t\t\treturn [\r\n\t\t\t\t{ type: 'video/mp4', src: uri },\r\n\t\t\t\t{ type: 'video/webm', src: uri },\r\n\t\t\t\t{ type: 'video/ogg; codecs=\"theora, vorbis\"', src: uri }\r\n\t\t\t];\r\n\t\t},\r\n\t\timgur: function ({ id }) {\r\n\t\t\tconst src = Y.DB.sites.imgur.embed.replace('%key', id);\r\n\r\n\t\t\treturn [\r\n\t\t\t\t{ type: 'video/webm', src: `${src}.webm` },\r\n\t\t\t\t{ type: 'video/mp4', src: `${src}.mp4` }\r\n\t\t\t];\r\n\t\t},\r\n\t\tyoutube: function ({ id }, { quality, start }) {\r\n\t\t\tconst params = `?html5=1&version=3&modestbranding=1&rel=0&showinfo=1&vq=${quality}&iv_load_policy=${Y.user.preferences.yt_annotation}&start=${start}&rel=0`;\r\n\r\n\t\t\treturn [\r\n\t\t\t\t{ type: 'text/html', src: Y.DB.sites.youtube.embed.replace('%key', id) + params }\r\n\t\t\t];\r\n\t\t}\r\n\t};\r\n\r\n\tPlayer.dimms = {\r\n\t\tratios: {\r\n\t\t\t1: 'sd',\r\n\t\t\t2: 'hd',\r\n\t\t\t3: 'pr'\r\n\t\t},\r\n\t\tsizes: {\r\n\t\t\t0: 'h',\r\n\t\t\t240: 's',\r\n\t\t\t360: 'm',\r\n\t\t\t480: 'l',\r\n\t\t\t720: 'xl'\r\n\t\t},\r\n\t\taspects: {\r\n\t\t\t1: 4 / 3,\r\n\t\t\t2: 16 / 9,\r\n\t\t\t3: 16 / 9\r\n\t\t}\r\n\t};\r\n\r\n\tPlayer.qualities = {\r\n\t\t240: 'small',\r\n\t\t360: 'medium',\r\n\t\t480: 'large',\r\n\t\t720: 'hd720',\r\n\t\t1080: 'hd1080',\r\n\t\t1081: 'highres'\r\n\t};\r\n\r\n\tPlayer.css = {\r\n\t\titem: function (key, value) {\r\n\t\t\tif (isNumber(value)) {\r\n\t\t\t\tvalue += 'px';\r\n\t\t\t}\r\n\r\n\t\t\treturn `\\t${key}: ${value};\\n`;\r\n\t\t},\r\n\t\titer: function (css, cssEntries) {\r\n\t\t\t_.o(cssEntries, (key, value) => {\r\n\t\t\t\tcss.push(Player.css.item(key, value));\r\n\t\t\t});\r\n\t\t\tcss.push('}');\r\n\t\t},\r\n\t\tgenerator: function () {\r\n\t\t\tconst css = [];\r\n\r\n\t\t\t_.o(this.sizes, (size, sizes) => {\r\n\t\t\t\t_.o(sizes, (dimm, keys) => {\r\n\t\t\t\t\tcss.push(`\\n.ytm_panel-${size}.ytm_panel-${dimm} .ytm_panel_size {\\n`);\r\n\t\t\t\t\tPlayer.css.iter(css, keys);\r\n\t\t\t\t});\r\n\t\t\t});\r\n\r\n\t\t\t// add site overrides\r\n\t\t\t_.o(this.sites, (site, data) => {\r\n\t\t\t\t_.o(data, (setting, keys) => {\r\n\t\t\t\t\tif (setting === 'all') {\r\n\t\t\t\t\t\tcss.push(`\\n.ytm_site_${site} .ytm_panel_size {\\n`);\r\n\t\t\t\t\t} else {\r\n\t\t\t\t\t\tcss.push(`\\n.ytm_site_${site} .ytm_panel-${setting} .ytm_panel_size {\\n`);\r\n\t\t\t\t\t}\r\n\t\t\t\t\tPlayer.css.iter(css, keys);\r\n\t\t\t\t});\r\n\t\t\t});\r\n\r\n\t\t\treturn css.join('');\r\n\t\t},\r\n\t\tsizes: (() => {\r\n\t\t\tconst merge = {};\r\n\r\n\t\t\t_.o(Player.dimms.sizes, (num, size) => {\r\n\t\t\t\tif (num >= 0) {\r\n\t\t\t\t\tmerge[size] = {};\r\n\r\n\t\t\t\t\t_.o(Player.dimms.ratios, (k, ratio) => {\r\n\t\t\t\t\t\tif (ratio === 'pr') {\r\n\t\t\t\t\t\t\tconst w = Math.floor(num * 0.95); // smaller than the normal sizes\r\n\t\t\t\t\t\t\tmerge[size][ratio] = {\r\n\t\t\t\t\t\t\t\twidth: w,\r\n\t\t\t\t\t\t\t\theight: Math.floor(w * Player.dimms.aspects[k])\r\n\t\t\t\t\t\t\t};\r\n\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\tmerge[size][ratio] = {\r\n\t\t\t\t\t\t\t\twidth: Math.floor(num * Player.dimms.aspects[k]),\r\n\t\t\t\t\t\t\t\theight: num\r\n\t\t\t\t\t\t\t};\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t});\r\n\t\t\t\t}\r\n\t\t\t});\r\n\r\n\t\t\treturn merge;\r\n\t\t})(),\r\n\t\tsites: { // custom sizes per site\r\n\t\t\tsoundcloud: {\r\n\t\t\t\tall: {\r\n\t\t\t\t\theight: '118px !important'\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tvine: {\r\n\t\t\t\ts: {\r\n\t\t\t\t\twidth: 240,\r\n\t\t\t\t\theight: 240\r\n\t\t\t\t},\r\n\t\t\t\tm: {\r\n\t\t\t\t\twidth: 360,\r\n\t\t\t\t\theight: 360\r\n\t\t\t\t},\r\n\t\t\t\tl: {\r\n\t\t\t\t\twidth: 480,\r\n\t\t\t\t\theight: 480\r\n\t\t\t\t},\r\n\t\t\t\txl: {\r\n\t\t\t\t\twidth: 720,\r\n\t\t\t\t\theight: 720\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\t};\r\n\r\n\tPlayer.makeMedia = {\r\n\t\t$css: function (type) {\r\n\t\t\treturn `ytm_panel_media ytm_panel_size ytm_block ytm_${type}`;\r\n\t\t},\r\n\t\tvideo: function ({ attrs }) {\r\n\t\t\tconst video = _.e('video', {\r\n\t\t\t\tcontrols: true,\r\n\t\t\t\tautoplay: false,\r\n\t\t\t\tloop: true,\r\n\t\t\t\tclassName: this.$css('video'),\r\n\t\t\t\t$allowscriptaccess: true,\r\n\t\t\t\tpreload: 'metadata'\r\n\t\t\t});\r\n\r\n\t\t\tconst links = [];\r\n\r\n\t\t\tattrs.sources.forEach(({ src, type }) => {\r\n\t\t\t\t_.e('source', { src, $type: type }, video);\r\n\r\n\t\t\t\tlinks.push(`<a href=\"${src}\">${src}</a>`);\r\n\t\t\t});\r\n\r\n\t\t\t_.e('p', { innerHTML: `Could not load source(s): ${links.join('<br />')}` }, video);\r\n\r\n\t\t\treturn video;\r\n\t\t},\r\n\t\tiframe: function ({ attrs }) {\r\n\t\t\treturn _.e('iframe', {\r\n\t\t\t\t$allowfullscreen: true,\r\n\t\t\t\t$referrerpolicy: 'no-referrer',\r\n\t\t\t\t// $sandbox: 'allow-same-origin allow-scripts allow-popups',\r\n\t\t\t\t$type: attrs.sources[0].type,\r\n\t\t\t\tsrc: attrs.sources[0].src,\r\n\t\t\t\tclassName: this.$css('iframe')\r\n\t\t\t});\r\n\t\t},\r\n\t\taudio: function ({ attrs }) {\r\n\t\t\treturn _.e('audio', {\r\n\t\t\t\tsrc: attrs.sources[0].src,\r\n\t\t\t\t$type: attrs.sources[0].type\r\n\t\t\t});\r\n\t\t}\r\n\t};\r\n\r\n\t/** S C R O L L CLASS\r\n\t * Window-Scroll Event Helper\r\n\t */\r\n\tclass Scroll {\r\n\t\tconstructor(selector, cb, delay = 500) {\r\n\t\t\tthis.selector = selector;\r\n\t\t\tthis.cb = cb;\r\n\t\t\tthis.monitor = this.monitor.bind(this);\r\n\r\n\t\t\t// console.log('YTMA.Scroll Monitor: ', selector);\r\n\t\t\tthis.bound = debounce(this.monitor, delay);\r\n\r\n\t\t\tthis.bound();\r\n\t\t\twindow.addEventListener('scroll', this.bound, false);\r\n\t\t}\r\n\r\n\t\tstop() {\r\n\t\t\t// console.log('clear scroll: ', this.selector);\r\n\t\t\twindow.removeEventListener('scroll', this.bound);\r\n\t\t}\r\n\r\n\t\tmonitor() {\r\n\t\t\t_.s(this.selector, this.cb);\r\n\t\t}\r\n\t}\r\n\r\n\tScroll.visible = el => {\r\n\t\tconst bound = el.getBoundingClientRect();\r\n\t\treturn (bound.top >= 0 && bound.top <= document.documentElement.clientHeight);\r\n\t};\r\n\r\n\tScroll.visibleAll = (el, offset) => {\r\n\t\tconst bound = el.getBoundingClientRect();\r\n\t\tconst height = document.documentElement.clientHeight;\r\n\t\toffset = isNumber(offset) ? +offset : 0;\r\n\t\treturn ((bound.bottom + offset >= 0)\r\n\t\t\t&& (bound.top <= height + offset || bound.bottom <= height - offset));\r\n\t};\r\n\r\n\t/** Returns 1, 0, -1 when el1 is above, exactly the same, or below el2 */\r\n\tScroll.compare = (el1, el2) => {\r\n\t\tconst a = el1.getBoundingClientRect().y;\r\n\t\tconst b = el2.getBoundingClientRect().y;\r\n\r\n\t\tif (a < b) { return 1; }\r\n\t\tif (a === b) { return 0; }\r\n\t\treturn -1;\r\n\t};\r\n\r\n\r\n\tY.main();\r\n})();\r\n","if (!Element.prototype.matches)\r\n\tElement.prototype.matches = Element.prototype.msMatchesSelector || Element.prototype.webkitMatchesSelector;\r\n\r\nif (!Element.prototype.closest) {\r\n\tElement.prototype.closest = function (s) {\r\n\t\tvar el = this;\r\n\t\tif (!document.documentElement.contains(el)) return null;\r\n\t\tdo {\r\n\t\t\tif (el.matches(s)) return el;\r\n\t\t\tel = el.parentElement || el.parentNode;\r\n\t\t} while (el !== null && el.nodeType === 1);\r\n\t\treturn null;\r\n\t};\r\n}\r\n\r\nexport const isNumber = n => !isNaN(parseFloat(n)) && isFinite(n);\r\n\r\nexport const removeSearch = (uri, keepHash) => {\r\n\t// removes search query from a uri\r\n\tconst s = uri.indexOf('?');\r\n\tconst h = uri.indexOf('#');\r\n\tlet hash = '';\r\n\tif (s > -1) {\r\n\t\tif (keepHash && h > -1) {\r\n\t\t\thash = uri.substr(h);\r\n\t\t}\r\n\t\turi = uri.substr(0, s) + hash;\r\n\t}\r\n\treturn uri;\r\n};\r\n\r\n/**\r\n * @param {HTMLElement} element HTML element\r\n * @param {string} events Space- or coma-separated string of one or more types, eg \"click dblclick\"\r\n * @param {string} selector CSS selector for the elements to trigger the event on\r\n * @param {Function} listener A callback\r\n * @param {Boolean} cancel Cancel\r\n */\r\nexport const on = (element, events, selector, listener, cancel = true) => {\r\n\tevents = events.split(/(?:\\s+|,)/).filter(f => f);\r\n\r\n\tif (events.length === 0) return;\r\n\r\n\tconst fn = event => {\r\n\t\tconst found = event.target.closest(selector);\r\n\t\tif (found) listener.call(found, event);\r\n\t};\r\n\r\n\tevents.forEach(type => element.addEventListener(type, fn, cancel));\r\n};\r\n\r\nexport const debounce = (fn, delay = 250) => {\r\n\tlet timeout;\r\n\r\n\treturn function (...args) {\r\n\t\tconst timed = () => {\r\n\t\t\ttimeout = null;\r\n\t\t\tfn.apply(this, args);\r\n\t\t};\r\n\r\n\t\twindow.clearTimeout(timeout);\r\n\t\ttimeout = window.setTimeout(timed, delay);\r\n\t};\r\n};","// H E L P E R Handle\r\nconst _ = {\r\n\ts: (selector, cb) => {\r\n\t\tconst elements = _.qsa(selector);\r\n\t\telements.some((element, index) => cb(element, index, elements) === false);\r\n\t},\r\n\to: function (object, cb) {\r\n\t\tObject.keys(object).some(key => cb(key, object[key], object) === false);\r\n\t},\r\n\te: function (t, o, e, p) {\r\n\t\tconst c = document.createElement(t);\r\n\t\t_.o(o, (k, v) => {\r\n\t\t\tconst b = k.charAt(0);\r\n\t\t\tif (b === '_')\r\n\t\t\t\tc.dataset[k.substring(1)] = v;\r\n\t\t\telse if (b === '$')\r\n\t\t\t\tc.setAttribute(k.substring(1), v);\r\n\t\t\telse\r\n\t\t\t\tc[k] = v;\r\n\t\t});\r\n\r\n\t\tif (e && p) {\r\n\t\t\tc.appendChild(e);\r\n\t\t} else if (e) {\r\n\t\t\te.appendChild(c);\r\n\t\t}\r\n\t\treturn c;\r\n\t},\r\n\tqsa: selector => Array.from(document.querySelectorAll(selector)),\r\n\tcss: function (text) {\r\n\t\tif (!this.style) {\r\n\t\t\tthis.style = document.createElement('style');\r\n\t\t\tthis.style.type = 'text/css';\r\n\t\t\tdocument.body.appendChild(this.style);\r\n\t\t}\r\n\t\tthis.style.appendChild(document.createTextNode(`${text}\\n`));\r\n\t},\r\n\tjs: t => {\r\n\t\tconst j = document.createElement('script');\r\n\t\tj.type = 'text/javascript';\r\n\t\tj[/^https?:\\/\\//i.test(t) ? 'src' : 'textContent'] = t;\r\n\t\tdocument.head.appendChild(j);\r\n\t}\r\n};\r\n\r\nexport default _;","// S T O R A G E HANDLE\r\nconst strg = {\r\n\tMAX: 5012,\r\n\ton: false,\r\n\ttest: () => {\r\n\t\ttry {\r\n\t\t\tconst l = localStorage;\r\n\t\t\tconst c = Math.random().toString(16).substr(2, 8);\r\n\t\t\tl.setItem(c, c);\r\n\t\t\treturn l.getItem(c) === c ? !l.removeItem(c) : false;\r\n\t\t} catch (e) { return false; }\r\n\t},\r\n\tread: (key) => {\r\n\t\ttry {\r\n\t\t\treturn JSON.parse(localStorage.getItem(key));\r\n\t\t} catch (e) {\r\n\t\t\treturn console.error(`${e.lineNumber}:${e.message}`);\r\n\t\t}\r\n\t},\r\n\tsave: (key, val) => strg.on ? !localStorage.setItem(key, JSON.stringify(val)) : false,\r\n\twipe: key => strg.on ? !localStorage.removeItem(key) : false,\r\n\tzero: o => { for (let k in o) { if (o.hasOwnProperty(k)) { return false; } } return true; }, // check if the object is empty\r\n\tgrab: (key, def) => { const s = strg.read(key); return strg.zero(s) ? def : s; },\r\n\tsize: () => {\r\n\t\tlet length = 0;\r\n\t\tlet key;\r\n\t\ttry {\r\n\t\t\tfor (key in window.localStorage) {\r\n\t\t\t\tif (window.localStorage.hasOwnProperty(key)) {\r\n\t\t\t\t\tlength += window.localStorage[key].length;\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t} catch (e) {\r\n\t\t\t// no more space\r\n\t\t}\r\n\t\treturn 3 + ((length * 16) / (8 * 1024));\r\n\t},\r\n\tfull: () => {\r\n\t\ttry {\r\n\t\t\tconst date = +(new Date());\r\n\t\t\tlocalStorage.setItem(date, date);\r\n\t\t\tlocalStorage.removeItem(date);\r\n\t\t\treturn false;\r\n\t\t} catch (e) {\r\n\t\t\tif (e.name === 'QuotaExceededError' ||\r\n\t\t\t\t\te.name === 'NS_ERROR_DOM_QUOTA_REACHED') {\r\n\t\t\t\treturn true;\r\n\t\t\t}\r\n\t\t}\r\n\t},\r\n\tinit: () => { strg.on = strg.test(); }\r\n};\r\nstrg.init();\r\n\r\nexport default strg;","import strg from './strg';\r\nimport _ from './_';\r\n\r\n// U P D A T E HANDLE\r\nconst update = {\r\n\tname: 'ytma!',\r\n\tversion: 8010,\r\n\tkey: 'ujs_YTMA_UPDT_HR',\r\n\tcallback: 'ytmaupdater',\r\n\tpage: 'https://greasyfork.org/scripts/1023-youtube-me-again',\r\n\tjson: 'https://hateradio.github.io/ytma/update.json',\r\n\tinterval: 7,\r\n\tday: (new Date()).getTime(),\r\n\tshow: false,\r\n\ttime: () => new Date(update.day + (1000 * 60 * 60 * 24 * update.interval)).getTime(),\r\n\tnotification: ({ version, page }) => {\r\n\t\tif (update.version < version) {\r\n\t\t\tstrg.save(update.key, { date: update.time(), version, page });\r\n\t\t\tupdate.link();\r\n\t\t}\r\n\t},\r\n\tlink: () => {\r\n\t\tif (update.show) { return; }\r\n\t\tupdate.show = true;\r\n\r\n\t\tconst { page } = strg.read(update.key);\r\n\t\tconst link = `\r\n\t\t\t\t<a href=\"${page || update.page}\" id=updatev3 title=Update target=_blank>\r\n\t\t\t\t\tAn update for ${update.name} is available.\r\n\t\t\t\t</a>`;\r\n\r\n\t\t_.css(update.css);\r\n\t\tdocument.body.insertAdjacentHTML('beforeend', link);\r\n\t\t_.on(document.body, 'click', '#updatev3', e => e.target.style.display = 'none');\r\n\t},\r\n\tcheck: (opt) => {\r\n\t\tif (!strg.on) { return; }\r\n\t\tif (window.chrome && window.chrome.extension) { return; }\r\n\t\tconst stored = strg.read(update.key);\r\n\t\tlet page;\r\n\r\n\t\tif (opt || !stored || stored.date < update.day) {\r\n\t\t\tpage = (stored && stored.page) || update.page;\r\n\t\t\tstrg.save(update.key, { date: update.time(), version: update.version, page });\r\n\t\t\tfetch(update.json).then(res => res.json()).then(update.notification);\r\n\t\t} else if (update.version < stored.version) {\r\n\t\t\tupdate.link();\r\n\t\t}\r\n\t},\r\n\tcss: '#updater3,#updater3:visited{box-shadow:1px 1px 6px #7776;border-bottom:3px solid #e39c2d;cursor:pointer;color:#555;font-family:sans-serif;font-size:12px;font-weight:700;text-align:justify;position:fixed;z-index:999999;right:10px;top:10px;background:#ebebeb url(data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgLTM0IDUxMiA1MTIiPjxwYXRoIGZpbGw9IiNlNWViZjUiIGQ9Ik0yNzAuNzgxIDcuNUgyNDEuMjJMOC42NiA0MTAuMzA1bDE0Ljc4MSAyNS42MDFINDg4LjU2bDE0Ljc4LTI1LjYwMXptMCAwIi8+PHBhdGggZmlsbD0iI2Y4ZTg2OCIgZD0iTTQ4LjcwNyA0MDAuOTM0TDI1My4xMjkgNDYuODY3aDUuNzQybDIwNC40MjIgMzU0LjA2N2MtMS4yNzcgMi4yMS0xLjU5NCAyLjc2NS0yLjg3MSA0Ljk3Nkg1MS41NzhjLTEuMjc3LTIuMjEtMS41OTQtMi43NjUtMi44NzEtNC45NzZ6bTAgMCIvPjxwYXRoIGZpbGw9IiNlMzljMmQiIGQ9Ik0yNzUuOTk2IDc2LjUyN2wtMTcuMTI1LTI5LjY2aC01Ljc0Mkw0OC43MDcgNDAwLjkzNGMxLjI3NyAyLjIxIDEuNTk0IDIuNzY1IDIuODcxIDQuOTc2aDM5Ljk5NmMtMS4yNzctMi4yMS0xLjU5Ny0yLjc2NS0yLjg3LTQuOTc2em0wIDAiLz48cGF0aCBmaWxsPSIjY2FkOGVhIiBkPSJNMjc1Ljk5NiAxNi41MzVMMjcwLjc4MSA3LjVIMjQxLjIyTDguNjYgNDEwLjMwNWM2LjU2NyAxMS4zNzkgOC4yMTEgMTQuMjIyIDE0Ljc4MSAyNS42MDFoMzkuOTk2Yy02LjU3LTExLjM3OS04LjIxNC0xNC4yMjItMTQuNzgtMjUuNjAxem0wIDAiLz48cGF0aCBmaWxsPSIjNzI4NjllIiBkPSJNMjcwLjg1NSAzMDIuNDU3aC0yOS43MWMtMy4wMDQtMy4wMDQtNC42OTItNC42ODctNy42OTYtNy42OTFWMTYzLjg2M2g0NS4xMDJ2MTMwLjkwM2MtMy4wMDggMy4wMDQtNC42OTIgNC42ODctNy42OTYgNy42OTF6bTAgMCIvPjxwYXRoIGZpbGw9IiM1MzYyNzUiIGQ9Ik0yNTMuNDUgMTYzLjg2M2gtMjB2MTMwLjkwM2MzLjAwMyAzLjAwNCA0LjY5IDQuNjg3IDcuNjk1IDcuNjkxaDE5Ljk5NmwtNy42OTItNy42OTF6bTAgMCIvPjxwYXRoIGZpbGw9IiM3Mjg2OWUiIGQ9Ik0yMzMuNDUgMzMwLjgxM2g0NS4xdjQ1LjEwMWgtNDUuMXptMCAwIi8+PHBhdGggZmlsbD0iIzUzNjI3NSIgZD0iTTIzMy40NSAzMzAuODEzaDIwdjQ1LjEwMWgtMjB6bTAgMCIvPjxwYXRoIGZpbGw9IiNmZmYiIGQ9Ik0yNzUuOTk2IDE0MS4zNjdIMjYxdi0xNWgxNC45OTZ6bS0yNC45OTYgMGgtMTV2LTE1aDE1em0wIDAiLz48cGF0aCBkPSJNNDAwLjQzIDIxNy4wNTlsLTEyLjk4OSA3LjVMNDk0LjY4IDQxMC4zMDVsLTEwLjQ1IDE4LjEwMUgyNy43N2wtMTAuNDU0LTE4LjEwMUwyNDUuNTQ2IDE1aDIwLjkwN2wxMDcuMjM4IDE4NS43NDYgMTIuOTg5LTcuNUwyNzUuMTEzIDBoLTM4LjIyMkwwIDQxMC4zMDVsMTkuMTEgMzMuMTAxaDQ3My43NzdMNTEyIDQxMC4zMDV6bTAgMCIvPjxwYXRoIGQ9Ik0xMjUuMTU2IDQxMy40MDZINDY0Ljc1bDcuMjAzLTEyLjQ3Mi0yMDguNzUtMzYxLjU2N2gtMTQuNDA2TDQwLjA0NyA0MDAuOTM0bDcuMiAxMi40NzJoNTcuOTEzVjM5OC40MUg1OC44MjRMMjU2IDU2Ljg5bDE5Ny4xNzYgMzQxLjUyaC0zMjguMDJ6bTAgMCIvPjxwYXRoIGQ9Ik0yODYuMDQ3IDIyOS41NTl2LTczLjE5NmgtNjAuMDk4djE0MS41MDhsMTIuMDkgMTIuMDg2aDM1LjkyMmwxMi4wODYtMTIuMDg2VjI0OS41NkgyNzEuMDV2NDIuMDk3bC0zLjMwMSAzLjMwNWgtMjMuNWwtMy4zLTMuMzA1VjE3MS4zNjNoMzAuMXY1OC4xOTZ6bTAgME0yMjUuOTUgMzgzLjQxaDYwLjA5N3YtNjAuMDk4aC02MC4wOTh6bTE1LTQ1LjA5OGgzMC4xdjMwLjEwMmgtMzAuMXptMCAwIi8+PC9zdmc+) no-repeat 10px center;background-size:40px;padding:0 20px 0 60px;height:55px;line-height:55px}#updater3:hover,#updater3:visited:hover{color:#b33a3a !important;border-color:#ce4b30;text-decoration: none;}' // Icon made by Freepik from www.flaticon.com \r\n};\r\nupdate.check();\r\n\r\nexport default update;"],"sourceRoot":""}