// ==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://greasyfork.org/users/137 // @version 1.0.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://greasyfork.org/scripts/19616/code/utilities.js?version=230651 // @require https://greasyfork.org/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://greasyfork.org/users/137 // @downloadURL none // ==/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', 'page'); tab.classList.remove(this.currentTabClass); } } if (!event.target.hasAttribute('aria-current')) { event.target.setAttribute('aria-current', 'page'); event.target.classList.add(this.currentTabClass); } }); 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.currentTabClass) { const currentTab = list.querySelector('[aria-current="page"]'); this.currentTabClass = currentTab.classList[1]; } const userPageTab = list.firstElementChild; const templateTab = userPageTab.cloneNode(true); templateTab.removeAttribute('aria-current'); templateTab.classList.remove(this.currentTabClass); 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 = ''; tab.classList.add(userPageTab.classList[0]); } } 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.currentTabClass); } } } 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 }); } })();