mirror of
https://github.com/vichan-devel/vichan.git
synced 2024-12-18 10:25:55 +01:00
455 lines
14 KiB
JavaScript
455 lines
14 KiB
JavaScript
/*
|
|
* watch.js - board watch, thread watch and board pinning
|
|
* https://github.com/vichan-devel/Tinyboard/blob/master/js/watch.js
|
|
*
|
|
* Released under the MIT license
|
|
* Copyright (c) 2014 Marcin Łabanowski <marcin@6irc.net>
|
|
*
|
|
* Usage:
|
|
* $config['api']['enabled'] = true;
|
|
* $config['additional_javascript'][] = 'js/jquery.min.js';
|
|
* $config['additional_javascript'][] = 'js/mobile-style.js';
|
|
* //$config['additional_javascript'][] = 'js/titlebar-notifications.js';
|
|
* //$config['additional_javascript'][] = 'js/auto-reload.js';
|
|
* //$config['additional_javascript'][] = 'js/hide-threads.js';
|
|
* //$config['additional_javascript'][] = 'js/compact-boardlist.js';
|
|
* $config['additional_javascript'][] = 'js/watch.js';
|
|
*
|
|
*/
|
|
|
|
$(function(){
|
|
// migrate from old name
|
|
if (typeof localStorage.watch == "string") {
|
|
localStorage.watch_js = localStorage.watch;
|
|
delete localStorage.watch;
|
|
}
|
|
|
|
var window_active = true;
|
|
$(window).focus(function() {
|
|
window_active = true;
|
|
$(window).trigger('scroll');
|
|
});
|
|
$(window).blur(function() {
|
|
window_active = false;
|
|
});
|
|
|
|
var status = {};
|
|
|
|
time_loaded = Date.now();
|
|
|
|
var updating_suspended = false;
|
|
|
|
var storage = function() {
|
|
var storage = JSON.parse(localStorage.watch_js !== undefined ? localStorage.watch_js : "{}");
|
|
delete storage.undefined; // fix for some bug
|
|
return storage;
|
|
};
|
|
|
|
var storage_save = function(s) {
|
|
localStorage.watch_js = JSON.stringify(s);
|
|
};
|
|
|
|
var osize = function(o) {
|
|
var size = 0;
|
|
for (var key in o) {
|
|
if (o.hasOwnProperty(key)) size++;
|
|
}
|
|
return size;
|
|
};
|
|
|
|
var is_pinned = function(boardconfig) {
|
|
return boardconfig.pinned || boardconfig.watched || (boardconfig.threads ? osize(boardconfig.threads) : false);
|
|
};
|
|
var is_boardwatched = function(boardconfig) {
|
|
return boardconfig.watched;
|
|
};
|
|
var is_threadwatched = function(boardconfig, thread) {
|
|
return boardconfig && boardconfig.threads && boardconfig.threads[thread];
|
|
};
|
|
var toggle_pinned = function(board) {
|
|
var st = storage();
|
|
var bc = st[board] || {};
|
|
if (is_pinned(bc)) {
|
|
bc.pinned = false;
|
|
bc.watched = false;
|
|
bc.threads = {};
|
|
}
|
|
else {
|
|
bc.pinned = true;
|
|
}
|
|
st[board] = bc;
|
|
storage_save(st);
|
|
return bc.pinned;
|
|
};
|
|
var toggle_boardwatched = function(board) {
|
|
var st = storage();
|
|
var bc = st[board] || {};
|
|
bc.watched = !is_boardwatched(bc) && Date.now();
|
|
st[board] = bc;
|
|
storage_save(st);
|
|
return bc.watched;
|
|
};
|
|
var toggle_threadwatched = function(board, thread) {
|
|
var st = storage();
|
|
var bc = st[board] || {};
|
|
if (is_threadwatched(bc, thread)) {
|
|
delete bc.threads[thread];
|
|
}
|
|
else {
|
|
bc.threads = bc.threads || {};
|
|
bc.threads[thread] = Date.now();
|
|
|
|
bc.slugs = bc.slugs || {};
|
|
bc.slugs[thread] = document.location.pathname + document.location.search;
|
|
}
|
|
st[board] = bc;
|
|
storage_save(st);
|
|
return is_threadwatched(bc, thread);
|
|
};
|
|
var construct_watchlist_for = function(board, variant) {
|
|
var list = $("<div class='boardlist top cb-menu watch-menu'></div>");
|
|
list.attr("data-board", board);
|
|
|
|
if (storage()[board] && storage()[board].threads)
|
|
for (var tid in storage()[board].threads) {
|
|
var newposts = "(0)";
|
|
if (status && status[board] && status[board].threads && status[board].threads[tid]) {
|
|
if (status[board].threads[tid] == -404) {
|
|
newposts = "<i class='fa fa-ban-circle'></i>";
|
|
}
|
|
else {
|
|
newposts = "("+status[board].threads[tid]+")";
|
|
}
|
|
}
|
|
|
|
var tag;
|
|
if (variant == 'desktop') {
|
|
tag = $("<a href='"+((storage()[board].slugs && storage()[board].slugs[tid]) || (modRoot+board+"/res/"+tid+".html"))+"'><span>#"+tid+"</span><span class='cb-uri watch-remove'>"+newposts+"</span>");
|
|
tag.find(".watch-remove").mouseenter(function() {
|
|
this.oldval = $(this).html();
|
|
$(this).css("min-width", $(this).width());
|
|
$(this).html("<i class='fa fa-minus'></i>");
|
|
})
|
|
.mouseleave(function() {
|
|
$(this).html(this.oldval);
|
|
})
|
|
}
|
|
else if (variant == 'mobile') {
|
|
tag = $("<a href='"+((storage()[board].slugs && storage()[board].slugs[tid]) || (modRoot+board+"/res/"+tid+".html"))+"'><span>#"+tid+"</span><span class='cb-uri'>"+newposts+"</span>"
|
|
+"<span class='cb-uri watch-remove'><i class='fa fa-minus'></i></span>");
|
|
}
|
|
|
|
tag.attr('data-thread', tid)
|
|
.addClass("cb-menuitem")
|
|
.appendTo(list)
|
|
.find(".watch-remove")
|
|
.click(function() {
|
|
var b = $(this).parent().parent().attr("data-board");
|
|
var t = $(this).parent().attr("data-thread");
|
|
toggle_threadwatched(b, t);
|
|
$(this).parent().parent().parent().mouseleave();
|
|
$(this).parent().remove();
|
|
return false;
|
|
});
|
|
}
|
|
return list;
|
|
};
|
|
|
|
var update_pinned = function() {
|
|
if (updating_suspended) return;
|
|
|
|
if (typeof update_title != "undefined") update_title();
|
|
|
|
var bl = $('.boardlist').first();
|
|
$('#watch-pinned, .watch-menu').remove();
|
|
var pinned = $('<div id="watch-pinned"></div>').appendTo(bl);
|
|
|
|
if (device_type == "desktop")
|
|
bl.off().on("mouseenter", function() {
|
|
updating_suspended = true;
|
|
}).on("mouseleave", function() {
|
|
updating_suspended = false;
|
|
});
|
|
|
|
var st = storage();
|
|
for (var i in st) {
|
|
if (is_pinned(st[i])) {
|
|
var link;
|
|
if (bl.find('[href*="'+modRoot+i+'/index.html"]:not(.cb-menuitem)').length) link = bl.find('[href*="'+modRoot+i+'/"]').first();
|
|
|
|
else link = $('<a href="'+modRoot+i+'/" class="cb-item cb-cat">/'+i+'/</a>').appendTo(pinned);
|
|
|
|
if (link[0].origtitle === undefined) {
|
|
link[0].origtitle = link.html();
|
|
}
|
|
else {
|
|
link.html(link[0].origtitle);
|
|
}
|
|
|
|
if (st[i].watched) {
|
|
link.css("font-weight", "bold");
|
|
if (status && status[i] && status[i].new_threads) {
|
|
link.html(link.html() + " (" + status[i].new_threads + ")");
|
|
}
|
|
}
|
|
else if (st[i].threads && osize(st[i].threads)) {
|
|
link.css("font-style", "italic");
|
|
|
|
link.attr("data-board", i);
|
|
|
|
if (status && status[i] && status[i].threads) {
|
|
var new_posts = 0;
|
|
for (var tid in status[i].threads) {
|
|
if (status[i].threads[tid] > 0) {
|
|
new_posts += status[i].threads[tid];
|
|
}
|
|
}
|
|
if (new_posts > 0) {
|
|
link.html(link.html() + " (" + new_posts + ")");
|
|
}
|
|
}
|
|
|
|
if (device_type == "desktop")
|
|
link.off().mouseenter(function() {
|
|
$('.cb-menu').remove();
|
|
|
|
var board = $(this).attr("data-board");
|
|
|
|
var wl = construct_watchlist_for(board, "desktop").appendTo($(this))
|
|
.css("top", $(this).position().top
|
|
+ ($(this).css('padding-top').replace('px', '')|0)
|
|
+ ($(this).css('padding-bottom').replace('px', '')|0)
|
|
+ $(this).height())
|
|
.css("left", $(this).position().left)
|
|
.css("right", "auto")
|
|
.css("font-style", "normal");
|
|
|
|
if (typeof init_hover != "undefined")
|
|
wl.find("a.cb-menuitem").each(init_hover);
|
|
|
|
}).mouseleave(function() {
|
|
$('.boardlist .cb-menu').remove();
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
if (device_type == "mobile" && (active_page == 'thread' || active_page == 'index')) {
|
|
var board = $('form[name="post"] input[name="board"]').val();
|
|
|
|
var where = $('div[style="text-align:right"]').first();
|
|
$('.watch-menu').remove();
|
|
construct_watchlist_for(board, "mobile").css("float", "left").insertBefore(where);
|
|
}
|
|
};
|
|
var fetch_jsons = function() {
|
|
if (window_active) check_scroll();
|
|
|
|
var st = storage();
|
|
|
|
var sched = 0;
|
|
var sched_diff = 2000;
|
|
|
|
for (var i in st) {
|
|
if (st[i].watched) {
|
|
(function(i) {
|
|
setTimeout(function() {
|
|
var r = $.getJSON(configRoot+i+"/threads.json", function(j, x, r) {
|
|
handle_board_json(r.board, j);
|
|
});
|
|
r.board = i;
|
|
}, sched);
|
|
sched += sched_diff;
|
|
})(i);
|
|
}
|
|
else if (st[i].threads) {
|
|
for (var j in st[i].threads) {
|
|
(function(i,j) {
|
|
setTimeout(function() {
|
|
var r = $.getJSON(configRoot+i+"/res/"+j+".json", function(k, x, r) {
|
|
handle_thread_json(r.board, r.thread, k);
|
|
}).error(function(r) {
|
|
if(r.status == 404) handle_thread_404(r.board, r.thread);
|
|
});
|
|
|
|
r.board = i;
|
|
r.thread = j;
|
|
}, sched);
|
|
})(i,j);
|
|
sched += sched_diff;
|
|
}
|
|
}
|
|
}
|
|
|
|
setTimeout(fetch_jsons, sched + sched_diff);
|
|
};
|
|
|
|
var handle_board_json = function(board, json) {
|
|
var last_thread;
|
|
|
|
var new_threads = 0;
|
|
|
|
var hidden_data = {};
|
|
if (localStorage.hiddenthreads) {
|
|
hidden_data = JSON.parse(localStorage.hiddenthreads);
|
|
}
|
|
|
|
for (var i in json) {
|
|
for (var j in json[i].threads) {
|
|
var thread = json[i].threads[j];
|
|
|
|
if (hidden_data[board]) { // hide threads integration
|
|
var cont = false;
|
|
for (var k in hidden_data[board]) {
|
|
if (parseInt(k) == thread.no) {
|
|
cont = true;
|
|
break;
|
|
}
|
|
}
|
|
if (cont) continue;
|
|
}
|
|
|
|
if (thread.last_modified > storage()[board].watched / 1000) {
|
|
last_thread = thread.no;
|
|
|
|
new_threads++;
|
|
}
|
|
}
|
|
}
|
|
|
|
status = status || {};
|
|
status[board] = status[board] || {};
|
|
if (status[board].last_thread != last_thread || status[board].new_threads != new_threads) {
|
|
status[board].last_thread = last_thread;
|
|
status[board].new_threads = new_threads;
|
|
update_pinned();
|
|
}
|
|
};
|
|
var handle_thread_json = function(board, threadid, json) {
|
|
var new_posts = 0;
|
|
for (var i in json.posts) {
|
|
var post = json.posts[i];
|
|
|
|
if (post.time > storage()[board].threads[threadid] / 1000) {
|
|
new_posts++;
|
|
}
|
|
}
|
|
|
|
status = status || {};
|
|
status[board] = status[board] || {};
|
|
status[board].threads = status[board].threads || {};
|
|
|
|
if (status[board].threads[threadid] != new_posts) {
|
|
status[board].threads[threadid] = new_posts;
|
|
update_pinned();
|
|
}
|
|
};
|
|
var handle_thread_404 = function(board, threadid) {
|
|
status = status || {};
|
|
status[board] = status[board] || {};
|
|
status[board].threads = status[board].threads || {};
|
|
if (status[board].threads[threadid] != -404) {
|
|
status[board].threads[threadid] = -404; //notify 404
|
|
update_pinned();
|
|
}
|
|
};
|
|
|
|
if (active_page == "thread") {
|
|
var board = $('form[name="post"] input[name="board"]').val();
|
|
var thread = $('form[name="post"] input[name="thread"]').val();
|
|
|
|
var boardconfig = storage()[board] || {};
|
|
|
|
$('hr:first').before('<div id="watch-thread" style="text-align:right"><a class="unimportant" href="javascript:void(0)">-</a></div>');
|
|
$('#watch-thread a').html(is_threadwatched(boardconfig, thread) ? _("Stop watching this thread") : _("Watch this thread")).click(function() {
|
|
$(this).html(toggle_threadwatched(board, thread) ? _("Stop watching this thread") : _("Watch this thread"));
|
|
update_pinned();
|
|
});
|
|
}
|
|
if (active_page == "index") {
|
|
var board = $('form[name="post"] input[name="board"]').val();
|
|
|
|
var boardconfig = storage()[board] || {};
|
|
|
|
$('hr:first').before('<div id="watch-pin" style="text-align:right"><a class="unimportant" href="javascript:void(0)">-</a></div>');
|
|
$('#watch-pin a').html(is_pinned(boardconfig) ? _("Unpin this board") : _("Pin this board")).click(function() {
|
|
$(this).html(toggle_pinned(board) ? _("Unpin this board") : _("Pin this board"));
|
|
$('#watch-board a').html(is_boardwatched(boardconfig) ? _("Stop watching this board") : _("Watch this board"));
|
|
update_pinned();
|
|
});
|
|
|
|
$('hr:first').before('<div id="watch-board" style="text-align:right"><a class="unimportant" href="javascript:void(0)">-</a></div>');
|
|
$('#watch-board a').html(is_boardwatched(boardconfig) ? _("Stop watching this board") : _("Watch this board")).click(function() {
|
|
$(this).html(toggle_boardwatched(board) ? _("Stop watching this board") : _("Watch this board"));
|
|
$('#watch-pin a').html(is_pinned(boardconfig) ? _("Unpin this board") : _("Pin this board"));
|
|
update_pinned();
|
|
});
|
|
|
|
}
|
|
|
|
var check_post = function(frame, post) {
|
|
return post.length && $(frame).scrollTop() + $(frame).height() >=
|
|
post.position().top + post.height();
|
|
}
|
|
|
|
var check_scroll = function() {
|
|
if (!status) return;
|
|
var refresh = false;
|
|
for(var bid in status) {
|
|
if (((status[bid].new_threads && (active_page == "ukko" || active_page == "index")) || status[bid].new_threads == 1)
|
|
&& check_post(this, $('[data-board="'+bid+'"]#thread_'+status[bid].last_thread))) {
|
|
var st = storage()
|
|
st[bid].watched = time_loaded;
|
|
storage_save(st);
|
|
refresh = true;
|
|
}
|
|
if (!status[bid].threads) continue;
|
|
|
|
for (var tid in status[bid].threads) {
|
|
if(status[bid].threads[tid] && check_post(this, $('[data-board="'+bid+'"]#thread_'+tid))) {
|
|
var st = storage();
|
|
st[bid].threads[tid] = time_loaded;
|
|
storage_save(st);
|
|
refresh = true;
|
|
}
|
|
}
|
|
}
|
|
return refresh;
|
|
};
|
|
|
|
$(window).scroll(function() {
|
|
var refresh = check_scroll();
|
|
if (refresh) {
|
|
//fetch_jsons();
|
|
refresh = false;
|
|
}
|
|
});
|
|
|
|
if (typeof add_title_collector != "undefined")
|
|
add_title_collector(function() {
|
|
if (!status) return 0;
|
|
var sum = 0;
|
|
for (var bid in status) {
|
|
if (status[bid].new_threads) {
|
|
sum += status[bid].new_threads;
|
|
if (!status[bid].threads) continue;
|
|
for (var tid in status[bid].threads) {
|
|
if (status[bid].threads[tid] > 0) {
|
|
if (auto_reload_enabled && active_page == "thread") {
|
|
var board = $('form[name="post"] input[name="board"]').val();
|
|
var thread = $('form[name="post"] input[name="thread"]').val();
|
|
|
|
if (board == bid && thread == tid) continue;
|
|
}
|
|
sum += status[bid].threads[tid];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return sum;
|
|
});
|
|
|
|
update_pinned();
|
|
fetch_jsons();
|
|
});
|