IdlePixel Chat Highlighter

Highlights messages containing specified words, or from specified users.

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 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);
})();