You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
557 lines
13 KiB
JavaScript
557 lines
13 KiB
JavaScript
let _infscroll_disable = 0;
|
|
let _infscroll_request_sent = 0;
|
|
|
|
let _search_query = false;
|
|
let _viewfeed_last = 0;
|
|
let _viewfeed_timeout = false;
|
|
|
|
let counters_last_request = 0;
|
|
let _counters_prev = [];
|
|
|
|
function resetCounterCache() {
|
|
_counters_prev = [];
|
|
}
|
|
|
|
function loadMoreHeadlines() {
|
|
console.log("loadMoreHeadlines");
|
|
|
|
let offset = 0;
|
|
|
|
const view_mode = document.forms["main_toolbar_form"].view_mode.value;
|
|
const unread_in_buffer = $$("#headlines-frame > div[id*=RROW][class*=Unread]").length;
|
|
const num_all = $$("#headlines-frame > div[id*=RROW]").length;
|
|
const num_unread = getFeedUnread(getActiveFeedId(), activeFeedIsCat());
|
|
|
|
// TODO implement marked & published
|
|
|
|
if (view_mode == "marked") {
|
|
console.warn("loadMoreHeadlines: marked is not implemented, falling back.");
|
|
offset = num_all;
|
|
} else if (view_mode == "published") {
|
|
console.warn("loadMoreHeadlines: published is not implemented, falling back.");
|
|
offset = num_all;
|
|
} else if (view_mode == "unread") {
|
|
offset = unread_in_buffer;
|
|
} else if (_search_query) {
|
|
offset = num_all;
|
|
} else if (view_mode == "adaptive" && !(getActiveFeedId() == -1 && !activeFeedIsCat())) {
|
|
// ^ starred feed shows both unread & read articles in adaptive mode
|
|
offset = num_unread > 0 ? unread_in_buffer : num_all;
|
|
} else {
|
|
offset = num_all;
|
|
}
|
|
|
|
console.log("offset: " + offset);
|
|
|
|
viewfeed({feed: getActiveFeedId(), is_cat: activeFeedIsCat(), offset: offset, infscroll_req: true});
|
|
|
|
}
|
|
|
|
function cleanup_memory(root) {
|
|
const dijits = dojo.query("[widgetid]", dijit.byId(root).domNode).map(dijit.byNode);
|
|
|
|
dijits.each(function (d) {
|
|
dojo.destroy(d.domNode);
|
|
});
|
|
|
|
$$("#" + root + " *").each(function (i) {
|
|
i.parentNode ? i.parentNode.removeChild(i) : true;
|
|
});
|
|
}
|
|
|
|
function viewfeed(params) {
|
|
const feed = params.feed;
|
|
let is_cat = !!params.is_cat || false;
|
|
let offset = params.offset || 0;
|
|
let background = params.background || false;
|
|
let infscroll_req = params.infscroll_req || false;
|
|
const can_wait = params.can_wait;
|
|
const viewfeed_debug = params.viewfeed_debug;
|
|
const method = params.method;
|
|
|
|
if (infscroll_req == undefined) infscroll_req = false;
|
|
|
|
last_requested_article = 0;
|
|
|
|
if (feed != getActiveFeedId() || activeFeedIsCat() != is_cat) {
|
|
if (!background && _search_query) _search_query = false;
|
|
}
|
|
|
|
if (!background) {
|
|
_viewfeed_last = get_timestamp();
|
|
|
|
if (getActiveFeedId() != feed || !infscroll_req) {
|
|
setActiveArticleId(0);
|
|
_infscroll_disable = 0;
|
|
|
|
cleanup_memory("headlines-frame");
|
|
_headlines_scroll_offset = 0;
|
|
}
|
|
|
|
if (infscroll_req) {
|
|
const timestamp = get_timestamp();
|
|
|
|
if (_infscroll_request_sent && _infscroll_request_sent + 30 > timestamp) {
|
|
//console.log("infscroll request in progress, aborting");
|
|
return;
|
|
}
|
|
|
|
_infscroll_request_sent = timestamp;
|
|
}
|
|
}
|
|
|
|
Form.enable("main_toolbar_form");
|
|
|
|
let query = Object.assign({op: "feeds", method: "view", feed: feed},
|
|
dojo.formToObject("main_toolbar_form"));
|
|
|
|
if (method) query.m = method;
|
|
|
|
if (offset > 0) {
|
|
if (current_first_id) {
|
|
query.fid = current_first_id;
|
|
}
|
|
}
|
|
|
|
if (!background) {
|
|
if (_search_query) {
|
|
query = Object.assign(query, _search_query);
|
|
}
|
|
|
|
if (offset != 0) {
|
|
query.skip = offset;
|
|
|
|
// to prevent duplicate feed titles when showing grouped vfeeds
|
|
if (vgroup_last_feed) {
|
|
query.vgrlf = vgroup_last_feed;
|
|
}
|
|
} else if (!is_cat && feed == getActiveFeedId() && !params.method) {
|
|
query.m = "ForceUpdate";
|
|
}
|
|
|
|
Form.enable("main_toolbar_form");
|
|
|
|
if (!setFeedExpandoIcon(feed, is_cat,
|
|
(is_cat) ? 'images/indicator_tiny.gif' : 'images/indicator_white.gif'))
|
|
notify_progress("Loading, please wait...", true);
|
|
}
|
|
|
|
query.cat = is_cat;
|
|
|
|
if (can_wait && _viewfeed_timeout) {
|
|
setFeedExpandoIcon(getActiveFeedId(), activeFeedIsCat(), 'images/blank_icon.gif');
|
|
clearTimeout(_viewfeed_timeout);
|
|
}
|
|
|
|
setActiveFeedId(feed, is_cat);
|
|
|
|
if (viewfeed_debug) {
|
|
window.open("backend.php?" +
|
|
dojo.objectToQuery(
|
|
Object.assign({debug: 1, csrf_token: getInitParam("csrf_token")}, query)
|
|
));
|
|
}
|
|
|
|
const timeout_ms = can_wait ? 250 : 0;
|
|
_viewfeed_timeout = setTimeout(() => {
|
|
|
|
xhrPost("backend.php", query, (transport) => {
|
|
try {
|
|
setFeedExpandoIcon(feed, is_cat, 'images/blank_icon.gif');
|
|
headlines_callback2(transport, offset, background, infscroll_req);
|
|
PluginHost.run(PluginHost.HOOK_FEED_LOADED, [feed, is_cat]);
|
|
} catch (e) {
|
|
exception_error(e);
|
|
}
|
|
});
|
|
|
|
}, timeout_ms); // Wait 250ms
|
|
|
|
}
|
|
|
|
function feedlist_init() {
|
|
console.log("in feedlist init");
|
|
|
|
loading_set_progress(50);
|
|
|
|
document.onkeydown = hotkey_handler;
|
|
setInterval(hotkey_prefix_timeout, 5*1000);
|
|
setInterval(catchupBatchedArticles, 3*1000);
|
|
|
|
if (!getActiveFeedId()) {
|
|
viewfeed({feed: -3});
|
|
} else {
|
|
viewfeed({feed: getActiveFeedId(), is_cat: activeFeedIsCat()});
|
|
}
|
|
|
|
hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
|
|
|
|
if (getInitParam("is_default_pw")) {
|
|
console.warn("user password is at default value");
|
|
|
|
const dialog = new dijit.Dialog({
|
|
title: __("Your password is at default value"),
|
|
href: "backend.php?op=dlg&method=defaultpasswordwarning",
|
|
id: 'infoBox',
|
|
style: "width: 600px",
|
|
onCancel: function() {
|
|
return true;
|
|
},
|
|
onExecute: function() {
|
|
return true;
|
|
},
|
|
onClose: function() {
|
|
return true;
|
|
}
|
|
});
|
|
|
|
dialog.show();
|
|
}
|
|
|
|
// bw_limit disables timeout() so we request initial counters separately
|
|
if (getInitParam("bw_limit") == "1") {
|
|
request_counters(true);
|
|
} else {
|
|
setTimeout(timeout, 250);
|
|
}
|
|
}
|
|
|
|
|
|
function request_counters(force) {
|
|
const date = new Date();
|
|
const timestamp = Math.round(date.getTime() / 1000);
|
|
|
|
if (force || timestamp - counters_last_request > 5) {
|
|
console.log("scheduling request of counters...");
|
|
|
|
counters_last_request = timestamp;
|
|
|
|
let query = {op: "rpc", method: "getAllCounters", seq: next_seq()};
|
|
|
|
if (!force)
|
|
query.last_article_id = getInitParam("last_article_id");
|
|
|
|
xhrPost("backend.php", query, (transport) => {
|
|
handle_rpc_json(transport);
|
|
});
|
|
|
|
} else {
|
|
console.log("request_counters: rate limit reached: " + (timestamp - counters_last_request));
|
|
}
|
|
}
|
|
|
|
// NOTE: this implementation is incomplete
|
|
// for general objects but good enough for counters
|
|
// http://adripofjavascript.com/blog/drips/object-equality-in-javascript.html
|
|
function counter_is_equal(a, b) {
|
|
// Create arrays of property names
|
|
const aProps = Object.getOwnPropertyNames(a);
|
|
const bProps = Object.getOwnPropertyNames(b);
|
|
|
|
// If number of properties is different,
|
|
// objects are not equivalent
|
|
if (aProps.length != bProps.length) {
|
|
return false;
|
|
}
|
|
|
|
for (let i = 0; i < aProps.length; i++) {
|
|
const propName = aProps[i];
|
|
|
|
// If values of same property are not equal,
|
|
// objects are not equivalent
|
|
if (a[propName] !== b[propName]) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// If we made it this far, objects
|
|
// are considered equivalent
|
|
return true;
|
|
}
|
|
|
|
|
|
function parse_counters(elems) {
|
|
for (let l = 0; l < elems.length; l++) {
|
|
|
|
if (_counters_prev[l] && counter_is_equal(elems[l], _counters_prev[l])) {
|
|
continue;
|
|
}
|
|
|
|
const id = elems[l].id;
|
|
const kind = elems[l].kind;
|
|
const ctr = parseInt(elems[l].counter);
|
|
const error = elems[l].error;
|
|
const has_img = elems[l].has_img;
|
|
const updated = elems[l].updated;
|
|
const auxctr = parseInt(elems[l].auxcounter);
|
|
|
|
if (id == "global-unread") {
|
|
global_unread = ctr;
|
|
updateTitle();
|
|
continue;
|
|
}
|
|
|
|
if (id == "subscribed-feeds") {
|
|
/* feeds_found = ctr; */
|
|
continue;
|
|
}
|
|
|
|
/*if (getFeedUnread(id, (kind == "cat")) != ctr ||
|
|
(kind == "cat")) {
|
|
}*/
|
|
|
|
setFeedUnread(id, (kind == "cat"), ctr);
|
|
setFeedValue(id, (kind == "cat"), 'auxcounter', auxctr);
|
|
|
|
if (kind != "cat") {
|
|
setFeedValue(id, false, 'error', error);
|
|
setFeedValue(id, false, 'updated', updated);
|
|
|
|
if (id > 0) {
|
|
if (has_img) {
|
|
setFeedIcon(id, false,
|
|
getInitParam("icons_url") + "/" + id + ".ico?" + has_img);
|
|
} else {
|
|
setFeedIcon(id, false, 'images/blank_icon.gif');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
|
|
|
|
_counters_prev = elems;
|
|
}
|
|
|
|
function getFeedUnread(feed, is_cat) {
|
|
try {
|
|
const tree = dijit.byId("feedTree");
|
|
|
|
if (tree && tree.model)
|
|
return tree.model.getFeedUnread(feed, is_cat);
|
|
|
|
} catch (e) {
|
|
//
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
function getFeedCategory(feed) {
|
|
try {
|
|
const tree = dijit.byId("feedTree");
|
|
|
|
if (tree && tree.model)
|
|
return tree.getFeedCategory(feed);
|
|
|
|
} catch (e) {
|
|
//
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function hideOrShowFeeds(hide) {
|
|
const tree = dijit.byId("feedTree");
|
|
|
|
if (tree)
|
|
return tree.hideRead(hide, getInitParam("hide_read_shows_special"));
|
|
}
|
|
|
|
function getFeedName(feed, is_cat) {
|
|
|
|
if (isNaN(feed)) return feed; // it's a tag
|
|
|
|
const tree = dijit.byId("feedTree");
|
|
|
|
if (tree && tree.model)
|
|
return tree.model.getFeedValue(feed, is_cat, 'name');
|
|
}
|
|
|
|
function getFeedValue(feed, is_cat, key) {
|
|
try {
|
|
const tree = dijit.byId("feedTree");
|
|
|
|
if (tree && tree.model)
|
|
return tree.model.getFeedValue(feed, is_cat, key);
|
|
|
|
} catch (e) {
|
|
//
|
|
}
|
|
return '';
|
|
}
|
|
|
|
function setFeedUnread(feed, is_cat, unread) {
|
|
const tree = dijit.byId("feedTree");
|
|
|
|
if (tree && tree.model)
|
|
return tree.model.setFeedUnread(feed, is_cat, unread);
|
|
}
|
|
|
|
function setFeedValue(feed, is_cat, key, value) {
|
|
try {
|
|
const tree = dijit.byId("feedTree");
|
|
|
|
if (tree && tree.model)
|
|
return tree.model.setFeedValue(feed, is_cat, key, value);
|
|
|
|
} catch (e) {
|
|
//
|
|
}
|
|
}
|
|
|
|
function selectFeed(feed, is_cat) {
|
|
const tree = dijit.byId("feedTree");
|
|
|
|
if (tree) return tree.selectFeed(feed, is_cat);
|
|
}
|
|
|
|
function setFeedIcon(feed, is_cat, src) {
|
|
const tree = dijit.byId("feedTree");
|
|
|
|
if (tree) return tree.setFeedIcon(feed, is_cat, src);
|
|
}
|
|
|
|
function setFeedExpandoIcon(feed, is_cat, src) {
|
|
const tree = dijit.byId("feedTree");
|
|
|
|
if (tree) return tree.setFeedExpandoIcon(feed, is_cat, src);
|
|
|
|
return false;
|
|
}
|
|
|
|
function getNextUnreadFeed(feed, is_cat) {
|
|
const tree = dijit.byId("feedTree");
|
|
const nuf = tree.model.getNextUnreadFeed(feed, is_cat);
|
|
|
|
if (nuf)
|
|
return tree.model.store.getValue(nuf, 'bare_id');
|
|
}
|
|
|
|
function catchupCurrentFeed(mode) {
|
|
catchupFeed(getActiveFeedId(), activeFeedIsCat(), mode);
|
|
}
|
|
|
|
function catchupFeedInGroup(id) {
|
|
const title = getFeedName(id);
|
|
|
|
const str = __("Mark all articles in %s as read?").replace("%s", title);
|
|
|
|
if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
|
|
|
|
const rows = $$("#headlines-frame > div[id*=RROW][data-orig-feed-id='"+id+"']");
|
|
|
|
if (rows.length > 0) {
|
|
|
|
rows.each(function (row) {
|
|
row.removeClassName("Unread");
|
|
|
|
if (row.getAttribute("data-article-id") != getActiveArticleId()) {
|
|
new Effect.Fade(row, {duration: 0.5});
|
|
}
|
|
|
|
});
|
|
|
|
const feedTitles = $$("#headlines-frame > div[class='feed-title']");
|
|
|
|
for (let i = 0; i < feedTitles.length; i++) {
|
|
if (feedTitles[i].getAttribute("data-feed-id") == id) {
|
|
|
|
if (i < feedTitles.length - 1) {
|
|
new Effect.Fade(feedTitles[i], {duration: 0.5});
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
updateFloatingTitle(true);
|
|
}
|
|
|
|
notify_progress("Loading, please wait...", true);
|
|
|
|
xhrPost("backend.php", { op: "rpc", method: "catchupFeed", feed_id: id, is_cat: false}, (transport) => {
|
|
handle_rpc_json(transport);
|
|
});
|
|
}
|
|
}
|
|
|
|
function catchupFeed(feed, is_cat, mode) {
|
|
if (is_cat == undefined) is_cat = false;
|
|
|
|
let str = false;
|
|
|
|
switch (mode) {
|
|
case "1day":
|
|
str = __("Mark %w in %s older than 1 day as read?");
|
|
break;
|
|
case "1week":
|
|
str = __("Mark %w in %s older than 1 week as read?");
|
|
break;
|
|
case "2week":
|
|
str = __("Mark %w in %s older than 2 weeks as read?");
|
|
break;
|
|
default:
|
|
str = __("Mark %w in %s as read?");
|
|
}
|
|
|
|
const mark_what = last_search_query && last_search_query[0] ? __("search results") : __("all articles");
|
|
const fn = getFeedName(feed, is_cat);
|
|
|
|
str = str.replace("%s", fn)
|
|
.replace("%w", mark_what);
|
|
|
|
if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
|
|
return;
|
|
}
|
|
|
|
const catchup_query = {op: 'rpc', method: 'catchupFeed', feed_id: feed,
|
|
is_cat: is_cat, mode: mode, search_query: last_search_query[0],
|
|
search_lang: last_search_query[1]};
|
|
|
|
notify_progress("Loading, please wait...", true);
|
|
|
|
xhrPost("backend.php", catchup_query, (transport) => {
|
|
handle_rpc_json(transport);
|
|
|
|
const show_next_feed = getInitParam("on_catchup_show_next_feed") == "1";
|
|
|
|
if (show_next_feed) {
|
|
const nuf = getNextUnreadFeed(feed, is_cat);
|
|
|
|
if (nuf) {
|
|
viewfeed({feed: nuf, is_cat: is_cat});
|
|
}
|
|
} else if (feed == getActiveFeedId() && is_cat == activeFeedIsCat()) {
|
|
viewCurrentFeed();
|
|
}
|
|
|
|
notify("");
|
|
});
|
|
}
|
|
|
|
function decrementFeedCounter(feed, is_cat) {
|
|
let ctr = getFeedUnread(feed, is_cat);
|
|
|
|
if (ctr > 0) {
|
|
setFeedUnread(feed, is_cat, ctr - 1);
|
|
global_unread = global_unread - 1;
|
|
updateTitle();
|
|
|
|
if (!is_cat) {
|
|
const cat = parseInt(getFeedCategory(feed));
|
|
|
|
if (!isNaN(cat)) {
|
|
ctr = getFeedUnread(cat, true);
|
|
|
|
if (ctr > 0) {
|
|
setFeedUnread(cat, true, ctr - 1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
|