pixiv コメントを展開

コメントを常に全件表示する / Expands pixiv comments. Always shows all comments.

当前为 2014-06-28 提交的版本,查看 最新版本

// ==UserScript==
// @name           pixiv コメントを展開
// @namespace      http://userscripts.org/users/347021
// @id             pixiv-expands-comments-347021
// @version        1.3.2
// @description    コメントを常に全件表示する / Expands pixiv comments. Always shows all comments.
// @match          http://www.pixiv.net/member_illust.php?*mode=medium*
// @match          http://www.pixiv.net/novel/show.php?*id=*
// @domain         www.pixiv.net
// @run-at         document-start
// @grant          dummy
// @icon           
// @author         100の人 https://gf.qytechs.cn/ja/users/137-100%E3%81%AE%E4%BA%BA
// @license        Creative Commons Attribution 4.0 International Public License; http://creativecommons.org/licenses/by/4.0/
// ==/UserScript==

/* @iconに、TMZ様のイラストを使わせていただきました。
 「iPhoneふうpixivアイコン」/「TMZ」のイラスト [pixiv]
 <http://www.pixiv.net/member_illust.php?mode=medium&illust_id=8572587> */

(function () {
'use strict';

/**
 * コメント取得回数(20コメント / 1回)制限
 * @constant {number}
 */
var LIMIT_LOADING_COMMENT = 10;

/**
 * 何ページ目のコメントから取得を開始するか
 * @constant {number}
 */
var DEFAULT_PAGE = 2;

var worksOptionRights = document.getElementsByClassName('worksOptionRight');
startScript(main,
	function (parent) { return parent.id === 'one_comment'; },
	function (target) { return target.classList.contains('worksOptionRight'); },
	function () { return worksOptionRights[0]; },
	{
		isTargetParent: function (parent) { return parent.nodeType === Node.ELEMENT_NODE && parent.classList.contains('work-detail-unit'); },
		isTarget: function (target) { return target.classList.contains('works_info'); },
	});
function main() {
	var illustIdElement, commentArea, commentList, moreCommentButton, illustId, userId,
			page = DEFAULT_PAGE, count = 1, client;

	illustIdElement = document.getElementsByName('illust_id')[0];
	if (illustIdElement) {
		// イラストページ
		moreCommentButton = document.getElementsByClassName('more-comment')[0];
		if (moreCommentButton) {
			// 「もっと見る」ボタンが存在すれば
			// イラストID、投稿者IDを取得
			illustId = illustIdElement.value;
			userId = document.getElementsByClassName('tab-profile')[0].search.replace('?id=', '');
			
			// コメントリスト要素を取得
			commentList = document.getElementsByClassName('_comment-items')[0];

			// コメントが取得されたとき
			client = new XMLHttpRequest();
			client.addEventListener('load', function (event) {
				var data = event.target.response.body;
				
				// コメントを追加
				commentList.insertAdjacentHTML('beforeend', data.html);
				
				if (data.more) {
					// さらにコメントがあれば
					if (++page >= DEFAULT_PAGE + LIMIT_LOADING_COMMENT * count) {
						// コメント取得回数制限を超えていれば
						// 遅延読み込みされる画像リストの更新
						var script = document.createElement('script');
						script.text = 'pixiv.scrollView.update();';
						commentList.appendChild(script);
					} else {
						getComments();
					}
				} else {
					// 全コメントを取得済みであれば
					// 「もっと見る」ボタンを削除
					moreCommentButton.remove();
					
					// 遅延読み込みされる画像リストの更新
					var script = document.createElement('script');
					script.text = 'pixiv.scrollView.update();';
					commentList.appendChild(script);
				}
			});
						
			// 「もっと見る」ボタンが押されたとき
			moreCommentButton.addEventListener('click', function (event) {
				// ページ側で設定されたイベントを実行しない
				event.stopPropagation();

				// コメントを取得
				count++;
				getComments();
			});

			// コメントを取得
			getComments();
		}
	} else {
		// 小説ページ
		// コメント表示領域を取得
		commentArea = document.getElementsByClassName('comment-area')[0];
		if (commentArea && commentArea.classList.contains('comment-more')) {
			// 「もっと見る」ボタンが表示されていれば
			// 小説ID、投稿者IDを取得
			illustId = document.getElementsByName('id')[0].value;
			userId = document.getElementById('rpc_u_id').textContent;
			
			// コメントリスト要素を取得
			commentList = commentArea.getElementsByClassName('comment-list')[0];

			// コメントが取得されたとき
			client = new XMLHttpRequest();
			client.addEventListener('load', function (event) {
				var data = event.target.response.data;
				
				// コメントを追加
				commentList.insertAdjacentHTML('beforeend', data.html_array.join(''));

				if (data.more) {
					// さらにコメントがあれば
					if (++page >= DEFAULT_PAGE + LIMIT_LOADING_COMMENT * count) {
						// コメント取得回数制限を超えていれば
					} else {
						getComments();
					}
				} else {
					// 全コメントを取得済みであれば
					// 「もっと見る」ボタンを非表示に
					commentArea.classList.remove('comment-more');
				}
			});

			// 「もっと見る」ボタンが押されたとき
			commentArea.addEventListener('click', function (event) {
				if (event.target.classList.contains('comment-more-button')) {
					// ページ側で設定されたイベントを実行しない
					event.stopPropagation();

					// コメントを取得
					count++;
					getComments();
				}
			}, true);

			// コメントを取得
			getComments();
		}
	}

	function getComments() {
		// POSTパラメータを生成
		var data = new FormData();
		data.append('i_id', illustId);
		data.append('u_id', userId);
		data.append('p', page);

		// POSTリクエスト送信
		client.open('POST', 'rpc_comment_history.php');
		client.responseType = 'json';
		client.send(data);
	}
}



/**
 * 挿入された節の親節が、目印となる節の親節か否かを返すコールバック関数。
 * @callback isTargetParent
 * @param {(Document|Element)} parent
 * @returns {boolean}
 */

/**
 * 挿入された節が、目印となる節か否かを返すコールバック関数。
 * @callback isTarget
 * @param {(DocumentType|Element)} target
 * @returns {boolean}
 */

/**
 * 目印となる節が文書に存在するか否かを返すコールバック関数。
 * @callback existsTarget
 * @returns {boolean}
 */

/**
 * 目印となる節が挿入された直後に関数を実行する。
 * @param {Function} main - 実行する関数
 * @param {isTargetParent} isTargetParent
 * @param {isTarget} isTarget
 * @param {existsTarget} existsTarget
 * @param {Object} [callbacksForFirefox]
 * @param {isTargetParent} [callbacksForFirefox.isTargetParent] - FirefoxにおけるisTargetParent
 * @param {isTarget} [callbacksForFirefox.isTarget] - FirefoxにおけるisTarget
 * @version 2014-04-21
 */
function startScript(main, isTargetParent, isTarget, existsTarget, callbacksForFirefox) {
	/**
	 * {@link checkExistingTarget}で{@link startMain}を実行する間隔(ミリ秒)。
	 * @constant {number}
	 */
	var INTERVAL = 10;
	/**
	 * {@link checkExistingTarget}で{@link startMain}を実行する回数。
	 * @constant {number}
	 */
	var LIMIT = 500;

	/**
	 * 実行済みなら真。
	 * @type {boolean}
	 */
	var alreadyCalled = false;

	// 指定した節が既に存在していれば、即実行
	startMain();
	if (alreadyCalled) {
		return;
	}

	// FirefoxのMutationObserverは、HTMLのDOM構築に関して要素をまとめて挿入したと見なすため、isTargetParent、isTargetを変更
	if (callbacksForFirefox && typeof sidebar !== 'undefined') {
		if (callbacksForFirefox.isTargetParent) {
			isTargetParent = callbacksForFirefox.isTargetParent;
		}
		if (callbacksForFirefox.isTarget) {
			isTarget = callbacksForFirefox.isTarget;
		}
	}

	var observer = new MutationObserver(mutationCallback);
	observer.observe(document, {
		childList: true,
		subtree: true,
	});

	// DOMContentLoadedまでにスクリプトを実行できなかった場合、監視を停止(指定した節が存在するか確認し、存在すれば実行)
	document.addEventListener('DOMContentLoaded', onDOMContentLoaded);

	/**
	 * {@link startMain}を実行し、スクリプトが開始されていなければ{@link stopObserving}を実行する。
	 */
	function onDOMContentLoaded() {
		startMain();
		if (!alreadyCalled) {
			stopObserving();
		}
	}

	/**
	 * 目印となる節が挿入されたら、監視を停止し、{@link checkExistingTarget}を実行する。
	 * @param {MutationRecord[]} mutations - A list of MutationRecord objects.
	 * @param {MutationObserver} observer - The constructed MutationObserver object.
	 */
	function mutationCallback(mutations, observer) {
		var mutation, target, nodeType, addedNodes, addedNode, i, j, l, l2;
		for (i = 0, l = mutations.length; i < l; i++) {
			mutation = mutations[i];
			target = mutation.target;
			nodeType = target.nodeType;
			if ((nodeType === Node.ELEMENT_NODE) && isTargetParent(target)) {
				// 子が追加された節が要素節で、かつその節についてisTargetParentが真を返せば
				addedNodes = Array.prototype.slice.call(mutation.addedNodes);
				for (j = 0, l2 = addedNodes.length; j < l2; j++) {
					addedNode = addedNodes[j];
					nodeType = addedNode.nodeType;
					if ((nodeType === Node.ELEMENT_NODE || nodeType === Node.DOCUMENT_TYPE_NODE) && isTarget(addedNode)) {
						// 追加された子が要素節か文書型節で、かつその節についてisTargetが真を返せば
						observer.disconnect();
						checkExistingTarget(0);
						return;
					}
				}
			}
		}
	}

	/**
	 * {@link startMain}を実行し、スクリプトが開始されていなければ再度実行。
	 * @param {number} count - {@link startMain}を実行した回数。
	 */
	function checkExistingTarget(count) {
		startMain();
		if (!alreadyCalled && count < LIMIT) {
			window.setTimeout(checkExistingTarget, INTERVAL, count + 1);
		}
	}

	/**
	 * 指定した節が存在するか確認し、存在すれば{@link stopObserving}を実行しスクリプトを開始。
	 */
	function startMain() {
		if (!alreadyCalled && existsTarget()) {
			stopObserving();
			main();
		}
	}

	/**
	 * 監視を停止する。
	 */
	function stopObserving() {
		alreadyCalled = true;
		if (observer) {
			observer.disconnect();
		}
		document.removeEventListener('DOMContentLoaded', onDOMContentLoaded);
	}
}

})();

QingJ © 2025

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