// ==UserScript== // @name pixiv タグクラウドからピックアップ // @name:ja pixiv タグクラウドからピックアップ // @name:en pixiv Tag Cloud Prioritizer // @description If there are tags attached to a work, this script brings those tags to the top of the tag cloud (illustration or novel tags column) on the left. // @description:ja 作品左側のタグクラウド (作品タグ) から、閲覧中の作品についているタグと同じものをピックアップします。 // @namespace https://userscripts.org/users/347021 // @version 2.0.0 // @match http://www.pixiv.net/member_illust.php?*mode=medium* // @match http://www.pixiv.net/novel/show.php?*id=* // @require https://greasyfork.org/scripts/17895/code/polyfill.js?version=113962 // @require https://greasyfork.org/scripts/17896/code/start-script.js?version=112958 // @license Mozilla Public License Version 2.0 (MPL 2.0); https://www.mozilla.org/MPL/2.0/ // @compatible Firefox // @compatible Opera // @compatible Chrome // @grant GM_setValue // @grant GM_getValue // @grant GM_deleteValue // @grant GM_listValues // @run-at document-start // @icon  // @author 100の人 // @homepage https://greasyfork.org/scripts/262 // @downloadURL none // ==/UserScript== (function () { 'use strict'; let frameElement = window.frameElement; if (frameElement && frameElement.src === 'about:blank') { return; } /** * messageイベントの選別等に使用するID。 * @type {string} */ const ID = 'pixiv-tag-cloud-pickup-347021'; /** * タグ一覧ページをキャッシュしておく期間(秒数)。 * @type {number} */ const CACHE_LIFETIME = 24 * 60 * 60; /** * 秒をミリ秒に変換するときの乗数 * @type {number} */ const MINUTES_TO_MILISECONDS = 1000; /** * 小説ページなら真。 * @type {boolean} */ const NOVEL = window.location.pathname === '/novel/show.php'; /** * タグ一覧ページをキャッシュする名前の接尾辞。 * @type {string} */ const CACHE_NAME_SUFFIX = (NOVEL ? '-novel' : '') + '-tags'; /** * タグ一覧ページのキャッシュ期限を記録する名前の接尾辞。 * @type {string} */ const CACHE_EXPIRE_NAME_SUFFIX = (NOVEL ? '-novel' : '') + '-expire'; let viewMypixivs = document.getElementsByClassName('view_mypixiv'); startScript( main, parent => parent.classList.contains('area_inside'), target => target.classList.contains('view_mypixiv'), () => viewMypixivs[0], { isTargetParent: parent => parent.classList.contains('ui-layout-west'), isTarget: target => target.classList.contains('user-tags'), } ); let nextCleaningDate = GM_getValue('next-cleaning-date'); if (nextCleaningDate) { if (new Date(nextCleaningDate).getTime() < Date.now()) { // 予定時刻を過ぎていれば、古いキャッシュを削除 let names = GM_listValues(); for (let name of names) { if (name && name.endsWith('-expire')) { if (new Date(GM_getValue(name)).getTime() < Date.now()) { // キャッシュの有効期限が切れていれば GM_deleteValue(name); let tagsName = name.replace('-expire', '-tags'); GM_deleteValue(tagsName); names[names.indexOf(tagsName)] = null; } } } nextCleaningDate = null; } } else { // バージョン1.0.0で生成されたデータの削除 GM_listValues().forEach(GM_deleteValue); } if (!nextCleaningDate) { GM_setValue('next-cleaning-date', new Date(Date.now() + CACHE_LIFETIME * MINUTES_TO_MILISECONDS).toISOString()); } function main() { // スタイルシートの設定 document.head.insertAdjacentHTML('beforeend', ``); /** * タグクラウド。 * @type {HTMLUListElement} */ let tagCloud = document.getElementsByClassName('tagCloud')[0]; /** * タグクラウドに無いタグ一覧。 * @type {HTMLLIElement[]} */ let minorityTags = []; let tagCloudItemTemplate; let tagCloudItemTemplateAnchor; let currentTags = new DocumentFragment(); // 表示している作品のタグを取得する for (let tagItem of document.querySelectorAll('.tag .text')) { currentTags.append(' '); /** * RFC 3986にもとづいてパーセント符号化されたタグ。 * @type {string} */ let urlencodedTag = /[^=]+$/.exec(tagItem.search)[0]; let anchor = tagCloud.querySelector('[href$="tag=' + urlencodedTag + '"]'); if (anchor) { // タグクラウドに同じタグが存在すれば、抜き出す currentTags.append(anchor.parentElement); } else { // 存在しなければ、もっとも出現度の低いタグとして追加しておく if (!tagCloudItemTemplate) { tagCloudItemTemplate = tagCloud.firstElementChild.cloneNode(true); tagCloudItemTemplate.className = 'level6'; tagCloudItemTemplateAnchor = tagCloudItemTemplate.firstElementChild; } tagCloudItemTemplateAnchor.search = tagCloudItemTemplateAnchor.search.replace(/[^=]+$/, urlencodedTag); tagCloudItemTemplateAnchor.textContent = tagItem.textContent; minorityTags.push(currentTags.append(tagCloudItemTemplate.cloneNode(true))); } } // 表示している作品のタグとそれ以外のタグとの区切りを示すクラスを設定 currentTags.lastElementChild.classList.add('last-current-tag'); /** * タグクラウドに出現数が2つ回以上のタグしか無ければ真。 * @type {boolean} */ let tagCloudHavingOnlymajorityTags = tagCloud.children.length === tagCloud.getElementsByClassName('cnt').length; // タグクラウドの先頭に挿入 tagCloud.prepend(currentTags); if (minorityTags.length > 0 && tagCloudHavingOnlymajorityTags) { // 表示している作品のタグのうち、タグクラウドに存在しないタグがあり、 // かつタグクラウドに出現数が2回以上のタグしか無ければ // タグ一覧を取得 getAllTags(new URLSearchParams(document.querySelector('.user-tags a').search).get('id'), function (tags) { for (let li of minorityTags) { let anchor = li.firstElementChild; // タグ一覧ページから出現数が2回以上のタグ数を取得 let tag = anchor.text; for (let count in tags) { if (tags[count].indexOf(tag) !== -1) { // タグの数を表示 anchor.insertAdjacentHTML('beforeend', '(' + count + ')'); break; } } } }); } } /** * 指定したユーザーの、出現数が2回以上のタグ一覧を取得する。 * @param {string} userId * @param {Function} callback - 第1引数に、イラスト数をキー、タグの配列を値としたオブジェクト。 */ function getAllTags(userId, callback) { let expire = GM_getValue(userId + CACHE_EXPIRE_NAME_SUFFIX); if (expire && new Date(expire).getTime() > Date.now()) { // キャッシュが存在し、有効期限が切れていなければ callback(JSON.parse(GM_getValue(userId + CACHE_NAME_SUFFIX))); } else { getAllTagsFromPage(userId, callback); } } /** * 指定したユーザーの、出現数が2回以上のタグ一覧をページから取得し、キャッシュとして保存する。 * @param {string} userId * @param {Function} callback - 第1引数に、イラスト数をキー、タグの配列を値としたオブジェクト。 */ function getAllTagsFromPage(userId, callback) { let client = new XMLHttpRequest(); client.open('GET', './member_tag_all.php?id=' + userId); client.responseType = 'document'; client.addEventListener('load', function (event) { let counts = event.target.response.querySelectorAll('.tag-list > dt'); if (counts.length > 0) { let tags = {}; for (let dt of counts) { let count = dt.textContent; if (count > 1) { tags[count] = Array.from(dt.nextElementSibling.getElementsByTagName('a')).map(function (anchor) { return anchor.text; }); } } GM_setValue(userId + CACHE_NAME_SUFFIX, JSON.stringify(tags)); // 有効期限(日時)の設定 GM_setValue( userId + CACHE_EXPIRE_NAME_SUFFIX, new Date(Date.now() + CACHE_LIFETIME * MINUTES_TO_MILISECONDS).toISOString() ); callback(tags); } }); client.send(); } })();