👑 EMPRESS 💅 v1.1.0

2 MODULES: Hide Torrents v1.0.0, Click to Nail v1.0.1

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

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

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

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

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

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

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

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

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

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

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

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

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

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

// ==UserScript==
// @name         👑 EMPRESS 💅 v1.1.0
// @namespace    https://empornium.is/
// @version      1.1.0
// @description  2 MODULES: Hide Torrents v1.0.0, Click to Nail v1.0.1
// @author       Anonymous + SerpentGPT
// @license      MIT
// @match        https://*.empornium.is/*
// @match        https://*.empornium.sx/*
// @match        https://*.empornium.me/*
// @grant        none
// ==/UserScript==


// 👑 HIDE TORRENTS v1.0.0
(function () {
    'use strict';

const HIDE_KEY = 'emp_hidden_torrents';
    let lastHidden = null;

    const get = (key) => {
        try {
            return JSON.parse(localStorage.getItem(key) || '{}');
        } catch {
            return {};
        }
    };
    const set = (key, value) => localStorage.setItem(key, JSON.stringify(value));

    const getId = (a) => (a.href.match(/id=(\d+)/) || [])[1];

    function styleButtons() {
        const style = document.createElement('style');
        style.textContent = `
            .empress-btn {
                margin-left: 6px;
                cursor: pointer;
                background: transparent;
                border: none;
                font-size: 14px;
                transition: transform 0.2s;
            }
            .empress-btn:hover {
                transform: scale(1.2);
                text-shadow: 0 0 4px hotpink;
            }
            #empress-modal {
                position: fixed;
                top: 50%;
                left: 50%;
                transform: translate(-50%, -50%);
                background: #222;
                color: white;
                padding: 20px;
                border-radius: 10px;
                box-shadow: 0 0 10px hotpink;
                z-index: 10000;
                min-width: 300px;
            }
            #empress-modal-close {
                float: right;
                cursor: pointer;
                color: hotpink;
                font-weight: bold;
            }
        `;
        document.head.appendChild(style);
    }

    function createButton(id, row, title) {
        const btn = document.createElement('button');
        btn.classList.add('empress-btn');
        btn.textContent = '🔕';
        btn.title = 'Hide this torrent';

        btn.onclick = () => {
            const list = get(HIDE_KEY);
            if (!list[id]) {
                list[id] = title;
                lastHidden = id;
                set(HIDE_KEY, list);
                row.style.display = 'none';
            }
        };
        return btn;
    }

    function processTorrents() {
        const hidden = get(HIDE_KEY);

        document.querySelectorAll('a[href*="torrents.php?id="]').forEach(a => {
            if (a.dataset.empressDone) return;
            const id = getId(a);
            const row = a.closest('tr');
            const td = a.closest('td');
            const title = a.textContent.trim();
            if (!id || !row || !td) return;

            a.dataset.empressDone = 'true';

            if (hidden[id]) {
                row.style.display = 'none';
                return;
            }

            td.appendChild(createButton(id, row, title));
        });
    }

    function showSettingsModal() {
        if (document.getElementById('empress-modal')) return;

        const modal = document.createElement('div');
        modal.id = 'empress-modal';
        modal.innerHTML = `
            <div id="empress-modal-close">✖</div>
            <h3>👑 EMPRESS 💅</h3>
            <button id="showHidden">Show Hidden Torrents</button>
            <button id="undoLastHide">Undo Last Hide</button>
        `;
        document.body.appendChild(modal);

        document.getElementById('empress-modal-close').onclick = () => {
            modal.remove();
        };

        document.getElementById('showHidden').onclick = () => {
            try {
                const hidden = get(HIDE_KEY);
                Object.keys(hidden).forEach(id => {
                    const links = Array.from(document.querySelectorAll(`a[href*="torrents.php?id=${id}"]`));
                    links.forEach(link => {
                        const row = link.closest('tr');
                        if (row) {
                            row.style.display = '';
                        }
                    });
                });
            } catch (e) {
                console.error('Error while showing hidden torrents:', e);
            }
        };

        document.getElementById('undoLastHide').onclick = () => {
            if (!lastHidden) return;
            const hidden = get(HIDE_KEY);
            delete hidden[lastHidden];
            set(HIDE_KEY, hidden);

            const links = Array.from(document.querySelectorAll(`a[href*="torrents.php?id=${lastHidden}"]`));
            links.forEach(link => {
                const row = link.closest('tr');
                if (row) {
                    row.style.display = '';
                }
            });

            lastHidden = null;
        };
    }

    function addSettingsLinkToNavbar() {
        const ul = document.createElement('ul');
        const li = document.createElement('li');
        ul.appendChild(li);
        ul.style.display = 'inline-block';

        const a = document.createElement('a');
        a.href = '#';
        a.textContent = '👑 EMPRESS 💅';
        a.addEventListener('click', showSettingsModal);
        li.appendChild(a);

        const stats = document.querySelector('#major_stats');
        if (stats) stats.prepend(ul);
    }

    function waitForTorrents() {
        const interval = setInterval(() => {
            const found = document.querySelector('a[href*="torrents.php?id="]');
            if (found) {
                clearInterval(interval);
                processTorrents();
            }
        }, 500);
    }

    window.addEventListener('load', () => {
        styleButtons();
        addSettingsLinkToNavbar();
        waitForTorrents();
        setInterval(processTorrents, 3000);
    });
})();


