diff --git a/Plugins/DisplayLargeMessages/DisplayLargeMessages.plugin.js b/Plugins/DisplayLargeMessages/DisplayLargeMessages.plugin.js new file mode 100644 index 0000000000..7a67274e08 --- /dev/null +++ b/Plugins/DisplayLargeMessages/DisplayLargeMessages.plugin.js @@ -0,0 +1,96 @@ +//META{"name":"DisplayLargeMessages","authorId":"278543574059057154","invite":"Jx3TjNS","donate":"https://www.paypal.me/MircoWittrien","patreon":"https://www.patreon.com/MircoWittrien","website":"https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/DisplayLargeMessages","source":"https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/DisplayLargeMessages/DisplayLargeMessages.plugin.js"}*// + +var DisplayLargeMessages = (_ => { + var encodedMessages, updateTimeout; + + return class DisplayLargeMessages { + getName () {return "DisplayLargeMessages";} + + getVersion () {return "1.0.0";} + + getAuthor () {return "DevilBro";} + + getDescription () {return "Injects the contents of large messages that were sent by discord via 'message.txt'.";} + + constructor () { + this.patchedModules = { + after: { + Messages: "render" + } + }; + } + + initConstructor () { + encodedMessages = {}; + } + + // Legacy + load () {} + + start () { + if (!window.BDFDB) window.BDFDB = {myPlugins:{}}; + if (window.BDFDB && window.BDFDB.myPlugins && typeof window.BDFDB.myPlugins == "object") window.BDFDB.myPlugins[this.getName()] = this; + let 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 (window.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 (window.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) { + if (this.started) return; + BDFDB.PluginUtils.init(this); + + 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 (window.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) { + this.stopping = true; + + BDFDB.ModuleUtils.forceAllUpdates(this); + + BDFDB.PluginUtils.clear(this); + } + } + + + // Begin of own functions + + processMessages (e) { + let [children, index] = BDFDB.ReactUtils.findChildren(e.returnvalue, {props: ["message", "channel"]}); + if (index > -1) for (let ele of children) if (ele && ele.props && ele.props.message) { + let encodedContent = encodedMessages[ele.props.message.id]; + if (encodedContent != null) { + ele.props.message = new BDFDB.DiscordObjects.Message(Object.assign({}, ele.props.message, { + content: (ele.props.message.content && (ele.props.message.content + "\n\n") || "") + encodedContent + })); + ele.props.message.attachments = ele.props.message.attachments.filter(n => n.filename != "message.txt"); + } + else for (let attachment of ele.props.message.attachments) { + if (attachment.filename == "message.txt") BDFDB.LibraryRequires.request(attachment.url, (error, response, body) => { + encodedMessages[ele.props.message.id] = body || ""; + BDFDB.TimeUtils.clear(updateTimeout); + updateTimeout = BDFDB.TimeUtils.timeout(_ => { + BDFDB.ReactUtils.forceUpdate(e.instance); + }, 1000); + }); + } + } + } + } +})(); \ No newline at end of file diff --git a/Plugins/DisplayLargeMessages/README.md b/Plugins/DisplayLargeMessages/README.md new file mode 100644 index 0000000000..40d898fa14 --- /dev/null +++ b/Plugins/DisplayLargeMessages/README.md @@ -0,0 +1,12 @@ +# Display Large Messages - [![Downlad][download-badge]][download-link] [![Paypal][paypal-badge]][paypal-link] [![Patreon][patreon-badge]][patreon-link] + +[download-badge]: https://img.shields.io/badge/Download-Plugin-brightgreen.svg?logo=&style=flat +[download-link]: https://betterdiscord.net/ghdl?url=https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/DisplayLargeMessages/DisplayLargeMessages.plugin.js + +[paypal-badge]: https://img.shields.io/badge/Paypal-Donate!-%23003087.svg?logo=paypal&style=flat +[paypal-link]: https://paypal.me/MircoWittrien + +[patreon-badge]: https://img.shields.io/badge/Patreon-Support!-%23F96854.svg?logo=patreon&style=flat +[patreon-link]: https://patreon.com/MircoWittrien + +Injects the contents of large messages that were sent by discord via 'message.txt' diff --git a/Plugins/LastMessageDate/LastMessageDate.plugin.js b/Plugins/LastMessageDate/LastMessageDate.plugin.js index f22d61fd76..f5830942e5 100644 --- a/Plugins/LastMessageDate/LastMessageDate.plugin.js +++ b/Plugins/LastMessageDate/LastMessageDate.plugin.js @@ -1,444 +1,450 @@ //META{"name":"LastMessageDate","authorId":"278543574059057154","invite":"Jx3TjNS","donate":"https://www.paypal.me/MircoWittrien","patreon":"https://www.patreon.com/MircoWittrien","website":"https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/LastMessageDate","source":"https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/LastMessageDate/LastMessageDate.plugin.js"}*// -class LastMessageDate { - getName () {return "LastMessageDate";} +var LastMessageDate = (_ => { + var loadedUsers, requestedUsers; + + return class LastMessageDate { + getName () {return "LastMessageDate";} - getVersion () {return "1.1.4";} + getVersion () {return "1.1.4";} - getAuthor () {return "DevilBro";} + getAuthor () {return "DevilBro";} - getDescription () {return "Displays the Date of the last sent Message of a Member for the current Server/DM in the UserPopout and UserModal.";} + getDescription () {return "Displays the Date of the last sent Message of a Member for the current Server/DM in the UserPopout and UserModal.";} - constructor () { - this.changelog = { - "added":[["Days Ago","Added a days ago $daysago placeholder, to check how it works read the guide in the settings"]], - "improved":[["New Library Structure & React","Restructured my Library and switched to React rendering instead of DOM manipulation"]] - }; + constructor () { + this.changelog = { + "added":[["Days Ago","Added a days ago $daysago placeholder, to check how it works read the guide in the settings"]], + "improved":[["New Library Structure & React","Restructured my Library and switched to React rendering instead of DOM manipulation"]] + }; - this.patchedModules = { - after: { - UserPopout: "render", - AnalyticsContext: "render" - } - }; - } - - initConstructor () { - this.loadedusers = {}; - this.requestedusers = {}; - - this.css = ` - ${BDFDB.dotCNS.userpopout + BDFDB.dotCN.userpopoutheadertext} { - margin-bottom: 8px; - } - .lastMessageDate + ${BDFDB.dotCN.userpopoutcustomstatus} { - margin-top: 4px; - } - ${BDFDB.dotCNS.themelight + BDFDB.dotCN.userpopoutheadernormal} .lastMessageDate { - color: #b9bbbe; - } - ${BDFDB.dotCNS.themelight + BDFDB.dotCN.userpopoutheader + BDFDB.notCN.userpopoutheadernormal} .lastMessageDate, - ${BDFDB.dotCNS.themedark + BDFDB.dotCN.userpopoutheader} .lastMessageDate { - color: hsla(0,0%,100%,.6); - } - ${BDFDB.dotCNS.themelight + BDFDB.dotCN.userprofiletopsectionnormal} .lastMessageDate { - color: hsla(216,4%,74%,.6); - } - ${BDFDB.dotCN.themelight} [class*='topSection']${BDFDB.notCN.userprofiletopsectionnormal} .lastMessageDate, - ${BDFDB.dotCN.themedark} [class*='topSection'] .lastMessageDate { - color: hsla(0,0%,100%,.6); - }`; - - - this.defaults = { - settings: { - addInUserPopout: {value:true, description:"Add in User Popouts:"}, - addInUserProfil: {value:true, description:"Add in User Profile Modal:"}, - displayTime: {value:true, description:"Display the Time in the Timestamp:"}, - displayDate: {value:true, description:"Display the Date in the Timestamp:"}, - cutSeconds: {value:false, description:"Cut off Seconds of the Time:"}, - forceZeros: {value:false, description:"Force leading Zeros:"}, - otherOrder: {value:false, description:"Show the Time before the Date:"} - }, - choices: { - lastMessageDateLang: {value:"$discord", description:"Last Message Date Format:"} - }, - formats: { - ownFormat: {value:"$hour:$minute:$second, $day.$month.$year", description:"Own Format:"} - } - }; - } - - getSettingsPanel (collapseStates = {}) { - if (!window.BDFDB || typeof BDFDB != "object" || !BDFDB.loaded || !this.started) return; - let settings = BDFDB.DataUtils.get(this, "settings"); - let choices = BDFDB.DataUtils.get(this, "choices"); - let formats = BDFDB.DataUtils.get(this, "formats"); - let settingsPanel, settingsItems = [], innerItems = []; - - for (let key in settings) settingsItems.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], - onChange: (e, instance) => { - BDFDB.ReactUtils.forceUpdate(BDFDB.ReactUtils.findOwner(BDFDB.ReactUtils.findOwner(instance, {name:"BDFDB_SettingsPanel", up:true}), {name:"BDFDB_Select", all:true, noCopies:true})); - } - })); - - settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormDivider, { - className: BDFDB.disCN.marginbottom8 - })); - - for (let key in choices) settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsSaveItem, { - className: BDFDB.disCN.marginbottom8, - type: "Select", - plugin: this, - keys: ["choices", key], - label: this.defaults.choices[key].description, - basis: "70%", - value: choices[key], - options: BDFDB.ObjectUtils.toArray(BDFDB.ObjectUtils.map(this.languages, (lang, id) => {return {value:id, label:lang.name}})), - searchable: true, - optionRenderer: lang => { - return BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex, { - align: BDFDB.LibraryComponents.Flex.Align.CENTER, - children: [ - BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, { - grow: 0, - shrink: 0, - basis: "40%", - children: lang.label - }), - BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, { - grow: 0, - shrink: 0, - basis: "60%", - children: this.getTimestamp(this.languages[lang.value].id) - }) - ] - }); - }, - valueRenderer: lang => { - return BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex, { - align: BDFDB.LibraryComponents.Flex.Align.CENTER, - children: [ - BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, { - grow: 0, - shrink: 0, - children: lang.label - }), - BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, { - grow: 1, - shrink: 0, - basis: "70%", - children: this.getTimestamp(this.languages[lang.value].id) - }) - ] - }); - } - })); - - settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormDivider, { - className: BDFDB.disCN.marginbottom8 - })); - - for (let key in formats) settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsSaveItem, { - className: BDFDB.disCN.marginbottom8, - type: "TextInput", - plugin: this, - keys: ["formats", key], - label: this.defaults.formats[key].description, - basis: "70%", - value: formats[key], - onChange: (e, instance) => { - BDFDB.ReactUtils.forceUpdate(BDFDB.ReactUtils.findOwner(BDFDB.ReactUtils.findOwner(instance, {name:"BDFDB_SettingsPanel", up:true}), {name:"BDFDB_Select", all:true, noCopies:true})); - } - })); - - settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.CollapseContainer, { - title: "Placeholder Guide", - dividertop: true, - collapseStates: collapseStates, - children: [ - "$hour will be replaced with the current hour", - "$minute will be replaced with the current minutes", - "$second will be replaced with the current seconds", - "$msecond will be replaced with the current milliseconds", - "$timemode will change $hour to a 12h format and will be replaced with AM/PM", - "$year will be replaced with the current year", - "$month will be replaced with the current month", - "$day will be replaced with the current day", - "$monthnameL will be replaced with the monthname in long format based on the Discord Language", - "$monthnameS will be replaced with the monthname in short format based on the Discord Language", - "$weekdayL will be replaced with the weekday in long format based on the Discord Language", - "$weekdayS will be replaced with the weekday in short format based on the Discord Language", - "$daysago will be replaced with a string to tell you how many days ago the event occured. For Example: " + BDFDB.LanguageUtils.LanguageStringsFormat("ACTIVITY_FEED_USER_PLAYED_DAYS_AGO", 3) - ].map(string => { - return BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormText, { - type: BDFDB.LibraryComponents.FormComponents.FormTextTypes.DESCRIPTION, - children: string - }); - }) - })); - - return settingsPanel = BDFDB.PluginUtils.createSettingsPanel(this, settingsItems); - } - - // Legacy - load () {} - - start () { - if (!window.BDFDB) window.BDFDB = {myPlugins:{}}; - if (window.BDFDB && window.BDFDB.myPlugins && typeof window.BDFDB.myPlugins == "object") window.BDFDB.myPlugins[this.getName()] = this; - let 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 (window.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 (window.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) { - if (this.started) return; - BDFDB.PluginUtils.init(this); - - this.languages = Object.assign({"own":{name:"Own",id:"own",integrated:false,dic:false}}, BDFDB.LanguageUtils.languages); - - BDFDB.ModuleUtils.patch(this, BDFDB.LibraryModules.DispatchApiUtils, "dirtyDispatch", {after: e => { - if (BDFDB.ObjectUtils.is(e.methodArguments[0]) && e.methodArguments[0].type == BDFDB.DiscordConstants.ActionTypes.MESSAGE_CREATE && e.methodArguments[0].message) { - let message = e.methodArguments[0].message; - let guildId = message.guild_id || message.channel_id; - if (guildId && this.loadedusers[guildId] && this.loadedusers[guildId][message.author.id]) this.loadedusers[guildId][message.author.id] = new Date(message.timestamp); + this.patchedModules = { + after: { + UserPopout: "render", + AnalyticsContext: "render" } - }}); - - 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 (window.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) { - this.stopping = true; - - BDFDB.ModuleUtils.forceAllUpdates(this); - - BDFDB.PluginUtils.clear(this); - } - } - - - // Begin of own functions - - processUserPopout (e) { - if (e.instance.props.user && BDFDB.DataUtils.get(this, "settings", "addInUserPopout")) { - let [children, index] = BDFDB.ReactUtils.findChildren(e.returnvalue, {name: "CustomStatus"}); - if (index > -1) this.injectDate(e.instance, children, 2, e.instance.props.user); - } - } - - processAnalyticsContext (e) { - if (typeof e.returnvalue.props.children == "function" && e.instance.props.section == "Profile Modal" && BDFDB.DataUtils.get(this, "settings", "addInUserProfil")) { - let renderChildren = e.returnvalue.props.children; - e.returnvalue.props.children = (...args) => { - let renderedChildren = renderChildren(...args); - let [children, index] = BDFDB.ReactUtils.findChildren(renderedChildren, {name: ["DiscordTag", "ColoredFluxTag"]}); - if (index > -1) this.injectDate(e.instance, children, 1, children[index].props.user); - return renderedChildren; }; } - } - injectDate (instance, children, index, user) { - if (!BDFDB.ArrayUtils.is(children) || !user || user.discriminator == "0000") return; - let guildid = BDFDB.LibraryModules.LastGuildStore.getGuildId(); - let isguild = !!guildid; - guildid = guildid || BDFDB.LibraryModules.LastChannelStore.getChannelId(); - if (!guildid) return; - if (!this.loadedusers[guildid]) this.loadedusers[guildid] = {}; - if (!this.requestedusers[guildid]) this.requestedusers[guildid] = {}; - if (!BDFDB.ArrayUtils.is(this.requestedusers[guildid][user.id])) { - this.requestedusers[guildid][user.id] = [instance]; - BDFDB.LibraryModules.APIUtils.get((isguild ? BDFDB.DiscordConstants.Endpoints.SEARCH_GUILD(guildid) : BDFDB.DiscordConstants.Endpoints.SEARCH_CHANNEL(guildid)) + "?author_id=" + user.id).then(result => { - if (typeof result.body.retry_after != "number") { - if (result.body.messages && Array.isArray(result.body.messages[0])) { - for (let message of result.body.messages[0]) if (message.hit && message.author.id == user.id) { - this.loadedusers[guildid][user.id] = new Date(message.timestamp); + initConstructor () { + loadedUsers = {}; + requestedUsers = {}; + + this.css = ` + ${BDFDB.dotCNS.userpopout + BDFDB.dotCN.userpopoutheadertext} { + margin-bottom: 8px; + } + .lastMessageDate + ${BDFDB.dotCN.userpopoutcustomstatus} { + margin-top: 4px; + } + ${BDFDB.dotCNS.themelight + BDFDB.dotCN.userpopoutheadernormal} .lastMessageDate { + color: #b9bbbe; + } + ${BDFDB.dotCNS.themelight + BDFDB.dotCN.userpopoutheader + BDFDB.notCN.userpopoutheadernormal} .lastMessageDate, + ${BDFDB.dotCNS.themedark + BDFDB.dotCN.userpopoutheader} .lastMessageDate { + color: hsla(0,0%,100%,.6); + } + ${BDFDB.dotCNS.themelight + BDFDB.dotCN.userprofiletopsectionnormal} .lastMessageDate { + color: hsla(216,4%,74%,.6); + } + ${BDFDB.dotCN.themelight} [class*='topSection']${BDFDB.notCN.userprofiletopsectionnormal} .lastMessageDate, + ${BDFDB.dotCN.themedark} [class*='topSection'] .lastMessageDate { + color: hsla(0,0%,100%,.6); + }`; + + + this.defaults = { + settings: { + addInUserPopout: {value:true, description:"Add in User Popouts:"}, + addInUserProfil: {value:true, description:"Add in User Profile Modal:"}, + displayTime: {value:true, description:"Display the Time in the Timestamp:"}, + displayDate: {value:true, description:"Display the Date in the Timestamp:"}, + cutSeconds: {value:false, description:"Cut off Seconds of the Time:"}, + forceZeros: {value:false, description:"Force leading Zeros:"}, + otherOrder: {value:false, description:"Show the Time before the Date:"} + }, + choices: { + lastMessageDateLang: {value:"$discord", description:"Last Message Date Format:"} + }, + formats: { + ownFormat: {value:"$hour:$minute:$second, $day.$month.$year", description:"Own Format:"} + } + }; + } + + getSettingsPanel (collapseStates = {}) { + if (!window.BDFDB || typeof BDFDB != "object" || !BDFDB.loaded || !this.started) return; + let settings = BDFDB.DataUtils.get(this, "settings"); + let choices = BDFDB.DataUtils.get(this, "choices"); + let formats = BDFDB.DataUtils.get(this, "formats"); + let settingsPanel, settingsItems = [], innerItems = []; + + for (let key in settings) settingsItems.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], + onChange: (e, instance) => { + BDFDB.ReactUtils.forceUpdate(BDFDB.ReactUtils.findOwner(BDFDB.ReactUtils.findOwner(instance, {name:"BDFDB_SettingsPanel", up:true}), {name:"BDFDB_Select", all:true, noCopies:true})); + } + })); + + settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormDivider, { + className: BDFDB.disCN.marginbottom8 + })); + + for (let key in choices) settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsSaveItem, { + className: BDFDB.disCN.marginbottom8, + type: "Select", + plugin: this, + keys: ["choices", key], + label: this.defaults.choices[key].description, + basis: "70%", + value: choices[key], + options: BDFDB.ObjectUtils.toArray(BDFDB.ObjectUtils.map(this.languages, (lang, id) => {return {value:id, label:lang.name}})), + searchable: true, + optionRenderer: lang => { + return BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex, { + align: BDFDB.LibraryComponents.Flex.Align.CENTER, + children: [ + BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, { + grow: 0, + shrink: 0, + basis: "40%", + children: lang.label + }), + BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, { + grow: 0, + shrink: 0, + basis: "60%", + children: this.getTimestamp(this.languages[lang.value].id) + }) + ] + }); + }, + valueRenderer: lang => { + return BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex, { + align: BDFDB.LibraryComponents.Flex.Align.CENTER, + children: [ + BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, { + grow: 0, + shrink: 0, + children: lang.label + }), + BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, { + grow: 1, + shrink: 0, + basis: "70%", + children: this.getTimestamp(this.languages[lang.value].id) + }) + ] + }); + } + })); + + settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormDivider, { + className: BDFDB.disCN.marginbottom8 + })); + + for (let key in formats) settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsSaveItem, { + className: BDFDB.disCN.marginbottom8, + type: "TextInput", + plugin: this, + keys: ["formats", key], + label: this.defaults.formats[key].description, + basis: "70%", + value: formats[key], + onChange: (e, instance) => { + BDFDB.ReactUtils.forceUpdate(BDFDB.ReactUtils.findOwner(BDFDB.ReactUtils.findOwner(instance, {name:"BDFDB_SettingsPanel", up:true}), {name:"BDFDB_Select", all:true, noCopies:true})); + } + })); + + settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.CollapseContainer, { + title: "Placeholder Guide", + dividertop: true, + collapseStates: collapseStates, + children: [ + "$hour will be replaced with the current hour", + "$minute will be replaced with the current minutes", + "$second will be replaced with the current seconds", + "$msecond will be replaced with the current milliseconds", + "$timemode will change $hour to a 12h format and will be replaced with AM/PM", + "$year will be replaced with the current year", + "$month will be replaced with the current month", + "$day will be replaced with the current day", + "$monthnameL will be replaced with the monthname in long format based on the Discord Language", + "$monthnameS will be replaced with the monthname in short format based on the Discord Language", + "$weekdayL will be replaced with the weekday in long format based on the Discord Language", + "$weekdayS will be replaced with the weekday in short format based on the Discord Language", + "$daysago will be replaced with a string to tell you how many days ago the event occured. For Example: " + BDFDB.LanguageUtils.LanguageStringsFormat("ACTIVITY_FEED_USER_PLAYED_DAYS_AGO", 3) + ].map(string => { + return BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormText, { + type: BDFDB.LibraryComponents.FormComponents.FormTextTypes.DESCRIPTION, + children: string + }); + }) + })); + + return settingsPanel = BDFDB.PluginUtils.createSettingsPanel(this, settingsItems); + } + + // Legacy + load () {} + + start () { + if (!window.BDFDB) window.BDFDB = {myPlugins:{}}; + if (window.BDFDB && window.BDFDB.myPlugins && typeof window.BDFDB.myPlugins == "object") window.BDFDB.myPlugins[this.getName()] = this; + let 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 (window.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 (window.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) { + if (this.started) return; + BDFDB.PluginUtils.init(this); + + this.languages = Object.assign({"own":{name:"Own",id:"own",integrated:false,dic:false}}, BDFDB.LanguageUtils.languages); + + BDFDB.ModuleUtils.patch(this, BDFDB.LibraryModules.DispatchApiUtils, "dirtyDispatch", {after: e => { + if (BDFDB.ObjectUtils.is(e.methodArguments[0]) && e.methodArguments[0].type == BDFDB.DiscordConstants.ActionTypes.MESSAGE_CREATE && e.methodArguments[0].message) { + let message = e.methodArguments[0].message; + let guildId = message.guild_id || message.channel_id; + if (guildId && loadedUsers[guildId] && loadedUsers[guildId][message.author.id]) { + loadedUsers[guildId][message.author.id] = new Date(message.timestamp); } } - else this.loadedusers[guildid][user.id] = null; - for (let queredinstance of this.requestedusers[guildid][user.id]) BDFDB.ReactUtils.forceUpdate(queredinstance); - } - else { - delete this.requestedusers[guildid][user.id]; - BDFDB.TimeUtils.timeout(_ => {this.injectDate(instance, children, index, user);}, result.body.retry_after + 500); - } - }); - } - else if (this.loadedusers[guildid][user.id] === undefined) this.requestedusers[guildid][user.id].push(instance); - else children.splice(index, 0, BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextScroller, { - className: "lastMessageDate " + BDFDB.disCN.textrow, - children: this.labels.lastmessage_text.replace("{{time}}", this.loadedusers[guildid][user.id] ? this.getTimestamp(this.languages[BDFDB.DataUtils.get(this, "choices", "lastMessageDateLang")].id, this.loadedusers[guildid][user.id]) : "---") - })); - } - - getTimestamp (languageid, time) { - let timeobj = time ? time : new Date(); - if (typeof time == "string") timeobj = new Date(time); - if (timeobj.toString() == "Invalid Date") timeobj = new Date(parseInt(time)); - if (timeobj.toString() == "Invalid Date") return; - let settings = BDFDB.DataUtils.get(this, "settings"), timestring = ""; - if (languageid != "own") { - let timestamp = []; - if (settings.displayDate) timestamp.push(timeobj.toLocaleDateString(languageid)); - if (settings.displayTime) timestamp.push(settings.cutSeconds ? this.cutOffSeconds(timeobj.toLocaleTimeString(languageid)) : timeobj.toLocaleTimeString(languageid)); - if (settings.otherOrder) timestamp.reverse(); - timestring = timestamp.length > 1 ? timestamp.join(", ") : (timestamp.length > 0 ? timestamp[0] : ""); - if (timestring && settings.forceZeros) timestring = this.addLeadingZeros(timestring); - } - else { - let ownformat = BDFDB.DataUtils.get(this, "formats", "ownFormat"); - languageid = BDFDB.LanguageUtils.getLanguage().id; - let hour = timeobj.getHours(), minute = timeobj.getMinutes(), second = timeobj.getSeconds(), msecond = timeobj.getMilliseconds(), day = timeobj.getDate(), month = timeobj.getMonth()+1, timemode = "", daysago = Math.round((new Date() - timeobj)/(1000*60*60*24)); - if (ownformat.indexOf("$timemode") > -1) { - timemode = hour >= 12 ? "PM" : "AM"; - hour = hour % 12; - hour = hour ? hour : 12; + }}); + + BDFDB.ModuleUtils.forceAllUpdates(this); } - timestring = ownformat - .replace("$hour", settings.forceZeros && hour < 10 ? "0" + hour : hour) - .replace("$minute", minute < 10 ? "0" + minute : minute) - .replace("$second", second < 10 ? "0" + second : second) - .replace("$msecond", settings.forceZeros ? (msecond < 10 ? "00" + msecond : (msecond < 100 ? "0" + msecond : msecond)) : msecond) - .replace("$timemode", timemode) - .replace("$weekdayL", timeobj.toLocaleDateString(languageid,{weekday: "long"})) - .replace("$weekdayS", timeobj.toLocaleDateString(languageid,{weekday: "short"})) - .replace("$monthnameL", timeobj.toLocaleDateString(languageid,{month: "long"})) - .replace("$monthnameS", timeobj.toLocaleDateString(languageid,{month: "short"})) - .replace("$daysago", daysago > 0 ? BDFDB.LanguageUtils.LanguageStringsFormat("ACTIVITY_FEED_USER_PLAYED_DAYS_AGO", daysago) : BDFDB.LanguageUtils.LanguageStrings.SEARCH_SHORTCUT_TODAY) - .replace("$day", settings.forceZeros && day < 10 ? "0" + day : day) - .replace("$month", settings.forceZeros && month < 10 ? "0" + month : month) - .replace("$year", timeobj.getFullYear()) - .trim().split(" ").filter(n => n).join(" "); - } - return timestring; - } - - cutOffSeconds (timestring) { - return timestring.replace(/(.{1,2}:.{1,2}):.{1,2}(.*)/, "$1$2").replace(/(.{1,2}\..{1,2})\..{1,2}(.*)/, "$1$2").replace(/(.{1,2} h .{1,2} min) .{1,2} s(.*)/, "$1$2"); - } - - addLeadingZeros (timestring) { - let chararray = timestring.split(""); - let numreg = /[0-9]/; - for (let i = 0; i < chararray.length; i++) { - if (!numreg.test(chararray[i-1]) && numreg.test(chararray[i]) && !numreg.test(chararray[i+1])) chararray[i] = "0" + chararray[i]; + else console.error(`%c[${this.getName()}]%c`, "color: #3a71c1; font-weight: 700;", "", "Fatal Error: Could not load BD functions!"); } - return chararray.join(""); - } - setLabelsByLanguage () { - switch (BDFDB.LanguageUtils.getLanguage().id) { - case "hr": //croatian - return { - lastmessage_text: "Posljednja poruka dana {{time}}" - }; - case "da": //danish - return { - lastmessage_text: "Sidste besked den {{time}}" - }; - case "de": //german - return { - lastmessage_text: "Letzte Nachricht am {{time}}" - }; - case "es": //spanish - return { - lastmessage_text: "Último mensaje el {{time}}" - }; - case "fr": //french - return { - lastmessage_text: "Dernier message le {{time}}" - }; - case "it": //italian - return { - lastmessage_text: "Ultimo messaggio il {{time}}" - }; - case "nl": //dutch - return { - lastmessage_text: "Laatste bericht op {{time}}" - }; - case "no": //norwegian - return { - lastmessage_text: "Siste melding på {{time}}" - }; - case "pl": //polish - return { - lastmessage_text: "Ostatnia wiadomość z {{time}}" - }; - case "pt-BR": //portuguese (brazil) - return { - lastmessage_text: "Última mensagem em {{time}}" - }; - case "fi": //finnish - return { - lastmessage_text: "Viimeisin viesti {{time}}" - }; - case "sv": //swedish - return { - lastmessage_text: "Senaste meddelandet den {{time}}" - }; - case "tr": //turkish - return { - lastmessage_text: "Son mesajı {{time}}" - }; - case "cs": //czech - return { - lastmessage_text: "Poslední zpráva dne {{time}}" - }; - case "bg": //bulgarian - return { - lastmessage_text: "Последно съобщение на {{time}}" - }; - case "ru": //russian - return { - lastmessage_text: "Последнее сообщение {{time}}" - }; - case "uk": //ukrainian - return { - lastmessage_text: "Останнє повідомлення {{time}}" - }; - case "ja": //japanese - return { - lastmessage_text: "{{time}} 最後のメッセージ" - }; - case "zh-TW": //chinese (traditional) - return { - lastmessage_text: "最後消息於 {{time}}" - }; - case "ko": //korean - return { - lastmessage_text: "{{time}} 마지막 메시지" - }; - default: //default: english - return { - lastmessage_text: "Last message on {{time}}" + stop () { + if (window.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) { + this.stopping = true; + + BDFDB.ModuleUtils.forceAllUpdates(this); + + BDFDB.PluginUtils.clear(this); + } + } + + + // Begin of own functions + + processUserPopout (e) { + if (e.instance.props.user && BDFDB.DataUtils.get(this, "settings", "addInUserPopout")) { + let [children, index] = BDFDB.ReactUtils.findChildren(e.returnvalue, {name: "CustomStatus"}); + if (index > -1) this.injectDate(e.instance, children, 2, e.instance.props.user); + } + } + + processAnalyticsContext (e) { + if (typeof e.returnvalue.props.children == "function" && e.instance.props.section == "Profile Modal" && BDFDB.DataUtils.get(this, "settings", "addInUserProfil")) { + let renderChildren = e.returnvalue.props.children; + e.returnvalue.props.children = (...args) => { + let renderedChildren = renderChildren(...args); + let [children, index] = BDFDB.ReactUtils.findChildren(renderedChildren, {name: ["DiscordTag", "ColoredFluxTag"]}); + if (index > -1) this.injectDate(e.instance, children, 1, children[index].props.user); + return renderedChildren; }; + } + } + + injectDate (instance, children, index, user) { + if (!BDFDB.ArrayUtils.is(children) || !user || user.discriminator == "0000") return; + let guildid = BDFDB.LibraryModules.LastGuildStore.getGuildId(); + let isguild = !!guildid; + guildid = guildid || BDFDB.LibraryModules.LastChannelStore.getChannelId(); + if (!guildid) return; + if (!loadedUsers[guildid]) loadedUsers[guildid] = {}; + if (!requestedUsers[guildid]) requestedUsers[guildid] = {}; + if (!BDFDB.ArrayUtils.is(requestedUsers[guildid][user.id])) { + requestedUsers[guildid][user.id] = [instance]; + BDFDB.LibraryModules.APIUtils.get((isguild ? BDFDB.DiscordConstants.Endpoints.SEARCH_GUILD(guildid) : BDFDB.DiscordConstants.Endpoints.SEARCH_CHANNEL(guildid)) + "?author_id=" + user.id).then(result => { + if (typeof result.body.retry_after != "number") { + if (result.body.messages && Array.isArray(result.body.messages[0])) { + for (let message of result.body.messages[0]) if (message.hit && message.author.id == user.id) { + loadedUsers[guildid][user.id] = new Date(message.timestamp); + } + } + else loadedUsers[guildid][user.id] = null; + for (let queredinstance of requestedUsers[guildid][user.id]) BDFDB.ReactUtils.forceUpdate(queredinstance); + } + else { + delete requestedUsers[guildid][user.id]; + BDFDB.TimeUtils.timeout(_ => {this.injectDate(instance, children, index, user);}, result.body.retry_after + 500); + } + }); + } + else if (loadedUsers[guildid][user.id] === undefined) requestedUsers[guildid][user.id].push(instance); + else children.splice(index, 0, BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextScroller, { + className: "lastMessageDate " + BDFDB.disCN.textrow, + children: this.labels.lastmessage_text.replace("{{time}}", loadedUsers[guildid][user.id] ? this.getTimestamp(this.languages[BDFDB.DataUtils.get(this, "choices", "lastMessageDateLang")].id, loadedUsers[guildid][user.id]) : "---") + })); + } + + getTimestamp (languageid, time) { + let timeobj = time ? time : new Date(); + if (typeof time == "string") timeobj = new Date(time); + if (timeobj.toString() == "Invalid Date") timeobj = new Date(parseInt(time)); + if (timeobj.toString() == "Invalid Date") return; + let settings = BDFDB.DataUtils.get(this, "settings"), timestring = ""; + if (languageid != "own") { + let timestamp = []; + if (settings.displayDate) timestamp.push(timeobj.toLocaleDateString(languageid)); + if (settings.displayTime) timestamp.push(settings.cutSeconds ? this.cutOffSeconds(timeobj.toLocaleTimeString(languageid)) : timeobj.toLocaleTimeString(languageid)); + if (settings.otherOrder) timestamp.reverse(); + timestring = timestamp.length > 1 ? timestamp.join(", ") : (timestamp.length > 0 ? timestamp[0] : ""); + if (timestring && settings.forceZeros) timestring = this.addLeadingZeros(timestring); + } + else { + let ownformat = BDFDB.DataUtils.get(this, "formats", "ownFormat"); + languageid = BDFDB.LanguageUtils.getLanguage().id; + let hour = timeobj.getHours(), minute = timeobj.getMinutes(), second = timeobj.getSeconds(), msecond = timeobj.getMilliseconds(), day = timeobj.getDate(), month = timeobj.getMonth()+1, timemode = "", daysago = Math.round((new Date() - timeobj)/(1000*60*60*24)); + if (ownformat.indexOf("$timemode") > -1) { + timemode = hour >= 12 ? "PM" : "AM"; + hour = hour % 12; + hour = hour ? hour : 12; + } + timestring = ownformat + .replace("$hour", settings.forceZeros && hour < 10 ? "0" + hour : hour) + .replace("$minute", minute < 10 ? "0" + minute : minute) + .replace("$second", second < 10 ? "0" + second : second) + .replace("$msecond", settings.forceZeros ? (msecond < 10 ? "00" + msecond : (msecond < 100 ? "0" + msecond : msecond)) : msecond) + .replace("$timemode", timemode) + .replace("$weekdayL", timeobj.toLocaleDateString(languageid,{weekday: "long"})) + .replace("$weekdayS", timeobj.toLocaleDateString(languageid,{weekday: "short"})) + .replace("$monthnameL", timeobj.toLocaleDateString(languageid,{month: "long"})) + .replace("$monthnameS", timeobj.toLocaleDateString(languageid,{month: "short"})) + .replace("$daysago", daysago > 0 ? BDFDB.LanguageUtils.LanguageStringsFormat("ACTIVITY_FEED_USER_PLAYED_DAYS_AGO", daysago) : BDFDB.LanguageUtils.LanguageStrings.SEARCH_SHORTCUT_TODAY) + .replace("$day", settings.forceZeros && day < 10 ? "0" + day : day) + .replace("$month", settings.forceZeros && month < 10 ? "0" + month : month) + .replace("$year", timeobj.getFullYear()) + .trim().split(" ").filter(n => n).join(" "); + } + return timestring; + } + + cutOffSeconds (timestring) { + return timestring.replace(/(.{1,2}:.{1,2}):.{1,2}(.*)/, "$1$2").replace(/(.{1,2}\..{1,2})\..{1,2}(.*)/, "$1$2").replace(/(.{1,2} h .{1,2} min) .{1,2} s(.*)/, "$1$2"); + } + + addLeadingZeros (timestring) { + let chararray = timestring.split(""); + let numreg = /[0-9]/; + for (let i = 0; i < chararray.length; i++) { + if (!numreg.test(chararray[i-1]) && numreg.test(chararray[i]) && !numreg.test(chararray[i+1])) chararray[i] = "0" + chararray[i]; + } + + return chararray.join(""); + } + + setLabelsByLanguage () { + switch (BDFDB.LanguageUtils.getLanguage().id) { + case "hr": //croatian + return { + lastmessage_text: "Posljednja poruka dana {{time}}" + }; + case "da": //danish + return { + lastmessage_text: "Sidste besked den {{time}}" + }; + case "de": //german + return { + lastmessage_text: "Letzte Nachricht am {{time}}" + }; + case "es": //spanish + return { + lastmessage_text: "Último mensaje el {{time}}" + }; + case "fr": //french + return { + lastmessage_text: "Dernier message le {{time}}" + }; + case "it": //italian + return { + lastmessage_text: "Ultimo messaggio il {{time}}" + }; + case "nl": //dutch + return { + lastmessage_text: "Laatste bericht op {{time}}" + }; + case "no": //norwegian + return { + lastmessage_text: "Siste melding på {{time}}" + }; + case "pl": //polish + return { + lastmessage_text: "Ostatnia wiadomość z {{time}}" + }; + case "pt-BR": //portuguese (brazil) + return { + lastmessage_text: "Última mensagem em {{time}}" + }; + case "fi": //finnish + return { + lastmessage_text: "Viimeisin viesti {{time}}" + }; + case "sv": //swedish + return { + lastmessage_text: "Senaste meddelandet den {{time}}" + }; + case "tr": //turkish + return { + lastmessage_text: "Son mesajı {{time}}" + }; + case "cs": //czech + return { + lastmessage_text: "Poslední zpráva dne {{time}}" + }; + case "bg": //bulgarian + return { + lastmessage_text: "Последно съобщение на {{time}}" + }; + case "ru": //russian + return { + lastmessage_text: "Последнее сообщение {{time}}" + }; + case "uk": //ukrainian + return { + lastmessage_text: "Останнє повідомлення {{time}}" + }; + case "ja": //japanese + return { + lastmessage_text: "{{time}} 最後のメッセージ" + }; + case "zh-TW": //chinese (traditional) + return { + lastmessage_text: "最後消息於 {{time}}" + }; + case "ko": //korean + return { + lastmessage_text: "{{time}} 마지막 메시지" + }; + default: //default: english + return { + lastmessage_text: "Last message on {{time}}" + }; + } } } -} +})(); \ No newline at end of file diff --git a/Plugins/PluginRepo/res/PluginList.txt b/Plugins/PluginRepo/res/PluginList.txt index 4ede7babb9..044562d685 100644 --- a/Plugins/PluginRepo/res/PluginList.txt +++ b/Plugins/PluginRepo/res/PluginList.txt @@ -105,6 +105,7 @@ https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/C https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/CopyRawMessage/CopyRawMessage.plugin.js https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/CreationDate/CreationDate.plugin.js https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/CustomQuoter/CustomQuoter.plugin.js +https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/DisplayLargeMessages/DisplayLargeMessages.plugin.js https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/DisplayServersAsChannels/DisplayServersAsChannels.plugin.js https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/EditChannels/EditChannels.plugin.js https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/EditServers/EditServers.plugin.js diff --git a/Plugins/README.md b/Plugins/README.md index 0d27c0970f..83d2e7bb83 100644 --- a/Plugins/README.md +++ b/Plugins/README.md @@ -17,6 +17,7 @@ - [Copy Raw Message](https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/CopyRawMessage) - Adds a entry in the contextmenu when you right click a message that allows you to copy the raw contents of a message. - [Creation Date](https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/CreationDate) - Displays the Creation Date of an Account in the UserPopout and UserModal. - [Custom Quoter](https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/CustomQuoter) - Let's you customize the output of the native quote feature of Discord + - [Display Large Messages](https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/DisplayLargeMessages) - Injects the contents of large messages that were sent by discord via 'message.txt' - [Display Servers As Channels](https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/DisplayServersAsChannels) - Display servers in a similar way as channels. - [Edit Channels](https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/EditChannels) - Allows the user to locally change the name and color of channels. - [Edit Servers](https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/EditServers) - Allows the user to locally change the name, initials, icon and color of servers.