//META{"name":"EditChannels","website":"https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/EditChannels","source":"https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/EditChannels/EditChannels.plugin.js"}*// class EditChannels { getName () {return "EditChannels";} getVersion () {return "4.0.7";} getAuthor () {return "DevilBro";} getDescription () {return "Allows you to rename and recolor channelnames.";} constructor () { this.changelog = { "fixed":[["Settings","Fixed issue where settings could not be saved"]] }; this.patchModules = { "ChannelTextArea":"componentDidMount", "AuditLog":"componentDidMount", "InviteCard":"render", "ChannelCategoryItem":["componentDidMount","componentDidUpdate"], "ChannelItem":["componentDidMount","componentDidUpdate"], "HeaderBar":["componentDidMount","componentDidUpdate"], "HeaderBarContainer":["componentDidMount","componentDidUpdate"], "Clickable":"componentDidMount", "StandardSidebarView":"componentWillUnmount" }; } initConstructor () { this.css = ` ${BDFDB.dotCN.guildsettingsinvitechannelname}[changed-by-editchannels] { opacity: 1; } `; this.defaults = { settings: { changeChannelIcon: {value:true, inner:false, description:"Change color of Channel Icon."}, changeUnreadIndicator: {value:true, inner:false, description:"Change color of Unread Indicator."}, changeInChatTextarea: {value:true, inner:true, description:"Chat Textarea"}, changeInMentions: {value:true, inner:true, description:"Mentions"}, changeInChannelList: {value:true, inner:true, description:"Channel List"}, changeInChannelHeader: {value:true, inner:true, description:"Channel Header"}, changeInRecentMentions: {value:true, inner:true, description:"Recent Mentions Popout"}, changeInAutoComplete: {value:true, inner:true, description:"Autocomplete Menu"}, changeInAuditLog: {value:true, inner:true, description:"Audit Log"}, changeInInviteLog: {value:true, inner:true, description:"Invite Log"}, changeInSearchPopout: {value:true, inner:true, description:"Search Popout"} } }; } getSettingsPanel () { if (!global.BDFDB || typeof BDFDB != "object" || !BDFDB.loaded || !this.started) return; let settings = BDFDB.DataUtils.get(this, "settings"); let settingsitems = [], inneritems = []; for (let key in settings) (!this.defaults.settings[key].inner ? settingsitems : inneritems).push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsSaveItem, { className: BDFDB.disCN.marginbottom8, type: "Switch", plugin: this, keys: ["settings", key], label: this.defaults.settings[key].description, value: settings[key] })); settingsitems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsPanelInner, { title: "Change Channels in:", first: settingsitems.length == 0, children: inneritems })); settingsitems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsItem, { type: "Button", className: BDFDB.disCN.marginbottom8, color: BDFDB.LibraryComponents.Button.Colors.RED, label: "Reset all Channels", onClick: _ => { BDFDB.ModalUtils.confirm(this, "Are you sure you want to reset all channels?", () => { BDFDB.DataUtils.remove(this, "channels"); this.forceUpdateAll(); }); }, children: BDFDB.LanguageUtils.LanguageStrings.RESET })); return BDFDB.PluginUtils.createSettingsPanel(this, settingsitems); } //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); var observer = new MutationObserver(() => {this.changeAppTitle();}); BDFDB.ObserverUtils.connect(this, document.head.querySelector("title"), {name:"appTitleObserver",instance:observer}, {childList:true}); this.forceUpdateAll(); } 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; let data = BDFDB.DataUtils.load(this, "channels"); BDFDB.DataUtils.remove(this, "channels"); try {this.forceUpdateAll();} catch (err) {} BDFDB.DataUtils.save(data, this, "channels"); BDFDB.DOMUtils.remove(".autocompleteEditChannels", ".autocompleteEditChannelsRow"); BDFDB.PluginUtils.clear(this); } } // begin of own functions onChannelContextMenu (instance, menu, returnvalue) { if (instance.props.channel && !BDFDB.DOMUtils.getParent(".container-hidden", instance.props.target) && !menu.querySelector(`${this.name}-contextMenuSubItem`)) { 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.ContextMenuSubItem, { label: this.labels.context_localchannelsettings_text, render: [BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ContextMenuItemGroup, { children: [ BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ContextMenuItem, { label: this.labels.submenu_channelsettings_text, action: e => { BDFDB.ContextMenuUtils.close(menu); this.showChannelSettings(instance.props.channel); } }), BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ContextMenuItem, { label: this.labels.submenu_resetsettings_text, disabled: !BDFDB.DataUtils.load(this, "channels", instance.props.channel.id), action: e => { BDFDB.ContextMenuUtils.close(menu); BDFDB.DataUtils.remove(this, "channels", instance.props.channel.id); this.forceUpdateAll(); } }) ] })] }) ] }); if (index > -1) children.splice(index, 0, itemgroup); else children.push(itemgroup); } } forceUpdateAll () { this.changeAppTitle(); BDFDB.ModuleUtils.forceAllUpdates(this); } showChannelSettings (info) { var data = BDFDB.DataUtils.load(this, "channels", info.id) || {}; BDFDB.ModalUtils.open(this, { size: "MEDIUM", header: this.labels.modal_header_text, subheader: info.name, children: [ BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormItem, { title: this.labels.modal_channelname_text, className: BDFDB.disCN.marginbottom20 + " input-channelname", children: [ BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextInput, { value: data.name, placeholder: info.name, autoFocus: true }), BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormDivider, { className: BDFDB.disCN.dividerdefault }) ] }), BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormItem, { title: this.labels.modal_colorpicker1_text, className: BDFDB.disCN.marginbottom20, children: [ BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ColorSwatches, { color: data.color, number: 1 }) ] }), BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsItem, { type: "Switch", className: BDFDB.disCN.marginbottom20 + " input-inheritcolor", label: this.labels.modal_inheritcolor_text, value: info.type == 4 && data.inheritColor, disabled: info.type != 4 }) ], buttons: [{ contents: BDFDB.LanguageUtils.LanguageStrings.SAVE, color: "BRAND", close: true, click: modal => { let olddata = Object.assign({}, data); let channelnameinput = modal.querySelector(".input-channelname " + BDFDB.dotCN.input); let inheritcolorinput = modal.querySelector(".input-inheritcolor " + BDFDB.dotCN.switchinner); data.name = channelnameinput.value.trim() || null; data.color = BDFDB.ColorUtils.getSwatchColor(modal, 1); console.log(data.color); if (data.color != null && !BDFDB.ObjectUtils.is(data.color)) { if (data.color[0] < 30 && data.color[1] < 30 && data.color[2] < 30) data.color = BDFDB.ColorUtils.change(data.color, 30); else if (data.color[0] > 225 && data.color[1] > 225 && data.color[2] > 225) data.color = BDFDB.ColorUtils.change(data.color, -30); } data.inheritColor = inheritcolorinput.checked; let changed = false; if (Object.keys(data).every(key => data[key] == null || data[key] == false) && (changed = true)) BDFDB.DataUtils.remove(this, "channels", info.id); else if (!BDFDB.equals(olddata, data) && (changed = true)) BDFDB.DataUtils.save(data, this, "channels", info.id); if (changed) this.forceUpdateAll(); } }] }); } processChannelTextArea (instance, wrapper, returnvalue) { let channel = BDFDB.ReactUtils.getValue(instance, "props.channel"); if (channel) { var textarea = wrapper.querySelector("textarea"); if (!textarea) return; if (channel.type == 0 && instance.props.type == "normal" && !instance.props.disabled) { let data = this.getChannelData(channel.id, wrapper); textarea.setAttribute("placeholder", BDFDB.LanguageUtils.LanguageStringsFormat("TEXTAREA_PLACEHOLDER", `#${data.name || channel.name}`)); } BDFDB.ListenerUtils.remove(this, textarea); if (BDFDB.DataUtils.get(this, "settings", "changeInAutoComplete")) { 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, "autocompleteEditChannelsRow")) { BDFDB.ListenerUtils.stopEvent(e); this.swapWordWithMention(textarea); } } else if (autocompletemenu && (e.which == 38 || e.which == 40)) { let autocompleteitems = autocompletemenu.querySelectorAll(BDFDB.dotCN.autocompleteselectable + ":not(.autocompleteEditChannelsSelector)"); let selected = autocompletemenu.querySelector(BDFDB.dotCN.autocompleteselected); if (BDFDB.DOMUtils.containsClass(selected, "autocompleteEditChannelsSelector") || 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 && textarea.selectionStart == textarea.selectionEnd && textarea.selectionEnd == textarea.value.length) { BDFDB.TimeUtils.clear(textarea.EditChannelsAutocompleteTimeout); textarea.EditChannelsAutocompleteTimeout = BDFDB.TimeUtils.timeout(() => {this.addAutoCompleteMenu(textarea, channel);},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(".autocompleteEditChannels", ".autocompleteEditChannelsRow"); }); BDFDB.ListenerUtils.add(this, textarea, "click", e => { if (textarea.selectionStart == textarea.selectionEnd && textarea.selectionEnd == textarea.value.length) BDFDB.TimeUtils.timeout(() => {this.addAutoCompleteMenu(textarea, channel);}); }); } } } processAuditLog (instance, wrapper, returnvalue) { let channel = BDFDB.ReactUtils.getValue(instance, "props.log.options.channel"); if (channel) { let hooks = wrapper.querySelectorAll(`${BDFDB.dotCN.flexchild} > span${BDFDB.notCN.auditloguserhook}`); if (hooks.length > 0) this.changeChannel2(channel, hooks[0].firstChild); } } processInviteCard (instance, wrapper, returnvalue) { let invite = BDFDB.ReactUtils.getValue(instance, "props.invite"); if (invite && invite.inviter && invite.channel) { let channelname = wrapper.querySelector(BDFDB.dotCN.guildsettingsinvitechannelname); if (channelname) this.changeChannel2(invite.channel, channelname); } } processChannelCategoryItem (instance, wrapper, returnvalue) { if (instance.props.channel) { this.changeChannel(instance.props.channel, wrapper.querySelector(BDFDB.dotCN.categoryname), true); } } processChannelItem (instance, wrapper, returnvalue) { if (instance.props.channel) { this.changeChannel(instance.props.channel, wrapper.querySelector(BDFDB.dotCN.channelname), true); } } processHeaderBarContainer (instance, wrapper, returnvalue) { this.processHeaderBar(instance, wrapper); } processHeaderBar (instance, wrapper, returnvalue) { let channel_id = BDFDB.ReactUtils.getValue(instance, "props.channelId") || BDFDB.ReactUtils.getValue(instance, "_reactInternalFiber.return.memoizedProps.channelId"); if (channel_id) { let channelname = wrapper.querySelector(BDFDB.dotCN.channelheaderheaderbartitle); if (channelname) { let channel = BDFDB.LibraryModules.ChannelStore.getChannel(channel_id); if (channel) { if (channel.type == 0 || channel.type == 2) this.changeChannel(channel, channelname); else { if (channel.type == 1) channel = BDFDB.LibraryModules.UserStore.getUser(channel.recipients[0]) || channel; if (channelname.EditChannelsChangeObserver && typeof channelname.EditChannelsChangeObserver.disconnect == "function") channelname.EditChannelsChangeObserver.disconnect(); if (BDFDB.BDUtils.isPluginEnabled("EditUsers")) BDFDB.BDUtils.getPlugin("EditUsers").changeName(channel, channelname); else { channelname.style.removeProperty("color"); channelname.style.removeProperty("background"); BDFDB.DOMUtils.setText(channelname, channel.name || channel.username); } } } } } } processClickable (instance, wrapper, returnvalue) { if (!instance.props || !instance.props.className) return; else if (instance.props.tag == "span" && instance.props.className.indexOf(BDFDB.disCN.mentionwrapper) > -1 && instance.props.className.indexOf(BDFDB.disCN.mention) == -1) { let children = BDFDB.ReactUtils.getValue(instance, "_reactInternalFiber.memoizedProps.children"); if (children && typeof children[0] == "string") { let channelname = children[0].slice(1); let categoryname = BDFDB.ReactUtils.getValue(instance, "_reactInternalFiber.return.return.type.displayName") == "Tooltip" ? BDFDB.ReactUtils.getValue(instance, "_reactInternalFiber.return.return.memoizedProps.text") : null let channelid = BDFDB.LibraryModules.LastGuildStore.getGuildId(); let channels = channelid ? (BDFDB.LibraryModules.GuildChannelStore.getChannels(channelid)[0] || BDFDB.LibraryModules.GuildChannelStore.getChannels(channelid).SELECTABLE) : null; if (Array.isArray(channels)) for (let channel of channels) { if (channelname == channel.channel.name) { let category = categoryname ? BDFDB.LibraryModules.ChannelStore.getChannel(channel.channel.parent_id) : null; if (!category || category && categoryname == category.name) { this.changeMention(channel.channel, wrapper, category || {}); break; } } } } } else if (instance.props.tag == "div" && instance.props.className.indexOf(BDFDB.disCN.quickswitchresult) > -1) { let channel = BDFDB.ReactUtils.getValue(instance, "_reactInternalFiber.return.return.memoizedProps.channel"); if (channel) { this.changeChannel(channel, wrapper.querySelector(BDFDB.dotCN.quickswitchresultmatch)); if (channel.parent_id) this.changeChannel(BDFDB.LibraryModules.ChannelStore.getChannel(channel.parent_id), wrapper.querySelector(BDFDB.dotCN.quickswitchresultnote)); } } else if (instance.props.tag == "div" && instance.props.className.indexOf(BDFDB.disCN.autocompleterow) > -1) { let channel = BDFDB.ReactUtils.getValue(instance, "_reactInternalFiber.return.memoizedProps.channel"); if (channel) { this.changeChannel(channel, wrapper.querySelector(BDFDB.dotCN.marginleft4)); let category = BDFDB.ReactUtils.getValue(instance, "_reactInternalFiber.return.memoizedProps.category"); if (category) this.changeChannel(category, wrapper.querySelector(BDFDB.dotCN.autocompletedescription)); } } else if (instance.props.tag == "span" && instance.props.className.indexOf(BDFDB.disCN.messagespopoutchannelname) > -1) { let channel = BDFDB.ReactUtils.getValue(instance, "_reactInternalFiber.return.sibling.child.child.memoizedProps.channel"); if (channel) this.changeChannel2(channel, wrapper); } } processStandardSidebarView (instance, wrapper, returnvalue) { if (this.SettingsUpdated) { delete this.SettingsUpdated; this.forceUpdateAll(); } } changeAppTitle () { let channel = BDFDB.LibraryModules.ChannelStore.getChannel(BDFDB.LibraryModules.LastChannelStore.getChannelId()); let title = document.head.querySelector("title"); if (title && channel && channel.type != 1) { let data = this.getChannelData(channel.id, channel.parent_id, title); BDFDB.DOMUtils.setText(title, "@" + (data.name || channel.name)); } } changeChannel (info, channelname, hoverlistener = false) { if (!info || !channelname || !channelname.parentElement) return; var change = () => { if (channelname.EditChannelsChangeObserver && typeof channelname.EditChannelsChangeObserver.disconnect == "function") channelname.EditChannelsChangeObserver.disconnect(); let data = this.getChannelData(info.id, info.parent_id, channelname); if (data.name || data.color || channelname.parentElement.getAttribute("changed-by-editchannels")) { let isgradient = data.color && BDFDB.ObjectUtils.is(data.color); let color = this.chooseColor(channelname, data.color); if (isgradient) { channelname.style.setProperty("color", BDFDB.ColorUtils.convert(data.color[Object.keys(data.color)[0]], "RGBA"), "important"); BDFDB.DOMUtils.setText(channelname, BDFDB.DOMUtils.create(`${BDFDB.StringUtils.htmlEscape(data.name || info.name)}`)); } else { channelname.style.setProperty("color", color, "important"); BDFDB.DOMUtils.setText(channelname, data.name || info.name); } let iconparent = BDFDB.DOMUtils.containsClass(channelname, BDFDB.disCN.quickswitchresultmatch) ? channelname.parentElement.parentElement : channelname.parentElement; if (!BDFDB.DOMUtils.containsClass(channelname, BDFDB.disCN.autocompletedescription)) { let settings = BDFDB.DataUtils.get(this, "settings"); iconparent.querySelectorAll('svg [stroke]:not([stroke="none"]').forEach(icon => { let iconcolor = color && BDFDB.DOMUtils.getParent(BDFDB.dotCN.channelheadertitle, icon) ? BDFDB.ColorUtils.setAlpha(isgradient ? color[0] : color, 0.6) : (isgradient ? color[0] : color); if (!icon.getAttribute("oldstroke")) icon.setAttribute("oldstroke", icon.getAttribute("stroke")); icon.setAttribute("stroke", iconcolor && settings.changeChannelIcon ? iconcolor : icon.getAttribute("oldstroke"), "important"); icon.style.setProperty("stroke", iconcolor && settings.changeChannelIcon ? iconcolor : icon.getAttribute("oldstroke"), "important"); }); iconparent.querySelectorAll('svg [fill]:not([fill="none"]').forEach(icon => { let iconcolor = color && BDFDB.DOMUtils.getParent(BDFDB.dotCN.channelheadertitle, icon) ? BDFDB.ColorUtils.setAlpha(isgradient ? color[0] : color, 0.6) : (isgradient ? color[0] : color); if (!icon.getAttribute("oldfill")) icon.setAttribute("oldfill", icon.getAttribute("fill")); icon.setAttribute("fill", iconcolor && settings.changeChannelIcon ? iconcolor : icon.getAttribute("oldfill"), "important"); icon.style.setProperty("fill", iconcolor && settings.changeChannelIcon ? iconcolor : icon.getAttribute("oldfill"), "important"); }); let unread = iconparent.parentElement.querySelector(BDFDB.dotCN.channelunread); if (unread) unread.style.setProperty("background-color", color && settings.changeUnreadIndicator ? (isgradient ? color[0] : color) : null, "important"); } if (data.name || data.color) { channelname.parentElement.setAttribute("changed-by-editchannels", true); channelname.EditChannelsChangeObserver = new MutationObserver((changes, _) => { changes.forEach( (change, i) => { if (change.type == "childList" && change.addedNodes.length && change.target.tagName && (change.target.tagName == "SVG" || change.target.querySelector("svg")) || change.type == "attributes" && change.attributeName == "class" && change.target.className.length && change.target.className.indexOf("name") > -1 || change.type == "attributes" && change.attributeName == "style" && BDFDB.DOMUtils.containsClass(change.target, BDFDB.disCN.channelheaderheaderbartitle)) { channelname.EditChannelsChangeObserver.disconnect(); this.changeChannel(info, channelname); } } ); }); channelname.EditChannelsChangeObserver.observe(iconparent, {attributes:true, childList:true, subtree:true}); } else channelname.parentElement.removeAttribute("changed-by-editchannels"); } } change(); if (hoverlistener) { let wrapper = info.type == 4 ? BDFDB.DOMUtils.getParent(BDFDB.dotCN.categorywrapper, channelname) : BDFDB.DOMUtils.getParent(BDFDB.dotCN.channelwrapper, channelname); if (wrapper) { wrapper.removeEventListener("mouseover", wrapper.mouseoverListenerEditChannels); wrapper.removeEventListener("mouseout", wrapper.mouseoutListenerEditChannels); wrapper.mouseoverListenerEditChannels = () => { channelname.EditChannelsHovered = true; change(); }; wrapper.mouseoutListenerEditChannels = () => { delete channelname.EditChannelsHovered; change(); }; wrapper.addEventListener("mouseover", wrapper.mouseoverListenerEditChannels); wrapper.addEventListener("mouseout", wrapper.mouseoutListenerEditChannels); } } } changeChannel2 (info, channelname) { if (!info || !channelname || !channelname.parentElement) return; if (channelname.EditChannelsChangeObserver && typeof channelname.EditChannelsChangeObserver.disconnect == "function") channelname.EditChannelsChangeObserver.disconnect(); let data = this.getChannelData(info.id, info.parent_id, channelname); if (data.name || data.color || channelname.getAttribute("changed-by-editchannels")) { if (BDFDB.ObjectUtils.is(data.color)) { channelname.style.setProperty("color", BDFDB.ColorUtils.convert(data.color[Object.keys(data.color)[0]], "RGBA"), "important"); BDFDB.DOMUtils.setText(channelname, BDFDB.DOMUtils.create(`${BDFDB.StringUtils.htmlEscape("#" + (data.name || info.name))}`)); } else { channelname.style.setProperty("color", this.chooseColor(channelname, data.color), "important"); BDFDB.DOMUtils.setText(channelname, "#" + (data.name || info.name)); } if (data.name || data.color) { channelname.setAttribute("changed-by-editchannels", true); channelname.EditChannelsChangeObserver = new MutationObserver((changes, _) => { changes.forEach( (change, i) => { if (change.type == "childList" && change.addedNodes.length && change.target.tagName && (change.target.tagName == "SVG" || change.target.querySelector("svg")) || change.type == "attributes" && change.attributeName == "class" && change.target.className.length && change.target.className.indexOf("name") > -1) { channelname.EditChannelsChangeObserver.disconnect(); this.changeChannel2(info, channelname); } } ); }); channelname.EditChannelsChangeObserver.observe(channelname.parentElement, {attributes:true, childList:true, subtree:true}); } else channelname.removeAttribute("changed-by-editchannels"); } } changeMention (info, mention, categoryinfo) { if (!info || !mention || !mention.parentElement) return; if (mention.EditChannelsChangeObserver && typeof mention.EditChannelsChangeObserver.disconnect == "function") mention.EditChannelsChangeObserver.disconnect(); mention.removeEventListener("mouseover", mention.mouseoverListenerEditChannels); mention.removeEventListener("mouseout", mention.mouseoutListenerEditChannels); let data = this.getChannelData(info.id, info.parent_id, mention); let name = "#" + (data.name || info.name); let isgradient = data.color && BDFDB.ObjectUtils.is(data.color); let color = isgradient ? BDFDB.ColorUtils.createGradient(data.color) : BDFDB.ColorUtils.convert(data.color, "RGBA"); let color0_1 = isgradient ? BDFDB.ColorUtils.createGradient(BDFDB.ColorUtils.setAlpha(data.color, 0.1, "RGBA")) : BDFDB.ColorUtils.setAlpha(data.color, 0.1, "RGBA"); let color0_7 = isgradient ? BDFDB.ColorUtils.createGradient(BDFDB.ColorUtils.setAlpha(data.color, 0.7, "RGBA")) : BDFDB.ColorUtils.setAlpha(data.color, 0.7, "RGBA"); if (mention.EditChannelsHovered) colorHover(); else colorDefault(); mention.mouseoverListenerEditChannels = () => { mention.EditChannelsHovered = true; colorHover(); let categorydata = this.getChannelData(categoryinfo.id, null, mention); if (categorydata.name) BDFDB.TooltipUtils.create(mention, categorydata.name, {type:"top", selector:"EditChannels-tooltip", hide:true}); }; mention.mouseoutListenerEditChannels = () => { delete mention.EditChannelsHovered; colorDefault(); }; mention.addEventListener("mouseover", mention.mouseoverListenerEditChannels); mention.addEventListener("mouseout", mention.mouseoutListenerEditChannels); mention.EditChannelsChangeObserver = new MutationObserver((changes, _) => { mention.EditChannelsChangeObserver.disconnect(); this.changeMention(info, mention, categoryinfo); }); mention.EditChannelsChangeObserver.observe(mention, {attributes:true}); function colorDefault() { mention.style.setProperty("background", color0_1, "important"); if (isgradient) { mention.style.setProperty("color", BDFDB.ColorUtils.convert(data.color[Object.keys(data.color)[0]], "RGBA"), "important"); BDFDB.DOMUtils.setText(mention, BDFDB.DOMUtils.create(`${BDFDB.StringUtils.htmlEscape(name)}`)); } else { mention.style.setProperty("color", color, "important"); BDFDB.DOMUtils.setText(mention, name); } } function colorHover() { mention.style.setProperty("background", color0_7, "important"); mention.style.setProperty("color", data.color ? "#FFFFFF" : null, "important"); BDFDB.DOMUtils.setText(mention, name); } } chooseColor (channelname, color) { if (color && channelname) { let hovered = channelname.EditChannelsHovered; channelname = BDFDB.DOMUtils.containsClass(channelname, BDFDB.disCN.channelname) ? channelname.parentElement.parentElement : channelname; let classname = channelname.className ? channelname.className.toLowerCase() : ""; if (classname.indexOf("muted") > -1 || classname.indexOf("locked") > -1) color = BDFDB.ColorUtils.change(color, -0.5); else if (hovered || classname.indexOf("selected") > -1 || classname.indexOf("hovered") > -1 || classname.indexOf("unread") > -1 || classname.indexOf("connected") > -1) color = BDFDB.ColorUtils.change(color, 0.5); return BDFDB.ObjectUtils.is(color) ? color : BDFDB.ColorUtils.convert(color, "RGBA"); } return null; } getChannelData (id, categoryid, wrapper) { let data = BDFDB.DataUtils.load(this, "channels", id); let categorydata = categoryid ? BDFDB.DataUtils.load(this, "channels", categoryid) : null; if (!data && (!categorydata || (categorydata && !categorydata.color && !categorydata.inheritColor))) return {}; if (!data) data = {}; data.color = data.color ? data.color : (categorydata && categorydata.color && categorydata.inheritColor ? categorydata.color : null); let allenabled = true, settings = BDFDB.DataUtils.get(this, "settings"); for (let i in settings) if (!settings[i]) { allenabled = false; break; } if (allenabled) return data; let key = null; if (BDFDB.DOMUtils.getParent(BDFDB.dotCN.textareawrapchat, wrapper)) key = "changeInChatTextarea"; else if (BDFDB.DOMUtils.containsClass(wrapper, BDFDB.disCN.mentionwrapper)) key = "changeInMentions"; else if (BDFDB.DOMUtils.getParent(BDFDB.dotCN.guildchannels, wrapper)) key = "changeInChannelList"; else if (BDFDB.DOMUtils.getParent(BDFDB.dotCN.channelheaderheaderbar, wrapper)) key = "changeInChannelHeader"; else if (BDFDB.DOMUtils.getParent(BDFDB.dotCN.recentmentionspopout, wrapper)) key = "changeInRecentMentions"; else if (BDFDB.DOMUtils.getParent(BDFDB.dotCN.autocomplete, wrapper)) key = "changeInAutoComplete"; else if (BDFDB.DOMUtils.getParent(BDFDB.dotCN.auditlog, wrapper)) key = "changeInAuditLog"; else if (BDFDB.DOMUtils.getParent(BDFDB.dotCN.guildsettingsinvitecard, wrapper)) key = "changeInInviteLog"; else if (BDFDB.DOMUtils.getParent(BDFDB.dotCN.searchpopout, wrapper) || BDFDB.DOMUtils.getParent(BDFDB.dotCN.quickswitcher, wrapper)) key = "changeInSearchPopout"; return !key || settings[key] ? data : {}; } addAutoCompleteMenu (textarea, channel) { if (textarea.parentElement.querySelector(".autocompleteEditChannelsRow")) return; let words = textarea.value.split(/\s/); let lastword = words[words.length-1].trim(); if (lastword && lastword.length > 1 && lastword[0] == "#") { let channels = BDFDB.DataUtils.load(this, "channels"); if (!channels) return; let channelarray = []; for (let id in channels) if (channels[id].name) { let channel = BDFDB.LibraryModules.ChannelStore.getChannel(id); let category = channel && channel.parent_id ? BDFDB.LibraryModules.ChannelStore.getChannel(channel.parent_id) : null; let catdata = (category ? channels[category.id] : null) || {}; if (channel && channel.type == 0) channelarray.push(Object.assign({lowercasename:channels[id].name.toLowerCase(),lowercasecatname:(catdata && catdata.name ? catdata.name.toLowerCase() : null),channel,category,catdata},channels[id])); } channelarray = BDFDB.ArrayUtils.keySort(channelarray.filter(n => n.lowercasename.indexOf(lastword.toLowerCase().slice(1)) != -1 || (n.lowercasecatname && n.lowercasecatname.indexOf(lastword.toLowerCase().slice(1)) != -1)), "lowercasename"); if (channelarray.length) { let settings = BDFDB.DataUtils.get(this, "settings"); let autocompletemenu = textarea.parentElement.querySelector(BDFDB.dotCNS.autocomplete + BDFDB.dotCN.autocompleteinner), amount = 15; if (!autocompletemenu) { autocompletemenu = BDFDB.DOMUtils.create(`