/* BetterDiscordApp Core JavaScript * Version: 1.52 * Author: Jiiks | http://jiiks.net * Date: 27/08/2015 - 16:36 * Last Update: 24/010/2015 - 17:27 * https://github.com/Jiiks/BetterDiscordApp */ var settingsPanel, emoteModule, utils, quickEmoteMenu, opublicServers, voiceMode, pluginModule; var jsVersion = 1.53; var supportedVersion = "0.2.3"; var mainObserver; var twitchEmoteUrlStart = "https://static-cdn.jtvnw.net/emoticons/v1/"; var twitchEmoteUrlEnd = "/1.0"; var ffzEmoteUrlStart = "https://cdn.frankerfacez.com/emoticon/"; var ffzEmoteUrlEnd = "/1"; var bttvEmoteUrlStart = "https://cdn.betterttv.net/emote/"; var bttvEmoteUrlEnd = "/1x"; var mainCore; var settings = { "Save logs locally": { "id": "bda-gs-0", "info": "Saves chat logs locally", "implemented": false }, "Public Servers": { "id": "bda-gs-1", "info": "Display public servers button", "implemented": true }, "Minimal Mode": { "id": "bda-gs-2", "info": "Hide elements and reduce the size of elements.", "implemented": true }, "Voice Mode": { "id": "bda-gs-4", "info": "Only show voice chat", "implemented": true }, "Hide Channels": { "id": "bda-gs-3", "info": "Hide channels in minimal mode", "implemented": true }, "Quick Emote Menu": { "id": "bda-es-0", "info": "Show quick emote menu for adding emotes", "implemented": true }, "Show Emotes": { "id": "bda-es-7", "info": "Show any emotes", "implemented": true }, "FrankerFaceZ Emotes": { "id": "bda-es-1", "info": "Show FrankerFaceZ Emotes", "implemented": true }, "BetterTTV Emotes": { "id": "bda-es-2", "info": "Show BetterTTV Emotes", "implemented": true }, "Emote Autocomplete": { "id": "bda-es-3", "info": "Autocomplete emote commands", "implemented": false }, "Emote Auto Capitalization": { "id": "bda-es-4", "info": "Autocapitalize emote commands", "implemented": true }, "Override Default Emotes": { "id": "bda-es-5", "info": "Override default emotes", "implemented": false }, "Show Names": { "id": "bda-es-6", "info": "Show emote names on hover", "implemented": true } }; var links = { "Jiiks.net": { "text": "Jiiks.net", "href": "http://jiiks.net", "target": "_blank" }, "twitter": { "text": "Twitter", "href": "http://twitter.com/jiiksi", "target": "_blank" }, "github": { "text": "Github", "href": "http://github.com/jiiks", "target": "_blank" } }; var defaultCookie = { "version": jsVersion, "bda-gs-0": false, "bda-gs-1": true, "bda-gs-2": false, "bda-gs-3": false, "bda-gs-4": false, "bda-es-0": true, "bda-es-1": false, "bda-es-2": false, "bda-es-3": false, "bda-es-4": false, "bda-es-5": true, "bda-es-6": true, "bda-es-7": true, "bda-jd": true }; var bdchangelog = { "changes": { "plugins": { "title": "Plugins!", "text": "Combined with Core 0.2.4, you can now write/use JavaScript plugins for Discord!", "img": "" } }, "fixes": { "eemotes": { "title": "Edit Emotes!", "text": "Edited messages now display emotes properly!", "img": "" } }, "upcoming": { "ignore": { "title": "Ignore User!", "text": "Ignore users you don't like", "img": "" }, "more": { "title": "More Things!", "text": "More things but probably not in the next version.", "img": "" } } }; var settingsCookie = {}; function Core() {} Core.prototype.init = function() { var self = this; if(version < supportedVersion) { alert("BetterDiscord v" + version + "(your version)" + " is not supported by the latest js("+jsVersion+"). Please download the latest version from GitHub."); return; } utils = new Utils(); utils.getHash(); emoteModule = new EmoteModule(); quickEmoteMenu = new QuickEmoteMenu(); voiceMode = new VoiceMode(); emoteModule.init(); this.initSettings(); this.initObserver(); //Incase were too fast function gwDefer() { console.log(new Date().getTime() + " Defer"); if($(".guilds-wrapper .guilds").children().length > 0) { console.log(new Date().getTime() + " Defer Loaded"); var guilds = $(".guilds li:first-child"); guilds.after($("
  • ", { id: "bd-pub-li", css: { "height": "20px", "display": settingsCookie["bda-gs-1"] == true ? "" : "none" } }).append($("
    ", { class: "guild-inner", css: { "height": "20px", "border-radius": "4px" } }).append($("").append($("
    ", { css: { "line-height": "20px", "font-size": "12px" }, text: "public", id: "bd-pub-button" }))))); var showChannelsButton = $("' + '
    ' + '
    ' + '
    '; if(bdchangelog.changes != null) { changeLog += '' + '

    ' + ' New Stuff' + '

    ' + '
      '; for(var change in bdchangelog.changes) { change = bdchangelog.changes[change]; changeLog += '' + '
    • ' + ' '+change.title+'' + '
      '+change.text+'
      ' + '
    • '; } changeLog += '
    '; } if(bdchangelog.fixes != null) { changeLog += '' + '

    ' + ' Fixed' + '

    ' + '
      '; for(var fix in bdchangelog.fixes) { fix = bdchangelog.fixes[fix]; changeLog += '' + '
    • ' + ' '+fix.title+'' + '
      '+fix.text+'
      ' + '
    • '; } changeLog += '
    '; } if(bdchangelog.upcoming != null) { changeLog += '' + '

    ' + ' Coming Soon' + '

    ' + '
      '; for(var upc in bdchangelog.upcoming) { upc = bdchangelog.upcoming[upc]; changeLog += '' + '
    • ' + ' '+upc.title+'' + '
      '+upc.text+'
      ' + '
    • '; } changeLog += '
    '; } changeLog += '' + '
    ' + '
    ' + ' ' + '
    ' + ' ' + ''; return changeLog; }; /* BetterDiscordApp EmoteModule JavaScript * Version: 1.5 * Author: Jiiks | http://jiiks.net * Date: 26/08/2015 - 15:29 * Last Update: 14/10/2015 - 09:48 * https://github.com/Jiiks/BetterDiscordApp * Note: Due to conflicts autocapitalize only supports global emotes */ /* * =Changelog= * -v1.5 * --Twitchemotes.com api */ var emotesFfz = {}; var emotesBTTV = {}; var emotesTwitch = { "emotes": { "emote": { "image_id": 0 } } }; //for ide var subEmotesTwitch = {}; function EmoteModule() { } EmoteModule.prototype.init = function() { }; EmoteModule.prototype.getBlacklist = function() { $.getJSON("https://cdn.rawgit.com/Jiiks/betterDiscordApp/"+_hash+"/emotefilter.json", function(data) { bemotes = data.blacklist; }); }; EmoteModule.prototype.obsCallback = function(mutation) { var self = this; if(!settingsCookie["bda-es-7"]) return; for(var i = 0 ; i < mutation.addedNodes.length ; ++i) { var next = mutation.addedNodes.item(i); if(next) { var nodes = self.getNodes(next); for(var node in nodes) { if(nodes.hasOwnProperty(node)) { self.injectEmote(nodes[node]); } } } } }; EmoteModule.prototype.getNodes = function(node) { var next; var nodes = []; var treeWalker = document.createTreeWalker(node, NodeFilter.SHOW_TEXT, null, false); while(next = treeWalker.nextNode()) { nodes.push(next); } return nodes; }; var bemotes = []; var spoilered = []; //TODO Edited emotes EmoteModule.prototype.injectEmote = function(node) { if(typeof emotesTwitch === 'undefined') return; if(!node.parentElement) return; var parent = node.parentElement; if(parent.tagName != "SPAN") return; var edited = false; if($(parent.parentElement).hasClass("edited")) { parent = parent.parentElement.parentElement.firstChild; //:D edited = true; } //if(!$(parent.parentElement).hasClass("markup") && !$(parent.parentElement).hasClass("message-content")) return; function inject() { if(!$(parent.parentElement).hasClass("markup") && !$(parent.parentElement).hasClass("message-content")) { return; } var parentInnerHTML = parent.innerHTML; var words = parentInnerHTML.split(/\s+/g); if(!words) return; words.some(function(word) { if(word.slice(0, 4) == "[!s]" ) { parentInnerHTML = parentInnerHTML.replace("[!s]", ""); var markup = $(parent).parent(); var reactId = markup.attr("data-reactid"); if(spoilered.indexOf(reactId) > -1) { return; } markup.addClass("spoiler"); markup.on("click", function() { $(this).removeClass("spoiler"); spoilered.push($(this).attr("data-reactid")); }); return; } if(word.length < 4) { return; } if($.inArray(word, bemotes) != -1) return; if (emotesTwitch.emotes.hasOwnProperty(word)) { var len = Math.round(word.length / 4); parentInnerHTML = parentInnerHTML.replace(word, '' + word.substr(0, len) + ' ); return; } if (typeof emotesFfz !== 'undefined' && settingsCookie["bda-es-1"]) { if (emotesFfz.hasOwnProperty(word)) { var len = Math.round(word.length / 4); parentInnerHTML = parentInnerHTML.replace(word, '' + word.substr(0, len) + '); return; } } if (typeof emotesBTTV !== 'undefined' && settingsCookie["bda-es-2"]) { if (emotesBTTV.hasOwnProperty(word)) { var len = Math.round(word.length / 4); parentInnerHTML = parentInnerHTML.replace(word, '' + word.substr(0, len) + '); return; } } if(typeof emotesBTTV2 !== 'undefined' && settingsCookie["bda-es-2"]) { if(emotesBTTV2.hasOwnProperty(word)) { var len = Math.round(word.length / 4); parentInnerHTML = parentInnerHTML.replace(word, '' + word.substr(0, len) + '); return; } } if (subEmotesTwitch.hasOwnProperty(word)) { var len = Math.round(word.length / 4); parentInnerHTML = parentInnerHTML.replace(word, '' + word.substr(0, len) + '); return; } }); if(parent.parentElement == null) return; var oldHeight = parent.parentElement.offsetHeight; parent.innerHTML = parentInnerHTML.replace(new RegExp("\uFDD9", "g"), ""); var newHeight = parent.parentElement.offsetHeight; //Scrollfix var scrollPane = $(".scroller.messages").first(); scrollPane.scrollTop(scrollPane.scrollTop() + (newHeight - oldHeight)); } if(edited) { setTimeout(inject, 250); } else { inject(); } }; EmoteModule.prototype.autoCapitalize = function() { var self = this; $('body').delegate($(".channel-textarea-inner textarea"), 'keyup change paste', function() { if(!settingsCookie["bda-es-4"]) return; var text = $(".channel-textarea-inner textarea").val(); if(text == undefined) return; var lastWord = text.split(" ").pop(); if(lastWord.length > 3) { var ret = self.capitalize(lastWord.toLowerCase()); if(ret != null) { $(".channel-textarea-inner textarea").val(text.replace(lastWord, ret)); } } }); }; EmoteModule.prototype.capitalize = function(value) { var res = emotesTwitch.emotes; for(var p in res){ if(res.hasOwnProperty(p) && value == (p+ '').toLowerCase()){ return p; } } }; /* BetterDiscordApp PublicSevers JavaScripts * Version: 1.0 * Author: Jiiks | http://jiiks.net * Date: 27/08/2015 - 14:16 * https://github.com/Jiiks/BetterDiscordApp */ var publicServers = { "servers": { "server": { "code": 0, "icon": null, "title": "title", "language": "EN", "description": "description" } } }; //for ide function PublicServers() { } PublicServers.prototype.getPanel = function() { return this.container; }; PublicServers.prototype.init = function() { var self = this; this.container = $("
    ", { id: "bd-ps-container", style: "display:none" }); var header = $("
    ", { id: "bd-ps-header" }); $("

    ", { text: "Public Servers" }).appendTo(header); $("", { id: "bd-ps-close", style:"cursor:pointer;", text: "X" }).appendTo(header); header.appendTo(this.getPanel()); var psbody = $("
    ", { id: "bd-ps-body" }); psbody.appendTo(this.getPanel()); var table = $("", { border:"0" }); var thead = $(""); thead.appendTo(table); var headers = $("", { }).append($("", { id: "bd-ps-tbody" }); tbody.appendTo(table); table.appendTo(psbody); $("body").append(this.getPanel()); $("#bd-ps-close").on("click", function() { self.show(); }); var servers = publicServers.servers; for(var server in servers) { if(servers.hasOwnProperty(server)) { var s = servers[server]; var code = s.code; var title = s.title; var language = s.language; var description = s.description; this.addServer(server, code, title, language, description); } } }; PublicServers.prototype.addServer = function(name, code, title, language, description) { var self = this; var tableBody = $("#bd-ps-tbody"); var desc = $(""); tr.append($("
    ", { text: "Name" })).append($("", { text: "Code" })).append($("", { text: "Language" })).append($("", { text: "Description" })).append($("", { text: "Join" })); headers.appendTo(thead); var tbody = $("
    ").append($("
    ", { class: "bd-ps-description", text: description })); var tr = $("
    ", { text: title })); tr.append($("", { css: { "-webkit-user-select":"initial", "user-select":"initial" }, text: code })); tr.append($("", { text: language })); tr.append(desc); tr.append($("").append($("