IdlePixel Chat Highlighter

Highlights messages containing specified words, or from specified users.

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

// ==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);
})();