98预告贴小助手

即点即看不跳转,周周预告看到爽

// ==UserScript==
// @name         98预告贴小助手
// @license      MIT
// @namespace    http://tampermonkey.net/
// @version      4.0.5
// @description  即点即看不跳转,周周预告看到爽
// @author       Lsssy
// @include      *://*.sehuatang.*/*
// @include      *://sehuatang.*/*
// @icon         
// @grant        none
// ==/UserScript==

(function () {
    'use strict';
    const regex = /([A-Za-z]{3,})-(\d{3})/g;
    const pad = s => s.padStart(5, '0');
    const buildUrl = (letters, digits) => {
        const l = letters.toLowerCase();
        const pd = pad(digits);
        return `https://cc3001.dmm.co.jp/litevideo/freepv/${l[0]}/${l.slice(0, 3)}/${l+pd}/${l+pd}hhb.mp4`;
    };

    const walker = document.createTreeWalker(document.body, NodeFilter.SHOW_TEXT, {
        acceptNode: node => {
            if (node.parentNode && /^(SCRIPT|STYLE|TEXTAREA|A)$/i.test(node.parentNode.tagName)) {
                return NodeFilter.FILTER_REJECT;
            }
            return NodeFilter.FILTER_ACCEPT;
        }
    });
    const nodes = [];
    while (walker.nextNode()) nodes.push(walker.currentNode);

    nodes.forEach(node => {
        if (node.parentNode && node.parentNode.closest('a')) return;

        const newContent = node.textContent.replace(regex, (m, letters, digits) => {
            if (/^(sta|ab|ff|wi|kb|ki)/i.test(letters)) return m;
            if (/\.jpg$|\.png$|\.gif$/i.test(node.textContent)) return m;
            return `<a href="${buildUrl(letters, digits)}" target="_blank">${m.toUpperCase()}</a>`;
        });

        if (newContent !== node.textContent) {
            const span = document.createElement('span');
            span.innerHTML = newContent;
            node.parentNode.replaceChild(span, node);
        }
    });
})();

(function () {
    'use strict';

    const fixUrl = url => {
        if (!/^https?:\/\//.test(url)) {
            url = 'https://' + url;
        }
        return url.replace(/^(https?:\/\/)?(?:[a-zA-Z0-9.-]+)?\.dmm\.co\.jp/, 'https://cc3001.dmm.co.jp');
    };

    const formatNumber = num => num.toUpperCase().replace(/^([a-zA-Z]+)0*(\d+)$/, (_, prefix, digits) => {
        return `${prefix}-${digits.padStart(3, '0')}`;
    });

    const replaceTextWithLink = node => {
        if (node.nodeType === Node.TEXT_NODE) {
            const regex = /((?:https?:\/\/)?(?:[a-zA-Z0-9.-]+)?\.dmm\.co\.jp\/.*?\/)([\w]+_\w+_\w+\.mp4)/g;
            let text = node.nodeValue, match, parent = node.parentNode, lastIndex = 0;

            while ((match = regex.exec(text)) !== null) {
                parent.insertBefore(document.createTextNode(text.slice(lastIndex, match.index)), node);
                const link = document.createElement('a');
                link.href = fixUrl(match[1] + match[2]);
                link.textContent = formatNumber(match[2].split('_')[0]);
                parent.insertBefore(link, node);
                lastIndex = regex.lastIndex;
            }

            parent.insertBefore(document.createTextNode(text.slice(lastIndex)), node);
            parent.removeChild(node);
        } else if (node.nodeType === Node.ELEMENT_NODE) {
            Array.from(node.childNodes).forEach(replaceTextWithLink);
        }
    };

    replaceTextWithLink(document.body);
})();

(function() {
    'use strict';

    document.querySelectorAll('a[href*="pv3001"], a[href*="dmm.com"], a[href*="_dm_w"]')
        .forEach(function(link) {
            link.href = link.href
                .replace('pv3001', 'cc3001')
                .replace('dmm.com', 'dmm.co.jp')
                .replace('_dm_w', 'hhb');
        });
})();