// 💎 CLICK TO NAIL v1.0.1

(function () {
  'use strict';

  const VIEWER_ID = 'emp-viewer-box';
  let stickyOffsetTop = null;
  let stickyActive = false;

  function insertViewerBox() {
    let box = document.getElementById(VIEWER_ID);
    if (!box) {
      box = document.createElement('div');
      box.id = VIEWER_ID;
      box.style = 'border: 2px solid hotpink; padding: 10px; margin: 10px 0; background:#111; z-index:9999;';
      box.innerHTML = `
        <div style="display:flex; justify-content:space-between; align-items:center;">
          <h3 style="color:hotpink;margin:0;">EMP ViewerBox 💎</h3>
          <button id="emp-viewer-close" style="font-size:20px; background:none; color:white; border:none; cursor:pointer;">&times;</button>
        </div>
        <div id="emp-viewer-gallery" style="margin-top:10px; display:flex; flex-wrap:nowrap; overflow-x:auto; overflow-y:hidden; gap:6px; align-items:flex-start; height:140px;"></div>
      `;
      const bottomPager = document.querySelector('div.linkbox.pager');
      if (bottomPager && bottomPager.parentElement) bottomPager.parentElement.insertBefore(box, bottomPager);
      else document.body.prepend(box);
      document.getElementById('emp-viewer-close').onclick = () => box.remove();
      stickyOffsetTop = box.offsetTop;
    }
  }

  function handleStickyBehavior() {
    const box = document.getElementById(VIEWER_ID);
    if (!box || stickyOffsetTop === null) return;
    const scrollTop = window.scrollY || document.documentElement.scrollTop;
    if (scrollTop > stickyOffsetTop && !stickyActive) {
      box.style.position = 'fixed';
      box.style.top = '0';
      box.style.left = '0';
      box.style.right = '0';
      box.style.boxShadow = '0 2px 6px rgba(255,105,180,0.5)';
      stickyActive = true;
    } else if (scrollTop <= stickyOffsetTop && stickyActive) {
      box.style.position = 'static';
      box.style.boxShadow = 'none';
      stickyActive = false;
    }
  }

  function showInViewerBox(imageUrls) {
    insertViewerBox();
    const gallery = document.getElementById('emp-viewer-gallery');
    gallery.innerHTML = '';

    let lightbox = document.getElementById('emp-lightbox');
    if (!lightbox) {
      lightbox = document.createElement('div');
      lightbox.id = 'emp-lightbox';
      lightbox.style = `
        position: fixed;
        top: 0; left: 0; right: 0; bottom: 0;
        background: rgba(0, 0, 0, 0.9);
        display: none;
        justify-content: center;
        align-items: center;
        z-index: 999999;
        cursor: zoom-out;
      `;
      lightbox.innerHTML = `<div id="emp-lightbox-pan" style="overflow: hidden; max-width: 90vw; max-height: 90vh;">
        <img id="emp-lightbox-img" style="border-radius:4px; transition: transform 0.3s ease; cursor: grab;">
      </div>`;
      document.body.appendChild(lightbox);
    }

    const panWrapper = document.getElementById('emp-lightbox-pan');
    const lightboxImg = document.getElementById('emp-lightbox-img');

    let currentIndex = 0;
    const allImages = imageUrls;

    panWrapper.addEventListener('wheel', (e) => {
      if (lightbox.style.display !== 'flex') return;
      if (Math.abs(e.deltaY) < Math.abs(e.deltaX)) return;
      e.preventDefault();
      currentIndex += e.deltaY > 0 ? 1 : -1;
      if (currentIndex < 0) currentIndex = allImages.length - 1;
      if (currentIndex >= allImages.length) currentIndex = 0;

      const nextUrl = allImages[currentIndex];
      lightboxImg.src = nextUrl;
      lightboxImg.dataset.zoomed = 'false';
      lightboxImg.style.width = 'auto';
      lightboxImg.style.height = 'auto';
      lightboxImg.style.maxWidth = '90vw';
      lightboxImg.style.maxHeight = '90vh';
      panWrapper.scrollTop = 0;
      panWrapper.scrollLeft = 0;
      panWrapper.style.overflow = 'hidden';
    }, { passive: false });

    let isDragging = false, startX = 0, startY = 0, scrollLeft = 0, scrollTop = 0, hasMoved = false;

    panWrapper.addEventListener('mousedown', (e) => {
      if (lightboxImg.dataset.zoomed === 'true') {
        isDragging = true;
        hasMoved = false;
        startX = e.pageX - panWrapper.offsetLeft;
        startY = e.pageY - panWrapper.offsetTop;
        scrollLeft = panWrapper.scrollLeft;
        scrollTop = panWrapper.scrollTop;
        lightboxImg.style.cursor = 'grabbing';
      }
    });

    panWrapper.addEventListener('mouseleave', () => {
      isDragging = false;
      lightboxImg.style.cursor = 'grab';
    });

    panWrapper.addEventListener('mouseup', () => {
      isDragging = false;
      setTimeout(() => { if (hasMoved) lightboxImg.style.cursor = 'grab'; }, 50);
    });

    panWrapper.addEventListener('mousemove', (e) => {
      if (!isDragging) return;
      e.preventDefault();
      hasMoved = true;
      const x = e.pageX - panWrapper.offsetLeft;
      const y = e.pageY - panWrapper.offsetTop;
      const walkX = (x - startX);
      const walkY = (y - startY);
      panWrapper.scrollLeft = scrollLeft - walkX;
      panWrapper.scrollTop = scrollTop - walkY;
    });

    lightbox.onclick = (e) => {
      if (e.target.id === 'emp-lightbox-img') {
        if (lightboxImg.dataset.zoomed === 'false') {
          lightboxImg.style.width = lightboxImg.naturalWidth + 'px';
          lightboxImg.style.height = lightboxImg.naturalHeight + 'px';
          lightboxImg.style.maxWidth = 'none';
          lightboxImg.style.maxHeight = 'none';
          lightboxImg.dataset.zoomed = 'true';
          lightbox.style.cursor = 'zoom-in';
          panWrapper.style.overflow = 'auto';
          panWrapper.scrollLeft = 0;
          panWrapper.scrollTop = 0;
        } else {
          lightboxImg.style.width = 'auto';
          lightboxImg.style.height = 'auto';
          lightboxImg.style.maxWidth = '90vw';
          lightboxImg.style.maxHeight = '90vh';
          lightboxImg.dataset.zoomed = 'false';
          lightbox.style.cursor = 'zoom-out';
          panWrapper.scrollTop = 0;
          panWrapper.scrollLeft = 0;
          panWrapper.style.overflow = 'hidden';
        }
      } else {
        lightbox.style.display = 'none';
      }
    };

    imageUrls.forEach((url) => {
      const thumb = document.createElement('img');
      const preview = url.replace(/(\.(jpg|jpeg|png|gif|webp))$/i, '.th$1');
      thumb.src = preview;
      thumb.loading = 'lazy';
      thumb.onerror = () => {
        thumb.src = url;
        thumb.style.opacity = 0.3;
        thumb.title = "Broken preview 😢";
      };
      thumb.style = 'height:100%; max-height:100%; width:auto; object-fit:contain; border-radius:4px; display:block; cursor: zoom-in;';
      thumb.onclick = () => {
        lightbox.style.display = 'flex';
        lightboxImg.src = url;
        currentIndex = imageUrls.indexOf(url);
        lightboxImg.dataset.zoomed = 'false';
        lightboxImg.style.width = 'auto';
        lightboxImg.style.height = 'auto';
        lightboxImg.style.maxWidth = '90vw';
        lightboxImg.style.maxHeight = '90vh';
        panWrapper.scrollTop = 0;
        panWrapper.scrollLeft = 0;
        panWrapper.style.overflow = 'hidden';
      };
      const wrapper = document.createElement('div');
      wrapper.style = 'flex: 0 0 auto;';
      wrapper.appendChild(thumb);
      gallery.appendChild(wrapper);
    });
  }

  function extractHamsterImages(container) {
    const imgs = new Set();
    container.querySelectorAll('img[src*="hamster.is"], img[data-src*="hamster.is"]').forEach(img => {
      const real = img.dataset.src || img.src;
      if (real) imgs.add(real.replace(/\.(th|md)\./i, '.'));
    });
    container.querySelectorAll('a[href*="hamster.is"]').forEach(a => {
      if (a.href.match(/\.(jpg|jpeg|png|gif|webp)$/i)) imgs.add(a.href);
    });
    return Array.from(imgs);
  }

  function interceptTorrentLinks() {
    document.querySelectorAll('a[href*="torrents.php?id="]').forEach(link => {
      if (link.dataset.viewerIntercepted) return;
      link.dataset.viewerIntercepted = 'true';

      link.addEventListener('click', async e => {
        e.preventDefault();
        const url = link.href;
        try {
          const html = await fetch(url).then(r => r.text());
          const parser = new DOMParser();
          const doc = parser.parseFromString(html, 'text/html');
          const descbox = doc.querySelector('#descbox') || doc.querySelector('#details_top');
          if (!descbox) return;
          const images = extractHamsterImages(descbox);
          if (images.length > 0) showInViewerBox(images);
        } catch (err) {
          console.error('[ViewerBox] Failed to fetch torrent page', err);
        }
      });
    });
  }

  window.addEventListener('scroll', handleStickyBehavior);
  window.addEventListener('load', () => {
    interceptTorrentLinks();
    setInterval(() => {
      try {
        interceptTorrentLinks();
      } catch (err) {
        console.warn('[ViewerBox] Retry failed:', err);
      }
    }, 3000);
  });
})();