pixiv Tabs Restorer

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

目前为 2019-02-19 提交的版本。查看 最新版本

// ==UserScript==
// @name        pixiv Tabs Restorer
// @name:ja     pixiv タブを復活
// @description Adds “All”, “Follow”, “My pixiv”, and “Tag Index” tabs to user pages etc. and “Illustrations”, “Manga”, and “Ugoira” tabs to search result pages.
// @description:ja ユーザーページなどに「すべて」「フォロー」「マイピク」「タグ一覧」タブを、検索結果に「イラスト」「マンガ」「うごイラ」タブを補完します。
// @namespace   https://gf.qytechs.cn/users/137
// @version     1.1.0
// @match       https://www.pixiv.net/*
// @exclude     https://www.pixiv.net/member_illust.php?*mode=manga*
// @exclude     https://www.pixiv.net/apps.php*
// @require     https://gf.qytechs.cn/scripts/19616/code/utilities.js?version=230651
// @require     https://gf.qytechs.cn/scripts/17896/code/start-script.js?version=112958
// @license     MPL-2.0
// @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/users/137
// ==/UserScript==

(function () {
	'use strict';
	
	
	// L10N
	Gettext.setLocalizedTexts({
		/*eslint-disable quote-props, max-len */
		'en': {
			'すべて': 'All',
			'フォロー': 'Follow',
			'マイピク': 'My pixiv',
			'タグ一覧': 'Tag Index',
			'イラスト': 'Illustrations',
			'マンガ': 'Manga',
			'うごイラ': 'Ugoira',
		},
		'ko': {
			'すべて': '전체',
			'フォロー': '팔로우',
			'マイピク': '마이픽',
			'タグ一覧': '태그 목록',
			'イラスト': '일러스트',
			'マンガ': '만화',
			'うごイラ': '움직이는 일러스트',
		},
		'zh': {
			'すべて': '全部',
			'フォロー': '关注',
			'マイピク': '好P友',
			'タグ一覧': '标签一览',
			'イラスト': '插画',
			'マンガ': '漫画',
			'うごイラ': '动图',
		},
		'zh-tw': {
			'すべて': '全部',
			'フォロー': '關注',
			'マイピク': '好P友',
			'タグ一覧': '標籤一覽',
			'イラスト': '插畫',
			'マンガ': '漫畫',
			'うごイラ': '動圖',
		},
		/*eslint-enable quote-props, max-len */
	});
	
	class UserPageTabCompleter
	{
		/**
		 * @access private
		 * @constant {number}
		 */
		static get URLS_AND_LABLES() {return [
			{ path: '/member_illust.php', label: _('すべて'), afterUserPageTab: true },
			{ 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 => {
				if (!event.defaultPrevented || !event.target.matches('#root > :not(header) nav > a')) {
					return;
				}
	
				for (const tab of event.target.parentElement.querySelectorAll('[aria-current]')) {
					if (tab.href !== location.href) {
						tab.removeAttribute('aria-current');
						tab.classList.remove(...this.currentTabClasses);
					}
				}
	
				if (!event.target.hasAttribute('aria-current')) {
					event.target.setAttribute('aria-current', 'page');
					event.target.classList.add(...this.currentTabClasses);
				}
			});
	
			new MutationObserver(mutations => {
				for (const mutation of mutations) {
					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});
		}
	
		/**
		 *
		 * @access private
		 */
		async complete()
		{
			const list = document.querySelector('#root > :not(header) nav');
			if (!this.currentTabClasses) {
				const currentTab = list.querySelector('[aria-current="page"]');
				const noCurrentTabClassList = Array.from(Array.from(list.children).find(tab => tab !== currentTab).classList);
				this.currentTabClasses = Array.from(currentTab.classList).filter(token => !noCurrentTabClassList.includes(token));
			}
	
			const userPageTab = list.firstElementChild;
	
			const templateTab = userPageTab.cloneNode(true);
			templateTab.removeAttribute('aria-current');
			templateTab.classList.remove(...this.currentTabClasses);
			const param = new URLSearchParams(userPageTab.search);
	
			for (const {path, type, label, afterUserPageTab} of UserPageTabCompleter.URLS_AND_LABLES) {
				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);
				}
	
				tab.pathname = path;
				if (type) {
					param.set('type', type);
					tab.search = param;
				}
				tab.text = label;
				if (afterUserPageTab) {
					userPageTab.after(tab);
				} else {
					list.append(tab);
				}
			}
	
			if (userPageTab.hasAttribute('aria-current') && location.pathname === '/member_illust.php') {
				userPageTab.replaceWith(templateTab);
				const illustAndMangaTab = list.querySelector('[href*="/member_illust.php"]:not([href*="type="])');
				illustAndMangaTab.setAttribute('aria-current', 'page');
				illustAndMangaTab.classList.add(...this.currentTabClasses);
			}
		}
	}
	
	if (['/search.php', '/novel/search.php', '/search_user.php'].includes(location.pathname)) {
		startScript(
			function () {
				Gettext.setLocale(document.documentElement.lang);
	
				const typesAndLabels = [
					{ type: 'illust', label: _('イラスト') },
					{ type: 'manga' , label: _('マンガ') },
					{ type: 'ugoira', label: _('うごイラ') },
				];
	
				const list = document.getElementsByClassName('tabs')[0];
				const allTab = list.firstElementChild;
				const afterTab = allTab.nextElementSibling;
	
				const allTabAnchor = allTab.getElementsByTagName('a')[0];
				const query = new URLSearchParams(allTabAnchor.search);
				query.delete('p');
	
				const currentType = new URLSearchParams(location.search).get('type');
	
				for (const {type, label} of typesAndLabels) {
					const tab = allTab.cloneNode(true);
					const anchor = tab.getElementsByTagName('a')[0];
					query.set('type', type);
					anchor.search = query;
					anchor.text = label;
					anchor.classList[currentType === type ? 'add' : 'remove']('current');
					afterTab.before(tab);
				}
	
				allTabAnchor.text = _('すべて');
				if (list.getElementsByClassName('current').length > 1) {
					allTabAnchor.classList.remove('current');
				}
			},
			parent => parent.classList.contains('tabs'),
			target => target.previousElementChild,
			() => document.querySelector('.tabs > li:nth-of-type(2)')
		);
	} else {
		document.addEventListener('DOMContentLoaded', function () {
			new UserPageTabCompleter();
		}, { passive: true, once: true });
	}
	
	})();
	

QingJ © 2025

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