// ==UserScript==
// @name futaba auto reloader k
// @namespace https://github.com/akoya-tomo
// @description ふたば☆ちゃんねるで自動更新しちゃう(実況モードもあるよ!)
// @author akoya_tomo
// @match http://*.2chan.net/*/res/*
// @match https://*.2chan.net/*/res/*
// @require http://ajax.googleapis.com/ajax/libs/jquery/2.0.3/jquery.min.js
// @version 1.9.0.0
// @grant GM_addStyle
// @grant GM_xmlhttpRequest
// @grant GM_getValue
// @grant GM_setValue
// @license MIT
// @icon 
// @noframes
// ==/UserScript==
/* globals jQuery */
this.$ = this.jQuery = jQuery.noConflict(true);
(function ($) {
/*
* 設定
*/
// =====================================================
var USE_SOUDANE = true; //そうだねをハイライト表示する
var USE_CLEAR_BUTTON = true; //フォームにクリアボタンを表示する
var USE_TITLE_NAME = true; //新着レス数・スレ消滅状態をタブに表示する
var RELOAD_INTERVAL_NORMAL = 60000; //リロード間隔[ミリ秒](通常時)
var RELOAD_INTERVAL_LIVE = 5000; //リロード間隔[ミリ秒](実況モード時)
var LIVE_SCROLL_INTERVAL = 12; //実況モードスクロール間隔[ミリ秒]
var LIVE_SCROLL_SPEED = 3; //実況モードスクロール幅[px]
var LIVE_TOGGLE_KEY = "l"; //実況モードON・OFF切り替えキー(With Alt)
var SHOW_NORMAL_BUTTON = true; //通常モードボタンを表示する
var USE_NOTIFICATION_DEFAULT = false; // 新着レスの通知をデフォルトで有効にする
var USE_SAVE_MHT = false; // スレ消滅時にMHTで保存する(赤福のみ)
var USE_BOARD_NAME = false; //板名をタブに表示する
var NOTIFY_THREAD_NOT_FOUND = false; //スレの消滅を通知する(通知ボタンとは独立して動作)
var KEEP_THREAD_NOT_FOUND_MARK = false; //タイトルのスレ消滅表示をリセット操作で消さない(true = 消さない : false = 消す)
// =====================================================
var res = 0; //新着レス数
var timerNormal, timerLiveReload, timerLiveJump, timerLiveScroll, timerSoudane;
var url = location.href;
var script_name = "futaba_auto_reloader_k";
var isWindowActive = true; // タブのアクティブ状態
var isNotificationEnable = USE_NOTIFICATION_DEFAULT; // 通知の有効フラグ
var normal_flag = true; //通常モード有効フラグ
var live_flag = false; //実況モード有効フラグ
var isSoudane = true;
var serverName = document.domain.match(/^[^.]+/);
var boardName = $("#tit").text().match(/^[^@]+/);
var newres_index = $(".rsc").length; // リロード前の総レス数
var isThreadNotFound = false; // スレ消滅フラグ
init();
function init() {
if(isFileNotFound()) return;
set_title();
loadSettings();
setNormalReload();
soudane();
makeButton();
addCss();
setWindowFocusEvent();
observeInserted();
showFindNextThread();
setWheelEvent();
setKeyEvent();
}
// 設定ロード
function loadSettings() {
isSoudane = GM_getValue("soudane", true);
console.log(isSoudane);
}
//通常リロード開始
function setNormalReload() {
timerNormal = setInterval(rel, RELOAD_INTERVAL_NORMAL);
console.log(script_name + ": Start auto reloading @" + url);
}
//通常リロード停止
function clearNormalReload() {
clearInterval(timerNormal);
console.log(script_name + ": Stop auto reloading @" + url);
}
/*
* 404チェック
*/
function isFileNotFound() {
if(document.title == "404 File Not Found") {
return true;
}
else {
return false;
}
}
/*
* ボタン作成
*/
function makeButton() {
//通常モードボタン
var normalButton = $("<a>", {
id: "GM_FAR_relButton_normal",
class: "GM_FAR_relButton",
text: "[通常]",
title: (RELOAD_INTERVAL_NORMAL / 1000) + "秒毎のリロード",
css: {
cursor: "pointer",
"background-color": "#ea8",
display: "inline-block"
},
click: function() {
toggleNormalMode();
}
});
//実況モードボタン
var liveButton = $("<a>", {
id: "GM_FAR_relButton_live",
class: "GM_FAR_relButton",
text: "[実況(Alt+" + LIVE_TOGGLE_KEY.toUpperCase() + ")]",
title: (RELOAD_INTERVAL_LIVE / 1000) + "秒毎のリロード + スクロール",
css: {
cursor: "pointer",
display: "inline-block"
},
click: function() {
liveMode();
}
});
// 通知ボタン
var notificationButton = $("<a>", {
id: "GM_FAR_notificationButton",
text: "[通知]",
title: "新着レスのポップアップ通知",
css: {
cursor: "pointer",
display: "inline-block"
},
click: function() {
toggleNotification();
}
});
if (isNotificationEnable) {
notificationButton.css("background-color", "#a9d8ff");
}
// フォームクリアボタン
if ( USE_CLEAR_BUTTON ) {
var formClearButton = $("<div>", {
id: "formClearButton",
text: "[クリア]",
css: {
cursor: "pointer",
margin: "0 5px"
},
click: function() {
clearForm();
}
});
var comeTd = $(".ftdc b:contains('コメント')");
comeTd.after(formClearButton);
}
// そボタン
var soudaneButton = $("<a>", {
id: "GM_FAR_soudaneButton",
text: "[そ]",
title: "そうだねに色を付ける",
css: {
cursor: "pointer",
display: "inline-block"
},
click: function() {
toggleSoudane();
}
});
if (isSoudane) soudaneButton.css("background-color" , "#ea8");
var input = $("input[value$='信する']");
input.after(soudaneButton);
input.after(notificationButton);
input.after(liveButton);
if(SHOW_NORMAL_BUTTON){
input.after(normalButton);
}
}
/*
* 通常モード切り替え
*/
function toggleNormalMode() {
var normalButton = $("#GM_FAR_relButton_normal");
if(normal_flag) {
clearNormalReload();
normalButton.css("background" , "none");
normal_flag = false;
} else {
setNormalReload();
normalButton.css("background-color" , "#ea8");
normal_flag = true;
}
}
/*
* 実況モード
* 呼出ごとにON/OFFトグル
*/
function liveMode() {
var live_button = $("#GM_FAR_relButton_live");
if (!live_flag) {
setLiveReload();
setLiveJump();
setLiveScroll();
startspin();
live_flag = true;
live_button.css("backgroundColor", "#ffa5f0");
console.log(script_name + ": Start live mode @" + url);
} else {
clearLiveReload();
clearLiveJump();
clearLiveScroll();
stopspin();
live_flag = false;
live_button.css("background", "none");
console.log(script_name + ": Stop live mode @" + url);
}
}
function setLiveReload() {
clearInterval(timerLiveReload);
timerLiveReload = setInterval(rel, RELOAD_INTERVAL_LIVE);
}
function clearLiveReload() {
clearInterval(timerLiveReload);
}
function setLiveJump() {
clearInterval(timerLiveJump);
timerLiveJump = setInterval(liveJump, RELOAD_INTERVAL_LIVE);
}
function clearLiveJump() {
clearInterval(timerLiveJump);
}
function setLiveScroll() {
clearInterval(timerLiveScroll);
timerLiveScroll = setInterval(live_scroll, LIVE_SCROLL_INTERVAL);
}
function clearLiveScroll() {
clearInterval(timerLiveScroll);
}
//自動スクロール
function live_scroll() {
window.scrollBy( 0, LIVE_SCROLL_SPEED );
}
//新着ジャンプ
function liveJump() {
$('html, body').animate({scrollTop:document.body.scrollHeight},"fast");
}
function startspin() {
$("#akahuku_throp_menu_opener").css(
"animation", "spin 2s infinite steps(8)"
);
}
function stopspin() {
$("#akahuku_throp_menu_opener").css(
"animation", "none"
);
}
/**
* 新着レスをリセット
*/
function reset_titlename() {
res = 0;
if (!KEEP_THREAD_NOT_FOUND_MARK || !isAkahukuNotFound()) {
var title_char = title_name();
document.title = title_char;
}
}
/**
* リロードボタンをクリック
*/
function rel() {
// if(isAkahukuNotFound()) {
// return;
// }
var relbutton = $("#akahuku_reload_button").get(0) ? $("#akahuku_reload_button").get(0) : $("#contres a").get(0);
if (relbutton){
var e = document.createEvent("MouseEvents");
e.initEvent("click", false, true);
relbutton.dispatchEvent(e);
} else {
return;
}
setTimeout(function(){
soudane();
var res = $(".rsc");
if (!isWindowActive && isNotificationEnable) {
getNewResContent(res.length);
}
newres_index = res.length;
if(isAkahukuNotFound() || res.length >= 1000) {
//404 or 1000レス時
if (live_flag) {
liveMode();
}
changeTitleWhenExpired();
clearNormalReload();
if (!isWindowActive && NOTIFY_THREAD_NOT_FOUND) {
var popupText = res.length >= 1000 ? "上限1000レスに達しました\r\n自動更新を停止しました" : "スレは落ちています\r\n自動更新を停止しました";
showNotification(popupText);
}
if (USE_SAVE_MHT) {
saveMHT();
}
findNextThread();
console.log(script_name + ": Page not found, Stop auto reloading @" + url);
}
}, 1000);
}
/**
* MHTで保存
*/
function saveMHT() {
var saveMHTButton = $("#akahuku_throp_savemht_button").get(0);
if (saveMHTButton) {
var e = document.createEvent("MouseEvents");
e.initEvent("click", false, true);
saveMHTButton.dispatchEvent(e);
}
}
/*
* そうだねの数に応じてレスを着色
*/
function soudane() {
if ( !isSoudane ) return;
clearTimeout(timerSoudane);
timerSoudane = setTimeout(function() {
clearSoudane();
$("td > .sod").each(function(){
var sodnum = $(this).text().match(/\d+/);
if (sodnum){
var col = "rgb(180, 240," + (Math.round(10 * sodnum + 180)) + ")";
$(this).parent().css("background-color", col);
}
});
}, 100);
}
function clearSoudane() {
var coloredNode = $(".rtd[style]");
coloredNode.each(function() {
$(this).removeAttr("style");
});
}
function toggleSoudane() {
var soudaneButton = $("#GM_FAR_soudaneButton");
if (isSoudane) {
isSoudane = false;
clearSoudane();
soudaneButton.css("background" , "none");
} else {
isSoudane = true;
soudane();
soudaneButton.css("background-color" , "#ea8");
}
GM_setValue("soudane", isSoudane);
}
// リロードで挿入される要素を監視
function observeInserted() {
var target = $(".thre").length ?
$(".thre").get(0) :
$("html > body > form[action]:not([enctype])").get(0);
var observer = new MutationObserver(function(mutations) {
soudane();
mutations.forEach(function(mutation) {
var $nodes = $(mutation.addedNodes);
replaceNodeInserted($nodes);
});
});
observer.observe(target, { childList: true });
}
// 挿入されたレス
function replaceNodeInserted($nodes) {
var insertedRes = $nodes.find(".rtd");
if( insertedRes.length ) {
changetitle(insertedRes.length);
}
}
/*
* タブタイトルに新着レス数・スレ消滅状態を表示
*/
function changetitle(newResNum) {
if ( !USE_TITLE_NAME ) return;
var title_char = title_name();
if (isAkahukuNotFound()) return;
res = res + newResNum;
document.title = "(" + res + ")" + title_char;
}
function changeTitleWhenExpired() {
if (!isAkahukuNotFound()) return;
var title_char = title_name();
if(document.title.substr(0,1) !== "#"){
document.title = "#" + title_char;
}
}
// 新着レスの内容を取得
function getNewResContent(resnum) {
if (resnum > newres_index) {
var $newrestable = $(".rtd").slice(newres_index);
if ($newrestable.length) {
var restexts = [];
$newrestable.each(function() {
var texts = [];
$(this).find("blockquote").contents().each(function() {
if ($(this).text() !== "") {
texts.push($(this).text());
}
});
restexts.push(texts.join("\r\n"));
});
var popupText = restexts.join("\r\n===============\r\n");
showNotification(popupText);
}
}
}
/*
* ステータスからスレ消滅状態をチェック
*/
function isAkahukuNotFound() {
if (isThreadNotFound) return true;
// KOSHIAN
isThreadNotFound = checkStatus("#KOSHIAN_NOTIFY", /CODE:404/);
// KOSHIAN 返信フォーム拡張(改)が#KOSHIAN_NOTIFYを書き込むので、true以外ではreturnしない
if (isThreadNotFound) return true;
// 赤福
isThreadNotFound = checkStatus("#akahuku_reload_status", /(No Future)|((M|N)ot Found)/);
if (isThreadNotFound !== null) return isThreadNotFound;
// ふたば
isThreadNotFound = checkStatus("#contdisp", /^スレッドがありません$/);
return isThreadNotFound;
function checkStatus(status_id, regexp) {
var $status = $(status_id);
if ($status.length) {
return ($status.text().match(regexp));
}
return null;
}
}
function title_name() {
var title = document.title;
var title_num = title.match(/^(#|\(\d+\))/);
var title_num_length;
if(!title_num){
title_num_length = 0;
}
else {
title_num_length = title_num[0].length;
}
var act_title_name = title.substr(title_num_length);
return act_title_name;
}
function clearForm() {
$("#ftxa").val("");
}
function addCss() {
GM_addStyle(
"@keyframes spin {" +
" 0% { transform: rotate(0deg); }" +
" 100% { transform: rotate(359deg); }" +
"}"
);
}
/**
* 通知切り替え
*/
function toggleNotification() {
var notificationButton = $("#GM_FAR_notificationButton");
if(isNotificationEnable) {
notificationButton.css("background" , "none");
isNotificationEnable = false;
} else {
Notification.requestPermission(function(result) {
if (result == "denied") {
notificationButton.attr("title",
"通知はブラウザの設定でブロックされています\n" +
"ロケーションバー(URL)の左のアイコンをクリックして\n" +
"通知を「許可」に設定してください");
return;
} else if (result == "default") {
console.log("default");
return;
}
notificationButton.attr("title", "新着レスのポップアップ通知");
notificationButton.css("background-color" , "#a9d8ff");
isNotificationEnable = true;
});
}
}
// タブのアクティブ状態を取得
function setWindowFocusEvent() {
$(window).bind("focus", function() {
// タブアクティブ時
isWindowActive = true;
}).bind("blur", function() {
// タブ非アクティブ時
isWindowActive = false;
});
}
// 新着レスをポップアップでデスクトップ通知する
function showNotification(body) {
Notification.requestPermission();
//ファビコンからアイコン取得
var icon = $("head > link[rel*='icon']").attr("href");
if (!icon) {
icon = "/favicon.ico";
}
var instance = new Notification(
document.title, {
body: body,
icon: icon,
}
);
}
/**
* 次スレ候補検索ボタン表示
*/
function showFindNextThread() {
$("body").append(
$("<div>", {
id: "GM_FAR_next_thread_area",
class: "GM_FAR"
}).append(
$("<div>").append(
$("<a>", {
id: "GM_FAR_find_next_thread",
class: "GM_FAR_Button",
text: "[次スレ候補検索]",
css: {
cursor: "pointer",
"font-size": "9pt"
},
click: function() {
findNextThread();
}
}),
$("<span>", {
id: "GM_FAR_next_thread_search_result",
css: {
"display": "none",
"font-size": "9pt"
}
}).append(
$("<span>", {
text: "検索結果:",
}),
$("<span>", {
id: "GM_FAR_next_thread_search_result_count",
text: "0"
})
)
),
$("<ul>", {
"id": "GM_FAR_next_thread_found"
}).append(
$("<span>", {
id: "GM_FAR_next_thread_search_status",
text: "次スレ候補検索中...",
css: {
"display": "none"
}
})
)
)
)
}
/**
* 次スレ候補検索
*/
function findNextThread() {
var foundList = $("#GM_FAR_next_thread_found");
foundList.empty()
var statusMessage = $("#GM_FAR_next_thread_search_status")
statusMessage.show();
var dir = location.href.substr(0, location.href.lastIndexOf('/') - 3);
var threadTitle = $(".thre > blockquote").text();
var catalogURL = dir + "futaba.php?mode=cat&sort=1"
var resultCount = 0;
GM_xmlhttpRequest({
method: "GET",
url: catalogURL,
onload: function(res) {
statusMessage.hide();
var catalog = $($.parseHTML(res.response));
var cattable = catalog.filter("#cattable");
var td = cattable.find("td small");
td.each(function() {
var tdText = $(this).text()
if (tdText.substr(0, 3) != threadTitle.substr(0, 3) || tdText.substr(0, 3) == "キタ━") return;
resultCount++;
var foundThread = $(this).parent().find("a");
var foundThreadResCount = $(this).parent().find("font").text();
var href = foundThread.attr("href");
foundThread.attr("href", dir + href);
foundList.append(
$("<li>").append(
$(this),
$("<span>", {
text: foundThreadResCount + "レス",
css: {
"margin-left": "2em"
}
}),
foundThread
)
);
});
$("#GM_FAR_next_thread_search_result_count").text(resultCount);
$("#GM_FAR_next_thread_search_result").show();
}
});
}
/**
* マウスホイールイベント
*/
function setWheelEvent() {
var wheelNum = 3; // ホイールダウン回数
var timerWheel;
var n = 0;
window.addEventListener("wheel", (e) => {
var y = window.pageYOffset;
var ym = getPageBottom();
//新着レス数をリセット
if (e.deltaY > 0 && y >= ym) {
reset_titlename();
}
// スレ落ち後の手動次スレ検索
var res = $(".rsc");
if(isAkahukuNotFound() || res.length >= 1000) {
if (e.deltaY > 0 && y >= ym) {
n++;
if (n >= wheelNum) {
clearTimeout(timerWheel);
n = 0;
timerWheel = setTimeout(() => {
findNextThread();
}, 200);
}
}
}
// 実況モード時上スクロールで自動スクロールの一時停止
if (live_flag && e.deltaY < 0) {
clearLiveScroll();
clearLiveJump();
$("#GM_FAR_relButton_live").css("backgroundColor", "#c0ffa5");
} else if (live_flag && y >= ym) {
setLiveScroll();
setLiveJump();
$("#GM_FAR_relButton_live").css("backgroundColor", "#ffa5f0");
}
} ,false);
/* ページ末尾 */
function getPageBottom() {
var pageBottom = document.body.scrollHeight - window.innerHeight;
return pageBottom;
}
}
/**
* キーボードイベント
*/
function setKeyEvent() {
//実況モードトグルショートカットキー
window.addEventListener("keydown",function(e) {
if ( e.altKey && e.key == LIVE_TOGGLE_KEY ) {
liveMode();
}
}, false);
}
/**
* タイトルに板名を追加する
*/
function set_title() {
if ( USE_BOARD_NAME ) {
if (boardName == "二次元裏") {
boardName = serverName;
}
document.title = boardName + " " + document.title;
}
}
})(jQuery);