]> git.wh0rd.org - tt-rss.git/blame - js/viewfeed.js
cdmScrollToArticle: set article read on selection
[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
8359ca6d 89 if (loaded_article_ids.indexOf(row.id) == -1 || row.hasClassName("feed-title")) {
1bfe1d7b 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 129
8359ca6d 130 if (loaded_article_ids.indexOf(row.id) == -1 || row.hasClassName("feed-title")) {
a71e571b
AD
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) {
02ec3984 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 346 xhrPost("backend.php", query, (transport) => {
02ec3984 347 article_callback2(transport, id);
560b9fdd 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) {
02ec3984 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 585 if (tmpClassName != row.className) {
02ec3984 586 if (cmode == undefined) cmode = 2;
933ba4ee 587
02ec3984
AD
588 const query = {op: "rpc", method: "catchupSelected",
589 cmode: cmode, ids: id};
933ba4ee 590
02ec3984 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 610 xhrPost("backend.php", query, (transport) => {
02ec3984
AD
611 handle_rpc_json(transport);
612 show_labels_in_headlines(transport);
560b9fdd 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
02ec3984
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
02ec3984
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
02ec3984
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) {
02ec3984
AD
727 const query = { op: "rpc", method: "publishSelected",
728 ids: rows.toString(), cmode: 2 };
e4f4b46f 729
02ec3984
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 849 xhrPost("backend.php", query, (transport) => {
02ec3984
AD
850 handle_rpc_json(transport);
851 viewCurrentFeed();
1bfe1d7b 852 });
e04c18a2
AD
853}
854
855function archiveSelection() {
856
02ec3984 857 const rows = getSelectedArticleIds2();
009646d2 858
02ec3984
AD
859 if (rows.length == 0) {
860 alert(__("No articles are selected."));
861 return;
862 }
b029f916 863
02ec3984
AD
864 const fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
865 let str;
866 let op;
b029f916 867
02ec3984
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
02ec3984 874 str += " " + __("Please note that unstarred articles might get purged on next feed update.");
009646d2 875
02ec3984
AD
876 op = "unarchive";
877 }
e04c18a2 878
02ec3984
AD
879 str = str.replace("%d", rows.length);
880 str = str.replace("%s", fn);
e04c18a2 881
02ec3984
AD
882 if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
883 return;
884 }
e04c18a2 885
02ec3984
AD
886 for (let i = 0; i < rows.length; i++) {
887 cache_delete("article:" + rows[i]);
888 }
a12eb9c3 889
02ec3984 890 const query = {op: "rpc", method: op, ids: rows.toString()};
e04c18a2 891
02ec3984
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 937 xhrPost("backend.php", this.attr('value'), (transport) => {
02ec3984
AD
938 try {
939 notify('');
940 dialog.hide();
009646d2 941
02ec3984 942 const data = JSON.parse(transport.responseText);
ddcbbea2 943
02ec3984
AD
944 if (data) {
945 const id = data.id;
ddcbbea2 946
02ec3984
AD
947 const tags = $("ATSTR-" + id);
948 const tooltip = dijit.byId("ATSTRTIP-" + id);
ddcbbea2 949
02ec3984
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 }
eaf7cfdb 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;
08bee978
AD
985
986 // article is selected manually, set it read
987 toggleUnread(id, 0);1
b8e6acea
AD
988 }
989}
990
18e1a773 991function setActiveArticleId(id) {
eaec06f9
AD
992 console.log("setActiveArticleId:" + id);
993
18e1a773 994 _active_article_id = id;
8d388f32 995 PluginHost.run(PluginHost.HOOK_ARTICLE_SET_ACTIVE, _active_article_id);
18e1a773
AD
996}
997
298f3f78 998function getActiveArticleId() {
18e1a773 999 return _active_article_id;
298f3f78 1000}
e4914b62 1001
d2f3467b 1002function postMouseIn(e, id) {
78b2c6ce 1003 post_under_pointer = id;
314fcd2b
AD
1004}
1005
1006function postMouseOut(id) {
78b2c6ce 1007 post_under_pointer = false;
314fcd2b
AD
1008}
1009
62b1f587 1010function unpackVisibleHeadlines() {
9563e3bc 1011 if (!isCdmMode()) return;
44018133 1012
02ec3984 1013 const rows = $$("#headlines-frame div[id*=RROW][data-content]");
59f5335f 1014 const threshold = $("headlines-frame").scrollTop + $("headlines-frame").offsetHeight + 300;
ad326dbf 1015
02ec3984
AD
1016 for (let i = 0; i < rows.length; i++) {
1017 const row = rows[i];
0c06bb5f 1018
02ec3984
AD
1019 if (row.offsetTop <= threshold) {
1020 console.log("unpacking: " + row.id);
b74c5134 1021
02ec3984 1022 const content = row.getAttribute("data-content");
d8e8e24e 1023
8359ca6d 1024 row.select(".content-inner")[0].innerHTML = content;
02ec3984 1025 row.removeAttribute("data-content");
d8e8e24e 1026
02ec3984
AD
1027 PluginHost.run(PluginHost.HOOK_ARTICLE_RENDERED_CDM, row);
1028 } else {
1029 break;
1030 }
1031 }
62b1f587
AD
1032}
1033
1034function headlines_scroll_handler(e) {
1035 try {
63c7446a
AD
1036
1037 // rate-limit in case of smooth scrolling and similar abominations
d7cc5e6c 1038 if (Math.max(e.scrollTop, _headlines_scroll_offset) - Math.min(e.scrollTop, _headlines_scroll_offset) < 25) {
63c7446a
AD
1039 return;
1040 }
1041
1042 _headlines_scroll_offset = e.scrollTop;
1043
62b1f587
AD
1044 unpackVisibleHeadlines();
1045
48be35ca 1046 // set topmost child in the buffer as active
2e35b3bd 1047 if (isCdmMode() && getInitParam("cdm_auto_catchup") == 1 &&
9563e3bc 1048 getSelectedArticleIds2().length <= 1) {
63c7446a 1049
424e28db 1050 const rows = $$("#headlines-frame > div[id*=RROW]");
48be35ca 1051
424e28db 1052 for (let i = 0; i < rows.length; i++) {
0c06bb5f 1053 const row = rows[i];
48be35ca 1054
0c06bb5f
AD
1055 if ($("headlines-frame").scrollTop <= row.offsetTop &&
1056 row.offsetTop - $("headlines-frame").scrollTop < 100 &&
1057 row.getAttribute("data-article-id") != _active_article_id) {
99d89d10 1058
48be35ca 1059 if (_active_article_id) {
424e28db 1060 const row = $("RROW-" + _active_article_id);
48be35ca
AD
1061 if (row) row.removeClassName("active");
1062 }
1063
0c06bb5f 1064 _active_article_id = row.getAttribute("data-article-id");
48be35ca 1065 showArticleInHeadlines(_active_article_id, true);
4bc6489a 1066 updateSelectedPrompt();
48be35ca
AD
1067 break;
1068 }
1069 }
1070 }
1071
18eb64a8 1072 if (!_infscroll_disable) {
02ec3984 1073 const hsp = $("headlines-spacer");
0c06bb5f 1074
eaec06f9 1075 if (hsp && hsp.offsetTop - 250 <= e.scrollTop + e.offsetHeight) {
ac541432 1076
d46af5a5
AD
1077 hsp.innerHTML = "<span class='loading'><img src='images/indicator_tiny.gif'> " +
1078 __("Loading, please wait...") + "</span>";
71317973
AD
1079
1080 loadMoreHeadlines();
79ddf6aa 1081 return;
71317973 1082
29dfb258 1083 }
ac541432
AD
1084 }
1085
7415fcf2 1086 if (isCdmMode()) {
08820be7
AD
1087 updateFloatingTitle();
1088 }
1089
ce68ec74 1090 if (getInitParam("cdm_auto_catchup") == 1) {
21426d3a 1091
0c06bb5f 1092 let rows = $$("#headlines-frame > div[id*=RROW][class*=Unread]");
21426d3a 1093
0c06bb5f 1094 for (let i = 0; i < rows.length; i++) {
02ec3984 1095 const row = rows[i];
0c06bb5f
AD
1096
1097 if ($("headlines-frame").scrollTop > (row.offsetTop + row.offsetHeight/2)) {
76495dfd 1098
02ec3984 1099 const id = row.getAttribute("data-article-id")
79ddf6aa 1100
02ec3984
AD
1101 if (catchup_id_batch.indexOf(id) == -1)
1102 catchup_id_batch.push(id);
48be35ca 1103
02ec3984
AD
1104 //console.log("auto_catchup_batch: " + catchup_id_batch.toString());
1105 } else {
0c06bb5f
AD
1106 break;
1107 }
02ec3984 1108 }
21426d3a 1109
0f85f483 1110 if (_infscroll_disable) {
0c06bb5f 1111 const row = $$("#headlines-frame div[id*=RROW]").last();
b7c20a37 1112
0c06bb5f
AD
1113 if (row && $("headlines-frame").scrollTop >
1114 (row.offsetTop + row.offsetHeight - 50)) {
b7c20a37
AD
1115
1116 console.log("we seem to be at an end");
1117
1118 if (getInitParam("on_catchup_show_next_feed") == "1") {
22f675e5 1119 openNextUnreadFeed();
b7c20a37
AD
1120 }
1121 }
76495dfd
AD
1122 }
1123 }
21426d3a 1124
76495dfd 1125 } catch (e) {
37c03d3a 1126 console.warn("headlines_scroll_handler: " + e);
76495dfd
AD
1127 }
1128}
009646d2 1129
22f675e5 1130function openNextUnreadFeed() {
424e28db
AD
1131 const is_cat = activeFeedIsCat();
1132 const nuf = getNextUnreadFeed(getActiveFeedId(), is_cat);
1bfe1d7b 1133 if (nuf) viewfeed({feed: nuf, is_cat: is_cat});
22f675e5
AD
1134}
1135
76495dfd 1136function catchupBatchedArticles() {
1bfe1d7b 1137 if (catchup_id_batch.length > 0 && !_infscroll_request_sent && !_catchup_request_sent) {
009646d2 1138
0c06bb5f 1139 console.log("catchupBatchedArticles, size=", catchup_id_batch.length);
0a11d235 1140
1bfe1d7b 1141 // make a copy of the array
424e28db 1142 const batch = catchup_id_batch.slice();
eaf7cfdb
AD
1143 const query = { op: "rpc", method: "catchupSelected",
1144 cmode: 0, ids: batch.toString() };
8292123e 1145
1bfe1d7b 1146 _catchup_request_sent = true;
76495dfd 1147
eaf7cfdb 1148 xhrPost("backend.php", query, (transport) => {
02ec3984 1149 const reply = handle_rpc_json(transport);
8292123e 1150
02ec3984 1151 _catchup_request_sent = false;
ae31704b 1152
02ec3984
AD
1153 if (reply) {
1154 const batch = reply.ids;
76495dfd 1155
02ec3984
AD
1156 batch.each(function (id) {
1157 const elem = $("RROW-" + id);
1158 if (elem) elem.removeClassName("Unread");
1159 catchup_id_batch.remove(id);
1160 });
1161 }
76495dfd 1162
02ec3984 1163 updateFloatingTitle(true);
1bfe1d7b 1164 });
ac541432
AD
1165 }
1166}
8be83f42 1167
2ea34cc1 1168function catchupRelativeToArticle(below, id) {
8be83f42 1169
1bfe1d7b 1170 if (!id) id = getActiveArticleId();
49fd1e94 1171
1bfe1d7b
AD
1172 if (!id) {
1173 alert(__("No article is selected."));
1174 return;
1175 }
8be83f42 1176
424e28db 1177 const visible_ids = getLoadedArticleIds();
8be83f42 1178
4508e310 1179 const ids_to_mark = [];
8be83f42 1180
1bfe1d7b
AD
1181 if (!below) {
1182 for (var i = 0; i < visible_ids.length; i++) {
1183 if (visible_ids[i] != id) {
1184 var e = $("RROW-" + visible_ids[i]);
8be83f42 1185
1bfe1d7b
AD
1186 if (e && e.hasClassName("Unread")) {
1187 ids_to_mark.push(visible_ids[i]);
8be83f42 1188 }
1bfe1d7b
AD
1189 } else {
1190 break;
8be83f42 1191 }
1bfe1d7b
AD
1192 }
1193 } else {
1194 for (var i = visible_ids.length - 1; i >= 0; i--) {
1195 if (visible_ids[i] != id) {
1196 var e = $("RROW-" + visible_ids[i]);
8be83f42 1197
1bfe1d7b
AD
1198 if (e && e.hasClassName("Unread")) {
1199 ids_to_mark.push(visible_ids[i]);
8be83f42 1200 }
1bfe1d7b
AD
1201 } else {
1202 break;
8be83f42
AD
1203 }
1204 }
1bfe1d7b 1205 }
8be83f42 1206
1bfe1d7b
AD
1207 if (ids_to_mark.length == 0) {
1208 alert(__("No articles found to mark"));
1209 } else {
424e28db 1210 const msg = ngettext("Mark %d article as read?", "Mark %d articles as read?", ids_to_mark.length).replace("%d", ids_to_mark.length);
8be83f42 1211
1bfe1d7b 1212 if (getInitParam("confirm_feed_catchup") != 1 || confirm(msg)) {
8be83f42 1213
1bfe1d7b
AD
1214 for (var i = 0; i < ids_to_mark.length; i++) {
1215 var e = $("RROW-" + ids_to_mark[i]);
1216 e.removeClassName("Unread");
1217 }
8be83f42 1218
eaf7cfdb
AD
1219 const query = { op: "rpc", method: "catchupSelected",
1220 cmode: 0, ids: ids_to_mark.toString() };
8be83f42 1221
eaf7cfdb 1222 xhrPost("backend.php", query, (transport) => {
02ec3984 1223 handle_rpc_json(transport);
1bfe1d7b 1224 });
8be83f42 1225 }
8be83f42
AD
1226 }
1227}
a04c8e8d 1228
314fcd2b
AD
1229function getArticleUnderPointer() {
1230 return post_under_pointer;
7a822893 1231}
eedfb635 1232
a411875b 1233function scrollArticle(offset) {
1bfe1d7b 1234 if (!isCdmMode()) {
424e28db 1235 const ci = $("content-insert");
1bfe1d7b
AD
1236 if (ci) {
1237 ci.scrollTop += offset;
a411875b 1238 }
1bfe1d7b 1239 } else {
424e28db 1240 const hi = $("headlines-frame");
1bfe1d7b
AD
1241 if (hi) {
1242 hi.scrollTop += offset;
1243 }
1244
a411875b
AD
1245 }
1246}
f9247195
AD
1247
1248function show_labels_in_headlines(transport) {
424e28db 1249 const data = JSON.parse(transport.responseText);
f9247195 1250
1bfe1d7b
AD
1251 if (data) {
1252 data['info-for-headlines'].each(function (elem) {
1253 $$(".HLLCTR-" + elem.id).each(function (ctr) {
1254 ctr.innerHTML = elem.labels;
8eb592ec 1255 });
1bfe1d7b 1256 });
f9247195
AD
1257 }
1258}
bf3c9838 1259
63f0ed3d 1260function cdmClicked(event, id, in_body) {
1bfe1d7b 1261 //var shift_key = event.shiftKey;
1ede5814 1262
1bfe1d7b 1263 if (!event.ctrlKey && !event.metaKey) {
62b800b4 1264
9563e3bc 1265 let elem = $("RROW-" + getActiveArticleId());
009646d2 1266
9563e3bc 1267 if (elem) elem.removeClassName("active");
009646d2 1268
9563e3bc
AD
1269 selectArticles("none");
1270 toggleSelected(id);
009646d2 1271
9563e3bc
AD
1272 elem = $("RROW-" + id);
1273 const article_is_unread = elem.hasClassName("Unread");
009646d2 1274
9563e3bc
AD
1275 elem.removeClassName("Unread");
1276 elem.addClassName("active");
9bfb11ed 1277
9563e3bc 1278 setActiveArticleId(id);
62b800b4 1279
9563e3bc
AD
1280 if (article_is_unread) {
1281 decrementFeedCounter(getActiveFeedId(), activeFeedIsCat());
1282 updateFloatingTitle(true);
bace5e9e 1283
9563e3bc
AD
1284 const query = {
1285 op: "rpc", method: "catchupSelected",
1286 cmode: 0, ids: id
1287 };
de44b9a6 1288
9563e3bc
AD
1289 xhrPost("backend.php", query, (transport) => {
1290 handle_rpc_json(transport);
1291 });
1bfe1d7b 1292 }
de44b9a6 1293
9563e3bc
AD
1294 return !event.shiftKey;
1295
63f0ed3d 1296 } else if (!in_body) {
de44b9a6 1297
1bfe1d7b 1298 toggleSelected(id, true);
38325ad6 1299
4508e310
AD
1300 let elem = $("RROW-" + id);
1301 const article_is_unread = elem.hasClassName("Unread");
1bfe1d7b
AD
1302
1303 if (article_is_unread) {
1304 decrementFeedCounter(getActiveFeedId(), activeFeedIsCat());
1ede5814
AD
1305 }
1306
1bfe1d7b 1307 toggleUnread(id, 0, false);
5b55e9e2 1308
1bfe1d7b 1309 openArticleInNewWindow(id);
63f0ed3d
AD
1310 } else {
1311 return true;
1ede5814
AD
1312 }
1313
424e28db 1314 const unread_in_buffer = $$("#headlines-frame > div[id*=RROW][class*=Unread]").length
1bfe1d7b
AD
1315 request_counters(unread_in_buffer == 0);
1316
1ede5814
AD
1317 return false;
1318}
1319
1320function hlClicked(event, id) {
1bfe1d7b
AD
1321 if (event.which == 2) {
1322 view(id);
1323 return true;
1324 } else if (event.ctrlKey || event.metaKey) {
1bfe1d7b
AD
1325 openArticleInNewWindow(id);
1326 return false;
1327 } else {
1328 view(id);
1329 return false;
1ede5814 1330 }
1ede5814 1331}
e69fb880 1332
e69fb880 1333function openArticleInNewWindow(id) {
b509d64e 1334 toggleUnread(id, 0, false);
829d478f 1335
424e28db 1336 const w = window.open("");
829d478f
AD
1337 w.opener = null;
1338 w.location = "backend.php?op=article&method=redirect&id=" + id;
e69fb880
AD
1339}
1340
1341function isCdmMode() {
8cc5e965 1342 return getInitParam("combined_display_mode");
e69fb880
AD
1343}
1344
414191d4
AD
1345function markHeadline(id, marked) {
1346 if (marked == undefined) marked = true;
1347
424e28db 1348 const row = $("RROW-" + id);
e69fb880 1349 if (row) {
424e28db 1350 const check = dijit.getEnclosingWidget(
7e27c914 1351 row.getElementsByClassName("rchk")[0]);
e69fb880
AD
1352
1353 if (check) {
414191d4 1354 check.attr("checked", marked);
e69fb880
AD
1355 }
1356
3ac153f1
AD
1357 if (marked)
1358 row.addClassName("Selected");
1359 else
1360 row.removeClassName("Selected");
e69fb880
AD
1361 }
1362}
1363
8cc5e965
AD
1364function getRelativePostIds(id, limit) {
1365
424e28db 1366 const tmp = [];
8cc5e965 1367
1bfe1d7b 1368 if (!limit) limit = 6; //3
009646d2 1369
424e28db 1370 const ids = getLoadedArticleIds();
009646d2 1371
424e28db 1372 for (let i = 0; i < ids.length; i++) {
1bfe1d7b 1373 if (ids[i] == id) {
424e28db 1374 for (let k = 1; k <= limit; k++) {
1bfe1d7b
AD
1375 //if (i > k-1) tmp.push(ids[i-k]);
1376 if (i < ids.length - k) tmp.push(ids[i + k]);
8cc5e965 1377 }
1bfe1d7b 1378 break;
8cc5e965 1379 }
8cc5e965
AD
1380 }
1381
1382 return tmp;
1383}
1384
1385function correctHeadlinesOffset(id) {
009646d2 1386
424e28db
AD
1387 const container = $("headlines-frame");
1388 const row = $("RROW-" + id);
e4f48f86 1389
1bfe1d7b 1390 if (!container || !row) return;
009646d2 1391
424e28db 1392 const viewport = container.offsetHeight;
009646d2 1393
424e28db
AD
1394 const rel_offset_top = row.offsetTop - container.scrollTop;
1395 const rel_offset_bottom = row.offsetTop + row.offsetHeight - container.scrollTop;
8cc5e965 1396
1bfe1d7b
AD
1397 //console.log("Rtop: " + rel_offset_top + " Rbtm: " + rel_offset_bottom);
1398 //console.log("Vport: " + viewport);
8cc5e965 1399
1bfe1d7b
AD
1400 if (rel_offset_top <= 0 || rel_offset_top > viewport) {
1401 container.scrollTop = row.offsetTop;
1402 } else if (rel_offset_bottom > viewport) {
8cc5e965 1403
1bfe1d7b
AD
1404 /* doesn't properly work with Opera in some cases because
1405 Opera fucks up element scrolling */
8cc5e965 1406
1bfe1d7b 1407 container.scrollTop = row.offsetTop + row.offsetHeight - viewport;
8cc5e965 1408 }
8cc5e965
AD
1409}
1410
fcf70c51 1411function headlineActionsChange(elem) {
1bfe1d7b
AD
1412 eval(elem.value);
1413 elem.attr('value', 'false');
fcf70c51 1414}
6f3976c9 1415
0b461ed5 1416function closeArticlePanel() {
e3387e2d 1417
38325ad6
AD
1418 if (dijit.byId("content-insert"))
1419 dijit.byId("headlines-wrap-inner").removeChild(
1420 dijit.byId("content-insert"));
6f3976c9 1421}
b509d64e 1422
65f0eb01 1423function initFloatingMenu() {
1bfe1d7b 1424 if (!dijit.byId("floatingMenu")) {
b509d64e 1425
424e28db 1426 const menu = new dijit.Menu({
1bfe1d7b
AD
1427 id: "floatingMenu",
1428 targetNodeIds: ["floatingTitle"]
1429 });
b509d64e 1430
1bfe1d7b 1431 headlinesMenuCommon(menu);
c4e21104 1432
1bfe1d7b 1433 menu.startup();
65f0eb01
AD
1434 }
1435}
b509d64e 1436
c4e21104 1437function headlinesMenuCommon(menu) {
7fc2e87e 1438
1bfe1d7b
AD
1439 menu.addChild(new dijit.MenuItem({
1440 label: __("Open original article"),
1441 onClick: function (event) {
1442 openArticleInNewWindow(this.getParent().currentTarget.getAttribute("data-article-id"));
1443 }
1444 }));
2ea34cc1 1445
1bfe1d7b
AD
1446 menu.addChild(new dijit.MenuItem({
1447 label: __("Display article URL"),
1448 onClick: function (event) {
1449 displayArticleUrl(this.getParent().currentTarget.getAttribute("data-article-id"));
1450 }
1451 }));
b835a528 1452
1bfe1d7b 1453 menu.addChild(new dijit.MenuSeparator());
be7bb7d5 1454
1bfe1d7b
AD
1455 menu.addChild(new dijit.MenuItem({
1456 label: __("Toggle unread"),
4508e310 1457 onClick: function () {
be7bb7d5 1458
424e28db 1459 let ids = getSelectedArticleIds2();
1bfe1d7b 1460 // cast to string
424e28db 1461 const id = (this.getParent().currentTarget.getAttribute("data-article-id")) + "";
4508e310 1462 ids = ids.length != 0 && ids.indexOf(id) != -1 ? ids : [id];
be7bb7d5 1463
1bfe1d7b
AD
1464 selectionToggleUnread(undefined, false, true, ids);
1465 }
1466 }));
be7bb7d5 1467
1bfe1d7b
AD
1468 menu.addChild(new dijit.MenuItem({
1469 label: __("Toggle starred"),
4508e310 1470 onClick: function () {
424e28db 1471 let ids = getSelectedArticleIds2();
1bfe1d7b 1472 // cast to string
424e28db 1473 const id = (this.getParent().currentTarget.getAttribute("data-article-id")) + "";
4508e310 1474 ids = ids.length != 0 && ids.indexOf(id) != -1 ? ids : [id];
be7bb7d5 1475
1bfe1d7b
AD
1476 selectionToggleMarked(undefined, false, true, ids);
1477 }
1478 }));
be7bb7d5 1479
1bfe1d7b
AD
1480 menu.addChild(new dijit.MenuItem({
1481 label: __("Toggle published"),
4508e310 1482 onClick: function () {
424e28db 1483 let ids = getSelectedArticleIds2();
1bfe1d7b 1484 // cast to string
424e28db 1485 const id = (this.getParent().currentTarget.getAttribute("data-article-id")) + "";
4508e310 1486 ids = ids.length != 0 && ids.indexOf(id) != -1 ? ids : [id];
be7bb7d5 1487
1bfe1d7b
AD
1488 selectionTogglePublished(undefined, false, true, ids);
1489 }
1490 }));
2ea34cc1 1491
1bfe1d7b 1492 menu.addChild(new dijit.MenuSeparator());
2ea34cc1 1493
1bfe1d7b
AD
1494 menu.addChild(new dijit.MenuItem({
1495 label: __("Mark above as read"),
4508e310 1496 onClick: function () {
1bfe1d7b
AD
1497 catchupRelativeToArticle(0, this.getParent().currentTarget.getAttribute("data-article-id"));
1498 }
1499 }));
10690c19 1500
1bfe1d7b
AD
1501 menu.addChild(new dijit.MenuItem({
1502 label: __("Mark below as read"),
4508e310 1503 onClick: function () {
1bfe1d7b
AD
1504 catchupRelativeToArticle(1, this.getParent().currentTarget.getAttribute("data-article-id"));
1505 }
1506 }));
1beea800 1507
1beea800 1508
424e28db 1509 const labels = getInitParam("labels");
1beea800 1510
b5fc9781 1511 if (labels && labels.length) {
1beea800 1512
1bfe1d7b 1513 menu.addChild(new dijit.MenuSeparator());
1beea800 1514
424e28db
AD
1515 const labelAddMenu = new dijit.Menu({ownerMenu: menu});
1516 const labelDelMenu = new dijit.Menu({ownerMenu: menu});
1beea800 1517
1bfe1d7b 1518 labels.each(function (label) {
424e28db
AD
1519 const bare_id = label.id;
1520 const name = label.caption;
3e7b0bd4 1521
1bfe1d7b
AD
1522 labelAddMenu.addChild(new dijit.MenuItem({
1523 label: name,
1524 labelId: bare_id,
4508e310 1525 onClick: function () {
3e7b0bd4 1526
424e28db 1527 let ids = getSelectedArticleIds2();
1bfe1d7b 1528 // cast to string
424e28db 1529 const id = (this.getParent().ownerMenu.currentTarget.getAttribute("data-article-id")) + "";
10690c19 1530
4508e310 1531 ids = ids.length != 0 && ids.indexOf(id) != -1 ? ids : [id];
3e7b0bd4 1532
1bfe1d7b
AD
1533 selectionAssignLabel(this.labelId, ids);
1534 }
1535 }));
3e7b0bd4 1536
1bfe1d7b
AD
1537 labelDelMenu.addChild(new dijit.MenuItem({
1538 label: name,
1539 labelId: bare_id,
4508e310 1540 onClick: function () {
424e28db 1541 let ids = getSelectedArticleIds2();
1bfe1d7b 1542 // cast to string
424e28db 1543 const id = (this.getParent().ownerMenu.currentTarget.getAttribute("data-article-id")) + "";
10690c19 1544
4508e310 1545 ids = ids.length != 0 && ids.indexOf(id) != -1 ? ids : [id];
1beea800 1546
1bfe1d7b
AD
1547 selectionRemoveLabel(this.labelId, ids);
1548 }
10690c19
AD
1549 }));
1550
1bfe1d7b 1551 });
10690c19 1552
1bfe1d7b
AD
1553 menu.addChild(new dijit.PopupMenuItem({
1554 label: __("Assign label"),
1555 popup: labelAddMenu
1556 }));
1beea800 1557
1bfe1d7b
AD
1558 menu.addChild(new dijit.PopupMenuItem({
1559 label: __("Remove label"),
1560 popup: labelDelMenu
1561 }));
65f0eb01 1562
65f0eb01
AD
1563 }
1564}
1565
51614600 1566function initHeadlinesMenu() {
1bfe1d7b 1567 if (!dijit.byId("headlinesMenu")) {
87065739 1568
4508e310 1569 const menu = new dijit.Menu({
1bfe1d7b
AD
1570 id: "headlinesMenu",
1571 targetNodeIds: ["headlines-frame"],
1572 selector: ".hlMenuAttach"
1573 });
87065739 1574
1bfe1d7b 1575 headlinesMenuCommon(menu);
87065739 1576
1bfe1d7b
AD
1577 menu.startup();
1578 }
87065739 1579
1bfe1d7b 1580 /* vgroup feed title menu */
87065739 1581
1bfe1d7b 1582 if (!dijit.byId("headlinesFeedTitleMenu")) {
e1f7b05b 1583
4508e310 1584 const menu = new dijit.Menu({
1bfe1d7b
AD
1585 id: "headlinesFeedTitleMenu",
1586 targetNodeIds: ["headlines-frame"],
1587 selector: "div.cdmFeedTitle"
1588 });
e1f7b05b 1589
1bfe1d7b
AD
1590 menu.addChild(new dijit.MenuItem({
1591 label: __("Select articles in group"),
1592 onClick: function (event) {
1593 selectArticles("all",
1594 "#headlines-frame > div[id*=RROW]" +
1595 "[data-orig-feed-id='" + this.getParent().currentTarget.getAttribute("data-feed-id") + "']");
e1f7b05b 1596
1bfe1d7b
AD
1597 }
1598 }));
e1f7b05b 1599
1bfe1d7b
AD
1600 menu.addChild(new dijit.MenuItem({
1601 label: __("Mark group as read"),
4508e310 1602 onClick: function () {
1bfe1d7b
AD
1603 selectArticles("none");
1604 selectArticles("all",
1605 "#headlines-frame > div[id*=RROW]" +
1606 "[data-orig-feed-id='" + this.getParent().currentTarget.getAttribute("data-feed-id") + "']");
e1f7b05b 1607
1bfe1d7b
AD
1608 catchupSelection();
1609 }
1610 }));
e1f7b05b 1611
1bfe1d7b
AD
1612 menu.addChild(new dijit.MenuItem({
1613 label: __("Mark feed as read"),
4508e310 1614 onClick: function () {
1bfe1d7b
AD
1615 catchupFeedInGroup(this.getParent().currentTarget.getAttribute("data-feed-id"));
1616 }
1617 }));
51614600 1618
1bfe1d7b
AD
1619 menu.addChild(new dijit.MenuItem({
1620 label: __("Edit feed"),
4508e310 1621 onClick: function () {
1bfe1d7b
AD
1622 editFeed(this.getParent().currentTarget.getAttribute("data-feed-id"));
1623 }
1624 }));
87065739 1625
1bfe1d7b 1626 menu.startup();
b509d64e
AD
1627 }
1628}
411fe209 1629
78b2c6ce 1630function cache_set(id, obj) {
37c03d3a 1631 //console.log("cache_set: " + id);
78b2c6ce
AD
1632 if (has_storage)
1633 try {
1634 sessionStorage[id] = obj;
1635 } catch (e) {
dea24b86 1636 sessionStorage.clear();
78b2c6ce
AD
1637 }
1638}
1639
1640function cache_get(id) {
1641 if (has_storage)
1642 return sessionStorage[id];
1643}
1644
1645function cache_clear() {
1646 if (has_storage)
1647 sessionStorage.clear();
1648}
1649
1650function cache_delete(id) {
1651 if (has_storage)
1652 sessionStorage.removeItem(id);
1653}
1654
74467907 1655function cancelSearch() {
1bfe1d7b
AD
1656 _search_query = "";
1657 viewCurrentFeed();
74467907 1658}
beb6ce27 1659
29064218 1660function setSelectionScore() {
424e28db 1661 const ids = getSelectedArticleIds2();
29064218 1662
1bfe1d7b
AD
1663 if (ids.length > 0) {
1664 console.log(ids);
29064218 1665
4508e310 1666 const score = prompt(__("Please enter new score for selected articles:"));
29064218 1667
1bfe1d7b 1668 if (score != undefined) {
eaf7cfdb
AD
1669 const query = { op: "article", method: "setScore", id: ids.toString(),
1670 score: score };
1671
1672 xhrJson("backend.php", query, (reply) => {
1673 if (reply) {
02ec3984
AD
1674 reply.id.each((id) => {
1675 const row = $("RROW-" + id);
1676
1677 if (row) {
8359ca6d 1678 const pic = row.getElementsByClassName("score-pic")[0];
02ec3984
AD
1679
1680 if (pic) {
1681 pic.src = pic.src.replace(/score_.*?\.png/,
1682 reply["score_pic"]);
1683 pic.setAttribute("score", reply["score"]);
1684 }
1685 }
1686 });
1687 }
1bfe1d7b 1688 });
29064218 1689 }
1bfe1d7b
AD
1690
1691 } else {
1692 alert(__("No articles are selected."));
29064218
AD
1693 }
1694}
1695
beb6ce27 1696function changeScore(id, pic) {
424e28db 1697 const score = pic.getAttribute("score");
beb6ce27 1698
424e28db 1699 const new_score = prompt(__("Please enter new score for this article:"), score);
beb6ce27 1700
1bfe1d7b 1701 if (new_score != undefined) {
eaf7cfdb
AD
1702 const query = { op: "article", method: "setScore", id: id, score: new_score };
1703
1704 xhrJson("backend.php", query, (reply) => {
02ec3984
AD
1705 if (reply) {
1706 pic.src = pic.src.replace(/score_.*?\.png/, reply["score_pic"]);
1707 pic.setAttribute("score", new_score);
1708 pic.setAttribute("title", new_score);
1709 }
1bfe1d7b 1710 });
beb6ce27
AD
1711 }
1712}
7fc2e87e
AD
1713
1714function displayArticleUrl(id) {
eaf7cfdb 1715 const query = { op: "rpc", method: "getlinktitlebyid", id: id };
7fc2e87e 1716
eaf7cfdb 1717 xhrJson("backend.php", query, (reply) => {
02ec3984
AD
1718 if (reply && reply.link) {
1719 prompt(__("Article URL:"), reply.link);
1720 }
1bfe1d7b 1721 });
eaf7cfdb 1722
7fc2e87e 1723}
f822a8e5 1724
8ee5e9e5 1725function scrollToRowId(id) {
424e28db 1726 const row = $(id);
8ee5e9e5 1727
1bfe1d7b
AD
1728 if (row)
1729 $("headlines-frame").scrollTop = row.offsetTop - 4;
8ee5e9e5
AD
1730}
1731
41d37fb2 1732function updateFloatingTitle(unread_only) {
1bfe1d7b 1733 if (!isCdmMode()) return;
08820be7 1734
424e28db 1735 const hf = $("headlines-frame");
08820be7 1736
424e28db 1737 const elems = $$("#headlines-frame > div[id*=RROW]");
4f62f8f6 1738
424e28db 1739 for (let i = 0; i < elems.length; i++) {
d1343b84 1740
424e28db 1741 const child = elems[i];
8ee5e9e5 1742
1bfe1d7b 1743 if (child && child.offsetTop + child.offsetHeight > hf.scrollTop) {
8ee5e9e5 1744
8359ca6d 1745 const header = child.getElementsByClassName("header")[0];
b6b5554d 1746
1bfe1d7b
AD
1747 if (unread_only || child.getAttribute("data-article-id") != $("floatingTitle").getAttribute("data-article-id")) {
1748 if (child.getAttribute("data-article-id") != $("floatingTitle").getAttribute("data-article-id")) {
8ee5e9e5 1749
1bfe1d7b
AD
1750 $("floatingTitle").setAttribute("data-article-id", child.getAttribute("data-article-id"));
1751 $("floatingTitle").innerHTML = header.innerHTML;
1752 $("floatingTitle").firstChild.innerHTML = "<img class='anchor markedPic' src='images/page_white_go.png' onclick=\"scrollToRowId('" + child.id + "')\">" + $("floatingTitle").firstChild.innerHTML;
65f0eb01 1753
1bfe1d7b 1754 initFloatingMenu();
41d37fb2 1755
424e28db 1756 const cb = $$("#floatingTitle .dijitCheckBox")[0];
0971cc61 1757
1bfe1d7b
AD
1758 if (cb)
1759 cb.parentNode.removeChild(cb);
8ee5e9e5
AD
1760 }
1761
1bfe1d7b
AD
1762 if (child.hasClassName("Unread"))
1763 $("floatingTitle").addClassName("Unread");
8ee5e9e5 1764 else
1bfe1d7b 1765 $("floatingTitle").removeClassName("Unread");
8ee5e9e5 1766
1bfe1d7b 1767 PluginHost.run(PluginHost.HOOK_FLOATING_TITLE, child);
8ee5e9e5 1768 }
d1343b84 1769
1bfe1d7b
AD
1770 $("floatingTitle").style.marginRight = hf.offsetWidth - child.offsetWidth + "px";
1771 if (header.offsetTop + header.offsetHeight < hf.scrollTop + $("floatingTitle").offsetHeight - 5 &&
1772 child.offsetTop + child.offsetHeight >= hf.scrollTop + $("floatingTitle").offsetHeight - 5)
1773 $("floatingTitle").style.visibility = "visible";
1774 else
1775 $("floatingTitle").style.visibility = "hidden";
1776
1777 return;
1778
1779 }
08820be7
AD
1780 }
1781}