//META{"name":"StalkerNotifications","website":"https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/StalkerNotifications","source":"https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/StalkerNotifications/StalkerNotifications.plugin.js"}*// class StalkerNotifications { getName () {return "StalkerNotifications";} getVersion () {return "1.1.2";} getAuthor () {return "DevilBro";} getDescription () {return "Lets you observe the status of people that aren't your friends.";} initConstructor () { this.patchModules = { "StandardSidebarView":"componentWillUnmount" }; this.stalkerOnlineList = {}; this.checkInterval = null; this.timeLog = []; this.timeLogModalMarkup = `

User LogIn/-Out Timelog

`; this.logEntryMarkup = `

`; this.dividerMarkup = `
`; this.css = ` .${this.name}-modal .log-time { width: 110px; } .${this.name}-modal .log-avatar { width: 35px; height: 35px; background-size: cover; background-position: center; border-radius: 50%; } .${this.name}-settings .avatar-list { display: flex; align-items: center; flex-wrap: wrap; } .${this.name}-settings .type-toast, .${this.name}-settings .type-desktop { border-radius: 3px; padding: 0 3px; } .${this.name}-settings .type-toast { background-color: #7289DA; } .${this.name}-settings .type-desktop { background-color: #43B581; } .${this.name}-settings .settings-avatar.desktop { border-color: #43B581; } .${this.name}-settings .settings-avatar { margin: 5px; width: 50px; height: 50px; background-size: cover; background-position: center; border: 5px solid #7289DA; border-radius: 50%; box-sizing: border-box; cursor: pointer; } .${this.name}-settings .settings-avatar.desktop { border-color: #43B581; } .${this.name}-settings .settings-avatar.disabled { border-color: #36393F; filter: grayscale(100%) brightness(50%); } .${this.name}-settings .settings-avatar ${BDFDB.dotCN.hovercardbutton} { position: relative; top: -10px; right: -25px; } .${this.name}-settings .settings-avatar:not(:hover) ${BDFDB.dotCN.hovercardbutton} { opacity: 1; }`; this.defaults = { settings: { muteOnDND: {value:false, description:"Do not notify me when I am DnD:"}, onlyOnOnline: {value:false, description:"Only notify me when a User logs in:"}, openOnClick: {value:false, description:"Open the DM when you click a Notification:"} }, notificationsounds: { toastonline: {value:{url:null,song:null,mute:false}}, toastoffline: {value:{url:null,song:null,mute:false}}, desktoponline: {value:{url:null,song:null,mute:false}}, desktopoffline: {value:{url:null,song:null,mute:false}} }, amounts: { checkInterval: {value:10, min:5, description:"Check Users every X seconds:"} } }; } getSettingsPanel () { if (!global.BDFDB || typeof BDFDB != "object" || !BDFDB.loaded || !this.started) return; let amounts = BDFDB.getAllData(this, "amounts"); let settings = BDFDB.getAllData(this, "settings"); let users = BDFDB.loadAllData(this, "users"); let notificationsounds = BDFDB.getAllData(this, "notificationsounds"); let settingshtml = `
${this.name}
`; settingshtml += `

Add User:

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

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

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

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

`; } for (let key in notificationsounds) { if (key.indexOf("desktop") == -1 || "Notification" in window) settingshtml += `
${key} notification sound:
Mute:
`; } settingshtml += `

Click on a Icon to toggle Notifications for that User:

`; if ("Notification" in window) settingshtml += `

Rightclick on a Icon to toggle Notifications for that User:

`; settingshtml += `
`; for (let id in users) { let user = this.UserUtils.getUser(id); if (user) settingshtml += this.createSettingsAvatarHtml(user, users[id]); } settingshtml += `
`; settingshtml += `

Batch set Users:

${"Notification" in window ? `` : ``}
`; settingshtml += `

Timelog of LogIns/-Outs:

`; settingshtml += `
`; let settingspanel = BDFDB.htmlToElement(settingshtml); BDFDB.initElements(settingspanel, this); BDFDB.addEventListener(this, settingspanel, "click", ".btn-savesong", e => {this.saveAudio(settingspanel, e.currentTarget.getAttribute("option"));}) BDFDB.addEventListener(this, settingspanel, "click", ".mute-checkbox", e => { let option = e.currentTarget.getAttribute("option"); let notificationsound = BDFDB.getData(option, this, "notificationsounds"); notificationsound.mute = e.currentTarget.checked; BDFDB.saveData(option, notificationsound, this, "notificationsounds"); }); BDFDB.addEventListener(this, settingspanel, "mouseenter", ".settings-avatar", e => { let user = this.UserUtils.getUser(e.currentTarget.getAttribute("user-id")); let data = BDFDB.loadData(user.id, "EditUsers", "users") || {}; BDFDB.createTooltip(data.name ? data.name : user.username, e.currentTarget, {type:"top"}); }); BDFDB.addEventListener(this, settingspanel, "contextmenu", ".settings-avatar", e => { if (!("Notification" in window)) return; let desktopoff = !BDFDB.containsClass(e.currentTarget, "desktop"); let id = e.currentTarget.getAttribute("user-id"); BDFDB.removeClass(e.currentTarget, "disabled"); BDFDB.toggleClass(e.currentTarget, "desktop", desktopoff); BDFDB.saveData(id, {"desktop":desktopoff,"disabled":false}, this, "users"); }); BDFDB.addEventListener(this, settingspanel, "click", ".settings-avatar", e => { if (BDFDB.containsClass(e.target, "remove-user")) return; let disableoff = !BDFDB.containsClass(e.currentTarget, "disabled"); let id = e.currentTarget.getAttribute("user-id"); BDFDB.removeClass(e.currentTarget, "desktop"); BDFDB.toggleClass(e.currentTarget, "disabled", disableoff); BDFDB.saveData(id, {"desktop":false,"disabled":disableoff}, this, "users"); }); BDFDB.addEventListener(this, settingspanel, "click", ".remove-user", e => { BDFDB.removeData(e.currentTarget.parentElement.getAttribute("user-id"), this, "users"); BDFDB.removeEles(this.name + "-settings .settings-avatar"); let listhtml = `
`; let users = BDFDB.loadAllData(this, "users"); for (let id in users) { let user = this.UserUtils.getUser(id); if (user) listhtml += this.createSettingsAvatarHtml(user, users[id]); } listhtml += `
`; settingspanel.querySelector(".avatar-list").innerHTML = listhtml; }); BDFDB.addEventListener(this, settingspanel, "click", ".disable-all, .toast-all, .desktop-all", e => { let disableon = e.currentTarget.getAttribute("do-disable"); let desktopon = e.currentTarget.getAttribute("do-desktop"); let users = BDFDB.loadAllData(this, "users"); settingspanel.querySelectorAll(".settings-avatar").forEach(avatar => { let id = avatar.getAttribute("user-id"); BDFDB.toggleClass(avatar, "disabled", disableon); BDFDB.toggleClass(avatar, "desktop", desktopon); users[id].desktop = desktopon ? true : false; users[id].disabled = disableon ? true : false; }); BDFDB.saveAllData(users, this, "users"); }); BDFDB.addEventListener(this, settingspanel, "click", ".btn-adduser", e => { let idinput = settingspanel.querySelector("#input-userid"); let user = this.UserUtils.getUser(idinput.value); if (user) { idinput.value = ""; BDFDB.saveData(user.id, {desktop:false,disabled:false}, this, "users"); BDFDB.removeEles(this.name + "-settings .settings-avatar"); let listhtml = `
`; let users = BDFDB.loadAllData(this, "users"); for (let id in users) { let user = this.UserUtils.getUser(id); if (user) listhtml += this.createSettingsAvatarHtml(user, users[id]); } listhtml += `
`; settingspanel.querySelector(".avatar-list").innerHTML = listhtml; } else BDFDB.showToast("Please enter a valid UserID.",{type:"error"}); }); BDFDB.addEventListener(this, settingspanel, "click", ".btn-timelog", () => {this.showTimeLog();}); return settingspanel; } //legacy load () {} start () { if (!global.BDFDB) global.BDFDB = {myPlugins:{}}; if (global.BDFDB && global.BDFDB.myPlugins && typeof global.BDFDB.myPlugins == "object") global.BDFDB.myPlugins[this.getName()] = this; var libraryScript = document.querySelector('head script[src="https://mwittrien.github.io/BetterDiscordAddons/Plugins/BDFDB.js"]'); if (!libraryScript || performance.now() - libraryScript.getAttribute("date") > 600000) { if (libraryScript) libraryScript.remove(); libraryScript = document.createElement("script"); libraryScript.setAttribute("type", "text/javascript"); libraryScript.setAttribute("src", "https://mwittrien.github.io/BetterDiscordAddons/Plugins/BDFDB.js"); libraryScript.setAttribute("date", performance.now()); libraryScript.addEventListener("load", () => {if (global.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) this.initialize();}); document.head.appendChild(libraryScript); } else if (global.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) this.initialize(); this.startTimeout = setTimeout(() => {this.initialize();}, 30000); } initialize () { if (global.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) { if (this.started) return; BDFDB.loadMessage(this); this.ChannelUtils = BDFDB.WebModules.findByProperties("getDMFromUserId"); this.ChannelSwitchUtils = BDFDB.WebModules.findByProperties("selectPrivateChannel"); this.PrivateChannelUtils = BDFDB.WebModules.findByProperties("openPrivateChannel"); this.UserMetaStore = BDFDB.WebModules.findByProperties("getStatus", "getOnlineFriendCount"); this.UserUtils = BDFDB.WebModules.findByProperties("getUsers"); for (let id in BDFDB.loadAllData(this, "users")) { this.stalkerOnlineList[id] = this.UserMetaStore.getStatus(id) != "offline"; } this.startInterval(); } 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) { clearInterval(this.checkInterval); BDFDB.unloadMessage(this); } } // begin of own functions saveAudio (settingspanel, option) { let successSavedAudio = (parsedurl, parseddata) => { if (parsedurl && parseddata) BDFDB.showToast(`Sound was saved successfully.`, {type:"success"}); let notificationsound = BDFDB.getData(option, this, "notificationsounds"); notificationsound.url = parsedurl; notificationsound.song = parseddata; BDFDB.saveData(option, notificationsound, this, "notificationsounds"); }; let url = settingspanel.querySelector(`.songInput[option="${option}"]`).value; if (url.length == 0) { BDFDB.showToast(`Sound file was removed.`, {type:"warn"}); successSavedAudio(url, url); } else if (url.indexOf("http") == 0) { require("request")(url, (error, response, result) => { if (response) { let type = response.headers["content-type"]; if (type && (type.indexOf("octet-stream") > -1 || type.indexOf("audio") > -1 || type.indexOf("video") > -1)) { successSavedAudio(url, url); return; } } BDFDB.showToast("Use a valid direct link to a video or audio source. They usually end on something like .mp3, .mp4 or .wav.", {type:"danger"}); }); } else { require("fs").readFile(url, (error, response) => { if (error) BDFDB.showToast("Could not fetch file. Please make sure the file exists.", {type:"danger"}); else successSavedAudio(url, `data:audio/mpeg;base64,${response.toString("base64")}`); }); } } createSettingsAvatarHtml (user, settings) { let data = BDFDB.loadData(user.id, "EditUsers", "users") || {}; return `
`; } processStandardSidebarView (instance, wrapper) { if (this.SettingsUpdated) { delete this.SettingsUpdated; this.startInterval(); } } startInterval () { clearInterval(this.checkInterval); this.checkInterval = setInterval(() => { let settings = BDFDB.getAllData(this, "settings"); let users = BDFDB.loadAllData(this, "users"); for (let id in users) { let online = this.UserMetaStore.getStatus(id) != "offline"; let user = this.UserUtils.getUser(id); if (user && !user.disabled && this.stalkerOnlineList[id] != online) { this.timeLog.push({user, online, time: new Date()}); if (!(settings.onlyOnOnline && !online) && !(settings.muteOnDND && BDFDB.getUserStatus() == "dnd")) { let data = BDFDB.loadData(user.id, "EditUsers", "users") || {}; let string = `${BDFDB.encodeToHTML(data.name ? data.name : user.username)} is ${online ? "online" : "offline"}.`; let avatar = data.removeIcon ? "" : (data.url ? data.url : BDFDB.getUserAvatar(user.id)); let openChannel = () => { if (settings.openOnClick) { let DMid = this.ChannelUtils.getDMFromUserId(user.id) if (DMid) this.ChannelSwitchUtils.selectPrivateChannel(DMid); else this.PrivateChannelUtils.openPrivateChannel(BDFDB.myData.id, user.id); require("electron").remote.getCurrentWindow().maximize(); } }; if (!user.desktop) { let toast = BDFDB.showToast(`
${string}
`, {html:true, timeout:5000, type:(online ? "success" : null), icon:false, selector:`stalkernotifications-${online ? "online" : "offline"}-toast`}); toast.addEventListener("click", openChannel); let notificationsound = BDFDB.getData(online ? "toastonline" : "toastoffline", this, "notificationsounds"); if (!notificationsound.mute && notificationsound.song) { let audio = new Audio(); audio.src = notificationsound.song; audio.play(); } } else { let notificationsound = BDFDB.getData(online ? "desktoponline" : "desktopoffline", this, "notificationsounds"); BDFDB.showDesktopNotification(string, {icon:avatar, timeout:5000, click:openChannel, silent:notificationsound.mute, sound:notificationsound.song}); } } } this.stalkerOnlineList[id] = online; } },BDFDB.getData("checkInterval", this, "amounts") * 1000); } showTimeLog () { let timeLogModal = BDFDB.htmlToElement(this.timeLogModalMarkup); let container = timeLogModal.querySelector(".entries"); if (!container) return; let logs = this.timeLog.slice(0).reverse(); for (let log of logs) { if (container.childElementCount) container.appendChild(BDFDB.htmlToElement(`
`)); let data = BDFDB.loadData(log.user.id, "EditUsers", "users") || {}; let entry = BDFDB.htmlToElement(this.logEntryMarkup); entry.querySelector(".log-time").innerText = `[${log.time.toLocaleTimeString()}]`; entry.querySelector(".log-avatar").style.setProperty("background-image", `url(${data.removeIcon ? "" : (data.url ? data.url : BDFDB.getUserAvatar(log.user.id))})`); entry.querySelector(".log-description").innerText = `${data.name || log.user.username} is ${log.online ? "online" : "offline"}.`; container.appendChild(entry) } BDFDB.appendModal(timeLogModal); } }