//META{"name":"PersonalPins","website":"https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/PersonalPins","source":"https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/PersonalPins/PersonalPins.plugin.js"}*// class PersonalPins { getName () {return "PersonalPins";} getDescription () {return "Similar to normal pins. Lets you save messages as notes for yourself.";} getVersion () {return "1.8.6";} getAuthor () {return "DevilBro";} constructor () { this.changelog = { "fixed":[["Embed Crash","Fixed issue where opening the pins could crash Discord because of old embed data"]], "improved":[["New Library Structure & React","Restructured my Library and switched to React rendering instead of DOM manipulation"]] }; this.patchedModules = { after: { HeaderBarContainer: "render" } }; } getSettingsPanel () { if (!global.BDFDB || typeof BDFDB != "object" || !BDFDB.loaded || !this.started) return; let settingspanel, settingsitems = []; settingsitems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsItem, { type: "Button", className: BDFDB.disCN.marginbottom8, color: BDFDB.LibraryComponents.Button.Colors.RED, label: "Delete all Notes", onClick: _ => { BDFDB.ModalUtils.confirm(this, "Are you sure you want to delete all pinned notes?", _ => { BDFDB.DataUtils.remove(this, "notes"); }); }, children: BDFDB.LanguageUtils.LanguageStrings.DELETE })); return settingspanel = 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); // REMOVE let pins = BDFDB.DataUtils.load(this, "pins"), notes = {}, olddata = 0, resolveddata = 0, loading = true, i = 0; for (let guild_id in pins) { notes[guild_id] = {}; for (let channel_id in pins[guild_id]) { notes[guild_id][channel_id] = {}; for (let message_idPOS in pins[guild_id][channel_id]) olddata++; } } if (olddata > 0) BDFDB.ModalUtils.confirm(this, `${this.name} found ${olddata} instances of old saved notes. Press the "Okay" button to refetch the messages and update your old config. This will take around ${(olddata + 1) * 5} seconds.`, _ => { BDFDB.TimeUtils.timeout(_ => { BDFDB.DataUtils.remove(this, "pins"); BDFDB.DataUtils.save(notes, this, "notes"); loading = false; BDFDB.ModalUtils.confirm(this, `${resolveddata} of ${olddata} pins could be transfered to the new format.${olddata - resolveddata > 0 ? " " + (olddata - resolveddata) + " could not be transfered because the message could no longer be found in discords data." : ""}`, _ => {}); }, (olddata + 1) * 5000); for (let guild_id in pins) for (let channel_id in pins[guild_id]) for (let message_idPOS in pins[guild_id][channel_id]) { BDFDB.TimeUtils.timeout(_ => {if (loading) BDFDB.LibraryModules.APIUtils.get(BDFDB.DiscordConstants.Endpoints.MESSAGES(pins[guild_id][channel_id][message_idPOS].channel_id) + "?limit=1&around=" + pins[guild_id][channel_id][message_idPOS].id).then(result => {if (result && result.body && result.body[0] && result.body[0].id == pins[guild_id][channel_id][message_idPOS].id) { let note = {}; note.addedat = pins[guild_id][channel_id][message_idPOS].addedat; note.channel = JSON.stringify(BDFDB.LibraryModules.ChannelStore.getChannel(channel_id)); note.id = pins[guild_id][channel_id][message_idPOS].id; note.message = JSON.stringify(result.body[0]); note.pos = pins[guild_id][channel_id][message_idPOS].pos; note.timestamp = pins[guild_id][channel_id][message_idPOS].timestamp; notes[guild_id][channel_id][message_idPOS] = note; resolveddata++; }})}, i * 5000); i++; } }); 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.ModuleUtils.forceAllUpdates(this); BDFDB.PluginUtils.clear(this); } } // begin of own functions onMessageContextMenu (e) { if (e.instance.props.message && e.instance.props.channel && e.instance.props.target) { let {messagediv, pos} = this.getMessageAndPos(e.instance.props.target); if (!messagediv || pos == -1) return; let note = this.getNoteData(e.instance.props.message, e.instance.props.target, e.instance.props.channel); let [children, index] = BDFDB.ReactUtils.findChildren(e.returnvalue, {name:"MessagePinItem"}); const pinUnpinItem = BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ContextMenuItems.Item, { label: this.labels[note ? "context_unpinoption_text" : "context_pinoption_text"], hint: BDFDB.BDUtils.isPluginEnabled("MessageUtilities") ? BDFDB.BDUtils.getPlugin("MessageUtilities").getActiveShortcutString("__Note_Message") : null, action: _ => { BDFDB.ContextMenuUtils.close(e.instance); this.addMessageToNotes(e.instance.props.message, e.instance.props.target, e.instance.props.channel); } }); if (index > -1) children.splice(index, 0, pinUnpinItem); else children.push(pinUnpinItem); if (this.isNoteOutdated(note, e.instance.props.message)) { const updateItem = BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ContextMenuItems.Item, { label: this.labels.context_updateoption_text, action: _ => { BDFDB.ContextMenuUtils.close(e.instance); this.updateNoteData(note, e.instance.props.message); } }); if (index > -1) children.splice(index, 0, updateItem); else children.push(updateItem); } } } onMessageOptionPopout (e) { if (e.instance.props.message && e.instance.props.channel && e.instance.props.target) { let {messagediv, pos} = this.getMessageAndPos(e.instance.props.target); if (!messagediv || pos == -1) return; let note = this.getNoteData(e.instance.props.message, e.instance.props.target, e.instance.props.channel); let [children, index] = BDFDB.ReactUtils.findChildren(e.returnvalue, {props:[["label", [BDFDB.LanguageUtils.LanguageStrings.PIN, BDFDB.LanguageUtils.LanguageStrings.UNPIN]]]}); children.splice(index + 1, 0, BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ContextMenuItems.Item, { label: this.labels[note ? "context_unpinoption_text" : "popout_pinoption_text"], className: BDFDB.disCN.optionpopoutitem, action: _ => { this.addMessageToNotes(e.instance.props.message, e.instance.props.target, e.instance.props.channel); e.instance.props.onClose(); } })); if (this.isNoteOutdated(note, e.instance.props.message)) children.splice(index + 1, 0, BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ContextMenuItems.Item, { label: this.labels.context_updateoption_text, className: BDFDB.disCN.optionpopoutitem, action: _ => { this.updateNoteData(note, e.instance.props.message); e.instance.props.onClose(); } })); } } processHeaderBarContainer (e) { let [children, index] = BDFDB.ReactUtils.findChildren(BDFDB.ReactUtils.getValue(e.returnvalue, "props.toolbar"), {name: "FluxContainer(Search)"}); if (index > -1) children.splice(index, 0, BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.PopoutContainer, { children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TooltipContainer, { text: this.labels.popout_note_text, tooltipConfig: {type: "bottom"}, children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Clickable, { className: BDFDB.disCNS.channelheadericonwrapper + BDFDB.disCN.channelheadericonclickable, children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SvgIcon, { className: BDFDB.disCNS.channelheadericon, iconSVG: `` }) }) }), popoutClassName: BDFDB.disCN.messagespopoutwrap, animation: BDFDB.LibraryComponents.PopoutContainer.Animation.SCALE, position: BDFDB.LibraryComponents.PopoutContainer.Positions.BOTTOM, align: BDFDB.LibraryComponents.PopoutContainer.Align.RIGHT, width: 500, maxHeight: "calc(100vh - 100px)", onClose: instance => { BDFDB.DOMUtils.removeClass(instance.domElementRef.current, BDFDB.disCN.channelheadericonselected); }, renderPopout: instance => { BDFDB.DOMUtils.addClass(instance.domElementRef.current, BDFDB.disCN.channelheadericonselected); return this.createNotesPopout(instance); } })); } createNotesPopout (buttoninstance) { let tabkeys = ["channel", "server", "all"], sortkeys = ["notetime", "messagetime"]; buttoninstance.props.selectedFilter = buttoninstance.props.selectedFilter || this.getValue(tabkeys[0], "filter"); buttoninstance.props.selectedSort = buttoninstance.props.selectedSort || this.getValue(sortkeys[0], "sort"); buttoninstance.props.searchKey = buttoninstance.props.searchKey || ""; return [ BDFDB.ReactUtils.createElement("div", { className: BDFDB.disCNS.popoutheader + BDFDB.disCN.messagespopoutheader, style: { paddingBottom: 0 }, children: [ BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex, { children: [ BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, { className: BDFDB.disCNS.popouttitle + BDFDB.disCN.messagespopouttitle, children: this.labels.popout_note_text }), BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SearchBar, { query: buttoninstance.props.searchKey, onChange: value => { BDFDB.TimeUtils.clear(this.searchTimeout); this.searchTimeout = BDFDB.TimeUtils.timeout(_ => { buttoninstance.props.searchKey = value; BDFDB.ReactUtils.forceUpdate(buttoninstance.popout._owner.stateNode); }, 1000); }, onClear: _ => { buttoninstance.props.searchKey = ""; BDFDB.ReactUtils.forceUpdate(buttoninstance.popout._owner.stateNode); } }) ] }), BDFDB.ReactUtils.createElement("div", { className: BDFDB.disCN.tabbarheadercontainer, children: [ BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TabBar, { className: BDFDB.disCN.tabbarheader, itemClassName: BDFDB.disCN.tabbarheaderitem, type: BDFDB.LibraryComponents.TabBar.Types.TOP, selectedItem: buttoninstance.props.selectedFilter.value, items: tabkeys.map(key => this.getValue(key, "filter")), onItemSelect: key => { buttoninstance.props.selectedFilter = this.getValue(key, "filter"); BDFDB.ReactUtils.forceUpdate(buttoninstance.popout._owner.stateNode); } }), BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.QuickSelect, { spacing: -25, label: this.labels.popout_sort_text + ":", value: buttoninstance.props.selectedSort, options: sortkeys.map(key => this.getValue(key, "sort")), onChange: key => { buttoninstance.props.selectedSort = this.getValue(key, "sort"); BDFDB.ReactUtils.forceUpdate(buttoninstance.popout._owner.stateNode); } }) ] }) ] }), BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ScrollerVertical, { className: BDFDB.disCN.messagespopout, children: this.filterMessages(buttoninstance, buttoninstance.props.selectedFilter.value, buttoninstance.props.selectedSort.value, buttoninstance.props.searchKey.toUpperCase()) }) ]; } getValue (key, type) { return { label: this.labels[`popout_${type}_${key}_text`], value: key } } filterMessages (buttoninstance, filter, sort, searchkey) { let lighttheme = BDFDB.DiscordUtils.getTheme() == BDFDB.disCN.themelight; let messages = [], notes = BDFDB.DataUtils.load(this, "notes"), updatedata = false; for (let guild_id in notes) for (let channel_id in notes[guild_id]) for (let message_idPOS in notes[guild_id][channel_id]) { let message = JSON.parse(notes[guild_id][channel_id][message_idPOS].message); message.author = new BDFDB.DiscordObjects.User(message.author); message.timestamp = new BDFDB.DiscordObjects.Timestamp(message.timestamp); message.editedTimestamp = message.editedTimestamp && new BDFDB.DiscordObjects.Timestamp(message.editedTimestamp); for (let embed of message.embeds) { embed.color = typeof embed.color != "string" ? null : embed.color; embed.timestamp = embed.timestamp && new BDFDB.DiscordObjects.Timestamp(embed.timestamp); } message = new BDFDB.DiscordObjects.Message(message); let channel = notes[guild_id][channel_id][message_idPOS].channel && new BDFDB.DiscordObjects.Channel(JSON.parse(notes[guild_id][channel_id][message_idPOS].channel)); if (!channel) { channel = BDFDB.LibraryModules.ChannelStore.getChannel(message.channel_id); if (channel) { updatedata = true; notes[guild_id][channel_id][message_idPOS].channel = JSON.stringify(channel); } } messages.push({ note: notes[guild_id][channel_id][message_idPOS], channel_id, guild_id, message, channel, messagetime: notes[guild_id][channel_id][message_idPOS].timestamp, notetime: notes[guild_id][channel_id][message_idPOS].addedat }); } if (updatedata) BDFDB.DataUtils.save(notes, this, "notes"); let currentchannel = BDFDB.LibraryModules.ChannelStore.getChannel(BDFDB.LibraryModules.LastChannelStore.getChannelId()) || {}; switch (filter) { case "channel": messages = messages.filter(messagedata => messagedata.channel_id == currentchannel.id); break; case "server": messages = messages.filter(messagedata => messagedata.guild_id == (currentchannel.guild_id || "@me")); break; case "allservers": messages = messages; break; } if (searchkey) { let searchvalues = ["content", "author.username", "rawDescription", "author.name"]; messages = messages.filter(messagedata => searchvalues.some(key => this.containsSearchkey(messagedata.message, key, searchkey) || messagedata.message.embeds.some(embed => this.containsSearchkey(embed, key, searchkey)))); } BDFDB.ArrayUtils.keySort(messages, sort); messages.reverse(); return Object.keys(messages).length ? messages.map(messagedata => this.renderMessage(buttoninstance, messagedata.note, messagedata.message, messagedata.channel, filter)).flat(10).filter(n => n) : BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.MessagesPopoutComponents.EmptyStateCenter, { msg: BDFDB.LanguageUtils.LanguageStrings.AUTOCOMPLETE_NO_RESULTS_HEADER, image: lighttheme ? "/assets/03c7541028afafafd1a9f6a81cb7f149.svg" : "/assets/6793e022dc1b065b21f12d6df02f91bd.svg" }); } containsSearchkey (data, key, searchkey) { let value = BDFDB.ReactUtils.getValue(data, key); return value && value.toUpperCase().indexOf(searchkey) > -1 } renderMessage (buttoninstance, note, message, channel, filter) { if (!message || !channel) return null; let channelname = channel.name; let guild = channel.guild_id && BDFDB.LibraryModules.GuildStore.getGuild(channel.guild_id); let role = guild && BDFDB.LibraryModules.PermissionRoleUtils.getHighestRole(guild, message.author.id); if (role) message.colorString = role.colorString; if (filter != "channel" && !channelname && channel.recipients.length > 0) { for (let dmuser_id of channel.recipients) { channelname = channelname ? channelname + ", @" : channelname; channelname = channelname + BDFDB.LibraryModules.UserStore.getUser(dmuser_id).username; } } let separator = filter == "channel" ? null : BDFDB.ReactUtils.createElement("div", { className: BDFDB.disCN.messagespopoutchannelseparator, children: [ BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Clickable, { tag: "span", className: BDFDB.disCN.messagespopoutchannelname, onClick: _ => { BDFDB.LibraryModules.SelectChannelUtils.selectChannel(channel.guild_id, channel.id); }, children: channelname ? ((channel.guild_id ? "#" : "@") + channelname) : "???" }), filter == "all" ? BDFDB.ReactUtils.createElement("span", { className: BDFDB.disCN.messagespopoutguildname, children: channel.guild_id ? (BDFDB.LibraryModules.GuildStore.getGuild(channel.guild_id) || {}).name || BDFDB.LanguageUtils.LanguageStrings.GUILD_UNAVAILABLE_HEADER : BDFDB.LanguageUtils.LanguageStrings.DIRECT_MESSAGES }) : null ] }); return [separator, BDFDB.ReactUtils.createElement("div", { className: BDFDB.disCN.messagegroupwrapper, key: message.id, children: BDFDB.ReactUtils.createElement("div", { className: BDFDB.disCN.messagegroupwrapperoffsetcorrection, children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.MessageGroup, { cozyClassName: BDFDB.disCN.messagespopoutmessagegroupcozy, messages: [message], channel: channel, inlineAttachmentMedia: true, inlineEmbedMedia: true, renderEmbeds: true, renderEmbedsSmall: true, compact: false, developerMode: false, hasDivider: false, popoutPosition: BDFDB.LibraryComponents.PopoutContainer.Positions.LEFT, groupOption: _ => { return BDFDB.ReactUtils.createElement("div", { className: BDFDB.disCN.messagespopoutactionbuttons, children: [ BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Clickable, { className: BDFDB.disCN.messagespopoutjumpbutton, onClick: _ => { BDFDB.LibraryModules.SelectChannelUtils.selectChannel(channel.guild_id, channel.id, message.id); }, children: BDFDB.ReactUtils.createElement("div", { className: BDFDB.disCN.messagespopouttext, children: BDFDB.LanguageUtils.LanguageStrings.JUMP }) }), BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Clickable, { className: BDFDB.disCN.messagespopoutjumpbutton, onClick: _ => { if (message.content || message.attachments.length > 1) { let text = message.content || ""; for (let attachment of message.attachments) if (attachment.url) text += ((text ? "\n" : "") + attachment.url); BDFDB.LibraryRequires.electron.clipboard.write({text}); } else if (message.attachments.length == 1 && message.attachments[0].url) { BDFDB.LibraryRequires.request({url: message.attachments[0].url, encoding: null}, (error, response, buffer) => { if (buffer) { if (BDFDB.LibraryRequires.process.platform === "win32" || BDFDB.LibraryRequires.process.platform === "darwin") { BDFDB.LibraryRequires.electron.clipboard.write({image: BDFDB.LibraryRequires.electron.nativeImage.createFromBuffer(buffer)}); } else { let file = BDFDB.LibraryRequires.path.join(BDFDB.LibraryRequires.process.env["HOME"], "personalpinstemp.png"); BDFDB.LibraryRequires.fs.writeFileSync(file, buffer, {encoding: null}); BDFDB.LibraryRequires.electron.clipboard.write({image: file}); BDFDB.LibraryRequires.fs.unlinkSync(file); } } }); } }, children: BDFDB.ReactUtils.createElement("div", { className: BDFDB.disCN.messagespopouttext, children: BDFDB.LanguageUtils.LanguageStrings.COPY }) }), BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Button, { look: BDFDB.LibraryComponents.Button.Looks.BLANK, size: BDFDB.LibraryComponents.Button.Sizes.NONE, onClick: (e, instance) => { this.removeNoteData(note); BDFDB.ReactUtils.forceUpdate(buttoninstance.popout._owner.stateNode); }, children: BDFDB.ReactUtils.createElement("div", { className: BDFDB.disCN.messagespopoutclosebutton }) }) ] }); }, messageOption: false, canEdit: false, accessoriesClassName: BDFDB.disCN.messagespopoutaccessories }) }) })]; } addMessageToNotes (message, target, channel) { if (!message || !target) return; let {messagediv, pos} = this.getMessageAndPos(target); if (!messagediv || pos == -1) return; let notes = BDFDB.DataUtils.load(this, "notes"); channel = channel || BDFDB.LibraryModules.ChannelStore.getChannel(message.channel_id); let guild_id = channel.guild_id || "@me"; notes[guild_id] = notes[guild_id] || {}; notes[guild_id][channel.id] = notes[guild_id][channel.id] || {} if (!notes[guild_id][channel.id][message.id + "_" + pos]) { notes[guild_id][channel.id][message.id + "_" + pos] = { addedat: new Date().getTime(), channel: JSON.stringify(channel), id: message.id, message: JSON.stringify(message), pos: pos, timestamp: message.timestamp._i.getTime() }; BDFDB.DataUtils.save(notes, this, "notes"); BDFDB.NotificationUtils.toast(this.labels.toast_noteadd_text, {type: "success"}); } else this.removeNoteData(notes[guild_id][channel.id][message.id + "_" + pos]); } isNoteOutdated (note, message) { let notemessage = note && JSON.parse(note.message), keys = ["content", "embeds", "attachment"]; return notemessage && !BDFDB.equals(BDFDB.ObjectUtils.extract(notemessage, keys), BDFDB.ObjectUtils.extract(message, keys)); } getNoteData (message, target, channel) { if (!message || !target) return; let {messagediv, pos} = this.getMessageAndPos(target); if (!messagediv || pos == -1) return; channel = channel || BDFDB.LibraryModules.ChannelStore.getChannel(message.channel_id); let guild_id = channel.guild_id || "@me"; let notes = BDFDB.DataUtils.load(this, "notes"); return notes[guild_id] && notes[guild_id][channel.id] && notes[guild_id][channel.id][message.id + "_" + pos]; } updateNoteData (note, newmessage) { let message = JSON.parse(note.message); let channel = JSON.parse(note.channel); if (!message || !channel) return; let guild_id = channel.guild_id || "@me"; let notes = BDFDB.DataUtils.load(this, "notes"); notes[guild_id][channel.id][note.id + "_" + note.pos].message = JSON.stringify(newmessage); BDFDB.DataUtils.save(notes, this, "notes"); BDFDB.NotificationUtils.toast(this.labels.toast_noteupdate_text, {type: "info"}); } removeNoteData (note) { let message = JSON.parse(note.message); let channel = JSON.parse(note.channel); if (!message || !channel) return; let guild_id = channel.guild_id || "@me"; let notes = BDFDB.DataUtils.load(this, "notes"); delete notes[guild_id][channel.id][note.id + "_" + note.pos]; if (BDFDB.ObjectUtils.isEmpty(notes[guild_id][channel.id])) { delete notes[guild_id][channel.id]; if (BDFDB.ObjectUtils.isEmpty(notes[guild_id])) delete notes[guild_id]; } BDFDB.DataUtils.save(notes, this, "notes"); BDFDB.NotificationUtils.toast(this.labels.toast_noteremove_text, {type: "danger"}); } getMessageAndPos (target) { let messagediv = BDFDB.DOMUtils.getParent(BDFDB.dotCN.messagegroup + "> [aria-disabled]," + BDFDB.dotCN.messagegroup + "> * > [aria-disabled]", target); let pos = messagediv ? Array.from(messagediv.parentElement.childNodes).filter(n => n.nodeType != Node.TEXT_NODE).indexOf(messagediv) : -1; return {messagediv, pos}; } setLabelsByLanguage () { switch (BDFDB.LanguageUtils.getLanguage().id) { case "hr": //croatian return { popout_note_text: "Bilješke", popout_filter_channel_text: "Kanal", popout_filter_server_text: "Poslužavnik", popout_filter_all_text: "Svi poslužitelji", popout_sort_text: "Poredaj po", popout_sort_messagetime_text: "Vijesti-Datum", popout_sort_notetime_text: "Bilješka-Datum", context_pinoption_text: "Napominjemo poruku", context_updateoption_text: "Ažuriraj bilješku", context_unpinoption_text: "Uklonite bilješku", popout_pinoption_text: "Bilješka", toast_noteadd_text: "Poruka dodana u bilježnicu.", toast_noteupdate_text: "Poruka je ažurirana u bilježnici.", toast_noteremove_text: "Poruka uklonjena iz bilježnice." }; case "da": //danish return { popout_note_text: "Noter", popout_filter_channel_text: "Kanal", popout_filter_server_text: "Server", popout_filter_all_text: "Alle servere", popout_sort_text: "Sorter efter", popout_sort_messagetime_text: "Meddelelse-Dato", popout_sort_notetime_text: "Note-Dato", context_pinoption_text: "Noter besked", context_updateoption_text: "Opdater note", context_unpinoption_text: "Fjern note", popout_pinoption_text: "Noter", toast_noteadd_text: "Meddelelse tilføjet til notesbog.", toast_noteupdate_text: "Meddelelse opdateret i den notesbog.", toast_noteremove_text: "Meddelelse fjernet fra notesbog." }; case "de": //german return { popout_note_text: "Notizen", popout_filter_channel_text: "Kanal", popout_filter_server_text: "Server", popout_filter_all_text: "Alle Server", popout_sort_text: "Sortieren nach", popout_sort_messagetime_text: "Nachrichten-Datum", popout_sort_notetime_text: "Notiz-Datum", context_pinoption_text: "Nachricht notieren", context_updateoption_text: "Notiz aktualisieren", context_unpinoption_text: "Notiz entfernen", popout_pinoption_text: "Notieren", toast_noteadd_text: "Nachricht zum Notizbuch hinzugefügt.", toast_noteupdate_text: "Nachricht im Notizbuch aktualisiert.", toast_noteremove_text: "Nachricht aus dem Notizbuch entfernt." }; case "es": //spanish return { popout_note_text: "Notas", popout_filter_channel_text: "Canal", popout_filter_server_text: "Servidor", popout_filter_all_text: "Todos los servidores", popout_sort_text: "Ordenar por", popout_sort_messagetime_text: "Mensaje-Fecha", popout_sort_notetime_text: "Nota-Fecha", context_pinoption_text: "Anotar mensaje", context_updateoption_text: "Actualiza la nota", context_unpinoption_text: "Eliminar la nota", popout_pinoption_text: "Anotar", toast_noteadd_text: "Mensaje agregado al cuaderno.", toast_noteupdate_text: "Mensaje actualizado en el cuaderno.", toast_noteremove_text: "Mensaje eliminado del cuaderno." }; case "fr": //french return { popout_note_text: "Notes", popout_filter_channel_text: "Canal", popout_filter_server_text: "Serveur", popout_filter_all_text: "Tous les serveurs", popout_sort_text: "Trier par", popout_sort_messagetime_text: "Message-Date", popout_sort_notetime_text: "Note-Date", context_pinoption_text: "Noter le message", context_updateoption_text: "Mettre à jour la note", context_unpinoption_text: "Enlevez la note", popout_pinoption_text: "Noter", toast_noteadd_text: "Message ajouté au cahier.", toast_noteupdate_text: "Message mis à jour dans le cahier.", toast_noteremove_text: "Message supprimé du cahier." }; case "it": //italian return { popout_note_text: "Note", popout_filter_channel_text: "Canale", popout_filter_server_text: "Server", popout_filter_all_text: "Tutti i server", popout_sort_text: "Ordina per", popout_sort_messagetime_text: "Messaggio-Data", popout_sort_notetime_text: "Nota-Data", context_pinoption_text: "Annotare il messaggio", context_updateoption_text: "Aggiorna la nota", context_unpinoption_text: "Rimuovi la nota", popout_pinoption_text: "Annotare", toast_noteadd_text: "Messaggio aggiunto al blocco note.", toast_noteupdate_text: "Messaggio aggiornato nel blocco note.", toast_noteremove_text: "Messaggio rimosso dal blocco note." }; case "nl": //dutch return { popout_note_text: "Notities", popout_filter_channel_text: "Kanaal", popout_filter_server_text: "Server", popout_filter_all_text: "Alle servers", popout_sort_text: "Sorteer op", popout_sort_messagetime_text: "Bericht-Datum", popout_sort_notetime_text: "Notitie-Datum", context_pinoption_text: "Noteer bericht", context_updateoption_text: "Update de notitie", context_unpinoption_text: "Verwijder de notitie", popout_pinoption_text: "Noteer", toast_noteadd_text: "Bericht toegevoegd aan notitieblok.", toast_noteupdate_text: "Bericht bijgewerkt in het notitieblok.", toast_noteremove_text: "Bericht verwijderd van notitieblok." }; case "no": //norwegian return { popout_note_text: "Notatene", popout_filter_channel_text: "Kanal", popout_filter_server_text: "Server", popout_filter_all_text: "Alle servere", popout_sort_text: "Sorter etter", popout_sort_messagetime_text: "Melding-Dato", popout_sort_notetime_text: "Merknad-Dato", context_pinoption_text: "Notat ned meldingen", context_updateoption_text: "Oppdater notatet", context_unpinoption_text: "Fjern notatet", popout_pinoption_text: "Notere", toast_noteadd_text: "Melding lagt til i notisboken.", toast_noteupdate_text: "Melding oppdatert i notisbok.", toast_noteremove_text: "Melding fjernet fra notatboken." }; case "pl": //polish return { popout_note_text: "Notatki", popout_filter_channel_text: "Kanał", popout_filter_server_text: "Serwer", popout_filter_all_text: "Wszystkie serwery", popout_sort_text: "Sortuj według", popout_sort_messagetime_text: "Wiadomość-Data", popout_sort_notetime_text: "Notatka-Data", context_pinoption_text: "Notuj wiadomość", context_updateoption_text: "Zaktualizuj notatkę", context_unpinoption_text: "Usuń notatkę", popout_pinoption_text: "Notuj", toast_noteadd_text: "Wiadomość została dodana do notatnika.", toast_noteupdate_text: "Wiadomość zaktualizowana w notatniku.", toast_noteremove_text: "Wiadomość została usunięta z notatnika." }; case "pt-BR": //portuguese (brazil) return { popout_note_text: "Notas", popout_filter_channel_text: "Canal", popout_filter_server_text: "Servidor", popout_filter_all_text: "Todos os servidores", popout_sort_text: "Ordenar por", popout_sort_messagetime_text: "Mensagem-Data", popout_sort_notetime_text: "Nota-Data", context_pinoption_text: "Anote a mensagem", context_updateoption_text: "Atualize a nota", context_unpinoption_text: "Remova a nota", popout_pinoption_text: "Anotar", toast_noteadd_text: "Mensagem adicionada ao caderno.", toast_noteupdate_text: "Mensagem atualizada no caderno.", toast_noteremove_text: "Mensagem removida do caderno." }; case "fi": //finnish return { popout_note_text: "Muistiinpanot", popout_filter_channel_text: "Kanava", popout_filter_server_text: "Palvelin", popout_filter_all_text: "Kaikki palvelimet", popout_sort_text: "Järjestä", popout_sort_messagetime_text: "Viesti-Päivämäärä", popout_sort_notetime_text: "Huomaa-Päivämäärä", context_pinoption_text: "Huomaa viesti", context_updateoption_text: "Päivitä muistiinpano", context_unpinoption_text: "Poista muistiinpano", popout_pinoption_text: "Huomaa", toast_noteadd_text: "Viesti lisätty muistikirjaan.", toast_noteupdate_text: "Viesti päivitetty muistikirjaan.", toast_noteremove_text: "Viesti poistettiin muistikirjaan." }; case "sv": //swedish return { popout_note_text: "Anteckningarna", popout_filter_channel_text: "Kanal", popout_filter_server_text: "Server", popout_filter_all_text: "Alla servrar", popout_sort_text: "Sortera efter", popout_sort_messagetime_text: "Meddelande-Datum", popout_sort_notetime_text: "Anteckningen-Datum", context_pinoption_text: "Anteckna meddelande", context_updateoption_text: "Uppdatera noten", context_unpinoption_text: "Ta bort noten", popout_pinoption_text: "Anteckna", toast_noteadd_text: "Meddelandet läggs till i anteckningsboken.", toast_noteupdate_text: "Meddelandet uppdateras i anteckningsboken.", toast_noteremove_text: "Meddelande borttaget från anteckningsboken." }; case "tr": //turkish return { popout_note_text: "Notlar", popout_filter_channel_text: "Kanal", popout_filter_server_text: "Sunucu", popout_filter_all_text: "Tüm Sunucular", popout_sort_text: "Göre sırala", popout_sort_messagetime_text: "Mesaj-Tarih", popout_sort_notetime_text: "Not-Tarih", context_pinoption_text: "Mesajı not alın", context_updateoption_text: "Notu güncelle", context_unpinoption_text: "Notu kaldırmak", popout_pinoption_text: "Not almak", toast_noteadd_text: "Mesaj not defteri'ya eklendi.", toast_noteupdate_text: "Mesaj defterde güncellendi.", toast_noteremove_text: "Mesaj not defteri'dan kaldırıldı." }; case "cs": //czech return { popout_note_text: "Poznámky", popout_filter_channel_text: "Kanál", popout_filter_server_text: "Server", popout_filter_all_text: "Všechny servery", popout_sort_text: "Seřazeno podle", popout_sort_messagetime_text: "Zpráva-datum", popout_sort_notetime_text: "Poznámka-datum", context_pinoption_text: "Poznámka dolů zprávu", context_updateoption_text: "Aktualizujte poznámku", context_unpinoption_text: "Odstraňte poznámku", popout_pinoption_text: "Poznámka dolů", toast_noteadd_text: "Zpráva byla přidána do notebooku.", toast_noteupdate_text: "Zpráva byla v notebooku aktualizována.", toast_noteremove_text: "Zpráva byla odebrána z notebooku." }; case "bg": //bulgarian return { popout_note_text: "бележките", popout_filter_channel_text: "Канал", popout_filter_server_text: "Сървър", popout_filter_all_text: "Всички сървъри", popout_sort_text: "Сортиране по", popout_sort_messagetime_text: "Съобщение-Дата", popout_sort_notetime_text: "Забележка-Дата", context_pinoption_text: "Oтбележете съобщението", context_updateoption_text: "Актуализирайте бележката", context_unpinoption_text: "Премахнете бележката", popout_pinoption_text: "Oтбележете", toast_noteadd_text: "Съобщението бе добавено към бележника.", toast_noteupdate_text: "Съобщението е актуализирано в бележника.", toast_noteremove_text: "Съобщението е премахнато от преносимия бележника." }; case "ru": //russian return { popout_note_text: "Заметки", popout_filter_channel_text: "Канал", popout_filter_server_text: "Cервер", popout_filter_all_text: "Все серверы", popout_sort_text: "Сортировать по", popout_sort_messagetime_text: "Сообщение-дата", popout_sort_notetime_text: "Заметки-Дата", context_pinoption_text: "Записывать вниз", context_updateoption_text: "Обновить заметку", context_unpinoption_text: "Удалить заметку", popout_pinoption_text: "Записывать", toast_noteadd_text: "Сообщение добавлено в блокнот.", toast_noteupdate_text: "Сообщение обновлено в записной блокнот.", toast_noteremove_text: "Сообщение удалено из записной блокнот." }; case "uk": //ukrainian return { popout_note_text: "Замітки", popout_filter_channel_text: "Канал", popout_filter_server_text: "Сервер", popout_filter_all_text: "Всі сервери", popout_sort_text: "Сортувати за", popout_sort_messagetime_text: "Повідомлення-дата", popout_sort_notetime_text: "Примітка-дата", context_pinoption_text: "Зверніть увагу на повідомлення", context_updateoption_text: "Оновіть нотатку", context_unpinoption_text: "Видаліть нотатку", popout_pinoption_text: "Занотуйте", toast_noteadd_text: "Повідомлення додається до ноутбука.", toast_noteupdate_text: "Повідомлення оновлено в ноутбука.", toast_noteremove_text: "Повідомлення видалено з ноутбука." }; case "ja": //japanese return { popout_note_text: "ノート", popout_filter_channel_text: "チャネル", popout_filter_server_text: "サーバ", popout_filter_all_text: "すべてのサーバー", popout_sort_text: "並び替え", popout_sort_messagetime_text: "メッセージ-日付", popout_sort_notetime_text: "注-日付", context_pinoption_text: "ノートダウンメッセージ", context_updateoption_text: "メモを更新する", context_unpinoption_text: "メモを削除", popout_pinoption_text: "書き留める", toast_noteadd_text: "ノートブックにメッセージが追加されました.", toast_noteupdate_text: "ノートブックで更新されたメッセージ.", toast_noteremove_text: "ノートブックからメッセージが削除されました." }; case "zh-TW": //chinese (traditional) return { popout_note_text: "筆記", popout_filter_channel_text: "渠道", popout_filter_server_text: "服務器", popout_filter_all_text: "所有服務器", popout_sort_text: "排序方式", popout_sort_messagetime_text: "消息-日期", popout_sort_notetime_text: "注-日期", context_pinoption_text: "記下下來的消息", context_updateoption_text: "更新說明", context_unpinoption_text: "刪除備註", popout_pinoption_text: "記下", toast_noteadd_text: "消息添加到筆記本.", toast_noteupdate_text: "消息在筆記本中更新.", toast_noteremove_text: "消息從筆記本中刪除." }; case "ko": //korean return { popout_note_text: "노트", popout_filter_channel_text: "채널", popout_filter_server_text: "섬기는 사람", popout_filter_all_text: "모든 서버", popout_sort_text: "정렬 기준", popout_sort_messagetime_text: "메시지-날짜", popout_sort_notetime_text: "주-날짜", context_pinoption_text: "메모 다운 메시지", context_updateoption_text: "메모 업데이트", context_unpinoption_text: "메모 삭제", popout_pinoption_text: "메모하다", toast_noteadd_text: "노트북에 메시지 추가됨.", toast_noteupdate_text: "노트북에서 메시지가 업데이트되었습니다.", toast_noteremove_text: "노트에서 메시지 삭제됨." }; default: //default: english return { popout_note_text: "Notes", popout_filter_channel_text: "Channel", popout_filter_server_text: "Server", popout_filter_all_text: "All Servers", popout_sort_text: "Sort by", popout_sort_messagetime_text: "Message-Date", popout_sort_notetime_text: "Note-Date", context_pinoption_text: "Note Message", context_updateoption_text: "Update Note", context_unpinoption_text: "Remove Note", popout_pinoption_text: "Note", toast_noteadd_text: "Message added to notebook.", toast_noteupdate_text: "Message updated in the notebook.", toast_noteremove_text: "Message removed from notebook." }; } } }