]> git.wh0rd.org - tt-rss.git/blame - js/feedlist.js
scroll handler: performance improvements
[tt-rss.git] / js / feedlist.js
CommitLineData
8572e010
AD
1let _infscroll_disable = 0;
2let _infscroll_request_sent = 0;
e7a66d12 3
8572e010
AD
4let _search_query = false;
5let _viewfeed_last = 0;
6let _viewfeed_timeout = false;
ac541432 7
8572e010
AD
8let counters_last_request = 0;
9let _counters_prev = [];
91cdbe52 10
6358d70d
AD
11function resetCounterCache() {
12 _counters_prev = [];
13}
14
71317973 15function loadMoreHeadlines() {
1bfe1d7b
AD
16 console.log("loadMoreHeadlines");
17
424e28db 18 let offset = 0;
1bfe1d7b 19
424e28db
AD
20 const view_mode = document.forms["main_toolbar_form"].view_mode.value;
21 const unread_in_buffer = $$("#headlines-frame > div[id*=RROW][class*=Unread]").length;
22 const num_all = $$("#headlines-frame > div[id*=RROW]").length;
23 const num_unread = getFeedUnread(getActiveFeedId(), activeFeedIsCat());
1bfe1d7b
AD
24
25 // TODO implement marked & published
26
27 if (view_mode == "marked") {
28 console.warn("loadMoreHeadlines: marked is not implemented, falling back.");
29 offset = num_all;
30 } else if (view_mode == "published") {
31 console.warn("loadMoreHeadlines: published is not implemented, falling back.");
32 offset = num_all;
33 } else if (view_mode == "unread") {
34 offset = unread_in_buffer;
35 } else if (_search_query) {
36 offset = num_all;
37 } else if (view_mode == "adaptive" && !(getActiveFeedId() == -1 && !activeFeedIsCat())) {
38 // ^ starred feed shows both unread & read articles in adaptive mode
39 offset = num_unread > 0 ? unread_in_buffer : num_all;
40 } else {
41 offset = num_all;
42 }
2a8534a0 43
1bfe1d7b 44 console.log("offset: " + offset);
c67cd7ab 45
1bfe1d7b 46 viewfeed({feed: getActiveFeedId(), is_cat: activeFeedIsCat(), offset: offset, infscroll_req: true});
ac541432 47
ac541432
AD
48}
49
0b41bd34 50function cleanup_memory(root) {
424e28db 51 const dijits = dojo.query("[widgetid]", dijit.byId(root).domNode).map(dijit.byNode);
0b41bd34 52
1bfe1d7b
AD
53 dijits.each(function (d) {
54 dojo.destroy(d.domNode);
55 });
0b41bd34 56
1bfe1d7b
AD
57 $$("#" + root + " *").each(function (i) {
58 i.parentNode ? i.parentNode.removeChild(i) : true;
59 });
0b41bd34
AD
60}
61
6b0a17ad 62function viewfeed(params) {
424e28db 63 const feed = params.feed;
0d272273
AD
64 let is_cat = !!params.is_cat || false;
65 let offset = params.offset || 0;
66 let background = params.background || false;
67 let infscroll_req = params.infscroll_req || false;
424e28db
AD
68 const can_wait = params.can_wait;
69 const viewfeed_debug = params.viewfeed_debug;
70 const method = params.method;
1bfe1d7b 71
1bfe1d7b
AD
72 if (infscroll_req == undefined) infscroll_req = false;
73
74 last_requested_article = 0;
75
76 if (feed != getActiveFeedId() || activeFeedIsCat() != is_cat) {
77 if (!background && _search_query) _search_query = false;
78 }
4d5eed55 79
1bfe1d7b
AD
80 if (!background) {
81 _viewfeed_last = get_timestamp();
0b41bd34 82
1bfe1d7b
AD
83 if (getActiveFeedId() != feed || !infscroll_req) {
84 setActiveArticleId(0);
85 _infscroll_disable = 0;
203de776 86
1bfe1d7b
AD
87 cleanup_memory("headlines-frame");
88 _headlines_scroll_offset = 0;
89 }
82764a41 90
1bfe1d7b 91 if (infscroll_req) {
424e28db 92 const timestamp = get_timestamp();
37c03d3a 93
1bfe1d7b
AD
94 if (_infscroll_request_sent && _infscroll_request_sent + 30 > timestamp) {
95 //console.log("infscroll request in progress, aborting");
96 return;
82764a41 97 }
1bfe1d7b
AD
98
99 _infscroll_request_sent = timestamp;
82764a41 100 }
1bfe1d7b 101 }
82764a41 102
1bfe1d7b 103 Form.enable("main_toolbar_form");
fd98576c 104
560b9fdd
AD
105 let query = Object.assign({op: "feeds", method: "view", feed: feed},
106 dojo.formToObject("main_toolbar_form"));
86b682ce 107
560b9fdd 108 if (method) query.m = method;
791877c3 109
1bfe1d7b
AD
110 if (offset > 0) {
111 if (current_first_id) {
560b9fdd 112 query.fid = current_first_id;
f5a0fb8b 113 }
1bfe1d7b 114 }
f5a0fb8b 115
1bfe1d7b
AD
116 if (!background) {
117 if (_search_query) {
560b9fdd 118 query = Object.assign(query, _search_query);
1bfe1d7b 119 }
164f4738 120
1bfe1d7b 121 if (offset != 0) {
560b9fdd 122 query.skip = offset;
60d4df4d 123
1bfe1d7b
AD
124 // to prevent duplicate feed titles when showing grouped vfeeds
125 if (vgroup_last_feed) {
560b9fdd 126 query.vgrlf = vgroup_last_feed;
60d4df4d 127 }
424e28db 128 } else if (!is_cat && feed == getActiveFeedId() && !params.method) {
560b9fdd 129 query.m = "ForceUpdate";
1bfe1d7b 130 }
37c03d3a 131
1bfe1d7b 132 Form.enable("main_toolbar_form");
37c03d3a 133
1bfe1d7b
AD
134 if (!setFeedExpandoIcon(feed, is_cat,
135 (is_cat) ? 'images/indicator_tiny.gif' : 'images/indicator_white.gif'))
136 notify_progress("Loading, please wait...", true);
137 }
203de776 138
560b9fdd 139 query.cat = is_cat;
6b4163cb 140
1bfe1d7b
AD
141 if (can_wait && _viewfeed_timeout) {
142 setFeedExpandoIcon(getActiveFeedId(), activeFeedIsCat(), 'images/blank_icon.gif');
143 clearTimeout(_viewfeed_timeout);
144 }
a95e14fb 145
1bfe1d7b 146 setActiveFeedId(feed, is_cat);
ba1c7d7c 147
1bfe1d7b 148 if (viewfeed_debug) {
560b9fdd
AD
149 window.open("backend.php?" +
150 dojo.objectToQuery(
151 Object.assign({debug: 1, csrf_token: getInitParam("csrf_token")}, query)
152 ));
1bfe1d7b 153 }
6b0a17ad 154
424e28db 155 const timeout_ms = can_wait ? 250 : 0;
0d272273
AD
156 _viewfeed_timeout = setTimeout(() => {
157
158 xhrPost("backend.php", query, (transport) => {
159 try {
160 setFeedExpandoIcon(feed, is_cat, 'images/blank_icon.gif');
161 headlines_callback2(transport, offset, background, infscroll_req);
162 PluginHost.run(PluginHost.HOOK_FEED_LOADED, [feed, is_cat]);
163 } catch (e) {
164 exception_error(e);
165 }
166 });
167
1bfe1d7b 168 }, timeout_ms); // Wait 250ms
6b4163cb 169
1a66d16e
AD
170}
171
6b4163cb 172function feedlist_init() {
1bfe1d7b 173 console.log("in feedlist init");
bd202c3f 174
1bfe1d7b 175 loading_set_progress(50);
ea18ccea 176
1bfe1d7b 177 document.onkeydown = hotkey_handler;
7e8be97b 178 setInterval(hotkey_prefix_timeout, 5*1000);
0c06bb5f 179 setInterval(catchupBatchedArticles, 3*1000);
164469f9 180
1bfe1d7b
AD
181 if (!getActiveFeedId()) {
182 viewfeed({feed: -3});
183 } else {
184 viewfeed({feed: getActiveFeedId(), is_cat: activeFeedIsCat()});
185 }
7086277c 186
1bfe1d7b 187 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
782ddd70 188
7c0eb1b6
AD
189 if (getInitParam("is_default_pw")) {
190 console.warn("user password is at default value");
191
424e28db 192 const dialog = new dijit.Dialog({
7c0eb1b6
AD
193 title: __("Your password is at default value"),
194 href: "backend.php?op=dlg&method=defaultpasswordwarning",
195 id: 'infoBox',
196 style: "width: 600px",
197 onCancel: function() {
198 return true;
199 },
200 onExecute: function() {
201 return true;
202 },
203 onClose: function() {
204 return true;
205 }
206 });
207
208 dialog.show();
209 }
210
03c56b04
AD
211 // bw_limit disables timeout() so we request initial counters separately
212 if (getInitParam("bw_limit") == "1") {
213 request_counters(true);
214 } else {
215 setTimeout(timeout, 250);
216 }
3745788e 217}
22e06406 218
997d9d7c 219
5b55e9e2 220function request_counters(force) {
424e28db
AD
221 const date = new Date();
222 const timestamp = Math.round(date.getTime() / 1000);
5225d420 223
1bfe1d7b
AD
224 if (force || timestamp - counters_last_request > 5) {
225 console.log("scheduling request of counters...");
5b55e9e2 226
1bfe1d7b 227 counters_last_request = timestamp;
5b55e9e2 228
0d272273 229 let query = {op: "rpc", method: "getAllCounters", seq: next_seq()};
5b55e9e2 230
1bfe1d7b 231 if (!force)
0d272273 232 query.last_article_id = getInitParam("last_article_id");
5b55e9e2 233
0d272273
AD
234 xhrPost("backend.php", query, (transport) => {
235 handle_rpc_json(transport);
236 });
997d9d7c 237
1bfe1d7b
AD
238 } else {
239 console.log("request_counters: rate limit reached: " + (timestamp - counters_last_request));
5aaf9b34
AD
240 }
241}
a4b906ec 242
d5f74019
AD
243// NOTE: this implementation is incomplete
244// for general objects but good enough for counters
245// http://adripofjavascript.com/blog/drips/object-equality-in-javascript.html
246function counter_is_equal(a, b) {
247 // Create arrays of property names
424e28db
AD
248 const aProps = Object.getOwnPropertyNames(a);
249 const bProps = Object.getOwnPropertyNames(b);
d5f74019
AD
250
251 // If number of properties is different,
252 // objects are not equivalent
253 if (aProps.length != bProps.length) {
254 return false;
255 }
256
424e28db
AD
257 for (let i = 0; i < aProps.length; i++) {
258 const propName = aProps[i];
d5f74019
AD
259
260 // If values of same property are not equal,
261 // objects are not equivalent
262 if (a[propName] !== b[propName]) {
263 return false;
264 }
265 }
266
267 // If we made it this far, objects
268 // are considered equivalent
269 return true;
270}
271
272
a4e04c49 273function parse_counters(elems) {
424e28db 274 for (let l = 0; l < elems.length; l++) {
e69fb880 275
1bfe1d7b
AD
276 if (_counters_prev[l] && counter_is_equal(elems[l], _counters_prev[l])) {
277 continue;
278 }
a4e04c49 279
424e28db
AD
280 const id = elems[l].id;
281 const kind = elems[l].kind;
282 const ctr = parseInt(elems[l].counter);
283 const error = elems[l].error;
284 const has_img = elems[l].has_img;
285 const updated = elems[l].updated;
286 const auxctr = parseInt(elems[l].auxcounter);
e69fb880 287
1bfe1d7b
AD
288 if (id == "global-unread") {
289 global_unread = ctr;
290 updateTitle();
291 continue;
292 }
13e785e0 293
1bfe1d7b
AD
294 if (id == "subscribed-feeds") {
295 /* feeds_found = ctr; */
296 continue;
297 }
78b2c6ce 298
1bfe1d7b
AD
299 /*if (getFeedUnread(id, (kind == "cat")) != ctr ||
300 (kind == "cat")) {
301 }*/
e69fb880 302
1bfe1d7b
AD
303 setFeedUnread(id, (kind == "cat"), ctr);
304 setFeedValue(id, (kind == "cat"), 'auxcounter', auxctr);
2ef5c21f 305
1bfe1d7b
AD
306 if (kind != "cat") {
307 setFeedValue(id, false, 'error', error);
308 setFeedValue(id, false, 'updated', updated);
309
310 if (id > 0) {
311 if (has_img) {
312 setFeedIcon(id, false,
fa3bcfa3 313 getInitParam("icons_url") + "/" + id + ".ico?" + has_img);
1bfe1d7b
AD
314 } else {
315 setFeedIcon(id, false, 'images/blank_icon.gif');
2ef5c21f 316 }
e69fb880 317 }
fcf70c51 318 }
1bfe1d7b 319 }
bd202c3f 320
1bfe1d7b 321 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
a4e04c49 322
1bfe1d7b 323 _counters_prev = elems;
e69fb880
AD
324}
325
6e88da82 326function getFeedUnread(feed, is_cat) {
e69fb880 327 try {
424e28db 328 const tree = dijit.byId("feedTree");
e032e2e7 329
bd202c3f 330 if (tree && tree.model)
e032e2e7
AD
331 return tree.model.getFeedUnread(feed, is_cat);
332
e69fb880 333 } catch (e) {
fcf70c51 334 //
e69fb880 335 }
fcf70c51
AD
336
337 return -1;
e69fb880
AD
338}
339
6c8e8fbd
AD
340function getFeedCategory(feed) {
341 try {
424e28db 342 const tree = dijit.byId("feedTree");
6c8e8fbd
AD
343
344 if (tree && tree.model)
345 return tree.getFeedCategory(feed);
346
347 } catch (e) {
348 //
349 }
350
351 return false;
352}
353
e69fb880 354function hideOrShowFeeds(hide) {
424e28db 355 const tree = dijit.byId("feedTree");
fcf70c51 356
05f224a3
AD
357 if (tree)
358 return tree.hideRead(hide, getInitParam("hide_read_shows_special"));
e69fb880
AD
359}
360
bd202c3f 361function getFeedName(feed, is_cat) {
9842b51c
AD
362
363 if (isNaN(feed)) return feed; // it's a tag
364
424e28db 365 const tree = dijit.byId("feedTree");
e032e2e7 366
bd202c3f 367 if (tree && tree.model)
e032e2e7 368 return tree.model.getFeedValue(feed, is_cat, 'name');
fcf70c51 369}
e69fb880 370
bd202c3f 371function getFeedValue(feed, is_cat, key) {
fcf70c51 372 try {
424e28db 373 const tree = dijit.byId("feedTree");
fcf70c51 374
bd202c3f 375 if (tree && tree.model)
e032e2e7 376 return tree.model.getFeedValue(feed, is_cat, key);
bd202c3f 377
fcf70c51
AD
378 } catch (e) {
379 //
e69fb880 380 }
fcf70c51 381 return '';
e69fb880
AD
382}
383
13e785e0 384function setFeedUnread(feed, is_cat, unread) {
424e28db 385 const tree = dijit.byId("feedTree");
e032e2e7 386
1bfe1d7b
AD
387 if (tree && tree.model)
388 return tree.model.setFeedUnread(feed, is_cat, unread);
fcf70c51
AD
389}
390
391function setFeedValue(feed, is_cat, key, value) {
392 try {
424e28db 393 const tree = dijit.byId("feedTree");
13e785e0 394
bd202c3f 395 if (tree && tree.model)
e032e2e7 396 return tree.model.setFeedValue(feed, is_cat, key, value);
c8a9fe5b 397
13e785e0 398 } catch (e) {
e032e2e7 399 //
fcf70c51
AD
400 }
401}
402
fcf70c51 403function selectFeed(feed, is_cat) {
424e28db 404 const tree = dijit.byId("feedTree");
fcf70c51 405
1bfe1d7b 406 if (tree) return tree.selectFeed(feed, is_cat);
13e785e0 407}
2ef5c21f
AD
408
409function setFeedIcon(feed, is_cat, src) {
424e28db 410 const tree = dijit.byId("feedTree");
2ef5c21f 411
1bfe1d7b 412 if (tree) return tree.setFeedIcon(feed, is_cat, src);
2ef5c21f 413}
9c99281f
AD
414
415function setFeedExpandoIcon(feed, is_cat, src) {
424e28db 416 const tree = dijit.byId("feedTree");
9c99281f 417
1bfe1d7b 418 if (tree) return tree.setFeedExpandoIcon(feed, is_cat, src);
9c99281f 419
f92d7c2b 420 return false;
9c99281f 421}
37c03d3a
AD
422
423function getNextUnreadFeed(feed, is_cat) {
424e28db
AD
424 const tree = dijit.byId("feedTree");
425 const nuf = tree.model.getNextUnreadFeed(feed, is_cat);
37c03d3a 426
1bfe1d7b
AD
427 if (nuf)
428 return tree.model.store.getValue(nuf, 'bare_id');
37c03d3a
AD
429}
430
e3ca32a8
AD
431function catchupCurrentFeed(mode) {
432 catchupFeed(getActiveFeedId(), activeFeedIsCat(), mode);
44245618
AD
433}
434
435function catchupFeedInGroup(id) {
424e28db 436 const title = getFeedName(id);
44245618 437
424e28db 438 const str = __("Mark all articles in %s as read?").replace("%s", title);
44245618 439
1bfe1d7b 440 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
701de97e 441
424e28db 442 const rows = $$("#headlines-frame > div[id*=RROW][data-orig-feed-id='"+id+"']");
701de97e 443
1bfe1d7b 444 if (rows.length > 0) {
701de97e 445
1bfe1d7b
AD
446 rows.each(function (row) {
447 row.removeClassName("Unread");
2d052e42 448
1bfe1d7b
AD
449 if (row.getAttribute("data-article-id") != getActiveArticleId()) {
450 new Effect.Fade(row, {duration: 0.5});
451 }
701de97e 452
1bfe1d7b 453 });
2d052e42 454
424e28db 455 const feedTitles = $$("#headlines-frame > div[class='cdmFeedTitle']");
2d052e42 456
424e28db 457 for (let i = 0; i < feedTitles.length; i++) {
1bfe1d7b 458 if (feedTitles[i].getAttribute("data-feed-id") == id) {
2d052e42 459
1bfe1d7b
AD
460 if (i < feedTitles.length - 1) {
461 new Effect.Fade(feedTitles[i], {duration: 0.5});
2d052e42 462 }
2d052e42 463
1bfe1d7b
AD
464 break;
465 }
dfaf806a 466 }
701de97e 467
1bfe1d7b
AD
468 updateFloatingTitle(true);
469 }
701de97e 470
1bfe1d7b 471 notify_progress("Loading, please wait...", true);
701de97e 472
0d272273
AD
473 xhrPost("backend.php", { op: "rpc", method: "catchupFeed", feed_id: id, is_cat: false}, (transport) => {
474 handle_rpc_json(transport);
475 });
44245618
AD
476 }
477}
478
c8b693cf 479function catchupFeed(feed, is_cat, mode) {
1bfe1d7b
AD
480 if (is_cat == undefined) is_cat = false;
481
424e28db 482 let str = false;
1bfe1d7b
AD
483
484 switch (mode) {
485 case "1day":
33753541 486 str = __("Mark %w in %s older than 1 day as read?");
1bfe1d7b
AD
487 break;
488 case "1week":
33753541 489 str = __("Mark %w in %s older than 1 week as read?");
1bfe1d7b
AD
490 break;
491 case "2week":
33753541 492 str = __("Mark %w in %s older than 2 weeks as read?");
1bfe1d7b
AD
493 break;
494 default:
33753541 495 str = __("Mark %w in %s as read?");
1bfe1d7b 496 }
c8b693cf 497
424e28db
AD
498 const mark_what = last_search_query && last_search_query[0] ? __("search results") : __("all articles");
499 const fn = getFeedName(feed, is_cat);
a593cb62 500
33753541
AD
501 str = str.replace("%s", fn)
502 .replace("%w", mark_what);
a593cb62 503
1bfe1d7b
AD
504 if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
505 return;
506 }
e3d2c029 507
424e28db 508 const catchup_query = {op: 'rpc', method: 'catchupFeed', feed_id: feed,
33753541
AD
509 is_cat: is_cat, mode: mode, search_query: last_search_query[0],
510 search_lang: last_search_query[1]};
184f5195 511
1bfe1d7b 512 notify_progress("Loading, please wait...", true);
e3d2c029 513
0d272273
AD
514 xhrPost("backend.php", catchup_query, (transport) => {
515 handle_rpc_json(transport);
44245618 516
0d272273 517 const show_next_feed = getInitParam("on_catchup_show_next_feed") == "1";
44245618 518
0d272273
AD
519 if (show_next_feed) {
520 const nuf = getNextUnreadFeed(feed, is_cat);
44245618 521
0d272273
AD
522 if (nuf) {
523 viewfeed({feed: nuf, is_cat: is_cat});
524 }
525 } else if (feed == getActiveFeedId() && is_cat == activeFeedIsCat()) {
526 viewCurrentFeed();
527 }
e3d2c029 528
0d272273
AD
529 notify("");
530 });
e3d2c029 531}
de44b9a6
AD
532
533function decrementFeedCounter(feed, is_cat) {
424e28db 534 let ctr = getFeedUnread(feed, is_cat);
de44b9a6 535
1bfe1d7b
AD
536 if (ctr > 0) {
537 setFeedUnread(feed, is_cat, ctr - 1);
538 global_unread = global_unread - 1;
539 updateTitle();
de44b9a6 540
1bfe1d7b 541 if (!is_cat) {
424e28db 542 const cat = parseInt(getFeedCategory(feed));
de44b9a6 543
1bfe1d7b
AD
544 if (!isNaN(cat)) {
545 ctr = getFeedUnread(cat, true);
de44b9a6 546
1bfe1d7b
AD
547 if (ctr > 0) {
548 setFeedUnread(cat, true, ctr - 1);
de44b9a6
AD
549 }
550 }
551 }
de44b9a6 552 }
1bfe1d7b 553
de44b9a6 554}
2196b517
AD
555
556