]> git.wh0rd.org - tt-rss.git/blame - js/viewfeed.js
scroll handler: performance improvements
[tt-rss.git] / js / viewfeed.js
CommitLineData
4508e310
AD
1/* global dijit, __ */
2
8572e010 3let _active_article_id = 0;
18e1a773 4
8572e010
AD
5let vgroup_last_feed = false;
6let post_under_pointer = false;
081e527d 7
0d272273 8let last_requested_article = 0;
022d9e9e 9
8572e010
AD
10let catchup_id_batch = [];
11let catchup_timeout_id = false;
37c03d3a 12
8572e010
AD
13let cids_requested = [];
14let loaded_article_ids = [];
15let _last_headlines_update = 0;
16let _headlines_scroll_offset = 0;
17let current_first_id = 0;
18let last_search_query;
76495dfd 19
8572e010 20let _catchup_request_sent = false;
8292123e 21
8572e010 22let has_storage = 'sessionStorage' in window && window['sessionStorage'] !== null;
78b2c6ce 23
28364067 24function headlines_callback2(transport, offset, background, infscroll_req) {
0d272273 25 const reply = handle_rpc_json(transport);
1bfe1d7b
AD
26
27 console.log("headlines_callback2 [offset=" + offset + "] B:" + background + " I:" + infscroll_req);
28
0d272273
AD
29 if (background)
30 return;
1bfe1d7b 31
0d272273
AD
32 var is_cat = false;
33 var feed_id = false;
9ca945a6 34
1bfe1d7b 35 if (reply) {
e19c1824 36
1bfe1d7b
AD
37 is_cat = reply['headlines']['is_cat'];
38 feed_id = reply['headlines']['id'];
33753541 39 last_search_query = reply['headlines']['search_query'];
ac541432 40
1bfe1d7b
AD
41 if (feed_id != -7 && (feed_id != getActiveFeedId() || is_cat != activeFeedIsCat()))
42 return;
e19c1824 43
1bfe1d7b
AD
44 try {
45 if (infscroll_req == false) {
46 $("headlines-frame").scrollTop = 0;
ff4019f0 47
1bfe1d7b
AD
48 $("floatingTitle").style.visibility = "hidden";
49 $("floatingTitle").setAttribute("data-article-id", 0);
50 $("floatingTitle").innerHTML = "";
37c03d3a 51 }
424e28db 52 } catch (e) { }
081e527d 53
1bfe1d7b
AD
54 $("headlines-frame").removeClassName("cdm");
55 $("headlines-frame").removeClassName("normal");
bd202c3f 56
1bfe1d7b 57 $("headlines-frame").addClassName(isCdmMode() ? "cdm" : "normal");
fa2cde39 58
424e28db 59 const headlines_count = reply['headlines-info']['count'];
5f9699a5 60
1bfe1d7b 61 vgroup_last_feed = reply['headlines-info']['vgroup_last_feed'];
e19c1824 62
1bfe1d7b
AD
63 if (parseInt(headlines_count) < 30) {
64 _infscroll_disable = 1;
65 } else {
66 _infscroll_disable = 0;
67 }
c068b98b 68
1bfe1d7b 69 current_first_id = reply['headlines']['first_id'];
424e28db
AD
70 const counters = reply['counters'];
71 const articles = reply['articles'];
c068b98b 72
1bfe1d7b
AD
73 if (infscroll_req == false) {
74 loaded_article_ids = [];
009646d2 75
1bfe1d7b
AD
76 dojo.html.set($("headlines-toolbar"),
77 reply['headlines']['toolbar'],
78 {parseContent: true});
081e527d 79
1bfe1d7b 80 $("headlines-frame").innerHTML = '';
16c93768 81
0d272273 82 let tmp = document.createElement("div");
1bfe1d7b
AD
83 tmp.innerHTML = reply['headlines']['content'];
84 dojo.parser.parse(tmp);
dd5865d4 85
1bfe1d7b
AD
86 while (tmp.hasChildNodes()) {
87 var row = tmp.removeChild(tmp.firstChild);
88
89 if (loaded_article_ids.indexOf(row.id) == -1 || row.hasClassName("cdmFeedTitle")) {
90 dijit.byId("headlines-frame").domNode.appendChild(row);
ac4c1383 91
f1706996 92 loaded_article_ids.push(row.id);
1bfe1d7b
AD
93 }
94 }
95
0d272273 96 let hsp = $("headlines-spacer");
1bfe1d7b
AD
97 if (!hsp) hsp = new Element("DIV", {"id": "headlines-spacer"});
98 dijit.byId('headlines-frame').domNode.appendChild(hsp);
99
100 initHeadlinesMenu();
101
102 if (_infscroll_disable)
103 hsp.innerHTML = "<a href='#' onclick='openNextUnreadFeed()'>" +
104 __("Click to open next unread feed.") + "</a>";
105
106 if (_search_query) {
107 $("feed_title").innerHTML += "<span id='cancel_search'>" +
108 " (<a href='#' onclick='cancelSearch()'>" + __("Cancel search") + "</a>)" +
109 "</span>";
110 }
111
424e28db 112 } else if (headlines_count > 0 && feed_id == getActiveFeedId() && is_cat == activeFeedIsCat()) {
1bfe1d7b
AD
113 console.log("adding some more headlines: " + headlines_count);
114
424e28db
AD
115 const c = dijit.byId("headlines-frame");
116 const ids = getSelectedArticleIds2();
1bfe1d7b 117
0d272273 118 let hsp = $("headlines-spacer");
1bfe1d7b
AD
119
120 if (hsp)
121 c.domNode.removeChild(hsp);
a71e571b 122
0d272273 123 let tmp = document.createElement("div");
a71e571b
AD
124 tmp.innerHTML = reply['headlines']['content'];
125 dojo.parser.parse(tmp);
126
127 while (tmp.hasChildNodes()) {
0d272273 128 let row = tmp.removeChild(tmp.firstChild);
a71e571b
AD
129
130 if (loaded_article_ids.indexOf(row.id) == -1 || row.hasClassName("cdmFeedTitle")) {
131 dijit.byId("headlines-frame").domNode.appendChild(row);
132
133 loaded_article_ids.push(row.id);
134 }
135 }
18eb64a8 136
06a02265 137 if (!hsp) hsp = new Element("DIV", {"id": "headlines-spacer"});
1bfe1d7b 138 c.domNode.appendChild(hsp);
18eb64a8 139
1bfe1d7b 140 if (headlines_count < 30) _infscroll_disable = true;
7e8832b3 141
1bfe1d7b 142 console.log("restore selected ids: " + ids);
22f675e5 143
0d272273 144 for (let i = 0; i < ids.length; i++) {
1bfe1d7b 145 markHeadline(ids[i]);
74467907
AD
146 }
147
1bfe1d7b 148 initHeadlinesMenu();
18eb64a8 149
1bfe1d7b
AD
150 if (_infscroll_disable) {
151 hsp.innerHTML = "<a href='#' onclick='openNextUnreadFeed()'>" +
152 __("Click to open next unread feed.") + "</a>";
153 }
c67cd7ab 154
1bfe1d7b
AD
155 } else {
156 console.log("no new headlines received");
c32cd48a 157
424e28db 158 const first_id_changed = reply['headlines']['first_id_changed'];
1bfe1d7b 159 console.log("first id changed:" + first_id_changed);
c32cd48a 160
0d272273 161 let hsp = $("headlines-spacer");
b509d64e 162
1bfe1d7b
AD
163 if (hsp) {
164 if (first_id_changed) {
165 hsp.innerHTML = "<a href='#' onclick='viewCurrentFeed()'>" +
166 __("New articles found, reload feed to continue.") + "</a>";
167 } else {
39d488a2
AD
168 hsp.innerHTML = "<a href='#' onclick='openNextUnreadFeed()'>" +
169 __("Click to open next unread feed.") + "</a>";
170 }
171
6c3b0198 172 }
009646d2 173
961f4c73 174 }
d36f5607 175
1bfe1d7b 176 if (articles) {
0d272273 177 for (let i = 0; i < articles.length; i++) {
424e28db 178 const a_id = articles[i]['id'];
1bfe1d7b
AD
179 cache_set("article:" + a_id, articles[i]['content']);
180 }
df63125d 181 } else {
1bfe1d7b 182 console.log("no cached articles received");
3de0261a 183 }
009646d2 184
1bfe1d7b
AD
185 if (counters)
186 parse_counters(counters);
187 else
22adcd74 188 request_counters();
82764a41 189
1bfe1d7b
AD
190 } else {
191 console.error("Invalid object received: " + transport.responseText);
192 dijit.byId("headlines-frame").attr('content', "<div class='whiteBox'>" +
193 __('Could not update headlines (invalid object received - see error console for details)') +
194 "</div>");
195 }
b74c5134 196
1bfe1d7b
AD
197 _infscroll_request_sent = 0;
198 _last_headlines_update = new Date().getTime();
2b28c8f2 199
1bfe1d7b 200 unpackVisibleHeadlines();
2b28c8f2 201
1bfe1d7b 202 // if we have some more space in the buffer, why not try to fill it
2b28c8f2 203
1bfe1d7b
AD
204 if (!_infscroll_disable && $("headlines-spacer") &&
205 $("headlines-spacer").offsetTop < $("headlines-frame").offsetHeight) {
730dbf19 206
1bfe1d7b
AD
207 window.setTimeout(function() {
208 loadMoreHeadlines();
209 }, 250);
6b4163cb 210 }
1bfe1d7b
AD
211
212 notify("");
6b4163cb
AD
213}
214
e097e8be 215function render_article(article) {
1bfe1d7b 216 cleanup_memory("content-insert");
0b41bd34 217
1bfe1d7b
AD
218 dijit.byId("headlines-wrap-inner").addChild(
219 dijit.byId("content-insert"));
6f3976c9 220
424e28db 221 const c = dijit.byId("content-insert");
17042fc0 222
1bfe1d7b
AD
223 try {
224 c.domNode.scrollTop = 0;
424e28db 225 } catch (e) { }
c559efcf 226
1bfe1d7b
AD
227 c.attr('content', article);
228 PluginHost.run(PluginHost.HOOK_ARTICLE_RENDERED, c.domNode);
53fd5495 229
1bfe1d7b 230 correctHeadlinesOffset(getActiveArticleId());
e097e8be 231
1bfe1d7b
AD
232 try {
233 c.focus();
424e28db 234 } catch (e) { }
e097e8be
AD
235}
236
8b35d171 237function showArticleInHeadlines(id, noexpand) {
424e28db 238 const row = $("RROW-" + id);
1bfe1d7b 239 if (!row) return;
e097e8be 240
1bfe1d7b
AD
241 if (!noexpand)
242 row.removeClassName("Unread");
11063ec6 243
1bfe1d7b 244 row.addClassName("active");
11063ec6 245
1bfe1d7b 246 selectArticles('none');
4c009c78 247
1bfe1d7b 248 markHeadline(id);
71d277de 249}
e097e8be 250
a491a7f1 251function article_callback2(transport, id) {
1bfe1d7b 252 console.log("article_callback2 " + id);
e097e8be 253
560b9fdd 254 const reply = handle_rpc_json(transport);
2184738a 255
1bfe1d7b 256 if (reply) {
37c03d3a 257
1bfe1d7b
AD
258 reply.each(function(article) {
259 if (getActiveArticleId() == article['id']) {
260 render_article(article['content']);
261 }
262 cids_requested.remove(article['id']);
37c03d3a 263
1bfe1d7b
AD
264 cache_set("article:" + article['id'], article['content']);
265 });
71d277de 266
1bfe1d7b
AD
267 } else {
268 console.error("Invalid object received: " + transport.responseText);
009646d2 269
1bfe1d7b
AD
270 render_article("<div class='whiteBox'>" +
271 __('Could not display article (invalid object received - see error console for details)') + "</div>");
272 }
4b6206fa 273
560b9fdd 274 const unread_in_buffer = $$("#headlines-frame > div[id*=RROW][class*=Unread]").length;
1bfe1d7b 275 request_counters(unread_in_buffer == 0);
9ec58704 276
1bfe1d7b 277 notify("");
6b4163cb
AD
278}
279
8b35d171 280function view(id, activefeed, noexpand) {
424e28db 281 const oldrow = $("RROW-" + getActiveArticleId());
1bfe1d7b 282 if (oldrow) oldrow.removeClassName("active");
a7c0e150 283
424e28db 284 const crow = $("RROW-" + id);
be5a5cd2 285
1bfe1d7b
AD
286 if (!crow) return;
287 if (noexpand) {
288 setActiveArticleId(id);
289 showArticleInHeadlines(id, noexpand);
290 return;
291 }
be5a5cd2 292
1bfe1d7b 293 console.log("loading article: " + id);
12e55b90 294
424e28db 295 const cached_article = cache_get("article:" + id);
e097e8be 296
1bfe1d7b 297 console.log("cache check result: " + (cached_article != false));
009646d2 298
560b9fdd 299 const query = {op: "article", method: "view", id: id};
6b4163cb 300
424e28db 301 const neighbor_ids = getRelativePostIds(id);
e097e8be 302
1bfe1d7b 303 /* only request uncached articles */
e097e8be 304
424e28db 305 const cids_to_request = [];
5a94a953 306
424e28db 307 for (let i = 0; i < neighbor_ids.length; i++) {
1bfe1d7b
AD
308 if (cids_requested.indexOf(neighbor_ids[i]) == -1)
309 if (!cache_get("article:" + neighbor_ids[i])) {
310 cids_to_request.push(neighbor_ids[i]);
311 cids_requested.push(neighbor_ids[i]);
312 }
313 }
5a94a953 314
1bfe1d7b 315 console.log("additional ids: " + cids_to_request.toString());
009646d2 316
560b9fdd 317 query.cids = cids_to_request.toString();
e097e8be 318
424e28db 319 const article_is_unread = crow.hasClassName("Unread");
e097e8be 320
1bfe1d7b
AD
321 setActiveArticleId(id);
322 showArticleInHeadlines(id);
3f70f90f 323
1bfe1d7b 324 if (cached_article && article_is_unread) {
560b9fdd 325 query.mode = "prefetch";
1bfe1d7b 326 render_article(cached_article);
1bfe1d7b 327 } else if (cached_article) {
560b9fdd 328 query.mode = "prefetch_old";
1bfe1d7b 329 render_article(cached_article);
997d9d7c 330
1bfe1d7b
AD
331 // if we don't need to request any relative ids, we might as well skip
332 // the server roundtrip altogether
333 if (cids_to_request.length == 0) {
334 return;
78b2c6ce 335 }
1bfe1d7b 336 }
86173d9a 337
1bfe1d7b 338 last_requested_article = id;
022d9e9e 339
1bfe1d7b 340 console.log(query);
f7cffd2c 341
1bfe1d7b
AD
342 if (article_is_unread) {
343 decrementFeedCounter(getActiveFeedId(), activeFeedIsCat());
344 }
6c8e8fbd 345
560b9fdd
AD
346 xhrPost("backend.php", query, (transport) => {
347 article_callback2(transport, id);
348 })
6b4163cb 349
1bfe1d7b 350 return false;
a7764e51 351
f0601b87
AD
352}
353
4ad8c93b 354function toggleMark(id, client_only) {
560b9fdd 355 const query = { op: "rpc", id: id, method: "mark" };
009646d2 356
424e28db 357 const row = $("RROW-" + id);
1bfe1d7b 358 if (!row) return;
56ff7564 359
424e28db 360 const imgs = [];
08820be7 361
424e28db 362 const row_imgs = row.getElementsByClassName("markedPic");
08820be7 363
560b9fdd 364 for (let i = 0; i < row_imgs.length; i++)
1bfe1d7b 365 imgs.push(row_imgs[i]);
08820be7 366
424e28db 367 const ft = $("floatingTitle");
08820be7 368
1bfe1d7b 369 if (ft && ft.getAttribute("data-article-id") == id) {
424e28db 370 const fte = ft.getElementsByClassName("markedPic");
08820be7 371
1bfe1d7b
AD
372 for (var i = 0; i < fte.length; i++)
373 imgs.push(fte[i]);
374 }
009646d2 375
1bfe1d7b 376 for (i = 0; i < imgs.length; i++) {
424e28db 377 const img = imgs[i];
ee8768db 378
1bfe1d7b
AD
379 if (!row.hasClassName("marked")) {
380 img.src = img.src.replace("mark_unset", "mark_set");
560b9fdd 381 query.mark = 1;
1bfe1d7b
AD
382 } else {
383 img.src = img.src.replace("mark_set", "mark_unset");
560b9fdd 384 query.mark = 0;
ace1a6da 385 }
1bfe1d7b 386 }
772bc83b 387
1bfe1d7b 388 row.toggleClassName("marked");
035d7a5a 389
560b9fdd
AD
390 if (!client_only)
391 xhrPost("backend.php", query, (transport) => {
392 handle_rpc_json(transport);
1bfe1d7b 393 });
b685ba25 394 }
f0601b87 395
c7e51de1 396function togglePub(id, client_only, no_effects, note) {
560b9fdd 397 const query = { op: "rpc", id: id, method: "publ" };
009646d2 398
1bfe1d7b 399 if (note != undefined) {
560b9fdd 400 query.note = note;
1bfe1d7b 401 } else {
560b9fdd 402 query.note = "undefined";
1bfe1d7b 403 }
56ff7564 404
424e28db 405 const row = $("RROW-" + id);
1bfe1d7b 406 if (!row) return;
56ff7564 407
424e28db 408 const imgs = [];
08820be7 409
424e28db 410 const row_imgs = row.getElementsByClassName("pubPic");
08820be7 411
560b9fdd 412 for (let i = 0; i < row_imgs.length; i++)
1bfe1d7b 413 imgs.push(row_imgs[i]);
08820be7 414
424e28db 415 const ft = $("floatingTitle");
08820be7 416
1bfe1d7b 417 if (ft && ft.getAttribute("data-article-id") == id) {
424e28db 418 const fte = ft.getElementsByClassName("pubPic");
08820be7 419
560b9fdd 420 for (let i = 0; i < fte.length; i++)
1bfe1d7b
AD
421 imgs.push(fte[i]);
422 }
009646d2 423
560b9fdd 424 for (let i = 0; i < imgs.length; i++) {
424e28db 425 const img = imgs[i];
ee8768db 426
1bfe1d7b
AD
427 if (!row.hasClassName("published") || note != undefined) {
428 img.src = img.src.replace("pub_unset", "pub_set");
560b9fdd 429 query.pub = 1;
1bfe1d7b
AD
430 } else {
431 img.src = img.src.replace("pub_set", "pub_unset");
560b9fdd 432 query.pub = 0;
e4f4b46f 433 }
1bfe1d7b 434 }
b685ba25 435
1bfe1d7b
AD
436 if (note != undefined)
437 row.addClassName("published");
438 else
439 row.toggleClassName("published");
741b6090 440
560b9fdd
AD
441 if (!client_only)
442 xhrPost("backend.php", query, (transport) => {
1bfe1d7b 443 handle_rpc_json(transport);
560b9fdd 444 });
e4f4b46f
AD
445}
446
8b35d171 447function moveToPost(mode, noscroll, noexpand) {
424e28db 448 const rows = getLoadedArticleIds();
f0601b87 449
424e28db
AD
450 let prev_id = false;
451 let next_id = false;
009646d2 452
1bfe1d7b
AD
453 if (!$('RROW-' + getActiveArticleId())) {
454 setActiveArticleId(0);
455 }
009646d2 456
1bfe1d7b
AD
457 if (!getActiveArticleId()) {
458 next_id = rows[0];
459 prev_id = rows[rows.length-1]
460 } else {
424e28db 461 for (let i = 0; i < rows.length; i++) {
1bfe1d7b 462 if (rows[i] == getActiveArticleId()) {
7b5ab2c3 463
1bfe1d7b
AD
464 // Account for adjacent identical article ids.
465 if (i > 0) prev_id = rows[i-1];
7b5ab2c3 466
424e28db 467 for (let j = i+1; j < rows.length; j++) {
1bfe1d7b
AD
468 if (rows[j] != getActiveArticleId()) {
469 next_id = rows[j];
470 break;
7b5ab2c3 471 }
b8e6acea 472 }
1bfe1d7b 473 break;
b8e6acea 474 }
5ad9d132 475 }
1bfe1d7b 476 }
009646d2 477
1bfe1d7b 478 console.log("cur: " + getActiveArticleId() + " next: " + next_id);
009646d2 479
1bfe1d7b
AD
480 if (mode == "next") {
481 if (next_id || getActiveArticleId()) {
482 if (isCdmMode()) {
5ad9d132 483
1bfe1d7b
AD
484 var article = $("RROW-" + getActiveArticleId());
485 var ctr = $("headlines-frame");
104e5c65 486
1bfe1d7b
AD
487 if (!noscroll && article && article.offsetTop + article.offsetHeight >
488 ctr.scrollTop + ctr.offsetHeight) {
104e5c65 489
1bfe1d7b 490 scrollArticle(ctr.offsetHeight/4);
104e5c65
AD
491
492 } else if (next_id) {
1bfe1d7b 493 cdmScrollToArticleId(next_id, true);
b8e6acea 494 }
1bfe1d7b
AD
495
496 } else if (next_id) {
497 correctHeadlinesOffset(next_id);
498 view(next_id, getActiveFeedId(), noexpand);
b8e6acea 499 }
5ad9d132 500 }
1bfe1d7b 501 }
009646d2 502
1bfe1d7b
AD
503 if (mode == "prev") {
504 if (prev_id || getActiveArticleId()) {
505 if (isCdmMode()) {
104e5c65 506
1bfe1d7b 507 var article = $("RROW-" + getActiveArticleId());
424e28db 508 const prev_article = $("RROW-" + prev_id);
1bfe1d7b 509 var ctr = $("headlines-frame");
104e5c65 510
9563e3bc
AD
511 if (!noscroll && article && article.offsetTop < ctr.scrollTop) {
512 scrollArticle(-ctr.offsetHeight/3);
513 } else if (!noscroll && prev_article &&
514 prev_article.offsetTop < ctr.scrollTop) {
515 scrollArticle(-ctr.offsetHeight/4);
516 } else if (prev_id) {
517 cdmScrollToArticleId(prev_id, noscroll);
518 }
1bfe1d7b
AD
519
520 } else if (prev_id) {
521 correctHeadlinesOffset(prev_id);
522 view(prev_id, getActiveFeedId(), noexpand);
5ad9d132 523 }
009646d2 524 }
bb7cface 525 }
1bfe1d7b 526
b8e6acea
AD
527}
528
6f3976c9 529function toggleSelected(id, force_on) {
424e28db 530 const row = $("RROW-" + id);
5ad9d132 531
1bfe1d7b 532 if (row) {
424e28db 533 const cb = dijit.getEnclosingWidget(
1bfe1d7b 534 row.getElementsByClassName("rchk")[0]);
7e27c914 535
1bfe1d7b
AD
536 if (row.hasClassName('Selected') && !force_on) {
537 row.removeClassName('Selected');
538 if (cb) cb.attr("checked", false);
539 } else {
540 row.addClassName('Selected');
541 if (cb) cb.attr("checked", true);
bb7cface 542 }
b8e6acea 543 }
1bfe1d7b
AD
544
545 updateSelectedPrompt();
bb7cface
AD
546}
547
e43a9c4a 548function updateSelectedPrompt() {
4508e310 549 const count = getSelectedArticleIds2().length;
424e28db 550 const elem = $("selected_prompt");
e43a9c4a 551
1bfe1d7b
AD
552 if (elem) {
553 elem.innerHTML = ngettext("%d article selected",
554 "%d articles selected", count).replace("%d", count);
e43a9c4a 555
1bfe1d7b
AD
556 if (count > 0)
557 Element.show(elem);
558 else
559 Element.hide(elem);
e43a9c4a 560 }
1bfe1d7b 561
e43a9c4a
AD
562}
563
560b9fdd 564function toggleUnread(id, cmode) {
424e28db 565 const row = $("RROW-" + id);
1bfe1d7b 566 if (row) {
424e28db 567 const tmpClassName = row.className;
ca8e3d75 568
1bfe1d7b
AD
569 if (cmode == undefined || cmode == 2) {
570 if (row.hasClassName("Unread")) {
ca8e3d75 571 row.removeClassName("Unread");
b8e6acea 572
1bfe1d7b 573 } else {
ca8e3d75 574 row.addClassName("Unread");
5f51022a
AD
575 }
576
1bfe1d7b 577 } else if (cmode == 0) {
933ba4ee 578
1bfe1d7b 579 row.removeClassName("Unread");
933ba4ee 580
1bfe1d7b
AD
581 } else if (cmode == 1) {
582 row.addClassName("Unread");
933ba4ee
AD
583 }
584
560b9fdd
AD
585 if (tmpClassName != row.className) {
586 if (cmode == undefined) cmode = 2;
933ba4ee 587
560b9fdd
AD
588 const query = {op: "rpc", method: "catchupSelected",
589 cmode: cmode, ids: id};
933ba4ee 590
560b9fdd 591 xhrPost("backend.php", query, (transport) => {
1bfe1d7b 592 handle_rpc_json(transport);
560b9fdd 593
1bfe1d7b
AD
594 });
595 }
933ba4ee
AD
596 }
597}
598
1bfe1d7b
AD
599function selectionRemoveLabel(id, ids) {
600 if (!ids) ids = getSelectedArticleIds2();
b8a637f3 601
1bfe1d7b
AD
602 if (ids.length == 0) {
603 alert(__("No articles are selected."));
604 return;
605 }
b8a637f3 606
560b9fdd
AD
607 const query = { op: "article", method: "removeFromLabel",
608 ids: ids.toString(), lid: id };
b8a637f3 609
560b9fdd
AD
610 xhrPost("backend.php", query, (transport) => {
611 handle_rpc_json(transport);
612 show_labels_in_headlines(transport);
613 });
1bfe1d7b 614}
b8a637f3 615
1bfe1d7b
AD
616function selectionAssignLabel(id, ids) {
617 if (!ids) ids = getSelectedArticleIds2();
b8a637f3 618
1bfe1d7b
AD
619 if (ids.length == 0) {
620 alert(__("No articles are selected."));
621 return;
b8a637f3 622 }
1bfe1d7b 623
560b9fdd
AD
624 const query = { op: "article", method: "assignToLabel",
625 ids: ids.toString(), lid: id };
1bfe1d7b 626
560b9fdd
AD
627 xhrPost("backend.php", query, (transport) => {
628 handle_rpc_json(transport);
629 show_labels_in_headlines(transport);
630 });
b8a637f3
AD
631}
632
be7bb7d5 633function selectionToggleUnread(set_state, callback, no_error, ids) {
424e28db 634 const rows = ids ? ids : getSelectedArticleIds2();
9cc600d1 635
1bfe1d7b
AD
636 if (rows.length == 0 && !no_error) {
637 alert(__("No articles are selected."));
638 return;
639 }
a5ae125a 640
424e28db
AD
641 for (let i = 0; i < rows.length; i++) {
642 const row = $("RROW-" + rows[i]);
1bfe1d7b
AD
643 if (row) {
644 if (set_state == undefined) {
645 if (row.hasClassName("Unread")) {
ca8e3d75 646 row.removeClassName("Unread");
1bfe1d7b 647 } else {
ca8e3d75 648 row.addClassName("Unread");
f1f2db64
AD
649 }
650 }
1572afe5 651
1bfe1d7b
AD
652 if (set_state == false) {
653 row.removeClassName("Unread");
654 }
b47b5af7 655
1bfe1d7b
AD
656 if (set_state == true) {
657 row.addClassName("Unread");
b47b5af7 658 }
1bfe1d7b
AD
659 }
660 }
b47b5af7 661
1bfe1d7b 662 updateFloatingTitle(true);
1572afe5 663
1bfe1d7b 664 if (rows.length > 0) {
9ec58704 665
424e28db 666 let cmode = "";
1572afe5 667
1bfe1d7b
AD
668 if (set_state == undefined) {
669 cmode = "2";
670 } else if (set_state == true) {
671 cmode = "1";
672 } else if (set_state == false) {
673 cmode = "0";
1572afe5
AD
674 }
675
560b9fdd
AD
676 const query = {op: "rpc", method: "catchupSelected",
677 cmode: cmode, ids: rows.toString() };
1bfe1d7b
AD
678
679 notify_progress("Loading, please wait...");
680
560b9fdd
AD
681 xhrPost("backend.php", query, (transport) => {
682 handle_rpc_json(transport);
683 if (callback) callback(transport);
684 });
1bfe1d7b 685
1572afe5
AD
686 }
687}
688
be7bb7d5
AD
689// sel_state ignored
690function selectionToggleMarked(sel_state, callback, no_error, ids) {
424e28db 691 const rows = ids ? ids : getSelectedArticleIds2();
009646d2 692
1bfe1d7b
AD
693 if (rows.length == 0 && !no_error) {
694 alert(__("No articles are selected."));
695 return;
696 }
1572afe5 697
424e28db 698 for (let i = 0; i < rows.length; i++) {
1bfe1d7b
AD
699 toggleMark(rows[i], true, true);
700 }
1572afe5 701
1bfe1d7b 702 if (rows.length > 0) {
560b9fdd
AD
703 const query = { op: "rpc", method: "markSelected",
704 ids: rows.toString(), cmode: 2 };
1572afe5 705
560b9fdd
AD
706 xhrPost("backend.php", query, (transport) => {
707 handle_rpc_json(transport);
708 if (callback) callback(transport);
709 });
1572afe5
AD
710 }
711}
712
be7bb7d5
AD
713// sel_state ignored
714function selectionTogglePublished(sel_state, callback, no_error, ids) {
424e28db 715 const rows = ids ? ids : getSelectedArticleIds2();
e4f4b46f 716
1bfe1d7b
AD
717 if (rows.length == 0 && !no_error) {
718 alert(__("No articles are selected."));
719 return;
720 }
e4f4b46f 721
424e28db 722 for (let i = 0; i < rows.length; i++) {
1bfe1d7b
AD
723 togglePub(rows[i], true, true);
724 }
e4f4b46f 725
1bfe1d7b 726 if (rows.length > 0) {
560b9fdd
AD
727 const query = { op: "rpc", method: "publishSelected",
728 ids: rows.toString(), cmode: 2 };
e4f4b46f 729
560b9fdd
AD
730 xhrPost("backend.php", query, (transport) => {
731 handle_rpc_json(transport);
732 if (callback) callback(transport);
733 });
e4f4b46f
AD
734 }
735}
736
e69fb880 737function getSelectedArticleIds2() {
386cbf27 738
424e28db 739 const rv = [];
e69fb880 740
e5df6e9e 741 $$("#headlines-frame > div[id*=RROW][class*=Selected]").each(
ca8e3d75 742 function(child) {
b6b5554d 743 rv.push(child.getAttribute("data-article-id"));
ca8e3d75 744 });
386cbf27 745
ca8e3d75 746 return rv;
386cbf27
AD
747}
748
e69fb880 749function getLoadedArticleIds() {
424e28db 750 const rv = [];
8be83f42 751
424e28db 752 const children = $$("#headlines-frame > div[id*=RROW-]");
3a40e8a2 753
ca8e3d75 754 children.each(function(child) {
2d052e42 755 if (Element.visible(child)) {
b6b5554d 756 rv.push(child.getAttribute("data-article-id"));
2d052e42
AD
757 }
758 });
8be83f42 759
ca8e3d75 760 return rv;
8be83f42 761
8be83f42
AD
762}
763
5482b388 764// mode = all,none,unread,invert,marked,published
87065739 765function selectArticles(mode, query) {
1bfe1d7b 766 if (!query) query = "#headlines-frame > div[id*=RROW]";
87065739 767
424e28db 768 const children = $$(query);
06d1a1c1 769
1bfe1d7b 770 children.each(function(child) {
560b9fdd 771 //const id = child.getAttribute("data-article-id");
7e27c914 772
424e28db 773 const cb = dijit.getEnclosingWidget(
1bfe1d7b 774 child.getElementsByClassName("rchk")[0]);
06d1a1c1 775
1bfe1d7b
AD
776 if (mode == "all") {
777 child.addClassName("Selected");
778 if (cb) cb.attr("checked", true);
779 } else if (mode == "unread") {
780 if (child.hasClassName("Unread")) {
ca8e3d75 781 child.addClassName("Selected");
524de8dc 782 if (cb) cb.attr("checked", true);
ca8e3d75
AD
783 } else {
784 child.removeClassName("Selected");
524de8dc 785 if (cb) cb.attr("checked", false);
ca8e3d75 786 }
1bfe1d7b
AD
787 } else if (mode == "marked") {
788 if (child.hasClassName("marked")) {
789 child.addClassName("Selected");
790 if (cb) cb.attr("checked", true);
791 } else {
792 child.removeClassName("Selected");
793 if (cb) cb.attr("checked", false);
794 }
795 } else if (mode == "published") {
796 if (child.hasClassName("published")) {
797 child.addClassName("Selected");
798 if (cb) cb.attr("checked", true);
799 } else {
800 child.removeClassName("Selected");
801 if (cb) cb.attr("checked", false);
802 }
803
804 } else if (mode == "invert") {
805 if (child.hasClassName("Selected")) {
806 child.removeClassName("Selected");
807 if (cb) cb.attr("checked", false);
808 } else {
809 child.addClassName("Selected");
810 if (cb) cb.attr("checked", true);
811 }
e69fb880 812
1bfe1d7b
AD
813 } else {
814 child.removeClassName("Selected");
815 if (cb) cb.attr("checked", false);
816 }
817 });
e43a9c4a 818
1bfe1d7b 819 updateSelectedPrompt();
386cbf27
AD
820}
821
e04c18a2
AD
822function deleteSelection() {
823
424e28db 824 const rows = getSelectedArticleIds2();
009646d2 825
1bfe1d7b
AD
826 if (rows.length == 0) {
827 alert(__("No articles are selected."));
828 return;
829 }
009646d2 830
424e28db
AD
831 const fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
832 let str;
009646d2 833
1bfe1d7b
AD
834 if (getActiveFeedId() != 0) {
835 str = ngettext("Delete %d selected article in %s?", "Delete %d selected articles in %s?", rows.length);
836 } else {
837 str = ngettext("Delete %d selected article?", "Delete %d selected articles?", rows.length);
838 }
009646d2 839
1bfe1d7b
AD
840 str = str.replace("%d", rows.length);
841 str = str.replace("%s", fn);
e04c18a2 842
1bfe1d7b
AD
843 if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
844 return;
845 }
e04c18a2 846
eaf7cfdb 847 const query = { op: "rpc", method: "delete", ids: rows.toString() };
e04c18a2 848
eaf7cfdb
AD
849 xhrPost("backend.php", query, (transport) => {
850 handle_rpc_json(transport);
851 viewCurrentFeed();
1bfe1d7b 852 });
e04c18a2
AD
853}
854
855function archiveSelection() {
856
eaf7cfdb 857 const rows = getSelectedArticleIds2();
009646d2 858
eaf7cfdb
AD
859 if (rows.length == 0) {
860 alert(__("No articles are selected."));
861 return;
862 }
b029f916 863
eaf7cfdb
AD
864 const fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
865 let str;
866 let op;
b029f916 867
eaf7cfdb
AD
868 if (getActiveFeedId() != 0) {
869 str = ngettext("Archive %d selected article in %s?", "Archive %d selected articles in %s?", rows.length);
870 op = "archive";
871 } else {
872 str = ngettext("Move %d archived article back?", "Move %d archived articles back?", rows.length);
009646d2 873
eaf7cfdb 874 str += " " + __("Please note that unstarred articles might get purged on next feed update.");
009646d2 875
eaf7cfdb
AD
876 op = "unarchive";
877 }
e04c18a2 878
eaf7cfdb
AD
879 str = str.replace("%d", rows.length);
880 str = str.replace("%s", fn);
e04c18a2 881
eaf7cfdb
AD
882 if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
883 return;
884 }
e04c18a2 885
eaf7cfdb
AD
886 for (let i = 0; i < rows.length; i++) {
887 cache_delete("article:" + rows[i]);
888 }
a12eb9c3 889
eaf7cfdb 890 const query = {op: "rpc", method: op, ids: rows.toString()};
e04c18a2 891
eaf7cfdb
AD
892 xhrPost("backend.php", query, (transport) => {
893 handle_rpc_json(transport);
894 viewCurrentFeed();
895 });
e04c18a2
AD
896}
897
a5ae125a
AD
898function catchupSelection() {
899
424e28db 900 const rows = getSelectedArticleIds2();
009646d2 901
1bfe1d7b
AD
902 if (rows.length == 0) {
903 alert(__("No articles are selected."));
904 return;
905 }
009646d2 906
424e28db 907 const fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
009646d2 908
424e28db 909 let str = ngettext("Mark %d selected article in %s as read?", "Mark %d selected articles in %s as read?", rows.length);
009646d2 910
1bfe1d7b
AD
911 str = str.replace("%d", rows.length);
912 str = str.replace("%s", fn);
a5ae125a 913
1bfe1d7b
AD
914 if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
915 return;
a5ae125a 916 }
1bfe1d7b
AD
917
918 selectionToggleUnread(false, 'viewCurrentFeed()', true);
a5ae125a
AD
919}
920
8386f861 921function editArticleTags(id) {
424e28db 922 const query = "backend.php?op=article&method=editArticleTags&param=" + param_escape(id);
88040f57 923
1bfe1d7b
AD
924 if (dijit.byId("editTagsDlg"))
925 dijit.byId("editTagsDlg").destroyRecursive();
0b126ac2 926
9dc5524d 927 const dialog = new dijit.Dialog({
1bfe1d7b
AD
928 id: "editTagsDlg",
929 title: __("Edit article Tags"),
930 style: "width: 600px",
931 execute: function() {
932 if (this.validate()) {
424e28db 933 const query = dojo.objectToQuery(this.attr('value'));
0b126ac2 934
1bfe1d7b 935 notify_progress("Saving article tags...", true);
14b6c54b 936
eaf7cfdb
AD
937 xhrPost("backend.php", this.attr('value'), (transport) => {
938 try {
939 notify('');
940 dialog.hide();
009646d2 941
eaf7cfdb 942 const data = JSON.parse(transport.responseText);
ddcbbea2 943
eaf7cfdb
AD
944 if (data) {
945 const id = data.id;
ddcbbea2 946
eaf7cfdb
AD
947 const tags = $("ATSTR-" + id);
948 const tooltip = dijit.byId("ATSTRTIP-" + id);
ddcbbea2 949
eaf7cfdb
AD
950 if (tags) tags.innerHTML = data.content;
951 if (tooltip) tooltip.attr('label', data.content_full);
952 }
953 } catch (e) {
954 exception_error(e);
955 }
956 });
1bfe1d7b
AD
957 }
958 },
959 href: query
960 });
d62a3b63 961
1bfe1d7b
AD
962 var tmph = dojo.connect(dialog, 'onLoad', function() {
963 dojo.disconnect(tmph);
d62a3b63 964
1bfe1d7b
AD
965 new Ajax.Autocompleter('tags_str', 'tags_choices',
966 "backend.php?op=article&method=completeTags",
967 { tokens: ',', paramName: "search" });
968 });
d62a3b63 969
1bfe1d7b 970 dialog.show();
d62a3b63 971
d62a3b63 972}
ba0978c8 973
18e1a773 974function cdmScrollToArticleId(id, force) {
424e28db
AD
975 const ctr = $("headlines-frame");
976 const e = $("RROW-" + id);
b8e6acea 977
1bfe1d7b 978 if (!e || !ctr) return;
c409d34a 979
1bfe1d7b
AD
980 if (force || e.offsetTop+e.offsetHeight > (ctr.scrollTop+ctr.offsetHeight) ||
981 e.offsetTop < ctr.scrollTop) {
b8e6acea 982
1bfe1d7b
AD
983 // expanded cdm has a 4px margin now
984 ctr.scrollTop = parseInt(e.offsetTop) - 4;
b8e6acea
AD
985 }
986}
987
18e1a773 988function setActiveArticleId(id) {
eaec06f9
AD
989 console.log("setActiveArticleId:" + id);
990
18e1a773 991 _active_article_id = id;
8d388f32 992 PluginHost.run(PluginHost.HOOK_ARTICLE_SET_ACTIVE, _active_article_id);
18e1a773
AD
993}
994
298f3f78 995function getActiveArticleId() {
18e1a773 996 return _active_article_id;
298f3f78 997}
e4914b62 998
d2f3467b 999function postMouseIn(e, id) {
78b2c6ce 1000 post_under_pointer = id;
314fcd2b
AD
1001}
1002
1003function postMouseOut(id) {
78b2c6ce 1004 post_under_pointer = false;
314fcd2b
AD
1005}
1006
62b1f587 1007function unpackVisibleHeadlines() {
9563e3bc 1008 if (!isCdmMode()) return;
44018133 1009
0c06bb5f 1010 const rows = $$("#headlines-frame div[id*=RROW][data-content]");
ad326dbf 1011
0c06bb5f
AD
1012 for (let i = 0; i < rows.length; i++) {
1013 const row = rows[i];
1014
1015 if (row.offsetTop <= $("headlines-frame").scrollTop + $("headlines-frame").offsetHeight) {
c8c9a26f 1016 console.log("unpacking: " + row.id);
b74c5134 1017
b9585004 1018 const content = row.getAttribute("data-content");
d8e8e24e 1019
0c06bb5f 1020 row.select(".cdmContentInner")[0].innerHTML = content;
c8c9a26f 1021 row.removeAttribute("data-content");
d8e8e24e 1022
c8c9a26f 1023 PluginHost.run(PluginHost.HOOK_ARTICLE_RENDERED_CDM, row);
0c06bb5f
AD
1024 } else {
1025 break;
1026 }
1027 }
62b1f587
AD
1028}
1029
1030function headlines_scroll_handler(e) {
1031 try {
63c7446a
AD
1032
1033 // rate-limit in case of smooth scrolling and similar abominations
d7cc5e6c 1034 if (Math.max(e.scrollTop, _headlines_scroll_offset) - Math.min(e.scrollTop, _headlines_scroll_offset) < 25) {
63c7446a
AD
1035 return;
1036 }
1037
1038 _headlines_scroll_offset = e.scrollTop;
1039
62b1f587
AD
1040 unpackVisibleHeadlines();
1041
48be35ca 1042 // set topmost child in the buffer as active
2e35b3bd 1043 if (isCdmMode() && getInitParam("cdm_auto_catchup") == 1 &&
9563e3bc 1044 getSelectedArticleIds2().length <= 1) {
63c7446a 1045
424e28db 1046 const rows = $$("#headlines-frame > div[id*=RROW]");
48be35ca 1047
424e28db 1048 for (let i = 0; i < rows.length; i++) {
0c06bb5f 1049 const row = rows[i];
48be35ca 1050
0c06bb5f
AD
1051 if ($("headlines-frame").scrollTop <= row.offsetTop &&
1052 row.offsetTop - $("headlines-frame").scrollTop < 100 &&
1053 row.getAttribute("data-article-id") != _active_article_id) {
99d89d10 1054
48be35ca 1055 if (_active_article_id) {
424e28db 1056 const row = $("RROW-" + _active_article_id);
48be35ca
AD
1057 if (row) row.removeClassName("active");
1058 }
1059
0c06bb5f 1060 _active_article_id = row.getAttribute("data-article-id");
48be35ca 1061 showArticleInHeadlines(_active_article_id, true);
4bc6489a 1062 updateSelectedPrompt();
48be35ca
AD
1063 break;
1064 }
1065 }
1066 }
1067
18eb64a8 1068 if (!_infscroll_disable) {
0c06bb5f
AD
1069 const hsp = $("headlines-spacer");
1070
eaec06f9 1071 if (hsp && hsp.offsetTop - 250 <= e.scrollTop + e.offsetHeight) {
ac541432 1072
d46af5a5
AD
1073 hsp.innerHTML = "<span class='loading'><img src='images/indicator_tiny.gif'> " +
1074 __("Loading, please wait...") + "</span>";
71317973
AD
1075
1076 loadMoreHeadlines();
79ddf6aa 1077 return;
71317973 1078
29dfb258 1079 }
ac541432
AD
1080 }
1081
7415fcf2 1082 if (isCdmMode()) {
08820be7
AD
1083 updateFloatingTitle();
1084 }
1085
ce68ec74 1086 if (getInitParam("cdm_auto_catchup") == 1) {
21426d3a 1087
0c06bb5f 1088 let rows = $$("#headlines-frame > div[id*=RROW][class*=Unread]");
21426d3a 1089
0c06bb5f
AD
1090 for (let i = 0; i < rows.length; i++) {
1091 const row = rows[i];
1092
1093 if ($("headlines-frame").scrollTop > (row.offsetTop + row.offsetHeight/2)) {
76495dfd 1094
0c06bb5f 1095 const id = row.getAttribute("data-article-id")
79ddf6aa 1096
0c06bb5f
AD
1097 if (catchup_id_batch.indexOf(id) == -1)
1098 catchup_id_batch.push(id);
48be35ca 1099
0c06bb5f
AD
1100 //console.log("auto_catchup_batch: " + catchup_id_batch.toString());
1101 } else {
1102 break;
1103 }
1104 }
21426d3a 1105
0f85f483 1106 if (_infscroll_disable) {
0c06bb5f 1107 const row = $$("#headlines-frame div[id*=RROW]").last();
b7c20a37 1108
0c06bb5f
AD
1109 if (row && $("headlines-frame").scrollTop >
1110 (row.offsetTop + row.offsetHeight - 50)) {
b7c20a37
AD
1111
1112 console.log("we seem to be at an end");
1113
1114 if (getInitParam("on_catchup_show_next_feed") == "1") {
22f675e5 1115 openNextUnreadFeed();
b7c20a37
AD
1116 }
1117 }
76495dfd
AD
1118 }
1119 }
21426d3a 1120
76495dfd 1121 } catch (e) {
37c03d3a 1122 console.warn("headlines_scroll_handler: " + e);
76495dfd
AD
1123 }
1124}
009646d2 1125
22f675e5 1126function openNextUnreadFeed() {
424e28db
AD
1127 const is_cat = activeFeedIsCat();
1128 const nuf = getNextUnreadFeed(getActiveFeedId(), is_cat);
1bfe1d7b 1129 if (nuf) viewfeed({feed: nuf, is_cat: is_cat});
22f675e5
AD
1130}
1131
76495dfd 1132function catchupBatchedArticles() {
1bfe1d7b 1133 if (catchup_id_batch.length > 0 && !_infscroll_request_sent && !_catchup_request_sent) {
009646d2 1134
0c06bb5f 1135 console.log("catchupBatchedArticles, size=", catchup_id_batch.length);
0a11d235 1136
1bfe1d7b 1137 // make a copy of the array
424e28db 1138 const batch = catchup_id_batch.slice();
eaf7cfdb
AD
1139 const query = { op: "rpc", method: "catchupSelected",
1140 cmode: 0, ids: batch.toString() };
8292123e 1141
1bfe1d7b 1142 _catchup_request_sent = true;
76495dfd 1143
eaf7cfdb
AD
1144 xhrPost("backend.php", query, (transport) => {
1145 const reply = handle_rpc_json(transport);
8292123e 1146
eaf7cfdb 1147 _catchup_request_sent = false;
ae31704b 1148
eaf7cfdb
AD
1149 if (reply) {
1150 const batch = reply.ids;
76495dfd 1151
eaf7cfdb 1152 batch.each(function (id) {
eaf7cfdb
AD
1153 const elem = $("RROW-" + id);
1154 if (elem) elem.removeClassName("Unread");
1155 catchup_id_batch.remove(id);
1156 });
1157 }
76495dfd 1158
eaf7cfdb 1159 updateFloatingTitle(true);
1bfe1d7b 1160 });
ac541432
AD
1161 }
1162}
8be83f42 1163
2ea34cc1 1164function catchupRelativeToArticle(below, id) {
8be83f42 1165
1bfe1d7b 1166 if (!id) id = getActiveArticleId();
49fd1e94 1167
1bfe1d7b
AD
1168 if (!id) {
1169 alert(__("No article is selected."));
1170 return;
1171 }
8be83f42 1172
424e28db 1173 const visible_ids = getLoadedArticleIds();
8be83f42 1174
4508e310 1175 const ids_to_mark = [];
8be83f42 1176
1bfe1d7b
AD
1177 if (!below) {
1178 for (var i = 0; i < visible_ids.length; i++) {
1179 if (visible_ids[i] != id) {
1180 var e = $("RROW-" + visible_ids[i]);
8be83f42 1181
1bfe1d7b
AD
1182 if (e && e.hasClassName("Unread")) {
1183 ids_to_mark.push(visible_ids[i]);
8be83f42 1184 }
1bfe1d7b
AD
1185 } else {
1186 break;
8be83f42 1187 }
1bfe1d7b
AD
1188 }
1189 } else {
1190 for (var i = visible_ids.length - 1; i >= 0; i--) {
1191 if (visible_ids[i] != id) {
1192 var e = $("RROW-" + visible_ids[i]);
8be83f42 1193
1bfe1d7b
AD
1194 if (e && e.hasClassName("Unread")) {
1195 ids_to_mark.push(visible_ids[i]);
8be83f42 1196 }
1bfe1d7b
AD
1197 } else {
1198 break;
8be83f42
AD
1199 }
1200 }
1bfe1d7b 1201 }
8be83f42 1202
1bfe1d7b
AD
1203 if (ids_to_mark.length == 0) {
1204 alert(__("No articles found to mark"));
1205 } else {
424e28db 1206 const msg = ngettext("Mark %d article as read?", "Mark %d articles as read?", ids_to_mark.length).replace("%d", ids_to_mark.length);
8be83f42 1207
1bfe1d7b 1208 if (getInitParam("confirm_feed_catchup") != 1 || confirm(msg)) {
8be83f42 1209
1bfe1d7b
AD
1210 for (var i = 0; i < ids_to_mark.length; i++) {
1211 var e = $("RROW-" + ids_to_mark[i]);
1212 e.removeClassName("Unread");
1213 }
8be83f42 1214
eaf7cfdb
AD
1215 const query = { op: "rpc", method: "catchupSelected",
1216 cmode: 0, ids: ids_to_mark.toString() };
8be83f42 1217
eaf7cfdb
AD
1218 xhrPost("backend.php", query, (transport) => {
1219 handle_rpc_json(transport);
1bfe1d7b 1220 });
8be83f42 1221 }
8be83f42
AD
1222 }
1223}
a04c8e8d 1224
314fcd2b
AD
1225function getArticleUnderPointer() {
1226 return post_under_pointer;
7a822893 1227}
eedfb635 1228
a411875b 1229function scrollArticle(offset) {
1bfe1d7b 1230 if (!isCdmMode()) {
424e28db 1231 const ci = $("content-insert");
1bfe1d7b
AD
1232 if (ci) {
1233 ci.scrollTop += offset;
a411875b 1234 }
1bfe1d7b 1235 } else {
424e28db 1236 const hi = $("headlines-frame");
1bfe1d7b
AD
1237 if (hi) {
1238 hi.scrollTop += offset;
1239 }
1240
a411875b
AD
1241 }
1242}
f9247195
AD
1243
1244function show_labels_in_headlines(transport) {
424e28db 1245 const data = JSON.parse(transport.responseText);
f9247195 1246
1bfe1d7b
AD
1247 if (data) {
1248 data['info-for-headlines'].each(function (elem) {
1249 $$(".HLLCTR-" + elem.id).each(function (ctr) {
1250 ctr.innerHTML = elem.labels;
8eb592ec 1251 });
1bfe1d7b 1252 });
f9247195
AD
1253 }
1254}
bf3c9838 1255
63f0ed3d 1256function cdmClicked(event, id, in_body) {
1bfe1d7b 1257 //var shift_key = event.shiftKey;
1ede5814 1258
1bfe1d7b 1259 if (!event.ctrlKey && !event.metaKey) {
62b800b4 1260
9563e3bc 1261 let elem = $("RROW-" + getActiveArticleId());
009646d2 1262
9563e3bc 1263 if (elem) elem.removeClassName("active");
009646d2 1264
9563e3bc
AD
1265 selectArticles("none");
1266 toggleSelected(id);
009646d2 1267
9563e3bc
AD
1268 elem = $("RROW-" + id);
1269 const article_is_unread = elem.hasClassName("Unread");
009646d2 1270
9563e3bc
AD
1271 elem.removeClassName("Unread");
1272 elem.addClassName("active");
9bfb11ed 1273
9563e3bc 1274 setActiveArticleId(id);
62b800b4 1275
9563e3bc
AD
1276 if (article_is_unread) {
1277 decrementFeedCounter(getActiveFeedId(), activeFeedIsCat());
1278 updateFloatingTitle(true);
bace5e9e 1279
9563e3bc
AD
1280 const query = {
1281 op: "rpc", method: "catchupSelected",
1282 cmode: 0, ids: id
1283 };
de44b9a6 1284
9563e3bc
AD
1285 xhrPost("backend.php", query, (transport) => {
1286 handle_rpc_json(transport);
1287 });
1bfe1d7b 1288 }
de44b9a6 1289
9563e3bc
AD
1290 return !event.shiftKey;
1291
63f0ed3d 1292 } else if (!in_body) {
de44b9a6 1293
1bfe1d7b 1294 toggleSelected(id, true);
38325ad6 1295
4508e310
AD
1296 let elem = $("RROW-" + id);
1297 const article_is_unread = elem.hasClassName("Unread");
1bfe1d7b
AD
1298
1299 if (article_is_unread) {
1300 decrementFeedCounter(getActiveFeedId(), activeFeedIsCat());
1ede5814
AD
1301 }
1302
1bfe1d7b 1303 toggleUnread(id, 0, false);
5b55e9e2 1304
1bfe1d7b 1305 openArticleInNewWindow(id);
63f0ed3d
AD
1306 } else {
1307 return true;
1ede5814
AD
1308 }
1309
424e28db 1310 const unread_in_buffer = $$("#headlines-frame > div[id*=RROW][class*=Unread]").length
1bfe1d7b
AD
1311 request_counters(unread_in_buffer == 0);
1312
1ede5814
AD
1313 return false;
1314}
1315
1316function hlClicked(event, id) {
1bfe1d7b
AD
1317 if (event.which == 2) {
1318 view(id);
1319 return true;
1320 } else if (event.ctrlKey || event.metaKey) {
1bfe1d7b
AD
1321 openArticleInNewWindow(id);
1322 return false;
1323 } else {
1324 view(id);
1325 return false;
1ede5814 1326 }
1ede5814 1327}
e69fb880 1328
e69fb880 1329function openArticleInNewWindow(id) {
b509d64e 1330 toggleUnread(id, 0, false);
829d478f 1331
424e28db 1332 const w = window.open("");
829d478f
AD
1333 w.opener = null;
1334 w.location = "backend.php?op=article&method=redirect&id=" + id;
e69fb880
AD
1335}
1336
1337function isCdmMode() {
8cc5e965 1338 return getInitParam("combined_display_mode");
e69fb880
AD
1339}
1340
414191d4
AD
1341function markHeadline(id, marked) {
1342 if (marked == undefined) marked = true;
1343
424e28db 1344 const row = $("RROW-" + id);
e69fb880 1345 if (row) {
424e28db 1346 const check = dijit.getEnclosingWidget(
7e27c914 1347 row.getElementsByClassName("rchk")[0]);
e69fb880
AD
1348
1349 if (check) {
414191d4 1350 check.attr("checked", marked);
e69fb880
AD
1351 }
1352
3ac153f1
AD
1353 if (marked)
1354 row.addClassName("Selected");
1355 else
1356 row.removeClassName("Selected");
e69fb880
AD
1357 }
1358}
1359
8cc5e965
AD
1360function getRelativePostIds(id, limit) {
1361
424e28db 1362 const tmp = [];
8cc5e965 1363
1bfe1d7b 1364 if (!limit) limit = 6; //3
009646d2 1365
424e28db 1366 const ids = getLoadedArticleIds();
009646d2 1367
424e28db 1368 for (let i = 0; i < ids.length; i++) {
1bfe1d7b 1369 if (ids[i] == id) {
424e28db 1370 for (let k = 1; k <= limit; k++) {
1bfe1d7b
AD
1371 //if (i > k-1) tmp.push(ids[i-k]);
1372 if (i < ids.length - k) tmp.push(ids[i + k]);
8cc5e965 1373 }
1bfe1d7b 1374 break;
8cc5e965 1375 }
8cc5e965
AD
1376 }
1377
1378 return tmp;
1379}
1380
1381function correctHeadlinesOffset(id) {
009646d2 1382
424e28db
AD
1383 const container = $("headlines-frame");
1384 const row = $("RROW-" + id);
e4f48f86 1385
1bfe1d7b 1386 if (!container || !row) return;
009646d2 1387
424e28db 1388 const viewport = container.offsetHeight;
009646d2 1389
424e28db
AD
1390 const rel_offset_top = row.offsetTop - container.scrollTop;
1391 const rel_offset_bottom = row.offsetTop + row.offsetHeight - container.scrollTop;
8cc5e965 1392
1bfe1d7b
AD
1393 //console.log("Rtop: " + rel_offset_top + " Rbtm: " + rel_offset_bottom);
1394 //console.log("Vport: " + viewport);
8cc5e965 1395
1bfe1d7b
AD
1396 if (rel_offset_top <= 0 || rel_offset_top > viewport) {
1397 container.scrollTop = row.offsetTop;
1398 } else if (rel_offset_bottom > viewport) {
8cc5e965 1399
1bfe1d7b
AD
1400 /* doesn't properly work with Opera in some cases because
1401 Opera fucks up element scrolling */
8cc5e965 1402
1bfe1d7b 1403 container.scrollTop = row.offsetTop + row.offsetHeight - viewport;
8cc5e965 1404 }
8cc5e965
AD
1405}
1406
fcf70c51 1407function headlineActionsChange(elem) {
1bfe1d7b
AD
1408 eval(elem.value);
1409 elem.attr('value', 'false');
fcf70c51 1410}
6f3976c9 1411
0b461ed5 1412function closeArticlePanel() {
e3387e2d 1413
38325ad6
AD
1414 if (dijit.byId("content-insert"))
1415 dijit.byId("headlines-wrap-inner").removeChild(
1416 dijit.byId("content-insert"));
6f3976c9 1417}
b509d64e 1418
65f0eb01 1419function initFloatingMenu() {
1bfe1d7b 1420 if (!dijit.byId("floatingMenu")) {
b509d64e 1421
424e28db 1422 const menu = new dijit.Menu({
1bfe1d7b
AD
1423 id: "floatingMenu",
1424 targetNodeIds: ["floatingTitle"]
1425 });
b509d64e 1426
1bfe1d7b 1427 headlinesMenuCommon(menu);
c4e21104 1428
1bfe1d7b 1429 menu.startup();
65f0eb01
AD
1430 }
1431}
b509d64e 1432
c4e21104 1433function headlinesMenuCommon(menu) {
7fc2e87e 1434
1bfe1d7b
AD
1435 menu.addChild(new dijit.MenuItem({
1436 label: __("Open original article"),
1437 onClick: function (event) {
1438 openArticleInNewWindow(this.getParent().currentTarget.getAttribute("data-article-id"));
1439 }
1440 }));
2ea34cc1 1441
1bfe1d7b
AD
1442 menu.addChild(new dijit.MenuItem({
1443 label: __("Display article URL"),
1444 onClick: function (event) {
1445 displayArticleUrl(this.getParent().currentTarget.getAttribute("data-article-id"));
1446 }
1447 }));
b835a528 1448
1bfe1d7b 1449 menu.addChild(new dijit.MenuSeparator());
be7bb7d5 1450
1bfe1d7b
AD
1451 menu.addChild(new dijit.MenuItem({
1452 label: __("Toggle unread"),
4508e310 1453 onClick: function () {
be7bb7d5 1454
424e28db 1455 let ids = getSelectedArticleIds2();
1bfe1d7b 1456 // cast to string
424e28db 1457 const id = (this.getParent().currentTarget.getAttribute("data-article-id")) + "";
4508e310 1458 ids = ids.length != 0 && ids.indexOf(id) != -1 ? ids : [id];
be7bb7d5 1459
1bfe1d7b
AD
1460 selectionToggleUnread(undefined, false, true, ids);
1461 }
1462 }));
be7bb7d5 1463
1bfe1d7b
AD
1464 menu.addChild(new dijit.MenuItem({
1465 label: __("Toggle starred"),
4508e310 1466 onClick: function () {
424e28db 1467 let ids = getSelectedArticleIds2();
1bfe1d7b 1468 // cast to string
424e28db 1469 const id = (this.getParent().currentTarget.getAttribute("data-article-id")) + "";
4508e310 1470 ids = ids.length != 0 && ids.indexOf(id) != -1 ? ids : [id];
be7bb7d5 1471
1bfe1d7b
AD
1472 selectionToggleMarked(undefined, false, true, ids);
1473 }
1474 }));
be7bb7d5 1475
1bfe1d7b
AD
1476 menu.addChild(new dijit.MenuItem({
1477 label: __("Toggle published"),
4508e310 1478 onClick: function () {
424e28db 1479 let ids = getSelectedArticleIds2();
1bfe1d7b 1480 // cast to string
424e28db 1481 const id = (this.getParent().currentTarget.getAttribute("data-article-id")) + "";
4508e310 1482 ids = ids.length != 0 && ids.indexOf(id) != -1 ? ids : [id];
be7bb7d5 1483
1bfe1d7b
AD
1484 selectionTogglePublished(undefined, false, true, ids);
1485 }
1486 }));
2ea34cc1 1487
1bfe1d7b 1488 menu.addChild(new dijit.MenuSeparator());
2ea34cc1 1489
1bfe1d7b
AD
1490 menu.addChild(new dijit.MenuItem({
1491 label: __("Mark above as read"),
4508e310 1492 onClick: function () {
1bfe1d7b
AD
1493 catchupRelativeToArticle(0, this.getParent().currentTarget.getAttribute("data-article-id"));
1494 }
1495 }));
10690c19 1496
1bfe1d7b
AD
1497 menu.addChild(new dijit.MenuItem({
1498 label: __("Mark below as read"),
4508e310 1499 onClick: function () {
1bfe1d7b
AD
1500 catchupRelativeToArticle(1, this.getParent().currentTarget.getAttribute("data-article-id"));
1501 }
1502 }));
1beea800 1503
1beea800 1504
424e28db 1505 const labels = getInitParam("labels");
1beea800 1506
b5fc9781 1507 if (labels && labels.length) {
1beea800 1508
1bfe1d7b 1509 menu.addChild(new dijit.MenuSeparator());
1beea800 1510
424e28db
AD
1511 const labelAddMenu = new dijit.Menu({ownerMenu: menu});
1512 const labelDelMenu = new dijit.Menu({ownerMenu: menu});
1beea800 1513
1bfe1d7b 1514 labels.each(function (label) {
424e28db
AD
1515 const bare_id = label.id;
1516 const name = label.caption;
3e7b0bd4 1517
1bfe1d7b
AD
1518 labelAddMenu.addChild(new dijit.MenuItem({
1519 label: name,
1520 labelId: bare_id,
4508e310 1521 onClick: function () {
3e7b0bd4 1522
424e28db 1523 let ids = getSelectedArticleIds2();
1bfe1d7b 1524 // cast to string
424e28db 1525 const id = (this.getParent().ownerMenu.currentTarget.getAttribute("data-article-id")) + "";
10690c19 1526
4508e310 1527 ids = ids.length != 0 && ids.indexOf(id) != -1 ? ids : [id];
3e7b0bd4 1528
1bfe1d7b
AD
1529 selectionAssignLabel(this.labelId, ids);
1530 }
1531 }));
3e7b0bd4 1532
1bfe1d7b
AD
1533 labelDelMenu.addChild(new dijit.MenuItem({
1534 label: name,
1535 labelId: bare_id,
4508e310 1536 onClick: function () {
424e28db 1537 let ids = getSelectedArticleIds2();
1bfe1d7b 1538 // cast to string
424e28db 1539 const id = (this.getParent().ownerMenu.currentTarget.getAttribute("data-article-id")) + "";
10690c19 1540
4508e310 1541 ids = ids.length != 0 && ids.indexOf(id) != -1 ? ids : [id];
1beea800 1542
1bfe1d7b
AD
1543 selectionRemoveLabel(this.labelId, ids);
1544 }
10690c19
AD
1545 }));
1546
1bfe1d7b 1547 });
10690c19 1548
1bfe1d7b
AD
1549 menu.addChild(new dijit.PopupMenuItem({
1550 label: __("Assign label"),
1551 popup: labelAddMenu
1552 }));
1beea800 1553
1bfe1d7b
AD
1554 menu.addChild(new dijit.PopupMenuItem({
1555 label: __("Remove label"),
1556 popup: labelDelMenu
1557 }));
65f0eb01 1558
65f0eb01
AD
1559 }
1560}
1561
51614600 1562function initHeadlinesMenu() {
1bfe1d7b 1563 if (!dijit.byId("headlinesMenu")) {
87065739 1564
4508e310 1565 const menu = new dijit.Menu({
1bfe1d7b
AD
1566 id: "headlinesMenu",
1567 targetNodeIds: ["headlines-frame"],
1568 selector: ".hlMenuAttach"
1569 });
87065739 1570
1bfe1d7b 1571 headlinesMenuCommon(menu);
87065739 1572
1bfe1d7b
AD
1573 menu.startup();
1574 }
87065739 1575
1bfe1d7b 1576 /* vgroup feed title menu */
87065739 1577
1bfe1d7b 1578 if (!dijit.byId("headlinesFeedTitleMenu")) {
e1f7b05b 1579
4508e310 1580 const menu = new dijit.Menu({
1bfe1d7b
AD
1581 id: "headlinesFeedTitleMenu",
1582 targetNodeIds: ["headlines-frame"],
1583 selector: "div.cdmFeedTitle"
1584 });
e1f7b05b 1585
1bfe1d7b
AD
1586 menu.addChild(new dijit.MenuItem({
1587 label: __("Select articles in group"),
1588 onClick: function (event) {
1589 selectArticles("all",
1590 "#headlines-frame > div[id*=RROW]" +
1591 "[data-orig-feed-id='" + this.getParent().currentTarget.getAttribute("data-feed-id") + "']");
e1f7b05b 1592
1bfe1d7b
AD
1593 }
1594 }));
e1f7b05b 1595
1bfe1d7b
AD
1596 menu.addChild(new dijit.MenuItem({
1597 label: __("Mark group as read"),
4508e310 1598 onClick: function () {
1bfe1d7b
AD
1599 selectArticles("none");
1600 selectArticles("all",
1601 "#headlines-frame > div[id*=RROW]" +
1602 "[data-orig-feed-id='" + this.getParent().currentTarget.getAttribute("data-feed-id") + "']");
e1f7b05b 1603
1bfe1d7b
AD
1604 catchupSelection();
1605 }
1606 }));
e1f7b05b 1607
1bfe1d7b
AD
1608 menu.addChild(new dijit.MenuItem({
1609 label: __("Mark feed as read"),
4508e310 1610 onClick: function () {
1bfe1d7b
AD
1611 catchupFeedInGroup(this.getParent().currentTarget.getAttribute("data-feed-id"));
1612 }
1613 }));
51614600 1614
1bfe1d7b
AD
1615 menu.addChild(new dijit.MenuItem({
1616 label: __("Edit feed"),
4508e310 1617 onClick: function () {
1bfe1d7b
AD
1618 editFeed(this.getParent().currentTarget.getAttribute("data-feed-id"));
1619 }
1620 }));
87065739 1621
1bfe1d7b 1622 menu.startup();
b509d64e
AD
1623 }
1624}
411fe209 1625
78b2c6ce 1626function cache_set(id, obj) {
37c03d3a 1627 //console.log("cache_set: " + id);
78b2c6ce
AD
1628 if (has_storage)
1629 try {
1630 sessionStorage[id] = obj;
1631 } catch (e) {
dea24b86 1632 sessionStorage.clear();
78b2c6ce
AD
1633 }
1634}
1635
1636function cache_get(id) {
1637 if (has_storage)
1638 return sessionStorage[id];
1639}
1640
1641function cache_clear() {
1642 if (has_storage)
1643 sessionStorage.clear();
1644}
1645
1646function cache_delete(id) {
1647 if (has_storage)
1648 sessionStorage.removeItem(id);
1649}
1650
74467907 1651function cancelSearch() {
1bfe1d7b
AD
1652 _search_query = "";
1653 viewCurrentFeed();
74467907 1654}
beb6ce27 1655
29064218 1656function setSelectionScore() {
424e28db 1657 const ids = getSelectedArticleIds2();
29064218 1658
1bfe1d7b
AD
1659 if (ids.length > 0) {
1660 console.log(ids);
29064218 1661
4508e310 1662 const score = prompt(__("Please enter new score for selected articles:"));
29064218 1663
1bfe1d7b 1664 if (score != undefined) {
eaf7cfdb
AD
1665 const query = { op: "article", method: "setScore", id: ids.toString(),
1666 score: score };
1667
1668 xhrJson("backend.php", query, (reply) => {
1669 if (reply) {
1670 reply.id.each((id) => {
1671 const row = $("RROW-" + id);
1672
1673 if (row) {
1674 const pic = row.getElementsByClassName("hlScorePic")[0];
1675
1676 if (pic) {
1677 pic.src = pic.src.replace(/score_.*?\.png/,
1678 reply["score_pic"]);
1679 pic.setAttribute("score", reply["score"]);
1680 }
1681 }
1682 });
1683 }
1bfe1d7b 1684 });
29064218 1685 }
1bfe1d7b
AD
1686
1687 } else {
1688 alert(__("No articles are selected."));
29064218
AD
1689 }
1690}
1691
eaf7cfdb 1692/*
a72cd54c 1693function updateScore(id) {
424e28db 1694 const pic = $$("#RROW-" + id + " .hlScorePic")[0];
a72cd54c 1695
1bfe1d7b 1696 if (pic) {
a72cd54c 1697
424e28db 1698 const query = "op=article&method=getScore&id=" + param_escape(id);
a72cd54c 1699
1bfe1d7b
AD
1700 new Ajax.Request("backend.php", {
1701 parameters: query,
1702 onComplete: function (transport) {
1703 console.log(transport.responseText);
a72cd54c 1704
424e28db 1705 const reply = JSON.parse(transport.responseText);
a72cd54c 1706
1bfe1d7b
AD
1707 if (reply) {
1708 pic.src = pic.src.replace(/score_.*?\.png/, reply["score_pic"]);
1709 pic.setAttribute("score", reply["score"]);
1710 pic.setAttribute("title", reply["score"]);
1711 }
1712 }
1713 });
a72cd54c 1714 }
eaf7cfdb 1715} */
a72cd54c 1716
beb6ce27 1717function changeScore(id, pic) {
424e28db 1718 const score = pic.getAttribute("score");
beb6ce27 1719
424e28db 1720 const new_score = prompt(__("Please enter new score for this article:"), score);
beb6ce27 1721
1bfe1d7b 1722 if (new_score != undefined) {
eaf7cfdb
AD
1723 const query = { op: "article", method: "setScore", id: id, score: new_score };
1724
1725 xhrJson("backend.php", query, (reply) => {
1726 if (reply) {
1727 pic.src = pic.src.replace(/score_.*?\.png/, reply["score_pic"]);
1728 pic.setAttribute("score", new_score);
1729 pic.setAttribute("title", new_score);
1730 }
1bfe1d7b 1731 });
beb6ce27
AD
1732 }
1733}
7fc2e87e
AD
1734
1735function displayArticleUrl(id) {
eaf7cfdb 1736 const query = { op: "rpc", method: "getlinktitlebyid", id: id };
7fc2e87e 1737
eaf7cfdb
AD
1738 xhrJson("backend.php", query, (reply) => {
1739 if (reply && reply.link) {
1740 prompt(__("Article URL:"), reply.link);
1741 }
1bfe1d7b 1742 });
eaf7cfdb 1743
7fc2e87e 1744}
f822a8e5 1745
8ee5e9e5 1746function scrollToRowId(id) {
424e28db 1747 const row = $(id);
8ee5e9e5 1748
1bfe1d7b
AD
1749 if (row)
1750 $("headlines-frame").scrollTop = row.offsetTop - 4;
8ee5e9e5
AD
1751}
1752
41d37fb2 1753function updateFloatingTitle(unread_only) {
1bfe1d7b 1754 if (!isCdmMode()) return;
08820be7 1755
424e28db 1756 const hf = $("headlines-frame");
08820be7 1757
424e28db 1758 const elems = $$("#headlines-frame > div[id*=RROW]");
4f62f8f6 1759
424e28db 1760 for (let i = 0; i < elems.length; i++) {
d1343b84 1761
424e28db 1762 const child = elems[i];
8ee5e9e5 1763
1bfe1d7b 1764 if (child && child.offsetTop + child.offsetHeight > hf.scrollTop) {
8ee5e9e5 1765
424e28db 1766 const header = child.getElementsByClassName("cdmHeader")[0];
b6b5554d 1767
1bfe1d7b
AD
1768 if (unread_only || child.getAttribute("data-article-id") != $("floatingTitle").getAttribute("data-article-id")) {
1769 if (child.getAttribute("data-article-id") != $("floatingTitle").getAttribute("data-article-id")) {
8ee5e9e5 1770
1bfe1d7b
AD
1771 $("floatingTitle").setAttribute("data-article-id", child.getAttribute("data-article-id"));
1772 $("floatingTitle").innerHTML = header.innerHTML;
1773 $("floatingTitle").firstChild.innerHTML = "<img class='anchor markedPic' src='images/page_white_go.png' onclick=\"scrollToRowId('" + child.id + "')\">" + $("floatingTitle").firstChild.innerHTML;
65f0eb01 1774
1bfe1d7b 1775 initFloatingMenu();
41d37fb2 1776
424e28db 1777 const cb = $$("#floatingTitle .dijitCheckBox")[0];
0971cc61 1778
1bfe1d7b
AD
1779 if (cb)
1780 cb.parentNode.removeChild(cb);
8ee5e9e5
AD
1781 }
1782
1bfe1d7b
AD
1783 if (child.hasClassName("Unread"))
1784 $("floatingTitle").addClassName("Unread");
8ee5e9e5 1785 else
1bfe1d7b 1786 $("floatingTitle").removeClassName("Unread");
8ee5e9e5 1787
1bfe1d7b 1788 PluginHost.run(PluginHost.HOOK_FLOATING_TITLE, child);
8ee5e9e5 1789 }
d1343b84 1790
1bfe1d7b
AD
1791 $("floatingTitle").style.marginRight = hf.offsetWidth - child.offsetWidth + "px";
1792 if (header.offsetTop + header.offsetHeight < hf.scrollTop + $("floatingTitle").offsetHeight - 5 &&
1793 child.offsetTop + child.offsetHeight >= hf.scrollTop + $("floatingTitle").offsetHeight - 5)
1794 $("floatingTitle").style.visibility = "visible";
1795 else
1796 $("floatingTitle").style.visibility = "hidden";
1797
1798 return;
1799
1800 }
08820be7
AD
1801 }
1802}