Greasy Fork

Greasy Fork is available in English.

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   http://greasyfork.icu/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     http://greasyfork.icu/scripts/19616/code/utilities.js?version=230651
// @require     http://greasyfork.icu/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 http://greasyfork.icu/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 });
}

})();