pixiv タグクラウドからピックアップ

Restores the tag cloud (illustration or novel tags column), and if there are tags attached to a work, this script brings those tags to the top of the tag cloud (illustration tags column).

目前為 2020-03-11 提交的版本,檢視 最新版本

// ==UserScript==
// @name        pixiv タグクラウドからピックアップ
// @name:ja     pixiv タグクラウドからピックアップ
// @name:en     pixiv Tag Cloud Prioritizer
// @description Restores the tag cloud (illustration or novel tags column), and if there are tags attached to a work, this script brings those tags to the top of the tag cloud (illustration tags column).
// @description:ja 作品ページへタグクラウド (作品タグ・小説タグ) を復活させ、閲覧中の作品についているタグと同じものをピックアップします。
// @namespace   https://userscripts.org/users/347021
// @version     2.14.0
// @match       https://www.pixiv.net/*
// @require     https://gitcdn.xyz/cdn/greasemonkey/gm4-polyfill/a834d46afcc7d6f6297829876423f58bb14a0d97/gm4-polyfill.js
// @require     https://gf.qytechs.cn/scripts/17895/code/polyfill.js?version=625392
// @require     https://gf.qytechs.cn/scripts/19616/code/utilities.js?version=752462
// @require     https://gf.qytechs.cn/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       GM.setValue
// @grant       GM_setValue
// @grant       GM.getValue
// @grant       GM_getValue
// @grant       GM.deleteValue
// @grant       GM_deleteValue
// @grant       GM.listValues
// @grant       GM_listValues
// @noframes
// @run-at      document-start
// @icon        
// @author      100の人
// @homepageURL https://gf.qytechs.cn/scripts/262
// ==/UserScript==

// 当スクリプトはpixivが作成、配布しているアプリケーションではありません。
// <https://www.pixiv.net/terms/?page=brand>

'use strict';

/**
 * タグ一覧ページをキャッシュしておく期間 (秒数)。
 * @constant {number}
 */
const CACHE_LIFETIME = 24 * 60 * 60;

/**
 * @typedef {Object} TagsData
 * @property {HTMLDivElement} tagCloudSection - タグクラウド。
 * @property {Object.<number>} tagsAndCounts - タグをキー、タグの出現数を値に持つ連想配列。
 */

if (typeof content !== 'undefined') {
	// For Greasemonkey 4
	XMLHttpRequest = content.XMLHttpRequest.bind(content); //eslint-disable-line no-global-assign, no-native-reassign, no-undef, max-len
}

/**
 * 小説ページなら真。
 * @type {boolean}
 */
const novel = location.pathname.startsWith('/novel/');

/** @type {Promise.<TagsData>} */
let tagsDataPromise;

getUserId().then(async function (userId) {
	new MutationObserver(function (mutations) {
		for (const mutation of mutations) {
			const addedNode = mutation.addedNodes[0];
			if (!addedNode) {
				continue;
			}

			const target = mutation.target;
			switch (target.localName) {
				case 'div':
					if (novel && addedNode.id === 'chapter_0_0') {
						// 作品ページから作品ページへの移動 (小説)
						pickup();
						return;
					}
					if (addedNode.localName !== 'div') {
						continue;
					}
					// 作品ページ外から作品ページへの移動
					insertTagCloud();
					return;
				case 'section':
					if (addedNode.localName !== 'div' || addedNode !== target.firstElementChild) {
						continue;
					}
					// 作品ページ外から作品ページへの移動
					pickup();
					return;
				case 'a': {
					if (novel || !target.closest('figure') || addedNode.localName !== 'img') {
						continue;
					}
					// 作品ページから作品ページへの移動 (イラスト)
					const newUserId = document.querySelector('[href*="/users/"]').pathname.replace('/users/', '');
					if (newUserId !== userId) {
						// 別ユーザーの関連作品への移動
						userId = newUserId;
						tagsDataPromise = getTagsData(userId);
						insertTagCloud();
					}
					pickup();
					return;
				}
			}
		}
	}).observe(document, { childList: true, subtree: true });

	tagsDataPromise = async function () {
		const version = await GM.getValue('version'); // v2.13.0以降
		let nextCleaningDate = await GM.getValue('next-cleaning-date'); // v1.1.0以降
		if (version && nextCleaningDate) {
			if (new Date(nextCleaningDate).getTime() < Date.now()) {
				// 予定時刻を過ぎていれば、古いキャッシュを削除
				for (const name of await GM.listValues()) {
					if (/-(?:tags|expire)$/.test(name)) {
						// バージョン2.2.0以前で生成されたデータの削除
						await GM.deleteValue(name);
						continue;
					}
					if (!/^[0-9]+(?:-novel)?$/.test(name)) {
						continue;
					}
					const data = await GM.getValue(name);
					if (new Date(data.expire).getTime() < Date.now()) {
						// キャッシュの有効期限が切れていれば
						await GM.deleteValue(name);
					}
				}
				nextCleaningDate = null;
			}
		} else {
			// v2.12.0以前に生成されたデータの削除
			await Promise.all((await GM.listValues()).map(GM.deleteValue));
		}
		if (!version || version !== GM.info.script.version) {
			await GM.setValue('version', GM.info.script.version);
		}
		if (!nextCleaningDate) {
			await GM.setValue(
				'next-cleaning-date',
				new Date(Date.now() + CACHE_LIFETIME * DateUtils.MINUTES_TO_MILISECONDS).toISOString()
			);
		}

		return getTagsData(userId);
	}();

	addStyleSheet();
});

