// ==UserScript==
// @name pixiv Tabs Restorer
// @name:ja pixiv タブを復活
// @description Adds “All”, “Follow”, “My pixiv”, and “Tag Index” tabs to user pages etc. and “All” and “Ugoira” tabs to search result pages.
// @description:ja ユーザーページなどに「すべて」「フォロー」「マイピク」「タグ一覧」タブを、検索結果に「すべて」「うごイラ」タブを補完します。
// @namespace https://gf.qytechs.cn/users/137
// @version 2.0.0
// @match https://www.pixiv.net/*
// @require https://gf.qytechs.cn/scripts/19616/code/utilities.js?version=752462
// @require https://gf.qytechs.cn/scripts/17896/code/start-script.js?version=112958
// @license MPL-2.0
// @contributionURL https://www.amazon.co.jp/registry/wishlist/E7PJ5C3K7AM2
// @compatible Edge 最新安定版 / Latest stable (非推奨 / Deprecated)
// @compatible Firefox
// @compatible Opera
// @compatible Chrome
// @grant dummy
// @noframes
// @run-at document-start
// @icon 
// @author 100の人
// @homepageURL https://gf.qytechs.cn/scripts/373026
// ==/UserScript==
// 当スクリプトはpixivが作成、配布しているアプリケーションではありません。
// <https://www.pixiv.net/terms/?page=brand>
'use strict';
// L10N
Gettext.setLocalizedTexts({
/*eslint-disable quote-props, max-len */
'en': {
'すべて': 'All',
'フォロー': 'Follow',
'マイピク': 'My pixiv',
'タグ一覧': 'Tag Index',
'うごイラ': 'Ugoira',
},
'ko': {
'すべて': '전체',
'フォロー': '팔로우',
'マイピク': '마이픽',
'タグ一覧': '태그 목록',
'うごイラ': '움직이는 일러스트',
},
'zh': {
'すべて': '全部',
'フォロー': '关注',
'マイピク': '好P友',
'タグ一覧': '标签一览',
'うごイラ': '动图',
},
'zh-tw': {
'すべて': '全部',
'フォロー': '關注',
'マイピク': '好P友',
'タグ一覧': '標籤一覽',
'うごイラ': '動圖',
},
/*eslint-enable quote-props, max-len */
});
class TabCompleter
{
/**
* @access private
* @constant {number}
*/
static get URLS_AND_LABLES() {return {
search: [
{ path: 'artworks', label: _('すべて'), position: 1 },
{ path: 'illustrations', type: 'ugoira', label: _('うごイラ'), position: 4 },
],
user: [
{ path: '/member_illust.php', label: _('すべて'), position: 1 },
{ path: '/bookmark.php', type: 'user', label: _('フォロー') },
{ path: '/mypixiv_all.php', label: _('マイピク') },
{ path: '/member_tag_all.php', label: _('タグ一覧') },
],
};}
constructor()
{
const root = document.getElementById('root');
if (!root) {
return;
}
Gettext.setLocale(document.documentElement.lang);
addEventListener('click', event => {
const tab = event.target.closest('a');
if (!tab || !tab.matches('#root > :not(header) nav > a')) {
return;
}
if (event.defaultPrevented) {
// 既存のタブ
this.markCurrentTab();
return;
}
if (tab.getAttribute('href').startsWith('/') || !location.pathname.startsWith('/tags/')
|| location.pathname.endsWith('/novels')) {
// 別サービスへのタブ
return;
}
// イラスト検索結果ページで「すべて」「うごイラ」タブをクリックした場合
event.preventDefault();
// 検索オプションボタンをクリック
document.querySelector('[d^="M0 1C0 0.447754"]').closest('button').click();
new MutationObserver(function (mutations, observer) {
let dialog;
for (const mutation of mutations) {
dialog = Array.from(mutation.addedNodes).find(
node => node.nodeType === Node.ELEMENT_NODE && node.getAttribute('role') === 'presentation'
);
if (dialog) {
break;
}
}
if (!dialog) {
return;
}
observer.disconnect();
dialog.hidden = true;
// 対象
const input = dialog.querySelector('[class$="-dummyInput"]');
// プルダウンメニューを開く
input.dispatchEvent(new KeyboardEvent('keydown', { bubbles: true, key: ' ' }));
new MutationObserver(function (mutations, observer) {
let select, options;
mutations: for (const mutation of mutations) {
for (const node of mutation.addedNodes) {
options = node.querySelectorAll('[role="option"]');
if (options.length === 0) {
continue;
}
select = node;
break mutations;
}
}
if (!select) {
return;
}
observer.disconnect();
// 項目を選択する
options[tab.pathname.endsWith('/artworks') ? /* すべて */0 : /* うごイラ */4].click();
new MutationObserver(function (mutations, observer) {
if (!mutations.some(mutation => Array.from(mutation.removedNodes).includes(select))) {
return;
}
observer.disconnect();
setTimeout(function () {
// 「適用する」ボタンを押す
dialog.querySelector('[type="submit"]').click();
this.markCurrentTab();
}, 100);
}).observe(select.parentElement, { childList: true });
}).observe(dialog, { subtree: true, childList: true });
}).observe(document.body, { childList: true });
});
new MutationObserver(mutations => {
for (const mutation of mutations) {
if (mutation.target.matches('#root > div[class]')) {
// 検索結果ページ
if (!Array.from(mutation.addedNodes).some(
node => node.nodeType === Node.ELEMENT_NODE && node.querySelector('nav > a[href^="/tags/"]')
)) {
continue;
}
this.complete();
return;
}
let findChild;
if (mutation.target.matches('#root > div[class] > div[class]')) {
findChild = node => node.localName === 'div' && node.hasAttribute('class');
} else if (mutation.target.matches('#root > div[class] > div[class] > div[class]')) {
findChild = node => node.localName === 'nav';
}
if (!findChild) {
continue;
}
const parent = Array.from(mutation.addedNodes).find(findChild);
if (parent) {
if (parent.querySelector('[href*="/mypixiv_all.php?"]')) {
return;
}
this.complete();
return;
}
}
}).observe(root, {childList: true, subtree: true});
}
markCurrentTab()
{
const currentTab = this.list.querySelector('[aria-current][href^="/"]');
const tabs = this.list.children;
for (const tab of tabs) {
if (tab.href === location.href) {
tab.setAttribute('aria-current', 'page');
tab.classList.add(...this.currentTabClasses);
} else {
tab.removeAttribute('aria-current');
tab.classList.remove(...this.currentTabClasses);
}
}
if (!this.list.querySelector('[aria-current]')) {
currentTab.setAttribute('aria-current', 'page');
currentTab.classList.add(...this.currentTabClasses);
}
}
/**
* タブを補完します。
* @returns {Promise.<void>}
*/
async complete()
{
/**
* 各タブの共通の親要素。
* @member {HTMLElement}
*/
this.list = document.querySelector('#root > :not(header) nav');
if (this.list.querySelector('[href*="type=ugoira"], [href*="/member_tag_all.php"]')) {
// すでに補完済みなら
this.markCurrentTab();
return;
}
const tabs = this.list.children;
if (!this.currentTabClasses) {
const currentTab = this.list.querySelector('[aria-current="page"]');
const noCurrentTabClassList = Array.from(Array.from(tabs).find(tab => tab !== currentTab).classList);
/**
* カレントタブに設定されるクラス。
* @member {string[]}
*/
this.currentTabClasses
= Array.from(currentTab.classList).filter(token => !noCurrentTabClassList.includes(token));
}
const pageType = location.pathname.startsWith('/tags/') ? 'search' : 'user';
// 挿入するタブのテンプレートを作成
const templateTab = tabs[pageType === 'search' ? 1 : 0].cloneNode(true);
templateTab.removeAttribute('aria-current');
templateTab.classList.remove(...this.currentTabClasses);
const param = new URLSearchParams(templateTab.search);
for (const { path, type, label, position } of TabCompleter.URLS_AND_LABLES[pageType]) {
let tab;
if (path === '/member_illust.php') {
tab = document.querySelector('[href^="/member_illust.php?id="]:not([href*="type="])');
if (tab) {
tab.classList = templateTab.classList;
}
}
if (!tab) {
tab = templateTab.cloneNode(true);
}
switch (pageType) {
case 'search':
tab.pathname = tab.pathname.replace(/[^/]+$/, path);
tab.firstElementChild.textContent = label;
break;
case 'user':
tab.pathname = path;
tab.text = label;
break;
}
if (type) {
param.set('type', type);
tab.search = param;
}
if (position) {
tabs[position].before(tab);
} else {
this.list.append(tab);
}
}
this.markCurrentTab();
}
}
document.addEventListener('DOMContentLoaded', function () {
new TabCompleter();
}, { passive: true, once: true });