// ==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 });
}
})();