//META{"name":"ChatAliases","website":"https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/ChatAliases","source":"https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/ChatAliases/ChatAliases.plugin.js"}*// class ChatAliases { getName () {return "ChatAliases";} getVersion () {return "2.0.1";} getAuthor () {return "DevilBro";} getDescription () {return "Allows the user to configure their own chat-aliases which will automatically be replaced before the message is being sent.";} constructor () { this.changelog = { "fixed":[["Light Theme Update","Fixed bugs for the Light Theme Update, which broke 99% of my plugins"]] }; this.patchModules = { "ChannelTextArea":"componentDidMount", "StandardSidebarView":"componentWillUnmount" }; } initConstructor () { this.defaults = { configs: { case: {value:false, description:"Handle the wordvalue case sensitive"}, exact: {value:true, description:"Handle the wordvalue as an exact word and not as part of a word"}, autoc: {value:true, description:"Add this alias in the autocomplete menu (not for RegExp)"}, regex: {value:false, description:"Handle the wordvalue as a RegExp string"}, file: {value:false, description:"Handle the replacevalue as a filepath"} }, settings: { addContextMenu: {value:true, description:"Add a ContextMenu entry to faster add new Aliases:"}, addAutoComplete: {value:true, description:"Add an Autocomplete-Menu for Non-RegExp Aliases:"} }, amounts: { minAliasLength: {value:2, min:1, description:"Minimal Character Length to open Autocomplete-Menu:"} } }; this.chataliasesAddModalMarkup = `

Add to ChatAliases

Replace:

With:

${BDFDB.ArrayUtils.remove(Object.keys(this.defaults.configs), "file").map((key, i) => `

${this.defaults.configs[key].description}

`).join('')}
`; } getSettingsPanel () { if (!global.BDFDB || typeof BDFDB != "object" || !BDFDB.loaded || !this.started) return; let settings = BDFDB.DataUtils.get(this, "settings"); var amounts = BDFDB.DataUtils.get(this, "amounts"); var settingshtml = `
${this.name}
`; for (let key in settings) { settingshtml += `

${this.defaults.settings[key].description}

`; } for (let key in amounts) { settingshtml += `

${this.defaults.amounts[key].description}

`; } settingshtml += `

Replace:

With:

`; settingshtml += `

List of Chataliases:

`; for (let config in this.defaults.configs) { settingshtml += `
${config.toUpperCase()}
`; } settingshtml += `
`; for (let word in this.aliases) { settingshtml += `
`; for (let config in this.defaults.configs) { settingshtml += `
`; } settingshtml += `
`; } settingshtml += `
`; settingshtml += `

Remove all added words.

