/* * 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(); });