// ==UserScript==
// @name sarkyScript
// @namespace https://www.destiny.gg/
// @version 1.5
// @description Tracks and logs the actions of conductors (or other users)
// @match *://*.destiny.gg/embed/chat*
// @run-at document-start
// @grant none
// @icon 
// @license MIT
// ==/UserScript==
(function() {
'use strict';
const menuIcon64 = "";
let stored = {
nicks: [],
joinAlerts: "default",
quitAlerts: "default",
embedAlerts: "default",
history: [],
};
const defaultNicks = ["destiny", "righttobeararmslol", "jaydrvernanda", "cake", "cyver", "4thot", "lemmiwinks", "ninou", "dancantstream", "aestudio", "mrmouton", "lilypichu", "pizza", "rin_lux", "yky", "drt0", "chacha", "tommyk", "csarky", "thatluckycamper", "zlxb", "pagi"]
let lastMsg;
try {
const sarkyScript = localStorage.getItem("sarkyScript");
if (sarkyScript) {
stored = JSON.parse(sarkyScript);
}
} catch (e) {
console.error(e);
}
function saveStored() {
localStorage.setItem("sarkyScript", JSON.stringify(stored));
}
function convertToJSON(prefixedData) {
const firstSpaceIndex = prefixedData.indexOf(" ");
const prefix = prefixedData.substring(0, firstSpaceIndex);
const data = JSON.parse(prefixedData.substring(firstSpaceIndex + 1));
return { prefix, data }
}
function isMainFrame() {
try {
return window == window.top;
} catch (e) {
return true;
}
}
class Msg {
constructor(nick, msg, color, time, embed) {
this.container = document.createElement("div");
this.container.className = "msg-chat msg-user";
this.container.setAttribute("data-username", nick.toLowerCase());
this.container.style.cssText = `background: ${color}; color: white;`;
const msgTime = document.createElement("time");
const date = new Date(time);
msgTime.textContent = date.getHours().toString().padStart(2, "0") + ":" + date.getMinutes().toString().padStart(2, "0");
msgTime.className = "time";
msgTime.setAttribute("data-unixtimestamp", time);
const msgUser = document.createElement("a");
msgUser.className = "user";
msgUser.textContent = nick;
const msgCtrl = document.createElement("span");
msgCtrl.className = "ctrl";
msgCtrl.textContent = ": "
const msgText = document.createElement("span");
msgText.className = "text";
msgText.textContent = msg;
this.container.append(msgTime,msgUser,msgCtrl,msgText);
if (embed) {
const embedLink = document.createElement("a");
embedLink.style.className = "externallink bookmarklink";
if (isMainFrame()) embedLink.setAttribute("target", "_blank");
else embedLink.setAttribute("target", "_top");
const embedStr = `#${embed.platform}/${embed.id}`;
embedLink.href = "https://www.destiny.gg/bigscreen" + embedStr;
embedLink.textContent = embedStr;
this.container.appendChild(embedLink);
}
}
}
function addMenu() {
const menuBtnContainer = document.createElement("a");
menuBtnContainer.setAttribute("title", "sarkyScript");
menuBtnContainer.className = "chat-tool-btn";
const menuBtn = document.createElement("img");
menuBtn.className = "btn-icon";
menuBtn.src = menuIcon64;
const menuTitle = document.createElement("h1");
menuTitle.textContent = "sarkyScript";
const menuDesc = document.createElement("p");
menuDesc.innerHTML = "This section is for the custom option <br> Seperate the names with commas using no spaces";
menuBtnContainer.addEventListener("click", function(e) {
const menuDialog = document.createElement("dialog");
menuDialog.style.cssText = "background: #333; color: #999; width: 50rem; height: 40rem; position: relative;";
document.body.appendChild(menuDialog);
menuDialog.showModal();
class CheckBox {
constructor(option, alert) {
this.container = document.createElement("div");
this.container.style.cssText = "display: flex; justify-content: space-between; padding: 2px;";
this.label = document.createElement("span");
this.label.textContent = option;
this.box = document.createElement("input");
this.box.setAttribute("type", "radio");
this.box.setAttribute("name", alert);
this.container.append(this.label, this.box);
if (alert == "JOIN" && stored.joinAlerts == option) this.box.checked = true;
else if (alert == "QUIT" && stored.quitAlerts == option) this.box.checked = true;
else if (alert == "EMBED" && stored.embedAlerts == option) this.box.checked = true;
this.box.addEventListener(("click"), function() {
if (alert == "JOIN") stored.joinAlerts = option;
else if (alert == "QUIT") stored.quitAlerts = option;
else stored.embedAlerts = option;
saveStored();
})
}
}
class AlertDiv {
constructor(alert) {
this.container = document.createElement("div");
this.container.style.cssText = "border: 1px solid black; padding: 8px;"
this.heading = document.createElement("h4");
this.heading.textContent = alert + " alerts";
const defaultCheckBox = new CheckBox("default", alert);
const allCheckBox = new CheckBox("all", alert);
const friendsCheckBox = new CheckBox("custom", alert);
const noneCheckBox = new CheckBox("none", alert);
this.container.append(this.heading, defaultCheckBox.container, allCheckBox.container, friendsCheckBox.container,noneCheckBox.container);
}
}
const alertsContainer = document.createElement("div");
alertsContainer.style.cssText = "display: flex; justify-content: space-around;";
const joinDiv = new AlertDiv("JOIN");
const quitDiv = new AlertDiv("QUIT");
const embedDiv = new AlertDiv("EMBED");
alertsContainer.append(joinDiv.container,quitDiv.container,embedDiv.container);
const friendsInput = document.createElement("textarea");
friendsInput.style.cssText = "width: 100%; height: 14rem; background: #030303; color: #b9b9b9";
friendsInput.value = stored.nicks;
friendsInput.addEventListener("input", function(e) {
let friendsInputValue = friendsInput.value.toLowerCase();
stored.nicks = friendsInputValue.split(",").map(f => f.trim()).filter(f => f);
saveStored();
});
const dialogBar = document.createElement("div");
dialogBar.style.cssText = "display: flex; justify-content: space-between;";
const closeMenuBtn = document.createElement("button");
closeMenuBtn.style.cssText = "background: red; padding: 4px;";
closeMenuBtn.textContent = "X";
closeMenuBtn.addEventListener("click", () => {
menuDialog.close();
});
const historyBtn = document.createElement("button");
historyBtn.textContent = "History";
historyBtn.style.cssText = "background: #b9b9b9; padding: 4px;"
historyBtn.addEventListener("click", function(e) {
menuDialog.close();
const historyDialog = document.createElement("dialog");
historyDialog.style.cssText = "background: #333; color: #999; width: 50rem; height: 40rem; position: relative;";
const backBtn = document.createElement("button");
backBtn.textContent = ">";
backBtn.style.cssText = "background: #b9b9b9; right: 10px; position: absolute; padding: 4px;";
backBtn.addEventListener("click", () => {
historyDialog.close();
menuDialog.showModal();
})
const historyHeading = document.createElement("h1");
historyHeading.textContent = "History";
const historyLines = document.createElement("div");
historyLines.className = "chat-lines";
historyLines.addEventListener("click", function(event) {
if (!historyLines.children[1].contains(event.target)) {
historyLines.classList.remove("focus");
Array.from(historyLines.children).forEach((element) => {
element.style.opacity = "";
})
}
})
function toggleOpacity(element) {
const currentOpacity = window.getComputedStyle(element).opacity;
if (currentOpacity == "1" || currentOpacity == "") element.style.opacity = ".3";
else element.style.opacity = "1";
}
function createDateHeading(date) {
const dateHeading = document.createElement("h2");
dateHeading.textContent = date.toLocaleDateString(undefined, { month: "long", day: "numeric" });
return dateHeading;
}
let lastDate = new Date(null);
stored.history.slice().reverse().forEach((item) => {
const date = new Date(item.time)
if (date.getDate() !== lastDate.getDate()) {
historyLines.append(createDateHeading(date));
}
const msg = new Msg(item.nick, item.msg, item.color, item.time, item.embed).container;
lastDate = new Date(item.time)
msg.style.cssText += "border: 1px solid black;";
msg.children[0].style.display = "inline";
msg.children[1].addEventListener("click", function(event) {
event.stopPropagation();
historyLines.classList.add("focus");
Array.from(historyLines.children).forEach((element) => {
if (msg.getAttribute("data-username") == element.getAttribute("data-username")) element.style.opacity = "1";
})
})
historyLines.append(msg);
})
historyDialog.append(backBtn, historyHeading, historyLines);
document.body.append(historyDialog);
historyDialog.showModal();
});
dialogBar.append(historyBtn, closeMenuBtn);
menuDialog.appendChild(dialogBar);
menuDialog.appendChild(menuTitle);
menuDialog.appendChild(alertsContainer);
menuDialog.appendChild(menuDesc);
menuDialog.appendChild(friendsInput);
});
const chatToolsGroup = document.querySelector(".chat-tools-group");
menuBtnContainer.appendChild(menuBtn);
chatToolsGroup.appendChild(menuBtnContainer);
}
function addMessage(nick, msg, color, time, embed) {
const chatLines = document.querySelector(".chat-lines");
if (!chatLines) {
console.error("couldnt find chat-lines");
}
let msgElement = new Msg(nick, msg, color, time).container;
if (embed) {
msgElement = new Msg(nick, "opened ", color, time, embed).container;
}
chatLines.appendChild(msgElement);
// console.log((chatLines.scrollHeight) - chatLines.scrollTop + chatLines.clientHeight < 1500);
if ((chatLines.scrollHeight) - chatLines.scrollTop + chatLines.clientHeight < 1500) {
chatLines.scrollTop = chatLines.scrollHeight;
}
if (stored.history.length > 200) stored.history.shift();
stored.history.push({
nick: nick,
msg: msg,
color: color,
time: time,
embed: embed
});
saveStored()
return msgElement;
}
function editMessage(nick, msg) {
msg.children[3].textContent = "REFRESH";
msg.style.cssText = "background: #CC5500; color: white;";
const time = parseInt(msg.children[0].getAttribute("data-unixtimestamp"));
stored.history.pop();
stored.history.push({
nick: nick,
msg: "REFRESH",
color: "#CC5500",
time: time,
embed: null
})
}
//
//
//
//
//
console.log('WebSocket interceptor starting...');
const OriginalWebSocket = window.WebSocket;
window.WebSocket = function(url, protocols) {
console.log('WebSocket intercepted to:', url);
const socket = new OriginalWebSocket(url, protocols);
socket.addEventListener('message', function(event) {
const friends = stored.nicks;
// console.log(event);
if (event.origin == "wss://chat.destiny.gg") {
try {
let { prefix, data } = convertToJSON(event.data);
if (!data.nick) return;
const isFriend = friends.includes(data.nick.toLowerCase());
const isDefault = defaultNicks.includes(data.nick.toLowerCase());
if (prefix == "JOIN") {
if (stored.joinAlerts == "none") return;
if ((isFriend && stored.joinAlerts == "custom") || (isDefault && stored.joinAlerts == "default") || stored.joinAlerts == "all") {
const timeDiff = data.timestamp - lastMsg.children[0].getAttribute("data-unixtimestamp")
if (data.nick.toLowerCase() == lastMsg.getAttribute("data-username").toLowerCase()) {
return editMessage(data.nick, lastMsg);
}
return addMessage(data.nick, "JOIN", "green", data.timestamp);
}
}
else if (prefix == "QUIT") {
if (stored.quitAlerts == "none") return;
if ((isFriend && stored.quitAlerts == "custom") || (isDefault && stored.quitAlerts == "default") || stored.quitAlerts == "all") {
const msg = addMessage(data.nick, "QUIT", "red", data.timestamp);
lastMsg = msg;
return
}
} else if (prefix == "UPDATEUSER") {
if (stored.embedAlerts == "none") return;
if ((isFriend && stored.embedAlerts == "custom") || (isDefault && stored.embedAlerts == "default") || stored.embedAlerts == "all") {
let hello = null;
for (let i = stored.history.length - 1; i >= 0; i--) {
const item = stored.history[i];
if (item.nick === data.nick && item.embed) {
hello = item;
break;
}
}
if (hello && data.watching && hello.embed.id == data.watching.id) return;
if (data.watching === null) {
return;
// return addMessage(data.nick, "closed_embed", "#2d1b4b");
}
return addMessage(data.nick, "", "#2d1b4b", Date.now(), data.watching);
}
}
} catch (e) {
console.error(e);
}
}
});
return socket;
};
for (const prop in OriginalWebSocket) {
if (OriginalWebSocket.hasOwnProperty(prop)) {
window.WebSocket[prop] = OriginalWebSocket[prop];
}
}
window.WebSocket.prototype = OriginalWebSocket.prototype;
window.addEventListener('load', function() {
setTimeout(function() {
addMenu();
console.log(window);
}, 1000);
});
console.log('WebSocket interceptor initialized');
})();