`; var infoHidden = BDFDB.DataUtils.load(this, "hideInfo", "hideInfo"); settingshtml += `
Information
`; settingshtml += `
Case: Will replace words while comparing lowercase/uppercase. apple => apple, not APPLE or AppLe
Not Case: Will replace words while ignoring lowercase/uppercase. apple => apple, APPLE and AppLe
Exact: Will replace words that are exactly the replaceword. apple to pear => applepie stays applepie
Not Exact: Will replace words anywhere they appear. apple to pear => applepieapple to pearpiepear
Autoc: Will appear in the Autocomplete Menu (if enabled).
Regex: Will treat the entered wordvalue as a regular expression. Help
File: If the replacevalue is a filepath it will try to upload the file located at the filepath.
`; settingshtml += `
`; let settingspanel = BDFDB.DOMUtils.create(settingshtml); BDFDB.initElements(settingspanel, this); BDFDB.ListenerUtils.add(this, settingspanel, "keypress", ".wordInputs", e => {if (e.which == 13) this.updateContainer(settingspanel, e.currentTarget);}); BDFDB.ListenerUtils.add(this, settingspanel, "keyup", BDFDB.dotCN.gamenameinput, e => {this.updateWord(e.currentTarget);}); BDFDB.ListenerUtils.add(this, settingspanel, "click", ".btn-addword, .remove-word, .remove-all", e => {this.updateContainer(settingspanel, e.currentTarget);}); BDFDB.ListenerUtils.add(this, settingspanel, "click", BDFDB.dotCN.checkboxinput, e => {this.updateConfig(e.currentTarget);}); BDFDB.ListenerUtils.add(this, settingspanel, "click", ".toggle-info", e => {this.toggleInfo(e.currentTarget);}); return settingspanel; } //legacy load () {} start () { if (!global.BDFDB) global.BDFDB = {myPlugins:{}}; if (global.BDFDB && global.BDFDB.myPlugins && typeof global.BDFDB.myPlugins == "object") global.BDFDB.myPlugins[this.getName()] = this; var libraryScript = document.querySelector('head script#BDFDBLibraryScript'); if (!libraryScript || (performance.now() - libraryScript.getAttribute("date")) > 600000) { if (libraryScript) libraryScript.remove(); libraryScript = document.createElement("script"); libraryScript.setAttribute("id", "BDFDBLibraryScript"); libraryScript.setAttribute("type", "text/javascript"); libraryScript.setAttribute("src", "https://mwittrien.github.io/BetterDiscordAddons/Plugins/BDFDB.min.js"); libraryScript.setAttribute("date", performance.now()); libraryScript.addEventListener("load", () => {this.initialize();}); document.head.appendChild(libraryScript); } else if (global.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) this.initialize(); this.startTimeout = setTimeout(() => { try {return this.initialize();} catch (err) {console.error(`%c[${this.getName()}]%c`, "color: #3a71c1; font-weight: 700;", "", "Fatal Error: Could not initiate plugin! " + err);} }, 30000); } initialize () { if (global.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) { if (this.started) return; BDFDB.PluginUtils.init(this); this.aliases = BDFDB.DataUtils.load(this, "words"); BDFDB.ListenerUtils.add(document, "click", e => { if (!e.target.tagName === "TEXTAREA") BDFDB.DOMUtils.remove(".autocompleteAliases", ".autocompleteAliasesRow"); }); BDFDB.ModuleUtils.forceAllUpdates(this); } else console.error(`%c[${this.getName()}]%c`, "color: #3a71c1; font-weight: 700;", "", "Fatal Error: Could not load BD functions!"); } stop () { if (global.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) { this.stopping = true; BDFDB.DOMUtils.remove(".autocompleteAliases", ".autocompleteAliasesRow"); BDFDB.PluginUtils.clear(this); } } // begin of own functions updateContainer (settingspanel, ele) { var wordvalue = null, replacevalue = null, action = ele.getAttribute("action"); var update = () => { BDFDB.DataUtils.save(this.aliases, this, "words"); var containerhtml = ``; for (let word in this.aliases) { containerhtml += `
`; for (let config in this.defaults.configs) { containerhtml += `
`; } containerhtml += `
`; } settingspanel.querySelector(".alias-list").innerHTML = containerhtml; BDFDB.initElements(settingspanel, this); }; if (action == "add") { var wordinput = settingspanel.querySelector("#input-wordvalue"); var replaceinput = settingspanel.querySelector("#input-replacevalue"); if (wordinput.value && wordinput.value.trim().length > 0 && replaceinput.value && replaceinput.value.trim().length > 0) { this.saveWord(wordinput.value.trim(), replaceinput.value.trim(), settingspanel.querySelector("#input-file")); wordinput.value = null; replaceinput.value = null; update(); } } else if (action == "remove") { wordvalue = ele.getAttribute("word"); if (wordvalue) { delete this.aliases[wordvalue]; update(); } } else if (action == "removeall") { BDFDB.ModalUtils.confirm(this, "Are you sure you want to remove all added Words from your list?", () => { this.aliases = {}; update(); }); } } saveWord (wordvalue, replacevalue, fileselection, configs = BDFDB.DataUtils.get(this, "configs")) { if (!wordvalue || !replacevalue || !fileselection) return; var filedata = null; if (fileselection.files && fileselection.files[0] && BDFDB.LibraryRequires.fs.existsSync(replacevalue)) { filedata = JSON.stringify({ data: BDFDB.LibraryRequires.fs.readFileSync(replacevalue).toString("base64"), name: fileselection.files[0].name, type: fileselection.files[0].type }); } this.aliases[wordvalue] = { replace: replacevalue, filedata: filedata, case: configs.case, exact: wordvalue.indexOf(" ") > -1 ? false : configs.exact, autoc: configs.regex ? false : configs.autoc, regex: configs.regex, file: filedata != null }; } updateWord (ele) { BDFDB.TimeUtils.clear(ele.updateTimeout); ele.updateTimeout = BDFDB.TimeUtils.timeout(() => { var card = ele.parentElement.parentElement; var oldwordvalue = ele.getAttribute("word"); if (oldwordvalue && this.aliases[oldwordvalue]) { var wordinput = card.querySelector(".word-name"); var replaceinput = card.querySelector(".replace-name"); var removebutton = card.querySelector(".remove-word"); var newwordvalue = wordinput.value; var newreplacevalue = replaceinput.value; wordinput.setAttribute("word", newwordvalue); wordinput.setAttribute("value", newwordvalue); replaceinput.setAttribute("word", newwordvalue); replaceinput.setAttribute("value", newreplacevalue); removebutton.setAttribute("word", newwordvalue); this.aliases[newwordvalue] = this.aliases[oldwordvalue]; this.aliases[newwordvalue].replace = newreplacevalue; if (newwordvalue != oldwordvalue) delete this.aliases[oldwordvalue]; BDFDB.DataUtils.save(this.aliases, this, "words"); } },500); } updateConfig (ele) { var wordvalue = ele.getAttribute("word"); var config = ele.getAttribute("config"); if (wordvalue && this.aliases[wordvalue] && config) { this.aliases[wordvalue][config] = ele.checked; BDFDB.DataUtils.save(this.aliases, this, "words"); } } toggleInfo (ele) { BDFDB.DOMUtils.toggleClass(ele.querySelector("svg"), BDFDB.disCN.directionright); BDFDB.DOMUtils.toggle(ele.nextElementSibling); BDFDB.DataUtils.save(BDFDB.DOMUtils.isHidden(ele.nextElementSibling), this, "hideInfo", "hideInfo"); } onNativeContextMenu (instance, menu, returnvalue) { if (instance.props.value && instance.props.value.trim()) { if ((instance.props.type == "NATIVE_TEXT" || instance.props.type == "CHANNEL_TEXT_AREA") && BDFDB.DataUtils.get(this, "settings", "addContextMenu")) this.appendItem(menu, returnvalue, instance.props.value.trim()); } } onMessageContextMenu (instance, menu, returnvalue) { if (instance.props.message && instance.props.channel && instance.props.target) { let text = document.getSelection().toString().trim(); if (text && BDFDB.DataUtils.get(this, "settings", "addContextMenu")) this.appendItem(menu, returnvalue, text); } } appendItem (menu, returnvalue, text) { let [children, index] = BDFDB.ReactUtils.findChildren(returnvalue, {name:["FluxContainer(MessageDeveloperModeGroup)", "DeveloperModeGroup"]}); const itemgroup = BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ContextMenuItemGroup, { children: [ BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ContextMenuItem, { label: "Add to ChatAliases", action: e => { BDFDB.ContextMenuUtils.close(menu); this.openAddModal(text); } }) ] }); if (index > -1) children.splice(index, 0, itemgroup); else children.push(itemgroup); } processStandardSidebarView (instance, wrapper, returnvalue) { if (this.SettingsUpdated) { delete this.SettingsUpdated; BDFDB.ModuleUtils.forceAllUpdates(this); } } processChannelTextArea (instance, wrapper, returnvalue) { if (instance.props.channel && instance.props.type) { var textarea = wrapper.querySelector("textarea"); if (!textarea) return; let settings = BDFDB.DataUtils.get(this, "settings"); BDFDB.ListenerUtils.add(this, textarea, "input", () => { if (this.format) { this.format = false; textarea.focus(); textarea.selectionStart = 0; textarea.selectionEnd = textarea.value.length; if (document.activeElement == textarea) { var messageInput = this.formatText(textarea.value); if (messageInput) { if (messageInput.text != null) { document.execCommand("insertText", false, messageInput.text ? messageInput.text + " " : ""); } if (messageInput.files.length > 0 && (instance.props.channel.type == 1 || BDFDB.UserUtils.can("ATTACH_FILES"))) { BDFDB.LibraryModules.UploadUtils.instantBatchUpload(instance.props.channel.id, messageInput.files); } } } } }); BDFDB.ListenerUtils.add(this, textarea, "keydown", e => { let autocompletemenu = textarea.parentElement.querySelector(BDFDB.dotCN.autocomplete); if (autocompletemenu && (e.which == 9 || e.which == 13)) { if (BDFDB.DOMUtils.containsClass(autocompletemenu.querySelector(BDFDB.dotCN.autocompleteselected).parentElement, "autocompleteAliasesRow")) { BDFDB.ListenerUtils.stopEvent(e); this.swapWordWithAlias(textarea); } } else if (autocompletemenu && (e.which == 38 || e.which == 40)) { let autocompleteitems = autocompletemenu.querySelectorAll(BDFDB.dotCN.autocompleteselectable + ":not(.autocompleteAliasesSelector)"); let selected = autocompletemenu.querySelector(BDFDB.dotCN.autocompleteselected); if (BDFDB.DOMUtils.containsClass(selected, "autocompleteAliasesSelector") || autocompleteitems[e.which == 38 ? 0 : (autocompleteitems.length-1)] == selected) { BDFDB.ListenerUtils.stopEvent(e); let next = this.getNextSelection(autocompletemenu, null, e.which == 38 ? false : true); BDFDB.DOMUtils.removeClass(selected, BDFDB.disCN.autocompleteselected); BDFDB.DOMUtils.addClass(selected, BDFDB.disCN.autocompleteselector); BDFDB.DOMUtils.addClass(next, BDFDB.disCN.autocompleteselected); } } else if (textarea.value && !e.shiftKey && e.which == 13 && !autocompletemenu && textarea.value.indexOf("s/") != 0) { this.format = true; textarea.dispatchEvent(new Event("input")); } else if (!e.ctrlKey && e.which != 16 && settings.addAutoComplete && textarea.selectionStart == textarea.selectionEnd && textarea.selectionEnd == textarea.value.length) { BDFDB.TimeUtils.clear(textarea.ChatAliasAutocompleteTimeout); textarea.ChatAliasAutocompleteTimeout = BDFDB.TimeUtils.timeout(() => {this.addAutoCompleteMenu(textarea);},100); } if (!e.ctrlKey && e.which != 38 && e.which != 40 && !(e.which == 39 && textarea.selectionStart == textarea.selectionEnd && textarea.selectionEnd == textarea.value.length)) BDFDB.DOMUtils.remove(".autocompleteAliases", ".autocompleteAliasesRow"); }); BDFDB.ListenerUtils.add(this, textarea, "click", e => { if (settings.addAutoComplete && textarea.selectionStart == textarea.selectionEnd && textarea.selectionEnd == textarea.value.length) BDFDB.TimeUtils.timeout(() => {this.addAutoCompleteMenu(textarea);}); }); } } addAutoCompleteMenu (textarea) { if (!textarea.value || textarea.parentElement.querySelector(".autocompleteAliasesRow")) return; let words = textarea.value.split(/\s/); let lastword = words[words.length-1].trim(); if (words.length == 1 && BDFDB.BDUtils.isPluginEnabled("WriteUpperCase")) { let first = lastword.charAt(0); if (first === first.toUpperCase() && lastword.toLowerCase().indexOf("http") == 0) { lastword = lastword.charAt(0).toLowerCase() + lastword.slice(1); } else if (first === first.toLowerCase() && first !== first.toUpperCase() && lastword.toLowerCase().indexOf("http") != 0) { lastword = lastword.charAt(0).toUpperCase() + lastword.slice(1); } } if (lastword && BDFDB.DataUtils.get(this, "amounts", "minAliasLength") <= lastword.length) { let matchedaliases = {}; for (let word in this.aliases) { let aliasdata = this.aliases[word]; if (!aliasdata.regex && aliasdata.autoc) { if (aliasdata.exact) { if (aliasdata.case && word.indexOf(lastword) == 0) matchedaliases[word] = aliasdata; else if (!aliasdata.case && word.toLowerCase().indexOf(lastword.toLowerCase()) == 0) matchedaliases[word] = aliasdata; } else { if (aliasdata.case && word.indexOf(lastword) > -1) matchedaliases[word] = aliasdata; else if (!aliasdata.case && word.toLowerCase().indexOf(lastword.toLowerCase()) > -1) matchedaliases[word] = aliasdata; } } } if (!BDFDB.ObjectUtils.isEmpty(matchedaliases)) { let autocompletemenu = textarea.parentElement.querySelector(BDFDB.dotCNS.autocomplete + BDFDB.dotCN.autocompleteinner), amount = 15; if (!autocompletemenu) { autocompletemenu = BDFDB.DOMUtils.create(`
`); textarea.parentElement.appendChild(autocompletemenu); autocompletemenu = autocompletemenu.firstElementChild; } else { amount -= autocompletemenu.querySelectorAll(BDFDB.dotCN.autocompleteselectable).length; } let autocompleterowheader = BDFDB.DOMUtils.create(`
Aliases: ${BDFDB.StringUtils.htmlEscape(lastword)}
`); autocompletemenu.appendChild(autocompleterowheader); BDFDB.ListenerUtils.add(this, autocompletemenu, "mouseenter", BDFDB.dotCN.autocompleteselectable, e => { var selected = autocompletemenu.querySelectorAll(BDFDB.dotCN.autocompleteselected); BDFDB.DOMUtils.removeClass(selected, BDFDB.disCN.autocompleteselected); BDFDB.DOMUtils.addClass(selected, BDFDB.disCN.autocompleteselector); BDFDB.DOMUtils.addClass(e.currentTarget, BDFDB.disCN.autocompleteselected); }); for (let word in matchedaliases) { if (amount-- < 1) break; let autocompleterow = BDFDB.DOMUtils.create(`
${BDFDB.StringUtils.htmlEscape(word)}
${BDFDB.StringUtils.htmlEscape(matchedaliases[word].replace)}
`); autocompleterow.querySelector(BDFDB.dotCN.autocompleteselectable).addEventListener("click", () => {this.swapWordWithAlias(textarea);}); autocompletemenu.appendChild(autocompleterow); } if (!autocompletemenu.querySelector(BDFDB.dotCN.autocompleteselected)) { BDFDB.DOMUtils.addClass(autocompletemenu.querySelector(".autocompleteAliasesRow " + BDFDB.dotCN.autocompleteselectable), BDFDB.disCN.autocompleteselected); } } } } getNextSelection (menu, selected, forward) { selected = selected ? selected : menu.querySelector(BDFDB.dotCN.autocompleteselected).parentElement; let next, sibling = forward ? selected.nextElementSibling : selected.previousElementSibling; if (sibling) { next = sibling.querySelector(BDFDB.dotCN.autocompleteselectable); } else { let items = menu.querySelectorAll(BDFDB.dotCN.autocompleteselectable); next = forward ? items[0] : items[items.length-1]; } return next ? next : this.getNextSelection(menu, sibling, forward); } swapWordWithAlias (textarea) { let aliasword = textarea.parentElement.querySelector(".autocompleteAliasesRow " + BDFDB.dotCN.autocompleteselected + " .aliasword").innerText; let lastword = textarea.parentElement.querySelector(".autocompleteAliasesRow .lastword").innerText; if (aliasword && lastword) { BDFDB.DOMUtils.remove(".autocompleteAliases", ".autocompleteAliasesRow"); textarea.focus(); textarea.selectionStart = textarea.value.length - lastword.length; textarea.selectionEnd = textarea.value.length; document.execCommand("insertText", false, aliasword); textarea.selectionStart = textarea.value.length; textarea.selectionEnd = textarea.value.length; } } formatText (text) { text = text.replace(/([\n\t\r])/g, " $1 "); var newText = [], files = [], wordAliases = {}, multiAliases = {}; for (let word in this.aliases) { if (!this.aliases[word].regex && word.indexOf(" ") == -1) wordAliases[word] = this.aliases[word]; else multiAliases[word] = this.aliases[word]; } for (let word of text.trim().split(" ")) { newText.push(this.useAliases(word, wordAliases, files, true)); } newText = newText.length == 1 ? newText[0] : newText.join(" "); newText = newText.replace(/ ([\n\t\r]) /g, "$1"); newText = this.useAliases(newText, multiAliases, files, false); return {text:newText, files}; } useAliases (string, aliases, files, singleword) { for (let word in aliases) { let aliasdata = aliases[word]; let escpAlias = aliasdata.regex ? word : BDFDB.StringUtils.regEscape(word); let result = true, replaced = false, tempstring1 = string, tempstring2 = ""; let regstring = aliasdata.exact ? "^" + escpAlias + "$" : escpAlias; while (result != null) { result = new RegExp(regstring, (aliasdata.case ? "" : "i") + (aliasdata.exact ? "" : "g")).exec(tempstring1); if (result) { replaced = true; let replace = aliasdata.file ? "" : BDFDB.StringUtils.insertNRST(aliasdata.replace); if (result.length > 1) for (var i = 1; i < result.length; i++) replace = replace.replace(new RegExp("\\\\" + i + "|\\$" + i, "g"), result[i]); tempstring2 += tempstring1.slice(0, result.index + result[0].length).replace(result[0], replace); tempstring1 = tempstring1.slice(result.index + result[0].length); if (aliasdata.file && typeof aliasdata.filedata == "string") { var filedata = JSON.parse(aliasdata.filedata); files.push(new File([Uint8Array.from(atob(filedata.data), c => c.charCodeAt(0))], filedata.name, {type:filedata.type})); } if (aliasdata.regex && regstring.indexOf("^") == 0) result = null; } if (!result) tempstring2 += tempstring1; } if (replaced) { string = tempstring2; if (singleword) break; } } return string; } openAddModal (wordvalue) { let chataliasesAddModal = BDFDB.DOMUtils.create(this.chataliasesAddModalMarkup); let wordvalueinput = chataliasesAddModal.querySelector("#input-wordvalue"); let replacevalueinput = chataliasesAddModal.querySelector("#input-replacevalue"); let addbutton = chataliasesAddModal.querySelector(".btn-add"); wordvalueinput.value = wordvalue || ""; BDFDB.appendModal(chataliasesAddModal); let checkInputs = () => { let validinputs = [wordvalueinput, replacevalueinput]; let invalidinputs = []; let type = ""; if (!wordvalueinput.value.trim()) { BDFDB.ArrayUtils.remove(validinputs, wordvalueinput); invalidinputs.push(wordvalueinput); type += "Wordvalue"; } if (!replacevalueinput.value.trim()) { BDFDB.ArrayUtils.remove(validinputs, replacevalueinput); invalidinputs.push(replacevalueinput); type += ((type ? " and " : "") + "Replacevalue"); } if (type) addDisabledTooltip(invalidinputs, type); else { addbutton.disabled = false; addbutton.style.removeProperty("pointer-events"); BDFDB.DOMUtils.remove(".chataliases-disabled-tooltip"); } BDFDB.DOMUtils.removeClass(validinputs, "invalid"); }; let addDisabledTooltip = (invalidinputs, type) => { BDFDB.DOMUtils.remove(".chataliases-disabled-tooltip"); addbutton.disabled = true; BDFDB.DOMUtils.addClass(invalidinputs, "invalid"); addbutton.style.setProperty("pointer-events", "none", "important"); BDFDB.TooltipUtils.create(addbutton, "Choose a " + type, {type: "right", color: "red", selector: "chataliases-disabled-tooltip"}); }; wordvalueinput.addEventListener("input", checkInputs); replacevalueinput.addEventListener("input", checkInputs); BDFDB.ListenerUtils.addToChildren(chataliasesAddModal, "click", BDFDB.dotCNC.backdrop + BDFDB.dotCNC.modalclose + ".btn-add", () => { BDFDB.DOMUtils.remove(".chataliases-disabled-tooltip"); }); addbutton.addEventListener("click", e => { let configs = {}; for (let key in this.defaults.configs) { let configinput = chataliasesAddModal.querySelector("#input-config" + key); if (configinput) configs[key] = configinput.checked; } this.saveWord(wordvalueinput.value.trim(), replacevalueinput.value.trim(), chataliasesAddModal.querySelector("#input-file"), configs); BDFDB.DataUtils.save(this.aliases, this, "words"); }); wordvalueinput.focus(); BDFDB.TimeUtils.timeout(checkInputs, 500); } }