IdlePixel Chat Highlighter

Highlights messages containing specified words, or from specified users.

  1. // ==UserScript==
  2. // @name IdlePixel Chat Highlighter
  3. // @namespace lbtechnology.info
  4. // @version 2.0.0
  5. // @description Highlights messages containing specified words, or from specified users.
  6. // @author Lux-Ferre
  7. // @license MIT
  8. // @match *://idle-pixel.com/login/play*
  9. // @grant none
  10. // @require https://gf.qytechs.cn/scripts/441206-idlepixel/code/IdlePixel+.js?anticache=20220905
  11. // ==/UserScript==
  12.  
  13. (function() {
  14. 'use strict';
  15. class HighlightPlugin extends IdlePixelPlusPlugin {
  16. constructor() {
  17. super("highlighting", {
  18. about: {
  19. name: `${GM_info.script.name} (ver: ${GM_info.script.version})`,
  20. version: GM_info.script.version,
  21. author: GM_info.script.author,
  22. description: GM_info.script.description
  23. },
  24. config: [
  25. {
  26. label: `<div class="d-flex w-100"><span class="align-self-center col-6">Word Highlighting</span><span class="col-6"><button class="btn btn-primary" type="button" onclick="IdlePixelPlus.plugins.highlighting.showModal('word_set')">Edit List</button></span></div>`,
  27. type: "label"
  28. },
  29. {
  30. id: "wordList",
  31. label: "List of trigger words (DEPRACATED! USE BUTTON INSTEAD!)",
  32. type: "string",
  33. max: 2000,
  34. default: ""
  35. },
  36. {
  37. label: `------------------------------------------------------------------------------------------------`,
  38. type: "label"
  39. },
  40. {
  41. label: `<div class="d-flex w-100"><span class="align-self-center col-6">Word Ignoring</span><span class="col-6"><button class="btn btn-primary" type="button" onclick="IdlePixelPlus.plugins.highlighting.showModal('ignore_word_set')">Edit List</button></span></div>`,
  42. type: "label"
  43. },
  44. {
  45. id: "ignoreWordList",
  46. label: "List of words to ignore on trigger (DEPRACATED! USE BUTTON INSTEAD!)",
  47. type: "string",
  48. max: 2000,
  49. default: ""
  50. },
  51. {
  52. label: `------------------------------------------------------------------------------------------------`,
  53. type: "label"
  54. },
  55. {
  56. label: `<div class="d-flex w-100"><span class="align-self-center col-6">Player Highlighting</span><span class="col-6"><button class="btn btn-primary" type="button" onclick="IdlePixelPlus.plugins.highlighting.showModal('user_set')">Edit List</button></span></div>`,
  57. type: "label"
  58. },
  59. {
  60. id: "friendList",
  61. label: "List of people to be highlighted (DEPRACATED! USE BUTTON INSTEAD!)",
  62. type: "string",
  63. max: 2000,
  64. default: ""
  65. },
  66. {
  67. label: `------------------------------------------------------------------------------------------------`,
  68. type: "label"
  69. },
  70. {
  71. label: `<div class="d-flex w-100"><span class="align-self-center col-6">Player Ignoring</span><span class="col-6"><button class="btn btn-primary" type="button" onclick="IdlePixelPlus.plugins.highlighting.showModal('ignore_user_set')">Edit List</button></span></div>`,
  72. type: "label"
  73. },
  74. {
  75. id: "ignoreNameList",
  76. label: "List of players to ignore triggers from (DEPRACATED! USE BUTTON INSTEAD!)",
  77. type: "string",
  78. max: 2000,
  79. default: ""
  80. },
  81. {
  82. label: `------------------------------------------------------------------------------------------------`,
  83. type: "label"
  84. },
  85. {
  86. id: "soundsEnabled",
  87. label: "Play a sound when being pinged?",
  88. type: "boolean",
  89. default: false
  90. },
  91. {
  92. id: "ignoreCase",
  93. label: "Ignore case-sensitivity?",
  94. type: "boolean",
  95. default: true
  96. },
  97. {
  98. id: "notificationsEnabled",
  99. label: "Enable popup notifications?",
  100. type: "boolean",
  101. default: false
  102. },
  103. {
  104. id: "considerSpaces",
  105. label: "Allow spaces in triggers?",
  106. type: "boolean",
  107. default: false
  108. },
  109. {
  110. id: "activeName",
  111. label: "Username for account having sound & popups (only useful if you have multiple accounts open.)",
  112. type: "string",
  113. max: 20,
  114. default: ""
  115. },
  116. {
  117. id: "colourWordHighlight",
  118. label: "Word highlighting colour:",
  119. type: "color",
  120. default: "#00FF00"
  121. },
  122. {
  123. id: "colourFriendHighlight",
  124. label: "Username highlighting colour",
  125. type: "color",
  126. default: "#8C00FF"
  127. }
  128. ]
  129. })
  130. this.word_set = new Set()
  131. this.ignore_word_set = new Set()
  132. this.user_set = new Set()
  133. this.ignore_user_set = new Set()
  134. }
  135.  
  136. onLogin(){
  137. this.addStyles()
  138. this.loadData()
  139. this.createModal()
  140. }
  141.  
  142. addStyles(){
  143. let backgroundColour
  144. let textColour
  145.  
  146. if ("ui-tweaks" in IdlePixelPlus.plugins){
  147. backgroundColour = IdlePixelPlus.plugins["ui-tweaks"].config["color-chat-area"]
  148. textColour = IdlePixelPlus.plugins["ui-tweaks"].config["font-color-chat-area"]
  149. } else {
  150. backgroundColour = "white"
  151. textColour = "black"
  152. }
  153. const styles = `
  154. #chatHighlighterListModalFooter {
  155. padding: calc(var(--bs-modal-padding) - var(--bs-modal-footer-gap) * .5);
  156. background-color: var(--bs-modal-footer-bg);
  157. border-top: var(--bs-modal-footer-border-width) solid var(--bs-modal-footer-border-color);
  158. border-bottom-right-radius: var(--bs-modal-inner-border-radius);
  159. border-bottom-left-radius: var(--bs-modal-inner-border-radius);
  160. }
  161.  
  162. .highlighterListItem {
  163. background-color: RGBA(1, 150, 150, 0.5);
  164. margin-bottom: 2px;
  165. }
  166.  
  167. .highlightListItemCross {
  168. border-right-style: ridge;
  169. margin-left: 5px;
  170. padding-right: 3px;
  171. }
  172.  
  173. .highlightListItemText {
  174. margin-left: 6px;
  175. }
  176.  
  177. #chatHighlighterListModalInner {
  178. background-color: ${backgroundColour};
  179. color: ${textColour};
  180. }
  181. #chatHighlighterListModalDialog {
  182. margin-top: 20vh;
  183. }
  184. #chatHighlighterListModalTitle{
  185. text-decoration: underline;
  186. }
  187. `
  188. const styleElement = `<style id="styles-highlighter">${styles}</style>`
  189. $("head").append(styleElement)
  190. }
  191.  
  192. loadData(){
  193. const configJSON = localStorage.getItem("chatHighlightingData")
  194.  
  195. if (configJSON){
  196. const configs = JSON.parse(configJSON)
  197. for (const [configType, configList] of Object.entries(configs)){
  198. configList.forEach(listItem=>{
  199. this[configType].add(listItem)
  200. })
  201. }
  202. }
  203. }
  204.  
  205. saveData(){
  206. const data = {
  207. word_set: [...this.word_set],
  208. ignore_word_set: [...this.ignore_word_set],
  209. user_set: [...this.user_set],
  210. ignore_user_set: [...this.ignore_user_set]
  211. }
  212.  
  213. const dataJSON = JSON.stringify(data)
  214. localStorage.setItem("chatHighlightingData", dataJSON)
  215. }
  216.  
  217. createModal(){
  218. const modalString = `
  219. <div id="chatHighlighterListModal" class="modal fade" role="dialog" tabindex="-1">
  220. <div id="chatHighlighterListModalDialog" class="modal-dialog" role="document">
  221. <div id="chatHighlighterListModalInner" class="modal-content">
  222. <div id="chatHighlighterListModalHeader" class="modal-header text-center">
  223. <h3 class="modal-title w-100" id="chatHighlighterListModalTitle">Pending Invitations</h3>
  224. </div>
  225. <div class="modal-body">
  226. <div id="chatHighlighterListModalList" class="overflow-auto"></div>
  227. </div>
  228. <div id="chatHighlighterListModalFooter">
  229. <form style="margin-right: 10px;margin-left: 10px;" onsubmit="event.preventDefault(); IdlePixelPlus.plugins.highlighting.addFromModal();">
  230. <div class="row d-flex flex-fill">
  231. <div class="col-10"><input id="highlightListInput" class="form-control w-100" type="text" /></div>
  232. <div class="col-2"><input id="highlightListButton" class="w-100 h-100 rounded-pill" type="submit" value="Add" /></div>
  233. </div>
  234. </form>
  235. </div>
  236. </div>
  237. </div>
  238. </div>
  239. `
  240.  
  241. const modalElement = $.parseHTML(modalString)
  242. $(document.body).append(modalElement)
  243. }
  244.  
  245. showModal(configType){
  246. const listModal = $("#chatHighlighterListModal")
  247. this.populateModal(configType)
  248. listModal.attr("data-configtype", configType)
  249. $("#chatHighlighterListModalTitle").text(configType)
  250. listModal.modal('show')
  251. document.body.scrollTop = document.documentElement.scrollTop = 0
  252. }
  253.  
  254. populateModal(configType){
  255. const data_set = this[configType]
  256. $("#chatHighlighterListModalList").empty()
  257.  
  258. data_set.forEach(item => {
  259. this.addToList(item)
  260. })
  261. }
  262.  
  263. addToList(item){
  264. const ident = item.match(/[a-z]/g).join('')
  265. const newItemString = `<div class="highlighterListItem rounded-pill" id="highlightItem${ident}" data-item="${item}" onclick="event.preventDefault(); IdlePixelPlus.plugins.highlighting.removeItem(this.getAttribute('data-item'), this.getAttribute('id'))"><span class="highlightListItemCross">❌</span><span class="highlightListItemText">${item}</span></div>`
  266. const newItemElement = $.parseHTML(newItemString)
  267. $("#chatHighlighterListModalList").append(newItemElement)
  268. }
  269.  
  270. addFromModal(){
  271. const inputBox = $("#highlightListInput")
  272. const configType = $("#chatHighlighterListModal").attr("data-configtype")
  273. const newItem = inputBox.val()
  274. inputBox.val("")
  275.  
  276. this.addItem(newItem, configType)
  277. }
  278.  
  279. addItem(newItem, configType){
  280. this[configType].add(newItem)
  281. this.addToList(newItem)
  282. this.saveData()
  283. }
  284.  
  285. removeItem(item, id){
  286. const configType = $("#chatHighlighterListModal").attr("data-configtype")
  287.  
  288. this[configType].delete(item)
  289. $(`#${id}`).remove()
  290. this.saveData()
  291. }
  292.  
  293. toRGBA(hex) {
  294. const r = parseInt(hex.slice(1, 3), 16);
  295. const g = parseInt(hex.slice(3, 5), 16);
  296. const b = parseInt(hex.slice(5, 7), 16);
  297. return `rgba(${r}, ${g}, ${b}, 0.15)`;
  298. }
  299.  
  300. highlightMessage(data, highlightType){
  301. const notificationsEnabled = this.getConfig("notificationsEnabled");
  302. const soundsEnabled = this.getConfig("soundsEnabled");
  303. const activeName = this.getConfig("activeName");
  304. let highlightColour = ""
  305. const wordColour = this.toRGBA(this.getConfig("colourWordHighlight"))
  306. const friendColour = this.toRGBA(this.getConfig("colourFriendHighlight"))
  307.  
  308. if(highlightType === "word") {highlightColour = wordColour}
  309. else if(highlightType === "user") {highlightColour = friendColour}
  310. else {highlightColour = "rgba(0, 0, 0, 0)"}
  311.  
  312. const element = $("#chat-area > *").last();
  313. element.attr("style", `background-color: ${highlightColour}`)
  314. if (highlightType === "word"){
  315. if (activeName === var_username || activeName === ""){
  316. if (soundsEnabled){Sounds.play(Sounds.VARIABLE_POWER_UP);}
  317. if (notificationsEnabled){this.notify(data.message, data.username)}
  318. }
  319. }
  320. }
  321.  
  322. notify(message, username){
  323. if (!window.Notification) {
  324. alert("Sorry, Notifications are not supported in this Browser!");
  325. } else {
  326. if (Notification.permission === 'default') {
  327. Notification.requestPermission(function(p) {
  328. if (p === 'denied') {
  329. alert('You have denied Notifications'); }
  330. else {
  331. var notify = new Notification('Chat Notification', {
  332. body: `${username}: ${message}`,
  333. requireInteraction: true,
  334. icon: bob
  335. });
  336. }
  337. });
  338. } else {
  339. var notify = new Notification('Chat Notification', {
  340. body: `${username}: ${message}`,
  341. icon: bob,
  342. requireInteraction: true
  343. });
  344. }
  345. }
  346. }
  347.  
  348. onChat(data) {
  349. const ignoreCase = this.getConfig("ignoreCase");
  350.  
  351. let message
  352.  
  353. let word_list
  354. let ignore_word_list
  355. const user_list = [...this.user_set]
  356. const ignore_user_list = [...this.ignore_user_set]
  357.  
  358. if (ignoreCase) {
  359. message = data.message.toLowerCase()
  360. word_list = [...this.word_set].map(word => word.toLowerCase())
  361. ignore_word_list = [...this.ignore_word_set].map(word => word.toLowerCase())
  362. }
  363. else {
  364. message = data.message
  365. word_list = [...this.word_set]
  366. ignore_word_list = [...this.ignore_word_set]
  367. }
  368.  
  369. if(ignore_user_list.includes(data.username)){
  370. return
  371. }
  372.  
  373. if (word_list.some(word => message.includes(word))) {
  374. if (!ignore_word_list.some(word => message.includes(word))){
  375. this.highlightMessage(data, "word");
  376. }
  377. } else if (user_list.includes(data.username)){
  378. this.highlightMessage(data, "user");
  379. }
  380. }
  381. }
  382. const plugin = new HighlightPlugin();
  383. var bob = ""
  384. IdlePixelPlus.registerPlugin(plugin);
  385. })();

QingJ © 2025

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