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.1
// @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或关注我们的公众号极客氢云获取最新地址