//META{"name":"PinDMs"}*// class PinDMs { initConstructor () { this.pinDMEntryMarkup = `
`; this.unpinDMEntryMarkup = ` `; } getName () {return "PinDMs";} getDescription () {return "Allows you to pin DMs, making them appear at the top of your DM-list.";} getVersion () {return "1.2.4";} getAuthor () {return "DevilBro";} //legacy load () {} start () { var libraryScript = null; if (typeof BDFDB !== "object" || typeof BDFDB.isLibraryOutdated !== "function" || BDFDB.isLibraryOutdated()) { libraryScript = document.querySelector('head script[src="https://mwittrien.github.io/BetterDiscordAddons/Plugins/BDFDB.js"]'); if (libraryScript) libraryScript.remove(); libraryScript = document.createElement("script"); libraryScript.setAttribute("type", "text/javascript"); libraryScript.setAttribute("src", "https://mwittrien.github.io/BetterDiscordAddons/Plugins/BDFDB.js"); document.head.appendChild(libraryScript); } this.startTimeout = setTimeout(() => {this.initialize();}, 30000); if (typeof BDFDB === "object" && typeof BDFDB.isLibraryOutdated === "function") this.initialize(); else libraryScript.addEventListener("load", () => {this.initialize();}); } initialize () { if (typeof BDFDB === "object") { BDFDB.loadMessage(this); this.UserStore = BDFDB.WebModules.findByProperties("getUsers", "getUser"); this.ChannelUtils = BDFDB.WebModules.findByProperties("getDMFromUserId"); var observer = null; observer = new MutationObserver((changes, _) => { changes.forEach( (change, i) => { if (change.addedNodes) { change.addedNodes.forEach((node) => { if (node && node.nodeType == 1 && node.className.includes(BDFDB.disCN.contextmenu)) { this.onContextMenu(node); } }); } } ); }); BDFDB.addObserver(this, BDFDB.dotCN.appmount, {name:"dmContextObserver",instance:observer}, {childList: true}); $(BDFDB.dotCN.appmount) .on("click." + this.getName(), BDFDB.dotCNS.dmchannels + BDFDB.dotCNS.dmchannel, (e) => { let instance = BDFDB.getReactInstance(e.currentTarget); if (instance.return.return.return.memoizedProps.ispin) { let dmsscroller = document.querySelector(BDFDB.dotCNS.dmchannels + BDFDB.dotCN.scroller); if (dmsscroller) { this.oldSrollerPos = dmsscroller.scrollTop; setTimeout(() => {this.oldSrollerPos = null;},1000); } } }) .on("click." + this.getName(), BDFDB.dotCNS.dmchannels + BDFDB.dotCNS.dmchannel + BDFDB.dotCN.dmchannelclose, (e) => { let instance = BDFDB.getReactInstance(e.currentTarget); if (instance.return.return.return.return.return.memoizedProps.ispin) { e.stopPropagation(); e.preventDefault(); let dmsscroller = document.querySelector(BDFDB.dotCNS.dmchannels + BDFDB.dotCN.scroller); if (dmsscroller) { this.removePinnedDM(instance.return.return.return.return.return.memoizedProps.channel.id, BDFDB.getReactInstance(dmsscroller).return.return.return.memoizedProps.children); this.forceUpdateScroller(dmsscroller); } } }); setTimeout(() => {this.patchDMsScroller();},1000); } else { console.error(this.getName() + ": Fatal Error: Could not load BD functions!"); } } stop () { if (typeof BDFDB === "object") { let dmsscroller = document.querySelector(BDFDB.dotCNS.dmchannels + BDFDB.dotCN.scroller); if (dmsscroller) { let dms = BDFDB.getReactInstance(dmsscroller).return.return.return.memoizedProps.children; let amount = 0; let insertpoint = null; for (let i in dms) { let ele = dms[i]; if (ele && ele.pinned) { delete ele.pinned; if (ele.props.ispin) { if (ele.type == "header") insertpoint = i; amount++; } } } dms.splice(insertpoint, amount); this.forceUpdateScroller(dmsscroller); } BDFDB.unloadMessage(this); } } // begin of own functions changeLanguageStrings () { this.pinDMEntryMarkup = this.pinDMEntryMarkup.replace("REPLACE_context_pindm_text", this.labels.context_pindm_text); this.unpinDMEntryMarkup = this.unpinDMEntryMarkup.replace("REPLACE_context_unpindm_text", this.labels.context_unpindm_text); } onContextMenu (context) { if (!document.querySelector(BDFDB.dotCNS.guildselected + BDFDB.dotCN.friendsicon) || !context || !context.tagName || !context.parentElement || context.querySelector(".pindm-item") || context.querySelector(".unpindm-item")) return; var info = BDFDB.getKeyInformation({"node":context, "key":"user"}), ele = null; if (info && BDFDB.getKeyInformation({"node":context, "key":"handleClose"})) { ele = context.querySelectorAll(BDFDB.dotCN.contextmenuitem)[3]; } else { info = BDFDB.getKeyInformation({"node":context, "key":"channel"}); if (info && BDFDB.getKeyInformation({"node":context, "key":"handleChangeIcon"})) { ele = context.querySelectorAll(BDFDB.dotCN.contextmenuitem)[1]; } } if (ele) { let proccesContextMenu = (id) => { let pinnedDMs = BDFDB.loadAllData(this, "pinnedDMs"); if (typeof pinnedDMs[id] == "undefined") { $(this.pinDMEntryMarkup).insertBefore(ele) .on("click", (e) => { $(context).hide(); let dmsscroller = document.querySelector(BDFDB.dotCNS.dmchannels + BDFDB.dotCN.scroller); if (dmsscroller) { let dms = BDFDB.getReactInstance(dmsscroller).return.return.return.memoizedProps.children; let insertpoint = this.getInsertPoint(dms); this.addPinnedDM(id, dms, insertpoint); this.forceUpdateScroller(dmsscroller); pinnedDMs[id] = Object.keys(pinnedDMs).length; BDFDB.saveAllData(pinnedDMs, this, "pinnedDMs"); } }); } else { $(this.unpinDMEntryMarkup).insertBefore(ele) .on("click", (e) => { $(context).hide(); let dmsscroller = document.querySelector(BDFDB.dotCNS.dmchannels + BDFDB.dotCN.scroller); if (dmsscroller) { let dms = BDFDB.getReactInstance(dmsscroller).return.return.return.memoizedProps.children; this.removePinnedDM(id, dms); this.forceUpdateScroller(dmsscroller); } }); } BDFDB.updateContextPosition(context); }; let id = info.type ? info.id : this.ChannelUtils.getDMFromUserId(info.id); if (id) proccesContextMenu(id); else this.PrivateChannelUtils.ensurePrivateChannel(BDFDB.myData.id, info.id).then((id) => {proccesContextMenu(id)}); } } patchDMsScroller () { let addAllDMs = (dmsarray) => { let sortedDMs = this.sortAndUpdate(); if (sortedDMs.length > 0) { let insertpoint = this.getInsertPoint(dmsarray); for (let pos in sortedDMs) this.addPinnedDM(sortedDMs[pos], dmsarray, insertpoint); } }; BDFDB.WebModules.patch(BDFDB.WebModules.findByName("LazyScroller").prototype, "render", this, { before: (e) => { if (e.thisObject._reactInternalFiber.return.memoizedProps.privateChannelIds && !e.thisObject.props.PinDMsPatched) { e.thisObject.props.PinDMsPatched = true; addAllDMs(e.thisObject.props.children); } if (e.thisObject._reactInternalFiber.return.memoizedProps.privateChannelIds && this.oldSrollerPos != null) { e.thisObject.getScrollerNode().scrollTop = this.oldSrollerPos; } } }); let dmsscroller = document.querySelector(BDFDB.dotCNS.dmchannels + BDFDB.dotCN.scroller); if (dmsscroller) { addAllDMs(BDFDB.getReactInstance(dmsscroller).return.return.return.memoizedProps.children); this.forceUpdateScroller(dmsscroller); } } getInsertPoint (dms) { let insertpoint = null; for (let i in dms) { let ele = dms[i]; if (ele && ele.type == "header") { insertpoint = parseInt(i); if (!ele.pinned && !ele.props.ispin) { ele.pinned = true; let headerpin = Object.assign({},ele); headerpin.key = "pin" + headerpin.key; headerpin.props = {children:this.labels.header_pinneddms_text,ispin:true}; dms.splice(insertpoint,0,headerpin); } insertpoint++; break; } } return insertpoint; } addPinnedDM (id, dms, insertpoint) { for (let ele of dms) { if (ele && !ele.pinned && id == ele.key) { ele.pinned = true; let dmpin = Object.assign({ispin:true},ele); dmpin.key = "pin" + ele.key; dmpin.props = {channel:ele.props.channel,selected:ele.props.selected,ispin:true}; dms.splice(insertpoint,0,dmpin); } } } removePinnedDM (id, dms) { BDFDB.removeData(id, this, "pinnedDMs"); let existingDMs = this.sortAndUpdate(); let removepoint = null; for (let i in dms) { let ele = dms[i]; if (ele && ele.pinned && (id == ele.key || ("pin" + id) == ele.key)) { delete ele.pinned; if (ele.props.ispin) removepoint = parseInt(i); } } if (removepoint) { let offset = existingDMs.length ? 0 : 1; if (offset) delete dms[removepoint + offset].pinned; dms.splice(removepoint-offset,1+offset); } } sortAndUpdate () { let pinnedDMs = BDFDB.loadAllData(this, "pinnedDMs"); delete pinnedDMs[""]; let sortedDMs = [], existingDMs = [], sortDM = (id) => { if (typeof sortedDMs[pinnedDMs[id]] == "undefined") sortedDMs[pinnedDMs[id]] = id; else sortDM(sortedDMs, pinnedDMs[id]+1, id); }; for (let id in pinnedDMs) sortDM(id); sortedDMs = sortedDMs.filter(n => n); for (let pos in sortedDMs) { pinnedDMs[sortedDMs[pos]] = parseInt(pos); if (this.ChannelUtils.getChannel(sortedDMs[pos])) existingDMs.push(sortedDMs[pos]); } BDFDB.saveAllData(pinnedDMs, this, "pinnedDMs"); return existingDMs; } forceUpdateScroller (scroller) { scroller.scrollTop += 10; scroller.scrollTop -= 10; } setLabelsByLanguage () { switch (BDFDB.getDiscordLanguage().id) { case "hr": //croatian return { context_pindm_text: "Prikljucite Izravnu Poruku", context_unpindm_text: "Otključaj Izravnu Poruku", header_pinneddms_text: "Prikvačene Izravne Poruke" }; case "da": //danish return { context_pindm_text: "Fastgør PB", context_unpindm_text: "Frigør PB", header_pinneddms_text: "Pinned Privat Beskeder" }; case "de": //german return { context_pindm_text: "Direktnachricht anheften", context_unpindm_text: "Direktnachricht loslösen", header_pinneddms_text: "Gepinnte Direktnachrichten" }; case "es": //spanish return { context_pindm_text: "Anclar MD", context_unpindm_text: "Desanclar MD", header_pinneddms_text: "Mensajes Directos Fijados" }; case "fr": //french return { context_pindm_text: "Épingler MP", context_unpindm_text: "Désépingler MP", header_pinneddms_text: "Messages Prives Épinglés" }; case "it": //italian return { context_pindm_text: "Fissa il messaggio diretto", context_unpindm_text: "Togli il messaggio diretto", header_pinneddms_text: "Messaggi Diretti Aggiunti" }; case "nl": //dutch return { context_pindm_text: "PB pinnen", context_unpindm_text: "PB losmaken", header_pinneddms_text: "Vastgezette Persoonluke Berichten" }; case "no": //norwegian return { context_pindm_text: "Fest DM", context_unpindm_text: "Løsne DM", header_pinneddms_text: "Pinned Direktemeldinger" }; case "pl": //polish return { context_pindm_text: "Przypnij PW", context_unpindm_text: "Odepnij PW", header_pinneddms_text: "Prywatne Wiadomości Bezpośrednie" }; case "pt-BR": //portuguese (brazil) return { context_pindm_text: "Fixar MD", context_unpindm_text: "Desafixar MD", header_pinneddms_text: "Mensagens diretas fixadas" }; case "fi": //finnish return { context_pindm_text: "Kiinnitä yksityisviestit", context_unpindm_text: "Poista yksityisviestit", header_pinneddms_text: "Liitetyt yksityisviestit" }; case "sv": //swedish return { context_pindm_text: "Fäst DM", context_unpindm_text: "Lossa DM", header_pinneddms_text: "Inlagda Direktmeddelanden" }; case "tr": //turkish return { context_pindm_text: "DM'yi Sabitle", context_unpindm_text: "DM'yi Kaldır", header_pinneddms_text: "Direkt Mesajlar Sabitleyin" }; case "cs": //czech return { context_pindm_text: "Připnout PZ", context_unpindm_text: "Odepnout PZ", header_pinneddms_text: "Připojené Přímá Zpráva" }; case "bg": //bulgarian return { context_pindm_text: "Закачени ДС", context_unpindm_text: "Откачи ДС", header_pinneddms_text: "Свързани директни съобщения" }; case "ru": //russian return { context_pindm_text: "Закрепить ЛС", context_unpindm_text: "Открепить ЛС", header_pinneddms_text: "Прикрепленные Личные Сообщения" }; case "uk": //ukrainian return { context_pindm_text: "Закріпити ОП", context_unpindm_text: "Відкріпити ОП", header_pinneddms_text: "Прикріплені oсобисті повідомлення" }; case "ja": //japanese return { context_pindm_text: "DMピン", context_unpindm_text: "DMをピン止めする", header_pinneddms_text: "固定された直接メッセージ" }; case "zh-TW": //chinese (traditional) return { context_pindm_text: "引腳直接留言", context_unpindm_text: "分離直接消息", header_pinneddms_text: "固定私人信息" }; case "ko": //korean return { context_pindm_text: "비공개 메시지 고정", context_unpindm_text: "비공개 메시지 고정 해제", header_pinneddms_text: "고정 된 비공개 메시지" }; default: //default: english return { context_pindm_text: "Pin DM", context_unpindm_text: "Unpin DM", header_pinneddms_text: "Pinned Direct Messages" }; } } }