pixiv ツリー型コメント

Changes the comment section's default linear mode to threaded mode.

当前为 2014-12-07 提交的版本,查看 最新版本

// ==UserScript==
// @name        pixiv ツリー型コメント
// @name:ja     pixiv ツリー型コメント
// @name:en     pixiv Threaded Comments
// @namespace   https://gf.qytechs.cn/users/137
// @version     2.0.0
// @description Changes the comment section's default linear mode to threaded mode.
// @description:ja 返信コメントを返信先の下に移動し、コメント欄を見やすく整理する
// @match       http://www.pixiv.net/member_illust.php?*mode=medium*
// @match       http://www.pixiv.net/novel/show.php?*id=*
// @match       http://www.pixiv.net/group/*?*id=*
// @run-at      document-start
// @grant       dummy
// @icon           
// @author      100の人
// @homepage    https://gf.qytechs.cn/scripts/5291-pixiv-%E3%83%84%E3%83%AA%E3%83%BC%E5%9E%8B%E3%82%B3%E3%83%A1%E3%83%B3%E3%83%88
// @license     Creative Commons Attribution 4.0 International Public License; http://creativecommons.org/licenses/by/4.0/
// ==/UserScript==

(function () {
'use strict';

/**
 * ページの種類。
 * @type {string}
 */
var pageType;
switch (window.location.pathname) {
	case '/member_illust.php':
		pageType = 'illust';
		break;
	case '/novel/show.php':
		pageType = 'novel';
		break;
	case '/group/':
	case '/group/show.php':
	case '/group/comment.php':
		pageType = 'group';
		break;
	default:
		return;
}

polyfill();

if (pageType !== 'group') {
	startScript(main,
		function (parent) { return parent.id === 'one_comment'; },
		function (target) { return target.classList.contains(pageType === 'illust' ? 'layout-column-2' : 'worksOptionRight'); },
		function () { return pageType === 'illust'
				? document.querySelector('#one_comment .layout-column-2')
				: document.getElementsByClassName('worksOptionRight')[0]; });
} else {
	startScript(main,
		function (parent) { return parent.id === 'wrapper'; },
		function (target) { return target.id === 'template-drawr-paint-ui'; },
		function () { return document.getElementById('template-drawr-paint-ui'); });
}

function main() {
	document.head.insertAdjacentHTML('beforeend', '<style>' + (pageType !== 'group' ? ' \
		/*==================================== \
			イラスト・小説のコメント欄 \
		*/ \
		/*------------------------------------ \
			余白を修正 \
		*/ \
		._comment-items ._comment-item, \
		._comment-items ._comment-sticker-item { \
			border-top: 1px solid #D6DEE5; \
			display: flex; \
			margin: initial; \
			padding: initial; \
		} \
		._comment-items ._comment-sticker-item { \
			display: block; \
			padding-bottom: 6px; \
		} \
		._comment-items ._comment-item > .user-icon-container, \
		._comment-items .comment:first-child::before { \
			/* アイコン */ \
			flex-shrink: 0; \
			position: initial; \
			margin: 0.2em 0.4em; \
		} \
		._comment-items .comment:first-child::before { \
			/* 削除されたユーザーのアイコン */ \
			content: ""; \
			position: absolute; \
			top: 0; \
			left: 0; \
			display: inline-block; \
			width: 40px; \
			height: 40px; \
			background: url("http://source.pixiv.net/common/images/no_profile_s.png") 0% 0% / 100% content-box; \
			opacity: 0.4; \
		} \
		._comment-items .comment:first-child { \
			margin-left: calc(40px + 1em); \
		} \
		._comment-items .comment, \
		._comment-items .has-action-list .comment, \
		._comment-items .host-user .comment { \
			/* アイコン以外の部分 */ \
			flex-grow: 1; \
			padding: 0; \
			margin: 0.2em; \
		} \
		._comment-items .meta { \
			/* 名前・投稿日時 */ \
			margin-bottom: initial; \
		} \
		#one_comment ._comment-sticker-item .sticker-item { \
			/* スタンプ */ \
			margin-top: 6px; \
			margin-left: 6px; \
		} \
		._comment-items .action-list { \
			/* 返信する・削除するボタン */ \
			position: absolute; \
			top: 0; \
			right: 0; \
		} \
		\
		/*------------------------------------ \
			作者コメント強調方法の修正 \
		*/ \
		._comment-items .host-user .comment > * { \
			/* 名前・投稿日時・返信先・スタンプ */ \
			text-align: initial; \
		} \
		._comment-items .host-user .body { \
			/* 本文 */ \
			float: initial; \
		} \
		._comment-items .host-user { \
			/* 背景色 */ \
			background: lavenderblush; \
		} \
		\
		/*------------------------------------ \
			返信コメント \
		*/ \
		._comment-items .tree > :nth-of-type(n+2) { \
			margin-left: 2em; \
		} \
		._comment-items .tree > :first-of-type.host-user { \
			border-bottom: 1px solid #D6DEE5; \
			margin-bottom: -1px; \
		} \
		\
		/*==================================== \
			イラスト・小説の返信ダイアログ \
		*/ \
		/*------------------------------------ \
			余白を修正 \
		*/ \
		._comment-modal .comment { \
			margin: initial; \
		} \
		._comment-modal .comment, \
		._comment-modal .host-user .comment { \
			padding-left: 90px; \
			padding-right: 20px; \
		} \
		/*------------------------------------ \
			作者コメント強調方法の修正 \
		*/ \
		._comment-modal .host-user .user-icon-container { \
			/* アイコン */ \
			left: 20px; \
		} \
		._comment-modal .host-user .comment > * { \
			/* 名前・投稿日時・返信先・スタンプ */ \
			text-align: initial; \
		} \
		._comment-modal .host-user .body { \
			/* 本文 */ \
			float: initial; \
		} \
		._comment-modal ._comment-item.host-user { \
			/* 背景色 */ \
			background: lavenderblush; \
		} \
		\
		/*==================================== \
			スタンプの表示方法の統一 \
		*/ \
		._comment-sticker-item .sticker { \
			/* 6個連続した時の大きさに */ \
			width: 67px; \
			height: 67px; \
		} \
		.sticker-item .user-icon { \
			/* 単体のスタンプを変換したアイテムのアイコン */ \
			width: 100%; \
		} \
		.comment > .body > p > img { \
			/* 単体の絵文字 */ \
			width: 28px; \
			height: initial; \
		} \
		' + (typeof MozSettingsEvent !== 'undefined' ? ' \
		/*==================================== \
			URLの取得に失敗しているアイコン \
		*/ \
		._comment-items ._comment-item > .user-icon-container[data-profile_img="_s."] { \
			background: url("http://source.pixiv.net/common/images/no_profile_s.png") 0% 0% / 100%; \
		}' : '')
	: ' \
		/*==================================== \
			グループのコメント欄 \
		*/ \
		/*------------------------------------ \
			区切り線 \
		*/ \
		#page-group #timeline li.post div.comment div.post, \
		#page-group #timeline li.post div.comment div.post ~ div.post { \
			border-top: dashed 1px #DEE0E8; \
			border-bottom: none; \
		} \
		#page-group #timeline li.post div.comment div.post::before { \
			content: ""; \
			position: absolute; \
			top: 0; \
			left: 0; \
			right: 0; \
			border-top: dashed 1px #FFFFFF; \
		} \
		#page-group #timeline li.post div.comment { \
			border-bottom: dashed 1px #DEE0E8; \
		} \
		/*------------------------------------ \
			最初のコメント \
		*/ \
		#page-group #timeline li.post div.comment > div.post:first-of-type, \
		#page-group #timeline li.post div.comment > .tree:first-of-type > div.post:first-of-type, \
		#page-group #timeline li.post div.comment > div.post:first-child::before, \
		#page-group #timeline li.post div.comment > .tree:first-child > div.post:first-of-type::before { \
			/* 一番上のコメント */ \
			/* 「以前のコメントを見る」ボタンがある時は、白色の破線は消さない */ \
			border-top: none; \
		} \
		/*------------------------------------ \
			返信コメント \
		*/ \
		#page-group #timeline li.post div.comment .tree > :nth-of-type(n+2) { \
			margin-left: 2em; \
		} \
		#page-group #timeline li.post div.comment .tree .postbody { \
			width: initial; \
		} \
	') + '</style>');

	/**
	 * コメントリスト要素。
	 * @type {HTMLDivElement}
	 */
	var commentList;

	/**
	 * コメント欄のすべてのスタンプを格納するリスト。
	 * @type {?HTMLDivElement}
	 */
	var allStickerList = null;

	/**
	 * スタンプの返信ボタン。
	 * @type {?HTMLUListElement}
	 */
	var itemActionsTemplate = document.getElementsByClassName('_item-actions')[0];

	if (pageType !== 'novel') {
		// 小説ページ以外なら
		/**
		 * {@link MutationObserver#observe}の第2引数に指定するオブジェクト。
		 * @type {MutationObserverInit}
		 */
		var observerOptions = {
			childList: true,
		};

		if (pageType === 'illust') {
			// イラストページ
			commentList = document.getElementsByClassName('_comment-items')[0];
		} else {
			// グループページ
			commentList = document.getElementById('timeline');
			observerOptions.subtree = true;
		}

		moveAllReplyComments();

		// コメントの増減を監視する
		new MutationObserver(function (mutations, observer) {
			var firstMutationRecord = mutations[0];
			var firstAddedNode = firstMutationRecord.addedNodes[0];
			if (firstAddedNode) {
				// コメントが増えていれば
				// 監視を一旦停止
				if (pageType === 'illust' && !firstMutationRecord.previousSibling) {
					// イラストページへのコメント投稿時
					firstAddedNode.style.display = '';
				}
				// 監視を一旦停止して返信コメントを移動する
				observer.disconnect();
				moveAllReplyComments();
				observer.observe(commentList, observerOptions);
			}
		}).observe(commentList, observerOptions);
	}

	/**
	 * すべての返信コメントを返信先コメントの下に移動する。
	 */
	function moveAllReplyComments() {
		if (pageType === 'illust') {
			// イラストページ

			// スタンプを一か所にまとめる
			if (allStickerList) {
				// まとめ先のリストが存在すれば
				if (allStickerList.nextElementSibling) {
					// コメント欄の一番下以外の場所にあれば、移動する
					commentList.appendChild(allStickerList);
				}
			}

			/**
			 * スタンプリスト、またはスタンプリストに含まれないスタンプ。
			 * @type {HTMLDivElement}
			 */
			for (var stickerListOrComment of commentList.querySelectorAll('._comment-sticker-item, ._comment-item .sticker-container')) {
				if (!allStickerList) {
					// まとめ先のリストがまだ作成されていなければ、作成
					commentList.insertAdjacentHTML('beforeend', '<div class="_comment-sticker-item sticker-container sticker-6-item"></div>');
					allStickerList = commentList.lastChild;
				}

				if (stickerListOrComment.classList.contains('_comment-sticker-item')) {
					// スタンプリストなら
					if (stickerListOrComment !== allStickerList) {
						// まとめ先のリストでなければ
						stickerListOrComment.remove();
						for (var sticker of stickerListOrComment.childNodes) {
							allStickerList.appendChild(sticker);
						}
					}
				} else {
					// スタンプなら
					/**
					 * コメントアイテム。
					 * @type {HTMLDivElement}
					 */
					var commentItem = stickerListOrComment.parentNode.parentNode;

					/**
					 * 返信先コメント投稿者名 (作者コメントの場合のみ)。
					 * @type {?HTMLSpanElement}
					 */
					var repliedUserName = commentItem.classList.contains('host-user') ? commentItem.getElementsByClassName('reply-to')[0] : null;

					if (repliedUserName) {
						// 作者コメント、かつ返信コメントなら、返信先のコメントIDを要素に保存
						commentItem.setAttribute('data-reply-to-id', repliedUserName.dataset.id);
					}

					// コメントアイテムをスタンプアイテムに変換
					commentItem.classList.remove('_comment-item');
					commentItem.classList.add('sticker-item');
					commentItem.replaceChild(commentItem.getElementsByClassName('sticker')[0], commentItem.getElementsByClassName('comment')[0]);

					// 返信ボタンを追加
					if (itemActionsTemplate) {
						commentItem.appendChild(itemActionsTemplate.cloneNode(true));
					} else {
						commentItem.insertAdjacentHTML('beforeend', '<ul class="_item-actions"> \
							<li class="reply item"> \
								<i class="_icon-12 _icon-reply"></i> \
							</li> \
						</ul>');
						itemActionsTemplate = commentItem.lastChild;
					}

					// スタンプリスト内にスタンプアイテムを移動
					allStickerList.appendChild(commentItem);
				}
			}

			// 返信コメントの移動
			/**
			 * 返信先コメント投稿者名、または返信スタンプ。
			 * @type {HTMLSpanElement}
			 */
			for (var repliedUserNameOrReplyStickerItem of commentList.querySelectorAll('.reply-to, .sticker-item[data-reply-to-id]')) {
				/**
				 * 返信スタンプなら真。
				 * @type {String}
				 */
				var replySticker = repliedUserNameOrReplyStickerItem.classList.contains('sticker-item');

				/**
				 * 返信先コメントのID。
				 * @type {string}
				 */
				var repliedCommentId = repliedUserNameOrReplyStickerItem.dataset[replySticker ? 'replyToId' : 'id'];

				/**
				 * 返信先コメント。
				 * @type {?HTMLDivElement}
				 */
				var repliedComment = commentList.querySelector('._comment-item[data-id="' + repliedCommentId + '"]');

				if (repliedComment) {
					// 返信先コメントが存在すれば
					/**
					 * 返信コメント。
					 * @type {?HTMLDivElement}
					 */
					var replyComment = null;
					if (replySticker) {
						// スタンプなら、スタンプリストに変換する
						replyComment = allStickerList.cloneNode();
						replyComment.classList.add('host-user');
						replyComment.appendChild(repliedUserNameOrReplyStickerItem);
					} else {
						// 通常のコメントなら
						replyComment = repliedUserNameOrReplyStickerItem.parentNode.parentNode;
						// 返信先コメント投稿者名を削除
						repliedUserNameOrReplyStickerItem.remove();
					}
					moveReplyComment(repliedComment, replyComment);
				}
			}
		} else {
			// グループページ
			var repliedUserNames = document.querySelectorAll('.body > p > a');
			for (var i = repliedUserNames.length - 1; i >= 0; i--) {
				/**
				 * 返信先コメント投稿者名。
				 * @type {HTMLSpanElement}
				 */
				var repliedUserName = repliedUserNames[i];

				/**
				 * 返信先コメント。
				 * @type {?HTMLDivElement}
				 */
				var repliedComment = document.getElementById(repliedUserName.hash.replace('#', ''));

				if (repliedComment) {
					// 返信先コメントが存在すれば
					moveReplyComment(repliedComment, repliedUserName.parentNode.parentNode.parentNode.parentNode.parentNode.parentNode);
					// 返信先コメント投稿者名を削除
					repliedUserName.parentNode.remove();
				}
			}
		}
	}

	/**
	 * 返信コメントを返信先コメントの下に移動する。
	 * @param {HTMLDivElement} repliedComment - 返信先コメント。
	 * @param {HTMLDivElement} replyComment - 返信コメント。
	 */
	function moveReplyComment(repliedComment, replyComment) {
		if (!replyComment.previousElementSibling) {
			var parent = replyComment.parentNode;
			if (parent && parent.classList.contains('tree')) {
				// 返信コメントにラッパー要素が存在すれば
				replyComment = parent;
			}
		}

		/**
		 * 返信先コメントと返信コメントを格納するラッパー要素。
		 * @type {HTMLDivElement}
		 */
		var tree = null;
		if (!repliedComment.previousElementSibling) {
			var parent = repliedComment.parentNode;
			if (parent.classList.contains('tree')) {
				// ラッパー要素がすでに存在すれば
				tree = parent;
			}
		}

		if (!tree) {
			// ラッパー要素が存在しなければ
			// ラッパー要素を作成
			tree = document.createElement('div');
			tree.classList.add('tree');
			// 返信先コメントをラッパー要素に置換
			repliedComment.parentNode.replaceChild(tree, repliedComment);
			// 返信先コメントをラッパーに追加
			tree.appendChild(repliedComment);
		}

		// 返信コメント移動
		if (pageType === 'illust') {
			// イラストページ
			tree.insertBefore(replyComment, repliedComment.nextSibling);
		} else {
			// グループページ
			tree.appendChild(replyComment);
		}
	}
}



/**
 * 挿入された節の親節が、目印となる節の親節か否かを返すコールバック関数。
 * @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における{@link isTargetParent}。
 * @param {isTarget} [callbacksForFirefox.isTarget] - Firefoxにおける{@link isTarget}。
 * @param {number} [timeoutSinceStopParsingDocument=0] - DOM構築完了後に監視を続けるミリ秒数。
 * @version 2014-11-25
 * @global
 */
function startScript(main, isTargetParent, isTarget, existsTarget) {
	/**
	 * {@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を変更
	var callbacksForFirefox = arguments[4];
	if (callbacksForFirefox && typeof MozSettingsEvent !== 'undefined') {
		isTargetParent = callbacksForFirefox.isTargetParent || isTargetParent;
		isTarget = callbacksForFirefox.isTarget || isTarget;
	}

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

	var timeoutSinceStopParsingDocument = arguments[5] || 0;
	if (document.readyState === 'complete') {
		// DOMの構築が完了していれば
		onDOMContentLoaded();
	} else {
		document.addEventListener('DOMContentLoaded', onDOMContentLoaded);
	}

	/**
	 * {@link startMain}を実行し、スクリプトが開始されていなければさらに{@link timeoutSinceStopParsingDocument}ミリ秒待機し、
	 * スクリプトが開始されていなければ{@link stopObserving}を実行する。
	 */
	function onDOMContentLoaded() {
		startMain();
		if (timeoutSinceStopParsingDocument === 0) {
			if (!alreadyCalled) {
				stopObserving();
			}
		} else {
			window.setTimeout(function () {
				if (!alreadyCalled) {
					stopObserving();
				}
			}, timeoutSinceStopParsingDocument);
		}
	}

	/**
	 * 目印となる節が挿入されたら、監視を停止し、{@link checkExistingTarget}を実行する。
	 * @param {MutationRecord[]} mutations - A list of MutationRecord objects.
	 * @param {MutationObserver} observer - The constructed MutationObserver object.
	 */
	function mutationCallback(mutations, observer) {
		for (var mutation of mutations) {
			var target = mutation.target;
			if (target.nodeType === Node.ELEMENT_NODE && isTargetParent(target)) {
				// 子が追加された節が要素節で、かつその節についてisTargetParentが真を返せば
				for (var addedNode of mutation.addedNodes) {
					if (addedNode.nodeType === Node.ELEMENT_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);
	}
}



/**
 * WHATWG仕様のPolyfill。
 */
function polyfill() {
	// Polyfill for Opera and Google Chrome
	if (!('@@iterator' in NodeList.prototype) && !(Symbol.iterator in NodeList.prototype)) {
		/** @version polyfill-2014-12-07 */
		Object.defineProperties(NodeList.prototype, /** @lends NodeList# */ {
			/**
			 * @returns {Iterator.<Array.<number, Node>>}
			 * @function
			 */
			entries: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function* () {
					for (var i = 0, l = this.length; i < l; i++) {
						yield [i, this[i]];
					}
				}
			},
			/**
			 * @returns {Iterator.<number>}
			 * @function
			 */
			keys: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function* () {
					for (var i = 0, l = this.length; i < l; i++) {
						yield i;
					}
				}
			},
			/**
			 * @returns {Iterator.<Node>}
			 * @function
			 */
			values: {
				writable: true,
				enumerable: false,
				configurable: true,
				value: function* () {
					for (var i = 0, l = this.length; i < l; i++) {
						yield this[i];
					}
				}
			},
		});
		/**
		 * @returns {Iterator.<Node>}
		 * @function NodeList#@@iterator
		 */
		Object.defineProperty(NodeList.prototype, Symbol.iterator, {
			writable: true,
			enumerable: false,
			configurable: true,
			value: function* () {
				for (var i = 0, l = this.length; i < l; i++) {
					yield this[i];
				}
			}
		});
	}
}


})();

QingJ © 2025

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