// ==UserScript==
// @name Pixiv Previewer
// @namespace https://github.com/lintmx
// @version 0.0.1
// @description preview image in pixiv.
// @author lintmx
// @license MIT
// @match https://www.pixiv.net*
// @grant none
// @run-at document-idle
// ==/UserScript==
const isDev = false;
const Logger = {
debug: (...msg) => isDev && console.log('[Pixiv Previewer]', 'DEBUG:', ...msg),
info: (...msg) => console.log('[Pixiv Previewer]', 'INFO:', ...msg),
error: (...msg) => console.log('[Pixiv Previewer]', 'ERROR:', ...msg),
};
const Icon = {
Expand:
'',
Redirect:
'',
};
const Previewer = {
addButton: () => {
Logger.debug('Start Add Button');
const illustList = document.querySelectorAll('div[type=illust]>div:not(.pixiv-previewer-btn-exist)');
for (const illust of illustList) {
const previewButton = document.createElement('div');
previewButton.className = 'pixiv-previewer-view-btn';
previewButton.style.cssText = [
'position: absolute',
'left: 0px',
'bottom: 0px',
'width: 32px',
'height: 32px',
'cursor: pointer',
'background: no-repeat center/50%',
'background-image: url("' + Icon.Expand + '")',
].join(';');
previewButton.innerHTML = ' ';
previewButton.addEventListener('click', (e) => {
const clickImgElm = e.target.parentNode.querySelector('a>div:last-child>img');
const pageSizeElm = e.target.parentNode.querySelector(
'a>div:first-child>div:last-child>div>span:last-child'
);
if (!clickImgElm) return;
const imgUrl = clickImgElm.src;
const pageSize = pageSizeElm ? pageSizeElm.innerText : '1';
Previewer.showPreview(imgUrl, pageSize);
});
illust.appendChild(previewButton);
illust.classList.add('pixiv-previewer-btn-exist');
}
},
showPreview: (url, pageSize = 1) => {
const { Time, IllustId } = Previewer.convert(url);
const Cover = document.querySelector('#pixiv-previewer-cover');
Cover.style.display = 'flex';
Cover.dataset.time = Time;
Cover.dataset.id = IllustId;
Cover.dataset.size = pageSize;
Cover.dataset.page = 0;
document.querySelector('#pixiv-previewer-redirect').href = 'https://www.pixiv.net/artworks/' + IllustId;
document.querySelector(
'#pixiv-previewer-image'
).src = `https://i.pximg.net/img-master/img/${Time}/${IllustId}_p0_master1200.jpg`;
},
convert: (baseUrl = '') => {
const [, Time = '', IllustId = ''] = baseUrl.match(
/^https:\/\/i\.pximg\.net\/.+?\/img\/([0-9\/]+?)\/(\d+?)_p.+?\.[a-z]+$/
);
return { Time, IllustId };
},
};
(function () {
('use strict');
Logger.debug('Script Start.');
const rootElm = document.querySelector('#root');
// 全屏预览遮罩
const previewCover = document.createElement('div');
previewCover.id = 'pixiv-previewer-cover';
previewCover.style.cssText = [
'z-index: 445',
'width: 100%',
'height: 100%',
'position: fixed',
'left: 0px',
'right: 0px',
'top: 0px',
'bottom: 0px',
'display: none',
'justify-content: center',
'align-items: center',
'flex-direction: column',
'background: rgba(0, 0, 0, 0.6)',
].join(';');
previewCover.addEventListener('click', (e) => {
Logger.debug('cover click.');
e.target.style.display = 'none';
});
const previewRedirect = document.createElement('a');
previewRedirect.id = 'pixiv-previewer-redirect';
previewRedirect.style.cssText = [
'width: 32px',
'height: 32px',
'background: red',
'margin-top: 16px',
'display: block',
'background: no-repeat center',
'background-image: url("' + Icon.Redirect + '")',
].join(';');
previewCover.appendChild(previewRedirect);
const previewImage = document.createElement('img');
previewImage.id = 'pixiv-previewer-image';
previewImage.style.cssText = ['max-width: 90%', 'max-height: 90%', 'margin: auto'].join(';');
previewImage.addEventListener('click', (e) => {
e.stopPropagation();
const Cover = document.querySelector('#pixiv-previewer-cover');
const { time, id, size } = Cover.dataset;
let { page = 0 } = Cover.dataset;
let next = Number.parseInt(page, 10) + 1;
if (next + 1 > size) {
next = 0;
}
if (next != page) {
document.querySelector(
'#pixiv-previewer-image'
).src = `https://i.pximg.net/img-master/img/${time}/${id}_p${next}_master1200.jpg`;
Cover.dataset.page = next;
}
Logger.debug('click image');
});
previewCover.appendChild(previewImage);
rootElm.appendChild(previewCover);
setInterval(Previewer.addButton, 1000);
Logger.debug('Script End.');
})();