pixiv Tabs Restorer

Adds “All”, “Follow”, “My pixiv”, and “Tag Index” tabs to user pages etc. and “All” and “Ugoira” tabs to search result pages.

目前为 2019-11-25 提交的版本。查看 最新版本

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

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址