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