// ==UserScript==
// @name OPENREC.tv Screen Comment Scroller [Fix]
// @description OPENREC.tv のコメントをニコニコ風にスクロールさせます。
// @version 1.1
// @author Yos_sy
// @match https://www.openrec.tv/*
// @namespace http://tampermonkey.net/
// @icon 
// @license MIT
// @grant none
// ==/UserScript==
(function () {
"use strict";
// スクリプト名
const SCRIPTNAME = "ScreenCommentScroller";
// カスタマイズ設定のデフォルト
const defaultSettings = {
COLOR: "FFFFFF", // コメント色
OCOLOR: "000000", // コメント縁取り色
OWIDTH: 0.1, // コメント縁取りの太さ(比率)
OPACITY: 0.5, // コメントの不透明度
MAXLINES: 15, // コメント最大行数
LINEHEIGHT: 1.5, // コメント行高さ
DURATION: 5, // スクロール秒数
FPS: 60, // 秒間コマ数
};
// 設定の保存と読み込み
const loadSettings = () => {
const savedSettings = JSON.parse(localStorage.getItem(SCRIPTNAME) || "{}");
return { ...defaultSettings, ...savedSettings };
};
const saveSettings = (settings) => {
localStorage.setItem(SCRIPTNAME, JSON.stringify(settings));
};
let settings = loadSettings();
// 色の検証と変換
const validateColor = (color) => {
color = color.replace(/^#/, "");
return /^[0-9A-Fa-f]{6}$/.test(color) ? `#${color}` : null;
};
// 設定パネルの作成
const createSettingsPanel = () => {
const panel = document.createElement("div");
panel.id = `${SCRIPTNAME}_settingsPanel`;
panel.style.position = "fixed";
panel.style.top = "10px";
panel.style.right = "10px";
panel.style.backgroundColor = "#535069CC";
panel.style.color = "#ffffff";
panel.style.padding = "20px";
panel.style.borderRadius = "10px";
panel.style.boxShadow = "0 0 10px #00000080";
panel.style.zIndex = "9999";
panel.style.display = "none";
panel.style.fontFamily = "Arial, sans-serif";
panel.style.maxWidth = "300px";
// ボタンスタイルの作成
const setButtonStyle = (button) => {
button.style.display = "block";
button.style.width = "100%";
button.style.padding = "8px";
button.style.backgroundColor = "#FF4C11";
button.style.color = "white";
button.style.border = "none";
button.style.borderRadius = "3px";
button.style.cursor = "pointer";
};
// リセットボタンの設定
const resetButton = document.createElement("button");
resetButton.textContent = "デフォルト値にリセット";
setButtonStyle(resetButton);
resetButton.style.marginBottom = "15px";
resetButton.onclick = () => {
if (
confirm(
"設定をデフォルト値にリセットしますか?この操作は元に戻せません。"
)
) {
settings = { ...defaultSettings };
saveSettings(settings);
updateSettingsUI();
core.applySettings();
}
};
panel.appendChild(resetButton);
// 入力フィールドの設定
const createInput = (labelText, key, type = "text") => {
const container = document.createElement("div");
container.style.marginBottom = "10px";
const label = document.createElement("label");
label.textContent = labelText;
label.style.display = "block";
label.style.marginBottom = "5px";
const input = document.createElement("input");
input.id = `${SCRIPTNAME}_${key}`;
input.type = type;
input.value = settings[key];
input.style.width = "100%";
input.style.padding = "5px";
input.style.boxSizing = "border-box";
input.style.backgroundColor = "#FFFFFF1A";
input.style.border = "1px solid #FFFFFF4D";
input.style.borderRadius = "3px";
input.style.color = "#FFFFFF";
if (type === "number") {
input.step = "0.1";
}
input.onchange = (e) => {
let value = e.target.value;
if (type === "color") {
value = validateColor(value);
if (!value) {
alert("無効な色形式です。6桁の16進数を使用してください。");
e.target.value = `#${settings[key]}`;
return;
}
}
settings[key] =
type === "number" ? parseFloat(value) : value.replace(/^#/, "");
saveSettings(settings);
core.applySettings();
updateSettingsUI();
};
if (type === "color") {
input.value = `#${settings[key]}`;
}
container.appendChild(label);
container.appendChild(input);
panel.appendChild(container);
};
createInput("コメント色", "COLOR", "color");
createInput("コメント縁取り色", "OCOLOR", "color");
createInput("縁取りの太さ", "OWIDTH", "number");
createInput("不透明度", "OPACITY", "number");
createInput("最大行数", "MAXLINES", "number");
createInput("行高さ", "LINEHEIGHT", "number");
createInput("スクロール秒数", "DURATION", "number");
createInput("秒間コマ数", "FPS", "number");
// 閉じるボタンの設定
const closeButton = document.createElement("button");
closeButton.textContent = "閉じる";
setButtonStyle(closeButton);
closeButton.style.marginTop = "15px";
closeButton.onclick = () => {
panel.style.display = "none";
};
panel.appendChild(closeButton);
document.body.appendChild(panel);
};
// 設定UIの更新
const updateSettingsUI = () => {
Object.keys(settings).forEach((key) => {
const input = document.getElementById(`${SCRIPTNAME}_${key}`);
if (input) {
if (input.type === "color") {
input.value = `#${settings[key]}`;
} else {
input.value = settings[key];
}
}
});
};
// 設定パネルの表示切り替え
const toggleSettingsPanel = () => {
const panel = document.getElementById(`${SCRIPTNAME}_settingsPanel`);
panel.style.display = panel.style.display === "none" ? "block" : "none";
};
// キーボードショートカットの設定
document.addEventListener("keydown", (e) => {
if (e.ctrlKey && e.altKey && e.key === "o") {
toggleSettingsPanel();
}
});
// キャッシュ
const cache = {
screen: null,
board: null,
play: null,
};
const getElement = (key, selector) => {
if (!cache[key]) {
cache[key] = document.querySelector(selector);
}
return cache[key];
};
// サイト定義
const site = {
getScreen: () => getElement("screen", ".video-player-wrapper"),
getBoard: () => getElement("board", ".chat-list-content"),
getComments: (node) => node.querySelectorAll(".chat-content"),
getPlay: () =>
getElement(
"play",
'[class^="MovieToolbar"] [class^="TextLabel__Wrapper"]'
),
isPlaying: () => true, // 常に再生中と仮定
};
// 処理本体
let screen,
board,
play,
canvas,
context,
lines = [],
fontsize;
const core = {
// 初期化
initialize: () => {
console.log(SCRIPTNAME, "initialize...");
screen = site.getScreen();
board = site.getBoard();
play = site.getPlay();
if (!screen || !board || !play) {
window.setTimeout(core.initialize, 1000);
return;
}
canvas = document.createElement("canvas");
canvas.id = SCRIPTNAME;
screen.appendChild(canvas);
context = canvas.getContext("2d");
core.applySettings();
core.listenComments();
core.scrollComments();
createSettingsPanel();
},
// 設定の適用
applySettings: () => {
core.modify();
core.addStyle();
},
// キャンバスのサイズ調整とフォント設定
modify: () => {
const newWidth = screen.offsetWidth;
const newHeight = screen.offsetHeight;
canvas.width = newWidth;
canvas.height = newHeight;
fontsize = newHeight / settings.MAXLINES / settings.LINEHEIGHT;
context.font = `bold ${fontsize}px sans-serif`;
context.fillStyle = validateColor(settings.COLOR) || "#FFFFFF";
context.strokeStyle = validateColor(settings.OCOLOR) || "#000000";
context.lineWidth = fontsize * settings.OWIDTH;
},
// スタイル追加
addStyle: () => {
let canvas = document.querySelector(`canvas#${SCRIPTNAME}`);
if (!canvas) {
canvas = document.createElement("canvas");
canvas.id = SCRIPTNAME;
document.body.appendChild(canvas);
}
// キャンバスのスタイル追加
canvas.style.pointerEvents = "none";
canvas.style.position = "absolute";
canvas.style.top = "0";
canvas.style.left = "0";
canvas.style.width = "100%";
canvas.style.height = "100%";
canvas.style.opacity = `${settings.OPACITY}`;
canvas.style.zIndex = "calc(infinity)";
},
// コメントの監視
listenComments: () => {
board.addEventListener("DOMNodeInserted", (e) => {
const comments = site.getComments(e.target);
if (!comments || !comments.length) return;
for (let comment of comments) {
core.attachComment(comment);
}
});
},
// コメントの追加
attachComment: (comment) => {
const text = comment.textContent;
const width = context.measureText(text).width;
const life = settings.DURATION * settings.FPS;
const left = canvas.width;
const delta = (canvas.width + width) / life;
for (let i = 0; i < settings.MAXLINES; i++) {
const line = lines[i] || [];
if (
line.length === 0 ||
line[line.length - 1].left < canvas.width - width
) {
lines[i] = line;
line.push({
text,
width,
life,
left,
delta,
top: (canvas.height / settings.MAXLINES) * i + fontsize,
});
break;
}
}
},
// コメントのスクロール
scrollComments: () => {
let lastTime = 0;
const animate = (currentTime) => {
if (site.isPlaying()) {
const deltaTime = (currentTime - lastTime) / 1000;
lastTime = currentTime;
context.clearRect(0, 0, canvas.width, canvas.height);
lines.forEach((line, i) => {
lines[i] = line.filter((comment) => {
comment.life -= deltaTime * settings.FPS;
comment.left -= comment.delta * deltaTime * settings.FPS;
if (comment.left + comment.width > 0) {
context.strokeText(comment.text, comment.left, comment.top);
context.fillText(comment.text, comment.left, comment.top);
return true;
}
return false;
});
});
}
requestAnimationFrame(animate);
};
requestAnimationFrame(animate);
},
};
core.initialize();
})();