您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Highlights messages containing specified words, or from specified users.
// ==UserScript== // @name IdlePixel Chat Highlighter // @namespace lbtechnology.info // @version 2.0.0 // @description Highlights messages containing specified words, or from specified users. // @author Lux-Ferre // @license MIT // @match *://idle-pixel.com/login/play* // @grant none // @require https://greasyfork.org/scripts/441206-idlepixel/code/IdlePixel+.js?anticache=20220905 // ==/UserScript== (function() { 'use strict'; class HighlightPlugin extends IdlePixelPlusPlugin { constructor() { super("highlighting", { about: { name: `${GM_info.script.name} (ver: ${GM_info.script.version})`, version: GM_info.script.version, author: GM_info.script.author, description: GM_info.script.description }, config: [ { 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>`, type: "label" }, { id: "wordList", label: "List of trigger words (DEPRACATED! USE BUTTON INSTEAD!)", type: "string", max: 2000, default: "" }, { label: `------------------------------------------------------------------------------------------------`, type: "label" }, { 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>`, type: "label" }, { id: "ignoreWordList", label: "List of words to ignore on trigger (DEPRACATED! USE BUTTON INSTEAD!)", type: "string", max: 2000, default: "" }, { label: `------------------------------------------------------------------------------------------------`, type: "label" }, { 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>`, type: "label" }, { id: "friendList", label: "List of people to be highlighted (DEPRACATED! USE BUTTON INSTEAD!)", type: "string", max: 2000, default: "" }, { label: `------------------------------------------------------------------------------------------------`, type: "label" }, { 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>`, type: "label" }, { id: "ignoreNameList", label: "List of players to ignore triggers from (DEPRACATED! USE BUTTON INSTEAD!)", type: "string", max: 2000, default: "" }, { label: `------------------------------------------------------------------------------------------------`, type: "label" }, { id: "soundsEnabled", label: "Play a sound when being pinged?", type: "boolean", default: false }, { id: "ignoreCase", label: "Ignore case-sensitivity?", type: "boolean", default: true }, { id: "notificationsEnabled", label: "Enable popup notifications?", type: "boolean", default: false }, { id: "considerSpaces", label: "Allow spaces in triggers?", type: "boolean", default: false }, { id: "activeName", label: "Username for account having sound & popups (only useful if you have multiple accounts open.)", type: "string", max: 20, default: "" }, { id: "colourWordHighlight", label: "Word highlighting colour:", type: "color", default: "#00FF00" }, { id: "colourFriendHighlight", label: "Username highlighting colour", type: "color", default: "#8C00FF" } ] }) this.word_set = new Set() this.ignore_word_set = new Set() this.user_set = new Set() this.ignore_user_set = new Set() } onLogin(){ this.addStyles() this.loadData() this.createModal() } addStyles(){ let backgroundColour let textColour if ("ui-tweaks" in IdlePixelPlus.plugins){ backgroundColour = IdlePixelPlus.plugins["ui-tweaks"].config["color-chat-area"] textColour = IdlePixelPlus.plugins["ui-tweaks"].config["font-color-chat-area"] } else { backgroundColour = "white" textColour = "black" } const styles = ` #chatHighlighterListModalFooter { padding: calc(var(--bs-modal-padding) - var(--bs-modal-footer-gap) * .5); background-color: var(--bs-modal-footer-bg); border-top: var(--bs-modal-footer-border-width) solid var(--bs-modal-footer-border-color); border-bottom-right-radius: var(--bs-modal-inner-border-radius); border-bottom-left-radius: var(--bs-modal-inner-border-radius); } .highlighterListItem { background-color: RGBA(1, 150, 150, 0.5); margin-bottom: 2px; } .highlightListItemCross { border-right-style: ridge; margin-left: 5px; padding-right: 3px; } .highlightListItemText { margin-left: 6px; } #chatHighlighterListModalInner { background-color: ${backgroundColour}; color: ${textColour}; } #chatHighlighterListModalDialog { margin-top: 20vh; } #chatHighlighterListModalTitle{ text-decoration: underline; } ` const styleElement = `<style id="styles-highlighter">${styles}</style>` $("head").append(styleElement) } loadData(){ const configJSON = localStorage.getItem("chatHighlightingData") if (configJSON){ const configs = JSON.parse(configJSON) for (const [configType, configList] of Object.entries(configs)){ configList.forEach(listItem=>{ this[configType].add(listItem) }) } } } saveData(){ const data = { word_set: [...this.word_set], ignore_word_set: [...this.ignore_word_set], user_set: [...this.user_set], ignore_user_set: [...this.ignore_user_set] } const dataJSON = JSON.stringify(data) localStorage.setItem("chatHighlightingData", dataJSON) } createModal(){ const modalString = ` <div id="chatHighlighterListModal" class="modal fade" role="dialog" tabindex="-1"> <div id="chatHighlighterListModalDialog" class="modal-dialog" role="document"> <div id="chatHighlighterListModalInner" class="modal-content"> <div id="chatHighlighterListModalHeader" class="modal-header text-center"> <h3 class="modal-title w-100" id="chatHighlighterListModalTitle">Pending Invitations</h3> </div> <div class="modal-body"> <div id="chatHighlighterListModalList" class="overflow-auto"></div> </div> <div id="chatHighlighterListModalFooter"> <form style="margin-right: 10px;margin-left: 10px;" onsubmit="event.preventDefault(); IdlePixelPlus.plugins.highlighting.addFromModal();"> <div class="row d-flex flex-fill"> <div class="col-10"><input id="highlightListInput" class="form-control w-100" type="text" /></div> <div class="col-2"><input id="highlightListButton" class="w-100 h-100 rounded-pill" type="submit" value="Add" /></div> </div> </form> </div> </div> </div> </div> ` const modalElement = $.parseHTML(modalString) $(document.body).append(modalElement) } showModal(configType){ const listModal = $("#chatHighlighterListModal") this.populateModal(configType) listModal.attr("data-configtype", configType) $("#chatHighlighterListModalTitle").text(configType) listModal.modal('show') document.body.scrollTop = document.documentElement.scrollTop = 0 } populateModal(configType){ const data_set = this[configType] $("#chatHighlighterListModalList").empty() data_set.forEach(item => { this.addToList(item) }) } addToList(item){ const ident = item.match(/[a-z]/g).join('') 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>` const newItemElement = $.parseHTML(newItemString) $("#chatHighlighterListModalList").append(newItemElement) } addFromModal(){ const inputBox = $("#highlightListInput") const configType = $("#chatHighlighterListModal").attr("data-configtype") const newItem = inputBox.val() inputBox.val("") this.addItem(newItem, configType) } addItem(newItem, configType){ this[configType].add(newItem) this.addToList(newItem) this.saveData() } removeItem(item, id){ const configType = $("#chatHighlighterListModal").attr("data-configtype") this[configType].delete(item) $(`#${id}`).remove() this.saveData() } toRGBA(hex) { const r = parseInt(hex.slice(1, 3), 16); const g = parseInt(hex.slice(3, 5), 16); const b = parseInt(hex.slice(5, 7), 16); return `rgba(${r}, ${g}, ${b}, 0.15)`; } highlightMessage(data, highlightType){ const notificationsEnabled = this.getConfig("notificationsEnabled"); const soundsEnabled = this.getConfig("soundsEnabled"); const activeName = this.getConfig("activeName"); let highlightColour = "" const wordColour = this.toRGBA(this.getConfig("colourWordHighlight")) const friendColour = this.toRGBA(this.getConfig("colourFriendHighlight")) if(highlightType === "word") {highlightColour = wordColour} else if(highlightType === "user") {highlightColour = friendColour} else {highlightColour = "rgba(0, 0, 0, 0)"} const element = $("#chat-area > *").last(); element.attr("style", `background-color: ${highlightColour}`) if (highlightType === "word"){ if (activeName === var_username || activeName === ""){ if (soundsEnabled){Sounds.play(Sounds.VARIABLE_POWER_UP);} if (notificationsEnabled){this.notify(data.message, data.username)} } } } notify(message, username){ if (!window.Notification) { alert("Sorry, Notifications are not supported in this Browser!"); } else { if (Notification.permission === 'default') { Notification.requestPermission(function(p) { if (p === 'denied') { alert('You have denied Notifications'); } else { var notify = new Notification('Chat Notification', { body: `${username}: ${message}`, requireInteraction: true, icon: bob }); } }); } else { var notify = new Notification('Chat Notification', { body: `${username}: ${message}`, icon: bob, requireInteraction: true }); } } } onChat(data) { const ignoreCase = this.getConfig("ignoreCase"); let message let word_list let ignore_word_list const user_list = [...this.user_set] const ignore_user_list = [...this.ignore_user_set] if (ignoreCase) { message = data.message.toLowerCase() word_list = [...this.word_set].map(word => word.toLowerCase()) ignore_word_list = [...this.ignore_word_set].map(word => word.toLowerCase()) } else { message = data.message word_list = [...this.word_set] ignore_word_list = [...this.ignore_word_set] } if(ignore_user_list.includes(data.username)){ return } if (word_list.some(word => message.includes(word))) { if (!ignore_word_list.some(word => message.includes(word))){ this.highlightMessage(data, "word"); } } else if (user_list.includes(data.username)){ this.highlightMessage(data, "user"); } } } const plugin = new HighlightPlugin(); var bob = "" IdlePixelPlus.registerPlugin(plugin); })();