async function addStyleSheet()
{
	let tagCloudStyles = await GM.getValue('tag-cloud-styles');
	if (!tagCloudStyles) {
		tagCloudStyles = (await (await fetch('https://s.pximg.net/www/css/global.css')).text())
			.match(/^(?:\.area_(?:new|title|inside)|\.view_mypixiv|ul\.tagCloud) .+?$/umg).join('\n');
		GM.setValue('tag-cloud-styles', tagCloudStyles);
	}
	document.head.insertAdjacentHTML('beforeend', h`<style>
		${tagCloudStyles}
		.area_new {
			width: unset;
			margin: 16px;
		}
		.area_new + section {
			/* 小説ページにもともと含まれる「作者の作品タグ」 */
			display: none;
		}
		.tagCloud {
			padding: 0;
		}
		.tagCloud .last-current-tag::after {
			content: "";
			display: inline-block;
			height: 18px;
			border-right: solid 1px #999;
			width: 10px;
			margin-bottom: -3px;
			-webkit-transform: rotate(0.3rad);
			transform: rotate(0.3rad);
		}
	`);
}

async function insertTagCloud()
{
	const tagCloudSection = (await tagsDataPromise).tagCloudSection.cloneNode(true);
	const currentTagCloudSection = document.getElementsByClassName('area_new')[0];
	if (currentTagCloudSection) {
		if (currentTagCloudSection.getElementsByTagName('a')[0].getAttribute('href')
			!== tagCloudSection.getElementsByTagName('a')[0].getAttribute('href')) {
			currentTagCloudSection.replaceWith(tagCloudSection);
		}
		return;
	}
	if (novel) {
		const h2 = document.querySelector('main + aside > div ~ section header > h2');
		if (!h2) {
			return;
		}
		const sibling = h2.closest('section');
		if (!sibling) {
			return;
		}
		sibling.before(tagCloudSection);
	} else {
		const sibling = document.querySelector('main + aside > div');
		if (!sibling) {
			return;
		}
		sibling.after(tagCloudSection);
	}
}

async function pickup()
{
	/** @type {TagsData} */
	const tagsData = await tagsDataPromise;

	if (!document.getElementsByClassName('area_new')[0]) {
		await insertTagCloud();
	}

	/** @type {HTMLUListElement} */
	const tagCloud = tagsData.tagCloudSection.getElementsByClassName('tagCloud')[0].cloneNode(true);

	let tagCloudItemTemplate;
	let tagCloudItemTemplateAnchor;

	const currentTags = [];

	// 表示している作品のタグを取得する
	for (const tagItem
		of document.getElementsByClassName('gtm-new-work-tag-event-click')[0].closest('ul').getElementsByTagName('a')) {
		/**
		 * RFC 3986にもとづいてパーセント符号化されたタグ。
		 * @type {string}
		 */
		const urlencodedTag = tagItem.pathname.split('/')[2];

		let tagCloudItem;

		const anchor = tagCloud.querySelector('[href$="/' + urlencodedTag + '"]');
		if (anchor) {
			// タグクラウドに同じタグが存在すれば、抜き出す
			tagCloudItem = anchor.parentElement;
		} else {
			// 存在しなければ、もっとも出現度の低いタグとして追加
			if (!tagCloudItemTemplate) {
				tagCloudItemTemplate = tagCloud.firstElementChild.cloneNode(true);
				tagCloudItemTemplate.className = 'level6';
				tagCloudItemTemplateAnchor = tagCloudItemTemplate.firstElementChild;
			}
			
			tagCloudItemTemplateAnchor.pathname = tagCloudItemTemplateAnchor.pathname.replace(/[^/]+$/, urlencodedTag);
			const tag = tagItem.textContent;
			tagCloudItemTemplateAnchor.text = tag;
			if (tag in tagsData.tagsAndCounts) {
				// タグの数を表示
				tagCloudItemTemplateAnchor
					.insertAdjacentHTML('beforeend', `<span class="cnt">(${tagsData.tagsAndCounts[tag]})</span>`);
			}
			tagCloudItem = tagCloudItemTemplate.cloneNode(true);
		}

		currentTags.push(' ', tagCloudItem);
	}

	// 表示している作品のタグとそれ以外のタグとの区切りを示すクラスを設定
	currentTags[currentTags.length - 1].classList.add('last-current-tag');

	// タグクラウドの先頭に挿入
	tagCloud.prepend(...currentTags);
	
	// 更新
	document.getElementsByClassName('tagCloud')[0].replaceWith(tagCloud);
}

