Skip to content

Instantly share code, notes, and snippets.

@cletusc
Created July 19, 2017 04:10
Show Gist options
  • Save cletusc/b33600190108f37088953a54223c1c17 to your computer and use it in GitHub Desktop.
Save cletusc/b33600190108f37088953a54223c1c17 to your computer and use it in GitHub Desktop.
2017-07-18 Emote Menu Test
// ==UserScript==
// @name Twitch Chat Emotes
// @namespace #Cletus
// @version 2.1.4
// @description Adds a button to Twitch that allows you to "click-to-insert" an emote.
// @copyright 2011+, Ryan Chatham <ryan.b.chatham@gmail.com> (https://github.com/cletusc)
// @author Ryan Chatham <ryan.b.chatham@gmail.com> (https://github.com/cletusc)
// @icon http://www.gravatar.com/avatar.php?gravatar_id=6875e83aa6c563790cb2da914aaba8b3&r=PG&s=48&default=identicon
// @license MIT; http://opensource.org/licenses/MIT
// @license CC BY-NC-SA 3.0; http://creativecommons.org/licenses/by-nc-sa/3.0/
// @homepage http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/
// @supportURL https://github.com/cletusc/Userscript--Twitch-Chat-Emotes/issues
// @contributionURL http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/#donate
// @grant none
// @include http://*.twitch.tv/*
// @include https://*.twitch.tv/*
// @exclude http://api.twitch.tv/*
// @exclude https://api.twitch.tv/*
// @exclude http://tmi.twitch.tv/*
// @exclude https://tmi.twitch.tv/*
// @exclude http://*.twitch.tv/*/dashboard
// @exclude https://*.twitch.tv/*/dashboard
// @exclude http://chatdepot.twitch.tv/*
// @exclude https://chatdepot.twitch.tv/*
// @exclude http://im.twitch.tv/*
// @exclude https://im.twitch.tv/*
// @exclude http://platform.twitter.com/*
// @exclude https://platform.twitter.com/*
// @exclude http://www.facebook.com/*
// @exclude https://www.facebook.com/*
// ==/UserScript==
/* Script compiled using build script. Script uses Browserify for CommonJS modules. */
(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
var pkg = require('../package.json');
var publicApi = require('./modules/public-api');
var ember = require('./modules/ember-api');
var logger = require('./modules/logger');
var emotes = require('./modules/emotes');
var ui = require('./modules/ui');
logger.log('(v'+ pkg.version + ') Initial load on ' + location.href);
// Only enable script if we have the right variables.
//---------------------------------------------------
var initTimer = 0;
(function init(time) {
if (!time) {
time = 0;
}
var objectsLoaded = (
window.Twitch !== undefined &&
window.jQuery !== undefined &&
ember.isLoaded()
);
if (!objectsLoaded) {
// Stops trying after 10 minutes.
if (initTimer >= 600000) {
logger.log('Taking too long to load, stopping. Refresh the page to try again. (' + initTimer + 'ms)');
return;
}
// Give an update every 10 seconds.
if (initTimer % 10000) {
logger.debug('Still waiting for objects to load. (' + initTimer + 'ms)');
}
// Bump time up after 1s to reduce possible lag.
time = time >= 1000 ? 1000 : time + 25;
initTimer += time;
setTimeout(init, time, time);
return;
}
// Expose public api.
if (typeof window.emoteMenu === 'undefined') {
window.emoteMenu = publicApi;
}
ember.hook('route:channel', activate, deactivate);
ember.hook('route:chat', activate, deactivate);
activate();
})();
function activate() {
ui.init();
emotes.init();
}
function deactivate() {
ui.hideMenu();
}
},{"../package.json":7,"./modules/ember-api":8,"./modules/emotes":9,"./modules/logger":10,"./modules/public-api":11,"./modules/ui":15}],2:[function(require,module,exports){
(function (doc, cssText) {
var id = "emote-menu-for-twitch-styles";
var styleEl = doc.getElementById(id);
if (!styleEl) {
styleEl = doc.createElement("style");
styleEl.id = id;
doc.getElementsByTagName("head")[0].appendChild(styleEl);
}
if (styleEl.styleSheet) {
if (!styleEl.styleSheet.disabled) {
styleEl.styleSheet.cssText = cssText;
}
} else {
try {
styleEl.innerHTML = cssText;
} catch (ignore) {
styleEl.innerText = cssText;
}
}
}(document, "/**\n" +
" * Minified style.\n" +
" * Original filename: \\node_modules\\jquery.scrollbar\\jquery.scrollbar.css\n" +
" */\n" +
".scroll-wrapper{overflow:hidden!important;padding:0!important;position:relative}.scroll-wrapper>.scroll-content{border:none!important;-moz-box-sizing:content-box!important;box-sizing:content-box!important;height:auto;left:0;margin:0;max-height:none!important;max-width:none!important;overflow:scroll!important;padding:0;position:relative!important;top:0;width:auto!important}.scroll-wrapper>.scroll-content::-webkit-scrollbar{height:0;width:0}.scroll-element{display:none}.scroll-element,.scroll-element div{-moz-box-sizing:content-box;box-sizing:content-box}.scroll-element.scroll-x.scroll-scrollx_visible,.scroll-element.scroll-y.scroll-scrolly_visible{display:block}.scroll-element .scroll-arrow,.scroll-element .scroll-bar{cursor:default}.scroll-textarea{border:1px solid #ccc;border-top-color:#999}.scroll-textarea>.scroll-content{overflow:hidden!important}.scroll-textarea>.scroll-content>textarea{border:none!important;-moz-box-sizing:border-box;box-sizing:border-box;height:100%!important;margin:0;max-height:none!important;max-width:none!important;overflow:scroll!important;outline:0;padding:2px;position:relative!important;top:0;width:100%!important}.scroll-textarea>.scroll-content>textarea::-webkit-scrollbar{height:0;width:0}.scrollbar-inner>.scroll-element,.scrollbar-inner>.scroll-element div{border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-inner>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-inner>.scroll-element.scroll-x{bottom:2px;height:8px;left:0;width:100%}.scrollbar-inner>.scroll-element.scroll-y{height:100%;right:2px;top:0;width:8px}.scrollbar-inner>.scroll-element .scroll-element_outer{overflow:hidden}.scrollbar-inner>.scroll-element .scroll-bar,.scrollbar-inner>.scroll-element .scroll-element_outer,.scrollbar-inner>.scroll-element .scroll-element_track{border-radius:8px}.scrollbar-inner>.scroll-element .scroll-bar,.scrollbar-inner>.scroll-element .scroll-element_track{-ms-filter:\"progid:DXImageTransform.Microsoft.Alpha(Opacity=40)\";filter:alpha(opacity=40);opacity:.4}.scrollbar-inner>.scroll-element .scroll-element_track{background-color:#e0e0e0}.scrollbar-inner>.scroll-element .scroll-bar{background-color:#c2c2c2}.scrollbar-inner>.scroll-element.scroll-draggable .scroll-bar,.scrollbar-inner>.scroll-element:hover .scroll-bar{background-color:#919191}.scrollbar-inner>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-12px}.scrollbar-inner>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-12px}.scrollbar-inner>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-12px}.scrollbar-inner>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-12px}.scrollbar-outer>.scroll-element,.scrollbar-outer>.scroll-element div{border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-outer>.scroll-element{background-color:#fff}.scrollbar-outer>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-outer>.scroll-element.scroll-x{bottom:0;height:12px;left:0;width:100%}.scrollbar-outer>.scroll-element.scroll-y{height:100%;right:0;top:0;width:12px}.scrollbar-outer>.scroll-element.scroll-x .scroll-element_outer{height:8px;top:2px}.scrollbar-outer>.scroll-element.scroll-y .scroll-element_outer{left:2px;width:8px}.scrollbar-outer>.scroll-element .scroll-element_outer{overflow:hidden}.scrollbar-outer>.scroll-element .scroll-element_track{background-color:#eee}.scrollbar-outer>.scroll-element .scroll-bar,.scrollbar-outer>.scroll-element .scroll-element_outer,.scrollbar-outer>.scroll-element .scroll-element_track{border-radius:8px}.scrollbar-outer>.scroll-element .scroll-bar{background-color:#d9d9d9}.scrollbar-outer>.scroll-element .scroll-bar:hover{background-color:#c2c2c2}.scrollbar-outer>.scroll-element.scroll-draggable .scroll-bar{background-color:#919191}.scrollbar-outer>.scroll-content.scroll-scrolly_visible{left:-12px;margin-left:12px}.scrollbar-outer>.scroll-content.scroll-scrollx_visible{top:-12px;margin-top:12px}.scrollbar-outer>.scroll-element.scroll-x .scroll-bar{min-width:10px}.scrollbar-outer>.scroll-element.scroll-y .scroll-bar{min-height:10px}.scrollbar-outer>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-14px}.scrollbar-outer>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-14px}.scrollbar-outer>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-14px}.scrollbar-outer>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-14px}.scrollbar-macosx>.scroll-element,.scrollbar-macosx>.scroll-element div{background:0 0;border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-macosx>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-macosx>.scroll-element .scroll-element_track{display:none}.scrollbar-macosx>.scroll-element .scroll-bar{background-color:#6C6E71;display:block;-ms-filter:\"progid:DXImageTransform.Microsoft.Alpha(Opacity=0)\";filter:alpha(opacity=0);opacity:0;border-radius:7px;transition:opacity .2s linear}.scrollbar-macosx:hover>.scroll-element .scroll-bar,.scrollbar-macosx>.scroll-element.scroll-draggable .scroll-bar{-ms-filter:\"progid:DXImageTransform.Microsoft.Alpha(Opacity=70)\";filter:alpha(opacity=70);opacity:.7}.scrollbar-macosx>.scroll-element.scroll-x{bottom:0;height:0;left:0;min-width:100%;overflow:visible;width:100%}.scrollbar-macosx>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:0}.scrollbar-macosx>.scroll-element.scroll-x .scroll-bar{height:7px;min-width:10px;top:-9px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-bar{left:-9px;min-height:10px;width:7px}.scrollbar-macosx>.scroll-element.scroll-x .scroll-element_outer{left:2px}.scrollbar-macosx>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-element_outer{top:2px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-macosx>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-11px}.scrollbar-macosx>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-11px}.scrollbar-light>.scroll-element,.scrollbar-light>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-light>.scroll-element{background-color:#fff}.scrollbar-light>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-light>.scroll-element .scroll-element_outer{border-radius:10px}.scrollbar-light>.scroll-element .scroll-element_size{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiA/Pgo8c3ZnIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgd2lkdGg9IjEwMCUiIGhlaWdodD0iMTAwJSIgdmlld0JveD0iMCAwIDEgMSIgcHJlc2VydmVBc3BlY3RSYXRpbz0ibm9uZSI+CiAgPGxpbmVhckdyYWRpZW50IGlkPSJncmFkLXVjZ2ctZ2VuZXJhdGVkIiBncmFkaWVudFVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeDE9IjAlIiB5MT0iMCUiIHgyPSIxMDAlIiB5Mj0iMCUiPgogICAgPHN0b3Agb2Zmc2V0PSIwJSIgc3RvcC1jb2xvcj0iI2RiZGJkYiIgc3RvcC1vcGFjaXR5PSIxIi8+CiAgICA8c3RvcCBvZmZzZXQ9IjEwMCUiIHN0b3AtY29sb3I9IiNlOGU4ZTgiIHN0b3Atb3BhY2l0eT0iMSIvPgogIDwvbGluZWFyR3JhZGllbnQ+CiAgPHJlY3QgeD0iMCIgeT0iMCIgd2lkdGg9IjEiIGhlaWdodD0iMSIgZmlsbD0idXJsKCNncmFkLXVjZ2ctZ2VuZXJhdGVkKSIgLz4KPC9zdmc+);background:linear-gradient(to right,#dbdbdb 0,#e8e8e8 100%);border-radius:10px}.scrollbar-light>.scroll-element.scroll-x{bottom:0;height:17px;left:0;min-width:100%;width:100%}.scrollbar-light>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:17px}.scrollbar-light>.scroll-element .scroll-bar{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiA/Pgo8c3ZnIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgd2lkdGg9IjEwMCUiIGhlaWdodD0iMTAwJSIgdmlld0JveD0iMCAwIDEgMSIgcHJlc2VydmVBc3BlY3RSYXRpbz0ibm9uZSI+CiAgPGxpbmVhckdyYWRpZW50IGlkPSJncmFkLXVjZ2ctZ2VuZXJhdGVkIiBncmFkaWVudFVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeDE9IjAlIiB5MT0iMCUiIHgyPSIxMDAlIiB5Mj0iMCUiPgogICAgPHN0b3Agb2Zmc2V0PSIwJSIgc3RvcC1jb2xvcj0iI2ZlZmVmZSIgc3RvcC1vcGFjaXR5PSIxIi8+CiAgICA8c3RvcCBvZmZzZXQ9IjEwMCUiIHN0b3AtY29sb3I9IiNmNWY1ZjUiIHN0b3Atb3BhY2l0eT0iMSIvPgogIDwvbGluZWFyR3JhZGllbnQ+CiAgPHJlY3QgeD0iMCIgeT0iMCIgd2lkdGg9IjEiIGhlaWdodD0iMSIgZmlsbD0idXJsKCNncmFkLXVjZ2ctZ2VuZXJhdGVkKSIgLz4KPC9zdmc+);background:linear-gradient(to right,#fefefe 0,#f5f5f5 100%);border:1px solid #dbdbdb;border-radius:10px}.scrollbar-light>.scroll-content.scroll-scrolly_visible{left:-17px;margin-left:17px}.scrollbar-light>.scroll-content.scroll-scrollx_visible{top:-17px;margin-top:17px}.scrollbar-light>.scroll-element.scroll-x .scroll-bar{height:10px;min-width:10px;top:0}.scrollbar-light>.scroll-element.scroll-y .scroll-bar{left:0;min-height:10px;width:10px}.scrollbar-light>.scroll-element.scroll-x .scroll-element_outer{height:12px;left:2px;top:2px}.scrollbar-light>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-light>.scroll-element.scroll-y .scroll-element_outer{left:2px;top:2px;width:12px}.scrollbar-light>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-light>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-19px}.scrollbar-light>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-19px}.scrollbar-light>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-19px}.scrollbar-light>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-19px}.scrollbar-rail>.scroll-element,.scrollbar-rail>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-rail>.scroll-element{background-color:#fff}.scrollbar-rail>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-rail>.scroll-element .scroll-element_size{background-color:#999;background-color:rgba(0,0,0,.3)}.scrollbar-rail>.scroll-element .scroll-element_outer:hover .scroll-element_size{background-color:#666;background-color:rgba(0,0,0,.5)}.scrollbar-rail>.scroll-element.scroll-x{bottom:0;height:12px;left:0;min-width:100%;padding:3px 0 2px;width:100%}.scrollbar-rail>.scroll-element.scroll-y{height:100%;min-height:100%;padding:0 2px 0 3px;right:0;top:0;width:12px}.scrollbar-rail>.scroll-element .scroll-bar{background-color:#d0b9a0;border-radius:2px;box-shadow:1px 1px 3px rgba(0,0,0,.5)}.scrollbar-rail>.scroll-element .scroll-element_outer:hover .scroll-bar{box-shadow:1px 1px 3px rgba(0,0,0,.6)}.scrollbar-rail>.scroll-content.scroll-scrolly_visible{left:-17px;margin-left:17px}.scrollbar-rail>.scroll-content.scroll-scrollx_visible{margin-top:17px;top:-17px}.scrollbar-rail>.scroll-element.scroll-x .scroll-bar{height:10px;min-width:10px;top:1px}.scrollbar-rail>.scroll-element.scroll-y .scroll-bar{left:1px;min-height:10px;width:10px}.scrollbar-rail>.scroll-element.scroll-x .scroll-element_outer{height:15px;left:5px}.scrollbar-rail>.scroll-element.scroll-x .scroll-element_size{height:2px;left:-10px;top:5px}.scrollbar-rail>.scroll-element.scroll-y .scroll-element_outer{top:5px;width:15px}.scrollbar-rail>.scroll-element.scroll-y .scroll-element_size{left:5px;top:-10px;width:2px}.scrollbar-rail>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-25px}.scrollbar-rail>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-25px}.scrollbar-rail>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-25px}.scrollbar-rail>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-25px}.scrollbar-dynamic>.scroll-element,.scrollbar-dynamic>.scroll-element div{background:0 0;border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-dynamic>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-dynamic>.scroll-element.scroll-x{bottom:2px;height:7px;left:0;min-width:100%;width:100%}.scrollbar-dynamic>.scroll-element.scroll-y{height:100%;min-height:100%;right:2px;top:0;width:7px}.scrollbar-dynamic>.scroll-element .scroll-element_outer{opacity:.3;border-radius:12px}.scrollbar-dynamic>.scroll-element .scroll-element_size{background-color:#ccc;opacity:0;border-radius:12px;transition:opacity .2s}.scrollbar-dynamic>.scroll-element .scroll-bar{background-color:#6c6e71;border-radius:7px}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-bar{bottom:0;height:7px;min-width:24px;top:auto}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-bar{left:auto;min-height:24px;right:0;width:7px}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-element_outer{bottom:0;top:auto;left:2px;transition:height .2s}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-element_outer{left:auto;right:0;top:2px;transition:width .2s}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-dynamic>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-11px}.scrollbar-dynamic>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-11px}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer{overflow:hidden;-ms-filter:\"progid:DXImageTransform.Microsoft.Alpha(Opacity=70)\";filter:alpha(opacity=70);opacity:.7}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer .scroll-element_size,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer .scroll-element_size{opacity:1}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer .scroll-bar,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer .scroll-bar{height:100%;width:100%;border-radius:12px}.scrollbar-dynamic>.scroll-element.scroll-x.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element.scroll-x:hover .scroll-element_outer{height:20px;min-height:7px}.scrollbar-dynamic>.scroll-element.scroll-y.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element.scroll-y:hover .scroll-element_outer{min-width:7px;width:20px}.scrollbar-chrome>.scroll-element,.scrollbar-chrome>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-chrome>.scroll-element{background-color:#fff}.scrollbar-chrome>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-chrome>.scroll-element .scroll-element_track{background:#f1f1f1;border:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-x{bottom:0;height:16px;left:0;min-width:100%;width:100%}.scrollbar-chrome>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:16px}.scrollbar-chrome>.scroll-element .scroll-bar{background-color:#d9d9d9;border:1px solid #bdbdbd;cursor:default;border-radius:2px}.scrollbar-chrome>.scroll-element .scroll-bar:hover{background-color:#c2c2c2;border-color:#a9a9a9}.scrollbar-chrome>.scroll-element.scroll-draggable .scroll-bar{background-color:#919191;border-color:#7e7e7e}.scrollbar-chrome>.scroll-content.scroll-scrolly_visible{left:-16px;margin-left:16px}.scrollbar-chrome>.scroll-content.scroll-scrollx_visible{top:-16px;margin-top:16px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-bar{height:8px;min-width:10px;top:3px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-bar{left:3px;min-height:10px;width:8px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_outer{border-left:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_track{height:14px;left:-3px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_size{height:14px;left:-4px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_outer{border-top:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_track{top:-3px;width:14px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_size{top:-4px;width:14px}.scrollbar-chrome>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-19px}.scrollbar-chrome>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-19px}.scrollbar-chrome>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-19px}.scrollbar-chrome>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-19px}\n" +
"/**\n" +
" * Minified style.\n" +
" * Original filename: \\src\\styles\\style.css\n" +
" */\n" +
"@-webkit-keyframes spin{100%{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}@keyframes spin{100%{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}#emote-menu-button{background-image:url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABIAAAAQCAYAAAAbBi9cAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAKUSURBVDhPfZTNi1JRGMZvMIsWUZts5SIXFYK0CME/IGghxVC7WUoU1NBixI+mRSD4MQzmxziKO3XUBhRmUGZKdBG40XEGU6d0GFGZcT4qxW1hi7fzvNwZqKwDD5z7vs/vueeee+6VMJxO5wUhhdvtfuHz+T4tLS2NhegfGsMDLxiwHIIhLi57PJ75VCr1Y39/n4bDIY1Go4lCDx54wYCVYzjoVjQa/dxutyfCkwSvYJpgOSQf708tuBa1yWRy/L+V/Cl4wYBFhhTxfLhum/esiiJ1u12KRCJksVhofX2dTk5OzkHMUUMPHnjB2F55VpEhPde/Lbx8FqBEIkHpdJoMBgNptVrS6XRUqVTOg7a3t2lmZob0ej2p1Wr2ggGLDOnJ3QSZH4coHo/TysoKhygUCtJoNFQsFmkwGLAwR7hSqSSVSsVeMGCRIT29F6fXJi8Xy+Uymc1mmp6eJofDQfV6nU5PT1mY2+127uHxSqUSh4FFhhQLvrvtcrm+YpkHBwdUrVZpa2uLarUadTodOjw8ZGGOGnrwwAsGLDLw1i4uLrzRYeOOj49pb2+Pdnd3qdVq8StGAIQ5ao1Ggz3wggGLDD4C4izcEcWfR0dHbMrlcrSxscGbjVAIK8lms7S5ucmB/X6fXz9YDsEQFzdjsVit2Wzyqc1kMrwfVquVjEYjzc3NkclkIpvNRmtra+yBVzAfBXtDjuGgS8FgcFbc8QvuhjNSKBQoFAqR6LFEn/L5PPfggXd5eXkWrBzDQdC1QCBgFoeut7Ozw/tyBp2FQzhPwtOFFwzY34Yo4A9wRXzdD8LhcE48wncE9no9Fuaoid574bkPLxgZ/3uI5pTQVfFlP/L7/Wmhb7JSXq/3IXrwyHZ5SNIvGCnqyh+J7+gAAAAASUVORK5CYII=)!important;background-position:50%;background-repeat:no-repeat;cursor:pointer;height:30px;width:30px}#emote-menu-button:focus{box-shadow:none}#emote-menu-button.active{box-shadow:0 0 6px 0 #7d5bbe,inset 0 0 0 1px rgba(100,65,164,.5)}.emote-menu{padding:5px;z-index:1000;display:none;background-color:#202020;position:absolute}.emote-menu a{color:#fff}.emote-menu a:hover{cursor:pointer;text-decoration:underline;color:#ccc}.emote-menu .emotes-starred{height:38px}.emote-menu .draggable{background-image:repeating-linear-gradient(45deg,transparent,transparent 5px,rgba(255,255,255,.05) 5px,rgba(255,255,255,.05) 10px);cursor:move;height:7px;margin-bottom:3px}.emote-menu .draggable:hover{background-image:repeating-linear-gradient(45deg,transparent,transparent 5px,rgba(255,255,255,.1) 5px,rgba(255,255,255,.1) 10px)}.emote-menu .header-info{border-top:1px solid #000;box-shadow:0 1px 0 rgba(255,255,255,.05) inset;background-image:linear-gradient(to top,transparent,rgba(0,0,0,.5));padding:2px;color:#ddd;text-align:center;position:relative}.emote-menu .header-info img{margin-right:8px}.emote-menu .emote{display:inline-block;padding:2px;margin:1px;cursor:pointer;border-radius:5px;text-align:center;position:relative;width:30px;height:30px;transition:all .25s ease;border:1px solid transparent}.emote-menu.editing .emote{cursor:auto}.emote-menu .emote img{max-width:100%;max-height:100%;margin:auto;position:absolute;top:0;bottom:0;left:0;right:0}.emote-menu .single-row .emote-container{overflow:hidden;height:37px}.emote-menu .single-row .emote{display:inline-block;margin-bottom:100px}.emote-menu .emote:hover{background-color:rgba(255,255,255,.1)}.emote-menu .pull-left{float:left}.emote-menu .pull-right{float:right}.emote-menu .footer{text-align:center;border-top:1px solid #000;box-shadow:0 1px 0 rgba(255,255,255,.05) inset;padding:5px 0 2px;margin-top:5px;height:18px}.emote-menu .footer .pull-left{margin-right:5px}.emote-menu .footer .pull-right{margin-left:5px}.emote-menu .icon{height:16px;width:16px;opacity:.5;background-size:contain!important}.emote-menu .icon:hover{opacity:1}.emote-menu .icon-home{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNjQiDQogICBoZWlnaHQ9IjY0Ig0KICAgdmlld0JveD0iMCAwIDY0IDY0Ig0KICAgaWQ9IkNhcGFfMSINCiAgIHhtbDpzcGFjZT0icHJlc2VydmUiPjxtZXRhZGF0YQ0KICAgaWQ9Im1ldGFkYXRhMzAwMSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczI5OTkiIC8+DQo8cGF0aA0KICAgZD0ibSA1Ny4wNjIsMzEuMzk4IGMgMC45MzIsLTEuMDI1IDAuODQyLC0yLjU5NiAtMC4yMDEsLTMuNTA4IEwgMzMuODg0LDcuNzg1IEMgMzIuODQxLDYuODczIDMxLjE2OSw2Ljg5MiAzMC4xNDgsNy44MjggTCA3LjA5MywyOC45NjIgYyAtMS4wMjEsMC45MzYgLTEuMDcxLDIuNTA1IC0wLjExMSwzLjUwMyBsIDAuNTc4LDAuNjAyIGMgMC45NTksMC45OTggMi41MDksMS4xMTcgMy40NiwwLjI2NSBsIDEuNzIzLC0xLjU0MyB2IDIyLjU5IGMgMCwxLjM4NiAxLjEyMywyLjUwOCAyLjUwOCwyLjUwOCBoIDguOTg3IGMgMS4zODUsMCAyLjUwOCwtMS4xMjIgMi41MDgsLTIuNTA4IFYgMzguNTc1IGggMTEuNDYzIHYgMTUuODA0IGMgLTAuMDIsMS4zODUgMC45NzEsMi41MDcgMi4zNTYsMi41MDcgaCA5LjUyNCBjIDEuMzg1LDAgMi41MDgsLTEuMTIyIDIuNTA4LC0yLjUwOCBWIDMyLjEwNyBjIDAsMCAwLjQ3NiwwLjQxNyAxLjA2MywwLjkzMyAwLjU4NiwwLjUxNSAxLjgxNywwLjEwMiAyLjc0OSwtMC45MjQgbCAwLjY1MywtMC43MTggeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNmZmZmZmY7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4=) 50% no-repeat}.emote-menu .icon-gear{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMjEuNTkiDQogICBoZWlnaHQ9IjIxLjEzNjk5OSINCiAgIHZpZXdCb3g9IjAgMCAyMS41OSAyMS4xMzciDQogICBpZD0iQ2FwYV8xIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGEzOSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczM3IiAvPg0KPHBhdGgNCiAgIGQ9Ik0gMTguNjIyLDguMTQ1IDE4LjA3Nyw2Ljg1IGMgMCwwIDEuMjY4LC0yLjg2MSAxLjE1NiwtMi45NzEgTCAxNy41NTQsMi4yNCBDIDE3LjQzOCwyLjEyNyAxNC41NzYsMy40MzMgMTQuNTc2LDMuNDMzIEwgMTMuMjU2LDIuOSBDIDEzLjI1NiwyLjkgMTIuMDksMCAxMS45MywwIEggOS41NjEgQyA5LjM5NiwwIDguMzE3LDIuOTA2IDguMzE3LDIuOTA2IEwgNi45OTksMy40NDEgYyAwLDAgLTIuOTIyLC0xLjI0MiAtMy4wMzQsLTEuMTMxIEwgMi4yODksMy45NTEgQyAyLjE3Myw0LjA2NCAzLjUwNyw2Ljg2NyAzLjUwNyw2Ljg2NyBMIDIuOTYyLDguMTYgQyAyLjk2Miw4LjE2IDAsOS4zMDEgMCw5LjQ1NSB2IDIuMzIyIGMgMCwwLjE2MiAyLjk2OSwxLjIxOSAyLjk2OSwxLjIxOSBsIDAuNTQ1LDEuMjkxIGMgMCwwIC0xLjI2OCwyLjg1OSAtMS4xNTcsMi45NjkgbCAxLjY3OCwxLjY0MyBjIDAuMTE0LDAuMTExIDIuOTc3LC0xLjE5NSAyLjk3NywtMS4xOTUgbCAxLjMyMSwwLjUzNSBjIDAsMCAxLjE2NiwyLjg5OCAxLjMyNywyLjg5OCBoIDIuMzY5IGMgMC4xNjQsMCAxLjI0NCwtMi45MDYgMS4yNDQsLTIuOTA2IGwgMS4zMjIsLTAuNTM1IGMgMCwwIDIuOTE2LDEuMjQyIDMuMDI5LDEuMTMzIGwgMS42NzgsLTEuNjQxIGMgMC4xMTcsLTAuMTE1IC0xLjIyLC0yLjkxNiAtMS4yMiwtMi45MTYgbCAwLjU0NCwtMS4yOTMgYyAwLDAgMi45NjMsLTEuMTQzIDIuOTYzLC0xLjI5OSBWIDkuMzYgQyAyMS41OSw5LjE5OSAxOC42MjIsOC4xNDUgMTguNjIyLDguMTQ1IHogbSAtNC4zNjYsMi40MjMgYyAwLDEuODY3IC0xLjU1MywzLjM4NyAtMy40NjEsMy4zODcgLTEuOTA2LDAgLTMuNDYxLC0xLjUyIC0zLjQ2MSwtMy4zODcgMCwtMS44NjcgMS41NTUsLTMuMzg1IDMuNDYxLC0zLjM4NSAxLjkwOSwwLjAwMSAzLjQ2MSwxLjUxOCAzLjQ2MSwzLjM4NSB6Ig0KICAgaWQ9InBhdGgzIg0KICAgc3R5bGU9ImZpbGw6I0ZGRkZGRiIgLz4NCjxnDQogICBpZD0iZzUiPg0KPC9nPg0KPGcNCiAgIGlkPSJnNyI+DQo8L2c+DQo8Zw0KICAgaWQ9Imc5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzExIj4NCjwvZz4NCjxnDQogICBpZD0iZzEzIj4NCjwvZz4NCjxnDQogICBpZD0iZzE1Ij4NCjwvZz4NCjxnDQogICBpZD0iZzE3Ij4NCjwvZz4NCjxnDQogICBpZD0iZzE5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzIxIj4NCjwvZz4NCjxnDQogICBpZD0iZzIzIj4NCjwvZz4NCjxnDQogICBpZD0iZzI1Ij4NCjwvZz4NCjxnDQogICBpZD0iZzI3Ij4NCjwvZz4NCjxnDQogICBpZD0iZzI5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzMxIj4NCjwvZz4NCjxnDQogICBpZD0iZzMzIj4NCjwvZz4NCjwvc3ZnPg0K) 50% no-repeat}.emote-menu.editing .icon-gear{-webkit-animation:spin 4s linear infinite;animation:spin 4s linear infinite}.emote-menu .icon-resize-handle{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTYiDQogICBoZWlnaHQ9IjE2Ig0KICAgdmlld0JveD0iMCAwIDE2IDE2Ig0KICAgaWQ9IkNhcGFfMSINCiAgIHhtbDpzcGFjZT0icHJlc2VydmUiPjxtZXRhZGF0YQ0KICAgaWQ9Im1ldGFkYXRhNDM1NyI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczQzNTUiIC8+DQo8cGF0aA0KICAgZD0iTSAxMy41LDggQyAxMy4yMjUsOCAxMyw4LjIyNCAxMyw4LjUgdiAzLjc5MyBMIDMuNzA3LDMgSCA3LjUgQyA3Ljc3NiwzIDgsMi43NzYgOCwyLjUgOCwyLjIyNCA3Ljc3NiwyIDcuNSwyIGggLTUgTCAyLjMwOSwyLjAzOSAyLjE1LDIuMTQ0IDIuMTQ2LDIuMTQ2IDIuMTQzLDIuMTUyIDIuMDM5LDIuMzA5IDIsMi41IHYgNSBDIDIsNy43NzYgMi4yMjQsOCAyLjUsOCAyLjc3Niw4IDMsNy43NzYgMyw3LjUgViAzLjcwNyBMIDEyLjI5MywxMyBIIDguNSBDIDguMjI0LDEzIDgsMTMuMjI1IDgsMTMuNSA4LDEzLjc3NSA4LjIyNCwxNCA4LjUsMTQgaCA1IGwgMC4xOTEsLTAuMDM5IGMgMC4xMjEsLTAuMDUxIDAuMjIsLTAuMTQ4IDAuMjcsLTAuMjcgTCAxNCwxMy41MDIgViA4LjUgQyAxNCw4LjIyNCAxMy43NzUsOCAxMy41LDggeiINCiAgIGlkPSJwYXRoNDM1MSINCiAgIHN0eWxlPSJmaWxsOiNmZmZmZmY7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4=) 50% no-repeat;cursor:nwse-resize!important}.emote-menu .icon-pin{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTYiDQogICBoZWlnaHQ9IjE2Ig0KICAgaWQ9InN2ZzMwMDUiPg0KICA8bWV0YWRhdGENCiAgICAgaWQ9Im1ldGFkYXRhMzAyMyI+DQogICAgPHJkZjpSREY+DQogICAgICA8Y2M6V29yaw0KICAgICAgICAgcmRmOmFib3V0PSIiPg0KICAgICAgICA8ZGM6Zm9ybWF0PmltYWdlL3N2Zyt4bWw8L2RjOmZvcm1hdD4NCiAgICAgICAgPGRjOnR5cGUNCiAgICAgICAgICAgcmRmOnJlc291cmNlPSJodHRwOi8vcHVybC5vcmcvZGMvZGNtaXR5cGUvU3RpbGxJbWFnZSIgLz4NCiAgICAgICAgPGRjOnRpdGxlPjwvZGM6dGl0bGU+DQogICAgICA8L2NjOldvcms+DQogICAgPC9yZGY6UkRGPg0KICA8L21ldGFkYXRhPg0KICA8ZGVmcw0KICAgICBpZD0iZGVmczMwMjEiIC8+DQogIDxnDQogICAgIHRyYW5zZm9ybT0ibWF0cml4KDAuNzkzMDc4MiwwLDAsMC43OTMwNzgyLC0yLjE3MDk4NSwtODE0LjY5Mjk5KSINCiAgICAgaWQ9ImczMDA3Ij4NCiAgICA8Zw0KICAgICAgIHRyYW5zZm9ybT0ibWF0cml4KDAuNzA3MTEsMC43MDcxMSwtMC43MDcxMSwwLjcwNzExLDczNy43MDc1NSwyOTUuNDg4MDgpIg0KICAgICAgIGlkPSJnMzAwOSI+DQogICAgICA8Zw0KICAgICAgICAgaWQ9ImczNzU1Ij4NCiAgICAgICAgPHBhdGgNCiAgICAgICAgICAgZD0iTSA5Ljc4MTI1LDAgQyA5LjQ3NDA1NjIsMC42ODkxMTIgOS41MjA2OCwxLjUyMzA4NTMgOS4zMTI1LDIuMTg3NSBMIDQuOTM3NSw2LjU5Mzc1IEMgMy45NTg5NjA4LDYuNDI5NDgzIDIuOTQ3NzU0OCw2LjUzMjc4OTkgMiw2LjgxMjUgTCA1LjAzMTI1LDkuODQzNzUgMC41NjI1LDE0LjMxMjUgMCwxNiBDIDAuNTY5Mjk2MjgsMTUuNzk1NjI2IDEuMTY3NzM3OCwxNS42NDAyMzcgMS43MTg3NSwxNS40MDYyNSBMIDYuMTU2MjUsMTAuOTY4NzUgOS4xODc1LDE0IGMgMC4yNzk2ODIzLC0wLjk0Nzc4MyAwLjM4MzE1MjgsLTEuOTU4OTM3IDAuMjE4NzUsLTIuOTM3NSAxLjUwMDAxMSwtMS40ODk1Nzk4IDMuMDAwMDAxLC0yLjk3OTE1OSA0LjUsLTQuNDY4NzUgMC42MDExMDIsLTAuMDMxMzYxIDEuODIyMTM4LC0wLjA5NjEzNyAyLC0wLjQ2ODc1IEMgMTMuODc5ODkyLDQuMDY5NDgwMyAxMS44NDI4NjUsMi4wMjAyMjgyIDkuNzgxMjUsMCB6Ig0KICAgICAgICAgICB0cmFuc2Zvcm09Im1hdHJpeCgwLjg5MTU5Mzc0LC0wLjg5MTU5Mzc0LDAuODkxNTkzNzQsMC44OTE1OTM3NCwtMi4yNjU1LDEwMzcuMTM0NSkiDQogICAgICAgICAgIGlkPSJwYXRoMzAxMSINCiAgICAgICAgICAgc3R5bGU9ImZpbGw6I2ZmZmZmZjtmaWxsLW9wYWNpdHk6MSIgLz4NCiAgICAgIDwvZz4NCiAgICA8L2c+DQogIDwvZz4NCjwvc3ZnPg0K) 50% no-repeat;transition:all .25s ease}.emote-menu .icon-pin:hover,.emote-menu.pinned .icon-pin{-webkit-transform:rotate(-45deg);transform:rotate(-45deg);opacity:1}.emote-menu .edit-tool{background-position:50%;background-repeat:no-repeat;background-size:14px;border-radius:4px;border:1px solid #000;cursor:pointer;display:none;height:14px;opacity:.25;position:absolute;transition:all .25s ease;width:14px;z-index:1}.emote-menu .edit-tool:hover,.emote-menu .emote:hover .edit-tool{opacity:1}.emote-menu .edit-visibility{background-color:#00c800;background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTAwIg0KICAgaGVpZ2h0PSIxMDAiDQogICB2aWV3Qm94PSIwIDAgMTAwIDEwMCINCiAgIGlkPSJMYXllcl8xIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGE5Ij48cmRmOlJERj48Y2M6V29yaw0KICAgICAgIHJkZjphYm91dD0iIj48ZGM6Zm9ybWF0PmltYWdlL3N2Zyt4bWw8L2RjOmZvcm1hdD48ZGM6dHlwZQ0KICAgICAgICAgcmRmOnJlc291cmNlPSJodHRwOi8vcHVybC5vcmcvZGMvZGNtaXR5cGUvU3RpbGxJbWFnZSIgLz48ZGM6dGl0bGU+PC9kYzp0aXRsZT48L2NjOldvcms+PC9yZGY6UkRGPjwvbWV0YWRhdGE+PGRlZnMNCiAgIGlkPSJkZWZzNyIgLz4NCjxwYXRoDQogICBkPSJNIDk3Ljk2NCw0Ni41NDggQyA5Ny4wOTgsNDUuNTI4IDc2LjQyNywyMS42MDMgNTAsMjEuNjAzIGMgLTI2LjQyNywwIC00Ny4wOTgsMjMuOTI1IC00Ny45NjUsMjQuOTQ2IC0xLjcwMSwyIC0xLjcwMSw0LjkwMiAxMGUtNCw2LjkwMyAwLjg2NiwxLjAyIDIxLjUzNywyNC45NDUgNDcuOTY0LDI0Ljk0NSAyNi40MjcsMCA0Ny4wOTgsLTIzLjkyNiA0Ny45NjUsLTI0Ljk0NiAxLjcwMSwtMiAxLjcwMSwtNC45MDIgLTAuMDAxLC02LjkwMyB6IE0gNTguMDczLDM1Ljk3NSBjIDEuNzc3LC0wLjk3IDQuMjU1LDAuMTQzIDUuNTM0LDIuNDg1IDEuMjc5LDIuMzQzIDAuODc1LDUuMDI5IC0wLjkwMiw1Ljk5OSAtMS43NzcsMC45NzEgLTQuMjU1LC0wLjE0MyAtNS41MzUsLTIuNDg1IC0xLjI3OSwtMi4zNDMgLTAuODc1LC01LjAyOSAwLjkwMywtNS45OTkgeiBNIDUwLDY5LjcyOSBDIDMxLjU0LDY5LjcyOSAxNi4wMDUsNTUuNTUzIDEwLjYyOCw1MCAxNC4yNTksNDYuMjQ5IDIyLjUyNiwzOC41NzEgMzMuMTk1LDMzLjk3OSAzMS4xMTQsMzcuMTQ1IDI5Ljg5NCw0MC45MjggMjkuODk0LDQ1IGMgMCwxMS4xMDQgOS4wMDEsMjAuMTA1IDIwLjEwNSwyMC4xMDUgMTEuMTA0LDAgMjAuMTA2LC05LjAwMSAyMC4xMDYsLTIwLjEwNSAwLC00LjA3MiAtMS4yMTksLTcuODU1IC0zLjMsLTExLjAyMSBDIDc3LjQ3NCwzOC41NzIgODUuNzQxLDQ2LjI1IDg5LjM3Miw1MCA4My45OTUsNTUuNTU1IDY4LjQ2LDY5LjcyOSA1MCw2OS43MjkgeiINCiAgIGlkPSJwYXRoMyIgLz4NCjwvc3ZnPg==)}.emote-menu .edit-starred{background-color:#323232;background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNTAiDQogICBoZWlnaHQ9IjUwIg0KICAgdmlld0JveD0iMCAwIDUwIDUwIg0KICAgaWQ9IkxheWVyXzEiDQogICB4bWw6c3BhY2U9InByZXNlcnZlIj48bWV0YWRhdGENCiAgIGlkPSJtZXRhZGF0YTMwMDEiPjxyZGY6UkRGPjxjYzpXb3JrDQogICAgICAgcmRmOmFib3V0PSIiPjxkYzpmb3JtYXQ+aW1hZ2Uvc3ZnK3htbDwvZGM6Zm9ybWF0PjxkYzp0eXBlDQogICAgICAgICByZGY6cmVzb3VyY2U9Imh0dHA6Ly9wdXJsLm9yZy9kYy9kY21pdHlwZS9TdGlsbEltYWdlIiAvPjxkYzp0aXRsZT48L2RjOnRpdGxlPjwvY2M6V29yaz48L3JkZjpSREY+PC9tZXRhZGF0YT48ZGVmcw0KICAgaWQ9ImRlZnMyOTk5IiAvPg0KPHBhdGgNCiAgIGQ9Im0gNDMuMDQsMjIuNjk2IC03LjU2OCw3LjM3NyAxLjc4NywxMC40MTcgYyAwLjEyNywwLjc1IC0wLjE4MiwxLjUwOSAtMC43OTcsMS45NTcgLTAuMzQ4LDAuMjUzIC0wLjc2MiwwLjM4MiAtMS4xNzYsMC4zODIgLTAuMzE4LDAgLTAuNjM4LC0wLjA3NiAtMC45MzEsLTAuMjMgTCAyNSwzNy42ODEgMTUuNjQ1LDQyLjU5OSBjIC0wLjY3NCwwLjM1NSAtMS40OSwwLjI5NSAtMi4xMDcsLTAuMTUxIEMgMTIuOTIzLDQyIDEyLjYxNCw0MS4yNDIgMTIuNzQzLDQwLjQ5MSBMIDE0LjUzLDMwLjA3NCA2Ljk2MiwyMi42OTcgQyA2LjQxNSwyMi4xNjYgNi4yMjEsMjEuMzcxIDYuNDU0LDIwLjY0NyA2LjY5LDE5LjkyMyA3LjMxNSwxOS4zOTYgOC4wNjksMTkuMjg2IGwgMTAuNDU5LC0xLjUyMSA0LjY4LC05LjQ3OCBDIDIzLjU0Myw3LjYwMyAyNC4yMzksNy4xNzEgMjUsNy4xNzEgYyAwLjc2MywwIDEuNDU2LDAuNDMyIDEuNzkzLDEuMTE1IGwgNC42NzksOS40NzggMTAuNDYxLDEuNTIxIGMgMC43NTIsMC4xMDkgMS4zNzksMC42MzcgMS42MTIsMS4zNjEgMC4yMzcsMC43MjQgMC4wMzgsMS41MTkgLTAuNTA1LDIuMDUgeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNjY2NjY2M7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4NCg==)}.emote-menu .emote>.edit-visibility{bottom:auto;left:auto;right:0;top:0}.emote-menu .emote>.edit-starred{bottom:auto;left:0;right:auto;top:0}.emote-menu .header-info>.edit-tool{margin-left:5px}.emote-menu.editing .edit-tool{display:inline-block}.emote-menu .emote-menu-hidden .edit-visibility{background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTAwIg0KICAgaGVpZ2h0PSIxMDAiDQogICB2aWV3Qm94PSIwIDAgMTAwIDEwMCINCiAgIGlkPSJMYXllcl8zIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGExNSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczEzIiAvPg0KPGcNCiAgIGlkPSJnMyI+DQoJPHBhdGgNCiAgIGQ9Ik0gNzAuMDgyLDQ1LjQ3NSA1MC40NzQsNjUuMDgyIEMgNjEuMTk4LDY0LjgzMSA2OS44MzEsNTYuMTk3IDcwLjA4Miw0NS40NzUgeiINCiAgIGlkPSJwYXRoNSINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQoJPHBhdGgNCiAgIGQ9Im0gOTcuOTY0LDQ2LjU0OCBjIC0wLjQ1LC0wLjUyOSAtNi4yNDUsLTcuMjMgLTE1LjQwMywtMTMuNTU0IGwgLTYuMiw2LjIgQyA4Mi4zNTEsNDMuMTQ4IDg2LjkyLDQ3LjQ2OSA4OS4zNzIsNTAgODMuOTk1LDU1LjU1NSA2OC40Niw2OS43MjkgNTAsNjkuNzI5IGMgLTEuMzM0LDAgLTIuNjUxLC0wLjA4MiAtMy45NTIsLTAuMjIyIGwgLTcuNDM5LDcuNDM5IGMgMy42MzksMC45MDkgNy40NDksMS40NSAxMS4zOTEsMS40NSAyNi40MjcsMCA0Ny4wOTgsLTIzLjkyNiA0Ny45NjUsLTI0Ljk0NiAxLjcwMSwtMS45OTkgMS43MDEsLTQuOTAxIC0wLjAwMSwtNi45MDIgeiINCiAgIGlkPSJwYXRoNyINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQoJPHBhdGgNCiAgIGQ9Im0gOTEuNDExLDE2LjY2IGMgMCwtMC4yNjYgLTAuMTA1LC0wLjUyIC0wLjI5MywtMC43MDcgbCAtNy4wNzEsLTcuMDcgYyAtMC4zOTEsLTAuMzkxIC0xLjAyMywtMC4zOTEgLTEuNDE0LDAgTCA2Ni44MDQsMjQuNzExIEMgNjEuNjAyLDIyLjgxOCA1NS45NDksMjEuNjAzIDUwLDIxLjYwMyBjIC0yNi40MjcsMCAtNDcuMDk4LDIzLjkyNiAtNDcuOTY1LDI0Ljk0NiAtMS43MDEsMiAtMS43MDEsNC45MDIgMTBlLTQsNi45MDMgMC41MTcsMC42MDcgOC4wODMsOS4zNTQgMTkuNzA3LDE2LjMyIEwgOC44ODMsODIuNjMyIEMgOC42OTUsODIuODIgOC41OSw4My4wNzMgOC41OSw4My4zMzkgYyAwLDAuMjY2IDAuMTA1LDAuNTIgMC4yOTMsMC43MDcgbCA3LjA3MSw3LjA3IGMgMC4xOTUsMC4xOTUgMC40NTEsMC4yOTMgMC43MDcsMC4yOTMgMC4yNTYsMCAwLjUxMiwtMC4wOTggMC43MDcsLTAuMjkzIGwgNzMuNzUsLTczLjc1IGMgMC4xODcsLTAuMTg2IDAuMjkzLC0wLjQ0IDAuMjkzLC0wLjcwNiB6IE0gMTAuNjI4LDUwIEMgMTQuMjU5LDQ2LjI0OSAyMi41MjYsMzguNTcxIDMzLjE5NSwzMy45NzkgMzEuMTE0LDM3LjE0NSAyOS44OTQsNDAuOTI4IDI5Ljg5NCw0NSBjIDAsNC42NjUgMS42MDEsOC45NDUgNC4yNywxMi4zNTEgTCAyOC4wNCw2My40NzUgQyAxOS44ODgsNTguOTU1IDEzLjY0OSw1My4xMiAxMC42MjgsNTAgeiINCiAgIGlkPSJwYXRoOSINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQo8L2c+DQo8L3N2Zz4NCg==);background-color:red}.emote-menu .emote-menu-starred .edit-starred{background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNTAiDQogICBoZWlnaHQ9IjUwIg0KICAgdmlld0JveD0iMCAwIDUwIDUwIg0KICAgaWQ9IkxheWVyXzEiDQogICB4bWw6c3BhY2U9InByZXNlcnZlIj48bWV0YWRhdGENCiAgIGlkPSJtZXRhZGF0YTMwMDEiPjxyZGY6UkRGPjxjYzpXb3JrDQogICAgICAgcmRmOmFib3V0PSIiPjxkYzpmb3JtYXQ+aW1hZ2Uvc3ZnK3htbDwvZGM6Zm9ybWF0PjxkYzp0eXBlDQogICAgICAgICByZGY6cmVzb3VyY2U9Imh0dHA6Ly9wdXJsLm9yZy9kYy9kY21pdHlwZS9TdGlsbEltYWdlIiAvPjxkYzp0aXRsZT48L2RjOnRpdGxlPjwvY2M6V29yaz48L3JkZjpSREY+PC9tZXRhZGF0YT48ZGVmcw0KICAgaWQ9ImRlZnMyOTk5IiAvPg0KPHBhdGgNCiAgIGQ9Im0gNDMuMDQsMjIuNjk2IC03LjU2OCw3LjM3NyAxLjc4NywxMC40MTcgYyAwLjEyNywwLjc1IC0wLjE4MiwxLjUwOSAtMC43OTcsMS45NTcgLTAuMzQ4LDAuMjUzIC0wLjc2MiwwLjM4MiAtMS4xNzYsMC4zODIgLTAuMzE4LDAgLTAuNjM4LC0wLjA3NiAtMC45MzEsLTAuMjMgTCAyNSwzNy42ODEgMTUuNjQ1LDQyLjU5OSBjIC0wLjY3NCwwLjM1NSAtMS40OSwwLjI5NSAtMi4xMDcsLTAuMTUxIEMgMTIuOTIzLDQyIDEyLjYxNCw0MS4yNDIgMTIuNzQzLDQwLjQ5MSBMIDE0LjUzLDMwLjA3NCA2Ljk2MiwyMi42OTcgQyA2LjQxNSwyMi4xNjYgNi4yMjEsMjEuMzcxIDYuNDU0LDIwLjY0NyA2LjY5LDE5LjkyMyA3LjMxNSwxOS4zOTYgOC4wNjksMTkuMjg2IGwgMTAuNDU5LC0xLjUyMSA0LjY4LC05LjQ3OCBDIDIzLjU0Myw3LjYwMyAyNC4yMzksNy4xNzEgMjUsNy4xNzEgYyAwLjc2MywwIDEuNDU2LDAuNDMyIDEuNzkzLDEuMTE1IGwgNC42NzksOS40NzggMTAuNDYxLDEuNTIxIGMgMC43NTIsMC4xMDkgMS4zNzksMC42MzcgMS42MTIsMS4zNjEgMC4yMzcsMC43MjQgMC4wMzgsMS41MTkgLTAuNTA1LDIuMDUgeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNmZmNjMDA7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4NCg==)}.emote-menu .emote.emote-menu-starred{border-color:rgba(200,200,0,.5)}.emote-menu .emote.emote-menu-hidden{border-color:rgba(255,0,0,.5)}.emote-menu #starred-emotes-group .emote:not(.emote-menu-starred),.emote-menu:not(.editing) .emote-menu-hidden{display:none}.emote-menu:not(.editing) #starred-emotes-group .emote-menu-starred{border-color:transparent}.emote-menu #starred-emotes-group{text-align:center;color:#646464}.emote-menu #starred-emotes-group:empty:before{content:\"Use the edit mode to star an emote!\";position:relative;top:8px}.emote-menu .scrollable{height:calc(100% - 101px);overflow-y:auto}.emote-menu .sticky{position:absolute;bottom:0;width:100%}.emote-menu .emote-menu-inner{position:relative;max-height:100%;height:100%}"));
},{}],3:[function(require,module,exports){
module.exports = (function() {
var Hogan = require('hogan.js/lib/template.js');
var templates = {};
templates['emote'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||"");t.b("<div class=\"emote");if(t.s(t.f("thirdParty",c,p,1),c,p,0,32,44,"{{ }}")){t.rs(c,p,function(c,p,t){t.b(" third-party");});c.pop();}if(!t.s(t.f("isVisible",c,p,1),c,p,1,0,0,"")){t.b(" emote-menu-hidden");};if(t.s(t.f("isStarred",c,p,1),c,p,0,119,138,"{{ }}")){t.rs(c,p,function(c,p,t){t.b(" emote-menu-starred");});c.pop();}t.b("\" data-emote=\"");t.b(t.v(t.f("text",c,p,0)));t.b("\" title=\"");t.b(t.v(t.f("text",c,p,0)));if(t.s(t.f("thirdParty",c,p,1),c,p,0,206,229,"{{ }}")){t.rs(c,p,function(c,p,t){t.b(" (from 3rd party addon)");});c.pop();}t.b("\">\r");t.b("\n" + i);t.b(" <img src=\"");t.b(t.t(t.f("url",c,p,0)));t.b("\">\r");t.b("\n" + i);t.b(" <div class=\"edit-tool edit-starred\" data-which=\"");t.b(t.v(t.f("text",c,p,0)));t.b("\" data-command=\"toggle-starred\" title=\"Star/unstar emote: ");t.b(t.v(t.f("text",c,p,0)));t.b("\"></div>\r");t.b("\n" + i);t.b(" <div class=\"edit-tool edit-visibility\" data-which=\"");t.b(t.v(t.f("text",c,p,0)));t.b("\" data-command=\"toggle-visibility\" title=\"Hide/show emote: ");t.b(t.v(t.f("text",c,p,0)));t.b("\"></div>\r");t.b("\n" + i);t.b("</div>\r");t.b("\n");return t.fl(); },partials: {}, subs: { }});
templates['emoteButton'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||"");t.b("<button class=\"button button--icon-only float-left\" title=\"Emote Menu\" id=\"emote-menu-button\"></button>\r");t.b("\n");return t.fl(); },partials: {}, subs: { }});
templates['emoteGroupHeader'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||"");t.b("<div class=\"group-header\" data-emote-channel=\"");t.b(t.v(t.f("channel",c,p,0)));t.b("\">\r");t.b("\n" + i);t.b(" <div class=\"header-info\">\r");t.b("\n" + i);t.b(" <img src=\"");t.b(t.v(t.f("badge",c,p,0)));t.b("\" />\r");t.b("\n" + i);t.b(" ");t.b(t.v(t.f("channelDisplayName",c,p,0)));t.b("\r");t.b("\n" + i);t.b(" <div class=\"edit-tool edit-visibility\" data-which=\"channel-");t.b(t.v(t.f("channel",c,p,0)));t.b("\" data-command=\"toggle-visibility\" title=\"Hide/show current emotes for ");t.b(t.v(t.f("channelDisplayName",c,p,0)));t.b(" (note: new emotes will still show up if they are added)\"></div>\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b(" <div class=\"emote-container\"></div>\r");t.b("\n" + i);t.b("</div>\r");t.b("\n");return t.fl(); },partials: {}, subs: { }});
templates['menu'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||"");t.b("<div class=\"emote-menu\" id=\"emote-menu-for-twitch\">\r");t.b("\n" + i);t.b(" <div class=\"emote-menu-inner\">\r");t.b("\n" + i);t.b("\r");t.b("\n" + i);t.b(" <div class=\"draggable\"></div>\r");t.b("\n" + i);t.b("\r");t.b("\n" + i);t.b(" <div class=\"scrollable scrollbar-macosx\">\r");t.b("\n" + i);t.b(" <div class=\"group-container\" id=\"all-emotes-group\"></div>\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b("\r");t.b("\n" + i);t.b(" <div class=\"sticky\">\r");t.b("\n" + i);t.b(" <div class=\"group-header single-row\" id=\"starred-emotes-group\">\r");t.b("\n" + i);t.b(" <div class=\"header-info\">Favorite Emotes</div>\r");t.b("\n" + i);t.b(" <div class=\"emote-container\"></div>\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b("\r");t.b("\n" + i);t.b(" <div class=\"footer\">\r");t.b("\n" + i);t.b(" <a class=\"pull-left icon icon-home\" href=\"http://cletusc.github.io/Userscript--Twitch-Chat-Emotes\" target=\"_blank\" title=\"Visit the homepage where you can donate, post a review, or contact the developer\"></a>\r");t.b("\n" + i);t.b(" <a class=\"pull-left icon icon-gear\" data-command=\"toggle-editing\" title=\"Toggle edit mode\"></a>\r");t.b("\n" + i);t.b(" <a class=\"pull-right icon icon-resize-handle\" data-command=\"resize-handle\"></a>\r");t.b("\n" + i);t.b(" <a class=\"pull-right icon icon-pin\" data-command=\"toggle-pinned\" title=\"Pin/unpin the emote menu to the screen\"></a>\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b("\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b("</div>\r");t.b("\n");return t.fl(); },partials: {}, subs: { }});
templates['newsMessage'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||"");t.b("\r");t.b("\n" + i);t.b("<div class=\"twitch-chat-emotes-news\">\r");t.b("\n" + i);t.b(" [");t.b(t.v(t.f("scriptName",c,p,0)));t.b("] News: ");t.b(t.t(t.f("message",c,p,0)));t.b(" (<a href=\"#\" data-command=\"twitch-chat-emotes:dismiss-news\" data-news-id=\"");t.b(t.v(t.f("id",c,p,0)));t.b("\">Dismiss</a>)\r");t.b("\n" + i);t.b("</div>\r");t.b("\n");return t.fl(); },partials: {}, subs: { }});
return templates;
})();
},{"hogan.js/lib/template.js":4}],4:[function(require,module,exports){
/*
* Copyright 2011 Twitter, Inc.
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
var Hogan = {};
(function (Hogan) {
Hogan.Template = function (codeObj, text, compiler, options) {
codeObj = codeObj || {};
this.r = codeObj.code || this.r;
this.c = compiler;
this.options = options || {};
this.text = text || '';
this.partials = codeObj.partials || {};
this.subs = codeObj.subs || {};
this.buf = '';
}
Hogan.Template.prototype = {
// render: replaced by generated code.
r: function (context, partials, indent) { return ''; },
// variable escaping
v: hoganEscape,
// triple stache
t: coerceToString,
render: function render(context, partials, indent) {
return this.ri([context], partials || {}, indent);
},
// render internal -- a hook for overrides that catches partials too
ri: function (context, partials, indent) {
return this.r(context, partials, indent);
},
// ensurePartial
ep: function(symbol, partials) {
var partial = this.partials[symbol];
// check to see that if we've instantiated this partial before
var template = partials[partial.name];
if (partial.instance && partial.base == template) {
return partial.instance;
}
if (typeof template == 'string') {
if (!this.c) {
throw new Error("No compiler available.");
}
template = this.c.compile(template, this.options);
}
if (!template) {
return null;
}
// We use this to check whether the partials dictionary has changed
this.partials[symbol].base = template;
if (partial.subs) {
// Make sure we consider parent template now
if (!partials.stackText) partials.stackText = {};
for (key in partial.subs) {
if (!partials.stackText[key]) {
partials.stackText[key] = (this.activeSub !== undefined && partials.stackText[this.activeSub]) ? partials.stackText[this.activeSub] : this.text;
}
}
template = createSpecializedPartial(template, partial.subs, partial.partials,
this.stackSubs, this.stackPartials, partials.stackText);
}
this.partials[symbol].instance = template;
return template;
},
// tries to find a partial in the current scope and render it
rp: function(symbol, context, partials, indent) {
var partial = this.ep(symbol, partials);
if (!partial) {
return '';
}
return partial.ri(context, partials, indent);
},
// render a section
rs: function(context, partials, section) {
var tail = context[context.length - 1];
if (!isArray(tail)) {
section(context, partials, this);
return;
}
for (var i = 0; i < tail.length; i++) {
context.push(tail[i]);
section(context, partials, this);
context.pop();
}
},
// maybe start a section
s: function(val, ctx, partials, inverted, start, end, tags) {
var pass;
if (isArray(val) && val.length === 0) {
return false;
}
if (typeof val == 'function') {
val = this.ms(val, ctx, partials, inverted, start, end, tags);
}
pass = !!val;
if (!inverted && pass && ctx) {
ctx.push((typeof val == 'object') ? val : ctx[ctx.length - 1]);
}
return pass;
},
// find values with dotted names
d: function(key, ctx, partials, returnFound) {
var found,
names = key.split('.'),
val = this.f(names[0], ctx, partials, returnFound),
doModelGet = this.options.modelGet,
cx = null;
if (key === '.' && isArray(ctx[ctx.length - 2])) {
val = ctx[ctx.length - 1];
} else {
for (var i = 1; i < names.length; i++) {
found = findInScope(names[i], val, doModelGet);
if (found !== undefined) {
cx = val;
val = found;
} else {
val = '';
}
}
}
if (returnFound && !val) {
return false;
}
if (!returnFound && typeof val == 'function') {
ctx.push(cx);
val = this.mv(val, ctx, partials);
ctx.pop();
}
return val;
},
// find values with normal names
f: function(key, ctx, partials, returnFound) {
var val = false,
v = null,
found = false,
doModelGet = this.options.modelGet;
for (var i = ctx.length - 1; i >= 0; i--) {
v = ctx[i];
val = findInScope(key, v, doModelGet);
if (val !== undefined) {
found = true;
break;
}
}
if (!found) {
return (returnFound) ? false : "";
}
if (!returnFound && typeof val == 'function') {
val = this.mv(val, ctx, partials);
}
return val;
},
// higher order templates
ls: function(func, cx, partials, text, tags) {
var oldTags = this.options.delimiters;
this.options.delimiters = tags;
this.b(this.ct(coerceToString(func.call(cx, text)), cx, partials));
this.options.delimiters = oldTags;
return false;
},
// compile text
ct: function(text, cx, partials) {
if (this.options.disableLambda) {
throw new Error('Lambda features disabled.');
}
return this.c.compile(text, this.options).render(cx, partials);
},
// template result buffering
b: function(s) { this.buf += s; },
fl: function() { var r = this.buf; this.buf = ''; return r; },
// method replace section
ms: function(func, ctx, partials, inverted, start, end, tags) {
var textSource,
cx = ctx[ctx.length - 1],
result = func.call(cx);
if (typeof result == 'function') {
if (inverted) {
return true;
} else {
textSource = (this.activeSub && this.subsText && this.subsText[this.activeSub]) ? this.subsText[this.activeSub] : this.text;
return this.ls(result, cx, partials, textSource.substring(start, end), tags);
}
}
return result;
},
// method replace variable
mv: function(func, ctx, partials) {
var cx = ctx[ctx.length - 1];
var result = func.call(cx);
if (typeof result == 'function') {
return this.ct(coerceToString(result.call(cx)), cx, partials);
}
return result;
},
sub: function(name, context, partials, indent) {
var f = this.subs[name];
if (f) {
this.activeSub = name;
f(context, partials, this, indent);
this.activeSub = false;
}
}
};
//Find a key in an object
function findInScope(key, scope, doModelGet) {
var val;
if (scope && typeof scope == 'object') {
if (scope[key] !== undefined) {
val = scope[key];
// try lookup with get for backbone or similar model data
} else if (doModelGet && scope.get && typeof scope.get == 'function') {
val = scope.get(key);
}
}
return val;
}
function createSpecializedPartial(instance, subs, partials, stackSubs, stackPartials, stackText) {
function PartialTemplate() {};
PartialTemplate.prototype = instance;
function Substitutions() {};
Substitutions.prototype = instance.subs;
var key;
var partial = new PartialTemplate();
partial.subs = new Substitutions();
partial.subsText = {}; //hehe. substext.
partial.buf = '';
stackSubs = stackSubs || {};
partial.stackSubs = stackSubs;
partial.subsText = stackText;
for (key in subs) {
if (!stackSubs[key]) stackSubs[key] = subs[key];
}
for (key in stackSubs) {
partial.subs[key] = stackSubs[key];
}
stackPartials = stackPartials || {};
partial.stackPartials = stackPartials;
for (key in partials) {
if (!stackPartials[key]) stackPartials[key] = partials[key];
}
for (key in stackPartials) {
partial.partials[key] = stackPartials[key];
}
return partial;
}
var rAmp = /&/g,
rLt = /</g,
rGt = />/g,
rApos = /\'/g,
rQuot = /\"/g,
hChars = /[&<>\"\']/;
function coerceToString(val) {
return String((val === null || val === undefined) ? '' : val);
}
function hoganEscape(str) {
str = coerceToString(str);
return hChars.test(str) ?
str
.replace(rAmp, '&amp;')
.replace(rLt, '&lt;')
.replace(rGt, '&gt;')
.replace(rApos, '&#39;')
.replace(rQuot, '&quot;') :
str;
}
var isArray = Array.isArray || function(a) {
return Object.prototype.toString.call(a) === '[object Array]';
};
})(typeof exports !== 'undefined' ? exports : Hogan);
},{}],5:[function(require,module,exports){
/**
* jQuery CSS Customizable Scrollbar
*
* Copyright 2014, Yuriy Khabarov
* Dual licensed under the MIT or GPL Version 2 licenses.
*
* If you found bug, please contact me via email <13real008@gmail.com>
*
* @author Yuriy Khabarov aka Gromo
* @version 0.2.6
* @url https://github.com/gromo/jquery.scrollbar/
*
*/
(function(e,t,n){"use strict";function h(t){if(o.webkit&&!t){return{height:0,width:0}}if(!o.data.outer){var n={border:"none","box-sizing":"content-box",height:"200px",margin:"0",padding:"0",width:"200px"};o.data.inner=e("<div>").css(e.extend({},n));o.data.outer=e("<div>").css(e.extend({left:"-1000px",overflow:"scroll",position:"absolute",top:"-1000px"},n)).append(o.data.inner).appendTo("body")}o.data.outer.scrollLeft(1e3).scrollTop(1e3);return{height:Math.ceil(o.data.outer.offset().top-o.data.inner.offset().top||0),width:Math.ceil(o.data.outer.offset().left-o.data.inner.offset().left||0)}}function p(n,r){e(t).on({"blur.scrollbar":function(){e(t).add("body").off(".scrollbar");n&&n()},"dragstart.scrollbar":function(e){e.preventDefault();return false},"mouseup.scrollbar":function(){e(t).add("body").off(".scrollbar");n&&n()}});e("body").on({"selectstart.scrollbar":function(e){e.preventDefault();return false}});r&&r.preventDefault();return false}function d(){var e=h(true);return!(e.height||e.width)}function v(e){var t=e.originalEvent;if(t.axis&&t.axis===t.HORIZONTAL_AXIS)return false;if(t.wheelDeltaX)return false;return true}var r=false;var i=1,s="px";var o={data:{},macosx:n.navigator.platform.toLowerCase().indexOf("mac")!==-1,mobile:/Android|webOS|iPhone|iPad|iPod|BlackBerry/i.test(n.navigator.userAgent),overlay:null,scroll:null,scrolls:[],webkit:/WebKit/.test(n.navigator.userAgent),log:r?function(t,r){var i=t;if(r&&typeof t!="string"){i=[];e.each(t,function(e,t){i.push('"'+e+'": '+t)});i=i.join(", ")}if(n.console&&n.console.log){n.console.log(i)}else{alert(i)}}:function(){}};var u={autoScrollSize:true,autoUpdate:true,debug:false,disableBodyScroll:false,duration:200,ignoreMobile:true,ignoreOverlay:true,scrollStep:30,showArrows:false,stepScrolling:true,type:"simple",scrollx:null,scrolly:null,onDestroy:null,onInit:null,onScroll:null,onUpdate:null};var a=function(t,r){if(!o.scroll){o.log("Init jQuery Scrollbar v0.2.6");o.overlay=d();o.scroll=h();c();e(n).resize(function(){var e=false;if(o.scroll&&(o.scroll.height||o.scroll.width)){var t=h();if(t.height!=o.scroll.height||t.width!=o.scroll.width){o.scroll=t;e=true}}c(e)})}this.container=t;this.options=e.extend({},u,n.jQueryScrollbarOptions||{});this.scrollTo=null;this.scrollx={};this.scrolly={};this.init(r)};a.prototype={destroy:function(){if(!this.wrapper){return}var n=this.container.scrollLeft();var r=this.container.scrollTop();this.container.insertBefore(this.wrapper).css({height:"",margin:""}).removeClass("scroll-content").removeClass("scroll-scrollx_visible").removeClass("scroll-scrolly_visible").off(".scrollbar").scrollLeft(n).scrollTop(r);this.scrollx.scrollbar.removeClass("scroll-scrollx_visible").find("div").andSelf().off(".scrollbar");this.scrolly.scrollbar.removeClass("scroll-scrolly_visible").find("div").andSelf().off(".scrollbar");this.wrapper.remove();e(t).add("body").off(".scrollbar");if(e.isFunction(this.options.onDestroy))this.options.onDestroy.apply(this,[this.container])},getScrollbar:function(t){var n=this.options["scroll"+t];var r={advanced:'<div class="scroll-element_corner"></div>'+'<div class="scroll-arrow scroll-arrow_less"></div>'+'<div class="scroll-arrow scroll-arrow_more"></div>'+'<div class="scroll-element_outer">'+' <div class="scroll-element_size"></div>'+' <div class="scroll-element_inner-wrapper">'+' <div class="scroll-element_inner scroll-element_track">'+' <div class="scroll-element_inner-bottom"></div>'+" </div>"+" </div>"+' <div class="scroll-bar">'+' <div class="scroll-bar_body">'+' <div class="scroll-bar_body-inner"></div>'+" </div>"+' <div class="scroll-bar_bottom"></div>'+' <div class="scroll-bar_center"></div>'+" </div>"+"</div>",simple:'<div class="scroll-element_outer">'+' <div class="scroll-element_size"></div>'+' <div class="scroll-element_track"></div>'+' <div class="scroll-bar"></div>'+"</div>"};var i=r[this.options.type]?this.options.type:"advanced";if(n){if(typeof n=="string"){n=e(n).appendTo(this.wrapper)}else{n=e(n)}}else{n=e("<div>").addClass("scroll-element").html(r[i]).appendTo(this.wrapper)}if(this.options.showArrows){n.addClass("scroll-element_arrows_visible")}return n.addClass("scroll-"+t)},init:function(n){var r=this;var u=this.container;var a=this.containerWrapper||u;var f=e.extend(this.options,n||{});var l={x:this.scrollx,y:this.scrolly};var c=this.wrapper;var h={scrollLeft:u.scrollLeft(),scrollTop:u.scrollTop()};if(o.mobile&&f.ignoreMobile||o.overlay&&f.ignoreOverlay||o.macosx&&!o.webkit){return false}if(!c){this.wrapper=c=e("<div>").addClass("scroll-wrapper").addClass(u.attr("class")).css("position",u.css("position")=="absolute"?"absolute":"relative").insertBefore(u).append(u);if(u.is("textarea")){this.containerWrapper=a=e("<div>").insertBefore(u).append(u);c.addClass("scroll-textarea")}a.addClass("scroll-content").css({height:"","margin-bottom":o.scroll.height*-1+s,"margin-right":o.scroll.width*-1+s});u.on("scroll.scrollbar",function(t){if(e.isFunction(f.onScroll)){f.onScroll.call(r,{maxScroll:l.y.maxScrollOffset,scroll:u.scrollTop(),size:l.y.size,visible:l.y.visible},{maxScroll:l.x.maxScrollOffset,scroll:u.scrollLeft(),size:l.x.size,visible:l.x.visible})}l.x.isVisible&&l.x.scroller.css("left",u.scrollLeft()*l.x.kx+s);l.y.isVisible&&l.y.scroller.css("top",u.scrollTop()*l.y.kx+s)});c.on("scroll",function(){c.scrollTop(0).scrollLeft(0)});if(f.disableBodyScroll){var d=function(e){v(e)?l.y.isVisible&&l.y.mousewheel(e):l.x.isVisible&&l.x.mousewheel(e)};c.on({"MozMousePixelScroll.scrollbar":d,"mousewheel.scrollbar":d});if(o.mobile){c.on("touchstart.scrollbar",function(n){var r=n.originalEvent.touches&&n.originalEvent.touches[0]||n;var i={pageX:r.pageX,pageY:r.pageY};var s={left:u.scrollLeft(),top:u.scrollTop()};e(t).on({"touchmove.scrollbar":function(e){var t=e.originalEvent.targetTouches&&e.originalEvent.targetTouches[0]||e;u.scrollLeft(s.left+i.pageX-t.pageX);u.scrollTop(s.top+i.pageY-t.pageY);e.preventDefault()},"touchend.scrollbar":function(){e(t).off(".scrollbar")}})})}}if(e.isFunction(f.onInit))f.onInit.apply(this,[u])}else{a.css({height:"","margin-bottom":o.scroll.height*-1+s,"margin-right":o.scroll.width*-1+s})}e.each(l,function(n,s){var o=null;var a=1;var c=n=="x"?"scrollLeft":"scrollTop";var h=f.scrollStep;var d=function(){var e=u[c]();u[c](e+h);if(a==1&&e+h>=m)e=u[c]();if(a==-1&&e+h<=m)e=u[c]();if(u[c]()==e&&o){o()}};var m=0;if(!s.scrollbar){s.scrollbar=r.getScrollbar(n);s.scroller=s.scrollbar.find(".scroll-bar");s.mousewheel=function(e){if(!s.isVisible||n=="x"&&v(e)){return true}if(n=="y"&&!v(e)){l.x.mousewheel(e);return true}var t=e.originalEvent.wheelDelta*-1||e.originalEvent.detail;var i=s.size-s.visible-s.offset;if(!(m<=0&&t<0||m>=i&&t>0)){m=m+t;if(m<0)m=0;if(m>i)m=i;r.scrollTo=r.scrollTo||{};r.scrollTo[c]=m;setTimeout(function(){if(r.scrollTo){u.stop().animate(r.scrollTo,240,"linear",function(){m=u[c]()});r.scrollTo=null}},1)}e.preventDefault();return false};s.scrollbar.on({"MozMousePixelScroll.scrollbar":s.mousewheel,"mousewheel.scrollbar":s.mousewheel,"mouseenter.scrollbar":function(){m=u[c]()}});s.scrollbar.find(".scroll-arrow, .scroll-element_track").on("mousedown.scrollbar",function(t){if(t.which!=i)return true;a=1;var l={eventOffset:t[n=="x"?"pageX":"pageY"],maxScrollValue:s.size-s.visible-s.offset,scrollbarOffset:s.scroller.offset()[n=="x"?"left":"top"],scrollbarSize:s.scroller[n=="x"?"outerWidth":"outerHeight"]()};var v=0,g=0;if(e(this).hasClass("scroll-arrow")){a=e(this).hasClass("scroll-arrow_more")?1:-1;h=f.scrollStep*a;m=a>0?l.maxScrollValue:0}else{a=l.eventOffset>l.scrollbarOffset+l.scrollbarSize?1:l.eventOffset<l.scrollbarOffset?-1:0;h=Math.round(s.visible*.75)*a;m=l.eventOffset-l.scrollbarOffset-(f.stepScrolling?a==1?l.scrollbarSize:0:Math.round(l.scrollbarSize/2));m=u[c]()+m/s.kx}r.scrollTo=r.scrollTo||{};r.scrollTo[c]=f.stepScrolling?u[c]()+h:m;if(f.stepScrolling){o=function(){m=u[c]();clearInterval(g);clearTimeout(v);v=0;g=0};v=setTimeout(function(){g=setInterval(d,40)},f.duration+100)}setTimeout(function(){if(r.scrollTo){u.animate(r.scrollTo,f.duration);r.scrollTo=null}},1);return p(o,t)});s.scroller.on("mousedown.scrollbar",function(r){if(r.which!=i)return true;var o=r[n=="x"?"pageX":"pageY"];var a=u[c]();s.scrollbar.addClass("scroll-draggable");e(t).on("mousemove.scrollbar",function(e){var t=parseInt((e[n=="x"?"pageX":"pageY"]-o)/s.kx,10);u[c](a+t)});return p(function(){s.scrollbar.removeClass("scroll-draggable");m=u[c]()},r)})}});e.each(l,function(e,t){var n="scroll-scroll"+e+"_visible";var r=e=="x"?l.y:l.x;t.scrollbar.removeClass(n);r.scrollbar.removeClass(n);a.removeClass(n)});e.each(l,function(t,n){e.extend(n,t=="x"?{offset:parseInt(u.css("left"),10)||0,size:u.prop("scrollWidth"),visible:c.width()}:{offset:parseInt(u.css("top"),10)||0,size:u.prop("scrollHeight"),visible:c.height()})});var m=function(t,n){var r="scroll-scroll"+t+"_visible";var i=t=="x"?l.y:l.x;var f=parseInt(u.css(t=="x"?"left":"top"),10)||0;var h=n.size;var p=n.visible+f;n.isVisible=h-p>1;if(n.isVisible){n.scrollbar.addClass(r);i.scrollbar.addClass(r);a.addClass(r)}else{n.scrollbar.removeClass(r);i.scrollbar.removeClass(r);a.removeClass(r)}if(t=="y"&&(n.isVisible||n.size<n.visible)){a.css("height",p+o.scroll.height+s)}if(l.x.size!=u.prop("scrollWidth")||l.y.size!=u.prop("scrollHeight")||l.x.visible!=c.width()||l.y.visible!=c.height()||l.x.offset!=(parseInt(u.css("left"),10)||0)||l.y.offset!=(parseInt(u.css("top"),10)||0)){e.each(l,function(t,n){e.extend(n,t=="x"?{offset:parseInt(u.css("left"),10)||0,size:u.prop("scrollWidth"),visible:c.width()}:{offset:parseInt(u.css("top"),10)||0,size:u.prop("scrollHeight"),visible:c.height()})});m(t=="x"?"y":"x",i)}};e.each(l,m);if(e.isFunction(f.onUpdate))f.onUpdate.apply(this,[u]);e.each(l,function(e,t){var n=e=="x"?"left":"top";var r=e=="x"?"outerWidth":"outerHeight";var i=e=="x"?"width":"height";var o=parseInt(u.css(n),10)||0;var a=t.size;var l=t.visible+o;var c=t.scrollbar.find(".scroll-element_size");c=c[r]()+(parseInt(c.css(n),10)||0);if(f.autoScrollSize){t.scrollbarSize=parseInt(c*l/a,10);t.scroller.css(i,t.scrollbarSize+s)}t.scrollbarSize=t.scroller[r]();t.kx=(c-t.scrollbarSize)/(a-l)||1;t.maxScrollOffset=a-l});u.scrollLeft(h.scrollLeft).scrollTop(h.scrollTop).trigger("scroll")}};e.fn.scrollbar=function(t,n){var r=this;if(t==="get")r=null;this.each(function(){var i=e(this);if(i.hasClass("scroll-wrapper")||i.get(0).nodeName=="body"){return true}var s=i.data("scrollbar");if(s){if(t==="get"){r=s;return false}var u=typeof t=="string"&&s[t]?t:"init";s[u].apply(s,e.isArray(n)?n:[]);if(t==="destroy"){i.removeData("scrollbar");while(e.inArray(s,o.scrolls)>=0)o.scrolls.splice(e.inArray(s,o.scrolls),1)}}else{if(typeof t!="string"){s=new a(i,t);i.data("scrollbar",s);o.scrolls.push(s)}}return true});return r};e.fn.scrollbar.options=u;if(n.angular){(function(e){var t=e.module("jQueryScrollbar",[]);t.directive("jqueryScrollbar",function(){return{link:function(e,t){t.scrollbar(e.options).on("$destroy",function(){t.scrollbar("destroy")})},restring:"AC",scope:{options:"=jqueryScrollbar"}}})})(n.angular)}var f=0,l=0;var c=function(e){var t,n,i,s,u,a,h;for(t=0;t<o.scrolls.length;t++){s=o.scrolls[t];n=s.container;i=s.options;u=s.wrapper;a=s.scrollx;h=s.scrolly;if(e||i.autoUpdate&&u&&u.is(":visible")&&(n.prop("scrollWidth")!=a.size||n.prop("scrollHeight")!=h.size||u.width()!=a.visible||u.height()!=h.visible)){s.init();if(r){o.log({scrollHeight:n.prop("scrollHeight")+":"+s.scrolly.size,scrollWidth:n.prop("scrollWidth")+":"+s.scrollx.size,visibleHeight:u.height()+":"+s.scrolly.visible,visibleWidth:u.width()+":"+s.scrollx.visible},true);l++}}}if(r&&l>10){o.log("Scroll updates exceed 10");c=function(){}}else{clearTimeout(f);f=setTimeout(c,300)}}})(jQuery,document,window);
},{}],6:[function(require,module,exports){
// Storage cache.
var cache = {};
// The store handling expiration of data.
var expiresStore = new Store({
namespace: '__storage-wrapper:expires'
});
/**
* Storage wrapper for making routine storage calls super easy.
* @class Store
* @constructor
* @param {object} [options] The options for the store. Options not overridden will use the defaults.
* @param {mixed} [options.namespace=''] See {{#crossLink "Store/setNamespace"}}Store#setNamespace{{/crossLink}}
* @param {mixed} [options.storageType='local'] See {{#crossLink "Store/setStorageType"}}Store#setStorageType{{/crossLink}}
*/
function Store(options) {
var settings = {
namespace: '',
storageType: 'local'
};
/**
* Sets the storage namespace.
* @method setNamespace
* @param {string|false|null} namespace The namespace to work under. To use no namespace (e.g. global namespace), pass in `false` or `null` or an empty string.
*/
this.setNamespace = function (namespace) {
var validNamespace = /^[\w-:]+$/;
// No namespace.
if (namespace === false || namespace == null || namespace === '') {
settings.namespace = '';
return;
}
if (typeof namespace !== 'string' || !validNamespace.test(namespace)) {
throw new Error('Invalid namespace.');
}
settings.namespace = namespace;
};
/**
* Gets the current storage namespace.
* @method getNamespace
* @return {string} The current namespace.
*/
this.getNamespace = function (includeSeparator) {
if (includeSeparator && settings.namespace !== '') {
return settings.namespace + ':';
}
return settings.namespace;
}
/**
* Sets the type of storage to use.
* @method setStorageType
* @param {string} type The type of storage to use. Use `session` for `sessionStorage` and `local` for `localStorage`.
*/
this.setStorageType = function (type) {
if (['session', 'local'].indexOf(type) < 0) {
throw new Error('Invalid storage type.');
}
settings.storageType = type;
};
/**
* Get the type of storage being used.
* @method getStorageType
* @return {string} The type of storage being used.
*/
this.getStorageType = function () {
return settings.storageType;
};
// Override default settings.
if (options) {
for (var key in options) {
switch (key) {
case 'namespace':
this.setNamespace(options[key]);
break;
case 'storageType':
this.setStorageType(options[key]);
break;
}
}
}
}
/**
* Gets the actual handler to use
* @method getStorageHandler
* @return {mixed} The storage handler.
*/
Store.prototype.getStorageHandler = function () {
var handlers = {
'local': localStorage,
'session': sessionStorage
};
return handlers[this.getStorageType()];
}
/**
* Gets the full storage name for a key, including the namespace, if any.
* @method getStorageKey
* @param {string} key The storage key name.
* @return {string} The full storage name that is used by the storage methods.
*/
Store.prototype.getStorageKey = function (key) {
if (!key || typeof key !== 'string' || key.length < 1) {
throw new Error('Key must be a string.');
}
return this.getNamespace(true) + key;
};
/**
* Gets a storage item from the current namespace.
* @method get
* @param {string} key The key that the data can be accessed under.
* @param {mixed} defaultValue The default value to return in case the storage value is not set or `null`.
* @return {mixed} The data for the storage.
*/
Store.prototype.get = function (key, defaultValue) {
// Prevent recursion. Only check expire date if it isn't called from `expiresStore`.
if (this !== expiresStore) {
// Check if key is expired.
var expireDate = expiresStore.get(this.getStorageKey(key));
if (expireDate !== null && expireDate.getTime() < Date.now()) {
// Expired, remove it.
this.remove(key);
expiresStore.remove(this.getStorageKey(key));
}
}
// Cached, read from memory.
if (cache[this.getStorageKey(key)] != null) {
return cache[this.getStorageKey(key)];
}
var val = this.getStorageHandler().getItem(this.getStorageKey(key));
// Value doesn't exist and we have a default, return default.
if (val === null && typeof defaultValue !== 'undefined') {
return defaultValue;
}
// Only pre-process strings.
if (typeof val === 'string') {
// Handle RegExps.
if (val.indexOf('~RegExp:') === 0) {
var matches = /^~RegExp:([gim]*?):(.*)/.exec(val);
val = new RegExp(matches[2], matches[1]);
}
// Handle Dates.
else if (val.indexOf('~Date:') === 0) {
val = new Date(val.replace(/^~Date:/, ''));
}
// Handle numbers.
else if (val.indexOf('~Number:') === 0) {
val = parseInt(val.replace(/^~Number:/, ''), 10);
}
// Handle booleans.
else if (val.indexOf('~Boolean:') === 0) {
val = val.replace(/^~Boolean:/, '') === 'true';
}
// Handle objects.
else if (val.indexOf('~JSON:') === 0) {
val = val.replace(/^~JSON:/, '');
// Try parsing it.
try {
val = JSON.parse(val);
}
// Parsing went wrong (invalid JSON), return default or null.
catch (e) {
if (typeof defaultValue !== 'undefined') {
return defaultValue;
}
return null;
}
}
}
// Return it.
cache[this.getStorageKey(key)] = val;
return val;
};
/**
* Sets a storage item on the current namespace.
* @method set
* @param {string} key The key that the data can be accessed under.
* @param {mixed} val The value to store. May be the following types of data: `RegExp`, `Date`, `Object`, `String`, `Boolean`, `Number`
* @param {Date|number} [expires] The date in the future to expire, or relative number of milliseconds from `Date#now` to expire.
*
* Note: This converts special data types that normally can't be stored in the following way:
*
* - `RegExp`: prefixed with type, flags stored, and source stored as string.
* - `Date`: prefixed with type, stored as string using `Date#toString`.
* - `Object`: prefixed with "JSON" indicator, stored as string using `JSON#stringify`.
*/
Store.prototype.set = function (key, val, expires) {
var parsedVal = null;
// Handle RegExps.
if (val instanceof RegExp) {
var flags = [
val.global ? 'g' : '',
val.ignoreCase ? 'i' : '',
val.multiline ? 'm' : '',
].join('');
parsedVal = '~RegExp:' + flags + ':' + val.source;
}
// Handle Dates.
else if (val instanceof Date) {
parsedVal = '~Date:' + val.toString();
}
// Handle objects.
else if (val === Object(val)) {
parsedVal = '~JSON:' + JSON.stringify(val);
}
// Handle numbers.
else if (typeof val === 'number') {
parsedVal = '~Number:' + val.toString();
}
// Handle booleans.
else if (typeof val === 'boolean') {
parsedVal = '~Boolean:' + val.toString();
}
// Handle strings.
else if (typeof val === 'string') {
parsedVal = val;
}
// Throw if we don't know what it is.
else {
throw new Error('Unable to store this value; wrong value type.');
}
// Set expire date if needed.
if (typeof expires !== 'undefined') {
// Convert to a relative date.
if (typeof expires === 'number') {
expires = new Date(Date.now() + expires);
}
// Make sure it is a date.
if (expires instanceof Date) {
expiresStore.set(this.getStorageKey(key), expires);
}
else {
throw new Error('Key expire must be a valid date or timestamp.');
}
}
// Save it.
cache[this.getStorageKey(key)] = val;
this.getStorageHandler().setItem(this.getStorageKey(key), parsedVal);
};
/**
* Gets all data for the current namespace.
* @method getAll
* @return {object} An object containing all data in the form of `{theKey: theData}` where `theData` is parsed using {{#crossLink "Store/get"}}Store#get{{/crossLink}}.
*/
Store.prototype.getAll = function () {
var keys = this.listKeys();
var data = {};
keys.forEach(function (key) {
data[key] = this.get(key);
}, this);
return data;
};
/**
* List all keys that are tied to the current namespace.
* @method listKeys
* @return {array} The storage keys.
*/
Store.prototype.listKeys = function () {
var keys = [];
var key = null;
var storageLength = this.getStorageHandler().length;
var prefix = new RegExp('^' + this.getNamespace(true));
for (var i = 0; i < storageLength; i++) {
key = this.getStorageHandler().key(i)
if (prefix.test(key)) {
keys.push(key.replace(prefix, ''));
}
}
return keys;
};
/**
* Removes a specific key and data from the current namespace.
* @method remove
* @param {string} key The key to remove the data for.
*/
Store.prototype.remove = function (key) {
cache[this.getStorageKey(key)] = null;
this.getStorageHandler().removeItem(this.getStorageKey(key));
};
/**
* Removes all data and keys from the current namespace.
* @method removeAll
*/
Store.prototype.removeAll = function () {
this.listKeys().forEach(this.remove, this);
};
/**
* Removes namespaced items from the cache so your next {{#crossLink "Store/get"}}Store#get{{/crossLink}} will be fresh from the storage.
* @method freshen
* @param {string} key The key to remove the cache data for.
*/
Store.prototype.freshen = function (key) {
var keys = key ? [key] : this.listKeys();
keys.forEach(function (key) {
cache[this.getStorageKey(key)] = null;
}, this);
};
/**
* Migrate data from a different namespace to current namespace.
* @method migrate
* @param {object} migration The migration object.
* @param {string} migration.toKey The key name under your current namespace the old data should change to.
* @param {string} migration.fromNamespace The old namespace that the old key belongs to.
* @param {string} migration.fromKey The old key name to migrate from.
* @param {string} [migration.fromStorageType] The storage type to migrate from. Defaults to same type as where you are migrating to.
* @param {boolean} [migration.keepOldData=false] Whether old data should be kept after it has been migrated.
* @param {boolean} [migration.overwriteNewData=false] Whether old data should overwrite currently stored data if it exists.
* @param {function} [migration.transform] The function to pass the old key data through before migrating.
* @example
*
* var Store = require('storage-wrapper');
* var store = new Store({
* namespace: 'myNewApp'
* });
*
* // Migrate from the old app.
* store.migrate({
* toKey: 'new-key',
* fromNamespace: 'myOldApp',
* fromKey: 'old-key'
* });
*
* // Migrate from global data. Useful when moving from other storage wrappers or regular ol' `localStorage`.
* store.migrate({
* toKey: 'other-new-key',
* fromNamespace: '',
* fromKey: 'other-old-key-on-global'
* });
*
* // Migrate some JSON data that was stored as a string.
* store.migrate({
* toKey: 'new-json-key',
* fromNamespace: 'myOldApp',
* fromKey: 'old-json-key',
* // Try converting some old JSON data.
* transform: function (data) {
* try {
* return JSON.parse(data);
* }
* catch (e) {
* return data;
* }
* }
* });
*/
Store.prototype.migrate = function (migration) {
// Save our current namespace.
var toNamespace = this.getNamespace();
var toStorageType = this.getStorageType();
// Create a temporary store to avoid changing namespace during actual get/sets.
var store = new Store({
namespace: toNamespace,
storageType: toStorageType
});
var data = null;
// Get data from old namespace.
store.setNamespace(migration.fromNamespace);
if (typeof migration.fromStorageType !== 'undefined') {
store.setStorageType(migration.fromStorageType);
}
data = store.get(migration.fromKey);
// Remove old if needed.
if (!migration.keepOldData) {
store.remove(migration.fromKey);
}
// No data, ignore this migration.
if (data === null) {
return;
}
// Transform data if needed.
if (typeof migration.transform === 'function') {
data = migration.transform(data);
}
else if (typeof migration.transform !== 'undefined') {
throw new Error('Invalid transform callback.');
}
// Go back to current namespace.
store.setNamespace(toNamespace);
store.setStorageType(toStorageType);
// Only overwrite new data if it doesn't exist or it's requested.
if (store.get(migration.toKey) === null || migration.overwriteNewData) {
store.set(migration.toKey, data);
}
};
/**
* Creates a substore that is nested in the current namespace.
* @method createSubstore
* @param {string} namespace The substore's namespace.
* @return {Store} The substore.
* @example
*
* var Store = require('storage-wrapper');
* // Create main store.
* var store = new Store({
* namespace: 'myapp'
* });
*
* // Create substore.
* var substore = store.createSubstore('things');
* substore.set('foo', 'bar');
*
* substore.get('foo') === store.get('things:foo');
* // true
*/
Store.prototype.createSubstore = function (namespace) {
return new Store({
namespace: this.getNamespace(true) + namespace,
storageType: this.getStorageType()
});
};
module.exports = Store;
},{}],7:[function(require,module,exports){
module.exports={
"name": "twitch-chat-emotes",
"version": "2.1.4",
"homepage": "http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/",
"bugs": "https://github.com/cletusc/Userscript--Twitch-Chat-Emotes/issues",
"author": "Ryan Chatham <ryan.b.chatham@gmail.com> (https://github.com/cletusc)",
"repository": {
"type": "git",
"url": "https://github.com/cletusc/Userscript--Twitch-Chat-Emotes.git"
},
"userscript": {
"name": "Twitch Chat Emotes",
"namespace": "#Cletus",
"version": "{{{pkg.version}}}",
"description": "Adds a button to Twitch that allows you to \"click-to-insert\" an emote.",
"copyright": "2011+, {{{pkg.author}}}",
"author": "{{{pkg.author}}}",
"icon": "http://www.gravatar.com/avatar.php?gravatar_id=6875e83aa6c563790cb2da914aaba8b3&r=PG&s=48&default=identicon",
"license": [
"MIT; http://opensource.org/licenses/MIT",
"CC BY-NC-SA 3.0; http://creativecommons.org/licenses/by-nc-sa/3.0/"
],
"homepage": "{{{pkg.homepage}}}",
"supportURL": "{{{pkg.bugs}}}",
"contributionURL": "http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/#donate",
"grant": "none",
"include": [
"http://*.twitch.tv/*",
"https://*.twitch.tv/*"
],
"exclude": [
"http://api.twitch.tv/*",
"https://api.twitch.tv/*",
"http://tmi.twitch.tv/*",
"https://tmi.twitch.tv/*",
"http://*.twitch.tv/*/dashboard",
"https://*.twitch.tv/*/dashboard",
"http://chatdepot.twitch.tv/*",
"https://chatdepot.twitch.tv/*",
"http://im.twitch.tv/*",
"https://im.twitch.tv/*",
"http://platform.twitter.com/*",
"https://platform.twitter.com/*",
"http://www.facebook.com/*",
"https://www.facebook.com/*"
]
},
"devDependencies": {
"browser-sync": "^1.3.2",
"browserify": "^5.9.1",
"generate-userscript-header": "^1.0.0",
"gulp": "^3.8.3",
"gulp-autoprefixer": "0.0.8",
"gulp-beautify": "1.1.0",
"gulp-changed": "^0.4.1",
"gulp-concat": "^2.2.0",
"gulp-conflict": "^0.1.2",
"gulp-css-base64": "^1.1.0",
"gulp-css2js": "^1.0.2",
"gulp-header": "^1.0.2",
"gulp-hogan-compile": "^0.2.1",
"gulp-minify-css": "^0.3.5",
"gulp-notify": "^1.4.1",
"gulp-rename": "^1.2.0",
"gulp-uglify": "^0.3.1",
"gulp-util": "^3.0.0",
"hogan.js": "^3.0.2",
"jquery-ui": "^1.10.5",
"jquery.scrollbar": "^0.2.7",
"pretty-hrtime": "^0.2.1",
"storage-wrapper": "cletusc/storage-wrapper#v0.1.1",
"vinyl-map": "^1.0.1",
"vinyl-source-stream": "^0.1.1",
"watchify": "^1.0.1"
}
}
},{}],8:[function(require,module,exports){
var logger = require('./logger');
var api = {};
var ember = null;
var hookedFactories = {};
api.getEmber = function () {
if (ember) {
return ember;
}
if (window.App && window.App.__container__) {
ember = window.App.__container__;
return ember;
}
return false;
};
api.isLoaded = function () {
return Boolean(api.getEmber());
};
api.lookup = function (lookupFactory) {
if (!api.isLoaded()) {
logger.debug('Factory lookup failure, Ember not loaded.');
return false;
}
return api.getEmber().lookup(lookupFactory);
};
api.hook = function (lookupFactory, activateCb, deactivateCb) {
if (!api.isLoaded()) {
logger.debug('Factory hook failure, Ember not loaded.');
return false;
}
if (hookedFactories[lookupFactory]) {
logger.debug('Factory already hooked: ' + lookupFactory);
return true;
}
var reopenOptions = {};
var factory = api.lookup(lookupFactory);
if (!factory) {
logger.debug('Factory hook failure, factory not found: ' + lookupFactory);
return false;
}
if (activateCb) {
reopenOptions.activate = function () {
this._super();
activateCb.call(this);
logger.debug('Hook run on activate: ' + lookupFactory);
};
}
if (deactivateCb) {
reopenOptions.deactivate = function () {
this._super();
deactivateCb.call(this);
logger.debug('Hook run on deactivate: ' + lookupFactory);
};
}
try {
factory.reopen(reopenOptions);
hookedFactories[lookupFactory] = true;
logger.debug('Factory hooked: ' + lookupFactory);
return true;
}
catch (err) {
logger.debug('Factory hook failure, unexpected error: ' + lookupFactory);
logger.debug(err);
return false;
}
};
api.get = function (lookupFactory, property) {
if (!api.isLoaded()) {
logger.debug('Factory get failure, Ember not loaded.');
return false;
}
var properties = property.split('.');
var getter = api.lookup(lookupFactory);
properties.some(function (property) {
// If getter fails, just exit, otherwise, keep looping.
if (getter == null || typeof getter === 'undefined') {
getter = null;
return true;
}
if (getter[property] == null || typeof getter[property] === 'undefined') {
getter = null;
return true;
}
if (typeof getter.get === 'function') {
getter = getter.get(property);
if (getter == null || typeof getter === 'undefined') {
getter = null;
return true;
}
return false;
}
getter = getter[property];
});
return getter;
};
module.exports = api;
},{"./logger":10}],9:[function(require,module,exports){
var storage = require('./storage');
var logger = require('./logger');
var ui = require('./ui');
var api = {};
var emoteStore = new EmoteStore();
var $ = window.jQuery;
/**
* The entire emote storing system.
*/
function EmoteStore() {
var getters = {};
var nativeEmotes = {};
var hasInitialized = false;
/**
* Get a list of usable emoticons.
* @param {function} [filters] A filter method to limit what emotes are returned. Passed to Array#filter.
* @param {function|string} [sortBy] How the emotes should be sorted. `function` will be passed to sort via Array#sort. `'channel'` sorts by channel name, globals first. All other values (or omitted) sort alphanumerically.
* @param {string} [returnType] `'object'` will return in object format, e.g. `{'Kappa': Emote(...), ...}`. All other values (or omitted) return an array format, e.g. `[Emote(...), ...]`.
* @return {object|array} See `returnType` param.
*/
this.getEmotes = function (filters, sortBy, returnType) {
var twitchApi = require('./twitch-api');
// Get native emotes.
var emotes = $.extend({}, nativeEmotes);
// Parse the custom emotes provided by third party addons.
Object.keys(getters).forEach(function (getterName) {
// Try the getter.
var results = null;
try {
results = getters[getterName]();
}
catch (err) {
logger.debug('Emote getter `' + getterName + '` failed unexpectedly, skipping.', err.toString());
return;
}
if (!Array.isArray(results)) {
logger.debug('Emote getter `' + getterName + '` must return an array, skipping.');
return;
}
// Override natives and previous getters.
results.forEach(function (emote) {
try {
// Create the emote.
var instance = new Emote(emote);
// Force the getter.
instance.setGetterName(getterName);
// Force emotes without channels to the getter's name.
if (!emote.channel) {
instance.setChannelName(getterName);
}
// Add/override it.
emotes[instance.getText()] = instance;
}
catch (err) {
logger.debug('Emote parsing for getter `' + getterName + '` failed, skipping.', err.toString(), emote);
}
});
});
// Covert to array.
emotes = Object.keys(emotes).map(function (emote) {
return emotes[emote];
});
// Filter results.
if (typeof filters === 'function') {
emotes = emotes.filter(filters);
}
// Return as an object if requested.
if (returnType === 'object') {
var asObject = {};
emotes.forEach(function (emote) {
asObject[emote.getText()] = emote;
});
return asObject;
}
// Sort results.
if (typeof sortBy === 'function') {
emotes.sort(sortBy);
}
else if (sortBy === 'channel') {
emotes.sort(sorting.allEmotesCategory);
}
else {
emotes.sort(sorting.byText);
}
// Return the emotes in array format.
return emotes;
};
/**
* Registers a 3rd party emote hook.
* @param {string} name The name of the 3rd party registering the hook.
* @param {function} getter The function called when looking for emotes. Must return an array of emote objects, e.g. `[emote, ...]`. See Emote class.
*/
this.registerGetter = function (name, getter) {
if (typeof name !== 'string') {
throw new Error('Name must be a string.');
}
if (getters[name]) {
throw new Error('Getter already exists.');
}
if (typeof getter !== 'function') {
throw new Error('Getter must be a function.');
}
logger.debug('Getter registered: ' + name);
getters[name] = getter;
ui.updateEmotes();
};
/**
* Registers a 3rd party emote hook.
* @param {string} name The name of the 3rd party hook to deregister.
*/
this.deregisterGetter = function (name) {
logger.debug('Getter unregistered: ' + name);
delete getters[name];
ui.updateEmotes();
};
/**
* Initializes the raw data from the API endpoints. Should be called at load and/or whenever the API may have changed. Populates internal objects with updated data.
*/
this.init = function () {
if (hasInitialized) {
logger.debug('Already initialized EmoteStore, stopping init.');
return;
}
logger.debug('Starting initialization.');
var twitchApi = require('./twitch-api');
var self = this;
// Hash of emote set to forced channel.
var forcedSetsToChannels = {
// Globals.
0: '~global',
// Bubble emotes.
33: 'turbo',
// Monkey emotes.
42: 'turbo',
// Hidden turbo emotes.
457: 'turbo',
793: 'turbo',
19151: 'twitch_prime',
19194: 'twitch_prime'
};
logger.debug('Initializing emote set change listener.');
twitchApi.getEmotes(function (emoteSets) {
logger.debug('Parsing emote sets.');
Object.keys(emoteSets).forEach(function (set) {
var emotes = emoteSets[set];
set = Number(set);
emotes.forEach(function (emote) {
// Set some required info.
emote.url = '//static-cdn.jtvnw.net/emoticons/v1/' + emote.id + '/1.0';
emote.text = getEmoteFromRegEx(emote.code);
emote.set = set;
// Hardcode the channels of certain sets.
if (forcedSetsToChannels[set]) {
emote.channel = forcedSetsToChannels[set];
}
var instance = new Emote(emote);
// Save the emote for use later.
nativeEmotes[emote.text] = instance;
});
});
logger.debug('Loading subscription data.');
// Get active subscriptions to find the channels.
twitchApi.getTickets(function (tickets) {
// Instances from each channel to preload channel data.
var deferredChannelGets = {};
logger.debug('Tickets loaded from the API.', tickets);
tickets.forEach(function (ticket) {
var product = ticket.product;
var channel = product.owner_name || product.short_name;
// Get subscriptions with emotes only.
if (!product.emoticons || !product.emoticons.length) {
return;
}
// Set the channel on the emotes.
product.emoticons.forEach(function (emote) {
var instance = nativeEmotes[getEmoteFromRegEx(emote.regex)];
instance.setChannelName(channel);
// Save instance for later, but only one instance per channel.
if (!deferredChannelGets[channel]) {
deferredChannelGets[channel] = instance;
}
});
});
// Preload channel data.
Object.keys(deferredChannelGets).forEach(function (key) {
var instance = deferredChannelGets[key];
instance.getChannelBadge();
instance.getChannelDisplayName();
});
ui.updateEmotes();
});
ui.updateEmotes();
});
hasInitialized = true;
logger.debug('Finished EmoteStore initialization.');
};
};
/**
* Gets a specific emote, if available.
* @param {string} text The text of the emote to get.
* @return {Emote|null} The Emote instance of the emote or `null` if it couldn't be found.
*/
EmoteStore.prototype.getEmote = function (text) {
return this.getEmotes(null, null, 'object')[text] || null;
};
/**
* Emote object.
* @param {object} details Object describing the emote.
* @param {string} details.text The text to use in the chat box when emote is clicked.
* @param {string} details.url The URL of the image for the emote.
* @param {string} [details.badge] The URL of the badge for the emote.
* @param {string} [details.channel] The channel the emote should be categorized under.
* @param {string} [details.getterName] The 3rd party getter that registered the emote. Used internally only.
*/
function Emote(details) {
var text = null;
var url = null;
var getterName = null;
var channel = {
name: null,
displayName: null,
badge: null
};
/**
* Gets the text of the emote.
* @return {string} The emote text.
*/
this.getText = function () {
return text;
};
/**
* Sets the text of the emote.
* @param {string} theText The text to set.
*/
this.setText = function (theText) {
if (typeof theText !== 'string' || theText.length < 1) {
throw new Error('Invalid text');
}
text = theText;
};
/**
* Gets the getter name this emote belongs to.
* @return {string} The getter's name.
*/
this.getGetterName = function () {
return getterName;
};
/**
* Sets the getter name this emote belongs to.
* @param {string} theGetterName The getter's name.
*/
this.setGetterName = function (theGetterName) {
if (typeof theGetterName !== 'string' || theGetterName.length < 1) {
throw new Error('Invalid getter name');
}
getterName = theGetterName;
};
/**
* Gets the emote's image URL.
* @return {string} The emote image URL.
*/
this.getUrl = function () {
return url;
};
/**
* Sets the emote's image URL.
* @param {string} theUrl The image URL to set.
*/
this.setUrl = function (theUrl) {
if (typeof theUrl !== 'string' || theUrl.length < 1) {
throw new Error('Invalid URL');
}
url = theUrl;
};
/**
* Gets the emote's channel name.
* @return {string} The emote's channel or an empty string if it doesn't have one.
*/
this.getChannelName = function () {
if (!channel.name) {
channel.name = storage.channelNames.get(this.getText());
}
return channel.name || '';
};
/**
* Sets the emote's channel name.
* @param {string} theChannel The channel name to set.
*/
this.setChannelName = function (theChannel) {
if (typeof theChannel !== 'string' || theChannel.length < 1) {
throw new Error('Invalid channel');
}
// Only save the channel to storage if it's dynamic.
if (theChannel !== '~global' && theChannel !== 'turbo' && theChannel !== 'twitch_prime') {
storage.channelNames.set(this.getText(), theChannel);
}
channel.name = theChannel;
};
/**
* Gets the emote channel's badge image URL.
* @return {string|null} The URL of the badge image for the emote's channel or `null` if it doesn't have a channel.
*/
this.getChannelBadge = function () {
var twitchApi = require('./twitch-api');
var channelName = this.getChannelName();
var defaultBadge = '//static-cdn.jtvnw.net/jtv_user_pictures/subscriber-star.png';
// No channel.
if (!channelName) {
return null;
}
// Give globals a default badge.
if (channelName === '~global') {
return '/favicon.ico';
}
// Already have one preset.
if (channel.badge) {
return channel.badge;
}
// Check storage.
channel.badge = storage.badges.get(channelName);
if (channel.badge !== null) {
return channel.badge;
}
// Set default until API returns something.
channel.badge = defaultBadge;
// Get from API.
logger.debug('Getting fresh badge for: ' + channelName);
twitchApi.getBadges(channelName, function (badges) {
var badge = null;
// Save turbo badge while we are here.
if (badges.turbo && badges.turbo.image) {
badge = badges.turbo.image;
storage.badges.set('turbo', badge, 86400000);
// Turbo is actually what we wanted, so we are done.
if (channelName === 'turbo') {
channel.badge = badge;
return;
}
}
// Save turbo badge while we are here.
if (badges.premium && badges.premium.image) {
badge = badges.premium.image;
storage.badges.set('twitch_prime', badge, 86400000);
// Turbo is actually what we wanted, so we are done.
if (channelName === 'twitch_prime') {
channel.badge = badge;
return;
}
}
// Save subscriber badge in storage.
if (badges.subscriber && badges.subscriber.image) {
channel.badge = badges.subscriber.image;
storage.badges.set(channelName, channel.badge, 86400000);
ui.updateEmotes();
}
// No subscriber badge.
else {
channel.badge = defaultBadge;
logger.debug('Failed to get subscriber badge for: ' + channelName);
}
});
return channel.badge || defaultBadge;
};
/**
* Sets the emote's channel badge image URL.
* @param {string} theBadge The badge image URL to set.
*/
this.setChannelBadge = function (theBadge) {
if (typeof theBadge !== 'string' || theBadge.length < 1) {
throw new Error('Invalid badge');
}
channel.badge = theBadge;
};
/**
* Get a channel's display name.
* @return {string} The channel's display name. May be equivalent to the channel the first time the API needs to be called.
*/
this.getChannelDisplayName = function () {
var twitchApi = require('./twitch-api');
var channelName = this.getChannelName();
var self = this;
var forcedChannelToDisplayNames = {
'~global': 'Global',
'turbo': 'Twitch Turbo',
'twitch_prime': 'Twitch Prime'
};
// No channel.
if (!channelName) {
return '';
}
// Forced display name.
if (forcedChannelToDisplayNames[channelName]) {
return forcedChannelToDisplayNames[channelName];
}
// Already have one preset.
if (channel.displayName) {
return channel.displayName;
}
// Look for obvious bad channel names that shouldn't hit the API or storage. Use channel name instead.
if (/[^a-z0-9_]/.test(channelName)) {
logger.debug('Unable to get display name due to obvious non-standard channel name for: ' + channelName);
return channelName;
}
// Check storage.
channel.displayName = storage.displayNames.get(channelName);
if (channel.displayName !== null) {
return channel.displayName;
}
// Get from API.
else {
// Set default until API returns something.
channel.displayName = channelName;
logger.debug('Getting fresh display name for: ' + channelName);
twitchApi.getUser(channelName, function (user) {
if (!user || !user.display_name) {
logger.debug('Failed to get display name for: ' + channelName);
return;
}
// Save it.
self.setChannelDisplayName(user.display_name);
ui.updateEmotes();
});
}
return channel.displayName;
};
/**
* Sets the emote's channel badge image URL.
* @param {string} theBadge The badge image URL to set.
*/
this.setChannelDisplayName = function (displayName) {
if (typeof displayName !== 'string' || displayName.length < 1) {
throw new Error('Invalid displayName');
}
channel.displayName = displayName;
storage.displayNames.set(this.getChannelName(), displayName, 86400000);
};
/**
* Initialize the details.
*/
// Required fields.
this.setText(details.text);
this.setUrl(details.url);
// Optional fields.
if (details.getterName) {
this.setGetterName(details.getterName);
}
if (details.channel) {
this.setChannelName(details.channel);
}
if (details.channelDisplayName) {
this.setChannelDisplayName(details.channelDisplayName);
}
if (details.badge) {
this.setChannelBadge(details.badge);
}
};
/**
* State changers.
*/
/**
* Toggle whether an emote should be a favorite.
* @param {boolean} [force] `true` forces the emote to be a favorite, `false` forces the emote to not be a favorite.
*/
Emote.prototype.toggleFavorite = function (force) {
if (typeof force !== 'undefined') {
storage.starred.set(this.getText(), !!force);
return;
}
storage.starred.set(this.getText(), !this.isFavorite());
};
/**
* Toggle whether an emote should be visible out of editing mode.
* @param {boolean} [force] `true` forces the emote to be visible, `false` forces the emote to be hidden.
*/
Emote.prototype.toggleVisibility = function (force) {
if (typeof force !== 'undefined') {
storage.visibility.set(this.getText(), !!force);
return;
}
storage.visibility.set(this.getText(), !this.isVisible());
};
/**
* State getters.
*/
/**
* Whether the emote is from a 3rd party.
* @return {boolean} Whether the emote is from a 3rd party.
*/
Emote.prototype.isThirdParty = function () {
return !!this.getGetterName();
};
/**
* Whether the emote was favorited.
* @return {boolean} Whether the emote was favorited.
*/
Emote.prototype.isFavorite = function () {
return storage.starred.get(this.getText(), false);
};
/**
* Whether the emote is visible outside of editing mode.
* @return {boolean} Whether the emote is visible outside of editing mode.
*/
Emote.prototype.isVisible = function () {
return storage.visibility.get(this.getText(), true);
};
/**
* Whether the emote is considered a simple smiley.
* @return {boolean} Whether the emote is considered a simple smiley.
*/
Emote.prototype.isSmiley = function () {
// The basic smiley emotes.
var emotes = [':(', ':)', ':/', ':\\', ':D', ':o', ':p', ':z', ';)', ';p', '<3', '>(', 'B)', 'R)', 'o_o', 'O_O', '#/', ':7', ':>', ':S', '<]'];
return emotes.indexOf(this.getText()) !== -1;
};
/**
* Property getters/setters.
*/
/**
* Gets the usable emote text from a regex.
*/
function getEmoteFromRegEx(regex) {
if (typeof regex === 'string') {
regex = new RegExp(regex);
}
if (!regex) {
throw new Error('`regex` must be a RegExp string or object.');
}
return decodeURI(regex.source)
// Replace HTML entity brackets with actual brackets.
.replace('&gt\\;', '>')
.replace('&lt\\;', '<')
// Remove negative groups.
//
// /
// \(\?! // (?!
// [^)]* // any amount of characters that are not )
// \) // )
// /g
.replace(/\(\?![^)]*\)/g, '')
// Pick first option from a group.
//
// /
// \( // (
// ([^|])* // any amount of characters that are not |
// \|? // an optional | character
// [^)]* // any amount of characters that are not )
// \) // )
// /g
.replace(/\(([^|])*\|?[^)]*\)/g, '$1')
// Pick first character from a character group.
//
// /
// \[ // [
// ([^|\]\[])* // any amount of characters that are not |, [, or ]
// \|? // an optional | character
// [^\]]* // any amount of characters that are not [, or ]
// \] // ]
// /g
.replace(/\[([^|\]\[])*\|?[^\]\[]*\]/g, '$1')
// Remove optional characters.
//
// /
// [^\\] // any character that is not \
// \? // ?
// /g
.replace(/[^\\]\?/g, '')
// Remove boundaries at beginning and end.
.replace(/^\\b|\\b$/g, '')
// Unescape only single backslash, not multiple.
//
// /
// \\ // \
// (?!\\) // look-ahead, any character that isn't \
// /g
.replace(/\\(?!\\)/g, '');
}
var sorting = {};
/**
* Sort by alphanumeric in this order: symbols -> numbers -> AaBb... -> numbers
*/
sorting.byText = function (a, b) {
textA = a.getText().toLowerCase();
textB = b.getText().toLowerCase();
if (textA < textB) {
return -1;
}
if (textA > textB) {
return 1;
}
return 0;
}
/**
* Basic smilies before non-basic smilies.
*/
sorting.bySmiley = function (a, b) {
if (a.isSmiley() && !b.isSmiley()) {
return -1;
}
if (b.isSmiley() && !a.isSmiley()) {
return 1;
}
return 0;
};
/**
* Globals before subscription emotes, subscriptions in alphabetical order.
*/
sorting.byChannelName = function (a, b) {
var channelA = a.getChannelName();
var channelB = b.getChannelName();
// Both don't have channels.
if (!channelA && !channelB) {
return 0;
}
// "A" has channel, "B" doesn't.
if (channelA && !channelB) {
return 1;
}
// "B" has channel, "A" doesn't.
if (channelB && !channelA) {
return -1;
}
channelA = channelA.toLowerCase();
channelB = channelB.toLowerCase();
if (channelA < channelB) {
return -1;
}
if (channelB > channelA) {
return 1;
}
// All the same
return 0;
};
/**
* The general sort order for the all emotes category.
* Smileys -> Channel grouping -> alphanumeric
*/
sorting.allEmotesCategory = function (a, b) {
var bySmiley = sorting.bySmiley(a, b);
var byChannelName = sorting.byChannelName(a, b);
var byText = sorting.byText(a, b);
if (bySmiley !== 0) {
return bySmiley;
}
if (byChannelName !== 0) {
return byChannelName;
}
return byText;
};
module.exports = emoteStore;
},{"./logger":10,"./storage":12,"./twitch-api":14,"./ui":15}],10:[function(require,module,exports){
var api = {};
var instance = '[instance ' + (Math.floor(Math.random() * (999 - 100)) + 100) + '] ';
var prefix = '[Emote Menu] ';
var storage = require('./storage');
api.log = function () {
if (typeof console.log === 'undefined') {
return;
}
arguments = [].slice.call(arguments).map(function (arg) {
if (typeof arg !== 'string') {
return JSON.stringify(arg);
}
return arg;
});
if (storage.global.get('debugMessagesEnabled', false)) {
arguments.unshift(instance);
}
arguments.unshift(prefix);
console.log.apply(console, arguments);
};
api.debug = function () {
if (!storage.global.get('debugMessagesEnabled', false)) {
return;
}
arguments = [].slice.call(arguments);
arguments.unshift('[DEBUG] ');
api.log.apply(null, arguments);
}
module.exports = api;
},{"./storage":12}],11:[function(require,module,exports){
var storage = require('./storage');
var logger = require('./logger');
var emotes = require('./emotes');
var api = {};
api.toggleDebug = function (forced) {
if (typeof forced === 'undefined') {
forced = !storage.global.get('debugMessagesEnabled', false);
}
else {
forced = !!forced;
}
storage.global.set('debugMessagesEnabled', forced);
logger.log('Debug messages are now ' + (forced ? 'enabled' : 'disabled'));
};
api.registerEmoteGetter = emotes.registerGetter;
api.deregisterEmoteGetter = emotes.deregisterGetter;
api.ember = require('./ember-api');
api.twitchAPI = require('./twitch-api');
module.exports = api;
},{"./ember-api":8,"./emotes":9,"./logger":10,"./storage":12,"./twitch-api":14}],12:[function(require,module,exports){
var Store = require('storage-wrapper');
var storage = {};
// General storage.
storage.global = new Store({
namespace: 'emote-menu-for-twitch'
});
// Emote visibility storage.
storage.visibility = storage.global.createSubstore('visibility');
// Emote starred storage.
storage.starred = storage.global.createSubstore('starred');
// Display name storage.
storage.displayNames = storage.global.createSubstore('displayNames');
// Channel name storage.
storage.channelNames = storage.global.createSubstore('channelNames');
// Badges storage.
storage.badges = storage.global.createSubstore('badges');
module.exports = storage;
},{"storage-wrapper":6}],13:[function(require,module,exports){
var templates = require('../../build/templates');
module.exports = (function () {
var data = {};
var key = null;
// Convert templates to their shorter "render" form.
for (key in templates) {
if (!templates.hasOwnProperty(key)) {
continue;
}
data[key] = render(key);
}
// Shortcut the render function. All templates will be passed in as partials by default.
function render(template) {
template = templates[template];
return function (context, partials, indent) {
return template.render(context, partials || templates, indent);
};
}
return data;
})();
},{"../../build/templates":3}],14:[function(require,module,exports){
var twitchApi = window.Twitch.api;
var jQuery = window.jQuery;
var logger = require('./logger');
var api = {};
api.getBadges = function (username, callback) {
if (
[
'~global',
'turbo',
'twitch_prime'
].indexOf(username) > -1
) {
if (!jQuery) {
callback({});
}
// Note: not a documented API endpoint.
jQuery.getJSON('https://badges.twitch.tv/v1/badges/global/display')
.done(function (api) {
var badges = {
turbo: {
image: api.badge_sets.turbo.versions['1'].image_url_1x
},
premium: {
image: api.badge_sets.premium.versions['1'].image_url_1x
}
};
callback(badges);
})
.fail(function () {
callback({});
});
}
else {
twitchApi.get('chat/' + username + '/badges')
.done(function (api) {
callback(api);
})
.fail(function () {
callback({});
});
}
};
api.getUser = function (username, callback) {
// Note: not a documented API endpoint.
twitchApi.get('users/' + username)
.done(function (api) {
callback(api);
})
.fail(function () {
callback({});
});
};
api.getTickets = function (callback) {
// Note: not a documented API endpoint.
twitchApi.get(
'/api/users/:login/tickets',
{
offset: 0,
limit: 100,
unended: true
}
)
.done(function (api) {
callback(api.tickets || []);
})
.fail(function () {
callback([]);
});
};
api.getEmotes = function (callback) {
twitchApi.get('users/:login/emotes')
.done(function (response) {
if (!response || !response.emoticon_sets) {
logger.debug('getEmotes emoticon_sets empty');
callback({});
return;
}
callback(response.emoticon_sets);
})
.fail(function () {
logger.debug('getEmotes API call failed');
callback({});
});
};
api.raw = twitchApi;
module.exports = api;
},{"./logger":10}],15:[function(require,module,exports){
var api = {};
var $ = jQuery = window.jQuery;
var templates = require('./templates');
var storage = require('./storage');
var emotes = require('./emotes');
var logger = require('./logger');
var theMenu = new UIMenu();
var theMenuButton = new UIMenuButton();
api.init = function () {
// Load CSS.
require('../../build/styles');
// Load jQuery plugins.
require('../plugins/resizable');
require('jquery.scrollbar');
theMenuButton.init();
theMenu.init();
};
api.hideMenu = function () {
if (theMenu.dom && theMenu.dom.length) {
theMenu.toggleDisplay(false);
}
};
api.updateEmotes = function () {
theMenu.updateEmotes();
}
function UIMenuButton() {
this.dom = null;
}
UIMenuButton.prototype.init = function (timesFailed) {
var self = this;
var chatButton = $('.send-chat-button, .chat-buttons-container button');
var failCounter = timesFailed || 0;
this.dom = $('#emote-menu-button');
// Element already exists.
if (this.dom.length) {
logger.debug('MenuButton already exists, stopping init.');
return this;
}
if (!chatButton.length) {
failCounter += 1;
if (failCounter === 1) {
logger.log('MenuButton container missing, trying again.');
}
if (failCounter >= 10) {
logger.log('MenuButton container missing, MenuButton unable to be added, stopping init.');
return this;
}
setTimeout(function () {
self.init(failCounter);
}, 1000);
return this;
}
// Create element.
this.dom = $(templates.emoteButton());
this.dom.insertBefore(chatButton);
// Hide then fade it in.
this.dom.hide();
this.dom.fadeIn();
// Enable clicking.
this.dom.on('click', function () {
theMenu.toggleDisplay();
});
return this;
};
UIMenuButton.prototype.toggleDisplay = function (forced) {
var state = typeof forced !== 'undefined' ? !!forced : !this.isVisible();
if (state) {
this.dom.addClass('active');
return this;
}
this.dom.removeClass('active');
return this;
};
UIMenuButton.prototype.isVisible = function () {
return this.dom.hasClass('active');
};
function UIMenu() {
this.dom = null;
this.groups = {};
this.emotes = [];
this.offset = null;
this.favorites = null;
}
UIMenu.prototype.init = function () {
var logger = require('./logger');
var self = this;
this.dom = $('#emote-menu-for-twitch');
// Element already exists.
if (this.dom.length) {
return this;
}
// Create element.
this.dom = $(templates.menu());
$(document.body).append(this.dom);
this.favorites = new UIFavoritesGroup();
// Enable dragging.
this.dom.draggable({
handle: '.draggable',
start: function () {
self.togglePinned(true);
self.toggleMovement(true);
},
stop: function () {
self.offset = self.dom.offset();
},
containment: $(document.body)
});
// Enable resizing.
this.dom.resizable({
handle: '[data-command="resize-handle"]',
stop: function () {
self.togglePinned(true);
self.toggleMovement(true);
},
alsoResize: self.dom.find('.scrollable'),
containment: $(document.body),
minHeight: 180,
minWidth: 200
});
// Enable pinning.
this.dom.find('[data-command="toggle-pinned"]').on('click', function () {
self.togglePinned();
});
// Enable editing.
this.dom.find('[data-command="toggle-editing"]').on('click', function () {
self.toggleEditing();
});
this.dom.find('.scrollable').scrollbar()
this.updateEmotes();
return this;
};
UIMenu.prototype._detectOutsideClick = function (event) {
// Not outside of the menu, ignore the click.
if ($(event.target).is('#emote-menu-for-twitch, #emote-menu-for-twitch *')) {
return;
}
// Clicked on the menu button, just remove the listener and let the normal listener handle it.
if (!this.isVisible() || $(event.target).is('#emote-menu-button, #emote-menu-button *')) {
$(document).off('mouseup', this._detectOutsideClick.bind(this));
return;
}
// Clicked outside, make sure the menu isn't pinned.
if (!this.isPinned()) {
// Menu wasn't pinned, remove listener.
$(document).off('mouseup', this._detectOutsideClick.bind(this));
this.toggleDisplay();
}
};
UIMenu.prototype.toggleDisplay = function (forced) {
var state = typeof forced !== 'undefined' ? !!forced : !this.isVisible();
var loggedIn = window.Twitch && window.Twitch.user.isLoggedIn();
// Menu should be shown.
if (state) {
// Check if user is logged in.
if (!loggedIn) {
// Call native login form.
$.login();
return this;
}
this.updateEmotes();
this.dom.show();
// Menu moved, move it back.
if (this.hasMoved()) {
this.dom.offset(this.offset);
}
// Never moved, make it the same size as the chat window.
else {
var chatContainer = $('.chat-messages');
// Adjust the size to be the same as the chat container.
this.dom.height(chatContainer.outerHeight() - (this.dom.outerHeight() - this.dom.height()));
this.dom.width(chatContainer.outerWidth() - (this.dom.outerWidth() - this.dom.width()));
// Adjust the offset to be the same as the chat container.
this.offset = chatContainer.offset();
this.dom.offset(this.offset);
}
// Listen for outside click.
$(document).on('mouseup', this._detectOutsideClick.bind(this));
}
// Menu should be hidden.
else {
this.dom.hide();
this.toggleEditing(false);
this.togglePinned(false);
}
// Also toggle the menu button.
theMenuButton.toggleDisplay(this.isVisible());
return this;
};
UIMenu.prototype.isVisible = function () {
return this.dom.is(':visible');
};
UIMenu.prototype.updateEmotes = function (which) {
var emote = which ? this.getEmote(which) : null;
var favoriteEmote = emote ? this.favorites.getEmote(which) : null;
if (emote) {
emote.update();
if (favoriteEmote) {
favoriteEmote.update();
}
return this;
}
var emotes = require('./emotes');
var theEmotes = emotes.getEmotes();
var theEmotesKeys = [];
var self = this;
theEmotes.forEach(function (emoteInstance) {
self.addEmote(emoteInstance);
theEmotesKeys.push(emoteInstance.getText());
});
// Difference the emotes and remove all non-valid emotes.
this.emotes.forEach(function (oldEmote) {
var text = oldEmote.getText()
if (theEmotesKeys.indexOf(text) < 0) {
logger.debug('Emote difference found, removing emote from UI: ' + text);
self.removeEmote(text);
}
});
// Save the emotes for next differencing.
this.emotes = theEmotes;
//Update groups.
Object.keys(this.groups).forEach(function (group) {
self.getGroup(group).init();
});
return this;
};
UIMenu.prototype.toggleEditing = function (forced) {
var state = typeof forced !== 'undefined' ? !!forced : !this.isEditing();
this.dom.toggleClass('editing', state);
return this;
};
UIMenu.prototype.isEditing = function () {
return this.dom.hasClass('editing');
};
UIMenu.prototype.togglePinned = function (forced) {
var state = typeof forced !== 'undefined' ? !!forced : !this.isPinned();
this.dom.toggleClass('pinned', state);
return this;
};
UIMenu.prototype.isPinned = function () {
return this.dom.hasClass('pinned');
};
UIMenu.prototype.toggleMovement = function (forced) {
var state = typeof forced !== 'undefined' ? !!forced : !this.hasMoved();
this.dom.toggleClass('moved', state);
return this;
};
UIMenu.prototype.hasMoved = function () {
return this.dom.hasClass('moved');
};
UIMenu.prototype.addGroup = function (emoteInstance) {
var channel = emoteInstance.getChannelName();
var self = this;
// Already added, don't add again.
if (this.getGroup(channel)) {
return this;
}
// Add to current menu groups.
var group = new UIGroup(emoteInstance);
this.groups[channel] = group;
// Sort group names, get index of where this group should go.
var keys = Object.keys(this.groups);
keys.sort(function (a, b) {
// Get the instances.
a = self.groups[a].emoteInstance;
b = self.groups[b].emoteInstance;
// Get the channel name.
var aChannel = a.getChannelName();
var bChannel = b.getChannelName();
// Get the channel display name.
a = a.getChannelDisplayName().toLowerCase();
b = b.getChannelDisplayName().toLowerCase();
// Prime goes first, always.
if (aChannel === 'twitch_prime' && bChannel !== 'twitch_prime') {
return -1;
}
if (bChannel === 'twitch_prime' && aChannel !== 'twitch_prime') {
return 1;
}
// Turbo goes after Prime, always.
if (aChannel === 'turbo' && bChannel !== 'turbo') {
return -1;
}
if (bChannel === 'turbo' && aChannel !== 'turbo') {
return 1;
}
// Global goes after Turbo, always.
if (aChannel === '~global' && bChannel !== '~global') {
return -1;
}
if (bChannel === '~global' && aChannel !== '~global') {
return 1;
}
// A goes first.
if (a < b) {
return -1;
}
// B goest first.
if (a > b) {
return 1;
}
// Both the same, doesn't matter.
return 0;
});
var index = keys.indexOf(channel);
// First in the sort, place at the beginning of the menu.
if (index === 0) {
group.dom.prependTo(this.dom.find('#all-emotes-group'));
}
// Insert after the previous group in the sort.
else {
group.dom.insertAfter(this.getGroup(keys[index - 1]).dom);
}
return group;
};
UIMenu.prototype.getGroup = function (name) {
return this.groups[name] || null;
};
UIMenu.prototype.addEmote = function (emoteInstance) {
// Get the group, or add if needed.
var group = this.getGroup(emoteInstance.getChannelName()) || this.addGroup(emoteInstance);
group.addEmote(emoteInstance);
group.toggleDisplay(group.isVisible(), true);
this.favorites.addEmote(emoteInstance);
return this;
};
UIMenu.prototype.removeEmote = function (name) {
var self = this;
Object.keys(this.groups).forEach(function (groupName) {
self.groups[groupName].removeEmote(name);
});
this.favorites.removeEmote(name);
return this;
};
UIMenu.prototype.getEmote = function (name) {
var groupName = null;
var group = null;
var emote = null;
for (groupName in this.groups) {
group = this.groups[groupName];
emote = group.getEmote(name);
if (emote) {
return emote;
}
}
return null;
};
function UIGroup(emoteInstance) {
this.dom = null;
this.emotes = {};
this.emoteInstance = emoteInstance;
this.init();
}
UIGroup.prototype.init = function () {
var self = this;
var emoteInstance = this.emoteInstance;
// First init, create new DOM.
if (this.dom === null) {
this.dom = $(templates.emoteGroupHeader({
badge: emoteInstance.getChannelBadge(),
channel: emoteInstance.getChannelName(),
channelDisplayName: emoteInstance.getChannelDisplayName()
}));
}
// Update DOM instead.
else {
this.dom.find('.header-info').replaceWith(
$(templates.emoteGroupHeader({
badge: emoteInstance.getChannelBadge(),
channel: emoteInstance.getChannelName(),
channelDisplayName: emoteInstance.getChannelDisplayName()
}))
.find('.header-info')
);
}
// Enable emote hiding.
this.dom.find('.header-info [data-command="toggle-visibility"]').on('click', function () {
if (!theMenu.isEditing()) {
return;
}
self.toggleDisplay();
});
this.toggleDisplay(this.isVisible(), true);
};
UIGroup.prototype.toggleDisplay = function (forced, skipUpdatingEmoteDisplay) {
var self = this;
var state = typeof forced !== 'undefined' ? !forced : this.isVisible();
this.dom.toggleClass('emote-menu-hidden', state);
// Update the display of all emotes.
if (!skipUpdatingEmoteDisplay) {
Object.keys(this.emotes).forEach(function (emoteName) {
self.emotes[emoteName].toggleDisplay(!state);
theMenu.updateEmotes(self.emotes[emoteName].instance.getText());
});
}
return this;
};
UIGroup.prototype.isVisible = function () {
var self = this;
// If any emote is visible, the group should be visible.
return Object.keys(this.emotes).some(function (emoteName) {
return self.emotes[emoteName].isVisible();
});
};
UIGroup.prototype.addEmote = function (emoteInstance) {
var self = this;
var emote = this.getEmote(emoteInstance.getText());
// Already added, update instead.
if (emote) {
emote.update();
return this;
}
// Add to current emotes.
emote = new UIEmote(emoteInstance);
this.emotes[emoteInstance.getText()] = emote;
var keys = Object.keys(this.emotes);
keys.sort(function (a, b) {
// Get the emote instances.
a = self.emotes[a].instance;
b = self.emotes[b].instance;
// A is a smiley, B isn't. A goes first.
if (a.isSmiley() && !b.isSmiley()) {
return -1;
}
// B is a smiley, A isn't. B goes first.
if (b.isSmiley() && !a.isSmiley()) {
return 1;
}
// Get the text of the emotes.
a = a.getText().toLowerCase();
b = b.getText().toLowerCase();
// A goes first.
if (a < b) {
return -1;
}
// B goest first.
if (a > b) {
return 1;
}
// Both the same, doesn't matter.
return 0;
});
var index = keys.indexOf(emoteInstance.getText());
// First in the sort, place at the beginning of the group.
if (index === 0) {
emote.dom.prependTo(this.dom.find('.emote-container'));
}
// Insert after the previous emote in the sort.
else {
emote.dom.insertAfter(this.getEmote(keys[index - 1]).dom);
}
return this;
};
UIGroup.prototype.getEmote = function (name) {
return this.emotes[name] || null;
};
UIGroup.prototype.removeEmote = function (name) {
var emote = this.getEmote(name);
if (!emote) {
return this;
}
emote.dom.remove();
delete this.emotes[name];
return this;
};
function UIFavoritesGroup() {
this.dom = $('#starred-emotes-group');
this.emotes = {};
}
UIFavoritesGroup.prototype.addEmote = UIGroup.prototype.addEmote;
UIFavoritesGroup.prototype.getEmote = UIGroup.prototype.getEmote;
UIFavoritesGroup.prototype.removeEmote = UIGroup.prototype.removeEmote;
function UIEmote(emoteInstance) {
this.dom = null;
this.instance = emoteInstance;
this.init();
}
UIEmote.prototype.init = function () {
var self = this;
// Create element.
this.dom = $(templates.emote({
url: this.instance.getUrl(),
text: this.instance.getText(),
thirdParty: this.instance.isThirdParty(),
isVisible: this.instance.isVisible(),
isStarred: this.instance.isFavorite()
}));
// Enable clicking.
this.dom.on('click', function () {
if (!theMenu.isEditing()) {
self.addToChat();
// Close the menu if not pinned.
if (!theMenu.isPinned()) {
theMenu.toggleDisplay();
}
}
});
// Enable emote hiding.
this.dom.find('[data-command="toggle-visibility"]').on('click', function () {
if (!theMenu.isEditing()) {
return;
}
self.toggleDisplay();
theMenu.updateEmotes(self.instance.getText());
});
// Enable emote favoriting.
this.dom.find('[data-command="toggle-starred"]').on('click', function () {
if (!theMenu.isEditing()) {
return;
}
self.toggleFavorite();
theMenu.updateEmotes(self.instance.getText());
});
return this;
};
UIEmote.prototype.toggleDisplay = function (forced, skipInstanceUpdate) {
var state = typeof forced !== 'undefined' ? !forced : this.isVisible();
this.dom.toggleClass('emote-menu-hidden', state);
if (!skipInstanceUpdate) {
this.instance.toggleVisibility(!state);
}
var group = this.getGroup();
group.toggleDisplay(group.isVisible(), true);
return this;
};
UIEmote.prototype.isVisible = function () {
return !this.dom.hasClass('emote-menu-hidden');
};
UIEmote.prototype.toggleFavorite = function (forced, skipInstanceUpdate) {
var state = typeof forced !== 'undefined' ? !!forced : !this.isFavorite();
this.dom.toggleClass('emote-menu-starred', state);
if (!skipInstanceUpdate) {
this.instance.toggleFavorite(state);
}
return this;
};
UIEmote.prototype.isFavorite = function () {
return this.dom.hasClass('emote-menu-starred');
};
UIEmote.prototype.addToChat = function () {
var ember = require('./ember-api');
// Get textarea element.
var element = $('.chat-interface textarea').get(0);
var text = this.instance.getText();
// Insert at cursor / replace selection.
// https://developer.mozilla.org/en-US/docs/Code_snippets/Miscellaneous
var selectionEnd = element.selectionStart + text.length;
var currentValue = element.value;
var beforeText = currentValue.substring(0, element.selectionStart);
var afterText = currentValue.substring(element.selectionEnd, currentValue.length);
// Smart padding, only put space at start if needed.
if (
beforeText !== '' &&
beforeText.substr(-1) !== ' '
) {
text = ' ' + text;
}
// Always put space at end.
text = beforeText + text + ' ' + afterText;
// Set the text.
ember.get('controller:chat', 'currentRoom').set('messageToSend', text);
element.focus();
// Put cursor at end.
selectionEnd = element.selectionStart + text.length;
element.setSelectionRange(selectionEnd, selectionEnd);
return this;
};
UIEmote.prototype.getGroup = function () {
return theMenu.getGroup(this.instance.getChannelName());
};
UIEmote.prototype.update = function () {
this.toggleDisplay(this.instance.isVisible(), true);
this.toggleFavorite(this.instance.isFavorite(), true);
};
module.exports = api;
},{"../../build/styles":2,"../plugins/resizable":16,"./ember-api":8,"./emotes":9,"./logger":10,"./storage":12,"./templates":13,"jquery.scrollbar":5}],16:[function(require,module,exports){
(function ($) {
$.fn.resizable = function (options) {
var settings = $.extend({
alsoResize: null,
alsoResizeType: 'both', // `height`, `width`, `both`
containment: null,
create: null,
destroy: null,
handle: '.resize-handle',
maxHeight: 9999,
maxWidth: 9999,
minHeight: 0,
minWidth: 0,
resize: null,
resizeOnce: null,
snapSize: 1,
start: null,
stop: null
}, options);
settings.element = $(this);
function recalculateSize(evt) {
var data = evt.data,
resized = {};
data.diffX = Math.round((evt.pageX - data.pageX) / settings.snapSize) * settings.snapSize;
data.diffY = Math.round((evt.pageY - data.pageY) / settings.snapSize) * settings.snapSize;
if (Math.abs(data.diffX) > 0 || Math.abs(data.diffY) > 0) {
if (
settings.element.height() !== data.height + data.diffY &&
data.height + data.diffY >= settings.minHeight &&
data.height + data.diffY <= settings.maxHeight &&
(settings.containment ? data.outerHeight + data.diffY + data.offset.top <= settings.containment.offset().top + settings.containment.outerHeight() : true)
) {
settings.element.height(data.height + data.diffY);
resized.height = true;
}
if (
settings.element.width() !== data.width + data.diffX &&
data.width + data.diffX >= settings.minWidth &&
data.width + data.diffX <= settings.maxWidth &&
(settings.containment ? data.outerWidth + data.diffX + data.offset.left <= settings.containment.offset().left + settings.containment.outerWidth() : true)
) {
settings.element.width(data.width + data.diffX);
resized.width = true;
}
if (resized.height || resized.width) {
if (settings.resizeOnce) {
settings.resizeOnce.bind(settings.element)(evt.data);
settings.resizeOnce = null;
}
if (settings.resize) {
settings.resize.bind(settings.element)(evt.data);
}
if (settings.alsoResize) {
if (resized.height && (settings.alsoResizeType === 'height' || settings.alsoResizeType === 'both')) {
settings.alsoResize.height(data.alsoResizeHeight + data.diffY);
}
if (resized.width && (settings.alsoResizeType === 'width' || settings.alsoResizeType === 'both')) {
settings.alsoResize.width(data.alsoResizeWidth + data.diffX);
}
}
}
}
}
function start(evt) {
evt.preventDefault();
if (settings.start) {
settings.start.bind(settings.element)();
}
var data = {
alsoResizeHeight: settings.alsoResize ? settings.alsoResize.height() : 0,
alsoResizeWidth: settings.alsoResize ? settings.alsoResize.width() : 0,
height: settings.element.height(),
offset: settings.element.offset(),
outerHeight: settings.element.outerHeight(),
outerWidth: settings.element.outerWidth(),
pageX: evt.pageX,
pageY: evt.pageY,
width: settings.element.width()
};
$(document).on('mousemove', '*', data, recalculateSize);
$(document).on('mouseup', '*', stop);
}
function stop() {
if (settings.stop) {
settings.stop.bind(settings.element)();
}
$(document).off('mousemove', '*', recalculateSize);
$(document).off('mouseup', '*', stop);
}
if (settings.handle) {
if (settings.alsoResize && ['both', 'height', 'width'].indexOf(settings.alsoResizeType) >= 0) {
settings.alsoResize = $(settings.alsoResize);
}
if (settings.containment) {
settings.containment = $(settings.containment);
}
settings.handle = $(settings.handle);
settings.snapSize = settings.snapSize < 1 ? 1 : settings.snapSize;
if (options === 'destroy') {
settings.handle.off('mousedown', start);
if (settings.destroy) {
settings.destroy.bind(this)();
}
return this;
}
settings.handle.on('mousedown', start);
if (settings.create) {
settings.create.bind(this)();
}
}
return this;
};
})(jQuery);
},{}]},{},[1])
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules\\browserify\\node_modules\\browser-pack\\_prelude.js","./src/script.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/build/styles.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/build/templates.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/node_modules/hogan.js/lib/template.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/node_modules/jquery.scrollbar/jquery.scrollbar.min.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/node_modules/storage-wrapper/index.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/package.json","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/ember-api.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/emotes.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/logger.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/public-api.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/storage.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/templates.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/twitch-api.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/ui.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/plugins/resizable.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/uBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7rBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","var pkg = require('../package.json');\r\nvar publicApi = require('./modules/public-api');\r\nvar ember = require('./modules/ember-api');\r\nvar logger = require('./modules/logger');\r\nvar emotes = require('./modules/emotes');\r\nvar ui = require('./modules/ui');\r\n\r\nlogger.log('(v'+ pkg.version + ') Initial load on ' + location.href);\r\n\r\n// Only enable script if we have the right variables.\r\n//---------------------------------------------------\r\nvar initTimer = 0;\r\n(function init(time) {\t\r\n\tif (!time) {\r\n\t\ttime = 0;\r\n\t}\r\n\r\n\tvar objectsLoaded = (\r\n\t\twindow.Twitch !== undefined &&\r\n\t\twindow.jQuery !== undefined &&\r\n\t\tember.isLoaded()\r\n\t);\r\n\tif (!objectsLoaded) {\r\n\t\t// Stops trying after 10 minutes.\r\n\t\tif (initTimer >= 600000) {\r\n\t\t\tlogger.log('Taking too long to load, stopping. Refresh the page to try again. (' + initTimer + 'ms)');\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\t// Give an update every 10 seconds.\r\n\t\tif (initTimer % 10000) {\r\n\t\t\tlogger.debug('Still waiting for objects to load. (' + initTimer + 'ms)');\r\n\t\t}\r\n\r\n\t\t// Bump time up after 1s to reduce possible lag.\r\n\t\ttime = time >= 1000 ? 1000 : time + 25;\r\n\t\tinitTimer += time;\r\n\r\n\t\tsetTimeout(init, time, time);\r\n\t\treturn;\r\n\t}\r\n\t\r\n\t// Expose public api.\r\n\tif (typeof window.emoteMenu === 'undefined') {\r\n\t\twindow.emoteMenu = publicApi;\r\n\t}\r\n\r\n\tember.hook('route:channel', activate, deactivate);\r\n\tember.hook('route:chat', activate, deactivate);\r\n\r\n\tactivate();\r\n})();\r\n\r\nfunction activate() {\r\n\tui.init();\r\n\temotes.init();\r\n}\r\nfunction deactivate() {\r\n\tui.hideMenu();\r\n}\r\n","(function (doc, cssText) {\n    var id = \"emote-menu-for-twitch-styles\";\n    var styleEl = doc.getElementById(id);\n    if (!styleEl) {\n        styleEl = doc.createElement(\"style\");\n        styleEl.id = id;\n        doc.getElementsByTagName(\"head\")[0].appendChild(styleEl);\n    }\n    if (styleEl.styleSheet) {\n        if (!styleEl.styleSheet.disabled) {\n            styleEl.styleSheet.cssText = cssText;\n        }\n    } else {\n        try {\n            styleEl.innerHTML = cssText;\n        } catch (ignore) {\n            styleEl.innerText = cssText;\n        }\n    }\n}(document, \"/**\\n\" +\n\" * Minified style.\\n\" +\n\" * Original filename: \\\\node_modules\\\\jquery.scrollbar\\\\jquery.scrollbar.css\\n\" +\n\" */\\n\" +\n\".scroll-wrapper{overflow:hidden!important;padding:0!important;position:relative}.scroll-wrapper>.scroll-content{border:none!important;-moz-box-sizing:content-box!important;box-sizing:content-box!important;height:auto;left:0;margin:0;max-height:none!important;max-width:none!important;overflow:scroll!important;padding:0;position:relative!important;top:0;width:auto!important}.scroll-wrapper>.scroll-content::-webkit-scrollbar{height:0;width:0}.scroll-element{display:none}.scroll-element,.scroll-element div{-moz-box-sizing:content-box;box-sizing:content-box}.scroll-element.scroll-x.scroll-scrollx_visible,.scroll-element.scroll-y.scroll-scrolly_visible{display:block}.scroll-element .scroll-arrow,.scroll-element .scroll-bar{cursor:default}.scroll-textarea{border:1px solid #ccc;border-top-color:#999}.scroll-textarea>.scroll-content{overflow:hidden!important}.scroll-textarea>.scroll-content>textarea{border:none!important;-moz-box-sizing:border-box;box-sizing:border-box;height:100%!important;margin:0;max-height:none!important;max-width:none!important;overflow:scroll!important;outline:0;padding:2px;position:relative!important;top:0;width:100%!important}.scroll-textarea>.scroll-content>textarea::-webkit-scrollbar{height:0;width:0}.scrollbar-inner>.scroll-element,.scrollbar-inner>.scroll-element div{border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-inner>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-inner>.scroll-element.scroll-x{bottom:2px;height:8px;left:0;width:100%}.scrollbar-inner>.scroll-element.scroll-y{height:100%;right:2px;top:0;width:8px}.scrollbar-inner>.scroll-element .scroll-element_outer{overflow:hidden}.scrollbar-inner>.scroll-element .scroll-bar,.scrollbar-inner>.scroll-element .scroll-element_outer,.scrollbar-inner>.scroll-element .scroll-element_track{border-radius:8px}.scrollbar-inner>.scroll-element .scroll-bar,.scrollbar-inner>.scroll-element .scroll-element_track{-ms-filter:\\\"progid:DXImageTransform.Microsoft.Alpha(Opacity=40)\\\";filter:alpha(opacity=40);opacity:.4}.scrollbar-inner>.scroll-element .scroll-element_track{background-color:#e0e0e0}.scrollbar-inner>.scroll-element .scroll-bar{background-color:#c2c2c2}.scrollbar-inner>.scroll-element.scroll-draggable .scroll-bar,.scrollbar-inner>.scroll-element:hover .scroll-bar{background-color:#919191}.scrollbar-inner>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-12px}.scrollbar-inner>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-12px}.scrollbar-inner>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-12px}.scrollbar-inner>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-12px}.scrollbar-outer>.scroll-element,.scrollbar-outer>.scroll-element div{border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-outer>.scroll-element{background-color:#fff}.scrollbar-outer>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-outer>.scroll-element.scroll-x{bottom:0;height:12px;left:0;width:100%}.scrollbar-outer>.scroll-element.scroll-y{height:100%;right:0;top:0;width:12px}.scrollbar-outer>.scroll-element.scroll-x .scroll-element_outer{height:8px;top:2px}.scrollbar-outer>.scroll-element.scroll-y .scroll-element_outer{left:2px;width:8px}.scrollbar-outer>.scroll-element .scroll-element_outer{overflow:hidden}.scrollbar-outer>.scroll-element .scroll-element_track{background-color:#eee}.scrollbar-outer>.scroll-element .scroll-bar,.scrollbar-outer>.scroll-element .scroll-element_outer,.scrollbar-outer>.scroll-element .scroll-element_track{border-radius:8px}.scrollbar-outer>.scroll-element .scroll-bar{background-color:#d9d9d9}.scrollbar-outer>.scroll-element .scroll-bar:hover{background-color:#c2c2c2}.scrollbar-outer>.scroll-element.scroll-draggable .scroll-bar{background-color:#919191}.scrollbar-outer>.scroll-content.scroll-scrolly_visible{left:-12px;margin-left:12px}.scrollbar-outer>.scroll-content.scroll-scrollx_visible{top:-12px;margin-top:12px}.scrollbar-outer>.scroll-element.scroll-x .scroll-bar{min-width:10px}.scrollbar-outer>.scroll-element.scroll-y .scroll-bar{min-height:10px}.scrollbar-outer>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-14px}.scrollbar-outer>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-14px}.scrollbar-outer>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-14px}.scrollbar-outer>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-14px}.scrollbar-macosx>.scroll-element,.scrollbar-macosx>.scroll-element div{background:0 0;border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-macosx>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-macosx>.scroll-element .scroll-element_track{display:none}.scrollbar-macosx>.scroll-element .scroll-bar{background-color:#6C6E71;display:block;-ms-filter:\\\"progid:DXImageTransform.Microsoft.Alpha(Opacity=0)\\\";filter:alpha(opacity=0);opacity:0;border-radius:7px;transition:opacity .2s linear}.scrollbar-macosx:hover>.scroll-element .scroll-bar,.scrollbar-macosx>.scroll-element.scroll-draggable .scroll-bar{-ms-filter:\\\"progid:DXImageTransform.Microsoft.Alpha(Opacity=70)\\\";filter:alpha(opacity=70);opacity:.7}.scrollbar-macosx>.scroll-element.scroll-x{bottom:0;height:0;left:0;min-width:100%;overflow:visible;width:100%}.scrollbar-macosx>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:0}.scrollbar-macosx>.scroll-element.scroll-x .scroll-bar{height:7px;min-width:10px;top:-9px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-bar{left:-9px;min-height:10px;width:7px}.scrollbar-macosx>.scroll-element.scroll-x .scroll-element_outer{left:2px}.scrollbar-macosx>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-element_outer{top:2px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-macosx>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-11px}.scrollbar-macosx>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-11px}.scrollbar-light>.scroll-element,.scrollbar-light>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-light>.scroll-element{background-color:#fff}.scrollbar-light>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-light>.scroll-element .scroll-element_outer{border-radius:10px}.scrollbar-light>.scroll-element .scroll-element_size{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiA/Pgo8c3ZnIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgd2lkdGg9IjEwMCUiIGhlaWdodD0iMTAwJSIgdmlld0JveD0iMCAwIDEgMSIgcHJlc2VydmVBc3BlY3RSYXRpbz0ibm9uZSI+CiAgPGxpbmVhckdyYWRpZW50IGlkPSJncmFkLXVjZ2ctZ2VuZXJhdGVkIiBncmFkaWVudFVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeDE9IjAlIiB5MT0iMCUiIHgyPSIxMDAlIiB5Mj0iMCUiPgogICAgPHN0b3Agb2Zmc2V0PSIwJSIgc3RvcC1jb2xvcj0iI2RiZGJkYiIgc3RvcC1vcGFjaXR5PSIxIi8+CiAgICA8c3RvcCBvZmZzZXQ9IjEwMCUiIHN0b3AtY29sb3I9IiNlOGU4ZTgiIHN0b3Atb3BhY2l0eT0iMSIvPgogIDwvbGluZWFyR3JhZGllbnQ+CiAgPHJlY3QgeD0iMCIgeT0iMCIgd2lkdGg9IjEiIGhlaWdodD0iMSIgZmlsbD0idXJsKCNncmFkLXVjZ2ctZ2VuZXJhdGVkKSIgLz4KPC9zdmc+);background:linear-gradient(to right,#dbdbdb 0,#e8e8e8 100%);border-radius:10px}.scrollbar-light>.scroll-element.scroll-x{bottom:0;height:17px;left:0;min-width:100%;width:100%}.scrollbar-light>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:17px}.scrollbar-light>.scroll-element .scroll-bar{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiA/Pgo8c3ZnIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgd2lkdGg9IjEwMCUiIGhlaWdodD0iMTAwJSIgdmlld0JveD0iMCAwIDEgMSIgcHJlc2VydmVBc3BlY3RSYXRpbz0ibm9uZSI+CiAgPGxpbmVhckdyYWRpZW50IGlkPSJncmFkLXVjZ2ctZ2VuZXJhdGVkIiBncmFkaWVudFVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeDE9IjAlIiB5MT0iMCUiIHgyPSIxMDAlIiB5Mj0iMCUiPgogICAgPHN0b3Agb2Zmc2V0PSIwJSIgc3RvcC1jb2xvcj0iI2ZlZmVmZSIgc3RvcC1vcGFjaXR5PSIxIi8+CiAgICA8c3RvcCBvZmZzZXQ9IjEwMCUiIHN0b3AtY29sb3I9IiNmNWY1ZjUiIHN0b3Atb3BhY2l0eT0iMSIvPgogIDwvbGluZWFyR3JhZGllbnQ+CiAgPHJlY3QgeD0iMCIgeT0iMCIgd2lkdGg9IjEiIGhlaWdodD0iMSIgZmlsbD0idXJsKCNncmFkLXVjZ2ctZ2VuZXJhdGVkKSIgLz4KPC9zdmc+);background:linear-gradient(to right,#fefefe 0,#f5f5f5 100%);border:1px solid #dbdbdb;border-radius:10px}.scrollbar-light>.scroll-content.scroll-scrolly_visible{left:-17px;margin-left:17px}.scrollbar-light>.scroll-content.scroll-scrollx_visible{top:-17px;margin-top:17px}.scrollbar-light>.scroll-element.scroll-x .scroll-bar{height:10px;min-width:10px;top:0}.scrollbar-light>.scroll-element.scroll-y .scroll-bar{left:0;min-height:10px;width:10px}.scrollbar-light>.scroll-element.scroll-x .scroll-element_outer{height:12px;left:2px;top:2px}.scrollbar-light>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-light>.scroll-element.scroll-y .scroll-element_outer{left:2px;top:2px;width:12px}.scrollbar-light>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-light>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-19px}.scrollbar-light>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-19px}.scrollbar-light>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-19px}.scrollbar-light>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-19px}.scrollbar-rail>.scroll-element,.scrollbar-rail>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-rail>.scroll-element{background-color:#fff}.scrollbar-rail>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-rail>.scroll-element .scroll-element_size{background-color:#999;background-color:rgba(0,0,0,.3)}.scrollbar-rail>.scroll-element .scroll-element_outer:hover .scroll-element_size{background-color:#666;background-color:rgba(0,0,0,.5)}.scrollbar-rail>.scroll-element.scroll-x{bottom:0;height:12px;left:0;min-width:100%;padding:3px 0 2px;width:100%}.scrollbar-rail>.scroll-element.scroll-y{height:100%;min-height:100%;padding:0 2px 0 3px;right:0;top:0;width:12px}.scrollbar-rail>.scroll-element .scroll-bar{background-color:#d0b9a0;border-radius:2px;box-shadow:1px 1px 3px rgba(0,0,0,.5)}.scrollbar-rail>.scroll-element .scroll-element_outer:hover .scroll-bar{box-shadow:1px 1px 3px rgba(0,0,0,.6)}.scrollbar-rail>.scroll-content.scroll-scrolly_visible{left:-17px;margin-left:17px}.scrollbar-rail>.scroll-content.scroll-scrollx_visible{margin-top:17px;top:-17px}.scrollbar-rail>.scroll-element.scroll-x .scroll-bar{height:10px;min-width:10px;top:1px}.scrollbar-rail>.scroll-element.scroll-y .scroll-bar{left:1px;min-height:10px;width:10px}.scrollbar-rail>.scroll-element.scroll-x .scroll-element_outer{height:15px;left:5px}.scrollbar-rail>.scroll-element.scroll-x .scroll-element_size{height:2px;left:-10px;top:5px}.scrollbar-rail>.scroll-element.scroll-y .scroll-element_outer{top:5px;width:15px}.scrollbar-rail>.scroll-element.scroll-y .scroll-element_size{left:5px;top:-10px;width:2px}.scrollbar-rail>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-25px}.scrollbar-rail>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-25px}.scrollbar-rail>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-25px}.scrollbar-rail>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-25px}.scrollbar-dynamic>.scroll-element,.scrollbar-dynamic>.scroll-element div{background:0 0;border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-dynamic>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-dynamic>.scroll-element.scroll-x{bottom:2px;height:7px;left:0;min-width:100%;width:100%}.scrollbar-dynamic>.scroll-element.scroll-y{height:100%;min-height:100%;right:2px;top:0;width:7px}.scrollbar-dynamic>.scroll-element .scroll-element_outer{opacity:.3;border-radius:12px}.scrollbar-dynamic>.scroll-element .scroll-element_size{background-color:#ccc;opacity:0;border-radius:12px;transition:opacity .2s}.scrollbar-dynamic>.scroll-element .scroll-bar{background-color:#6c6e71;border-radius:7px}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-bar{bottom:0;height:7px;min-width:24px;top:auto}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-bar{left:auto;min-height:24px;right:0;width:7px}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-element_outer{bottom:0;top:auto;left:2px;transition:height .2s}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-element_outer{left:auto;right:0;top:2px;transition:width .2s}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-dynamic>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-11px}.scrollbar-dynamic>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-11px}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer{overflow:hidden;-ms-filter:\\\"progid:DXImageTransform.Microsoft.Alpha(Opacity=70)\\\";filter:alpha(opacity=70);opacity:.7}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer .scroll-element_size,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer .scroll-element_size{opacity:1}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer .scroll-bar,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer .scroll-bar{height:100%;width:100%;border-radius:12px}.scrollbar-dynamic>.scroll-element.scroll-x.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element.scroll-x:hover .scroll-element_outer{height:20px;min-height:7px}.scrollbar-dynamic>.scroll-element.scroll-y.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element.scroll-y:hover .scroll-element_outer{min-width:7px;width:20px}.scrollbar-chrome>.scroll-element,.scrollbar-chrome>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-chrome>.scroll-element{background-color:#fff}.scrollbar-chrome>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-chrome>.scroll-element .scroll-element_track{background:#f1f1f1;border:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-x{bottom:0;height:16px;left:0;min-width:100%;width:100%}.scrollbar-chrome>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:16px}.scrollbar-chrome>.scroll-element .scroll-bar{background-color:#d9d9d9;border:1px solid #bdbdbd;cursor:default;border-radius:2px}.scrollbar-chrome>.scroll-element .scroll-bar:hover{background-color:#c2c2c2;border-color:#a9a9a9}.scrollbar-chrome>.scroll-element.scroll-draggable .scroll-bar{background-color:#919191;border-color:#7e7e7e}.scrollbar-chrome>.scroll-content.scroll-scrolly_visible{left:-16px;margin-left:16px}.scrollbar-chrome>.scroll-content.scroll-scrollx_visible{top:-16px;margin-top:16px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-bar{height:8px;min-width:10px;top:3px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-bar{left:3px;min-height:10px;width:8px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_outer{border-left:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_track{height:14px;left:-3px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_size{height:14px;left:-4px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_outer{border-top:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_track{top:-3px;width:14px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_size{top:-4px;width:14px}.scrollbar-chrome>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-19px}.scrollbar-chrome>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-19px}.scrollbar-chrome>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-19px}.scrollbar-chrome>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-19px}\\n\" +\n\"/**\\n\" +\n\" * Minified style.\\n\" +\n\" * Original filename: \\\\src\\\\styles\\\\style.css\\n\" +\n\" */\\n\" +\n\"@-webkit-keyframes spin{100%{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}@keyframes spin{100%{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}#emote-menu-button{background-image:url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABIAAAAQCAYAAAAbBi9cAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAKUSURBVDhPfZTNi1JRGMZvMIsWUZts5SIXFYK0CME/IGghxVC7WUoU1NBixI+mRSD4MQzmxziKO3XUBhRmUGZKdBG40XEGU6d0GFGZcT4qxW1hi7fzvNwZqKwDD5z7vs/vueeee+6VMJxO5wUhhdvtfuHz+T4tLS2NhegfGsMDLxiwHIIhLi57PJ75VCr1Y39/n4bDIY1Go4lCDx54wYCVYzjoVjQa/dxutyfCkwSvYJpgOSQf708tuBa1yWRy/L+V/Cl4wYBFhhTxfLhum/esiiJ1u12KRCJksVhofX2dTk5OzkHMUUMPHnjB2F55VpEhPde/Lbx8FqBEIkHpdJoMBgNptVrS6XRUqVTOg7a3t2lmZob0ej2p1Wr2ggGLDOnJ3QSZH4coHo/TysoKhygUCtJoNFQsFmkwGLAwR7hSqSSVSsVeMGCRIT29F6fXJi8Xy+Uymc1mmp6eJofDQfV6nU5PT1mY2+127uHxSqUSh4FFhhQLvrvtcrm+YpkHBwdUrVZpa2uLarUadTodOjw8ZGGOGnrwwAsGLDLw1i4uLrzRYeOOj49pb2+Pdnd3qdVq8StGAIQ5ao1Ggz3wggGLDD4C4izcEcWfR0dHbMrlcrSxscGbjVAIK8lms7S5ucmB/X6fXz9YDsEQFzdjsVit2Wzyqc1kMrwfVquVjEYjzc3NkclkIpvNRmtra+yBVzAfBXtDjuGgS8FgcFbc8QvuhjNSKBQoFAqR6LFEn/L5PPfggXd5eXkWrBzDQdC1QCBgFoeut7Ozw/tyBp2FQzhPwtOFFwzY34Yo4A9wRXzdD8LhcE48wncE9no9Fuaoid574bkPLxgZ/3uI5pTQVfFlP/L7/Wmhb7JSXq/3IXrwyHZ5SNIvGCnqyh+J7+gAAAAASUVORK5CYII=)!important;background-position:50%;background-repeat:no-repeat;cursor:pointer;height:30px;width:30px}#emote-menu-button:focus{box-shadow:none}#emote-menu-button.active{box-shadow:0 0 6px 0 #7d5bbe,inset 0 0 0 1px rgba(100,65,164,.5)}.emote-menu{padding:5px;z-index:1000;display:none;background-color:#202020;position:absolute}.emote-menu a{color:#fff}.emote-menu a:hover{cursor:pointer;text-decoration:underline;color:#ccc}.emote-menu .emotes-starred{height:38px}.emote-menu .draggable{background-image:repeating-linear-gradient(45deg,transparent,transparent 5px,rgba(255,255,255,.05) 5px,rgba(255,255,255,.05) 10px);cursor:move;height:7px;margin-bottom:3px}.emote-menu .draggable:hover{background-image:repeating-linear-gradient(45deg,transparent,transparent 5px,rgba(255,255,255,.1) 5px,rgba(255,255,255,.1) 10px)}.emote-menu .header-info{border-top:1px solid #000;box-shadow:0 1px 0 rgba(255,255,255,.05) inset;background-image:linear-gradient(to top,transparent,rgba(0,0,0,.5));padding:2px;color:#ddd;text-align:center;position:relative}.emote-menu .header-info img{margin-right:8px}.emote-menu .emote{display:inline-block;padding:2px;margin:1px;cursor:pointer;border-radius:5px;text-align:center;position:relative;width:30px;height:30px;transition:all .25s ease;border:1px solid transparent}.emote-menu.editing .emote{cursor:auto}.emote-menu .emote img{max-width:100%;max-height:100%;margin:auto;position:absolute;top:0;bottom:0;left:0;right:0}.emote-menu .single-row .emote-container{overflow:hidden;height:37px}.emote-menu .single-row .emote{display:inline-block;margin-bottom:100px}.emote-menu .emote:hover{background-color:rgba(255,255,255,.1)}.emote-menu .pull-left{float:left}.emote-menu .pull-right{float:right}.emote-menu .footer{text-align:center;border-top:1px solid #000;box-shadow:0 1px 0 rgba(255,255,255,.05) inset;padding:5px 0 2px;margin-top:5px;height:18px}.emote-menu .footer .pull-left{margin-right:5px}.emote-menu .footer .pull-right{margin-left:5px}.emote-menu .icon{height:16px;width:16px;opacity:.5;background-size:contain!important}.emote-menu .icon:hover{opacity:1}.emote-menu .icon-home{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNjQiDQogICBoZWlnaHQ9IjY0Ig0KICAgdmlld0JveD0iMCAwIDY0IDY0Ig0KICAgaWQ9IkNhcGFfMSINCiAgIHhtbDpzcGFjZT0icHJlc2VydmUiPjxtZXRhZGF0YQ0KICAgaWQ9Im1ldGFkYXRhMzAwMSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczI5OTkiIC8+DQo8cGF0aA0KICAgZD0ibSA1Ny4wNjIsMzEuMzk4IGMgMC45MzIsLTEuMDI1IDAuODQyLC0yLjU5NiAtMC4yMDEsLTMuNTA4IEwgMzMuODg0LDcuNzg1IEMgMzIuODQxLDYuODczIDMxLjE2OSw2Ljg5MiAzMC4xNDgsNy44MjggTCA3LjA5MywyOC45NjIgYyAtMS4wMjEsMC45MzYgLTEuMDcxLDIuNTA1IC0wLjExMSwzLjUwMyBsIDAuNTc4LDAuNjAyIGMgMC45NTksMC45OTggMi41MDksMS4xMTcgMy40NiwwLjI2NSBsIDEuNzIzLC0xLjU0MyB2IDIyLjU5IGMgMCwxLjM4NiAxLjEyMywyLjUwOCAyLjUwOCwyLjUwOCBoIDguOTg3IGMgMS4zODUsMCAyLjUwOCwtMS4xMjIgMi41MDgsLTIuNTA4IFYgMzguNTc1IGggMTEuNDYzIHYgMTUuODA0IGMgLTAuMDIsMS4zODUgMC45NzEsMi41MDcgMi4zNTYsMi41MDcgaCA5LjUyNCBjIDEuMzg1LDAgMi41MDgsLTEuMTIyIDIuNTA4LC0yLjUwOCBWIDMyLjEwNyBjIDAsMCAwLjQ3NiwwLjQxNyAxLjA2MywwLjkzMyAwLjU4NiwwLjUxNSAxLjgxNywwLjEwMiAyLjc0OSwtMC45MjQgbCAwLjY1MywtMC43MTggeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNmZmZmZmY7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4=) 50% no-repeat}.emote-menu .icon-gear{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMjEuNTkiDQogICBoZWlnaHQ9IjIxLjEzNjk5OSINCiAgIHZpZXdCb3g9IjAgMCAyMS41OSAyMS4xMzciDQogICBpZD0iQ2FwYV8xIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGEzOSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczM3IiAvPg0KPHBhdGgNCiAgIGQ9Ik0gMTguNjIyLDguMTQ1IDE4LjA3Nyw2Ljg1IGMgMCwwIDEuMjY4LC0yLjg2MSAxLjE1NiwtMi45NzEgTCAxNy41NTQsMi4yNCBDIDE3LjQzOCwyLjEyNyAxNC41NzYsMy40MzMgMTQuNTc2LDMuNDMzIEwgMTMuMjU2LDIuOSBDIDEzLjI1NiwyLjkgMTIuMDksMCAxMS45MywwIEggOS41NjEgQyA5LjM5NiwwIDguMzE3LDIuOTA2IDguMzE3LDIuOTA2IEwgNi45OTksMy40NDEgYyAwLDAgLTIuOTIyLC0xLjI0MiAtMy4wMzQsLTEuMTMxIEwgMi4yODksMy45NTEgQyAyLjE3Myw0LjA2NCAzLjUwNyw2Ljg2NyAzLjUwNyw2Ljg2NyBMIDIuOTYyLDguMTYgQyAyLjk2Miw4LjE2IDAsOS4zMDEgMCw5LjQ1NSB2IDIuMzIyIGMgMCwwLjE2MiAyLjk2OSwxLjIxOSAyLjk2OSwxLjIxOSBsIDAuNTQ1LDEuMjkxIGMgMCwwIC0xLjI2OCwyLjg1OSAtMS4xNTcsMi45NjkgbCAxLjY3OCwxLjY0MyBjIDAuMTE0LDAuMTExIDIuOTc3LC0xLjE5NSAyLjk3NywtMS4xOTUgbCAxLjMyMSwwLjUzNSBjIDAsMCAxLjE2NiwyLjg5OCAxLjMyNywyLjg5OCBoIDIuMzY5IGMgMC4xNjQsMCAxLjI0NCwtMi45MDYgMS4yNDQsLTIuOTA2IGwgMS4zMjIsLTAuNTM1IGMgMCwwIDIuOTE2LDEuMjQyIDMuMDI5LDEuMTMzIGwgMS42NzgsLTEuNjQxIGMgMC4xMTcsLTAuMTE1IC0xLjIyLC0yLjkxNiAtMS4yMiwtMi45MTYgbCAwLjU0NCwtMS4yOTMgYyAwLDAgMi45NjMsLTEuMTQzIDIuOTYzLC0xLjI5OSBWIDkuMzYgQyAyMS41OSw5LjE5OSAxOC42MjIsOC4xNDUgMTguNjIyLDguMTQ1IHogbSAtNC4zNjYsMi40MjMgYyAwLDEuODY3IC0xLjU1MywzLjM4NyAtMy40NjEsMy4zODcgLTEuOTA2LDAgLTMuNDYxLC0xLjUyIC0zLjQ2MSwtMy4zODcgMCwtMS44NjcgMS41NTUsLTMuMzg1IDMuNDYxLC0zLjM4NSAxLjkwOSwwLjAwMSAzLjQ2MSwxLjUxOCAzLjQ2MSwzLjM4NSB6Ig0KICAgaWQ9InBhdGgzIg0KICAgc3R5bGU9ImZpbGw6I0ZGRkZGRiIgLz4NCjxnDQogICBpZD0iZzUiPg0KPC9nPg0KPGcNCiAgIGlkPSJnNyI+DQo8L2c+DQo8Zw0KICAgaWQ9Imc5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzExIj4NCjwvZz4NCjxnDQogICBpZD0iZzEzIj4NCjwvZz4NCjxnDQogICBpZD0iZzE1Ij4NCjwvZz4NCjxnDQogICBpZD0iZzE3Ij4NCjwvZz4NCjxnDQogICBpZD0iZzE5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzIxIj4NCjwvZz4NCjxnDQogICBpZD0iZzIzIj4NCjwvZz4NCjxnDQogICBpZD0iZzI1Ij4NCjwvZz4NCjxnDQogICBpZD0iZzI3Ij4NCjwvZz4NCjxnDQogICBpZD0iZzI5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzMxIj4NCjwvZz4NCjxnDQogICBpZD0iZzMzIj4NCjwvZz4NCjwvc3ZnPg0K) 50% no-repeat}.emote-menu.editing .icon-gear{-webkit-animation:spin 4s linear infinite;animation:spin 4s linear infinite}.emote-menu .icon-resize-handle{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTYiDQogICBoZWlnaHQ9IjE2Ig0KICAgdmlld0JveD0iMCAwIDE2IDE2Ig0KICAgaWQ9IkNhcGFfMSINCiAgIHhtbDpzcGFjZT0icHJlc2VydmUiPjxtZXRhZGF0YQ0KICAgaWQ9Im1ldGFkYXRhNDM1NyI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczQzNTUiIC8+DQo8cGF0aA0KICAgZD0iTSAxMy41LDggQyAxMy4yMjUsOCAxMyw4LjIyNCAxMyw4LjUgdiAzLjc5MyBMIDMuNzA3LDMgSCA3LjUgQyA3Ljc3NiwzIDgsMi43NzYgOCwyLjUgOCwyLjIyNCA3Ljc3NiwyIDcuNSwyIGggLTUgTCAyLjMwOSwyLjAzOSAyLjE1LDIuMTQ0IDIuMTQ2LDIuMTQ2IDIuMTQzLDIuMTUyIDIuMDM5LDIuMzA5IDIsMi41IHYgNSBDIDIsNy43NzYgMi4yMjQsOCAyLjUsOCAyLjc3Niw4IDMsNy43NzYgMyw3LjUgViAzLjcwNyBMIDEyLjI5MywxMyBIIDguNSBDIDguMjI0LDEzIDgsMTMuMjI1IDgsMTMuNSA4LDEzLjc3NSA4LjIyNCwxNCA4LjUsMTQgaCA1IGwgMC4xOTEsLTAuMDM5IGMgMC4xMjEsLTAuMDUxIDAuMjIsLTAuMTQ4IDAuMjcsLTAuMjcgTCAxNCwxMy41MDIgViA4LjUgQyAxNCw4LjIyNCAxMy43NzUsOCAxMy41LDggeiINCiAgIGlkPSJwYXRoNDM1MSINCiAgIHN0eWxlPSJmaWxsOiNmZmZmZmY7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4=) 50% no-repeat;cursor:nwse-resize!important}.emote-menu .icon-pin{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTYiDQogICBoZWlnaHQ9IjE2Ig0KICAgaWQ9InN2ZzMwMDUiPg0KICA8bWV0YWRhdGENCiAgICAgaWQ9Im1ldGFkYXRhMzAyMyI+DQogICAgPHJkZjpSREY+DQogICAgICA8Y2M6V29yaw0KICAgICAgICAgcmRmOmFib3V0PSIiPg0KICAgICAgICA8ZGM6Zm9ybWF0PmltYWdlL3N2Zyt4bWw8L2RjOmZvcm1hdD4NCiAgICAgICAgPGRjOnR5cGUNCiAgICAgICAgICAgcmRmOnJlc291cmNlPSJodHRwOi8vcHVybC5vcmcvZGMvZGNtaXR5cGUvU3RpbGxJbWFnZSIgLz4NCiAgICAgICAgPGRjOnRpdGxlPjwvZGM6dGl0bGU+DQogICAgICA8L2NjOldvcms+DQogICAgPC9yZGY6UkRGPg0KICA8L21ldGFkYXRhPg0KICA8ZGVmcw0KICAgICBpZD0iZGVmczMwMjEiIC8+DQogIDxnDQogICAgIHRyYW5zZm9ybT0ibWF0cml4KDAuNzkzMDc4MiwwLDAsMC43OTMwNzgyLC0yLjE3MDk4NSwtODE0LjY5Mjk5KSINCiAgICAgaWQ9ImczMDA3Ij4NCiAgICA8Zw0KICAgICAgIHRyYW5zZm9ybT0ibWF0cml4KDAuNzA3MTEsMC43MDcxMSwtMC43MDcxMSwwLjcwNzExLDczNy43MDc1NSwyOTUuNDg4MDgpIg0KICAgICAgIGlkPSJnMzAwOSI+DQogICAgICA8Zw0KICAgICAgICAgaWQ9ImczNzU1Ij4NCiAgICAgICAgPHBhdGgNCiAgICAgICAgICAgZD0iTSA5Ljc4MTI1LDAgQyA5LjQ3NDA1NjIsMC42ODkxMTIgOS41MjA2OCwxLjUyMzA4NTMgOS4zMTI1LDIuMTg3NSBMIDQuOTM3NSw2LjU5Mzc1IEMgMy45NTg5NjA4LDYuNDI5NDgzIDIuOTQ3NzU0OCw2LjUzMjc4OTkgMiw2LjgxMjUgTCA1LjAzMTI1LDkuODQzNzUgMC41NjI1LDE0LjMxMjUgMCwxNiBDIDAuNTY5Mjk2MjgsMTUuNzk1NjI2IDEuMTY3NzM3OCwxNS42NDAyMzcgMS43MTg3NSwxNS40MDYyNSBMIDYuMTU2MjUsMTAuOTY4NzUgOS4xODc1LDE0IGMgMC4yNzk2ODIzLC0wLjk0Nzc4MyAwLjM4MzE1MjgsLTEuOTU4OTM3IDAuMjE4NzUsLTIuOTM3NSAxLjUwMDAxMSwtMS40ODk1Nzk4IDMuMDAwMDAxLC0yLjk3OTE1OSA0LjUsLTQuNDY4NzUgMC42MDExMDIsLTAuMDMxMzYxIDEuODIyMTM4LC0wLjA5NjEzNyAyLC0wLjQ2ODc1IEMgMTMuODc5ODkyLDQuMDY5NDgwMyAxMS44NDI4NjUsMi4wMjAyMjgyIDkuNzgxMjUsMCB6Ig0KICAgICAgICAgICB0cmFuc2Zvcm09Im1hdHJpeCgwLjg5MTU5Mzc0LC0wLjg5MTU5Mzc0LDAuODkxNTkzNzQsMC44OTE1OTM3NCwtMi4yNjU1LDEwMzcuMTM0NSkiDQogICAgICAgICAgIGlkPSJwYXRoMzAxMSINCiAgICAgICAgICAgc3R5bGU9ImZpbGw6I2ZmZmZmZjtmaWxsLW9wYWNpdHk6MSIgLz4NCiAgICAgIDwvZz4NCiAgICA8L2c+DQogIDwvZz4NCjwvc3ZnPg0K) 50% no-repeat;transition:all .25s ease}.emote-menu .icon-pin:hover,.emote-menu.pinned .icon-pin{-webkit-transform:rotate(-45deg);transform:rotate(-45deg);opacity:1}.emote-menu .edit-tool{background-position:50%;background-repeat:no-repeat;background-size:14px;border-radius:4px;border:1px solid #000;cursor:pointer;display:none;height:14px;opacity:.25;position:absolute;transition:all .25s ease;width:14px;z-index:1}.emote-menu .edit-tool:hover,.emote-menu .emote:hover .edit-tool{opacity:1}.emote-menu .edit-visibility{background-color:#00c800;background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTAwIg0KICAgaGVpZ2h0PSIxMDAiDQogICB2aWV3Qm94PSIwIDAgMTAwIDEwMCINCiAgIGlkPSJMYXllcl8xIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGE5Ij48cmRmOlJERj48Y2M6V29yaw0KICAgICAgIHJkZjphYm91dD0iIj48ZGM6Zm9ybWF0PmltYWdlL3N2Zyt4bWw8L2RjOmZvcm1hdD48ZGM6dHlwZQ0KICAgICAgICAgcmRmOnJlc291cmNlPSJodHRwOi8vcHVybC5vcmcvZGMvZGNtaXR5cGUvU3RpbGxJbWFnZSIgLz48ZGM6dGl0bGU+PC9kYzp0aXRsZT48L2NjOldvcms+PC9yZGY6UkRGPjwvbWV0YWRhdGE+PGRlZnMNCiAgIGlkPSJkZWZzNyIgLz4NCjxwYXRoDQogICBkPSJNIDk3Ljk2NCw0Ni41NDggQyA5Ny4wOTgsNDUuNTI4IDc2LjQyNywyMS42MDMgNTAsMjEuNjAzIGMgLTI2LjQyNywwIC00Ny4wOTgsMjMuOTI1IC00Ny45NjUsMjQuOTQ2IC0xLjcwMSwyIC0xLjcwMSw0LjkwMiAxMGUtNCw2LjkwMyAwLjg2NiwxLjAyIDIxLjUzNywyNC45NDUgNDcuOTY0LDI0Ljk0NSAyNi40MjcsMCA0Ny4wOTgsLTIzLjkyNiA0Ny45NjUsLTI0Ljk0NiAxLjcwMSwtMiAxLjcwMSwtNC45MDIgLTAuMDAxLC02LjkwMyB6IE0gNTguMDczLDM1Ljk3NSBjIDEuNzc3LC0wLjk3IDQuMjU1LDAuMTQzIDUuNTM0LDIuNDg1IDEuMjc5LDIuMzQzIDAuODc1LDUuMDI5IC0wLjkwMiw1Ljk5OSAtMS43NzcsMC45NzEgLTQuMjU1LC0wLjE0MyAtNS41MzUsLTIuNDg1IC0xLjI3OSwtMi4zNDMgLTAuODc1LC01LjAyOSAwLjkwMywtNS45OTkgeiBNIDUwLDY5LjcyOSBDIDMxLjU0LDY5LjcyOSAxNi4wMDUsNTUuNTUzIDEwLjYyOCw1MCAxNC4yNTksNDYuMjQ5IDIyLjUyNiwzOC41NzEgMzMuMTk1LDMzLjk3OSAzMS4xMTQsMzcuMTQ1IDI5Ljg5NCw0MC45MjggMjkuODk0LDQ1IGMgMCwxMS4xMDQgOS4wMDEsMjAuMTA1IDIwLjEwNSwyMC4xMDUgMTEuMTA0LDAgMjAuMTA2LC05LjAwMSAyMC4xMDYsLTIwLjEwNSAwLC00LjA3MiAtMS4yMTksLTcuODU1IC0zLjMsLTExLjAyMSBDIDc3LjQ3NCwzOC41NzIgODUuNzQxLDQ2LjI1IDg5LjM3Miw1MCA4My45OTUsNTUuNTU1IDY4LjQ2LDY5LjcyOSA1MCw2OS43MjkgeiINCiAgIGlkPSJwYXRoMyIgLz4NCjwvc3ZnPg==)}.emote-menu .edit-starred{background-color:#323232;background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNTAiDQogICBoZWlnaHQ9IjUwIg0KICAgdmlld0JveD0iMCAwIDUwIDUwIg0KICAgaWQ9IkxheWVyXzEiDQogICB4bWw6c3BhY2U9InByZXNlcnZlIj48bWV0YWRhdGENCiAgIGlkPSJtZXRhZGF0YTMwMDEiPjxyZGY6UkRGPjxjYzpXb3JrDQogICAgICAgcmRmOmFib3V0PSIiPjxkYzpmb3JtYXQ+aW1hZ2Uvc3ZnK3htbDwvZGM6Zm9ybWF0PjxkYzp0eXBlDQogICAgICAgICByZGY6cmVzb3VyY2U9Imh0dHA6Ly9wdXJsLm9yZy9kYy9kY21pdHlwZS9TdGlsbEltYWdlIiAvPjxkYzp0aXRsZT48L2RjOnRpdGxlPjwvY2M6V29yaz48L3JkZjpSREY+PC9tZXRhZGF0YT48ZGVmcw0KICAgaWQ9ImRlZnMyOTk5IiAvPg0KPHBhdGgNCiAgIGQ9Im0gNDMuMDQsMjIuNjk2IC03LjU2OCw3LjM3NyAxLjc4NywxMC40MTcgYyAwLjEyNywwLjc1IC0wLjE4MiwxLjUwOSAtMC43OTcsMS45NTcgLTAuMzQ4LDAuMjUzIC0wLjc2MiwwLjM4MiAtMS4xNzYsMC4zODIgLTAuMzE4LDAgLTAuNjM4LC0wLjA3NiAtMC45MzEsLTAuMjMgTCAyNSwzNy42ODEgMTUuNjQ1LDQyLjU5OSBjIC0wLjY3NCwwLjM1NSAtMS40OSwwLjI5NSAtMi4xMDcsLTAuMTUxIEMgMTIuOTIzLDQyIDEyLjYxNCw0MS4yNDIgMTIuNzQzLDQwLjQ5MSBMIDE0LjUzLDMwLjA3NCA2Ljk2MiwyMi42OTcgQyA2LjQxNSwyMi4xNjYgNi4yMjEsMjEuMzcxIDYuNDU0LDIwLjY0NyA2LjY5LDE5LjkyMyA3LjMxNSwxOS4zOTYgOC4wNjksMTkuMjg2IGwgMTAuNDU5LC0xLjUyMSA0LjY4LC05LjQ3OCBDIDIzLjU0Myw3LjYwMyAyNC4yMzksNy4xNzEgMjUsNy4xNzEgYyAwLjc2MywwIDEuNDU2LDAuNDMyIDEuNzkzLDEuMTE1IGwgNC42NzksOS40NzggMTAuNDYxLDEuNTIxIGMgMC43NTIsMC4xMDkgMS4zNzksMC42MzcgMS42MTIsMS4zNjEgMC4yMzcsMC43MjQgMC4wMzgsMS41MTkgLTAuNTA1LDIuMDUgeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNjY2NjY2M7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4NCg==)}.emote-menu .emote>.edit-visibility{bottom:auto;left:auto;right:0;top:0}.emote-menu .emote>.edit-starred{bottom:auto;left:0;right:auto;top:0}.emote-menu .header-info>.edit-tool{margin-left:5px}.emote-menu.editing .edit-tool{display:inline-block}.emote-menu .emote-menu-hidden .edit-visibility{background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTAwIg0KICAgaGVpZ2h0PSIxMDAiDQogICB2aWV3Qm94PSIwIDAgMTAwIDEwMCINCiAgIGlkPSJMYXllcl8zIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGExNSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczEzIiAvPg0KPGcNCiAgIGlkPSJnMyI+DQoJPHBhdGgNCiAgIGQ9Ik0gNzAuMDgyLDQ1LjQ3NSA1MC40NzQsNjUuMDgyIEMgNjEuMTk4LDY0LjgzMSA2OS44MzEsNTYuMTk3IDcwLjA4Miw0NS40NzUgeiINCiAgIGlkPSJwYXRoNSINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQoJPHBhdGgNCiAgIGQ9Im0gOTcuOTY0LDQ2LjU0OCBjIC0wLjQ1LC0wLjUyOSAtNi4yNDUsLTcuMjMgLTE1LjQwMywtMTMuNTU0IGwgLTYuMiw2LjIgQyA4Mi4zNTEsNDMuMTQ4IDg2LjkyLDQ3LjQ2OSA4OS4zNzIsNTAgODMuOTk1LDU1LjU1NSA2OC40Niw2OS43MjkgNTAsNjkuNzI5IGMgLTEuMzM0LDAgLTIuNjUxLC0wLjA4MiAtMy45NTIsLTAuMjIyIGwgLTcuNDM5LDcuNDM5IGMgMy42MzksMC45MDkgNy40NDksMS40NSAxMS4zOTEsMS40NSAyNi40MjcsMCA0Ny4wOTgsLTIzLjkyNiA0Ny45NjUsLTI0Ljk0NiAxLjcwMSwtMS45OTkgMS43MDEsLTQuOTAxIC0wLjAwMSwtNi45MDIgeiINCiAgIGlkPSJwYXRoNyINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQoJPHBhdGgNCiAgIGQ9Im0gOTEuNDExLDE2LjY2IGMgMCwtMC4yNjYgLTAuMTA1LC0wLjUyIC0wLjI5MywtMC43MDcgbCAtNy4wNzEsLTcuMDcgYyAtMC4zOTEsLTAuMzkxIC0xLjAyMywtMC4zOTEgLTEuNDE0LDAgTCA2Ni44MDQsMjQuNzExIEMgNjEuNjAyLDIyLjgxOCA1NS45NDksMjEuNjAzIDUwLDIxLjYwMyBjIC0yNi40MjcsMCAtNDcuMDk4LDIzLjkyNiAtNDcuOTY1LDI0Ljk0NiAtMS43MDEsMiAtMS43MDEsNC45MDIgMTBlLTQsNi45MDMgMC41MTcsMC42MDcgOC4wODMsOS4zNTQgMTkuNzA3LDE2LjMyIEwgOC44ODMsODIuNjMyIEMgOC42OTUsODIuODIgOC41OSw4My4wNzMgOC41OSw4My4zMzkgYyAwLDAuMjY2IDAuMTA1LDAuNTIgMC4yOTMsMC43MDcgbCA3LjA3MSw3LjA3IGMgMC4xOTUsMC4xOTUgMC40NTEsMC4yOTMgMC43MDcsMC4yOTMgMC4yNTYsMCAwLjUxMiwtMC4wOTggMC43MDcsLTAuMjkzIGwgNzMuNzUsLTczLjc1IGMgMC4xODcsLTAuMTg2IDAuMjkzLC0wLjQ0IDAuMjkzLC0wLjcwNiB6IE0gMTAuNjI4LDUwIEMgMTQuMjU5LDQ2LjI0OSAyMi41MjYsMzguNTcxIDMzLjE5NSwzMy45NzkgMzEuMTE0LDM3LjE0NSAyOS44OTQsNDAuOTI4IDI5Ljg5NCw0NSBjIDAsNC42NjUgMS42MDEsOC45NDUgNC4yNywxMi4zNTEgTCAyOC4wNCw2My40NzUgQyAxOS44ODgsNTguOTU1IDEzLjY0OSw1My4xMiAxMC42MjgsNTAgeiINCiAgIGlkPSJwYXRoOSINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQo8L2c+DQo8L3N2Zz4NCg==);background-color:red}.emote-menu .emote-menu-starred .edit-starred{background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNTAiDQogICBoZWlnaHQ9IjUwIg0KICAgdmlld0JveD0iMCAwIDUwIDUwIg0KICAgaWQ9IkxheWVyXzEiDQogICB4bWw6c3BhY2U9InByZXNlcnZlIj48bWV0YWRhdGENCiAgIGlkPSJtZXRhZGF0YTMwMDEiPjxyZGY6UkRGPjxjYzpXb3JrDQogICAgICAgcmRmOmFib3V0PSIiPjxkYzpmb3JtYXQ+aW1hZ2Uvc3ZnK3htbDwvZGM6Zm9ybWF0PjxkYzp0eXBlDQogICAgICAgICByZGY6cmVzb3VyY2U9Imh0dHA6Ly9wdXJsLm9yZy9kYy9kY21pdHlwZS9TdGlsbEltYWdlIiAvPjxkYzp0aXRsZT48L2RjOnRpdGxlPjwvY2M6V29yaz48L3JkZjpSREY+PC9tZXRhZGF0YT48ZGVmcw0KICAgaWQ9ImRlZnMyOTk5IiAvPg0KPHBhdGgNCiAgIGQ9Im0gNDMuMDQsMjIuNjk2IC03LjU2OCw3LjM3NyAxLjc4NywxMC40MTcgYyAwLjEyNywwLjc1IC0wLjE4MiwxLjUwOSAtMC43OTcsMS45NTcgLTAuMzQ4LDAuMjUzIC0wLjc2MiwwLjM4MiAtMS4xNzYsMC4zODIgLTAuMzE4LDAgLTAuNjM4LC0wLjA3NiAtMC45MzEsLTAuMjMgTCAyNSwzNy42ODEgMTUuNjQ1LDQyLjU5OSBjIC0wLjY3NCwwLjM1NSAtMS40OSwwLjI5NSAtMi4xMDcsLTAuMTUxIEMgMTIuOTIzLDQyIDEyLjYxNCw0MS4yNDIgMTIuNzQzLDQwLjQ5MSBMIDE0LjUzLDMwLjA3NCA2Ljk2MiwyMi42OTcgQyA2LjQxNSwyMi4xNjYgNi4yMjEsMjEuMzcxIDYuNDU0LDIwLjY0NyA2LjY5LDE5LjkyMyA3LjMxNSwxOS4zOTYgOC4wNjksMTkuMjg2IGwgMTAuNDU5LC0xLjUyMSA0LjY4LC05LjQ3OCBDIDIzLjU0Myw3LjYwMyAyNC4yMzksNy4xNzEgMjUsNy4xNzEgYyAwLjc2MywwIDEuNDU2LDAuNDMyIDEuNzkzLDEuMTE1IGwgNC42NzksOS40NzggMTAuNDYxLDEuNTIxIGMgMC43NTIsMC4xMDkgMS4zNzksMC42MzcgMS42MTIsMS4zNjEgMC4yMzcsMC43MjQgMC4wMzgsMS41MTkgLTAuNTA1LDIuMDUgeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNmZmNjMDA7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4NCg==)}.emote-menu .emote.emote-menu-starred{border-color:rgba(200,200,0,.5)}.emote-menu .emote.emote-menu-hidden{border-color:rgba(255,0,0,.5)}.emote-menu #starred-emotes-group .emote:not(.emote-menu-starred),.emote-menu:not(.editing) .emote-menu-hidden{display:none}.emote-menu:not(.editing) #starred-emotes-group .emote-menu-starred{border-color:transparent}.emote-menu #starred-emotes-group{text-align:center;color:#646464}.emote-menu #starred-emotes-group:empty:before{content:\\\"Use the edit mode to star an emote!\\\";position:relative;top:8px}.emote-menu .scrollable{height:calc(100% - 101px);overflow-y:auto}.emote-menu .sticky{position:absolute;bottom:0;width:100%}.emote-menu .emote-menu-inner{position:relative;max-height:100%;height:100%}\"));\n","module.exports = (function() {\n    var Hogan = require('hogan.js/lib/template.js');\n    var templates = {};\n    templates['emote'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||\"\");t.b(\"<div class=\\\"emote\");if(t.s(t.f(\"thirdParty\",c,p,1),c,p,0,32,44,\"{{ }}\")){t.rs(c,p,function(c,p,t){t.b(\" third-party\");});c.pop();}if(!t.s(t.f(\"isVisible\",c,p,1),c,p,1,0,0,\"\")){t.b(\" emote-menu-hidden\");};if(t.s(t.f(\"isStarred\",c,p,1),c,p,0,119,138,\"{{ }}\")){t.rs(c,p,function(c,p,t){t.b(\" emote-menu-starred\");});c.pop();}t.b(\"\\\" data-emote=\\\"\");t.b(t.v(t.f(\"text\",c,p,0)));t.b(\"\\\" title=\\\"\");t.b(t.v(t.f(\"text\",c,p,0)));if(t.s(t.f(\"thirdParty\",c,p,1),c,p,0,206,229,\"{{ }}\")){t.rs(c,p,function(c,p,t){t.b(\" (from 3rd party addon)\");});c.pop();}t.b(\"\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t<img src=\\\"\");t.b(t.t(t.f(\"url\",c,p,0)));t.b(\"\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t<div class=\\\"edit-tool edit-starred\\\" data-which=\\\"\");t.b(t.v(t.f(\"text\",c,p,0)));t.b(\"\\\" data-command=\\\"toggle-starred\\\" title=\\\"Star/unstar emote: \");t.b(t.v(t.f(\"text\",c,p,0)));t.b(\"\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"\t<div class=\\\"edit-tool edit-visibility\\\" data-which=\\\"\");t.b(t.v(t.f(\"text\",c,p,0)));t.b(\"\\\" data-command=\\\"toggle-visibility\\\" title=\\\"Hide/show emote: \");t.b(t.v(t.f(\"text\",c,p,0)));t.b(\"\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"</div>\\r\");t.b(\"\\n\");return t.fl(); },partials: {}, subs: {  }});\n    templates['emoteButton'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||\"\");t.b(\"<button class=\\\"button button--icon-only float-left\\\" title=\\\"Emote Menu\\\" id=\\\"emote-menu-button\\\"></button>\\r\");t.b(\"\\n\");return t.fl(); },partials: {}, subs: {  }});\n    templates['emoteGroupHeader'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||\"\");t.b(\"<div class=\\\"group-header\\\" data-emote-channel=\\\"\");t.b(t.v(t.f(\"channel\",c,p,0)));t.b(\"\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t<div class=\\\"header-info\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t\t<img src=\\\"\");t.b(t.v(t.f(\"badge\",c,p,0)));t.b(\"\\\" />\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\");t.b(t.v(t.f(\"channelDisplayName\",c,p,0)));t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t\t<div class=\\\"edit-tool edit-visibility\\\" data-which=\\\"channel-\");t.b(t.v(t.f(\"channel\",c,p,0)));t.b(\"\\\" data-command=\\\"toggle-visibility\\\" title=\\\"Hide/show current emotes for \");t.b(t.v(t.f(\"channelDisplayName\",c,p,0)));t.b(\" (note: new emotes will still show up if they are added)\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"\t<div class=\\\"emote-container\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"</div>\\r\");t.b(\"\\n\");return t.fl(); },partials: {}, subs: {  }});\n    templates['menu'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||\"\");t.b(\"<div class=\\\"emote-menu\\\" id=\\\"emote-menu-for-twitch\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t<div class=\\\"emote-menu-inner\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t\t<div class=\\\"draggable\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t\t<div class=\\\"scrollable scrollbar-macosx\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t<div class=\\\"group-container\\\" id=\\\"all-emotes-group\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t\t<div class=\\\"sticky\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t<div class=\\\"group-header single-row\\\" id=\\\"starred-emotes-group\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<div class=\\\"header-info\\\">Favorite Emotes</div>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<div class=\\\"emote-container\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t<div class=\\\"footer\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<a class=\\\"pull-left icon icon-home\\\" href=\\\"http://cletusc.github.io/Userscript--Twitch-Chat-Emotes\\\" target=\\\"_blank\\\" title=\\\"Visit the homepage where you can donate, post a review, or contact the developer\\\"></a>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<a class=\\\"pull-left icon icon-gear\\\" data-command=\\\"toggle-editing\\\" title=\\\"Toggle edit mode\\\"></a>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<a class=\\\"pull-right icon icon-resize-handle\\\" data-command=\\\"resize-handle\\\"></a>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<a class=\\\"pull-right icon icon-pin\\\" data-command=\\\"toggle-pinned\\\" title=\\\"Pin/unpin the emote menu to the screen\\\"></a>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"</div>\\r\");t.b(\"\\n\");return t.fl(); },partials: {}, subs: {  }});\n    templates['newsMessage'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||\"\");t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"<div class=\\\"twitch-chat-emotes-news\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t[\");t.b(t.v(t.f(\"scriptName\",c,p,0)));t.b(\"] News: \");t.b(t.t(t.f(\"message\",c,p,0)));t.b(\" (<a href=\\\"#\\\" data-command=\\\"twitch-chat-emotes:dismiss-news\\\" data-news-id=\\\"\");t.b(t.v(t.f(\"id\",c,p,0)));t.b(\"\\\">Dismiss</a>)\\r\");t.b(\"\\n\" + i);t.b(\"</div>\\r\");t.b(\"\\n\");return t.fl(); },partials: {}, subs: {  }});\n    return templates;\n})();","/*\n *  Copyright 2011 Twitter, Inc.\n *  Licensed under the Apache License, Version 2.0 (the \"License\");\n *  you may not use this file except in compliance with the License.\n *  You may obtain a copy of the License at\n *\n *  http://www.apache.org/licenses/LICENSE-2.0\n *\n *  Unless required by applicable law or agreed to in writing, software\n *  distributed under the License is distributed on an \"AS IS\" BASIS,\n *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n *  See the License for the specific language governing permissions and\n *  limitations under the License.\n */\n\nvar Hogan = {};\n\n(function (Hogan) {\n  Hogan.Template = function (codeObj, text, compiler, options) {\n    codeObj = codeObj || {};\n    this.r = codeObj.code || this.r;\n    this.c = compiler;\n    this.options = options || {};\n    this.text = text || '';\n    this.partials = codeObj.partials || {};\n    this.subs = codeObj.subs || {};\n    this.buf = '';\n  }\n\n  Hogan.Template.prototype = {\n    // render: replaced by generated code.\n    r: function (context, partials, indent) { return ''; },\n\n    // variable escaping\n    v: hoganEscape,\n\n    // triple stache\n    t: coerceToString,\n\n    render: function render(context, partials, indent) {\n      return this.ri([context], partials || {}, indent);\n    },\n\n    // render internal -- a hook for overrides that catches partials too\n    ri: function (context, partials, indent) {\n      return this.r(context, partials, indent);\n    },\n\n    // ensurePartial\n    ep: function(symbol, partials) {\n      var partial = this.partials[symbol];\n\n      // check to see that if we've instantiated this partial before\n      var template = partials[partial.name];\n      if (partial.instance && partial.base == template) {\n        return partial.instance;\n      }\n\n      if (typeof template == 'string') {\n        if (!this.c) {\n          throw new Error(\"No compiler available.\");\n        }\n        template = this.c.compile(template, this.options);\n      }\n\n      if (!template) {\n        return null;\n      }\n\n      // We use this to check whether the partials dictionary has changed\n      this.partials[symbol].base = template;\n\n      if (partial.subs) {\n        // Make sure we consider parent template now\n        if (!partials.stackText) partials.stackText = {};\n        for (key in partial.subs) {\n          if (!partials.stackText[key]) {\n            partials.stackText[key] = (this.activeSub !== undefined && partials.stackText[this.activeSub]) ? partials.stackText[this.activeSub] : this.text;\n          }\n        }\n        template = createSpecializedPartial(template, partial.subs, partial.partials,\n          this.stackSubs, this.stackPartials, partials.stackText);\n      }\n      this.partials[symbol].instance = template;\n\n      return template;\n    },\n\n    // tries to find a partial in the current scope and render it\n    rp: function(symbol, context, partials, indent) {\n      var partial = this.ep(symbol, partials);\n      if (!partial) {\n        return '';\n      }\n\n      return partial.ri(context, partials, indent);\n    },\n\n    // render a section\n    rs: function(context, partials, section) {\n      var tail = context[context.length - 1];\n\n      if (!isArray(tail)) {\n        section(context, partials, this);\n        return;\n      }\n\n      for (var i = 0; i < tail.length; i++) {\n        context.push(tail[i]);\n        section(context, partials, this);\n        context.pop();\n      }\n    },\n\n    // maybe start a section\n    s: function(val, ctx, partials, inverted, start, end, tags) {\n      var pass;\n\n      if (isArray(val) && val.length === 0) {\n        return false;\n      }\n\n      if (typeof val == 'function') {\n        val = this.ms(val, ctx, partials, inverted, start, end, tags);\n      }\n\n      pass = !!val;\n\n      if (!inverted && pass && ctx) {\n        ctx.push((typeof val == 'object') ? val : ctx[ctx.length - 1]);\n      }\n\n      return pass;\n    },\n\n    // find values with dotted names\n    d: function(key, ctx, partials, returnFound) {\n      var found,\n          names = key.split('.'),\n          val = this.f(names[0], ctx, partials, returnFound),\n          doModelGet = this.options.modelGet,\n          cx = null;\n\n      if (key === '.' && isArray(ctx[ctx.length - 2])) {\n        val = ctx[ctx.length - 1];\n      } else {\n        for (var i = 1; i < names.length; i++) {\n          found = findInScope(names[i], val, doModelGet);\n          if (found !== undefined) {\n            cx = val;\n            val = found;\n          } else {\n            val = '';\n          }\n        }\n      }\n\n      if (returnFound && !val) {\n        return false;\n      }\n\n      if (!returnFound && typeof val == 'function') {\n        ctx.push(cx);\n        val = this.mv(val, ctx, partials);\n        ctx.pop();\n      }\n\n      return val;\n    },\n\n    // find values with normal names\n    f: function(key, ctx, partials, returnFound) {\n      var val = false,\n          v = null,\n          found = false,\n          doModelGet = this.options.modelGet;\n\n      for (var i = ctx.length - 1; i >= 0; i--) {\n        v = ctx[i];\n        val = findInScope(key, v, doModelGet);\n        if (val !== undefined) {\n          found = true;\n          break;\n        }\n      }\n\n      if (!found) {\n        return (returnFound) ? false : \"\";\n      }\n\n      if (!returnFound && typeof val == 'function') {\n        val = this.mv(val, ctx, partials);\n      }\n\n      return val;\n    },\n\n    // higher order templates\n    ls: function(func, cx, partials, text, tags) {\n      var oldTags = this.options.delimiters;\n\n      this.options.delimiters = tags;\n      this.b(this.ct(coerceToString(func.call(cx, text)), cx, partials));\n      this.options.delimiters = oldTags;\n\n      return false;\n    },\n\n    // compile text\n    ct: function(text, cx, partials) {\n      if (this.options.disableLambda) {\n        throw new Error('Lambda features disabled.');\n      }\n      return this.c.compile(text, this.options).render(cx, partials);\n    },\n\n    // template result buffering\n    b: function(s) { this.buf += s; },\n\n    fl: function() { var r = this.buf; this.buf = ''; return r; },\n\n    // method replace section\n    ms: function(func, ctx, partials, inverted, start, end, tags) {\n      var textSource,\n          cx = ctx[ctx.length - 1],\n          result = func.call(cx);\n\n      if (typeof result == 'function') {\n        if (inverted) {\n          return true;\n        } else {\n          textSource = (this.activeSub && this.subsText && this.subsText[this.activeSub]) ? this.subsText[this.activeSub] : this.text;\n          return this.ls(result, cx, partials, textSource.substring(start, end), tags);\n        }\n      }\n\n      return result;\n    },\n\n    // method replace variable\n    mv: function(func, ctx, partials) {\n      var cx = ctx[ctx.length - 1];\n      var result = func.call(cx);\n\n      if (typeof result == 'function') {\n        return this.ct(coerceToString(result.call(cx)), cx, partials);\n      }\n\n      return result;\n    },\n\n    sub: function(name, context, partials, indent) {\n      var f = this.subs[name];\n      if (f) {\n        this.activeSub = name;\n        f(context, partials, this, indent);\n        this.activeSub = false;\n      }\n    }\n\n  };\n\n  //Find a key in an object\n  function findInScope(key, scope, doModelGet) {\n    var val;\n\n    if (scope && typeof scope == 'object') {\n\n      if (scope[key] !== undefined) {\n        val = scope[key];\n\n      // try lookup with get for backbone or similar model data\n      } else if (doModelGet && scope.get && typeof scope.get == 'function') {\n        val = scope.get(key);\n      }\n    }\n\n    return val;\n  }\n\n  function createSpecializedPartial(instance, subs, partials, stackSubs, stackPartials, stackText) {\n    function PartialTemplate() {};\n    PartialTemplate.prototype = instance;\n    function Substitutions() {};\n    Substitutions.prototype = instance.subs;\n    var key;\n    var partial = new PartialTemplate();\n    partial.subs = new Substitutions();\n    partial.subsText = {};  //hehe. substext.\n    partial.buf = '';\n\n    stackSubs = stackSubs || {};\n    partial.stackSubs = stackSubs;\n    partial.subsText = stackText;\n    for (key in subs) {\n      if (!stackSubs[key]) stackSubs[key] = subs[key];\n    }\n    for (key in stackSubs) {\n      partial.subs[key] = stackSubs[key];\n    }\n\n    stackPartials = stackPartials || {};\n    partial.stackPartials = stackPartials;\n    for (key in partials) {\n      if (!stackPartials[key]) stackPartials[key] = partials[key];\n    }\n    for (key in stackPartials) {\n      partial.partials[key] = stackPartials[key];\n    }\n\n    return partial;\n  }\n\n  var rAmp = /&/g,\n      rLt = /</g,\n      rGt = />/g,\n      rApos = /\\'/g,\n      rQuot = /\\\"/g,\n      hChars = /[&<>\\\"\\']/;\n\n  function coerceToString(val) {\n    return String((val === null || val === undefined) ? '' : val);\n  }\n\n  function hoganEscape(str) {\n    str = coerceToString(str);\n    return hChars.test(str) ?\n      str\n        .replace(rAmp, '&amp;')\n        .replace(rLt, '&lt;')\n        .replace(rGt, '&gt;')\n        .replace(rApos, '&#39;')\n        .replace(rQuot, '&quot;') :\n      str;\n  }\n\n  var isArray = Array.isArray || function(a) {\n    return Object.prototype.toString.call(a) === '[object Array]';\n  };\n\n})(typeof exports !== 'undefined' ? exports : Hogan);\n","/**\n * jQuery CSS Customizable Scrollbar\n *\n * Copyright 2014, Yuriy Khabarov\n * Dual licensed under the MIT or GPL Version 2 licenses.\n *\n * If you found bug, please contact me via email <13real008@gmail.com>\n *\n * @author Yuriy Khabarov aka Gromo\n * @version 0.2.6\n * @url https://github.com/gromo/jquery.scrollbar/\n *\n */\n(function(e,t,n){\"use strict\";function h(t){if(o.webkit&&!t){return{height:0,width:0}}if(!o.data.outer){var n={border:\"none\",\"box-sizing\":\"content-box\",height:\"200px\",margin:\"0\",padding:\"0\",width:\"200px\"};o.data.inner=e(\"<div>\").css(e.extend({},n));o.data.outer=e(\"<div>\").css(e.extend({left:\"-1000px\",overflow:\"scroll\",position:\"absolute\",top:\"-1000px\"},n)).append(o.data.inner).appendTo(\"body\")}o.data.outer.scrollLeft(1e3).scrollTop(1e3);return{height:Math.ceil(o.data.outer.offset().top-o.data.inner.offset().top||0),width:Math.ceil(o.data.outer.offset().left-o.data.inner.offset().left||0)}}function p(n,r){e(t).on({\"blur.scrollbar\":function(){e(t).add(\"body\").off(\".scrollbar\");n&&n()},\"dragstart.scrollbar\":function(e){e.preventDefault();return false},\"mouseup.scrollbar\":function(){e(t).add(\"body\").off(\".scrollbar\");n&&n()}});e(\"body\").on({\"selectstart.scrollbar\":function(e){e.preventDefault();return false}});r&&r.preventDefault();return false}function d(){var e=h(true);return!(e.height||e.width)}function v(e){var t=e.originalEvent;if(t.axis&&t.axis===t.HORIZONTAL_AXIS)return false;if(t.wheelDeltaX)return false;return true}var r=false;var i=1,s=\"px\";var o={data:{},macosx:n.navigator.platform.toLowerCase().indexOf(\"mac\")!==-1,mobile:/Android|webOS|iPhone|iPad|iPod|BlackBerry/i.test(n.navigator.userAgent),overlay:null,scroll:null,scrolls:[],webkit:/WebKit/.test(n.navigator.userAgent),log:r?function(t,r){var i=t;if(r&&typeof t!=\"string\"){i=[];e.each(t,function(e,t){i.push('\"'+e+'\": '+t)});i=i.join(\", \")}if(n.console&&n.console.log){n.console.log(i)}else{alert(i)}}:function(){}};var u={autoScrollSize:true,autoUpdate:true,debug:false,disableBodyScroll:false,duration:200,ignoreMobile:true,ignoreOverlay:true,scrollStep:30,showArrows:false,stepScrolling:true,type:\"simple\",scrollx:null,scrolly:null,onDestroy:null,onInit:null,onScroll:null,onUpdate:null};var a=function(t,r){if(!o.scroll){o.log(\"Init jQuery Scrollbar v0.2.6\");o.overlay=d();o.scroll=h();c();e(n).resize(function(){var e=false;if(o.scroll&&(o.scroll.height||o.scroll.width)){var t=h();if(t.height!=o.scroll.height||t.width!=o.scroll.width){o.scroll=t;e=true}}c(e)})}this.container=t;this.options=e.extend({},u,n.jQueryScrollbarOptions||{});this.scrollTo=null;this.scrollx={};this.scrolly={};this.init(r)};a.prototype={destroy:function(){if(!this.wrapper){return}var n=this.container.scrollLeft();var r=this.container.scrollTop();this.container.insertBefore(this.wrapper).css({height:\"\",margin:\"\"}).removeClass(\"scroll-content\").removeClass(\"scroll-scrollx_visible\").removeClass(\"scroll-scrolly_visible\").off(\".scrollbar\").scrollLeft(n).scrollTop(r);this.scrollx.scrollbar.removeClass(\"scroll-scrollx_visible\").find(\"div\").andSelf().off(\".scrollbar\");this.scrolly.scrollbar.removeClass(\"scroll-scrolly_visible\").find(\"div\").andSelf().off(\".scrollbar\");this.wrapper.remove();e(t).add(\"body\").off(\".scrollbar\");if(e.isFunction(this.options.onDestroy))this.options.onDestroy.apply(this,[this.container])},getScrollbar:function(t){var n=this.options[\"scroll\"+t];var r={advanced:'<div class=\"scroll-element_corner\"></div>'+'<div class=\"scroll-arrow scroll-arrow_less\"></div>'+'<div class=\"scroll-arrow scroll-arrow_more\"></div>'+'<div class=\"scroll-element_outer\">'+'    <div class=\"scroll-element_size\"></div>'+'    <div class=\"scroll-element_inner-wrapper\">'+'        <div class=\"scroll-element_inner scroll-element_track\">'+'            <div class=\"scroll-element_inner-bottom\"></div>'+\"        </div>\"+\"    </div>\"+'    <div class=\"scroll-bar\">'+'        <div class=\"scroll-bar_body\">'+'            <div class=\"scroll-bar_body-inner\"></div>'+\"        </div>\"+'        <div class=\"scroll-bar_bottom\"></div>'+'        <div class=\"scroll-bar_center\"></div>'+\"    </div>\"+\"</div>\",simple:'<div class=\"scroll-element_outer\">'+'    <div class=\"scroll-element_size\"></div>'+'    <div class=\"scroll-element_track\"></div>'+'    <div class=\"scroll-bar\"></div>'+\"</div>\"};var i=r[this.options.type]?this.options.type:\"advanced\";if(n){if(typeof n==\"string\"){n=e(n).appendTo(this.wrapper)}else{n=e(n)}}else{n=e(\"<div>\").addClass(\"scroll-element\").html(r[i]).appendTo(this.wrapper)}if(this.options.showArrows){n.addClass(\"scroll-element_arrows_visible\")}return n.addClass(\"scroll-\"+t)},init:function(n){var r=this;var u=this.container;var a=this.containerWrapper||u;var f=e.extend(this.options,n||{});var l={x:this.scrollx,y:this.scrolly};var c=this.wrapper;var h={scrollLeft:u.scrollLeft(),scrollTop:u.scrollTop()};if(o.mobile&&f.ignoreMobile||o.overlay&&f.ignoreOverlay||o.macosx&&!o.webkit){return false}if(!c){this.wrapper=c=e(\"<div>\").addClass(\"scroll-wrapper\").addClass(u.attr(\"class\")).css(\"position\",u.css(\"position\")==\"absolute\"?\"absolute\":\"relative\").insertBefore(u).append(u);if(u.is(\"textarea\")){this.containerWrapper=a=e(\"<div>\").insertBefore(u).append(u);c.addClass(\"scroll-textarea\")}a.addClass(\"scroll-content\").css({height:\"\",\"margin-bottom\":o.scroll.height*-1+s,\"margin-right\":o.scroll.width*-1+s});u.on(\"scroll.scrollbar\",function(t){if(e.isFunction(f.onScroll)){f.onScroll.call(r,{maxScroll:l.y.maxScrollOffset,scroll:u.scrollTop(),size:l.y.size,visible:l.y.visible},{maxScroll:l.x.maxScrollOffset,scroll:u.scrollLeft(),size:l.x.size,visible:l.x.visible})}l.x.isVisible&&l.x.scroller.css(\"left\",u.scrollLeft()*l.x.kx+s);l.y.isVisible&&l.y.scroller.css(\"top\",u.scrollTop()*l.y.kx+s)});c.on(\"scroll\",function(){c.scrollTop(0).scrollLeft(0)});if(f.disableBodyScroll){var d=function(e){v(e)?l.y.isVisible&&l.y.mousewheel(e):l.x.isVisible&&l.x.mousewheel(e)};c.on({\"MozMousePixelScroll.scrollbar\":d,\"mousewheel.scrollbar\":d});if(o.mobile){c.on(\"touchstart.scrollbar\",function(n){var r=n.originalEvent.touches&&n.originalEvent.touches[0]||n;var i={pageX:r.pageX,pageY:r.pageY};var s={left:u.scrollLeft(),top:u.scrollTop()};e(t).on({\"touchmove.scrollbar\":function(e){var t=e.originalEvent.targetTouches&&e.originalEvent.targetTouches[0]||e;u.scrollLeft(s.left+i.pageX-t.pageX);u.scrollTop(s.top+i.pageY-t.pageY);e.preventDefault()},\"touchend.scrollbar\":function(){e(t).off(\".scrollbar\")}})})}}if(e.isFunction(f.onInit))f.onInit.apply(this,[u])}else{a.css({height:\"\",\"margin-bottom\":o.scroll.height*-1+s,\"margin-right\":o.scroll.width*-1+s})}e.each(l,function(n,s){var o=null;var a=1;var c=n==\"x\"?\"scrollLeft\":\"scrollTop\";var h=f.scrollStep;var d=function(){var e=u[c]();u[c](e+h);if(a==1&&e+h>=m)e=u[c]();if(a==-1&&e+h<=m)e=u[c]();if(u[c]()==e&&o){o()}};var m=0;if(!s.scrollbar){s.scrollbar=r.getScrollbar(n);s.scroller=s.scrollbar.find(\".scroll-bar\");s.mousewheel=function(e){if(!s.isVisible||n==\"x\"&&v(e)){return true}if(n==\"y\"&&!v(e)){l.x.mousewheel(e);return true}var t=e.originalEvent.wheelDelta*-1||e.originalEvent.detail;var i=s.size-s.visible-s.offset;if(!(m<=0&&t<0||m>=i&&t>0)){m=m+t;if(m<0)m=0;if(m>i)m=i;r.scrollTo=r.scrollTo||{};r.scrollTo[c]=m;setTimeout(function(){if(r.scrollTo){u.stop().animate(r.scrollTo,240,\"linear\",function(){m=u[c]()});r.scrollTo=null}},1)}e.preventDefault();return false};s.scrollbar.on({\"MozMousePixelScroll.scrollbar\":s.mousewheel,\"mousewheel.scrollbar\":s.mousewheel,\"mouseenter.scrollbar\":function(){m=u[c]()}});s.scrollbar.find(\".scroll-arrow, .scroll-element_track\").on(\"mousedown.scrollbar\",function(t){if(t.which!=i)return true;a=1;var l={eventOffset:t[n==\"x\"?\"pageX\":\"pageY\"],maxScrollValue:s.size-s.visible-s.offset,scrollbarOffset:s.scroller.offset()[n==\"x\"?\"left\":\"top\"],scrollbarSize:s.scroller[n==\"x\"?\"outerWidth\":\"outerHeight\"]()};var v=0,g=0;if(e(this).hasClass(\"scroll-arrow\")){a=e(this).hasClass(\"scroll-arrow_more\")?1:-1;h=f.scrollStep*a;m=a>0?l.maxScrollValue:0}else{a=l.eventOffset>l.scrollbarOffset+l.scrollbarSize?1:l.eventOffset<l.scrollbarOffset?-1:0;h=Math.round(s.visible*.75)*a;m=l.eventOffset-l.scrollbarOffset-(f.stepScrolling?a==1?l.scrollbarSize:0:Math.round(l.scrollbarSize/2));m=u[c]()+m/s.kx}r.scrollTo=r.scrollTo||{};r.scrollTo[c]=f.stepScrolling?u[c]()+h:m;if(f.stepScrolling){o=function(){m=u[c]();clearInterval(g);clearTimeout(v);v=0;g=0};v=setTimeout(function(){g=setInterval(d,40)},f.duration+100)}setTimeout(function(){if(r.scrollTo){u.animate(r.scrollTo,f.duration);r.scrollTo=null}},1);return p(o,t)});s.scroller.on(\"mousedown.scrollbar\",function(r){if(r.which!=i)return true;var o=r[n==\"x\"?\"pageX\":\"pageY\"];var a=u[c]();s.scrollbar.addClass(\"scroll-draggable\");e(t).on(\"mousemove.scrollbar\",function(e){var t=parseInt((e[n==\"x\"?\"pageX\":\"pageY\"]-o)/s.kx,10);u[c](a+t)});return p(function(){s.scrollbar.removeClass(\"scroll-draggable\");m=u[c]()},r)})}});e.each(l,function(e,t){var n=\"scroll-scroll\"+e+\"_visible\";var r=e==\"x\"?l.y:l.x;t.scrollbar.removeClass(n);r.scrollbar.removeClass(n);a.removeClass(n)});e.each(l,function(t,n){e.extend(n,t==\"x\"?{offset:parseInt(u.css(\"left\"),10)||0,size:u.prop(\"scrollWidth\"),visible:c.width()}:{offset:parseInt(u.css(\"top\"),10)||0,size:u.prop(\"scrollHeight\"),visible:c.height()})});var m=function(t,n){var r=\"scroll-scroll\"+t+\"_visible\";var i=t==\"x\"?l.y:l.x;var f=parseInt(u.css(t==\"x\"?\"left\":\"top\"),10)||0;var h=n.size;var p=n.visible+f;n.isVisible=h-p>1;if(n.isVisible){n.scrollbar.addClass(r);i.scrollbar.addClass(r);a.addClass(r)}else{n.scrollbar.removeClass(r);i.scrollbar.removeClass(r);a.removeClass(r)}if(t==\"y\"&&(n.isVisible||n.size<n.visible)){a.css(\"height\",p+o.scroll.height+s)}if(l.x.size!=u.prop(\"scrollWidth\")||l.y.size!=u.prop(\"scrollHeight\")||l.x.visible!=c.width()||l.y.visible!=c.height()||l.x.offset!=(parseInt(u.css(\"left\"),10)||0)||l.y.offset!=(parseInt(u.css(\"top\"),10)||0)){e.each(l,function(t,n){e.extend(n,t==\"x\"?{offset:parseInt(u.css(\"left\"),10)||0,size:u.prop(\"scrollWidth\"),visible:c.width()}:{offset:parseInt(u.css(\"top\"),10)||0,size:u.prop(\"scrollHeight\"),visible:c.height()})});m(t==\"x\"?\"y\":\"x\",i)}};e.each(l,m);if(e.isFunction(f.onUpdate))f.onUpdate.apply(this,[u]);e.each(l,function(e,t){var n=e==\"x\"?\"left\":\"top\";var r=e==\"x\"?\"outerWidth\":\"outerHeight\";var i=e==\"x\"?\"width\":\"height\";var o=parseInt(u.css(n),10)||0;var a=t.size;var l=t.visible+o;var c=t.scrollbar.find(\".scroll-element_size\");c=c[r]()+(parseInt(c.css(n),10)||0);if(f.autoScrollSize){t.scrollbarSize=parseInt(c*l/a,10);t.scroller.css(i,t.scrollbarSize+s)}t.scrollbarSize=t.scroller[r]();t.kx=(c-t.scrollbarSize)/(a-l)||1;t.maxScrollOffset=a-l});u.scrollLeft(h.scrollLeft).scrollTop(h.scrollTop).trigger(\"scroll\")}};e.fn.scrollbar=function(t,n){var r=this;if(t===\"get\")r=null;this.each(function(){var i=e(this);if(i.hasClass(\"scroll-wrapper\")||i.get(0).nodeName==\"body\"){return true}var s=i.data(\"scrollbar\");if(s){if(t===\"get\"){r=s;return false}var u=typeof t==\"string\"&&s[t]?t:\"init\";s[u].apply(s,e.isArray(n)?n:[]);if(t===\"destroy\"){i.removeData(\"scrollbar\");while(e.inArray(s,o.scrolls)>=0)o.scrolls.splice(e.inArray(s,o.scrolls),1)}}else{if(typeof t!=\"string\"){s=new a(i,t);i.data(\"scrollbar\",s);o.scrolls.push(s)}}return true});return r};e.fn.scrollbar.options=u;if(n.angular){(function(e){var t=e.module(\"jQueryScrollbar\",[]);t.directive(\"jqueryScrollbar\",function(){return{link:function(e,t){t.scrollbar(e.options).on(\"$destroy\",function(){t.scrollbar(\"destroy\")})},restring:\"AC\",scope:{options:\"=jqueryScrollbar\"}}})})(n.angular)}var f=0,l=0;var c=function(e){var t,n,i,s,u,a,h;for(t=0;t<o.scrolls.length;t++){s=o.scrolls[t];n=s.container;i=s.options;u=s.wrapper;a=s.scrollx;h=s.scrolly;if(e||i.autoUpdate&&u&&u.is(\":visible\")&&(n.prop(\"scrollWidth\")!=a.size||n.prop(\"scrollHeight\")!=h.size||u.width()!=a.visible||u.height()!=h.visible)){s.init();if(r){o.log({scrollHeight:n.prop(\"scrollHeight\")+\":\"+s.scrolly.size,scrollWidth:n.prop(\"scrollWidth\")+\":\"+s.scrollx.size,visibleHeight:u.height()+\":\"+s.scrolly.visible,visibleWidth:u.width()+\":\"+s.scrollx.visible},true);l++}}}if(r&&l>10){o.log(\"Scroll updates exceed 10\");c=function(){}}else{clearTimeout(f);f=setTimeout(c,300)}}})(jQuery,document,window);","// Storage cache.\r\nvar cache = {};\r\n// The store handling expiration of data.\r\nvar expiresStore = new Store({\r\n\tnamespace: '__storage-wrapper:expires'\r\n});\r\n\r\n/**\r\n * Storage wrapper for making routine storage calls super easy.\r\n * @class Store\r\n * @constructor\r\n * @param {object} [options]                     The options for the store. Options not overridden will use the defaults.\r\n * @param {mixed}  [options.namespace='']        See {{#crossLink \"Store/setNamespace\"}}Store#setNamespace{{/crossLink}}\r\n * @param {mixed}  [options.storageType='local'] See {{#crossLink \"Store/setStorageType\"}}Store#setStorageType{{/crossLink}}\r\n */\r\nfunction Store(options) {\r\n\tvar settings = {\r\n\t\tnamespace: '',\r\n\t\tstorageType: 'local'\r\n\t};\r\n\r\n\t/**\r\n\t * Sets the storage namespace.\r\n\t * @method setNamespace\r\n\t * @param {string|false|null} namespace The namespace to work under. To use no namespace (e.g. global namespace), pass in `false` or `null` or an empty string.\r\n\t */\r\n\tthis.setNamespace = function (namespace) {\r\n\t\tvar validNamespace = /^[\\w-:]+$/;\r\n\t\t// No namespace.\r\n\t\tif (namespace === false || namespace == null || namespace === '') {\r\n\t\t\tsettings.namespace = '';\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tif (typeof namespace !== 'string' || !validNamespace.test(namespace)) {\r\n\t\t\tthrow new Error('Invalid namespace.');\r\n\t\t}\r\n\t\tsettings.namespace = namespace;\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the current storage namespace.\r\n\t * @method getNamespace\r\n\t * @return {string} The current namespace.\r\n\t */\r\n\tthis.getNamespace = function (includeSeparator) {\r\n\t\tif (includeSeparator && settings.namespace !== '') {\r\n\t\t\treturn settings.namespace + ':';\r\n\t\t}\r\n\t\treturn settings.namespace;\r\n\t}\r\n\r\n\t/**\r\n\t * Sets the type of storage to use.\r\n\t * @method setStorageType\r\n\t * @param {string} type The type of storage to use. Use `session` for `sessionStorage` and `local` for `localStorage`.\r\n\t */\r\n\tthis.setStorageType = function (type) {\r\n\t\tif (['session', 'local'].indexOf(type) < 0) {\r\n\t\t\tthrow new Error('Invalid storage type.');\r\n\t\t}\r\n\t\tsettings.storageType = type;\r\n\t};\r\n\t/**\r\n\t * Get the type of storage being used.\r\n\t * @method getStorageType\r\n\t * @return {string} The type of storage being used.\r\n\t */\r\n\tthis.getStorageType = function () {\r\n\t\treturn settings.storageType;\r\n\t};\r\n\r\n\t// Override default settings.\r\n\tif (options) {\r\n\t\tfor (var key in options) {\r\n\t\t\tswitch (key) {\r\n\t\t\t\tcase 'namespace':\r\n\t\t\t\t\tthis.setNamespace(options[key]);\r\n\t\t\t\t\tbreak;\r\n\t\t\t\tcase 'storageType':\r\n\t\t\t\t\tthis.setStorageType(options[key]);\r\n\t\t\t\t\tbreak;\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n}\r\n\r\n/**\r\n * Gets the actual handler to use\r\n * @method getStorageHandler\r\n * @return {mixed} The storage handler.\r\n */\r\nStore.prototype.getStorageHandler = function () {\r\n\tvar handlers = {\r\n\t\t'local': localStorage,\r\n\t\t'session': sessionStorage\r\n\t};\r\n\treturn handlers[this.getStorageType()];\r\n}\r\n\r\n/**\r\n * Gets the full storage name for a key, including the namespace, if any.\r\n * @method getStorageKey\r\n * @param  {string} key The storage key name.\r\n * @return {string}     The full storage name that is used by the storage methods.\r\n */\r\nStore.prototype.getStorageKey = function (key) {\r\n\tif (!key || typeof key !== 'string' || key.length < 1) {\r\n\t\tthrow new Error('Key must be a string.');\r\n\t}\r\n\treturn this.getNamespace(true) + key;\r\n};\r\n\r\n/**\r\n * Gets a storage item from the current namespace.\r\n * @method get\r\n * @param  {string} key          The key that the data can be accessed under.\r\n * @param  {mixed}  defaultValue The default value to return in case the storage value is not set or `null`.\r\n * @return {mixed}               The data for the storage.\r\n */\r\nStore.prototype.get = function (key, defaultValue) {\r\n\t// Prevent recursion. Only check expire date if it isn't called from `expiresStore`.\r\n\tif (this !== expiresStore) {\r\n\t\t// Check if key is expired.\r\n\t\tvar expireDate = expiresStore.get(this.getStorageKey(key));\r\n\t\tif (expireDate !== null && expireDate.getTime() < Date.now()) {\r\n\t\t\t// Expired, remove it.\r\n\t\t\tthis.remove(key);\r\n\t\t\texpiresStore.remove(this.getStorageKey(key));\r\n\t\t}\r\n\t}\r\n\r\n\t// Cached, read from memory.\r\n\tif (cache[this.getStorageKey(key)] != null) {\r\n\t\treturn cache[this.getStorageKey(key)];\r\n\t}\r\n\r\n\tvar val = this.getStorageHandler().getItem(this.getStorageKey(key));\r\n\r\n\t// Value doesn't exist and we have a default, return default.\r\n\tif (val === null && typeof defaultValue !== 'undefined') {\r\n\t\treturn defaultValue;\r\n\t}\r\n\r\n\t// Only pre-process strings.\r\n\tif (typeof val === 'string') {\r\n\t\t// Handle RegExps.\r\n\t\tif (val.indexOf('~RegExp:') === 0) {\r\n\t\t\tvar matches = /^~RegExp:([gim]*?):(.*)/.exec(val);\r\n\t\t\tval = new RegExp(matches[2], matches[1]);\r\n\t\t}\r\n\t\t// Handle Dates.\r\n\t\telse if (val.indexOf('~Date:') === 0) {\r\n\t\t\tval = new Date(val.replace(/^~Date:/, ''));\r\n\t\t}\r\n\t\t// Handle numbers.\r\n\t\telse if (val.indexOf('~Number:') === 0) {\r\n\t\t\tval = parseInt(val.replace(/^~Number:/, ''), 10);\r\n\t\t}\r\n\t\t// Handle booleans.\r\n\t\telse if (val.indexOf('~Boolean:') === 0) {\r\n\t\t\tval = val.replace(/^~Boolean:/, '') === 'true';\r\n\t\t}\r\n\t\t// Handle objects.\r\n\t\telse if (val.indexOf('~JSON:') === 0) {\r\n\t\t\tval = val.replace(/^~JSON:/, '');\r\n\t\t\t// Try parsing it.\r\n\t\t\ttry {\r\n\t\t\t\tval = JSON.parse(val);\r\n\t\t\t}\r\n\t\t\t// Parsing went wrong (invalid JSON), return default or null.\r\n\t\t\tcatch (e) {\r\n\t\t\t\tif (typeof defaultValue !== 'undefined') {\r\n\t\t\t\t\treturn defaultValue;\r\n\t\t\t\t}\r\n\t\t\t\treturn null;\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n\r\n\t// Return it.\r\n\tcache[this.getStorageKey(key)] = val;\r\n\treturn val;\r\n};\r\n\r\n/**\r\n * Sets a storage item on the current namespace.\r\n * @method set\r\n * @param {string}      key       The key that the data can be accessed under.\r\n * @param {mixed}       val       The value to store. May be the following types of data: `RegExp`, `Date`, `Object`, `String`, `Boolean`, `Number`\r\n * @param {Date|number} [expires] The date in the future to expire, or relative number of milliseconds from `Date#now` to expire.\r\n *\r\n * Note: This converts special data types that normally can't be stored in the following way:\r\n * \r\n * - `RegExp`: prefixed with type, flags stored, and source stored as string.\r\n * - `Date`: prefixed with type, stored as string using `Date#toString`.\r\n * - `Object`: prefixed with \"JSON\" indicator, stored as string using `JSON#stringify`.\r\n */\r\nStore.prototype.set = function (key, val, expires) {\r\n\tvar parsedVal = null;\r\n\t// Handle RegExps.\r\n\tif (val instanceof RegExp) {\r\n\t\tvar flags = [\r\n\t\t\tval.global ? 'g' : '',\r\n\t\t\tval.ignoreCase ? 'i' : '',\r\n\t\t\tval.multiline ? 'm' : '',\r\n\t\t].join('');\r\n\t\tparsedVal = '~RegExp:' + flags + ':' + val.source;\r\n\t}\r\n\t// Handle Dates.\r\n\telse if (val instanceof Date) {\r\n\t\tparsedVal = '~Date:' + val.toString();\r\n\t}\r\n\t// Handle objects.\r\n\telse if (val === Object(val)) {\r\n\t\tparsedVal = '~JSON:' + JSON.stringify(val);\r\n\t}\r\n\t// Handle numbers.\r\n\telse if (typeof val === 'number') {\r\n\t\tparsedVal = '~Number:' + val.toString();\r\n\t}\r\n\t// Handle booleans.\r\n\telse if (typeof val === 'boolean') {\r\n\t\tparsedVal = '~Boolean:' + val.toString();\r\n\t}\r\n\t// Handle strings.\r\n\telse if (typeof val === 'string') {\r\n\t\tparsedVal = val;\r\n\t}\r\n\t// Throw if we don't know what it is.\r\n\telse {\r\n\t\tthrow new Error('Unable to store this value; wrong value type.');\r\n\t}\r\n\t// Set expire date if needed.\r\n\tif (typeof expires !== 'undefined') {\r\n\t\t// Convert to a relative date.\r\n\t\tif (typeof expires === 'number') {\r\n\t\t\texpires = new Date(Date.now() + expires);\r\n\t\t}\r\n\t\t// Make sure it is a date.\r\n\t\tif (expires instanceof Date) {\r\n\t\t\texpiresStore.set(this.getStorageKey(key), expires);\r\n\t\t}\r\n\t\telse {\r\n\t\t\tthrow new Error('Key expire must be a valid date or timestamp.');\r\n\t\t}\r\n\t}\r\n\t// Save it.\r\n\tcache[this.getStorageKey(key)] = val;\r\n\tthis.getStorageHandler().setItem(this.getStorageKey(key), parsedVal);\r\n};\r\n\r\n/**\r\n * Gets all data for the current namespace.\r\n * @method getAll\r\n * @return {object} An object containing all data in the form of `{theKey: theData}` where `theData` is parsed using {{#crossLink \"Store/get\"}}Store#get{{/crossLink}}.\r\n */\r\nStore.prototype.getAll = function () {\r\n\tvar keys = this.listKeys();\r\n\tvar data = {};\r\n\tkeys.forEach(function (key) {\r\n\t\tdata[key] = this.get(key);\r\n\t}, this);\r\n\treturn data;\r\n};\r\n\r\n/**\r\n * List all keys that are tied to the current namespace.\r\n * @method listKeys\r\n * @return {array} The storage keys.\r\n */\r\nStore.prototype.listKeys = function () {\r\n\tvar keys = [];\r\n\tvar key = null;\r\n\tvar storageLength = this.getStorageHandler().length;\r\n\tvar prefix = new RegExp('^' + this.getNamespace(true));\r\n\tfor (var i = 0; i < storageLength; i++) {\r\n\t\tkey = this.getStorageHandler().key(i)\r\n\t\tif (prefix.test(key)) {\r\n\t\t\tkeys.push(key.replace(prefix, ''));\r\n\t\t}\r\n\t}\r\n\treturn keys;\r\n};\r\n\r\n/**\r\n * Removes a specific key and data from the current namespace.\r\n * @method remove\r\n * @param {string} key The key to remove the data for.\r\n */\r\nStore.prototype.remove = function (key) {\r\n\tcache[this.getStorageKey(key)] = null;\r\n\tthis.getStorageHandler().removeItem(this.getStorageKey(key));\r\n};\r\n\r\n/**\r\n * Removes all data and keys from the current namespace.\r\n * @method removeAll\r\n */\r\nStore.prototype.removeAll = function () {\r\n\tthis.listKeys().forEach(this.remove, this);\r\n};\r\n\r\n/**\r\n * Removes namespaced items from the cache so your next {{#crossLink \"Store/get\"}}Store#get{{/crossLink}} will be fresh from the storage.\r\n * @method freshen\r\n * @param {string} key The key to remove the cache data for.\r\n */\r\nStore.prototype.freshen = function (key) {\r\n\tvar keys = key ? [key] : this.listKeys();\r\n\tkeys.forEach(function (key) {\r\n\t\tcache[this.getStorageKey(key)] = null;\r\n\t}, this);\r\n};\r\n\r\n/**\r\n * Migrate data from a different namespace to current namespace.\r\n * @method migrate\r\n * @param {object}   migration                          The migration object.\r\n * @param {string}   migration.toKey                    The key name under your current namespace the old data should change to.\r\n * @param {string}   migration.fromNamespace            The old namespace that the old key belongs to.\r\n * @param {string}   migration.fromKey                  The old key name to migrate from.\r\n * @param {string}   [migration.fromStorageType]        The storage type to migrate from. Defaults to same type as where you are migrating to.\r\n * @param {boolean}  [migration.keepOldData=false]      Whether old data should be kept after it has been migrated.\r\n * @param {boolean}  [migration.overwriteNewData=false] Whether old data should overwrite currently stored data if it exists.\r\n * @param {function} [migration.transform]              The function to pass the old key data through before migrating.\r\n * @example\r\n * \r\n *     var Store = require('storage-wrapper');\r\n *     var store = new Store({\r\n *         namespace: 'myNewApp'\r\n *     });\r\n *\r\n *     // Migrate from the old app.\r\n *     store.migrate({\r\n *         toKey: 'new-key',\r\n *         fromNamespace: 'myOldApp',\r\n *         fromKey: 'old-key'\r\n *     });\r\n *     \r\n *     // Migrate from global data. Useful when moving from other storage wrappers or regular ol' `localStorage`.\r\n *     store.migrate({\r\n *         toKey: 'other-new-key',\r\n *         fromNamespace: '',\r\n *         fromKey: 'other-old-key-on-global'\r\n *     });\r\n *     \r\n *     // Migrate some JSON data that was stored as a string.\r\n *     store.migrate({\r\n *         toKey: 'new-json-key',\r\n *         fromNamespace: 'myOldApp',\r\n *         fromKey: 'old-json-key',\r\n *         // Try converting some old JSON data.\r\n *         transform: function (data) {\r\n *             try {\r\n *                 return JSON.parse(data);\r\n *             }\r\n *             catch (e) {\r\n *                 return data;\r\n *             }\r\n *         }\r\n *     });\r\n */\r\n\r\nStore.prototype.migrate = function (migration) {\r\n\t// Save our current namespace.\r\n\tvar toNamespace = this.getNamespace();\r\n\tvar toStorageType = this.getStorageType();\r\n\r\n\t// Create a temporary store to avoid changing namespace during actual get/sets.\r\n\tvar store = new Store({\r\n\t\tnamespace: toNamespace,\r\n\t\tstorageType: toStorageType\r\n\t});\r\n\r\n\tvar data = null;\r\n\r\n\t// Get data from old namespace.\r\n\tstore.setNamespace(migration.fromNamespace);\r\n\tif (typeof migration.fromStorageType !== 'undefined') {\r\n\t\tstore.setStorageType(migration.fromStorageType);\r\n\t}\r\n\tdata = store.get(migration.fromKey);\r\n\r\n\t// Remove old if needed.\r\n\tif (!migration.keepOldData) {\r\n\t\tstore.remove(migration.fromKey);\r\n\t}\r\n\t\r\n\t// No data, ignore this migration.\r\n\tif (data === null) {\r\n\t\treturn;\r\n\t}\r\n\r\n\t// Transform data if needed.\r\n\tif (typeof migration.transform === 'function') {\r\n\t\tdata = migration.transform(data);\r\n\t}\r\n\telse if (typeof migration.transform !== 'undefined') {\r\n\t\tthrow new Error('Invalid transform callback.');\r\n\t}\r\n\r\n\t// Go back to current namespace.\r\n\tstore.setNamespace(toNamespace);\r\n\tstore.setStorageType(toStorageType);\r\n\r\n\t// Only overwrite new data if it doesn't exist or it's requested.\r\n\tif (store.get(migration.toKey) === null || migration.overwriteNewData) {\r\n\t\tstore.set(migration.toKey, data);\r\n\t}\r\n};\r\n\r\n/**\r\n * Creates a substore that is nested in the current namespace.\r\n * @method createSubstore\r\n * @param  {string} namespace The substore's namespace.\r\n * @return {Store}            The substore.\r\n * @example\r\n * \r\n *     var Store = require('storage-wrapper');\r\n *     // Create main store.\r\n *     var store = new Store({\r\n *         namespace: 'myapp'\r\n *     });\r\n *\r\n *     // Create substore.\r\n *     var substore = store.createSubstore('things');\r\n *     substore.set('foo', 'bar');\r\n *\r\n *     substore.get('foo') === store.get('things:foo');\r\n *     // true\r\n */\r\nStore.prototype.createSubstore = function (namespace) {\r\n\treturn new Store({\r\n\t\tnamespace: this.getNamespace(true) + namespace,\r\n\t\tstorageType: this.getStorageType()\r\n\t});\r\n};\r\n\r\nmodule.exports = Store;\r\n","module.exports={\r\n  \"name\": \"twitch-chat-emotes\",\r\n  \"version\": \"2.1.4\",\r\n  \"homepage\": \"http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/\",\r\n  \"bugs\": \"https://github.com/cletusc/Userscript--Twitch-Chat-Emotes/issues\",\r\n  \"author\": \"Ryan Chatham <ryan.b.chatham@gmail.com> (https://github.com/cletusc)\",\r\n  \"repository\": {\r\n    \"type\": \"git\",\r\n    \"url\": \"https://github.com/cletusc/Userscript--Twitch-Chat-Emotes.git\"\r\n  },\r\n  \"userscript\": {\r\n    \"name\": \"Twitch Chat Emotes\",\r\n    \"namespace\": \"#Cletus\",\r\n    \"version\": \"{{{pkg.version}}}\",\r\n    \"description\": \"Adds a button to Twitch that allows you to \\\"click-to-insert\\\" an emote.\",\r\n    \"copyright\": \"2011+, {{{pkg.author}}}\",\r\n    \"author\": \"{{{pkg.author}}}\",\r\n    \"icon\": \"http://www.gravatar.com/avatar.php?gravatar_id=6875e83aa6c563790cb2da914aaba8b3&r=PG&s=48&default=identicon\",\r\n    \"license\": [\r\n      \"MIT; http://opensource.org/licenses/MIT\",\r\n      \"CC BY-NC-SA 3.0; http://creativecommons.org/licenses/by-nc-sa/3.0/\"\r\n    ],\r\n    \"homepage\": \"{{{pkg.homepage}}}\",\r\n    \"supportURL\": \"{{{pkg.bugs}}}\",\r\n    \"contributionURL\": \"http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/#donate\",\r\n    \"grant\": \"none\",\r\n    \"include\": [\r\n      \"http://*.twitch.tv/*\",\r\n      \"https://*.twitch.tv/*\"\r\n    ],\r\n    \"exclude\": [\r\n      \"http://api.twitch.tv/*\",\r\n      \"https://api.twitch.tv/*\",\r\n      \"http://tmi.twitch.tv/*\",\r\n      \"https://tmi.twitch.tv/*\",\r\n      \"http://*.twitch.tv/*/dashboard\",\r\n      \"https://*.twitch.tv/*/dashboard\",\r\n      \"http://chatdepot.twitch.tv/*\",\r\n      \"https://chatdepot.twitch.tv/*\",\r\n      \"http://im.twitch.tv/*\",\r\n      \"https://im.twitch.tv/*\",\r\n      \"http://platform.twitter.com/*\",\r\n      \"https://platform.twitter.com/*\",\r\n      \"http://www.facebook.com/*\",\r\n      \"https://www.facebook.com/*\"\r\n    ]\r\n  },\r\n  \"devDependencies\": {\r\n    \"browser-sync\": \"^1.3.2\",\r\n    \"browserify\": \"^5.9.1\",\r\n    \"generate-userscript-header\": \"^1.0.0\",\r\n    \"gulp\": \"^3.8.3\",\r\n    \"gulp-autoprefixer\": \"0.0.8\",\r\n    \"gulp-beautify\": \"1.1.0\",\r\n    \"gulp-changed\": \"^0.4.1\",\r\n    \"gulp-concat\": \"^2.2.0\",\r\n    \"gulp-conflict\": \"^0.1.2\",\r\n    \"gulp-css-base64\": \"^1.1.0\",\r\n    \"gulp-css2js\": \"^1.0.2\",\r\n    \"gulp-header\": \"^1.0.2\",\r\n    \"gulp-hogan-compile\": \"^0.2.1\",\r\n    \"gulp-minify-css\": \"^0.3.5\",\r\n    \"gulp-notify\": \"^1.4.1\",\r\n    \"gulp-rename\": \"^1.2.0\",\r\n    \"gulp-uglify\": \"^0.3.1\",\r\n    \"gulp-util\": \"^3.0.0\",\r\n    \"hogan.js\": \"^3.0.2\",\r\n    \"jquery-ui\": \"^1.10.5\",\r\n    \"jquery.scrollbar\": \"^0.2.7\",\r\n    \"pretty-hrtime\": \"^0.2.1\",\r\n    \"storage-wrapper\": \"cletusc/storage-wrapper#v0.1.1\",\r\n    \"vinyl-map\": \"^1.0.1\",\r\n    \"vinyl-source-stream\": \"^0.1.1\",\r\n    \"watchify\": \"^1.0.1\"\r\n  }\r\n}\r\n","var logger = require('./logger');\r\nvar api = {};\r\nvar ember = null;\r\nvar hookedFactories = {};\r\n\r\napi.getEmber = function () {\r\n\tif (ember) {\r\n\t\treturn ember;\r\n\t}\r\n\tif (window.App && window.App.__container__) {\r\n\t\tember = window.App.__container__;\r\n\t\treturn ember;\r\n\t}\r\n\treturn false;\r\n};\r\n\r\napi.isLoaded = function () {\r\n\treturn Boolean(api.getEmber());\r\n};\r\n\r\napi.lookup = function (lookupFactory) {\r\n\tif (!api.isLoaded()) {\r\n\t\tlogger.debug('Factory lookup failure, Ember not loaded.');\r\n\t\treturn false;\r\n\t}\r\n\treturn api.getEmber().lookup(lookupFactory);\r\n};\r\n\r\napi.hook = function (lookupFactory, activateCb, deactivateCb) {\r\n\tif (!api.isLoaded()) {\r\n\t\tlogger.debug('Factory hook failure, Ember not loaded.');\r\n\t\treturn false;\r\n\t}\r\n\tif (hookedFactories[lookupFactory]) {\r\n\t\tlogger.debug('Factory already hooked: ' + lookupFactory);\r\n\t\treturn true;\r\n\t}\r\n\tvar reopenOptions = {};\r\n\tvar factory = api.lookup(lookupFactory);\r\n\r\n\tif (!factory) {\r\n\t\tlogger.debug('Factory hook failure, factory not found: ' + lookupFactory);\r\n\t\treturn false;\r\n\t}\r\n\r\n\tif (activateCb) {\r\n\t\treopenOptions.activate = function () {\r\n\t\t\tthis._super();\r\n\t\t\tactivateCb.call(this);\r\n\t\t\tlogger.debug('Hook run on activate: ' + lookupFactory);\r\n\t\t};\r\n\t}\r\n\tif (deactivateCb) {\r\n\t\treopenOptions.deactivate = function () {\r\n\t\t\tthis._super();\r\n\t\t\tdeactivateCb.call(this);\r\n\t\t\tlogger.debug('Hook run on deactivate: ' + lookupFactory);\r\n\t\t};\r\n\t}\r\n\r\n\ttry {\r\n\t\tfactory.reopen(reopenOptions);\r\n\t\thookedFactories[lookupFactory] = true;\r\n\t\tlogger.debug('Factory hooked: ' + lookupFactory);\r\n\t\treturn true;\r\n\t}\r\n\tcatch (err) {\r\n\t\tlogger.debug('Factory hook failure, unexpected error: ' + lookupFactory);\r\n\t\tlogger.debug(err);\r\n\t\treturn false;\r\n\t}\r\n};\r\n\r\napi.get = function (lookupFactory, property) {\r\n\tif (!api.isLoaded()) {\r\n\t\tlogger.debug('Factory get failure, Ember not loaded.');\r\n\t\treturn false;\r\n\t}\r\n\tvar properties = property.split('.');\r\n\tvar getter = api.lookup(lookupFactory);\r\n\r\n\tproperties.some(function (property) {\r\n\t\t// If getter fails, just exit, otherwise, keep looping.\r\n\t\tif (getter == null || typeof getter === 'undefined') {\r\n\t\t\tgetter = null;\r\n\t\t\treturn true;\r\n\t\t}\r\n\t\tif (getter[property] == null || typeof getter[property] === 'undefined') {\r\n\t\t\tgetter = null;\r\n\t\t\treturn true;\r\n\t\t}\r\n\t\tif (typeof getter.get === 'function') {\r\n\t\t\tgetter = getter.get(property);\r\n\t\t\tif (getter == null || typeof getter === 'undefined') {\r\n\t\t\t\tgetter = null;\r\n\t\t\t\treturn true;\r\n\t\t\t}\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tgetter = getter[property];\r\n\t});\r\n\r\n\treturn getter;\r\n};\r\n\r\nmodule.exports = api;\r\n","var storage = require('./storage');\r\nvar logger = require('./logger');\r\nvar ui = require('./ui');\r\nvar api = {};\r\nvar emoteStore = new EmoteStore();\r\nvar $ = window.jQuery;\r\n\r\n/**\r\n * The entire emote storing system.\r\n */\r\nfunction EmoteStore() {\r\n\tvar getters = {};\r\n\tvar nativeEmotes = {};\r\n\tvar hasInitialized = false;\r\n\r\n\t/**\r\n\t * Get a list of usable emoticons.\r\n\t * @param  {function} [filters]       A filter method to limit what emotes are returned. Passed to Array#filter.\r\n\t * @param  {function|string} [sortBy] How the emotes should be sorted. `function` will be passed to sort via Array#sort. `'channel'` sorts by channel name, globals first. All other values (or omitted) sort alphanumerically.\r\n\t * @param  {string} [returnType]      `'object'` will return in object format, e.g. `{'Kappa': Emote(...), ...}`. All other values (or omitted) return an array format, e.g. `[Emote(...), ...]`.\r\n\t * @return {object|array}             See `returnType` param.\r\n\t */\r\n\tthis.getEmotes = function (filters, sortBy, returnType) {\r\n\t\tvar twitchApi = require('./twitch-api');\r\n\r\n\t\t// Get native emotes.\r\n\t\tvar emotes = $.extend({}, nativeEmotes);\r\n\r\n\t\t// Parse the custom emotes provided by third party addons.\r\n\t\tObject.keys(getters).forEach(function (getterName) {\r\n\t\t\t// Try the getter.\r\n\t\t\tvar results = null;\r\n\t\t\ttry {\r\n\t\t\t\tresults = getters[getterName]();\r\n\t\t\t}\r\n\t\t\tcatch (err) {\r\n\t\t\t\tlogger.debug('Emote getter `' + getterName + '` failed unexpectedly, skipping.', err.toString());\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tif (!Array.isArray(results)) {\r\n\t\t\t\tlogger.debug('Emote getter `' + getterName + '` must return an array, skipping.');\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\t// Override natives and previous getters.\r\n\t\t\tresults.forEach(function (emote) {\r\n\t\t\t\ttry {\r\n\t\t\t\t\t// Create the emote.\r\n\t\t\t\t\tvar instance = new Emote(emote);\r\n\r\n\t\t\t\t\t// Force the getter.\r\n\t\t\t\t\tinstance.setGetterName(getterName);\r\n\r\n\t\t\t\t\t// Force emotes without channels to the getter's name.\r\n\t\t\t\t\tif (!emote.channel) {\r\n\t\t\t\t\t\tinstance.setChannelName(getterName);\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t\t// Add/override it.\r\n\t\t\t\t\temotes[instance.getText()] = instance;\r\n\t\t\t\t}\r\n\t\t\t\tcatch (err) {\r\n\t\t\t\t\tlogger.debug('Emote parsing for getter `' + getterName + '` failed, skipping.', err.toString(), emote);\r\n\t\t\t\t}\r\n\t\t\t});\r\n\t\t});\r\n\r\n\t\t// Covert to array.\r\n\t\temotes = Object.keys(emotes).map(function (emote) {\r\n\t\t\treturn emotes[emote];\r\n\t\t});\r\n\r\n\t\t// Filter results.\r\n\t\tif (typeof filters === 'function') {\r\n\t\t\temotes = emotes.filter(filters);\r\n\t\t}\r\n\t\t\r\n\t\t// Return as an object if requested.\r\n\t\tif (returnType === 'object') {\r\n\t\t\tvar asObject = {};\r\n\t\t\temotes.forEach(function (emote) {\r\n\t\t\t\tasObject[emote.getText()] = emote;\r\n\t\t\t});\r\n\t\t\treturn asObject;\r\n\t\t}\r\n\r\n\t\t// Sort results.\r\n\t\tif (typeof sortBy === 'function') {\r\n\t\t\temotes.sort(sortBy);\r\n\t\t}\r\n\t\telse if (sortBy === 'channel') {\r\n\t\t\temotes.sort(sorting.allEmotesCategory);\r\n\t\t}\r\n\t\telse {\r\n\t\t\temotes.sort(sorting.byText);\r\n\t\t}\r\n\r\n\t\t// Return the emotes in array format.\r\n\t\treturn emotes;\r\n\t};\r\n\r\n\t/**\r\n\t * Registers a 3rd party emote hook.\r\n\t * @param  {string} name   The name of the 3rd party registering the hook.\r\n\t * @param  {function} getter The function called when looking for emotes. Must return an array of emote objects, e.g. `[emote, ...]`. See Emote class.\r\n\t */\r\n\tthis.registerGetter = function (name, getter) {\r\n\t\tif (typeof name !== 'string') {\r\n\t\t\tthrow new Error('Name must be a string.');\r\n\t\t}\r\n\t\tif (getters[name]) {\r\n\t\t\tthrow new Error('Getter already exists.');\r\n\t\t}\r\n\t\tif (typeof getter !== 'function') {\r\n\t\t\tthrow new Error('Getter must be a function.');\r\n\t\t}\r\n\t\tlogger.debug('Getter registered: ' + name);\r\n\t\tgetters[name] = getter;\r\n\t\tui.updateEmotes();\r\n\t};\r\n\r\n\t/**\r\n\t * Registers a 3rd party emote hook.\r\n\t * @param  {string} name   The name of the 3rd party hook to deregister.\r\n\t */\r\n\tthis.deregisterGetter = function (name) {\r\n\t\tlogger.debug('Getter unregistered: ' + name);\r\n\t\tdelete getters[name];\r\n\t\tui.updateEmotes();\r\n\t};\r\n\r\n\t/**\r\n\t * Initializes the raw data from the API endpoints. Should be called at load and/or whenever the API may have changed. Populates internal objects with updated data.\r\n\t */\r\n\tthis.init = function () {\r\n\t\tif (hasInitialized) {\r\n\t\t\tlogger.debug('Already initialized EmoteStore, stopping init.');\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tlogger.debug('Starting initialization.');\r\n\r\n\t\tvar twitchApi = require('./twitch-api');\r\n\t\tvar self = this;\r\n\r\n\t\t// Hash of emote set to forced channel.\r\n\t\tvar forcedSetsToChannels = {\r\n\t\t\t// Globals.\r\n\t\t\t0: '~global',\r\n\t\t\t// Bubble emotes.\r\n\t\t\t33: 'turbo',\r\n\t\t\t// Monkey emotes.\r\n\t\t\t42: 'turbo',\r\n\t\t\t// Hidden turbo emotes.\r\n\t\t\t457: 'turbo',\r\n\t\t\t793: 'turbo',\r\n\t\t\t19151: 'twitch_prime',\r\n\t\t\t19194: 'twitch_prime'\r\n\r\n\t\t};\r\n\r\n\t\tlogger.debug('Initializing emote set change listener.');\r\n\r\n\t\ttwitchApi.getEmotes(function (emoteSets) {\r\n\t\t\tlogger.debug('Parsing emote sets.');\r\n\r\n\t\t\tObject.keys(emoteSets).forEach(function (set) {\r\n\t\t\t\tvar emotes = emoteSets[set];\r\n\t\t\t\tset = Number(set);\r\n\t\t\t\temotes.forEach(function (emote) {\r\n\t\t\t\t\t// Set some required info.\r\n\t\t\t\t\temote.url = '//static-cdn.jtvnw.net/emoticons/v1/' + emote.id + '/1.0';\r\n\t\t\t\t\temote.text = getEmoteFromRegEx(emote.code);\r\n\t\t\t\t\temote.set = set;\r\n\r\n\t\t\t\t\t// Hardcode the channels of certain sets.\r\n\t\t\t\t\tif (forcedSetsToChannels[set]) {\r\n\t\t\t\t\t\temote.channel = forcedSetsToChannels[set];\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t\tvar instance = new Emote(emote);\r\n\r\n\t\t\t\t\t// Save the emote for use later.\r\n\t\t\t\t\tnativeEmotes[emote.text] = instance;\r\n\t\t\t\t});\r\n\t\t\t});\r\n\r\n\t\t\tlogger.debug('Loading subscription data.');\r\n\r\n\t\t\t// Get active subscriptions to find the channels.\r\n\t\t\ttwitchApi.getTickets(function (tickets) {\r\n\t\t\t\t// Instances from each channel to preload channel data.\r\n\t\t\t\tvar deferredChannelGets = {};\r\n\r\n\t\t\t\tlogger.debug('Tickets loaded from the API.', tickets);\r\n\t\t\t\ttickets.forEach(function (ticket) {\r\n\t\t\t\t\tvar product = ticket.product;\r\n\t\t\t\t\tvar channel = product.owner_name || product.short_name;\r\n\r\n\t\t\t\t\t// Get subscriptions with emotes only.\r\n\t\t\t\t\tif (!product.emoticons || !product.emoticons.length) {\r\n\t\t\t\t\t\treturn;\r\n\t\t\t\t\t}\r\n\t\t\t\t\t\r\n\t\t\t\t\t// Set the channel on the emotes.\r\n\t\t\t\t\tproduct.emoticons.forEach(function (emote) {\r\n\t\t\t\t\t\tvar instance = nativeEmotes[getEmoteFromRegEx(emote.regex)];\r\n\t\t\t\t\t\tinstance.setChannelName(channel);\r\n\r\n\t\t\t\t\t\t// Save instance for later, but only one instance per channel.\r\n\t\t\t\t\t\tif (!deferredChannelGets[channel]) {\r\n\t\t\t\t\t\t\tdeferredChannelGets[channel] = instance;\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t});\r\n\t\t\t\t});\r\n\r\n\t\t\t\t// Preload channel data.\r\n\t\t\t\tObject.keys(deferredChannelGets).forEach(function (key) {\r\n\t\t\t\t\tvar instance = deferredChannelGets[key];\r\n\t\t\t\t\tinstance.getChannelBadge();\r\n\t\t\t\t\tinstance.getChannelDisplayName();\r\n\t\t\t\t});\r\n\t\t\t\tui.updateEmotes();\r\n\t\t\t});\r\n\t\t\tui.updateEmotes();\r\n\t\t});\r\n\r\n\t\thasInitialized = true;\r\n\t\tlogger.debug('Finished EmoteStore initialization.');\r\n\t};\r\n};\r\n\r\n/**\r\n * Gets a specific emote, if available.\r\n * @param  {string}     text The text of the emote to get.\r\n * @return {Emote|null}      The Emote instance of the emote or `null` if it couldn't be found.\r\n */\r\nEmoteStore.prototype.getEmote = function (text) {\r\n\treturn this.getEmotes(null, null, 'object')[text] || null;\r\n};\r\n\r\n/**\r\n * Emote object.\r\n * @param {object} details              Object describing the emote.\r\n * @param {string} details.text         The text to use in the chat box when emote is clicked.\r\n * @param {string} details.url          The URL of the image for the emote.\r\n * @param {string} [details.badge]      The URL of the badge for the emote.\r\n * @param {string} [details.channel]    The channel the emote should be categorized under.\r\n * @param {string} [details.getterName] The 3rd party getter that registered the emote. Used internally only.\r\n */\r\nfunction Emote(details) {\r\n\tvar text = null;\r\n\tvar url = null;\r\n\tvar getterName = null;\r\n\tvar channel = {\r\n\t\tname: null,\r\n\t\tdisplayName: null,\r\n\t\tbadge: null\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the text of the emote.\r\n\t * @return {string} The emote text.\r\n\t */\r\n\tthis.getText = function () {\r\n\t\treturn text;\r\n\t};\r\n\r\n\t/**\r\n\t * Sets the text of the emote.\r\n\t * @param {string} theText The text to set.\r\n\t */\r\n\tthis.setText = function (theText) {\r\n\t\tif (typeof theText !== 'string' || theText.length < 1) {\r\n\t\t\tthrow new Error('Invalid text');\r\n\t\t}\r\n\t\ttext = theText;\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the getter name this emote belongs to.\r\n\t * @return {string} The getter's name.\r\n\t */\r\n\tthis.getGetterName = function () {\r\n\t\treturn getterName;\r\n\t};\r\n\r\n\t/**\r\n\t * Sets the getter name this emote belongs to.\r\n\t * @param {string} theGetterName The getter's name.\r\n\t */\r\n\tthis.setGetterName = function (theGetterName) {\r\n\t\tif (typeof theGetterName !== 'string' || theGetterName.length < 1) {\r\n\t\t\tthrow new Error('Invalid getter name');\r\n\t\t}\r\n\t\tgetterName = theGetterName;\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the emote's image URL.\r\n\t * @return {string} The emote image URL.\r\n\t */\r\n\tthis.getUrl = function () {\r\n\t\treturn url;\r\n\t};\r\n\t/**\r\n\t * Sets the emote's image URL.\r\n\t * @param {string} theUrl The image URL to set.\r\n\t */\r\n\tthis.setUrl = function (theUrl) {\r\n\t\tif (typeof theUrl !== 'string' || theUrl.length < 1) {\r\n\t\t\tthrow new Error('Invalid URL');\r\n\t\t}\r\n\t\turl = theUrl;\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the emote's channel name.\r\n\t * @return {string} The emote's channel or an empty string if it doesn't have one.\r\n\t */\r\n\tthis.getChannelName = function () {\r\n\t\tif (!channel.name) {\r\n\t\t\tchannel.name = storage.channelNames.get(this.getText());\r\n\t\t}\r\n\t\treturn channel.name || '';\r\n\t};\r\n\t/**\r\n\t * Sets the emote's channel name.\r\n\t * @param {string} theChannel The channel name to set.\r\n\t */\r\n\tthis.setChannelName = function (theChannel) {\r\n\t\tif (typeof theChannel !== 'string' || theChannel.length < 1) {\r\n\t\t\tthrow new Error('Invalid channel');\r\n\t\t}\r\n\r\n\t\t// Only save the channel to storage if it's dynamic.\r\n\t\tif (theChannel !== '~global' && theChannel !== 'turbo' && theChannel !== 'twitch_prime') {\r\n\t\t\tstorage.channelNames.set(this.getText(), theChannel);\r\n\t\t}\r\n\t\tchannel.name = theChannel;\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the emote channel's badge image URL.\r\n\t * @return {string|null} The URL of the badge image for the emote's channel or `null` if it doesn't have a channel.\r\n\t */\r\n\tthis.getChannelBadge = function () {\r\n\t\tvar twitchApi = require('./twitch-api');\r\n\t\tvar channelName = this.getChannelName();\r\n\t\tvar defaultBadge = '//static-cdn.jtvnw.net/jtv_user_pictures/subscriber-star.png';\r\n\r\n\t\t// No channel.\r\n\t\tif (!channelName) {\r\n\t\t\treturn null;\r\n\t\t}\r\n\r\n\t\t// Give globals a default badge.\r\n\t\tif (channelName === '~global') {\r\n\t\t\treturn '/favicon.ico';\r\n\t\t}\r\n\r\n\t\t// Already have one preset.\r\n\t\tif (channel.badge) {\r\n\t\t\treturn channel.badge;\r\n\t\t}\r\n\r\n\t\t// Check storage.\r\n\t\tchannel.badge = storage.badges.get(channelName);\r\n\t\tif (channel.badge !== null) {\r\n\t\t\treturn channel.badge;\r\n\t\t}\r\n\r\n\t\t// Set default until API returns something.\r\n\t\tchannel.badge = defaultBadge;\r\n\r\n\t\t// Get from API.\r\n\t\tlogger.debug('Getting fresh badge for: ' + channelName);\r\n\t\ttwitchApi.getBadges(channelName, function (badges) {\r\n\t\t\tvar badge = null;\r\n\r\n\t\t\t// Save turbo badge while we are here.\r\n\t\t\tif (badges.turbo && badges.turbo.image) {\r\n\t\t\t\tbadge = badges.turbo.image;\r\n\t\t\t\tstorage.badges.set('turbo', badge, 86400000);\r\n\r\n\t\t\t\t// Turbo is actually what we wanted, so we are done.\r\n\t\t\t\tif (channelName === 'turbo') {\r\n\t\t\t\t\tchannel.badge = badge;\r\n\t\t\t\t\treturn;\r\n\t\t\t\t}\r\n\t\t\t}\r\n\r\n\t\t\t// Save turbo badge while we are here.\r\n\t\t\tif (badges.premium && badges.premium.image) {\r\n\t\t\t\tbadge = badges.premium.image;\r\n\t\t\t\tstorage.badges.set('twitch_prime', badge, 86400000);\r\n\r\n\t\t\t\t// Turbo is actually what we wanted, so we are done.\r\n\t\t\t\tif (channelName === 'twitch_prime') {\r\n\t\t\t\t\tchannel.badge = badge;\r\n\t\t\t\t\treturn;\r\n\t\t\t\t}\r\n\t\t\t}\r\n\r\n\t\t\t// Save subscriber badge in storage.\r\n\t\t\tif (badges.subscriber && badges.subscriber.image) {\r\n\t\t\t\tchannel.badge = badges.subscriber.image;\r\n\t\t\t\tstorage.badges.set(channelName, channel.badge, 86400000);\r\n\t\t\t\tui.updateEmotes();\r\n\t\t\t}\r\n\t\t\t// No subscriber badge.\r\n\t\t\telse {\r\n\t\t\t\tchannel.badge = defaultBadge;\r\n\t\t\t\tlogger.debug('Failed to get subscriber badge for: ' + channelName);\r\n\t\t\t}\r\n\t\t});\r\n\t\t\r\n\t\treturn channel.badge || defaultBadge;\r\n\t};\r\n\r\n\t/**\r\n\t * Sets the emote's channel badge image URL.\r\n\t * @param {string} theBadge The badge image URL to set.\r\n\t */\r\n\tthis.setChannelBadge = function (theBadge) {\r\n\t\tif (typeof theBadge !== 'string' || theBadge.length < 1) {\r\n\t\t\tthrow new Error('Invalid badge');\r\n\t\t}\r\n\t\tchannel.badge = theBadge;\r\n\t};\r\n\r\n\t/**\r\n\t * Get a channel's display name.\r\n\t * @return {string} The channel's display name. May be equivalent to the channel the first time the API needs to be called.\r\n\t */\r\n\tthis.getChannelDisplayName = function () {\r\n\t\tvar twitchApi = require('./twitch-api');\r\n\t\tvar channelName = this.getChannelName();\r\n\t\tvar self = this;\r\n\r\n\t\tvar forcedChannelToDisplayNames = {\r\n\t\t\t'~global': 'Global',\r\n\t\t\t'turbo': 'Twitch Turbo',\r\n\t\t\t'twitch_prime': 'Twitch Prime'\r\n\t\t};\r\n\r\n\t\t// No channel.\r\n\t\tif (!channelName) {\r\n\t\t\treturn '';\r\n\t\t}\r\n\r\n\t\t// Forced display name.\r\n\t\tif (forcedChannelToDisplayNames[channelName]) {\r\n\t\t\treturn forcedChannelToDisplayNames[channelName];\r\n\t\t}\r\n\r\n\t\t// Already have one preset.\r\n\t\tif (channel.displayName) {\r\n\t\t\treturn channel.displayName;\r\n\t\t}\r\n\r\n\t\t// Look for obvious bad channel names that shouldn't hit the API or storage. Use channel name instead.\r\n\t\tif (/[^a-z0-9_]/.test(channelName)) {\r\n\t\t\tlogger.debug('Unable to get display name due to obvious non-standard channel name for: ' + channelName);\r\n\t\t\treturn channelName;\r\n\t\t}\r\n\r\n\t\t// Check storage.\r\n\t\tchannel.displayName = storage.displayNames.get(channelName);\r\n\t\tif (channel.displayName !== null) {\r\n\t\t\treturn channel.displayName;\r\n\t\t}\r\n\t\t// Get from API.\r\n\t\telse {\r\n\t\t\t// Set default until API returns something.\r\n\t\t\tchannel.displayName = channelName;\r\n\r\n\t\t\tlogger.debug('Getting fresh display name for: ' + channelName);\r\n\t\t\ttwitchApi.getUser(channelName, function (user) {\r\n\t\t\t\tif (!user || !user.display_name) {\r\n\t\t\t\t\tlogger.debug('Failed to get display name for: ' + channelName);\r\n\t\t\t\t\treturn;\r\n\t\t\t\t}\r\n\r\n\t\t\t\t// Save it.\r\n\t\t\t\tself.setChannelDisplayName(user.display_name);\r\n\t\t\t\tui.updateEmotes();\r\n\t\t\t});\r\n\t\t}\r\n\t\t\r\n\t\treturn channel.displayName;\r\n\t};\r\n\r\n\t/**\r\n\t * Sets the emote's channel badge image URL.\r\n\t * @param {string} theBadge The badge image URL to set.\r\n\t */\r\n\tthis.setChannelDisplayName = function (displayName) {\r\n\t\tif (typeof displayName !== 'string' || displayName.length < 1) {\r\n\t\t\tthrow new Error('Invalid displayName');\r\n\t\t}\r\n\t\tchannel.displayName = displayName;\r\n\t\tstorage.displayNames.set(this.getChannelName(), displayName, 86400000);\r\n\t};\r\n\r\n\t/**\r\n\t * Initialize the details.\r\n\t */\r\n\t\r\n\t// Required fields.\r\n\tthis.setText(details.text);\r\n\tthis.setUrl(details.url);\r\n\r\n\t// Optional fields.\r\n\tif (details.getterName) {\r\n\t\tthis.setGetterName(details.getterName);\r\n\t}\r\n\tif (details.channel) {\r\n\t\tthis.setChannelName(details.channel);\r\n\t}\r\n\tif (details.channelDisplayName) {\r\n\t\tthis.setChannelDisplayName(details.channelDisplayName);\r\n\t}\r\n\tif (details.badge) {\r\n\t\tthis.setChannelBadge(details.badge);\r\n\t}\r\n};\r\n\r\n/**\r\n * State changers.\r\n */\r\n\r\n/**\r\n * Toggle whether an emote should be a favorite.\r\n * @param {boolean} [force] `true` forces the emote to be a favorite, `false` forces the emote to not be a favorite.\r\n */\r\nEmote.prototype.toggleFavorite = function (force) {\r\n\tif (typeof force !== 'undefined') {\r\n\t\tstorage.starred.set(this.getText(), !!force);\r\n\t\treturn;\r\n\t}\r\n\tstorage.starred.set(this.getText(), !this.isFavorite());\r\n};\r\n\r\n/**\r\n * Toggle whether an emote should be visible out of editing mode.\r\n * @param {boolean} [force] `true` forces the emote to be visible, `false` forces the emote to be hidden.\r\n */\r\nEmote.prototype.toggleVisibility = function (force) {\r\n\tif (typeof force !== 'undefined') {\r\n\t\tstorage.visibility.set(this.getText(), !!force);\r\n\t\treturn;\r\n\t}\r\n\tstorage.visibility.set(this.getText(), !this.isVisible());\r\n};\r\n\r\n/**\r\n * State getters.\r\n */\r\n\r\n/**\r\n * Whether the emote is from a 3rd party.\r\n * @return {boolean} Whether the emote is from a 3rd party.\r\n */\r\nEmote.prototype.isThirdParty = function () {\r\n\treturn !!this.getGetterName();\r\n};\r\n\r\n/**\r\n * Whether the emote was favorited.\r\n * @return {boolean} Whether the emote was favorited.\r\n */\r\nEmote.prototype.isFavorite = function () {\r\n\treturn storage.starred.get(this.getText(), false);\r\n};\r\n\r\n/**\r\n * Whether the emote is visible outside of editing mode.\r\n * @return {boolean} Whether the emote is visible outside of editing mode.\r\n */\r\nEmote.prototype.isVisible = function () {\r\n\treturn storage.visibility.get(this.getText(), true);\r\n};\r\n\r\n/**\r\n * Whether the emote is considered a simple smiley.\r\n * @return {boolean} Whether the emote is considered a simple smiley.\r\n */\r\nEmote.prototype.isSmiley = function () {\r\n\t// The basic smiley emotes.\r\n\tvar emotes = [':(', ':)', ':/', ':\\\\', ':D', ':o', ':p', ':z', ';)', ';p', '<3', '>(', 'B)', 'R)', 'o_o', 'O_O', '#/', ':7', ':>', ':S', '<]'];\r\n\treturn emotes.indexOf(this.getText()) !== -1;\r\n};\r\n\r\n/**\r\n * Property getters/setters.\r\n */\r\n\r\n/**\r\n * Gets the usable emote text from a regex.\r\n */\r\nfunction getEmoteFromRegEx(regex) {\r\n\tif (typeof regex === 'string') {\r\n\t\tregex = new RegExp(regex);\r\n\t}\r\n\tif (!regex) {\r\n\t\tthrow new Error('`regex` must be a RegExp string or object.');\r\n\t}\r\n\r\n\treturn decodeURI(regex.source)\r\n\r\n\t\t// Replace HTML entity brackets with actual brackets.\r\n\t\t.replace('&gt\\\\;', '>')\r\n\t\t.replace('&lt\\\\;', '<')\r\n\r\n\t\t// Remove negative groups.\r\n\t\t//\r\n\t\t// /\r\n\t\t//   \\(\\?!              // (?!\r\n\t\t//   [^)]*              // any amount of characters that are not )\r\n\t\t//   \\)                 // )\r\n\t\t// /g\r\n\t\t.replace(/\\(\\?![^)]*\\)/g, '')\r\n\r\n\t\t// Pick first option from a group.\r\n\t\t//\r\n\t\t// /\r\n\t\t//   \\(                 // (\r\n\t\t//   ([^|])*            // any amount of characters that are not |\r\n\t\t//   \\|?                // an optional | character\r\n\t\t//   [^)]*              // any amount of characters that are not )\r\n\t\t//   \\)                 // )\r\n\t\t// /g\r\n\t\t.replace(/\\(([^|])*\\|?[^)]*\\)/g, '$1')\r\n\r\n\t\t// Pick first character from a character group.\r\n\t\t//\r\n\t\t// /\r\n\t\t//   \\[                 // [\r\n\t\t//   ([^|\\]\\[])*        // any amount of characters that are not |, [, or ]\r\n\t\t//   \\|?                // an optional | character\r\n\t\t//   [^\\]]*             // any amount of characters that are not [, or ]\r\n\t\t//   \\]                 // ]\r\n\t\t// /g\r\n\t\t.replace(/\\[([^|\\]\\[])*\\|?[^\\]\\[]*\\]/g, '$1')\r\n\r\n\t\t// Remove optional characters.\r\n\t\t//\r\n\t\t// /\r\n\t\t//   [^\\\\]              // any character that is not \\\r\n\t\t//   \\?                 // ?\r\n\t\t// /g\r\n\t\t.replace(/[^\\\\]\\?/g, '')\r\n\r\n\t\t// Remove boundaries at beginning and end.\r\n\t\t.replace(/^\\\\b|\\\\b$/g, '') \r\n\r\n\t\t// Unescape only single backslash, not multiple.\r\n\t\t//\r\n\t\t// /\r\n\t\t//   \\\\                 // \\\r\n\t\t//   (?!\\\\)             // look-ahead, any character that isn't \\\r\n\t\t// /g\r\n\t\t.replace(/\\\\(?!\\\\)/g, '');\r\n}\r\n\r\nvar sorting = {};\r\n\r\n/**\r\n * Sort by alphanumeric in this order: symbols -> numbers -> AaBb... -> numbers\r\n */\r\nsorting.byText = function (a, b) {\r\n\ttextA = a.getText().toLowerCase();\r\n\ttextB = b.getText().toLowerCase();\r\n\r\n\tif (textA < textB) {\r\n\t\treturn -1;\r\n\t}\r\n\tif (textA > textB) {\r\n\t\treturn 1;\r\n\t}\r\n\treturn 0;\r\n}\r\n\r\n/**\r\n * Basic smilies before non-basic smilies.\r\n */\r\nsorting.bySmiley = function (a, b) {\r\n\tif (a.isSmiley() &&\t!b.isSmiley()) {\r\n\t\treturn -1;\r\n\t}\r\n\tif (b.isSmiley() &&\t!a.isSmiley()) {\r\n\t\treturn 1;\r\n\t}\r\n\treturn 0;\r\n};\r\n\r\n/**\r\n * Globals before subscription emotes, subscriptions in alphabetical order.\r\n */\r\nsorting.byChannelName = function (a, b) {\r\n\tvar channelA = a.getChannelName();\r\n\tvar channelB = b.getChannelName();\r\n\r\n\t// Both don't have channels.\r\n\tif (!channelA && !channelB) {\r\n\t\treturn 0;\r\n\t}\r\n\r\n\t// \"A\" has channel, \"B\" doesn't.\r\n\tif (channelA && !channelB) {\r\n\t\treturn 1;\r\n\t}\r\n\t// \"B\" has channel, \"A\" doesn't.\r\n\tif (channelB && !channelA) {\r\n\t\treturn -1;\r\n\t}\r\n\r\n\tchannelA = channelA.toLowerCase();\r\n\tchannelB = channelB.toLowerCase();\r\n\r\n\tif (channelA < channelB) {\r\n\t\treturn -1;\r\n\t}\r\n\tif (channelB > channelA) {\r\n\t\treturn 1;\r\n\t}\r\n\r\n\t// All the same\r\n\treturn 0;\r\n};\r\n\r\n/**\r\n * The general sort order for the all emotes category.\r\n * Smileys -> Channel grouping -> alphanumeric\r\n */\r\nsorting.allEmotesCategory = function (a, b) {\r\n\tvar bySmiley = sorting.bySmiley(a, b);\r\n\tvar byChannelName  = sorting.byChannelName(a, b);\r\n\tvar byText = sorting.byText(a, b);\r\n\r\n\tif (bySmiley !== 0) {\r\n\t\treturn bySmiley;\r\n\t}\r\n\tif (byChannelName !== 0) {\r\n\t\treturn byChannelName;\r\n\t}\r\n\treturn byText;\r\n};\r\n\r\nmodule.exports = emoteStore;\r\n","var api = {};\r\nvar instance = '[instance ' + (Math.floor(Math.random() * (999 - 100)) + 100) + '] ';\r\nvar prefix = '[Emote Menu] ';\r\nvar storage = require('./storage');\r\n\r\napi.log = function () {\r\n\tif (typeof console.log === 'undefined') {\r\n\t\treturn;\r\n\t}\r\n\targuments = [].slice.call(arguments).map(function (arg) {\r\n\t\tif (typeof arg !== 'string') {\r\n\t\t\treturn JSON.stringify(arg);\r\n\t\t}\r\n\t\treturn arg;\r\n\t});\r\n\tif (storage.global.get('debugMessagesEnabled', false)) {\r\n\t\targuments.unshift(instance);\r\n\t}\r\n\targuments.unshift(prefix);\r\n\tconsole.log.apply(console, arguments);\r\n};\r\n\r\napi.debug = function () {\r\n\tif (!storage.global.get('debugMessagesEnabled', false)) {\r\n\t\treturn;\r\n\t}\r\n\targuments = [].slice.call(arguments);\r\n\targuments.unshift('[DEBUG] ');\r\n\tapi.log.apply(null, arguments);\r\n}\r\n\r\nmodule.exports = api;\r\n","var storage = require('./storage');\r\nvar logger = require('./logger');\r\nvar emotes = require('./emotes');\r\nvar api = {};\r\n\r\napi.toggleDebug = function (forced) {\r\n\tif (typeof forced === 'undefined') {\r\n\t\tforced = !storage.global.get('debugMessagesEnabled', false);\r\n\t}\r\n\telse {\r\n\t\tforced = !!forced;\r\n\t}\r\n\tstorage.global.set('debugMessagesEnabled', forced);\r\n\tlogger.log('Debug messages are now ' + (forced ? 'enabled' : 'disabled'));\r\n};\r\n\r\napi.registerEmoteGetter = emotes.registerGetter;\r\napi.deregisterEmoteGetter = emotes.deregisterGetter;\r\n\r\napi.ember = require('./ember-api');\r\napi.twitchAPI = require('./twitch-api');\r\n\r\nmodule.exports = api;\r\n","var Store = require('storage-wrapper');\r\nvar storage = {};\r\n\r\n// General storage.\r\nstorage.global = new Store({\r\n\tnamespace: 'emote-menu-for-twitch'\r\n});\r\n\r\n// Emote visibility storage.\r\nstorage.visibility = storage.global.createSubstore('visibility');\r\n// Emote starred storage.\r\nstorage.starred = storage.global.createSubstore('starred');\r\n// Display name storage.\r\nstorage.displayNames = storage.global.createSubstore('displayNames');\r\n// Channel name storage.\r\nstorage.channelNames = storage.global.createSubstore('channelNames');\r\n// Badges storage.\r\nstorage.badges = storage.global.createSubstore('badges');\r\n\r\nmodule.exports = storage;\r\n","var templates = require('../../build/templates');\r\n\r\nmodule.exports = (function () {\r\n\tvar data = {};\r\n\tvar key = null;\r\n\r\n\t// Convert templates to their shorter \"render\" form.\r\n\tfor (key in templates) {\r\n\t\tif (!templates.hasOwnProperty(key)) {\r\n\t\t\tcontinue;\r\n\t\t}\r\n\t\tdata[key] = render(key);\r\n\t}\r\n\r\n\t// Shortcut the render function. All templates will be passed in as partials by default.\r\n\tfunction render(template) {\r\n\t\ttemplate = templates[template];\r\n\t\treturn function (context, partials, indent) {\r\n\t\t\treturn template.render(context, partials || templates, indent);\r\n\t\t};\r\n\t}\r\n\r\n\treturn data;\r\n})();\r\n","var twitchApi = window.Twitch.api;\r\nvar jQuery = window.jQuery;\r\nvar logger = require('./logger');\r\nvar api = {};\r\n\r\napi.getBadges = function (username, callback) {\r\n\tif (\r\n\t\t[\r\n\t\t\t'~global',\r\n\t\t\t'turbo',\r\n\t\t\t'twitch_prime'\r\n\t\t].indexOf(username) > -1\r\n\t) {\r\n\t\tif (!jQuery) {\r\n\t\t\tcallback({});\r\n\t\t}\r\n\t\t// Note: not a documented API endpoint.\r\n\t\tjQuery.getJSON('https://badges.twitch.tv/v1/badges/global/display')\r\n\t\t\t.done(function (api) {\r\n\t\t\t\tvar badges = {\r\n\t\t\t\t\tturbo: {\r\n\t\t\t\t\t\timage: api.badge_sets.turbo.versions['1'].image_url_1x\r\n\t\t\t\t\t},\r\n\t\t\t\t\tpremium: {\r\n\t\t\t\t\t\timage: api.badge_sets.premium.versions['1'].image_url_1x\r\n\t\t\t\t\t}\r\n\t\t\t\t};\r\n\t\t\t\tcallback(badges);\r\n\t\t\t})\r\n\t\t\t.fail(function () {\r\n\t\t\t\tcallback({});\r\n\t\t\t});\r\n\t}\r\n\telse {\r\n\t\ttwitchApi.get('chat/' + username + '/badges')\r\n\t\t\t.done(function (api) {\r\n\t\t\t\tcallback(api);\r\n\t\t\t})\r\n\t\t\t.fail(function () {\r\n\t\t\t\tcallback({});\r\n\t\t\t});\r\n\t}\r\n};\r\n\r\napi.getUser = function (username, callback) {\r\n\t// Note: not a documented API endpoint.\r\n\ttwitchApi.get('users/' + username)\r\n\t\t.done(function (api) {\r\n\t\t\tcallback(api);\r\n\t\t})\r\n\t\t.fail(function () {\r\n\t\t\tcallback({});\r\n\t\t});\r\n};\r\n\r\napi.getTickets = function (callback) {\r\n\t// Note: not a documented API endpoint.\r\n\ttwitchApi.get(\r\n\t\t'/api/users/:login/tickets',\r\n\t\t{\r\n\t\t\toffset: 0,\r\n\t\t\tlimit: 100,\r\n\t\t\tunended: true\r\n\t\t}\r\n\t)\r\n\t\t.done(function (api) {\r\n\t\t\tcallback(api.tickets || []);\r\n\t\t})\r\n\t\t.fail(function () {\r\n\t\t\tcallback([]);\r\n\t\t});\r\n};\r\n\r\napi.getEmotes = function (callback) {\r\n\ttwitchApi.get('users/:login/emotes')\r\n\t\t.done(function (response) {\r\n\t\t\tif (!response || !response.emoticon_sets) {\r\n\t\t\t\tlogger.debug('getEmotes emoticon_sets empty');\r\n\t\t\t\tcallback({});\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\tcallback(response.emoticon_sets);\r\n\t\t})\r\n\t\t.fail(function () {\r\n\t\t\tlogger.debug('getEmotes API call failed');\r\n\t\t\tcallback({});\r\n\t\t});\r\n};\r\n\r\napi.raw = twitchApi;\r\n\r\nmodule.exports = api;\r\n","var api = {};\r\nvar $ = jQuery = window.jQuery;\r\nvar templates = require('./templates');\r\nvar storage = require('./storage');\r\nvar emotes = require('./emotes');\r\nvar logger = require('./logger');\r\n\r\nvar theMenu = new UIMenu();\r\nvar theMenuButton = new UIMenuButton();\r\n\r\napi.init = function () {\r\n\t// Load CSS.\r\n\trequire('../../build/styles');\r\n\r\n\t// Load jQuery plugins.\r\n\trequire('../plugins/resizable');\r\n\trequire('jquery.scrollbar');\r\n\r\n\ttheMenuButton.init();\r\n\ttheMenu.init();\r\n};\r\n\r\napi.hideMenu = function () {\r\n\tif (theMenu.dom && theMenu.dom.length) {\r\n\t\ttheMenu.toggleDisplay(false);\r\n\t}\r\n};\r\n\r\napi.updateEmotes = function () {\r\n\ttheMenu.updateEmotes();\r\n}\r\n\r\nfunction UIMenuButton() {\r\n\tthis.dom = null;\r\n}\r\n\r\nUIMenuButton.prototype.init = function (timesFailed) {\r\n\tvar self = this;\r\n\tvar chatButton = $('.send-chat-button, .chat-buttons-container button');\r\n\tvar failCounter = timesFailed || 0;\r\n\tthis.dom = $('#emote-menu-button');\r\n\r\n\t// Element already exists.\r\n\tif (this.dom.length) {\r\n\t\tlogger.debug('MenuButton already exists, stopping init.');\r\n\t\treturn this;\r\n\t}\r\n\r\n\tif (!chatButton.length) {\r\n\t\tfailCounter += 1;\r\n\t\tif (failCounter === 1) {\r\n\t\t\tlogger.log('MenuButton container missing, trying again.');\r\n\t\t}\r\n\t\tif (failCounter >= 10) {\r\n\t\t\tlogger.log('MenuButton container missing, MenuButton unable to be added, stopping init.');\r\n\t\t\treturn this;\r\n\t\t}\r\n\t\tsetTimeout(function () {\r\n\t\t\tself.init(failCounter);\r\n\t\t}, 1000);\r\n\t\treturn this;\r\n\t}\r\n\r\n\t// Create element.\r\n\tthis.dom = $(templates.emoteButton());\r\n\tthis.dom.insertBefore(chatButton);\r\n\r\n\t// Hide then fade it in.\r\n\tthis.dom.hide();\r\n\tthis.dom.fadeIn();\r\n\r\n\t// Enable clicking.\r\n\tthis.dom.on('click', function () {\r\n\t\ttheMenu.toggleDisplay();\r\n\t});\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenuButton.prototype.toggleDisplay = function (forced) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.isVisible();\r\n\tif (state) {\r\n\t\tthis.dom.addClass('active');\r\n\t\treturn this;\r\n\t}\r\n\tthis.dom.removeClass('active');\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenuButton.prototype.isVisible = function () {\r\n\treturn this.dom.hasClass('active');\r\n};\r\n\r\nfunction UIMenu() {\r\n\tthis.dom = null;\r\n\tthis.groups = {};\r\n\tthis.emotes = [];\r\n\tthis.offset = null;\r\n\tthis.favorites = null;\r\n}\r\n\r\nUIMenu.prototype.init = function () {\r\n\tvar logger = require('./logger');\r\n\tvar self = this;\r\n\r\n\tthis.dom = $('#emote-menu-for-twitch');\r\n\r\n\t// Element already exists.\r\n\tif (this.dom.length) {\r\n\t\treturn this;\r\n\t}\r\n\r\n\t// Create element.\r\n\tthis.dom = $(templates.menu());\r\n\t$(document.body).append(this.dom);\r\n\r\n\tthis.favorites = new UIFavoritesGroup();\r\n\r\n\t// Enable dragging.\r\n\tthis.dom.draggable({\r\n\t\thandle: '.draggable',\r\n\t\tstart: function () {\r\n\t\t\tself.togglePinned(true);\r\n\t\t\tself.toggleMovement(true);\r\n\t\t},\r\n\t\tstop: function () {\r\n\t\t\tself.offset = self.dom.offset();\r\n\t\t},\r\n\t\tcontainment: $(document.body)\r\n\t});\r\n\r\n\t// Enable resizing.\r\n\tthis.dom.resizable({\r\n\t\thandle: '[data-command=\"resize-handle\"]',\r\n\t\tstop: function () {\r\n\t\t\tself.togglePinned(true);\r\n\t\t\tself.toggleMovement(true);\r\n\t\t},\r\n\t\talsoResize: self.dom.find('.scrollable'),\r\n\t\tcontainment: $(document.body),\r\n\t\tminHeight: 180,\r\n\t\tminWidth: 200\r\n\t});\r\n\r\n\t// Enable pinning.\r\n\tthis.dom.find('[data-command=\"toggle-pinned\"]').on('click', function () {\r\n\t\tself.togglePinned();\r\n\t});\r\n\r\n\t// Enable editing.\r\n\tthis.dom.find('[data-command=\"toggle-editing\"]').on('click', function () {\r\n\t\tself.toggleEditing();\r\n\t});\r\n\r\n\tthis.dom.find('.scrollable').scrollbar()\r\n\r\n\tthis.updateEmotes();\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype._detectOutsideClick = function (event) {\r\n\t// Not outside of the menu, ignore the click.\r\n\tif ($(event.target).is('#emote-menu-for-twitch, #emote-menu-for-twitch *')) {\r\n\t\treturn;\r\n\t}\r\n\r\n\t// Clicked on the menu button, just remove the listener and let the normal listener handle it.\r\n\tif (!this.isVisible() || $(event.target).is('#emote-menu-button, #emote-menu-button *')) {\r\n\t\t$(document).off('mouseup', this._detectOutsideClick.bind(this));\r\n\t\treturn;\r\n\t}\r\n\r\n\t// Clicked outside, make sure the menu isn't pinned.\r\n\tif (!this.isPinned()) {\r\n\t\t// Menu wasn't pinned, remove listener.\r\n\t\t$(document).off('mouseup', this._detectOutsideClick.bind(this));\r\n\t\tthis.toggleDisplay();\r\n\t}\r\n};\r\n\r\nUIMenu.prototype.toggleDisplay = function (forced) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.isVisible();\r\n\tvar loggedIn = window.Twitch && window.Twitch.user.isLoggedIn();\r\n\r\n\t// Menu should be shown.\r\n\tif (state) {\r\n\t\t// Check if user is logged in.\r\n\t\tif (!loggedIn) {\r\n\t\t\t// Call native login form.\r\n\t\t\t$.login();\r\n\t\t\treturn this;\r\n\t\t}\r\n\r\n\t\tthis.updateEmotes();\r\n\t\tthis.dom.show();\r\n\r\n\t\t// Menu moved, move it back.\r\n\t\tif (this.hasMoved()) {\r\n\t\t\tthis.dom.offset(this.offset);\r\n\t\t}\r\n\t\t// Never moved, make it the same size as the chat window.\r\n\t\telse {\r\n\t\t\tvar chatContainer = $('.chat-messages');\r\n\t\t\t\r\n\t\t\t// Adjust the size to be the same as the chat container.\r\n\t\t\tthis.dom.height(chatContainer.outerHeight() - (this.dom.outerHeight() - this.dom.height()));\r\n\t\t\tthis.dom.width(chatContainer.outerWidth() - (this.dom.outerWidth() - this.dom.width()));\r\n\r\n\t\t\t// Adjust the offset to be the same as the chat container.\r\n\t\t\tthis.offset = chatContainer.offset();\r\n\t\t\tthis.dom.offset(this.offset);\r\n\t\t}\r\n\r\n\t\t// Listen for outside click.\r\n\t\t$(document).on('mouseup', this._detectOutsideClick.bind(this));\r\n\t}\r\n\t// Menu should be hidden.\r\n\telse {\r\n\t\tthis.dom.hide();\r\n\t\tthis.toggleEditing(false);\r\n\t\tthis.togglePinned(false);\r\n\t}\r\n\r\n\t// Also toggle the menu button.\r\n\ttheMenuButton.toggleDisplay(this.isVisible());\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.isVisible = function () {\r\n\treturn this.dom.is(':visible');\r\n};\r\n\r\nUIMenu.prototype.updateEmotes = function (which) {\r\n\tvar emote = which ? this.getEmote(which) : null;\r\n\tvar favoriteEmote = emote ? this.favorites.getEmote(which) : null;\r\n\tif (emote) {\r\n\t\temote.update();\r\n\t\tif (favoriteEmote) {\r\n\t\t\tfavoriteEmote.update();\r\n\t\t}\r\n\t\treturn this;\r\n\t}\r\n\tvar emotes = require('./emotes');\r\n\tvar theEmotes = emotes.getEmotes();\r\n\tvar theEmotesKeys = [];\r\n\tvar self = this;\r\n\r\n\ttheEmotes.forEach(function (emoteInstance) {\r\n\t\tself.addEmote(emoteInstance);\r\n\t\ttheEmotesKeys.push(emoteInstance.getText());\r\n\t});\r\n\r\n\t// Difference the emotes and remove all non-valid emotes.\r\n\tthis.emotes.forEach(function (oldEmote) {\r\n\t\tvar text = oldEmote.getText()\r\n\t\tif (theEmotesKeys.indexOf(text) < 0) {\r\n\t\t\tlogger.debug('Emote difference found, removing emote from UI: ' + text);\r\n\t\t\tself.removeEmote(text);\r\n\t\t}\r\n\t});\r\n\r\n\t// Save the emotes for next differencing.\r\n\tthis.emotes = theEmotes;\r\n\r\n\t//Update groups.\r\n\tObject.keys(this.groups).forEach(function (group) {\r\n\t\tself.getGroup(group).init();\r\n\t});\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.toggleEditing = function (forced) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.isEditing();\r\n\tthis.dom.toggleClass('editing', state);\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.isEditing = function () {\r\n\treturn this.dom.hasClass('editing');\r\n};\r\n\r\nUIMenu.prototype.togglePinned = function (forced) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.isPinned();\r\n\tthis.dom.toggleClass('pinned', state);\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.isPinned = function () {\r\n\treturn this.dom.hasClass('pinned');\r\n};\r\n\r\nUIMenu.prototype.toggleMovement = function (forced) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.hasMoved();\r\n\tthis.dom.toggleClass('moved', state);\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.hasMoved = function () {\r\n\treturn this.dom.hasClass('moved');\r\n};\r\n\r\nUIMenu.prototype.addGroup = function (emoteInstance) {\r\n\tvar channel = emoteInstance.getChannelName();\r\n\tvar self = this;\r\n\r\n\t// Already added, don't add again.\r\n\tif (this.getGroup(channel)) {\r\n\t\treturn this;\r\n\t}\r\n\r\n\t// Add to current menu groups.\r\n\tvar group = new UIGroup(emoteInstance);\r\n\tthis.groups[channel] = group;\r\n\r\n\t// Sort group names, get index of where this group should go.\r\n\tvar keys = Object.keys(this.groups);\r\n\tkeys.sort(function (a, b) {\r\n\t\t// Get the instances.\r\n\t\ta = self.groups[a].emoteInstance;\r\n\t\tb = self.groups[b].emoteInstance;\r\n\r\n\t\t// Get the channel name.\r\n\t\tvar aChannel = a.getChannelName();\r\n\t\tvar bChannel = b.getChannelName();\r\n\r\n\t\t// Get the channel display name.\r\n\t\ta = a.getChannelDisplayName().toLowerCase();\r\n\t\tb = b.getChannelDisplayName().toLowerCase();\r\n\r\n\t\t// Prime goes first, always.\r\n\t\tif (aChannel === 'twitch_prime' && bChannel !== 'twitch_prime') {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\tif (bChannel === 'twitch_prime' && aChannel !== 'twitch_prime') {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\r\n\t\t// Turbo goes after Prime, always.\r\n\t\tif (aChannel === 'turbo' && bChannel !== 'turbo') {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\tif (bChannel === 'turbo' && aChannel !== 'turbo') {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\r\n\t\t// Global goes after Turbo, always.\r\n\t\tif (aChannel === '~global' && bChannel !== '~global') {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\tif (bChannel === '~global' && aChannel !== '~global') {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\r\n\t\t// A goes first.\r\n\t\tif (a < b) {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\t// B goest first.\r\n\t\tif (a > b) {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\t\t// Both the same, doesn't matter.\r\n\t\treturn 0;\r\n\t});\r\n\r\n\tvar index = keys.indexOf(channel);\r\n\r\n\t// First in the sort, place at the beginning of the menu.\r\n\tif (index === 0) {\r\n\t\tgroup.dom.prependTo(this.dom.find('#all-emotes-group'));\r\n\t}\r\n\t// Insert after the previous group in the sort.\r\n\telse {\r\n\t\tgroup.dom.insertAfter(this.getGroup(keys[index - 1]).dom);\r\n\t}\r\n\r\n\treturn group;\r\n};\r\n\r\nUIMenu.prototype.getGroup = function (name) {\r\n\treturn this.groups[name] || null;\r\n};\r\n\r\nUIMenu.prototype.addEmote = function (emoteInstance) {\r\n\t// Get the group, or add if needed.\r\n\tvar group = this.getGroup(emoteInstance.getChannelName()) || this.addGroup(emoteInstance);\r\n\r\n\tgroup.addEmote(emoteInstance);\r\n\tgroup.toggleDisplay(group.isVisible(), true);\r\n\r\n\tthis.favorites.addEmote(emoteInstance);\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.removeEmote = function (name) {\r\n\tvar self = this;\r\n\tObject.keys(this.groups).forEach(function (groupName) {\r\n\t\tself.groups[groupName].removeEmote(name);\r\n\t});\r\n\tthis.favorites.removeEmote(name);\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.getEmote = function (name) {\r\n\tvar groupName = null;\r\n\tvar group = null;\r\n\tvar emote = null;\r\n\r\n\tfor (groupName in this.groups) {\r\n\t\tgroup = this.groups[groupName];\r\n\t\temote = group.getEmote(name);\r\n\r\n\t\tif (emote) {\r\n\t\t\treturn emote;\r\n\t\t}\r\n\t}\r\n\r\n\treturn null;\r\n};\r\n\r\nfunction UIGroup(emoteInstance) {\r\n\tthis.dom = null;\r\n\tthis.emotes = {};\r\n\tthis.emoteInstance = emoteInstance;\r\n\r\n\tthis.init();\r\n}\r\n\r\nUIGroup.prototype.init = function () {\r\n\tvar self = this;\r\n\tvar emoteInstance = this.emoteInstance;\r\n\r\n\t// First init, create new DOM.\r\n\tif (this.dom === null) {\r\n\t\tthis.dom = $(templates.emoteGroupHeader({\r\n\t\t\tbadge: emoteInstance.getChannelBadge(),\r\n\t\t\tchannel: emoteInstance.getChannelName(),\r\n\t\t\tchannelDisplayName: emoteInstance.getChannelDisplayName()\r\n\t\t}));\r\n\t}\r\n\t// Update DOM instead.\r\n\telse {\r\n\t\tthis.dom.find('.header-info').replaceWith(\r\n\t\t\t$(templates.emoteGroupHeader({\r\n\t\t\t\tbadge: emoteInstance.getChannelBadge(),\r\n\t\t\t\tchannel: emoteInstance.getChannelName(),\r\n\t\t\t\tchannelDisplayName: emoteInstance.getChannelDisplayName()\r\n\t\t\t}))\r\n\t\t\t.find('.header-info')\r\n\t\t);\r\n\t}\r\n\r\n\t// Enable emote hiding.\r\n\tthis.dom.find('.header-info [data-command=\"toggle-visibility\"]').on('click', function () {\r\n\t\tif (!theMenu.isEditing()) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tself.toggleDisplay();\r\n\t});\r\n\r\n\tthis.toggleDisplay(this.isVisible(), true);\r\n};\r\n\r\nUIGroup.prototype.toggleDisplay = function (forced, skipUpdatingEmoteDisplay) {\r\n\tvar self = this;\r\n\tvar state = typeof forced !== 'undefined' ? !forced : this.isVisible();\r\n\r\n\tthis.dom.toggleClass('emote-menu-hidden', state);\r\n\r\n\t// Update the display of all emotes.\r\n\tif (!skipUpdatingEmoteDisplay) {\r\n\t\tObject.keys(this.emotes).forEach(function (emoteName) {\r\n\t\t\tself.emotes[emoteName].toggleDisplay(!state);\r\n\t\t\ttheMenu.updateEmotes(self.emotes[emoteName].instance.getText());\r\n\t\t});\r\n\t}\r\n\r\n\treturn this;\r\n};\r\n\r\nUIGroup.prototype.isVisible = function () {\r\n\tvar self = this;\r\n\r\n\t// If any emote is visible, the group should be visible.\r\n\treturn Object.keys(this.emotes).some(function (emoteName) {\r\n\t\treturn self.emotes[emoteName].isVisible();\r\n\t});\r\n};\r\n\r\nUIGroup.prototype.addEmote = function (emoteInstance) {\r\n\tvar self = this;\r\n\tvar emote = this.getEmote(emoteInstance.getText());\r\n\r\n\t// Already added, update instead.\r\n\tif (emote) {\r\n\t\temote.update();\r\n\t\treturn this;\r\n\t}\r\n\r\n\t// Add to current emotes.\r\n\temote = new UIEmote(emoteInstance);\r\n\tthis.emotes[emoteInstance.getText()] = emote;\r\n\r\n\tvar keys = Object.keys(this.emotes);\r\n\r\n\tkeys.sort(function (a, b) {\r\n\t\t// Get the emote instances.\r\n\t\ta = self.emotes[a].instance;\r\n\t\tb = self.emotes[b].instance;\r\n\r\n\t\t// A is a smiley, B isn't. A goes first.\r\n\t\tif (a.isSmiley() &&\t!b.isSmiley()) {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\t// B is a smiley, A isn't. B goes first.\r\n\t\tif (b.isSmiley() &&\t!a.isSmiley()) {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\r\n\t\t// Get the text of the emotes.\r\n\t\ta = a.getText().toLowerCase();\r\n\t\tb = b.getText().toLowerCase();\r\n\r\n\t\t// A goes first.\r\n\t\tif (a < b) {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\t// B goest first.\r\n\t\tif (a > b) {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\t\t// Both the same, doesn't matter.\r\n\t\treturn 0;\r\n\t});\r\n\r\n\tvar index = keys.indexOf(emoteInstance.getText());\r\n\r\n\t// First in the sort, place at the beginning of the group.\r\n\tif (index === 0) {\r\n\t\temote.dom.prependTo(this.dom.find('.emote-container'));\r\n\t}\r\n\t// Insert after the previous emote in the sort.\r\n\telse {\r\n\t\temote.dom.insertAfter(this.getEmote(keys[index - 1]).dom);\r\n\t}\r\n\r\n\treturn this;\r\n};\r\n\r\nUIGroup.prototype.getEmote = function (name) {\r\n\treturn this.emotes[name] || null;\r\n};\r\n\r\nUIGroup.prototype.removeEmote = function (name) {\r\n\tvar emote = this.getEmote(name);\r\n\tif (!emote) {\r\n\t\treturn this;\r\n\t}\r\n\temote.dom.remove();\r\n\tdelete this.emotes[name];\r\n\r\n\treturn this;\r\n};\r\n\r\nfunction UIFavoritesGroup() {\r\n\tthis.dom = $('#starred-emotes-group');\r\n\tthis.emotes = {};\r\n}\r\n\r\nUIFavoritesGroup.prototype.addEmote = UIGroup.prototype.addEmote;\r\nUIFavoritesGroup.prototype.getEmote = UIGroup.prototype.getEmote;\r\nUIFavoritesGroup.prototype.removeEmote = UIGroup.prototype.removeEmote;\r\n\r\nfunction UIEmote(emoteInstance) {\r\n\tthis.dom = null;\r\n\tthis.instance = emoteInstance;\r\n\tthis.init();\r\n}\r\n\r\nUIEmote.prototype.init = function () {\r\n\tvar self = this;\r\n\r\n\t// Create element.\r\n\tthis.dom = $(templates.emote({\r\n\t\turl: this.instance.getUrl(),\r\n\t\ttext: this.instance.getText(),\r\n\t\tthirdParty: this.instance.isThirdParty(),\r\n\t\tisVisible: this.instance.isVisible(),\r\n\t\tisStarred: this.instance.isFavorite()\r\n\t}));\r\n\r\n\t// Enable clicking.\r\n\tthis.dom.on('click', function () {\r\n\t\tif (!theMenu.isEditing()) {\r\n\t\t\tself.addToChat();\r\n\r\n\t\t\t// Close the menu if not pinned.\r\n\t\t\tif (!theMenu.isPinned()) {\r\n\t\t\t\ttheMenu.toggleDisplay();\r\n\t\t\t}\r\n\t\t}\r\n\t});\r\n\r\n\t// Enable emote hiding.\r\n\tthis.dom.find('[data-command=\"toggle-visibility\"]').on('click', function () {\r\n\t\tif (!theMenu.isEditing()) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tself.toggleDisplay();\r\n\t\ttheMenu.updateEmotes(self.instance.getText());\r\n\t});\r\n\r\n\t// Enable emote favoriting.\r\n\tthis.dom.find('[data-command=\"toggle-starred\"]').on('click', function () {\r\n\t\tif (!theMenu.isEditing()) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tself.toggleFavorite();\r\n\t\ttheMenu.updateEmotes(self.instance.getText());\r\n\t});\r\n\r\n\treturn this;\r\n};\r\n\r\nUIEmote.prototype.toggleDisplay = function (forced, skipInstanceUpdate) {\r\n\tvar state = typeof forced !== 'undefined' ? !forced : this.isVisible();\r\n\tthis.dom.toggleClass('emote-menu-hidden', state);\r\n\tif (!skipInstanceUpdate) {\r\n\t\tthis.instance.toggleVisibility(!state);\r\n\t}\r\n\r\n\tvar group = this.getGroup();\r\n\tgroup.toggleDisplay(group.isVisible(), true);\r\n\r\n\treturn this;\r\n};\r\n\r\nUIEmote.prototype.isVisible = function () {\r\n\treturn !this.dom.hasClass('emote-menu-hidden');\r\n};\r\n\r\nUIEmote.prototype.toggleFavorite = function (forced, skipInstanceUpdate) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.isFavorite();\r\n\tthis.dom.toggleClass('emote-menu-starred', state);\r\n\tif (!skipInstanceUpdate) {\r\n\t\tthis.instance.toggleFavorite(state);\r\n\t}\r\n\treturn this;\r\n};\r\n\r\nUIEmote.prototype.isFavorite = function () {\r\n\treturn this.dom.hasClass('emote-menu-starred');\r\n};\r\n\r\nUIEmote.prototype.addToChat = function () {\r\n\tvar ember = require('./ember-api');\r\n\t// Get textarea element.\r\n\tvar element = $('.chat-interface textarea').get(0);\r\n\tvar text = this.instance.getText();\r\n\r\n\t// Insert at cursor / replace selection.\r\n\t// https://developer.mozilla.org/en-US/docs/Code_snippets/Miscellaneous\r\n\tvar selectionEnd = element.selectionStart + text.length;\r\n\tvar currentValue = element.value;\r\n\tvar beforeText = currentValue.substring(0, element.selectionStart);\r\n\tvar afterText = currentValue.substring(element.selectionEnd, currentValue.length);\r\n\t// Smart padding, only put space at start if needed.\r\n\tif (\r\n\t\tbeforeText !== '' &&\r\n\t\tbeforeText.substr(-1) !== ' '\r\n\t) {\r\n\t\ttext = ' ' + text;\r\n\t}\r\n\t// Always put space at end.\r\n\ttext = beforeText + text + ' ' + afterText;\r\n\t// Set the text.\r\n\tember.get('controller:chat', 'currentRoom').set('messageToSend', text);\r\n\telement.focus();\r\n\t// Put cursor at end.\r\n\tselectionEnd = element.selectionStart + text.length;\r\n\telement.setSelectionRange(selectionEnd, selectionEnd);\r\n\r\n\treturn this;\r\n};\r\n\r\nUIEmote.prototype.getGroup = function () {\r\n\treturn theMenu.getGroup(this.instance.getChannelName());\r\n};\r\n\r\nUIEmote.prototype.update = function () {\r\n\tthis.toggleDisplay(this.instance.isVisible(), true);\r\n\tthis.toggleFavorite(this.instance.isFavorite(), true);\r\n};\r\n\r\nmodule.exports = api;\r\n","(function ($) {\r\n\t$.fn.resizable = function (options) {\r\n\t\tvar settings = $.extend({\r\n\t\t\talsoResize: null,\r\n\t\t\talsoResizeType: 'both', // `height`, `width`, `both`\r\n\t\t\tcontainment: null,\r\n\t\t\tcreate: null,\r\n\t\t\tdestroy: null,\r\n\t\t\thandle: '.resize-handle',\r\n\t\t\tmaxHeight: 9999,\r\n\t\t\tmaxWidth: 9999,\r\n\t\t\tminHeight: 0,\r\n\t\t\tminWidth: 0,\r\n\t\t\tresize: null,\r\n\t\t\tresizeOnce: null,\r\n\t\t\tsnapSize: 1,\r\n\t\t\tstart: null,\r\n\t\t\tstop: null\r\n\t\t}, options);\r\n\r\n\t\tsettings.element = $(this);\r\n\r\n\t\tfunction recalculateSize(evt) {\r\n\t\t\tvar data = evt.data,\r\n\t\t\t\tresized = {};\r\n\t\t\tdata.diffX = Math.round((evt.pageX - data.pageX) / settings.snapSize) * settings.snapSize;\r\n\t\t\tdata.diffY = Math.round((evt.pageY - data.pageY) / settings.snapSize) * settings.snapSize;\r\n\t\t\tif (Math.abs(data.diffX) > 0 || Math.abs(data.diffY) > 0) {\r\n\t\t\t\tif (\r\n\t\t\t\t\tsettings.element.height() !== data.height + data.diffY &&\r\n\t\t\t\t\tdata.height + data.diffY >= settings.minHeight &&\r\n\t\t\t\t\tdata.height + data.diffY <= settings.maxHeight &&\r\n\t\t\t\t\t(settings.containment ? data.outerHeight + data.diffY + data.offset.top <= settings.containment.offset().top + settings.containment.outerHeight() : true)\r\n\t\t\t\t) {\r\n\t\t\t\t\tsettings.element.height(data.height + data.diffY);\r\n\t\t\t\t\tresized.height = true;\r\n\t\t\t\t}\r\n\t\t\t\tif (\r\n\t\t\t\t\tsettings.element.width() !== data.width + data.diffX &&\r\n\t\t\t\t\tdata.width + data.diffX >= settings.minWidth &&\r\n\t\t\t\t\tdata.width + data.diffX <= settings.maxWidth &&\r\n\t\t\t\t\t(settings.containment ? data.outerWidth + data.diffX + data.offset.left <= settings.containment.offset().left + settings.containment.outerWidth() : true)\r\n\t\t\t\t) {\r\n\t\t\t\t\tsettings.element.width(data.width + data.diffX);\r\n\t\t\t\t\tresized.width = true;\r\n\t\t\t\t}\r\n\t\t\t\tif (resized.height || resized.width) {\r\n\t\t\t\t\tif (settings.resizeOnce) {\r\n\t\t\t\t\t\tsettings.resizeOnce.bind(settings.element)(evt.data);\r\n\t\t\t\t\t\tsettings.resizeOnce = null;\r\n\t\t\t\t\t}\r\n\t\t\t\t\tif (settings.resize) {\r\n\t\t\t\t\t\tsettings.resize.bind(settings.element)(evt.data);\r\n\t\t\t\t\t}\r\n\t\t\t\t\tif (settings.alsoResize) {\r\n\t\t\t\t\t\tif (resized.height && (settings.alsoResizeType === 'height' || settings.alsoResizeType === 'both')) {\r\n\t\t\t\t\t\t\tsettings.alsoResize.height(data.alsoResizeHeight + data.diffY);\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t\tif (resized.width && (settings.alsoResizeType === 'width' || settings.alsoResizeType === 'both')) {\r\n\t\t\t\t\t\t\tsettings.alsoResize.width(data.alsoResizeWidth + data.diffX);\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}\r\n\r\n\t\tfunction start(evt) {\r\n\t\t\tevt.preventDefault();\r\n\t\t\tif (settings.start) {\r\n\t\t\t\tsettings.start.bind(settings.element)();\r\n\t\t\t}\r\n\t\t\tvar data = {\r\n\t\t\t\talsoResizeHeight: settings.alsoResize ? settings.alsoResize.height() : 0,\r\n\t\t\t\talsoResizeWidth: settings.alsoResize ? settings.alsoResize.width() : 0,\r\n\t\t\t\theight: settings.element.height(),\r\n\t\t\t\toffset: settings.element.offset(),\r\n\t\t\t\touterHeight: settings.element.outerHeight(),\r\n\t\t\t\touterWidth: settings.element.outerWidth(),\r\n\t\t\t\tpageX: evt.pageX,\r\n\t\t\t\tpageY: evt.pageY,\r\n\t\t\t\twidth: settings.element.width()\r\n\t\t\t};\r\n\t\t\t$(document).on('mousemove', '*', data, recalculateSize);\r\n\t\t\t$(document).on('mouseup', '*', stop);\r\n\t\t}\r\n\r\n\t\tfunction stop() {\r\n\t\t\tif (settings.stop) {\r\n\t\t\t\tsettings.stop.bind(settings.element)();\r\n\t\t\t}\r\n\t\t\t$(document).off('mousemove', '*', recalculateSize);\r\n\t\t\t$(document).off('mouseup', '*', stop);\r\n\t\t}\r\n\r\n\t\tif (settings.handle) {\r\n\t\t\tif (settings.alsoResize && ['both', 'height', 'width'].indexOf(settings.alsoResizeType) >= 0) {\r\n\t\t\t\tsettings.alsoResize = $(settings.alsoResize);\r\n\t\t\t}\r\n\t\t\tif (settings.containment) {\r\n\t\t\t\tsettings.containment = $(settings.containment);\r\n\t\t\t}\r\n\t\t\tsettings.handle = $(settings.handle);\r\n\t\t\tsettings.snapSize = settings.snapSize < 1 ? 1 : settings.snapSize;\r\n\r\n\t\t\tif (options === 'destroy') {\r\n\t\t\t\tsettings.handle.off('mousedown', start);\r\n\r\n\t\t\t\tif (settings.destroy) {\r\n\t\t\t\t\tsettings.destroy.bind(this)();\r\n\t\t\t\t}\r\n\t\t\t\treturn this;\r\n\t\t\t}\r\n\r\n\t\t\tsettings.handle.on('mousedown', start);\r\n\r\n\t\t\tif (settings.create) {\r\n\t\t\t\tsettings.create.bind(this)();\r\n\t\t\t}\r\n\t\t}\r\n\t\treturn this;\r\n\t};\r\n})(jQuery);\r\n"]}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment