sarkyScript

Tracks and logs the actions of conductors (or other users)

// ==UserScript==
// @name         sarkyScript
// @namespace    https://www.destiny.gg/
// @version      1.3
// @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 = HTMLElement;
  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: 90%; height: 70%; 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: 90%; height: 70%; 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");

        stored.history.slice().reverse().forEach((item) => {
          const msg = new Msg(item.nick, item.msg, item.color, item.time, item.embed).container;
          console.log(msg);
          msg.style.cssText += "border: 1px solid black;";
          msg.children[0].style.display = "inline";
          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"));
    console.log(time);
    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);
                    }
                    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;
                  }
                } 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");
                    }
                    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');
})();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址