TXTbonk

Bonk!

目前為 2023-03-12 提交的版本,檢視 最新版本

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

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

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

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

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

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

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name         TXTbonk
// @namespace    http://tampermonkey.net/
// @version      0.2
// @description  Bonk!
// @author       tharglet
// @license      MIT
// @match        https://myfigurecollection.net/*
// @icon         https://www.google.com/s2/favicons?sz=64&domain=myfigurecollection.net
// @grant        GM_addStyle
// @grant        GM_setValue
// @grant        GM_getValue
// ==/UserScript==
//Polyfill for GM_addStyle for Greasemonkey...
if(typeof GM_addStyle == 'undefined') {
    GM_addStyle = (aCss) => {
        'use strict';
        let head = document.getElementsByTagName('head')[0];
        if (head) {
            let style = document.createElement('style');
            style.setAttribute('type', 'text/css');
            style.textContent = aCss;
            head.appendChild(style);
            return style;
        }
        return null;
    };
}

GM_addStyle(`
.bonk_menu {
  position: relative;
  display: inline-block;
  cursor: pointer;
}

.bonk_menu-body {
  display: none;
  position: absolute;
  background-color: #f9f9f9;
  min-width: 160px;
  box-shadow: 0px 8px 16px 0px rgba(0,0,0,0.2);
  padding: 12px 16px;
  z-index: 1;
}

.bonk_alert {
  padding: 20px;
  font-size: 50px;
  background:#f7f7f7;
  border: 2px black solid;
  color: green;
  position:fixed;
  left:25%;
  top:40%;
}
`);

(function() {
    'use strict';
    // The default graphic
    let bonkGraphic = ''
    // Array of {itemId: {toBonk: bool, date: Date}}
    let bonkables = JSON.parse(GM_getValue('txtbonk', '{}'));
    //Stores the alert timeout
    let alertDivTimeout;
    let ONE_WEEK_AGO = new Date(Date.now() - 604800000);

    // Main "bonk" action to change the image of a .TXT icon
    let bonk = (imgTag) => {
        imgTag.setAttribute('src', bonkGraphic);
    };

    // Saves if a NSFW+ item should be "bonked" or not to the cache array
    let saveBonkable = (itemId, toBonk) => {
        bonkables[itemId] = {toBonk: toBonk, date: (new Date()).toString()}
        GM_setValue("txtbonk", JSON.stringify(bonkables));
    };

    // Fetches the item page for a NSFW+ item and see if it should be "bonked"
    // The result is stored in the bonkables array so that items do not have to be constantly checked
    let fetchIfBonkable = (itemId, imgTag) => {
        fetch(`/item/${itemId}`).then((response) => {
                    return response.text();
                }).then((html) => {
                    var parser = new DOMParser();
                    var doc = parser.parseFromString(html, 'text/html');
                    var img = doc.querySelector('.item-picture');
                    if(!img) {
                        saveBonkable(itemId, true);
                        bonk(imgTag);
                    } else {
                        saveBonkable(itemId, false);
                    }
                }).catch(function (err) {
                    console.warn('Something went wrong.', err);
                });
    };

    // Function that scans the page for bonkable thumbnails
    let scanForBonkables = (icons, cacheReset) => {
        for(let icon of icons) {
            let imgTag = icon.getElementsByTagName('img')[0];
            if(imgTag.getAttribute('alt').startsWith('[NSFW+]')) {
                let metaTag = icon.getElementsByTagName('meta')[0];
                let itemId = metaTag.getAttribute('content').substring(2);
                let bonkable = bonkables[itemId];
                if(bonkable) {
                    if(new Date(bonkable.date) < ONE_WEEK_AGO || cacheReset) {
                        fetchIfBonkable(itemId, imgTag);
                    } else if(bonkable.toBonk) {
                        bonk(imgTag);
                    }
                } else {
                    fetchIfBonkable(itemId, imgTag);
                }
            }
        };
    }

    // Toggle menu display on click
    let clickBonkMenu = () => {
        let bonkMenu = document.querySelector('.bonk_menu-body');
        if(bonkMenu.style.display === 'block') {
            bonkMenu.style.display = 'none';
        } else {
            bonkMenu.style.display = 'block';
        }
    }

    // Hides the alert div
    let hideAlertDiv = () => {
        let alertDiv = document.querySelector('.bonk_alert');
        alertDiv.style.display = 'none';
        if(alertDivTimeout) {
            window.clearTimeout(alertDivTimeout);
        }
    }

    // Shows the alert div with given message
    let showAlertDiv = (message) => {
        if(alertDivTimeout) {
            window.clearTimeout(alertDivTimeout);
        }
        let alertDiv = document.querySelector('.bonk_alert');
        alertDiv.innerText = message;
        alertDiv.style.display = 'block';
        alertDivTimeout = window.setTimeout(hideAlertDiv, 5000);
    }

    // Scans page again for bonkables
    let resetCache = () => {
        scanForBonkables(icons, true);
        showAlertDiv('Refreshed');
    }

    // Sets up the alert div for displaying messages
    let setupAlertDiv = () => {
        let alertDiv = document.createElement('div');
        alertDiv.classList.add('bonk_alert');
        alertDiv.style.display = 'none';
        alertDiv.addEventListener('click', hideAlertDiv, false);
        let innerText = document.createTextNode('bonk');
        alertDiv.appendChild(innerText);
        let body = document.querySelector('body');
        body.appendChild(alertDiv);
    }

    // Sets up the bonkable menu
    let setupMenu = () => {
        let pageMenuDesktop = document.querySelector('.h1-meta-actions.desktop');
        if(!pageMenuDesktop) {
            pageMenuDesktop = document.querySelector('.h1-meta-actions');
        }
        let pageMenuMobile = document.querySelector('.h1-meta-actions.mobile');

        let menuContainer = document.createElement('a');
        menuContainer.classList.add('bonk_menu', 'action');
        menuContainer.addEventListener('click', clickBonkMenu, false);
        let innerSpanIcon = document.createElement('span');
        innerSpanIcon.classList.add('icon', 'icon-sliders');
        let innerText = document.createTextNode('Bonk!');
        menuContainer.appendChild(innerSpanIcon);
        menuContainer.appendChild(innerText);
        pageMenuDesktop.appendChild(menuContainer);

        let menuBody = document.createElement('div');
        menuBody.classList.add('bonk_menu-body');
        let menuItemsUl = document.createElement('ul');
        let resetCacheLi = document.createElement('li');
        let resetCacheText = document.createTextNode('Reset cache & and reload for this page');
        resetCacheLi.appendChild(resetCacheText);
        resetCacheLi.addEventListener('click', resetCache, false);
        menuItemsUl.appendChild(resetCacheLi);

        menuBody.appendChild(menuItemsUl);
        menuContainer.appendChild(menuBody);

        if(pageMenuMobile) {
            //Setup mobile menu
        }

    }

    // Bonk initialisation
    let icons = document.getElementsByClassName('item-icon');
    if(icons) {
        setupMenu();
        setupAlertDiv();
        scanForBonkables(icons, false);
    }
})();