// ==UserScript==
// @name Nejire Refine
// @namespace http://nejiten.halfmoon.jp/
// @version 1.0.0
// @description ねじれ天国のUIを使いやすくするスクリプトです。
// @author euro_s
// @match https://nejiten.halfmoon.jp/index.cgi?vid=*
// @icon https://www.google.com/s2/favicons?sz=64&domain=halfmoon.jp
// @grant GM_addStyle
// @grant GM_setValue
// @grant GM_getValue
// @require https://code.jquery.com/jquery-3.6.0.min.js
// @run-at document-start
// @license MIT
// ==/UserScript==
(function () {
"use strict";
////////////////////////////////////////////////////////////////////////////////
// 設定
////////////////////////////////////////////////////////////////////////////////
// サイドバーの表示(デフォルト非表示)
let sideBarDisplay = GM_getValue("sideBarDisplay", "none");
// 表示の場合の値
// let sideBarDisplay = 'table-cell';
// ページリンクの表示(デフォルト表示)
let alllogAnnounceDisplay = GM_getValue("alllogAnnounceDisplay", "block");
// 非表示の場合の値
// let alllogAnnounceDisplay = 'none';
// テキストエリアのフォントサイズ(デフォルト1.5)
let textareaFontSize = GM_getValue("textareaFontSize", 1.5);
// メッセージの最大行数 これを超えると省略される(デフォルト20)
let maxBrTags = GM_getValue("maxBrTags", 20);
// メッセージの最大文字数 これを超えると省略される(デフォルト800)
let maxTextLength = GM_getValue("maxTextLength", 800);
// 発言に含まれる画像の最大数 これを超えると削除される(デフォルト10)
let maxImgCount = GM_getValue("maxImgCount", 10);
// 連投制限(デフォルトで省略=truncate, 削除=delete, 何もしない=none)
let spamOption = GM_getValue("spamOption", "truncate");
// ページ右上に移動ボタンを表示するか(デフォルト非表示)
let showScrollButton = GM_getValue("showScrollButton", false);
// ページ右下に検索ボタンを表示するか(デフォルト非表示)
let showSearchButton = GM_getValue("showSearchButton", false);
////////////////////////////////////////////////////////////////////////////////
// メタ情報
////////////////////////////////////////////////////////////////////////////////
let vid = 0;
let date = 0;
let idAndNames = new Map();
function getMetadata() {
const url = location.href;
const vidMatch = url.match(/vid=(\d+)/);
if (vidMatch) {
vid = vidMatch[1];
}
const dateMatch = url.match(/date=(\d+)/);
if (dateMatch) {
date = dateMatch[1];
} else {
const str = document.querySelector("span.today").innerText;
const dateMatch = str.match(/(\d+)日目/);
if (dateMatch) {
date = dateMatch[1];
}
}
const list = document.getElementById("list");
const aTags = list.getElementsByTagName("a");
for (let i = 0; i < aTags.length; i++) {
const aTag = aTags[i];
const href = aTag.href;
const idMatch = href.match(/&id=(\d+)/);
if (idMatch) {
const id = idMatch[1];
const name = aTag.innerText;
idAndNames.set(id, name);
}
}
}
////////////////////////////////////////////////////////////////////////////////
// メイン
////////////////////////////////////////////////////////////////////////////////
window.addEventListener("DOMContentLoaded", async (event) => {
getMetadata();
createTopButtonContainer();
createBottomButtonContainer();
disableDoubleSubmit();
createSettingDialog();
upDownButtons();
new MutationObserver(processMessage).observe(
document.querySelector("#content"),
{ childList: true }
);
processMessage();
new MutationObserver(processAnnounce).observe(
document.querySelector("#content"),
{ childList: true }
);
processAnnounce();
await addCopyAnchorEvent();
disableLeaveButtonAndAddCheckbox();
pagination();
setAnchorToggle();
addSearchIcon();
applySettings();
});
////////////////////////////////////////////////////////////////////////////////
// 各種処理
////////////////////////////////////////////////////////////////////////////////
// 設定の反映
function applySettings() {
// サイドバーの表示
const sideBars = document.querySelectorAll(
"body > table > tbody > tr > td > table > tbody > tr > td > table > tbody > tr > td:nth-child(1)"
);
sideBars.forEach((sideBar) => {
sideBar.style.display = sideBarDisplay;
});
// ページリンクの表示
const alllogAnnounces = document.querySelectorAll(".alllog_announce");
alllogAnnounces.forEach((alllogAnnounce) => {
alllogAnnounce.style.display = alllogAnnounceDisplay;
});
// 発言欄のフォントサイズ
const textarea = document.querySelectorAll("textarea");
textarea.forEach((ta) => {
ta.style.fontSize = `${textareaFontSize}rem`;
});
// 上下移動ボタンの表示
const buttonUp = document.getElementById("scrollToTopButton");
const buttonDown = document.getElementById("scrollToBottomButton");
if (showScrollButton) {
buttonUp.style.display = "block";
buttonDown.style.display = "block";
} else {
buttonUp.style.display = "none";
buttonDown.style.display = "none";
}
// 検索ボタンの表示
const searchButton = document.getElementById("searchIcon");
if (showSearchButton) {
searchButton.style.display = "block";
} else {
searchButton.style.display = "none";
}
}
// 設定ボタンを作成
function createSettingDialog() {
const btn = document.createElement("button");
btn.type = "button";
btn.title = "設定";
btn.id = "settingButton";
// 歯車アイコンのSVGをBase64でエンコードしたもの 白背景用
btn.innerHTML = `
<img src=""/>
`;
// ボタンのクリックイベントリスナーを設定
btn.addEventListener("click", function () {
// ボタンがクリックされたときにポップアップウィンドウを表示
dialog.showModal();
});
// ボタンを挿入する位置
const action_box = document.querySelector(".action_box");
const entryForm = document.querySelector("form[name=entryForm]");
let target;
if (action_box && !entryForm) {
target = action_box.querySelector(
"td.action_body" // 発言欄の名前の右
);
// replace innerHTML
target.innerHTML = target.innerHTML.replace('</span>', '</span></div>');
target.innerHTML = `<div>${target.innerHTML}`;
target.classList.add("container");
const span = document.createElement("span");
span.classList.add("right");
btn.classList.add("with_action_box");
// ボタンを追加
span.appendChild(btn);
target.appendChild(span);
// hopeForm の位置を変更
const hopeForm = document.querySelector("form[name=hopeForm]");
const boxHope = document.querySelector("#box_hope");
if (hopeForm && boxHope) {
boxHope.parentElement.appendChild(hopeForm);
boxHope.parentElement.removeChild(boxHope);
hopeForm.appendChild(boxHope);
}
} else {
// ログアウト中などで発言欄がない場合、または入村前の場合
// aタグ、title=にゃおーんの要素を取得してそこに追加
target = document.querySelector('a[title="にゃおーん"]');
// 歯車アイコンのSVGをBase64でエンコードしたもの 黒背景用
btn.innerHTML = `
<img src=""/>
`;
btn.classList.add("no_action_box");
if (target) {
target.insertAdjacentElement("afterend", btn);
}
}
// 設定値の設定
const sidebarChecked = sideBarDisplay == "table-cell" ? "checked" : "";
const pageLinkChecked = alllogAnnounceDisplay == "block" ? "checked" : "";
const showScrollButtonChecked = showScrollButton ? "checked" : "";
const showSearchButtonChecked = showSearchButton ? "checked" : "";
const truncateSelected = spamOption == "truncate" ? "selected" : "";
const deleteSelected = spamOption == "delete" ? "selected" : "";
const noneSelected = spamOption == "none" ? "selected" : "";
// ポップアップウィンドウ(<dialog>)を作成
const dialog = document.createElement("dialog");
dialog.innerHTML = `
<div class="dialog">
<h2>Nejire Refine 設定</h2>
<label>
<input type="checkbox" id="sidebarCheckbox" ${sidebarChecked}>
サイドバーを表示する
</label>
<br/>
<label>
<input type="checkbox" id="pageLinkCheckbox" ${pageLinkChecked}>
ページリンクを表示する
</label>
<br/>
<label>
<input type="checkbox" id="showScrollButton" ${showScrollButtonChecked}>
一番上/一番下に移動するボタンを表示する
</label>
<br/>
<label>
<input type="checkbox" id="showSearchButton" ${showSearchButtonChecked}>
検索ボタンを表示する
</label>
<br/>
<hr>
<div>ここから下は保存したあと画面リロードで反映</div>
<label for="spamControl">連投制御オプション:</label>
<select id="spamControl">
<option value="truncate" ${truncateSelected}>省略表示</option>
<option value="delete" ${deleteSelected}>発言全体を削除</option>
<option value="none" ${noneSelected}>何もしない</option>
</select>
<br/>
<label>
<input type="number" id="fontSizeInput" value="${textareaFontSize}">
発言欄のフォントサイズ(rem) デフォルトは 1.5
</label>
<br/>
<label>
<input type="number" id="maxLinesInput" value="${maxBrTags}">
発言の最大行数(超えると省略) デフォルトは 20
</label>
<br/>
<label>
<input type="number" id="maxCharsInput" value="${maxTextLength}">
発言の最大文字数(超えると省略) デフォルトは 800
</label>
<br/>
<label>
<input type="number" id="maxImagesInput" value="${maxImgCount}">
発言の最大画像数(タペストリー対策) デフォルトは 10
</label>
<br/>
<br/>
</div>
`;
document.body.appendChild(dialog);
// 設定を保存し、ダイアログを閉じるためのボタンを作成
let saveButton = document.createElement("button");
saveButton.classList.add("primary");
saveButton.type = "button";
saveButton.innerText = "保存";
saveButton.addEventListener("click", () => {
// 設定値を変数に反映
sideBarDisplay = document.querySelector("#sidebarCheckbox").checked
? "table-cell"
: "none";
alllogAnnounceDisplay = document.querySelector("#pageLinkCheckbox")
.checked
? "block"
: "none";
spamOption = document.querySelector("#spamControl").value;
textareaFontSize = document.querySelector("#fontSizeInput").value;
maxBrTags = document.querySelector("#maxLinesInput").value;
maxTextLength = document.querySelector("#maxCharsInput").value;
maxImgCount = document.querySelector("#maxImagesInput").value;
showScrollButton = document.querySelector("#showScrollButton").checked;
showSearchButton = document.querySelector("#showSearchButton").checked;
// 設定値を保存
GM_setValue("sideBarDisplay", sideBarDisplay);
GM_setValue("alllogAnnounceDisplay", alllogAnnounceDisplay);
GM_setValue("spamOption", spamOption);
GM_setValue("textareaFontSize", textareaFontSize);
GM_setValue("maxBrTags", maxBrTags);
GM_setValue("maxTextLength", maxTextLength);
GM_setValue("maxImgCount", maxImgCount);
GM_setValue("showScrollButton", showScrollButton);
GM_setValue("showSearchButton", showSearchButton);
// 変数からDOMに反映
applySettings();
dialog.close();
});
dialog.appendChild(saveButton);
// 設定をキャンセルし、ダイアログを閉じるためのボタンを作成
let cancelButton = document.createElement("button");
cancelButton.classList.add("secondary");
cancelButton.type = "button";
cancelButton.innerText = "キャンセル";
cancelButton.addEventListener("click", () => {
// 設定を元に戻す
document.querySelector("#sidebarCheckbox").checked =
sideBarDisplay == "table-cell";
document.querySelector("#pageLinkCheckbox").checked =
alllogAnnounceDisplay == "block";
document.querySelector("#spamControl").value = spamOption;
document.querySelector("#fontSizeInput").value = textareaFontSize;
document.querySelector("#maxLinesInput").value = maxBrTags;
document.querySelector("#maxCharsInput").value = maxTextLength;
document.querySelector("#maxImagesInput").value = maxImgCount;
document.querySelector("#showScrollButton").checked = showScrollButton;
document.querySelector("#showSearchButton").checked = showSearchButton;
// ダイアログを閉じる
dialog.close();
});
dialog.appendChild(cancelButton);
}
// 右上ボタン群のコンテナ作成
function createTopButtonContainer() {
if (document.querySelector("#topButtonContainer")) {
return;
}
// Create a new div element for buttons
const topButtonContainer = document.createElement("div");
topButtonContainer.id = "topButtonContainer";
// Append the new div element to the bottom of the page
document.body.appendChild(topButtonContainer);
}
// 右下ボタン群のコンテナ作成
function createBottomButtonContainer() {
if (document.querySelector("#bottomButtonContainer")) {
return;
}
// Create a new div element for buttons
const bottomButtonContainer = document.createElement("div");
bottomButtonContainer.id = "bottomButtonContainer";
// Append the new div element to the bottom of the page
document.body.appendChild(bottomButtonContainer);
}
// 上下スクロールボタンを作成
function upDownButtons() {
const topButtonContainer = document.querySelector("#topButtonContainer");
// Create a new button element for scrolling to bottom
const buttonDown = document.createElement("button");
buttonDown.id = "scrollToBottomButton";
buttonDown.innerHTML = `
<img src=""/>
`;
// Create a new button element for scrolling to top
const buttonUp = document.createElement("button");
buttonUp.id = "scrollToTopButton";
buttonUp.innerHTML = `
<img src=""/>
`;
// Add the buttons to the document body
topButtonContainer.append(buttonUp, buttonDown);
// Attach an event listener to the buttons to handle clicks
buttonDown.addEventListener("click", function () {
window.scrollTo({
top: document.body.scrollHeight, // Scroll to the bottom of the page
behavior: "smooth", // Animate the scroll
});
});
buttonUp.addEventListener("click", function () {
window.scrollTo({
top: 0, // Scroll to the top of the page
behavior: "smooth", // Animate the scroll
});
});
}
// 発言ボタン2度押し防止
function disableDoubleSubmit() {
const submitButtons = document.querySelectorAll('input[type="submit"]');
for (const submitButton of submitButtons) {
submitButton.addEventListener("click", (event) => {
setTimeout(() => (this.disabled = true), 0);
});
}
}
// メッセージの情報を取得する
function getMessageInformationFromMessageTable(messageTable) {
// mark processed message, add data-processed: true
messageTable.setAttribute("data-processed", true);
const meta = messageTable.querySelector("tbody > tr:nth-child(1)");
const icon = meta
.querySelector("td:nth-child(1) > img")
.getAttribute("src");
const number = meta.querySelector(
"td:nth-child(2) > span.mes_number"
)?.textContent;
const nameATag = meta.querySelector("td:nth-child(2) > a:nth-child(3)");
let id;
let name;
if (!nameATag) {
// nameATagがないのは匿名ユーザー
id = "anonymous";
name = "汝はねじれなりや?";
} else {
name = nameATag.textContent;
let match;
// 参加中のユーザーは数値のidを持つ
match = /&id=(\d+)/.exec(nameATag.href);
if (match) {
id = match[1];
} else {
// 観戦発言などはuidを持つ
match = /&uid=(\w+)/.exec(nameATag.href);
if (match) {
id = match[1];
}
}
}
if (!id) {
console.error("IDが取得できませんでした");
console.error(nameATag.href);
}
const content = messageTable.querySelector("[class$=body1]");
const original = content.innerHTML;
return { id, name, number, icon, content, original };
}
// div.announceの処理
function processAnnounce() {
const announces = document.querySelectorAll("div.announce:not([data-processed])");
for (const announce of announces) {
// mark processed announce, add data-processed: true
announce.setAttribute("data-processed", true);
if (announce.classList.contains("testament")) {
// 遺言
let tempHTML = announce.innerHTML;
const brTags = tempHTML.split("<br>");
const text = announce.textContent;
const originalContentHTML = announce.innerHTML;
if (brTags && brTags.length > maxBrTags) {
// brタグの数がmaxBrTagsを超えていたら、それ以降のHTMLを削除し、最後に...を追加したHTMLを作成する
const tempBrTags = brTags.slice(0, maxBrTags);
tempHTML =
tempBrTags.join("<br>") +
"<br><a class='ellipsis' title='省略されています'>...</a>";
} else if (text.length > maxTextLength) {
// textContent の長さがmaxTextLengthを超えていたら、まずtextをmaxTextLengthで切り取る
announce.textContent = text.slice(0, maxTextLength);
announce.innerHTML +=
"<a class='ellipsis' title='省略されています'>...</a>";
tempHTML = announce.innerHTML;
}
announce.innerHTML = tempHTML;
// ...にはイベントリスナーを追加し、クリックされたら全文を表示する
announce
.querySelector("a.ellipsis")
?.addEventListener("click", (event) => {
event.preventDefault();
announce.innerHTML = originalContentHTML;
});
}
}
}
// messageの処理
function processMessage() {
const messages = document.querySelectorAll(
"table.message:not([data-processed])"
);
const originalContentHTML = [];
const latestMessages = {};
for (const [i, messageTable] of messages.entries()) {
const message = getMessageInformationFromMessageTable(messageTable);
// 発言者のIDでグループ化
if (!latestMessages[message.id]) {
latestMessages[message.id] = [];
}
latestMessages[message.id].push(message);
// originalの本文を保存しておく
originalContentHTML.push(message.content.innerHTML);
// 連投対策。同じ発言者の連続した発言は、発言内容を省略する
if (spamOption == "truncate" || spamOption == "delete") {
if (i > 0) {
if (latestMessages[message.id].length > 1) {
// 連投していたら、最後の発言の内容を省略する
const lastMessage =
latestMessages[message.id][latestMessages[message.id].length - 2];
if (lastMessage.original == message.original) {
if (spamOption == "truncate") {
message.content.innerHTML =
"<a class='ellipsis' title='省略されています'>...</a>";
} else if (spamOption == "delete") {
messageTable.parentElement.removeChild(messageTable);
}
}
}
}
}
// タペストリー対策。imgタグがmaxImgCountを超えたら削除する
const imgs = message.content.querySelectorAll("img");
if (imgs.length > maxImgCount) {
// 画像を削除したことを通知するメッセージ要素を作成
const messageElement = document.createElement("span");
messageElement.classList.add("ellipsis");
messageElement.innerText = `この発言の画像は省略されました。`;
imgs[0].parentElement.appendChild(messageElement);
imgs.forEach((img) => {
img.parentElement.removeChild(img);
});
}
// 長文対策
const innerHTML = message.content.innerHTML;
const text = message.content.textContent;
let count = 0;
const savedLinks = {};
// { と } をエスケープ
let tempHTML = innerHTML.replace(/({|})/g, "\\$1");
// aタグをプレースホルダーに置換
tempHTML = tempHTML.replace(/<a\b[^>]*>(.*?)<\/a>/gi, function (match) {
const placeholder = `{{link${count}}}`;
savedLinks[placeholder] = match;
count++;
return placeholder;
});
// 置換したHTMLを<br>で分割
const brTags = tempHTML.split("<br>");
if (brTags && brTags.length > maxBrTags) {
// brタグの数がmaxBrTagsを超えていたら、それ以降のHTMLを削除し、最後に...を追加したHTMLを作成する
const tempBrTags = brTags.slice(0, maxBrTags);
tempHTML =
tempBrTags.join("<br>") +
"<br><a class='ellipsis' title='省略されています'>...</a>";
} else if (text.length > maxTextLength) {
// textContent の長さがmaxTextLengthを超えていたら、まずtextをmaxTextLengthで切り取る
// この場合、リンクやアンカーなどのタグが消えてしまうが、今のところしょうがない
message.content.textContent = text.slice(0, maxTextLength);
message.content.innerHTML +=
"<a class='ellipsis' title='省略されています'>...</a>";
tempHTML = message.content.innerHTML;
}
// プレースホルダーを元のaタグに戻す
for (let placeholder in savedLinks) {
tempHTML = tempHTML.replace(placeholder, savedLinks[placeholder]);
}
// エスケープした { と } を元に戻す
tempHTML = tempHTML.replace(/\\({|})/g, "$1");
message.content.innerHTML = tempHTML;
// ...にはイベントリスナーを追加し、クリックされたら全文を表示する
message.content
.querySelector("a.ellipsis")
?.addEventListener("click", (event) => {
event.preventDefault();
message.content.innerHTML = originalContentHTML[i];
});
}
}
// 過去ログのアンカーをコピーする処理の追加
async function addCopyAnchorEvent() {
const match = /&date=(\d+)/.exec(location.href);
if (match) {
const date = match[1];
const mesNumbers = document.querySelectorAll("span.mes_number");
for (const mesNumberElm of mesNumbers) {
mesNumberElm.addEventListener("click", async (event) => {
const mesNumber = mesNumberElm.textContent;
const anchor = `>>${date}:${mesNumber}`;
// コピーするテキストを一時的なテキストエリアにセット
const tempTextArea = document.createElement("textarea");
tempTextArea.style.position = "absolute";
tempTextArea.style.left = "-9999px";
tempTextArea.value = anchor;
document.body.appendChild(tempTextArea);
tempTextArea.select();
// クリップボードにコピー
document.execCommand("copy");
document.body.removeChild(tempTextArea);
// コピーしたことを通知
const notification = document.createElement("div");
notification.textContent = `${anchor}をコピーしました`;
notification.style.position = "fixed";
notification.style.top = `${event.clientY + 20}px`;
notification.style.left = `${event.clientX + 20}px`;
notification.style.zIndex = "9999";
notification.style.backgroundColor = "black";
notification.style.padding = "1rem";
notification.style.border = "1px solid black";
notification.style.borderRadius = "1rem";
notification.style.opacity = "0";
notification.style.transition = "opacity 0.5s";
document.body.appendChild(notification);
setTimeout(() => {
notification.style.opacity = "1";
}, 10);
setTimeout(() => {
notification.style.opacity = "0";
setTimeout(() => {
document.body.removeChild(notification);
}, 500);
}, 1000);
});
}
}
}
// 村を出るボタンの無効化とチェックボックスの追加
function disableLeaveButtonAndAddCheckbox() {
const leaeveButton = document.querySelector("input[value='村を出る']");
if (leaeveButton) {
leaeveButton.disabled = true;
leaeveButton.style.opacity = "0.5";
const checkbox = document.createElement("input");
checkbox.type = "checkbox";
checkbox.id = "vil_leave_checkbox";
checkbox.style.marginRight = "0.5rem";
checkbox.style.verticalAlign = "middle";
checkbox.addEventListener("change", (event) => {
leaeveButton.disabled = !event.target.checked;
leaeveButton.style.opacity = event.target.checked ? "1" : "0.5";
});
// add checkbox element before leave button
leaeveButton.parentNode.insertBefore(checkbox, leaeveButton);
}
}
// ページネーションの追加
function pagination() {
const announce = document.querySelector(".alllog_announce");
if (!announce) {
return;
}
const match = window.location.href.match(/part=(\d+)/);
if (match) {
const currentPart = parseInt(match[1]);
const parts = Array.from(announce.querySelectorAll("a[href*='part']"))
.map((a) => parseInt(a.innerText))
.filter((n) => !isNaN(n));
const minPart = 1;
const maxPart = Math.max(parts[parts.length - 1], currentPart);
const firstMessage = getFirstMessageOrAnnounce();
const lastMessage = getLastMessageOrAnnounce();
if (currentPart > minPart) {
const prevButton = createPaginationButton(
"▲前のページを読み込む",
currentPart - 1,
firstMessage,
false,
maxPart
);
firstMessage.parentNode.insertBefore(prevButton, firstMessage.previousSibling);
}
if (currentPart < maxPart) {
const nextButton = createPaginationButton(
"▼次のページを読み込む",
currentPart + 1,
lastMessage,
true,
maxPart
);
// insert next button after last message
lastMessage.parentNode.insertBefore(nextButton, lastMessage.nextSibling);
}
}
}
function getFirstMessageOrAnnounce() {
const content = document.querySelector("#content");
const firstChild = content.firstElementChild;
if (firstChild.classList.contains("alllog_announce")) {
return firstChild.nextElementSibling;
} else {
return firstChild;
}
}
function getLastMessageOrAnnounce() {
const content = document.querySelector("#content");
const lastChild = content.lastElementChild;
if (lastChild.classList.contains("alllog_announce")) {
return lastChild.previousSibling;
} else {
return lastChild;
}
}
// ページネーションのボタンを作成
function createPaginationButton(text, part, anchorMessage, isNext, maxPart) {
const button = document.createElement("button");
button.innerText = text;
button.style.cursor = "pointer";
button.style.width = "100%";
button.style.margin = "10px 0";
button.style.backgroundColor = "black";
button.style.color = "#994";
button.style.borderWidth = "0";
button.setAttribute("data-part", part);
button.addEventListener("click", async function () {
const part = parseInt(this.getAttribute("data-part"));
if ((part <= 0 && !isNext) || (part > maxPart && isNext)) {
alert(isNext ? "最後のページです。" : "最初のページです");
return;
}
const messages = await fetchPage(part, isNext);
const anchorRectBefore = anchorMessage.getBoundingClientRect();
messages.forEach((message) => {
anchorMessage.parentElement.insertBefore(
message,
isNext ? anchorMessage.nextSibling : anchorMessage
);
});
const anchorRectAfter = anchorMessage.getBoundingClientRect();
const topDiff = anchorRectAfter.top - anchorRectBefore.top;
window.scrollBy(0, topDiff);
anchorMessage = messages[0];
button.setAttribute("data-part", isNext ? part + 1 : part - 1);
});
return button;
}
// ページを取得
async function fetchPage(part, isNext) {
const url = window.location.href.replace(/part=\d+/, `part=${part}`);
const response = await fetch(url);
const arrayBuffer = await response.arrayBuffer();
const html = new TextDecoder("euc-jp").decode(arrayBuffer);
const parser = new DOMParser();
const doc = parser.parseFromString(html, "text/html");
const messages = Array.from(doc.querySelectorAll("#content > *")).filter(
(e) => !e.classList.contains("alllog_announce")
);
return isNext ? messages.reverse() : messages;
}
// アンカーのAjaxを再設定
function setAnchorToggle() {
$(document).on("click touchstart touchend", ".say", function (event) {
event.preventDefault();
if ($(this).data("clicked")) {
onSayToggleOff.call(this);
} else {
onSayClick.call(this);
}
$(this).data("clicked", !$(this).data("clicked"));
});
$(document).on("click", ".mes_res", function (event) {
if ($(this).data("clicked")) {
onMesResToggleOff.call(this);
} else {
onMesResClick.call(this);
}
$(this).data("clicked", !$(this).data("clicked"));
});
}
// アンカークリック時の処理
function onSayClick() {
const ank = $(this);
if (ank.text().startsWith(">>")) {
if (!ank.attr("onmouseover")) {
if (
confirm(
"このアンカーの遷移先が見つかりません。それでも移動しますか?"
)
) {
window.location.href = this.href;
}
return false;
}
const href = this.href.replace("all", "anc").replace("#", "&num=");
$.get(href, function (data) {
const mes = $(data).find(".anchor");
mes.addClass("ajax");
insertElementAfterParents(mes, ank, [".message", ".announce"]);
mes.find(".mes_res").show();
mes.hide().toggle("slide");
});
} else {
window.open(this.href, "_blank");
}
return false;
}
// アンカー閉じる処理
function onSayToggleOff() {
toggleSlideAndRemove($(this), [".message", ".announce"]);
return false;
}
// レスクリック時の処理
function onMesResClick() {
const ank = $(this);
const text = markEscape(ank.attr("id"));
const reg = new RegExp(text + "(?![\\d:])");
const res = $("<div>");
$("#content")
.children('.message, [class^="announce"]')
.each(function () {
const elementText = $(this).find(".say").text();
if (elementText.search(reg) === -1) return;
const cloned = $(this).clone().show();
res.append(cloned).addClass("ajax");
});
insertElementAfterParents(res, ank, [".message", ".announce"]);
res.hide().toggle("slide");
}
// レス閉じる処理
function onMesResToggleOff() {
toggleSlideAndRemove($(this), [".message", ".announce"]);
return false;
}
// 要素を削除
function toggleSlideAndRemove(element, parentSelectors) {
parentSelectors.forEach((selector) => {
const parent = element.parents(selector);
parent.nextAll(".ajax").toggle("slide", function () {
parent.nextAll(".ajax").remove();
});
});
}
// 親要素の後ろに要素を挿入する
function insertElementAfterParents(element, reference, parentSelectors) {
parentSelectors.forEach((selector) => {
const parent = reference.parents(selector);
parent.after(element);
});
}
// 検索ボタンを追加
function addSearchIcon() {
const searchIcon = `
<img src="">
`;
const $searchBoxContainer = $("<div>", { id: "searchBoxContainer" })
.html(
`
<div class="container">
<input id="searchBox" type="text" placeholder="Search...">
<input id="searchButton" type="button" value="検索" class="submit2" onclick="window.open('?vid=${vid}&date=${date}&hash=' + encodeURIComponent(metaEscape(document.querySelector('#searchBox').value)));">
</div>
<hr>
<div class="container">
<select id="filter">
</select>
<input id="filterButton" type="button" value="抽出" class="submit2" onclick="window.open('?vid=${vid}&date=${date}&id=' + encodeURIComponent(document.querySelector('#filter').value));">
</div>
`
)
.hide();
// idAndNamesの各要素を<select>ボックスに追加します
idAndNames.forEach((name, id) => {
const $option = $("<option>", {
value: id,
text: name,
});
$("#filter", $searchBoxContainer).append($option);
});
const $searchIcon = $("<button>", { id: "searchIcon" }).html(searchIcon);
$("#bottomButtonContainer").append($searchBoxContainer).append($searchIcon);
$searchIcon.on("click", function (event) {
$searchBoxContainer.toggle();
event.stopPropagation();
});
$searchBoxContainer.on("click", function (event) {
event.stopPropagation();
});
$("#searchButton, #filterButton", $searchBoxContainer).on("click", function () {
$searchBoxContainer.hide();
});
$(document).on("click", function () {
$searchBoxContainer.hide();
});
$("#searchBox").on("keydown", function (event) {
if (event.which === 13 || event.keyCode === 13) {
// 13 is the key code for Enter
$("#searchButton").click(); // Trigger the click event on the button
event.preventDefault(); // Prevent the default behavior of Enter key
}
});
}
////////////////////////////////////////////////////////////////////////////////
// ねじれスクリプト置換
////////////////////////////////////////////////////////////////////////////////
unsafeWindow.getElementsByClass = getElementsByClass;
unsafeWindow.setAjaxEvent = setAjaxEvent;
function getElementsByClass(searchClass) {
// もしsearchClassが"announce"の場合、そのクラス名を含むすべての要素を取得
if (searchClass === "announce") {
return Array.from(document.querySelectorAll(`[class*="${searchClass}"]`));
}
// それ以外の場合、クラス名が正確にマッチする要素のみを取得
return Array.from(document.getElementsByClassName(searchClass));
}
// ねじれスクリプトのsetAjaxEvent関数を置換して無効化
function setAjaxEvent(target) {
return false;
}
////////////////////////////////////////////////////////////////////////////////
// CSS
////////////////////////////////////////////////////////////////////////////////
GM_addStyle(`
div {
overflow-wrap: anywhere;
line-break: anywhere;
}
table.main {
width: 100%;
}
body > table > tbody > tr > td > table > tbody > tr > td > table > tbody > tr > td:nth-child(1) {
display: ${sideBarDisplay};
}
.vil_main {
margin: auto;
width: 552px;
}
.alllog_announce {
display: ${alllogAnnounceDisplay};
}
a.ellipsis,
span.ellipsis {
font-weight: bold;
}
a.ellipsis {
color: blue !important;
font-size: 1.2rem;
cursor: pointer;
}
span.ellipsis {
font-size: 0.8rem;
}
textarea {
font-size: ${textareaFontSize}rem !important;
}
.dialog {
text-align: left;
margin: 0 auto;
}
input[type=number] {
width: 3rem;
text-align: right;
}
button.primary, button.secondary, button.no_action_box {
border: none;
color: white;
text-align: center;
text-decoration: none;
font-size: 1rem;
cursor: pointer;
}
button.primary, button.secondary {
padding: 0.5rem 1rem;
display: inline-block;
margin: 0.5rem 0.5rem;
}
button.primary {
background-color: #4CAF50;
}
button.secondary {
background-color: #008CBA;
}
button.with_action_box {
border: none;
cursor: pointer;
background-color: white;
padding: 0;
}
button.no_action_box {
background-color: #000000;
padding: 0rem 1rem;
}
.container {
display: flex;
position: relative;
justify-content: space-between;
}
span.right {
margin-left: auto;
}
#scrollToBottomButton, #scrollToTopButton, #searchIcon {
padding: 5px;
cursor: pointer;
background: #ddd;
border: none;
border-radius: 5px;
transition: background 0.2s;
margin-bottom: 10px;
}
#scrollToTopButton:hover, #scrollToBottomButton:hover, #searchIcon:hover {
background: #bbb;
}
#scrollToTopButton {
order: 1;
}
#scrollToBottomButton {
order: 2;
}
#topButtonContainer, #bottomButtonContainer {
display: flex;
flex-direction: column;
position: fixed;
right: 20px;
z-index: 1000;
}
#topButtonContainer {
top: 20px;
}
#bottomButtonContainer {
bottom: 20px;
}
#searchBoxContainer {
order: 1;
align-items: center;
padding: 5px;
position: absolute;
right: 55px;
bottom: 0;
display: none;
background: #fff;
}
#searchBox, select#filter {
width: 200px;
height: 30px;
border-radius: 5px;
border: 1px solid #aaa;
}
input#searchBox, select#filter {
margin-right: 10px;
}
hr {
border: none;
border-top: 1px solid #e0e0e0;
margin: 5px 0;
}
`);
})();