(function () {
    'use strict';

    const isMobile = /Mobi|Android|iPhone/i.test(navigator.userAgent);

    function createEl(tag, { style = {}, attrs = {}, textContent = '', innerHTML = '' } = {}) {
        const el = document.createElement(tag);
        Object.assign(el.style, style);
        Object.entries(attrs).forEach(([key, value]) => el.setAttribute(key, value));
        if (textContent) el.textContent = textContent;
        if (innerHTML) el.innerHTML = innerHTML;
        return el;
    }

    const videoModal = createEl('div', {
        style: {
            position: 'fixed',
            top: 0,
            left: 0,
            width: '100vw',
            height: '100vh',
            backgroundColor: 'rgba(0,0,0,0.7)',
            display: 'none',
            justifyContent: 'center',
            alignItems: 'center',
            zIndex: 1000,
            cursor: 'pointer'
        }
    });
    document.body.appendChild(videoModal);

    const videoElement = createEl('video', {
        style: isMobile
            ? { width: '90%', height: 'auto', border: 'none' }
            : { maxWidth: '80%', maxHeight: '80%', border: 'none' },
        attrs: { controls: '' }
    });
    videoElement.muted = true;
    videoModal.appendChild(videoElement);

    const videoTitle = createEl('div', {
        style: isMobile
            ? { position: 'absolute', top: '5%', width: '90%', textAlign: 'center', fontSize: '18px', color: 'white', zIndex: 1001, fontFamily: 'Arial, sans-serif', opacity: 0.9 }
            : { position: 'absolute', top: '10px', width: '100%', textAlign: 'center', fontSize: '22px', color: 'white', zIndex: 1001, fontFamily: 'Arial, sans-serif', opacity: 0.9 }
    });
    videoModal.appendChild(videoTitle);

    const navBtnStyle = isMobile
        ? { position: 'absolute', bottom: '10%', backgroundColor: 'rgba(0,0,0,0.5)', border: 'none', padding: '10px', cursor: 'pointer', transition: 'background-color 0.3s ease' }
        : { position: 'absolute', top: '50%', transform: 'translateY(-50%)', backgroundColor: 'rgba(0,0,0,0.5)', border: 'none', padding: '10px', cursor: 'pointer', transition: 'background-color 0.3s ease' };

    const leftButton = createEl('button', {
        innerHTML: `<svg viewBox="0 0 24 24" width="32" height="32" fill="white" style="vertical-align:middle;">
            <path d="M15.41 7.41L14 6l-6 6 6 6 1.41-1.41L10.83 12z"/>
        </svg>`
    });
    Object.assign(leftButton.style, navBtnStyle, isMobile ? { left: '10%' } : { left: '20px' });
    videoModal.appendChild(leftButton);

    const rightButton = createEl('button', {
        innerHTML: `<svg viewBox="0 0 24 24" width="32" height="32" fill="white" style="vertical-align:middle;">
            <path d="M8.59 16.59L10 18l6-6-6-6-1.41 1.41L13.17 12z"/>
        </svg>`
    });
    Object.assign(rightButton.style, navBtnStyle, isMobile ? { right: '10%' } : { right: '20px' });
    videoModal.appendChild(rightButton);

    const videoIndexDisplay = createEl('div', {
        style: isMobile
            ? { position: 'fixed', top: '5%', left: '5%', fontSize: '16px', color: 'white', zIndex: 1002, fontFamily: 'Arial, sans-serif', opacity: 0.9 }
            : { position: 'fixed', top: '10px', left: '10px', fontSize: '18px', color: 'white', zIndex: 1002, fontFamily: 'Arial, sans-serif', opacity: 0.9 }
    });
    document.body.appendChild(videoIndexDisplay);

    let videoLinks = [];
    let currentIndex = -1;
    let isNavigatingVideos = false;

    function bindVideoLinks() {
        const links = Array.from(document.querySelectorAll('a[href*=".mp4"]'));
        links.forEach(link => {
            link.removeEventListener('click', handleLinkClick);
            link.addEventListener('click', handleLinkClick);
        });
    }

    function handleLinkClick(e) {
        e.preventDefault();
        const links = Array.from(document.querySelectorAll('a[href*=".mp4"]'));
        videoLinks = links.map(link => ({
            href: link.href,
            title: link.textContent.trim()
        }));
        currentIndex = videoLinks.findIndex(video => video.href === this.href);
        if (currentIndex === -1) return;
        playVideo(videoLinks[currentIndex]);
        videoModal.style.display = 'flex';
    }

    function playVideo({ href, title }) {
        videoElement.src = href;
        videoElement.play();
        videoTitle.textContent = title;
        updateVideoIndexDisplay();
    }

    function updateVideoIndexDisplay() {
        videoIndexDisplay.textContent = `${currentIndex + 1} / ${videoLinks.length}`;
    }

    videoModal.addEventListener('click', (e) => {
        if (e.target === videoModal) {
            videoElement.pause();
            videoModal.style.display = 'none';
        }
    });

    window.addEventListener('keydown', (e) => {
        if (e.key === 'ArrowLeft') {
            isNavigatingVideos ? loadPreviousVideo() : videoElement.currentTime = Math.max(videoElement.currentTime - 5, 0);
        } else if (e.key === 'ArrowRight') {
            isNavigatingVideos ? loadNextVideo() : videoElement.currentTime += 5;
        }
    });

    leftButton.addEventListener('click', (e) => {
        e.stopPropagation();
        isNavigatingVideos = true;
        loadPreviousVideo();
    });

    rightButton.addEventListener('click', (e) => {
        e.stopPropagation();
        isNavigatingVideos = true;
        loadNextVideo();
    });

    function loadPreviousVideo() {
        if (!videoLinks.length) return;
        currentIndex = (currentIndex === 0) ? videoLinks.length - 1 : currentIndex - 1;
        playVideo(videoLinks[currentIndex]);
    }

    function loadNextVideo() {
        if (!videoLinks.length) return;
        currentIndex = (currentIndex === videoLinks.length - 1) ? 0 : currentIndex + 1;
        playVideo(videoLinks[currentIndex]);
    }

    videoElement.addEventListener('fullscreenchange', () => {
        videoElement.style.border = 'none';
    });

    videoElement.addEventListener('click', (e) => {
        e.stopPropagation();
        isNavigatingVideos = false;
    });

    window.addEventListener('load', bindVideoLinks);

    const observer = new MutationObserver(bindVideoLinks);
    observer.observe(document.body, { childList: true, subtree: true });
})();