sarkyScript

Tracks and logs the actions of dgg users

目前为 2025-04-06 提交的版本。查看 最新版本

  1. // ==UserScript==
  2. // @name sarkyScript
  3. // @namespace https://www.destiny.gg/
  4. // @version 0.6
  5. // @description Tracks and logs the actions of dgg users
  6. // @match *://*.destiny.gg/embed/chat*
  7. // @run-at document-start
  8. // @grant none
  9. // @icon 
  10. // @license MIT
  11. // ==/UserScript==
  12.  
  13. (function() {
  14. 'use strict';
  15.  
  16. const menuIcon64 = "";
  17. let friends = [];
  18. let lastMsg = {};
  19.  
  20. try {
  21. const storedFriends = localStorage.getItem("sark_friends");
  22. if (storedFriends) {
  23. friends = JSON.parse(storedFriends);
  24. }
  25. } catch (e) {
  26. console.error(e);
  27. friends = [];
  28. }
  29.  
  30.  
  31. function convertToJSON(prefixedData) {
  32. const firstSpaceIndex = prefixedData.indexOf(" ");
  33. const prefix = prefixedData.substring(0, firstSpaceIndex);
  34. const data = JSON.parse(prefixedData.substring(firstSpaceIndex + 1));
  35.  
  36. return { prefix, data }
  37. }
  38.  
  39. function addMenu() {
  40. const menuBtnContainer = document.createElement("a");
  41. menuBtnContainer.setAttribute("title", "sarky");
  42. menuBtnContainer.className = "chat-tool-btn";
  43.  
  44. const menuBtn = document.createElement("img");
  45. menuBtn.className = "btn-icon";
  46. menuBtn.src = menuIcon64;
  47.  
  48. const menuTitle = document.createElement("h1");
  49. menuTitle.textContent = "List of names to track";
  50.  
  51. const menuDesc = document.createElement("p");
  52. menuDesc.textContent = "Seperate the names with commas using no spaces";
  53.  
  54.  
  55. menuBtnContainer.addEventListener("click", function(e) {
  56. // console.log("clicked menu button");
  57.  
  58. const menuDialog = document.createElement("dialog");
  59. menuDialog.style.cssText = "background: #333; color: #999; width: 80%; height: 70%; position: relative;";
  60. document.body.appendChild(menuDialog);
  61. menuDialog.showModal();
  62.  
  63.  
  64. const friendsInput = document.createElement("textarea");
  65. friendsInput.style.cssText = "width: 100%; height: 100%; background: #030303; color: #b9b9b9";
  66. // console.log("sark_friends", friends);
  67. friendsInput.value = friends;
  68.  
  69. friendsInput.addEventListener("input", function(e) {
  70. let friendsInputValue = friendsInput.value.toLowerCase();
  71. friends = friendsInputValue.split(",").map(f => f.trim()).filter(f => f);
  72. localStorage.setItem("sark_friends", JSON.stringify(friends));
  73. });
  74.  
  75. const closeMenuBtn = document.createElement("button");
  76. closeMenuBtn.style.cssText = "background: red; padding: 2px;";
  77. closeMenuBtn.textContent = "X";
  78. closeMenuBtn.addEventListener("click", () => {
  79. menuDialog.close();
  80. });
  81. menuDialog.appendChild(closeMenuBtn);
  82. menuDialog.appendChild(menuTitle);
  83. menuDialog.appendChild(menuDesc);
  84. menuDialog.appendChild(friendsInput);
  85.  
  86. });
  87.  
  88.  
  89. const chatToolsGroup = document.querySelector(".chat-tools-group");
  90. menuBtnContainer.appendChild(menuBtn);
  91. chatToolsGroup.appendChild(menuBtnContainer);
  92. }
  93.  
  94. function addMessage(nick, msg, color) {
  95. console.log(nick, msg, color);
  96. const chatLines = document.querySelector(".chat-lines");
  97. if (!chatLines) {
  98. console.error("couldnt find chat-lines");
  99. }
  100. const msgWords = msg.split(" ");
  101.  
  102. const msgElement = document.createElement("div");
  103. msgElement.className = "msg-chat msg-user";
  104. msgElement.setAttribute("data-username", nick.toLowerCase());
  105. msgElement.style.cssText = `background: ${color}; color: white; margin: 2px`;
  106.  
  107. const msgUser = document.createElement("a");
  108. msgUser.className = "user";
  109. msgUser.textContent = nick;
  110. msgElement.appendChild(msgUser);
  111.  
  112. const msgCtrl = document.createElement("span");
  113. msgCtrl.className = "ctrl";
  114. msgCtrl.textContent = ":"
  115. msgElement.appendChild(msgCtrl);
  116.  
  117. const msgText = document.createElement("span");
  118. msgText.className = "text";
  119. msgElement.appendChild(msgText);
  120. if (msgWords[1]) {
  121. console.log("embed");
  122. const embedLink = document.createElement("a");
  123. embedLink.style.className = "externallink bookmarklink";
  124. embedLink.setAttribute("target", "_top");
  125. embedLink.href = "https://www.destiny.gg/bigscreen" + msgWords[1];
  126. embedLink.textContent = msgWords[1];
  127. msgText.textContent = " opened ";
  128. msgElement.appendChild(embedLink);
  129. } else {
  130. msgText.textContent += " " + msg;
  131. }
  132.  
  133. chatLines.appendChild(msgElement);
  134.  
  135.  
  136. return true;
  137. }
  138.  
  139.  
  140. //
  141. //
  142. //
  143. //
  144. //
  145.  
  146. console.log('WebSocket interceptor starting...');
  147.  
  148. // Store the original WebSocket constructor
  149. const OriginalWebSocket = window.WebSocket;
  150.  
  151. // Override the WebSocket constructor
  152. window.WebSocket = function(url, protocols) {
  153. console.log('WebSocket intercepted to:', url);
  154.  
  155. // Create a WebSocket instance using the original constructor
  156. const socket = new OriginalWebSocket(url, protocols);
  157.  
  158. // Add event listeners instead of overriding onmessage
  159. socket.addEventListener('message', function(event) {
  160. // console.log(event);
  161. if (event.origin == "wss://chat.destiny.gg") {
  162. try {
  163. let { prefix, data } = convertToJSON(event.data);
  164.  
  165. if (!data.nick) return;
  166. if (friends.includes(data.nick.toLowerCase())) {
  167. if (prefix == "JOIN") {
  168. addMessage(data.nick, "JOIN", "green");
  169. } else if (prefix == "QUIT") {
  170. addMessage(data.nick, "QUIT", "red")
  171. } else if (prefix == "UPDATEUSER") {
  172. // console.log("UPDATE USER!");
  173. if (lastMsg.watching && data.watching) {
  174. if (lastMsg.watching.id == data.watching.id) return;
  175. }
  176. if (data.watching === null) {
  177. lastMsg = data;
  178. return addMessage(data.nick, "closed_embed", "#2d1b4b");
  179. }
  180. addMessage(data.nick, ("opened " + "#" + data.watching.platform + "/" + data.watching.id), "#2d1b4b");
  181. lastMsg = data
  182. }
  183. }
  184. } catch (e) {
  185. console.error(e);
  186. }
  187. }
  188. });
  189.  
  190. return socket;
  191. };
  192.  
  193. // Copy properties from original WebSocket to our overridden version
  194. for (const prop in OriginalWebSocket) {
  195. if (OriginalWebSocket.hasOwnProperty(prop)) {
  196. window.WebSocket[prop] = OriginalWebSocket[prop];
  197. }
  198. }
  199.  
  200. window.WebSocket.prototype = OriginalWebSocket.prototype;
  201.  
  202. window.addEventListener('load', function() {
  203. setTimeout(function() {
  204. addMenu();
  205. console.log(window);
  206. }, 1000);
  207. });
  208.  
  209. console.log('WebSocket interceptor initialized');
  210. })();