/**
 * 表示している作品の作者のユーザーIDを取得します。
 * @returns {Promise.<string>}
 */
function getUserId()
{
	return new Promise(function (resolve) {
		new MutationObserver(function (mutations, observer) {
			for (const mutation of mutations) {
				if (mutation.target.localName !== 'head') {
					continue;
				}
				for (const node of mutation.addedNodes) {
					if (node.id !== 'meta-preload-data') {
						continue;
					}
					observer.disconnect();
					resolve(Object.keys(JSON.parse(node.content).user)[0]);
					return;
				}
			}
		}).observe(document, {childList: true, subtree: true});
	});
}

/**
 * 指定したユーザーのタグクラウド、および出現数が2回以上のタグ一覧を取得します。
 * @param {string} userId
 * @returns {Promise.<TagsData>}
 */
async function getTagsData(userId)
{
	const serializedTagsData = await GM.getValue(userId);
	if (serializedTagsData && new Date(serializedTagsData.expire).getTime() > Date.now()) {
		const body = document.implementation.createHTMLDocument().body;
		body.innerHTML = serializedTagsData.tagCloudSection;
		return { tagCloudSection: body.firstElementChild, tagsAndCounts: serializedTagsData.tagsAndCounts };
	}
	return getTagsDataFromPage(userId);
}

/**
 * 指定したユーザーのタグクラウド、および出現数が2回以上のタグ一覧をページから取得し、キャッシュとして保存します。
 * @param {string} userId
 * @returns {Promise.<TagsData>}
 */
function getTagsDataFromPage(userId)
{
	return new Promise(function (resolve) {
		const client = new XMLHttpRequest();
		client.open('GET', new URL((novel ? '/novel' : '') + '/member_tag_all.php?id=' + userId, location));
		client.responseType = 'document';
		client.addEventListener('load', function (event) {
			const doc = event.target.response;
			
			const tagCloudSection = doc.getElementsByClassName(novel
				? 'area_new'
				: /* class="area_new promotion-comic" の回避 */'user-tags')[0];
			const tagCouldAnchors = {};
			for (const anchor of tagCloudSection.querySelectorAll('li a')) {
				tagCouldAnchors[anchor.firstChild.data] = anchor;
			}
			
			const counts = doc.querySelectorAll('.tag-list > dt');
			const tagsAndCounts = {};
			for (const dt of counts) {
				const count = Number.parseInt(dt.textContent);
				for (const anchor of dt.nextElementSibling.getElementsByTagName('a')) {
					const tag = anchor.text;
					if (count > 1) {
						tagsAndCounts[tag] = count;
					}

					// タグクラウドのリンクが旧URLになっている不具合を修正
					if (tag in tagCouldAnchors) {
						tagCouldAnchors[tag].href = anchor;
					}
				}
			}

			GM.setValue(userId + (novel ? '-novel' : ''), {
				expire: new Date(Date.now() + CACHE_LIFETIME * DateUtils.MINUTES_TO_MILISECONDS).toISOString(),
				tagCloudSection: tagCloudSection.outerHTML,
				tagsAndCounts,
			});

			resolve({ tagCloudSection, tagsAndCounts });
		});
		client.send();
	});
}

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址