Advanced Agma

clickable chat links, discord in game chat (/discord ... hi), remove specific animations, remove food completely (toggle mouse), advanced user stats (agma.io/stats.php), increase number of stackable animations

  1. // ==UserScript==
  2. // @name Advanced Agma
  3. // @namespace http://tampermonkey.net/
  4. // @version 0.1.3
  5. // @author Big watermelon (credits: Nersai, Vintrex)
  6. // @description clickable chat links, discord in game chat (/discord ... hi), remove specific animations, remove food completely (toggle mouse), advanced user stats (agma.io/stats.php), increase number of stackable animations
  7. // @match https://agma.io/*
  8. // @match https://discord.com/*
  9. // @require https://cdn.jsdelivr.net/npm/pako@2.0.4/dist/pako.min.js
  10. // @license GPL-3.0-or-later
  11. // @icon 
  12. // @grant unsafeWindow
  13. // @grant GM_setValue
  14. // @grant GM_getValue
  15. // @run-at document-start
  16. // ==/UserScript==
  17.  
  18. /*
  19. FIXIT: token isnt always grabbed
  20. FIXIT: GM_ functions don't seem to work always so maybe use localStorage
  21. but that's kinda risky so maybe hash it with some values inside the script
  22. TODO: message edit => edit message
  23. TODO: message too long to fit in chat => wrap arround (well maybe not worth it)
  24. */
  25.  
  26. (function() {
  27. "use strict";
  28.  
  29. if (unsafeWindow.top !== unsafeWindow.self || document.querySelector("title")?.textContent?.includes("Just a moment"))
  30. return;
  31.  
  32. const settings = Object.assign({
  33. removeFood: true,
  34. removeAnimations: [3, 6, 7, 8, 11],
  35. maxStackableAnimations: 3,
  36. chatSize: 15,
  37. chatMaxRows: 12,
  38. discordChat: false,
  39. discordSavedChannels: [],
  40. discordPresence: false
  41. }, GM_getValue("settings", {}));
  42.  
  43. const discordToken = GM_getValue("discordToken", null);
  44. if (settings.discordChat && unsafeWindow.location.href.startsWith("https://discord.com")) {
  45. if (!discordToken) {
  46. GM_setValue("discordToken", JSON.parse(unsafeWindow.localStorage.token));
  47. unsafeWindow.alert("Discord Chat for agma.io has updated your token !");
  48. }
  49. return;
  50. }
  51. const numberFormat = Intl.NumberFormat("fr-FR");
  52. const userprofiles = unsafeWindow.localStorage.userprofiles ? JSON.parse(unsafeWindow.localStorage.userprofiles) : {};
  53. if (settings.discordChat && !discordToken) {
  54. unsafeWindow.alert("Discord Chat can not work since you didnt log in into your browser with discord.\nIf you want to use the discord chat feature wait for an alert saying that your token has been saved.\n\nhttps://discord.com");
  55. settings.discordChat = false;
  56. }
  57.  
  58. const discordPresence = {
  59. status: "online",
  60. since: 0,
  61. activities: [{
  62. name: "Agma.io",
  63. type: 0,
  64. url: "https://agma.io",
  65. details: "Playing Agma.io",
  66. timestamps: {
  67. start: Date.now()
  68. }
  69. }],
  70. afk: false
  71. };
  72.  
  73. const animations = {
  74. 1: { name: "Recombine", style: "border-color: #337ab7; background-image: url('./img/store/recombine-min.png'); padding-left: 30px; padding-left: 30px;" },
  75. 2: { name: "Cell Select", style: "border-color: #fe3f3f;" },
  76. 3: { name: "Spin", style: "border-color: #b3b3b3;" },
  77. 4: { name: "360 Shot", style: "border-color: #337ab7; background-image: url('./img/push_lo.png'); padding-left: 30px;" },
  78. 5: { name: "Level Up", style: "border-color: #fe3f3f;" },
  79. 6: { name: "Flip Spin", style: "border-color: #b3b3b3;" },
  80. 7: { name: "Flip", style: "border-color: #b3b3b3;" },
  81. 8: { name: "Shake", style: "border-color: #b3b3b3;" },
  82. 9: { name: "Explosion", style: "border-color: #f0ad4e; background-image: url('./emotes/1f4a5.png'); padding-left: 30px;" },
  83. 10: { name: "1st Medal", style: "border-color: #f0ad4e;" },
  84. 11: { name: "Jump", style: "border-color: #b3b3b3;" },
  85. 12: { name: "Wacky", style: "border-color: #f0ad4e; background-image: url('./emotes/1f61c.png'); padding-left: 30px;" },
  86. 13: { name: "White cell for 1 frame", style: "border-color: #fe3f3f;" },
  87. 14: { name: "Freeze", style: "border-color: #337ab7; background-image: url('./img/inv_freeze2.png'); padding-left: 30px;" },
  88. 15: { name: "Speed", style: "border-color: #337ab7; background-image: url('./img/store/speed-min.png'); padding-left: 30px;" },
  89. 16: { name: "Idk", style: "border-color: #fe3f3f;" }, // weird nothing
  90. 17: { name: "Upgrade", style: "border-color: #fe3f3f;" },
  91. 18: { name: "Snowball", style: "border-color: #fe3f3f;" },
  92. 20: { name: "Anti freeze", style: "border-color: #337ab7; background-image: url('./skins/objects/20.png'); padding-left: 30px;" },
  93. 21: { name: "Anti recombine", style: "border-color: #337ab7; background-image: url('./skins/objects/21.png'); padding-left: 30px;" },
  94. 23: { name: "Shield", style: "border-color: #337ab7; background-image: url('img/inv_shield5.png'); color: rgba(82, 152, 203, 0.6); padding-left: 30px;" },
  95. 24: { name: "Shield", style: "border-color: #337ab7; background-image: url('img/inv_shield5.png'); color: rgba(84, 211, 77, 0.6); padding-left: 30px;" },
  96. 25: { name: "Shield", style: "border-color: #337ab7; background-image: url('img/inv_shield5.png'); color: rgba(243, 46, 46, 0.6); padding-left: 30px;" },
  97. 26: { name: "Shield", style: "border-color: #337ab7; background-image: url('img/inv_shield5.png'); color: rgba(127, 59, 227, 0.6); padding-left: 30px;" },
  98. 30: { name: "Wave", style: "border-color: #f0ad4e; background-image: url('./emotes/1f44b.png'); padding-left: 30px;" },
  99. 31: { name: "Head Explosion", style: "border-color: #f0ad4e; background-image: url('./emotes/1f61cd.png'); padding-left: 30px;" },
  100. 32: { name: "Hearts Face", style: "border-color: #f0ad4e; background-image: url('./emotes/1f60d.png'); padding-left: 30px;" },
  101. 41: { name: "Angry Pumpkin", style: "border-color: #f0ad4e; background-image: url('./emotes/angry_emote3.png'); padding-left: 30px;" },
  102. 42: { name: "Scared Pumpkin", style: "border-color: #f0ad4e; background-image: url('./emotes/scared_emote.png'); padding-left: 30px;" },
  103. 43: { name: "Yawn Pumpkin", style: "border-color: #f0ad4e; background-image: url('./emotes/yawn_emote.png'); padding-left: 30px;" },
  104. 44: { name: "Throwup", style: "border-color: #f0ad4e; background-image: url('./emotes/throwup.png'); padding-left: 30px;" },
  105. 45: { name: "Hot face", style: "border-color: #f0ad4e; background-image: url('./emotes/hotface.png'); padding-left: 30px;" },
  106. 46: { name: "Tears Joy", style: "border-color: #f0ad4e; background-image: url('./emotes/tearsjoy.png'); padding-left: 30px;" },
  107. 47: { name: "No No", style: "border-color: #f0ad4e; background-image: url('./emotes/nonu.png'); padding-left: 30px;" },
  108. 48: { name: "Clap", style: "border-color: #f0ad4e;" },
  109. 49: { name: "Crying", style: "border-color: #f0ad4e;" },
  110. 50: { name: "Devil Smile", style: "border-color: #f0ad4e;" },
  111. 51: { name: "Eatman", style: "border-color: #f0ad4e;" },
  112. 52: { name: "Trophy", style: "border-color: #f0ad4e;" },
  113. 53: { name: "Hearts", style: "border-color: #f0ad4e;" }
  114. };
  115. const animationsIds = [
  116. 1, 4, 14, 15, 20, 21, 23, 24, 25, 26, // Powers
  117. 3, 6, 7, 8, 11, // Annoying
  118. 9, 10, 12, 30, 31, 32, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, // Emotes
  119. 2, 5, 13, 16, 17, 18 // System
  120. ];
  121.  
  122. const discordIcon = new Image();
  123. discordIcon.src = "";
  124. const urlRegex = /https?:\/\/[^\s]+/g;
  125. const invisibleChar = "᠋";
  126. const wordsToReplace = ["https", "www", ".gg", ".com", ".io", ".net", ".biz", "miracle", "palestine"].map(word => [
  127. new RegExp(`\\b(${word})\\b`, "gi"),
  128. word[0] + invisibleChar + word.slice(1)
  129. ]);
  130.  
  131. var discordUsername;
  132. var discordUserContext = null;
  133.  
  134. var discordWebSocket;
  135. var discordHeartbeatInterval;
  136. var discordReconnectRetryCount = 0;
  137. const discordGatewayUrl = "wss://gateway.discord.gg/?v=9&encoding=json";
  138. const sentMessagesIds = [];
  139. const originalSend = WebSocket.prototype.send;
  140. const sendWebsocketDiscordMessage = (op, d) => originalSend.call(discordWebSocket, JSON.stringify({ op, d }));
  141.  
  142. var chtTabs;
  143. var currentTab = '';
  144. var localChatMessages;
  145. var cellProtoOverwritten = false;
  146.  
  147. function initDiscordWebSocket() {
  148. discordWebSocket = new WebSocket(discordGatewayUrl);
  149. discordWebSocket.onopen = () => console.debug("[🔵] Connected");
  150. discordWebSocket.onmessage = ({ data }) => {
  151. if (data instanceof Blob) {
  152. const reader = new FileReader();
  153. reader.onload = () => {
  154. handleMessage(JSON.parse(pako.inflate(new Uint8Array(reader.result), { to: 'string' })));
  155. };
  156. reader.readAsArrayBuffer(data);
  157. } else {
  158. handleMessage(JSON.parse(data));
  159. }
  160. };
  161. discordWebSocket.onclose = () => {
  162. console.debug("[🔵] Disconnected");
  163. clearInterval(discordHeartbeatInterval);
  164. if (++discordReconnectRetryCount < 5) {
  165. initDiscordWebSocket();
  166. }
  167. };
  168. }
  169. function createDiscordChatMessage(message) {
  170. if (!message.channel_id || ![0, 19, 1, 2, 3, 6].includes(message.type) || sentMessagesIds.includes(message.id)) {
  171. return;
  172. }
  173. const channel = settings.discordSavedChannels.find(channel => channel.id == message.channel_id);
  174. if (!channel) {
  175. return;
  176. }
  177. sentMessagesIds.push(message.id);
  178. console.debug("[🔵] New message:", message);
  179. const tab = chtTabs.querySelector(`div[data-username="discord-${channel.id}"]`);
  180. if (!tab) {
  181. newDiscordTab(channel.name, channel.id);
  182. } else if (!tab.classList.contains('selected')) {
  183. tab.classList.add('blink');
  184. setTimeout(() => tab.classList.remove('blink'), 2000);
  185. }
  186. const author = message.author.global_name || message.author.username;
  187. switch (message.type) {
  188. case 0: // DEFAULT
  189. case 19: // REPLY
  190. var content = message.content || '';
  191. message.mentions.forEach(mention => content = content.replaceAll(
  192. `<@${mention.id}>`,
  193. mention.global_name || mention.username
  194. ));
  195. if (message.attachments.length) {
  196. content += ` +${message.attachments.length} files`;
  197. }
  198. localChatMessages.push({
  199. O: true,
  200. get v() {
  201. discordUserContext = this.discordAuthor;
  202. return -1;
  203. },
  204. o: 0,
  205. P: 0,
  206. U: 0,
  207. name: author,
  208. G: "discord-" + channel.id,
  209. color: discordUsername == message.author.username ? "#313338" : "#5662E9",
  210. message: content,
  211. category: 2,
  212. goldMember: 0,
  213. L: 0,
  214. Y: 0,
  215. q: 0,
  216. j: 0,
  217. J: 0,
  218. time: Date.now(),
  219. _cache: null,
  220. get cache() {
  221. return this._cache;
  222. },
  223. set cache(value) {
  224. value.icons = [ 14 ];
  225. return this._cache = value;
  226. },
  227. discordAuthor: message.author
  228. });
  229. break;
  230. case 1: // RECIPIENT_ADD
  231. case 2: // RECIPIENT_REMOVE
  232. const member = message.mentions[0];
  233. localChatMessages.push({
  234. O: false,
  235. v: 0,
  236. o: 0,
  237. P: 0,
  238. U: 0,
  239. name: '',
  240. G: "discord-" + channel.id,
  241. color: message.type == 1 ? "#00FF00" : "#FF0000",
  242. message: `${message.type == 1 ? "➡️" : "⬅️"} ${member.global_name || member.username} ${message.type == 1 ? "joined" : "left"} the group`,
  243. category: 2,
  244. goldMember: 0,
  245. L: 0,
  246. Y: 0,
  247. q: 0,
  248. j: 0,
  249. J: 0,
  250. time: Date.now(),
  251. _cache: null,
  252. get cache() {
  253. return this._cache;
  254. },
  255. set cache(value) {
  256. value.icons = [ 14 ];
  257. Object.defineProperties(value, {
  258. ie: { get: () => this.message },
  259. color2: { get: () => this.color }
  260. });
  261. return this._cache = value;
  262. }
  263. });
  264. break;
  265. case 3: // CALL
  266. localChatMessages.push({
  267. O: false,
  268. v: 0,
  269. o: 0,
  270. P: 0,
  271. U: 0,
  272. name: '',
  273. G: "discord-" + channel.id,
  274. color: "#00FF00",
  275. message: `${author} started a call`,
  276. category: 2,
  277. goldMember: 0,
  278. L: 0,
  279. Y: 0,
  280. q: 0,
  281. j: 0,
  282. J: 0,
  283. time: Date.now(),
  284. _cache: null,
  285. get cache() {
  286. return this._cache;
  287. },
  288. set cache(value) {
  289. value.icons = [ 14 ];
  290. Object.defineProperties(value, {
  291. ie: { get: () => this.message },
  292. color2: { get: () => this.color }
  293. });
  294. return this._cache = value;
  295. }
  296. });
  297. break;
  298. case 6: // CHANNEL_PINNED_MESSAGE
  299. // message_reference: { message_id: "1259902171630145651", channel_id: "853556382417420298" }
  300. localChatMessages.push({
  301. O: false,
  302. v: 0,
  303. o: 0,
  304. P: 0,
  305. U: 0,
  306. name: '',
  307. G: "discord-" + channel.id,
  308. color: "#00FF00",
  309. message: `${author} pinned a message`,
  310. category: 2,
  311. goldMember: 0,
  312. L: 0,
  313. Y: 0,
  314. q: 0,
  315. j: 0,
  316. J: 0,
  317. time: Date.now(),
  318. _cache: null,
  319. get cache() {
  320. return this._cache;
  321. },
  322. set cache(value) {
  323. value.icons = [ 14 ];
  324. Object.defineProperties(value, {
  325. ie: { get: () => this.message },
  326. color2: { get: () => this.color }
  327. });
  328. return this._cache = value;
  329. }
  330. });
  331. break;
  332. }
  333. }
  334. function handleMessage(message) {
  335. const { op, t, d } = message;
  336. switch (op) {
  337. case 10: // Hello
  338. console.debug("[🔵] Connection accepted");
  339. discordHeartbeatInterval = setInterval(() => sendWebsocketDiscordMessage(1, null), d.heartbeat_interval);
  340. sendWebsocketDiscordMessage(2, {
  341. token: discordToken,
  342. capabilities: 30717,
  343. properties: {}
  344. });
  345. settings.discordPresence && setTimeout(() => sendWebsocketDiscordMessage(3, discordPresence), 2000);
  346. break;
  347. case 0: // Dispatch
  348. switch (t) {
  349. case "MESSAGE_CREATE":
  350. createDiscordChatMessage(d);
  351. break;
  352. case "READY":
  353. discordUsername = d.user?.username || discordUsername;
  354. console.debug("[🔵] Logged as", discordUsername);
  355. break;
  356. // default:
  357. // console.debug("[🔵] Unhandled message type:", t);
  358. }
  359. break;
  360. case 1: // HEARTBEAT
  361. sendWebsocketDiscordMessage(1, null);
  362. break;
  363. case 11: // HEARTBEAT_ACK
  364. break;
  365. default:
  366. console.debug("[🔵] Unhandled opcode type:", op);
  367. }
  368. }
  369. function findUrlToOpen(event) {
  370. const x = event.clientX - event.target.offsetLeft,
  371. y = event.clientY - event.target.offsetTop;
  372. for (const message of localChatMessages) {
  373. if (message.cache && message.cache.urls) {
  374. for (const { url, rect } of message.cache.urls) {
  375. if (x >= rect.x0 && y >= rect.y0 && x <= rect.x1 && y <= rect.y1) {
  376. unsafeWindow.open(url, '_blank');
  377. }
  378. }
  379. }
  380. }
  381. }
  382. function timeFormat(s) {
  383. const h = Math.floor(s / 3600);
  384. const m = Math.floor(s % 3600 / 60);
  385. return (h ? h + 'h ' : '') + (m ? m + 'm ' : '') + Math.floor(s % 3600 % 60) + 's';
  386. }
  387. function getTable(payload) {
  388. if (payload == null)
  389. return "<table style=\"width: 100%; text-align: left;\"><tr><td>Player not registered</td><tr><td>You need to play at least 1 game of batle royale to be registered on <a href=\"https://agma.io/stats.php\" target=\"_blank\">Agma Stats</a></td></tr></table>";
  390. return (
  391. "<table style=\"width: 100%; text-align: left;\"><tr><td>Players Consumed: </td><td>"
  392. + numberFormat.format(payload.players_consumed)
  393. + "</td></tr><tr><td>Death Count: </td><td>"
  394. + numberFormat.format(payload.death_count)
  395. + "</td></tr><tr><td>K/D: </td><td>"
  396. + (payload.players_consumed / payload.death_count).toFixed(2)
  397. + "</td></tr><tr><td>Splits count: </td><td>"
  398. + numberFormat.format(payload.splits_count)
  399. + "</td></tr><tr><td>Total time alive: </td><td>"
  400. + timeFormat(payload.total_time_alive)
  401. + "</td></tr><tr><td>Splits per seconds: </td><td>"
  402. + (payload.splits_count / payload.total_time_alive).toFixed(2)
  403. + "</td></tr></table>"
  404. );
  405. }
  406.  
  407. let flag = false
  408. const originalMax = unsafeWindow.Math.max;
  409. unsafeWindow.Math.max = function(...args) {
  410. flag = false;
  411. if (args.length == 2) {
  412. if (args[1] == 8) {
  413. args[0] = settings.chatSize;
  414. } else if (args[1] == 3) {
  415. flag = true;
  416. }
  417. }
  418. return originalMax(...args);
  419. };
  420. const originalMin = unsafeWindow.Math.min;
  421. unsafeWindow.Math.min = function(...args) {
  422. if (flag && !(flag = false) && args.length == 2 && args[0] == 12)
  423. return settings.chatMaxRows;
  424. return originalMin(...args);
  425. };
  426. const originalDrawImage = CanvasRenderingContext2D.prototype.drawImage;
  427. unsafeWindow.CanvasRenderingContext2D.prototype.drawImage = function() {
  428. if (arguments[0] instanceof Image && arguments[0].src.startsWith("https://agma.io/img/chaticons") && arguments[1] == 280) {
  429. arguments[0] = discordIcon;
  430. arguments[1] = 0;
  431. arguments[6] += 3;
  432. }
  433. return originalDrawImage.apply(this, arguments);
  434. }
  435. const originalReplace = unsafeWindow.String.prototype.replace;
  436. unsafeWindow.String.prototype.replace = function(pattern, replacement) {
  437. return originalReplace.call(this, pattern, replacement instanceof Function && pattern instanceof RegExp && pattern.source.startsWith(":rolling") ? (match, offset) => match == ":/" && this?.[offset + 1] == '/' ? match : replacement(match, offset) : replacement);
  438. }
  439. const originalPush = unsafeWindow.Array.prototype.push;
  440. unsafeWindow.Array.prototype.push = function(elem) {
  441. if (elem?.time !== undefined && elem?.cache === null) {
  442. if (localChatMessages != this)
  443. unsafeWindow.localChatMessages = localChatMessages = this;
  444. if ((elem.message = elem.message.replaceAll(invisibleChar, "")).includes("https://")) {
  445. Object.defineProperty(elem, "cache", {
  446. get: function() {
  447. return this._cache;
  448. },
  449. set: function(cache) {
  450. if (elem._cache !== undefined) {
  451. cache.icons = [ 14 ];
  452. // see if this breaks calls and stuff but shouldnt
  453. }
  454. Object.defineProperty(cache, "ctx", {
  455. get: function() {
  456. return this._ctx;
  457. },
  458. set: function(ctx) {
  459. ctx.fillText = function(text, x, y) {
  460. if (this.fillStyle != "#f5f6ce" && this.fillStyle != "#444444") {
  461. return CanvasRenderingContext2D.prototype.fillText.call(this, text, x, y);
  462. }
  463. // ngl I asked chatgpt cuz Im lazy (and I edited some stuff)
  464. let match;
  465. let lastIndex = 0;
  466. let currentX = x;
  467. cache.urls = [];
  468. while ((match = urlRegex.exec(text)) != null) {
  469. const url = match[0];
  470. const urlStart = match.index;
  471. const urlEnd = urlStart + url.length;
  472.  
  473. const preText = text.slice(lastIndex, urlStart);
  474. if (preText) {
  475. CanvasRenderingContext2D.prototype.fillText.call(this, preText, currentX, y);
  476. currentX += ctx.measureText(preText).width;
  477. }
  478. ctx.save();
  479. ctx.fillStyle = '#1E90FF';
  480. CanvasRenderingContext2D.prototype.fillText.call(this, url, currentX, y);
  481.  
  482. const urlWidth = ctx.measureText(url).width;
  483. const textSize = parseInt(ctx.font.match(/\d+/), 10);
  484. const underlineY = y + 2;
  485.  
  486. ctx.beginPath();
  487. ctx.moveTo(currentX, underlineY);
  488. ctx.lineTo(currentX + urlWidth, underlineY);
  489. ctx.strokeStyle = '#1E90FF';
  490. ctx.lineWidth = 1;
  491. ctx.stroke();
  492. ctx.restore();
  493. cache.urls.push({
  494. url,
  495. rect: {
  496. x0: currentX,
  497. yd0: y - textSize,
  498. x1: currentX + urlWidth,
  499. yd1: y
  500. }
  501. });
  502. currentX += urlWidth;
  503. lastIndex = urlEnd;
  504. }
  505. const remainingText = text.slice(lastIndex);
  506. if (remainingText) {
  507. CanvasRenderingContext2D.prototype.fillText.call(this, remainingText, currentX, y);
  508. }
  509. }
  510. return this._ctx = ctx;
  511. },
  512. configurable: true
  513. });
  514. return this._cache = cache;
  515. },
  516. configurable: true
  517. });
  518. }
  519. } else if (elem?.ch !== undefined && elem?.x0 !== undefined) {
  520. elem.ch.cache?.urls?.forEach(urlObject => {
  521. urlObject.rect.y0 = urlObject.rect.yd0 + elem.y0;
  522. urlObject.rect.y1 = urlObject.rect.yd1 + elem.y0;
  523. });
  524. } else if (elem?.namePart !== undefined && elem?.clanPart !== undefined) {
  525. if (!cellProtoOverwritten) {
  526. if (settings.removeAnimations.length) {
  527. elem.constructor.prototype.ge = function(animation) {
  528. if (1 == this.a || settings.removeAnimations.includes(animation.H)) {
  529. return;
  530. }
  531. animation = { H: animation.H, K: animation.K, received: animation.received };
  532. if (this.Ne) {
  533. for (var i = 0; i < this.Ne.length; i++) {
  534. if (this.Ne[i].received > animation.received) {
  535. this.Ne.splice(i, 0, animation);
  536. if (this.Ne.length > settings.maxStackableAnimations) {
  537. this.Ne.splice(this.Ne.length - 2, 1);
  538. }
  539. return;
  540. }
  541. }
  542. if (this.Ne.length < settings.maxStackableAnimations) {
  543. originalPush.call(this.Ne, animation);
  544. } else {
  545. this.Ne[this.Ne.length - 1] = animation;
  546. }
  547. } else {
  548. this.Ne = [animation];
  549. }
  550. }
  551. }
  552. cellProtoOverwritten = true;
  553. }
  554. if (settings.removeFood && elem.Pe) { // elem.a === 1
  555. return 0;
  556. }
  557. }
  558. return originalPush.apply(this, arguments);
  559. }
  560.  
  561. unsafeWindow.setAnimation = (id, value) => settings.removeAnimations[settings.removeAnimations.includes(id) ? 'remove' : 'push'](id);
  562. unsafeWindow.setCustomChatSize = value => settings.chatSize = parseInt(value);
  563. unsafeWindow.setCustomChatMaxRows = value => settings.chatMaxRows = parseInt(value);
  564. unsafeWindow.setMaxStackableAnimations = value => settings.maxStackableAnimations = parseInt(value);
  565. unsafeWindow.setRemoveFood = value => settings.removeFood = value;
  566. unsafeWindow.setDiscordChat = value => {
  567. if (settings.discordChat = value) {
  568. !discordWebSocket && initDiscordWebSocket();
  569. } else if (discordWebSocket != null) {
  570. discordWebSocket.close();
  571. discordWebSocket = null;
  572. }
  573. };
  574. unsafeWindow.setDiscordPresenceChat = value => settings.discordPresence = value;
  575. unsafeWindow.newDiscordTab = (name, channel) => chtTabs && (chtTabs.innerHTML += `<div data-category="2" data-username="discord-${channel}" data-insert data-tooltip="Discord chat: ${name}" class="chat-tab semi-selected">${name}</div>`);
  576. /*
  577. // could make your own messages show up faster but too annoying so no
  578. createDiscordChatMessage({
  579. id: null,
  580. channel_id: channel,
  581. type: 0,
  582. author: discordUser,
  583. content
  584. });
  585. */
  586. unsafeWindow.sendDiscordMessage = (channel, content) => settings.discordChat && discordUsername && fetch(`https://discord.com/api/v9/channels/${channel}/messages`, {
  587. method: 'POST',
  588. headers: {
  589. 'Authorization': discordToken,
  590. 'Content-Type': 'application/json'
  591. },
  592. body: JSON.stringify({
  593. content,
  594. tts: false
  595. })
  596. })
  597. .then(async response => createDiscordChatMessage(await response.json()))
  598. .catch(error => console.error('[🔵] Error while trying to send a message:', error));
  599.  
  600. var loaded = false;
  601. unsafeWindow.addEventListener("load", () => {
  602. if (loaded || typeof swal == "undefined") return;
  603. loaded = true;
  604. const chtCanvas = document.getElementById("chtCanvas");
  605. chtCanvas.addEventListener("dblclick", findUrlToOpen);
  606. chtCanvas.addEventListener("click", event => event.ctrlKey && findUrlToOpen(event));
  607. document.getElementById("chtbox").addEventListener("keydown", function(event) {
  608. if (event.keyCode == 13) {
  609. if (!currentTab && this.value.startsWith("/discord ")) {
  610. const args = this.value.slice(9).split(' ');
  611. const channelName = args.shift();
  612. this.value = args.join(' ').trim();
  613. const channel = settings.discordSavedChannels.find(channel => channel.name == channelName);
  614. if (channel && !chtTabs.querySelector(`div[data-username="discord-${channel.id}"]`)) {
  615. newDiscordTab(channelName, channel.id);
  616. }
  617. if (this.value) {
  618. sendDiscordMessage(channel.id, this.value);
  619. this.value = '';
  620. }
  621. } else if (currentTab.startsWith("discord-")) {
  622. if (this.value = this.value.trim()) {
  623. sendDiscordMessage(currentTab.slice(8), this.value);
  624. this.value = '';
  625. }
  626. } else {
  627. for (const [reg, rep] of wordsToReplace) {
  628. this.value = this.value.replaceAll(reg, rep);
  629. }
  630. }
  631. }
  632. });
  633. // Script devs so annoying can't even use that for settings pages
  634. // typeof GM_info == "undefined" ? Date.now() : GM_info?.script?.position + 3;
  635. const settingPageId = Math.random() * 10e17;
  636. chtTabs = document.getElementById("chtTabs");
  637. chtTabs.addEventListener("click", () => setTimeout(() => currentTab = chtTabs.querySelector("div.chat-tab.selected").dataset.username, 1));
  638. $('#settingTab2').after(`<button id="settingTab${settingPageId}" class="setting-tablink" onclick="openSettingPage(${settingPageId});">Advanced</button>`);
  639. $('#settingPage3').after(`
  640. <div id="settingPage${settingPageId}" class="setting-tabcontent">
  641. <div class="col-md-10 col-md-offset-1 stng" style="padding-left:20px;padding-right:10px;max-height:550px;overflow-y:auto;overflow-x:hidden;margin:0;width:calc(100% - 5px);">
  642. <span style="margin:0;" class="hotkey-paragraph"> Animations</span>
  643. <div class="row stng-row" style="font-size:14px;">
  644. <div style="width:100%;padding:4px;">
  645. <div style="display:flex;flex-wrap:wrap;padding:0px;width:100%;">
  646. ${animationsIds.map(id => `<div style="${animations[id].style}" onclick="setAnimation(${id}, !this.classList.toggle('disabled'));" class="emote${settings.removeAnimations.includes(id) ? " disabled" : ''}">${animations[id].name}</div>`).join('')}
  647. </div>
  648. </div>
  649. </div>
  650. <span style="margin:0;" class="hotkey-paragraph"> Other</span>
  651. <div class="row stng-row" style="font-size:14px;">
  652. <label>
  653. <input type="number" min="1" value="${settings.chatSize}" onchange="setCustomChatSize(this.value);">
  654. <span> Custom Chat Size *</span>
  655. </label>
  656. <br>
  657. <label>
  658. <input type="number" min="1" value="${settings.chatMaxRows}" onchange="setCustomChatMaxRows(this.value);">
  659. <span> Custom Chat Max Rows *</span>
  660. </label>
  661. <br>
  662. <label>
  663. <input type="number" min="0" value="${settings.maxStackableAnimations}" onchange="setMaxStackableAnimations(this.value);">
  664. <span> Stackable Animations *</span>
  665. </label>
  666. <br>
  667. <label>
  668. <input type="checkbox" onchange="setRemoveFood(this.checked);" ${settings.removeFood ? " checked" : ''}>
  669. <span> Remove Food *</span>
  670. </label>
  671. </div>
  672. <span style="margin:0;" class="hotkey-paragraph"> Discord</span>
  673. <div class="row stng-row" style="font-size:14px;">
  674. <label>
  675. <input type="checkbox" onchange="setDiscordChat(this.checked);" ${settings.discordChat ? " checked" : ''}>
  676. <span> Discord Chat *</span>
  677. </label>
  678. <br>
  679. <label>
  680. <input type="checkbox" onchange="setDiscordPresenceChat(this.checked);" ${settings.discordPresence ? " checked" : ''}>
  681. <span> Discord Agma.io Presence *</span>
  682. </label>
  683. <br>
  684. <label>
  685. <span> Discord Saved Channels *</span>
  686. <textarea id="discordSavedChannels" style="resize: vertical; min-height: 100px; width: 100%;" placeholder="agma.io,942193976063197214\nname,channelId">${settings.discordSavedChannels.map(channel => channel.name + ',' + channel.id).join('\n')}</textarea>
  687. </label>
  688. </div>
  689. </div>
  690. </div>
  691. `);
  692. const discordSavedChannelsTextarea = document.getElementById("discordSavedChannels");
  693. discordSavedChannelsTextarea.addEventListener("blur", function(event) {
  694. settings.discordSavedChannels = [];
  695. for (const line of this.value.split('\n')) {
  696. let [name, id] = line.split(',');
  697. name = name?.trim();
  698. id = id?.trim();
  699. if (name && id) {
  700. settings.discordSavedChannels.push({ name, id });
  701. } else {
  702. swal('Invalid channels', 'Please ensure each line has exactly one comma separating the key and value.', 'error');
  703. break;
  704. }
  705. }
  706. });
  707. discordSavedChannelsTextarea.addEventListener("keydown", function(event) {
  708. event.stopPropagation();
  709. });
  710. const style = document.createElement("style");
  711. style.innerHTML = `
  712. #settingPage${settingPageId}#settingPage${settingPageId} input[type="number"]::-webkit-outer-spin-button,
  713. #settingPage${settingPageId} input[type="number"]::-webkit-inner-spin-button {
  714. -webkit-appearance: none;
  715. margin: 0;
  716. }
  717. #settingPage${settingPageId} input[type="number"] {
  718. -moz-appearance: textfield;
  719. }
  720. #settingPage${settingPageId} input[type="number"] {
  721. max-width: 30px;
  722. }
  723. div.chat-tab[data-username^="discord-"] {
  724. background: #4954c5;
  725. }
  726. #settingPage${settingPageId} > div::-webkit-scrollbar {
  727. width: 8px;
  728. height: 8px;
  729. }
  730. #settingPage${settingPageId} > div::-webkit-scrollbar-track {
  731. background: #282934;
  732. border-radius: 10px;
  733. }
  734. #settingPage${settingPageId} > div::-webkit-scrollbar-thumb {
  735. background-color: #df8500;
  736. border-radius: 10px;
  737. border: 2px solid #282934;
  738. }
  739. #settingPage${settingPageId} div.emote {
  740. min-width: 30px;
  741. height: 30px;
  742. padding: 3px;
  743. border-radius: 5px;
  744. border-width: 1px;
  745. border-style: solid;
  746. margin: 4px;
  747. background-size: contain;
  748. background-repeat: no-repeat;
  749. cursor: pointer;
  750. text-overflow: ellipsis;
  751. }
  752. #settingPage${settingPageId} div.emote.disabled {
  753. text-decoration: line-through;
  754. text-decoration-thickness: 3px;
  755. text-decoration-color: red;
  756. }
  757. `;
  758. document.body.appendChild(style);
  759.  
  760. const originalFind = $.prototype.find;
  761. unsafeWindow.$.prototype.find = function() {
  762. const res = originalFind.apply(this, arguments);
  763. if (
  764. discordUserContext
  765. && this.selector == "#contextMenu"
  766. && arguments?.[0] == "li.enabled.hover"
  767. && res?.length
  768. ) {
  769. if (res[0].id == "contextUserProfile") {
  770. swal({
  771. title: `<img src="https://cdn.discordapp.com/avatars/${discordUserContext.id}/${discordUserContext.avatar}" width="128" height="128" style="border-radius:50%;"><br><br><span>${discordUserContext.global_name || discordUserContext.username}</span><br><span style="font-size:12px;">${discordUserContext.username}</span>`,
  772. html: true
  773. });
  774. return [];
  775. } else if (res[0].id == "contextDiscordMention") {
  776. document.getElementById("chtbox").value += `<@${discordUserContext.id}> `;
  777. return [];
  778. }
  779. }
  780. return res;
  781. }
  782. const originalAddClass = $.prototype.addClass;
  783. unsafeWindow.$.prototype.addClass = function() {
  784. if (discordUserContext && this.selector == "#contextMute") {
  785. $("#contextDiscordMention").show();
  786. originalAddClass.call($("#contextUserProfile"), "enabled");
  787. $("#contextPlayerSkin").css({ 'background-image': `url(https://cdn.discordapp.com/avatars/${discordUserContext.id}/${discordUserContext.avatar})` });
  788. $("#contextPartyLeave").hide();
  789. $("#contextPartyMessage").hide();
  790. $("#contextModerate").hide();
  791. $("#contextPartyInvite").hide();
  792. $("#contextFriendAdd").hide();
  793. $("#contextPrivateMessage").hide();
  794. $("#contextSpectate").hide();
  795. $("#contextPickpocket").hide();
  796. $("#contextMute").hide();
  797. return this;
  798. }
  799. return originalAddClass.apply(this, arguments);
  800. }
  801. const originalHide = $.prototype.hide;
  802. unsafeWindow.$.prototype.hide = function() {
  803. if (discordUserContext && this.selector == "#contextMenu") {
  804. $("#contextDiscordMention").hide();
  805. $("#contextPartyLeave").show();
  806. $("#contextPartyMessage").show();
  807. $("#contextModerate").show();
  808. $("#contextPartyInvite").show();
  809. $("#contextFriendAdd").show();
  810. $("#contextPrivateMessage").show();
  811. $("#contextSpectate").show();
  812. $("#contextPickpocket").show();
  813. $("#contextMute").show();
  814. discordUserContext = null;
  815. }
  816. return originalHide.apply(this, arguments);
  817. }
  818. $("#contextUserProfile").after('<li id="contextDiscordMention" class="contextmenu-item enabled" style="display: none;"><div class="fa fa-at fa-2x context-icon"></div><p>Mention</p></li>');
  819.  
  820. let waitingResponse = false;
  821. const originalSwal = unsafeWindow.swal;
  822. unsafeWindow.swal = function() {
  823. if (
  824. !discordUserContext
  825. && typeof arguments[0] == "object"
  826. && "title" in arguments[0]
  827. && arguments[0].title.startsWith("<img src=\"")
  828. ) {
  829. if (waitingResponse) {
  830. return;
  831. }
  832. const username = arguments[0].title.match(/>([^>]+)<\/span>/)?.[1];
  833. if (!username) {
  834. return originalSwal.apply(this, arguments);
  835. }
  836. if (Date.now() - userprofiles[username]?.at < 86400000) {
  837. arguments[0].text += getTable(userprofiles[username].data);
  838. } else {
  839. waitingResponse = true;
  840. $.getJSON("https://agma.io/royale_stats.php?user=" + username, payload => {
  841. userprofiles[username] = { at: Date.now(), data: payload };
  842. arguments[0].text += getTable(payload);
  843. waitingResponse = false;
  844. originalSwal.apply(this, arguments);
  845. }).fail(() => {
  846. userprofiles[username] = { at: Date.now(), data: null };
  847. arguments[0].text += getTable(null);
  848. waitingResponse = false;
  849. originalSwal.apply(this, arguments);
  850. }).always(() => {
  851. waitingResponse = false;
  852. });
  853. return;
  854. }
  855. }
  856. return originalSwal.apply(this, arguments);
  857. }
  858. unsafeWindow.swal.close = originalSwal.close;
  859. });
  860. unsafeWindow.addEventListener("beforeunload", () => {
  861. discordWebSocket && discordWebSocket.close();
  862. GM_setValue("settings", settings);
  863. unsafeWindow.localStorage.userprofiles = JSON.stringify(userprofiles);
  864. });
  865. settings.discordChat && initDiscordWebSocket();
  866. console.log(`%cAdvanced Agma - ${GM_info?.script?.version} Loaded`, "font-weight: bold; font-size: 20pt; color: black;");
  867. })();