// ==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://greasyfork.org/users/137 // @version 2.0.0 // @match https://www.pixiv.net/* // @require https://greasyfork.org/scripts/19616/code/utilities.js?version=752462 // @require https://greasyfork.org/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://greasyfork.org/scripts/373026 // @downloadURL none // ==/UserScript== // 当スクリプトはpixivが作成、配布しているアプリケーションではありません。 // '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.} */ 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 });