]> git.wh0rd.org - tt-rss.git/blob - js/viewfeed.js
assorted fixes for keyboard navigation which hopefully don't break everything else
[tt-rss.git] / js / viewfeed.js
1 var article_cache = new Array();
2
3 var _active_article_id = 0;
4
5 var vgroup_last_feed = false;
6 var post_under_pointer = false;
7
8 var last_requested_article = false;
9
10 var catchup_id_batch = [];
11 var catchup_timeout_id = false;
12
13 var cids_requested = [];
14 var loaded_article_ids = [];
15 var _last_headlines_update = 0;
16 var current_first_id = 0;
17
18 var _catchup_request_sent = false;
19
20 var has_storage = 'sessionStorage' in window && window['sessionStorage'] !== null;
21
22 function headlines_callback2(transport, offset, background, infscroll_req) {
23 try {
24 handle_rpc_json(transport);
25
26 console.log("headlines_callback2 [offset=" + offset + "] B:" + background + " I:" + infscroll_req);
27
28 var is_cat = false;
29 var feed_id = false;
30
31 var reply = false;
32
33 try {
34 reply = JSON.parse(transport.responseText);
35 } catch (e) {
36 console.error(e);
37 }
38
39 if (reply) {
40
41 is_cat = reply['headlines']['is_cat'];
42 feed_id = reply['headlines']['id'];
43
44 if (background) {
45 var content = reply['headlines']['content'];
46
47 content = content + "<div id='headlines-spacer'></div>";
48 return;
49 }
50
51 if (feed_id != -7 && (feed_id != getActiveFeedId() || is_cat != activeFeedIsCat()))
52 return;
53
54 /* dijit.getEnclosingWidget(
55 document.forms["main_toolbar_form"].update).attr('disabled',
56 is_cat || feed_id <= 0); */
57
58 try {
59 if (infscroll_req == false) {
60 $("headlines-frame").scrollTop = 0;
61
62 $("floatingTitle").style.visibility = "hidden";
63 $("floatingTitle").setAttribute("rowid", 0);
64 $("floatingTitle").innerHTML = "";
65 }
66 } catch (e) { };
67
68 $("headlines-frame").removeClassName("cdm");
69 $("headlines-frame").removeClassName("normal");
70
71 $("headlines-frame").addClassName(isCdmMode() ? "cdm" : "normal");
72
73 var headlines_count = reply['headlines-info']['count'];
74
75 vgroup_last_feed = reply['headlines-info']['vgroup_last_feed'];
76
77 if (parseInt(headlines_count) < 30) {
78 _infscroll_disable = 1;
79 } else {
80 _infscroll_disable = 0;
81 }
82
83 current_first_id = reply['headlines']['first_id'];
84 var counters = reply['counters'];
85 var articles = reply['articles'];
86 //var runtime_info = reply['runtime-info'];
87
88 if (infscroll_req == false) {
89 loaded_article_ids = [];
90
91 dojo.html.set($("headlines-toolbar"),
92 reply['headlines']['toolbar'],
93 {parseContent: true});
94
95 dojo.html.set($("headlines-frame"),
96 reply['headlines']['content'],
97 {parseContent: true});
98
99 $$("#headlines-frame div[id*='RROW']").each(function(row) {
100 loaded_article_ids.push(row.id);
101 });
102
103 var hsp = $("headlines-spacer");
104 if (!hsp) hsp = new Element("DIV", {"id": "headlines-spacer"});
105 dijit.byId('headlines-frame').domNode.appendChild(hsp);
106
107 initHeadlinesMenu();
108
109 if (_infscroll_disable)
110 hsp.innerHTML = "<a href='#' onclick='openNextUnreadFeed()'>" +
111 __("Click to open next unread feed.") + "</a>";
112
113 if (_search_query) {
114 $("feed_title").innerHTML += "<span id='cancel_search'>" +
115 " (<a href='#' onclick='cancelSearch()'>" + __("Cancel search") + "</a>)" +
116 "</span>";
117 }
118
119 } else {
120
121 if (headlines_count > 0 && feed_id == getActiveFeedId() && is_cat == activeFeedIsCat()) {
122 console.log("adding some more headlines: " + headlines_count);
123
124 var c = dijit.byId("headlines-frame");
125 var ids = getSelectedArticleIds2();
126
127 var hsp = $("headlines-spacer");
128
129 if (hsp)
130 c.domNode.removeChild(hsp);
131
132 var tmp = new Element("div");
133 tmp.innerHTML = reply['headlines']['content'];
134 dojo.parser.parse(tmp);
135
136 while (tmp.hasChildNodes()) {
137 var row = tmp.removeChild(tmp.firstChild);
138
139 if (loaded_article_ids.indexOf(row.id) == -1 || row.hasClassName("cdmFeedTitle")) {
140 dijit.byId("headlines-frame").domNode.appendChild(row);
141 //Element.hide(row);
142 //new Effect.Appear(row, {duration:0.5});
143 loaded_article_ids.push(row.id);
144 }
145 }
146
147 if (!hsp) hsp = new Element("DIV", {"id": "headlines-spacer"});
148 c.domNode.appendChild(hsp);
149
150 if (headlines_count < 30) _infscroll_disable = true;
151
152 console.log("restore selected ids: " + ids);
153
154 for (var i = 0; i < ids.length; i++) {
155 markHeadline(ids[i]);
156 }
157
158 initHeadlinesMenu();
159
160 if (_infscroll_disable) {
161 hsp.innerHTML = "<a href='#' onclick='openNextUnreadFeed()'>" +
162 __("Click to open next unread feed.") + "</a>";
163 }
164
165 } else {
166 console.log("no new headlines received");
167
168 var first_id_changed = reply['headlines']['first_id_changed'];
169 console.log("first id changed:" + first_id_changed);
170
171 var hsp = $("headlines-spacer");
172
173 if (hsp) {
174 if (first_id_changed) {
175 hsp.innerHTML = "<a href='#' onclick='viewCurrentFeed()'>" +
176 __("New articles found, reload feed to continue.") + "</a>";
177 } else {
178 hsp.innerHTML = "<a href='#' onclick='openNextUnreadFeed()'>" +
179 __("Click to open next unread feed.") + "</a>";
180 }
181
182 }
183
184 }
185 }
186
187 if (articles) {
188 for (var i = 0; i < articles.length; i++) {
189 var a_id = articles[i]['id'];
190 cache_set("article:" + a_id, articles[i]['content']);
191 }
192 } else {
193 console.log("no cached articles received");
194 }
195
196 if (counters)
197 parse_counters(counters);
198 else
199 request_counters(true);
200
201 } else {
202 console.error("Invalid object received: " + transport.responseText);
203 dijit.byId("headlines-frame").attr('content', "<div class='whiteBox'>" +
204 __('Could not update headlines (invalid object received - see error console for details)') +
205 "</div>");
206 }
207
208 _infscroll_request_sent = 0;
209 _last_headlines_update = new Date().getTime();
210
211 unpackVisibleHeadlines();
212
213 // if we have some more space in the buffer, why not try to fill it
214
215 if (!_infscroll_disable && $("headlines-spacer") &&
216 $("headlines-spacer").offsetTop < $("headlines-frame").offsetHeight) {
217
218 window.setTimeout(function() {
219 loadMoreHeadlines();
220 }, 250);
221 }
222
223 notify("");
224
225 } catch (e) {
226 exception_error("headlines_callback2", e, transport);
227 }
228 }
229
230 function render_article(article) {
231 try {
232 dijit.byId("headlines-wrap-inner").addChild(
233 dijit.byId("content-insert"));
234
235 var c = dijit.byId("content-insert");
236
237 try {
238 c.domNode.scrollTop = 0;
239 } catch (e) { };
240
241 c.attr('content', article);
242 PluginHost.run(PluginHost.HOOK_ARTICLE_RENDERED, c.domNode);
243
244 correctHeadlinesOffset(getActiveArticleId());
245
246 try {
247 c.focus();
248 } catch (e) { };
249
250 } catch (e) {
251 exception_error("render_article", e);
252 }
253 }
254
255 function showArticleInHeadlines(id, noexpand) {
256
257 try {
258 selectArticles("none");
259
260 var crow = $("RROW-" + id);
261
262 if (!crow) return;
263
264 var article_is_unread = crow.hasClassName("Unread");
265
266 if (!noexpand)
267 crow.removeClassName("Unread");
268 crow.addClassName("active");
269
270 selectArticles('none');
271
272 var view_mode = false;
273
274 try {
275 view_mode = document.forms['main_toolbar_form'].view_mode;
276 view_mode = view_mode[view_mode.selectedIndex].value;
277 } catch (e) {
278 //
279 }
280
281 markHeadline(id);
282
283 if (article_is_unread && !noexpand)
284 _force_scheduled_update = true;
285
286 } catch (e) {
287 exception_error("showArticleInHeadlines", e);
288 }
289 }
290
291 function article_callback2(transport, id) {
292 try {
293 console.log("article_callback2 " + id);
294
295 handle_rpc_json(transport);
296
297 var reply = false;
298
299 try {
300 reply = JSON.parse(transport.responseText);
301 } catch (e) {
302 console.error(e);
303 }
304
305 if (reply) {
306
307 reply.each(function(article) {
308 if (getActiveArticleId() == article['id']) {
309 render_article(article['content']);
310 }
311 cids_requested.remove(article['id']);
312
313 cache_set("article:" + article['id'], article['content']);
314 });
315
316 // if (id != last_requested_article) {
317 // console.log("requested article id is out of sequence, aborting");
318 // return;
319 // }
320
321 } else {
322 console.error("Invalid object received: " + transport.responseText);
323
324 render_article("<div class='whiteBox'>" +
325 __('Could not display article (invalid object received - see error console for details)') + "</div>");
326 }
327
328 var unread_in_buffer = $$("#headlines-frame > div[id*=RROW][class*=Unread]").length
329 request_counters(unread_in_buffer == 0);
330
331 notify("");
332 } catch (e) {
333 exception_error("article_callback2", e, transport);
334 }
335 }
336
337 function view(id, activefeed, noexpand) {
338 try {
339 var oldrow = $("RROW-" + getActiveArticleId());
340 if (oldrow) oldrow.removeClassName("active");
341
342 var crow = $("RROW-" + id);
343
344 if (!crow) return;
345 if (noexpand) {
346 setActiveArticleId(id);
347 showArticleInHeadlines(id, noexpand);
348 return;
349 }
350
351 console.log("loading article: " + id);
352
353 var cached_article = cache_get("article:" + id);
354
355 console.log("cache check result: " + (cached_article != false));
356
357 var query = "?op=article&method=view&id=" + param_escape(id);
358
359 var neighbor_ids = getRelativePostIds(id);
360
361 /* only request uncached articles */
362
363 var cids_to_request = [];
364
365 for (var i = 0; i < neighbor_ids.length; i++) {
366 if (cids_requested.indexOf(neighbor_ids[i]) == -1)
367 if (!cache_get("article:" + neighbor_ids[i])) {
368 cids_to_request.push(neighbor_ids[i]);
369 cids_requested.push(neighbor_ids[i]);
370 }
371 }
372
373 console.log("additional ids: " + cids_to_request.toString());
374
375 query = query + "&cids=" + cids_to_request.toString();
376
377 var article_is_unread = crow.hasClassName("Unread");
378
379 setActiveArticleId(id);
380 showArticleInHeadlines(id);
381
382 if (cached_article && article_is_unread) {
383
384 query = query + "&mode=prefetch";
385
386 render_article(cached_article);
387
388 } else if (cached_article) {
389
390 query = query + "&mode=prefetch_old";
391 render_article(cached_article);
392
393 // if we don't need to request any relative ids, we might as well skip
394 // the server roundtrip altogether
395 if (cids_to_request.length == 0) {
396 return;
397 }
398 }
399
400 last_requested_article = id;
401
402 console.log(query);
403
404 if (article_is_unread) {
405 decrementFeedCounter(getActiveFeedId(), activeFeedIsCat());
406 }
407
408 new Ajax.Request("backend.php", {
409 parameters: query,
410 onComplete: function(transport) {
411 article_callback2(transport, id);
412 } });
413
414 return false;
415
416 } catch (e) {
417 exception_error("view", e);
418 }
419 }
420
421 function toggleMark(id, client_only) {
422 try {
423 var query = "?op=rpc&id=" + id + "&method=mark";
424
425 var row = $("RROW-" + id);
426 if (!row) return;
427
428 var imgs = [];
429
430 var row_imgs = row.getElementsByClassName("markedPic");
431
432 for (var i = 0; i < row_imgs.length; i++)
433 imgs.push(row_imgs[i]);
434
435 var ft = $("floatingTitle");
436
437 if (ft && ft.getAttribute("rowid") == "RROW-" + id) {
438 var fte = ft.getElementsByClassName("markedPic");
439
440 for (var i = 0; i < fte.length; i++)
441 imgs.push(fte[i]);
442 }
443
444 for (i = 0; i < imgs.length; i++) {
445 var img = imgs[i];
446
447 if (!row.hasClassName("marked")) {
448 img.src = img.src.replace("mark_unset", "mark_set");
449 img.alt = __("Unstar article");
450 query = query + "&mark=1";
451 } else {
452 img.src = img.src.replace("mark_set", "mark_unset");
453 img.alt = __("Star article");
454 query = query + "&mark=0";
455 }
456 }
457
458 row.toggleClassName("marked");
459
460 if (!client_only) {
461 new Ajax.Request("backend.php", {
462 parameters: query,
463 onComplete: function(transport) {
464 handle_rpc_json(transport);
465 } });
466 }
467
468 } catch (e) {
469 exception_error("toggleMark", e);
470 }
471 }
472
473 function togglePub(id, client_only, no_effects, note) {
474 try {
475 var query = "?op=rpc&id=" + id + "&method=publ";
476
477 if (note != undefined) {
478 query = query + "&note=" + param_escape(note);
479 } else {
480 query = query + "&note=undefined";
481 }
482
483 var row = $("RROW-" + id);
484 if (!row) return;
485
486 var imgs = [];
487
488 var row_imgs = row.getElementsByClassName("pubPic");
489
490 for (var i = 0; i < row_imgs.length; i++)
491 imgs.push(row_imgs[i]);
492
493 var ft = $("floatingTitle");
494
495 if (ft && ft.getAttribute("rowid") == "RROW-" + id) {
496 var fte = ft.getElementsByClassName("pubPic");
497
498 for (var i = 0; i < fte.length; i++)
499 imgs.push(fte[i]);
500 }
501
502 for (i = 0; i < imgs.length; i++) {
503 var img = imgs[i];
504
505 if (!row.hasClassName("published") || note != undefined) {
506 img.src = img.src.replace("pub_unset", "pub_set");
507 img.alt = __("Unpublish article");
508 query = query + "&pub=1";
509 } else {
510 img.src = img.src.replace("pub_set", "pub_unset");
511 img.alt = __("Publish article");
512 query = query + "&pub=0";
513 }
514 }
515
516 if (note != undefined)
517 row.addClassName("published");
518 else
519 row.toggleClassName("published");
520
521 if (!client_only) {
522 new Ajax.Request("backend.php", {
523 parameters: query,
524 onComplete: function(transport) {
525 handle_rpc_json(transport);
526 } });
527 }
528
529 } catch (e) {
530 exception_error("togglePub", e);
531 }
532 }
533
534 function moveToPost(mode, noscroll, noexpand) {
535
536 try {
537
538 var rows = getVisibleArticleIds();
539
540 var prev_id = false;
541 var next_id = false;
542
543 if (!$('RROW-' + getActiveArticleId())) {
544 setActiveArticleId(0);
545 }
546
547 if (!getActiveArticleId()) {
548 next_id = rows[0];
549 prev_id = rows[rows.length-1]
550 } else {
551 for (var i = 0; i < rows.length; i++) {
552 if (rows[i] == getActiveArticleId()) {
553
554 // Account for adjacent identical article ids.
555 if (i > 0) prev_id = rows[i-1];
556
557 for (var j = i+1; j < rows.length; j++) {
558 if (rows[j] != getActiveArticleId()) {
559 next_id = rows[j];
560 break;
561 }
562 }
563 break;
564 }
565 }
566 }
567
568 if (mode == "next") {
569 if (next_id || getActiveArticleId()) {
570 if (isCdmMode()) {
571
572 var article = $("RROW-" + getActiveArticleId());
573 var ctr = $("headlines-frame");
574
575 if (!noscroll && article && article.offsetTop + article.offsetHeight >
576 ctr.scrollTop + ctr.offsetHeight) {
577
578 scrollArticle(ctr.offsetHeight/4);
579
580 } else if (next_id) {
581 cdmExpandArticle(next_id, noexpand);
582 cdmScrollToArticleId(next_id, true);
583 }
584
585 } else if (next_id) {
586 correctHeadlinesOffset(next_id);
587 view(next_id, getActiveFeedId(), noexpand);
588 }
589 }
590 }
591
592 if (mode == "prev") {
593 if (prev_id || getActiveArticleId()) {
594 if (isCdmMode()) {
595
596 var article = $("RROW-" + getActiveArticleId());
597 var prev_article = $("RROW-" + prev_id);
598 var ctr = $("headlines-frame");
599
600 if (!getInitParam("cdm_expanded")) {
601
602 if (!noscroll && article && article.offsetTop < ctr.scrollTop) {
603 scrollArticle(-ctr.offsetHeight/4);
604 } else {
605 cdmExpandArticle(prev_id, noexpand);
606 cdmScrollToArticleId(prev_id, true);
607 }
608 } else {
609
610 if (!noscroll && article && article.offsetTop < ctr.scrollTop) {
611 scrollArticle(-ctr.offsetHeight/3);
612 } else if (!noscroll && prev_article &&
613 prev_article.offsetTop < ctr.scrollTop) {
614 cdmExpandArticle(prev_id, noexpand);
615 scrollArticle(-ctr.offsetHeight/4);
616 } else if (prev_id) {
617 cdmExpandArticle(prev_id, noexpand);
618 cdmScrollToArticleId(prev_id, noscroll);
619 }
620 }
621
622 } else if (prev_id) {
623 correctHeadlinesOffset(prev_id);
624 view(prev_id, getActiveFeedId(), noexpand);
625 }
626 }
627 }
628
629 } catch (e) {
630 exception_error("moveToPost", e);
631 }
632 }
633
634 function toggleSelected(id, force_on) {
635 try {
636 var row = $("RROW-" + id);
637
638 if (row) {
639 var cb = dijit.getEnclosingWidget(
640 row.getElementsByClassName("rchk")[0]);
641
642 if (row.hasClassName('Selected') && !force_on) {
643 row.removeClassName('Selected');
644 if (cb) cb.attr("checked", false);
645 } else {
646 row.addClassName('Selected');
647 if (cb) cb.attr("checked", true);
648 }
649 }
650
651 updateSelectedPrompt();
652 } catch (e) {
653 exception_error("toggleSelected", e);
654 }
655 }
656
657 function updateSelectedPrompt() {
658 try {
659 var count = getSelectedArticleIds2().size();
660 var elem = $("selected_prompt");
661
662 if (elem) {
663 elem.innerHTML = ngettext("%d article selected",
664 "%d articles selected", count).replace("%d", count);
665
666 if (count > 0)
667 Element.show(elem);
668 else
669 Element.hide(elem);
670 }
671
672 } catch (e) {
673 exception_error("updateSelectedPrompt", e);
674 }
675 }
676
677 function toggleUnread_afh(effect) {
678 try {
679
680 var elem = effect.element;
681 elem.style.backgroundColor = "";
682
683 } catch (e) {
684 exception_error("toggleUnread_afh", e);
685 }
686 }
687
688 function toggleUnread(id, cmode, effect) {
689 try {
690
691 var row = $("RROW-" + id);
692 if (row) {
693 if (cmode == undefined || cmode == 2) {
694 if (row.hasClassName("Unread")) {
695 row.removeClassName("Unread");
696
697 } else {
698 row.addClassName("Unread");
699 }
700
701 } else if (cmode == 0) {
702
703 row.removeClassName("Unread");
704
705 } else if (cmode == 1) {
706 row.addClassName("Unread");
707 }
708
709 if (cmode == undefined) cmode = 2;
710
711 var query = "?op=rpc&method=catchupSelected" +
712 "&cmode=" + param_escape(cmode) + "&ids=" + param_escape(id);
713
714 // notify_progress("Loading, please wait...");
715
716 new Ajax.Request("backend.php", {
717 parameters: query,
718 onComplete: function(transport) {
719 handle_rpc_json(transport);
720 } });
721
722 }
723
724 } catch (e) {
725 exception_error("toggleUnread", e);
726 }
727 }
728
729 function selectionRemoveLabel(id, ids) {
730 try {
731
732 if (!ids) ids = getSelectedArticleIds2();
733
734 if (ids.length == 0) {
735 alert(__("No articles are selected."));
736 return;
737 }
738
739 var query = "?op=article&method=removeFromLabel&ids=" +
740 param_escape(ids.toString()) + "&lid=" + param_escape(id);
741
742 console.log(query);
743
744 new Ajax.Request("backend.php", {
745 parameters: query,
746 onComplete: function(transport) {
747 handle_rpc_json(transport);
748 show_labels_in_headlines(transport);
749 } });
750
751 } catch (e) {
752 exception_error("selectionAssignLabel", e);
753
754 }
755 }
756
757 function selectionAssignLabel(id, ids) {
758 try {
759
760 if (!ids) ids = getSelectedArticleIds2();
761
762 if (ids.length == 0) {
763 alert(__("No articles are selected."));
764 return;
765 }
766
767 var query = "?op=article&method=assignToLabel&ids=" +
768 param_escape(ids.toString()) + "&lid=" + param_escape(id);
769
770 console.log(query);
771
772 new Ajax.Request("backend.php", {
773 parameters: query,
774 onComplete: function(transport) {
775 handle_rpc_json(transport);
776 show_labels_in_headlines(transport);
777 } });
778
779 } catch (e) {
780 exception_error("selectionAssignLabel", e);
781
782 }
783 }
784
785 function selectionToggleUnread(set_state, callback, no_error, ids) {
786 try {
787 var rows = ids ? ids : getSelectedArticleIds2();
788
789 if (rows.length == 0 && !no_error) {
790 alert(__("No articles are selected."));
791 return;
792 }
793
794 for (var i = 0; i < rows.length; i++) {
795 var row = $("RROW-" + rows[i]);
796 if (row) {
797 if (set_state == undefined) {
798 if (row.hasClassName("Unread")) {
799 row.removeClassName("Unread");
800 } else {
801 row.addClassName("Unread");
802 }
803 }
804
805 if (set_state == false) {
806 row.removeClassName("Unread");
807 }
808
809 if (set_state == true) {
810 row.addClassName("Unread");
811 }
812 }
813 }
814
815 updateFloatingTitle(true);
816
817 if (rows.length > 0) {
818
819 var cmode = "";
820
821 if (set_state == undefined) {
822 cmode = "2";
823 } else if (set_state == true) {
824 cmode = "1";
825 } else if (set_state == false) {
826 cmode = "0";
827 }
828
829 var query = "?op=rpc&method=catchupSelected" +
830 "&cmode=" + cmode + "&ids=" + param_escape(rows.toString());
831
832 notify_progress("Loading, please wait...");
833
834 new Ajax.Request("backend.php", {
835 parameters: query,
836 onComplete: function(transport) {
837 handle_rpc_json(transport);
838 if (callback) callback(transport);
839 } });
840
841 }
842
843 } catch (e) {
844 exception_error("selectionToggleUnread", e);
845 }
846 }
847
848 // sel_state ignored
849 function selectionToggleMarked(sel_state, callback, no_error, ids) {
850 try {
851
852 var rows = ids ? ids : getSelectedArticleIds2();
853
854 if (rows.length == 0 && !no_error) {
855 alert(__("No articles are selected."));
856 return;
857 }
858
859 for (var i = 0; i < rows.length; i++) {
860 toggleMark(rows[i], true, true);
861 }
862
863 if (rows.length > 0) {
864
865 var query = "?op=rpc&method=markSelected&ids=" +
866 param_escape(rows.toString()) + "&cmode=2";
867
868 new Ajax.Request("backend.php", {
869 parameters: query,
870 onComplete: function(transport) {
871 handle_rpc_json(transport);
872 if (callback) callback(transport);
873 } });
874
875 }
876
877 } catch (e) {
878 exception_error("selectionToggleMarked", e);
879 }
880 }
881
882 // sel_state ignored
883 function selectionTogglePublished(sel_state, callback, no_error, ids) {
884 try {
885
886 var rows = ids ? ids : getSelectedArticleIds2();
887
888 if (rows.length == 0 && !no_error) {
889 alert(__("No articles are selected."));
890 return;
891 }
892
893 for (var i = 0; i < rows.length; i++) {
894 togglePub(rows[i], true, true);
895 }
896
897 if (rows.length > 0) {
898
899 var query = "?op=rpc&method=publishSelected&ids=" +
900 param_escape(rows.toString()) + "&cmode=2";
901
902 new Ajax.Request("backend.php", {
903 parameters: query,
904 onComplete: function(transport) {
905 handle_rpc_json(transport);
906 } });
907
908 }
909
910 } catch (e) {
911 exception_error("selectionToggleMarked", e);
912 }
913 }
914
915 function getSelectedArticleIds2() {
916
917 var rv = [];
918
919 $$("#headlines-frame > div[id*=RROW][class*=Selected]").each(
920 function(child) {
921 rv.push(child.id.replace("RROW-", ""));
922 });
923
924 return rv;
925 }
926
927 function getLoadedArticleIds() {
928 var rv = [];
929
930 var children = $$("#headlines-frame > div[id*=RROW-]");
931
932 children.each(function(child) {
933 rv.push(child.id.replace("RROW-", ""));
934 });
935
936 return rv;
937
938 }
939
940 // mode = all,none,unread,invert,marked,published
941 function selectArticles(mode, query) {
942 try {
943
944 if (!query) query = "#headlines-frame > div[id*=RROW]";
945
946 var children = $$(query);
947
948 children.each(function(child) {
949 var id = child.id.replace("RROW-", "");
950
951 var cb = dijit.getEnclosingWidget(
952 child.getElementsByClassName("rchk")[0]);
953
954 if (mode == "all") {
955 child.addClassName("Selected");
956 if (cb) cb.attr("checked", true);
957 } else if (mode == "unread") {
958 if (child.hasClassName("Unread")) {
959 child.addClassName("Selected");
960 if (cb) cb.attr("checked", true);
961 } else {
962 child.removeClassName("Selected");
963 if (cb) cb.attr("checked", false);
964 }
965 } else if (mode == "marked") {
966 if (child.hasClassName("marked")) {
967 child.addClassName("Selected");
968 if (cb) cb.attr("checked", true);
969 } else {
970 child.removeClassName("Selected");
971 if (cb) cb.attr("checked", false);
972 }
973 } else if (mode == "published") {
974 if (child.hasClassName("published")) {
975 child.addClassName("Selected");
976 if (cb) cb.attr("checked", true);
977 } else {
978 child.removeClassName("Selected");
979 if (cb) cb.attr("checked", false);
980 }
981
982 } else if (mode == "invert") {
983 if (child.hasClassName("Selected")) {
984 child.removeClassName("Selected");
985 if (cb) cb.attr("checked", false);
986 } else {
987 child.addClassName("Selected");
988 if (cb) cb.attr("checked", true);
989 }
990
991 } else {
992 child.removeClassName("Selected");
993 if (cb) cb.attr("checked", false);
994 }
995 });
996
997 updateSelectedPrompt();
998
999 } catch (e) {
1000 exception_error("selectArticles", e);
1001 }
1002 }
1003
1004 function deleteSelection() {
1005
1006 try {
1007
1008 var rows = getSelectedArticleIds2();
1009
1010 if (rows.length == 0) {
1011 alert(__("No articles are selected."));
1012 return;
1013 }
1014
1015 var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
1016 var str;
1017
1018 if (getActiveFeedId() != 0) {
1019 str = ngettext("Delete %d selected article in %s?", "Delete %d selected articles in %s?" , rows.length);
1020 } else {
1021 str = ngettext("Delete %d selected article?", "Delete %d selected articles?", rows.length);
1022 }
1023
1024 str = str.replace("%d", rows.length);
1025 str = str.replace("%s", fn);
1026
1027 if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
1028 return;
1029 }
1030
1031 query = "?op=rpc&method=delete&ids=" + param_escape(rows);
1032
1033 console.log(query);
1034
1035 new Ajax.Request("backend.php", {
1036 parameters: query,
1037 onComplete: function(transport) {
1038 handle_rpc_json(transport);
1039 viewCurrentFeed();
1040 } });
1041
1042 } catch (e) {
1043 exception_error("deleteSelection", e);
1044 }
1045 }
1046
1047 function archiveSelection() {
1048
1049 try {
1050
1051 var rows = getSelectedArticleIds2();
1052
1053 if (rows.length == 0) {
1054 alert(__("No articles are selected."));
1055 return;
1056 }
1057
1058 var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
1059 var str;
1060 var op;
1061
1062 if (getActiveFeedId() != 0) {
1063 str = ngettext("Archive %d selected article in %s?", "Archive %d selected articles in %s?", rows.length);
1064 op = "archive";
1065 } else {
1066 str = ngettext("Move %d archived article back?", "Move %d archived articles back?", rows.length);
1067
1068 str += " " + __("Please note that unstarred articles might get purged on next feed update.");
1069
1070 op = "unarchive";
1071 }
1072
1073 str = str.replace("%d", rows.length);
1074 str = str.replace("%s", fn);
1075
1076 if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
1077 return;
1078 }
1079
1080 query = "?op=rpc&method="+op+"&ids=" + param_escape(rows);
1081
1082 console.log(query);
1083
1084 for (var i = 0; i < rows.length; i++) {
1085 cache_delete("article:" + rows[i]);
1086 }
1087
1088 new Ajax.Request("backend.php", {
1089 parameters: query,
1090 onComplete: function(transport) {
1091 handle_rpc_json(transport);
1092 viewCurrentFeed();
1093 } });
1094
1095 } catch (e) {
1096 exception_error("archiveSelection", e);
1097 }
1098 }
1099
1100 function catchupSelection() {
1101
1102 try {
1103
1104 var rows = getSelectedArticleIds2();
1105
1106 if (rows.length == 0) {
1107 alert(__("No articles are selected."));
1108 return;
1109 }
1110
1111 var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
1112
1113 var str = ngettext("Mark %d selected article in %s as read?", "Mark %d selected articles in %s as read?", rows.length);
1114
1115 str = str.replace("%d", rows.length);
1116 str = str.replace("%s", fn);
1117
1118 if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
1119 return;
1120 }
1121
1122 selectionToggleUnread(false, 'viewCurrentFeed()', true);
1123
1124 } catch (e) {
1125 exception_error("catchupSelection", e);
1126 }
1127 }
1128
1129 function editArticleTags(id) {
1130 var query = "backend.php?op=article&method=editArticleTags&param=" + param_escape(id);
1131
1132 if (dijit.byId("editTagsDlg"))
1133 dijit.byId("editTagsDlg").destroyRecursive();
1134
1135 dialog = new dijit.Dialog({
1136 id: "editTagsDlg",
1137 title: __("Edit article Tags"),
1138 style: "width: 600px",
1139 execute: function() {
1140 if (this.validate()) {
1141 var query = dojo.objectToQuery(this.attr('value'));
1142
1143 notify_progress("Saving article tags...", true);
1144
1145 new Ajax.Request("backend.php", {
1146 parameters: query,
1147 onComplete: function(transport) {
1148 try {
1149 notify('');
1150 dialog.hide();
1151
1152 var data = JSON.parse(transport.responseText);
1153
1154 if (data) {
1155 var id = data.id;
1156
1157 console.log(id);
1158
1159 var tags = $("ATSTR-" + id);
1160 var tooltip = dijit.byId("ATSTRTIP-" + id);
1161
1162 if (tags) tags.innerHTML = data.content;
1163 if (tooltip) tooltip.attr('label', data.content_full);
1164 }
1165 } catch (e) {
1166 exception_error("editArticleTags/inner", e);
1167 }
1168
1169 }});
1170 }
1171 },
1172 href: query
1173 });
1174
1175 var tmph = dojo.connect(dialog, 'onLoad', function() {
1176 dojo.disconnect(tmph);
1177
1178 new Ajax.Autocompleter('tags_str', 'tags_choices',
1179 "backend.php?op=article&method=completeTags",
1180 { tokens: ',', paramName: "search" });
1181 });
1182
1183 dialog.show();
1184
1185 }
1186
1187 function cdmScrollToArticleId(id, force) {
1188 try {
1189 var ctr = $("headlines-frame");
1190 var e = $("RROW-" + id);
1191
1192 if (!e || !ctr) return;
1193
1194 if (force || e.offsetTop+e.offsetHeight > (ctr.scrollTop+ctr.offsetHeight) ||
1195 e.offsetTop < ctr.scrollTop) {
1196
1197 // expanded cdm has a 4px margin now
1198 ctr.scrollTop = parseInt(e.offsetTop) - 4;
1199 }
1200
1201 } catch (e) {
1202 exception_error("cdmScrollToArticleId", e);
1203 }
1204 }
1205
1206 function setActiveArticleId(id) {
1207 _active_article_id = id;
1208 PluginHost.run(PluginHost.HOOK_ARTICLE_SET_ACTIVE, _active_article_id);
1209 }
1210
1211 function getActiveArticleId() {
1212 return _active_article_id;
1213 }
1214
1215 function postMouseIn(e, id) {
1216 post_under_pointer = id;
1217 }
1218
1219 function postMouseOut(id) {
1220 post_under_pointer = false;
1221 }
1222
1223 function unpackVisibleHeadlines() {
1224 try {
1225 if (!isCdmMode() || !getInitParam("cdm_expanded")) return;
1226
1227 $$("#headlines-frame > div[id*=RROW]").each(
1228 function(child) {
1229 if (child.offsetTop <= $("headlines-frame").scrollTop +
1230 $("headlines-frame").offsetHeight) {
1231
1232 var cencw = $("CENCW-" + child.id.replace("RROW-", ""));
1233
1234 if (cencw) {
1235 cencw.innerHTML = htmlspecialchars_decode(cencw.innerHTML);
1236 cencw.setAttribute('id', '');
1237
1238 PluginHost.run(PluginHost.HOOK_ARTICLE_RENDERED_CDM, child);
1239
1240 Element.show(cencw);
1241 }
1242 }
1243 }
1244 );
1245
1246 } catch (e) {
1247 exception_error("unpackVisibleHeadlines", e);
1248 }
1249 }
1250
1251 function headlines_scroll_handler(e) {
1252 try {
1253 var hsp = $("headlines-spacer");
1254
1255 unpackVisibleHeadlines();
1256
1257 // set topmost child in the buffer as active
1258 if (isCdmMode() && getInitParam("cdm_auto_catchup") == 1 &&
1259 getSelectedArticleIds2().length <= 1 &&
1260 getInitParam("cdm_expanded")) {
1261 var rows = $$("#headlines-frame > div[id*=RROW]");
1262
1263 for (var i = 0; i < rows.length; i++) {
1264 var child = rows[i];
1265
1266 if ($("headlines-frame").scrollTop <= child.offsetTop &&
1267 child.offsetTop - $("headlines-frame").scrollTop < 100 &&
1268 child.id.replace("RROW-", "") != _active_article_id) {
1269
1270 if (_active_article_id) {
1271 var row = $("RROW-" + _active_article_id);
1272 if (row) row.removeClassName("active");
1273 }
1274
1275 _active_article_id = child.id.replace("RROW-", "");
1276 showArticleInHeadlines(_active_article_id, true);
1277 updateSelectedPrompt();
1278 break;
1279 }
1280 }
1281 }
1282
1283 if (!_infscroll_disable) {
1284 if (hsp && hsp.offsetTop - 100 <= e.scrollTop + e.offsetHeight) {
1285
1286 hsp.innerHTML = "<span class='loading'><img src='images/indicator_tiny.gif'> " +
1287 __("Loading, please wait...") + "</span>";
1288
1289 loadMoreHeadlines();
1290 return;
1291
1292 }
1293 }
1294
1295 if (isCdmMode()) {
1296 updateFloatingTitle();
1297 }
1298
1299 if (getInitParam("cdm_auto_catchup") == 1) {
1300
1301 // let's get DOM some time to settle down
1302 var ts = new Date().getTime();
1303 if (ts - _last_headlines_update < 100) return;
1304
1305 $$("#headlines-frame > div[id*=RROW][class*=Unread]").each(
1306 function(child) {
1307 if (child.hasClassName("Unread") && $("headlines-frame").scrollTop >
1308 (child.offsetTop + child.offsetHeight/2)) {
1309
1310 var id = child.id.replace("RROW-", "");
1311
1312 if (catchup_id_batch.indexOf(id) == -1)
1313 catchup_id_batch.push(id);
1314
1315 //console.log("auto_catchup_batch: " + catchup_id_batch.toString());
1316 }
1317
1318 });
1319
1320 if (catchup_id_batch.length > 0) {
1321 window.clearTimeout(catchup_timeout_id);
1322
1323 if (!_infscroll_request_sent) {
1324 if (catchup_id_batch.length < 10) {
1325 catchup_timeout_id = window.setTimeout('catchupBatchedArticles()',
1326 500);
1327 } else {
1328 catchupBatchedArticles();
1329 }
1330 }
1331
1332 catchupBatchedArticles();
1333 }
1334
1335 if (_infscroll_disable) {
1336 var child = $$("#headlines-frame div[id*=RROW]").last();
1337
1338 if (child && $("headlines-frame").scrollTop >
1339 (child.offsetTop + child.offsetHeight - 50)) {
1340
1341 console.log("we seem to be at an end");
1342
1343 if (getInitParam("on_catchup_show_next_feed") == "1") {
1344 openNextUnreadFeed();
1345 }
1346 }
1347 }
1348 }
1349
1350 } catch (e) {
1351 console.warn("headlines_scroll_handler: " + e);
1352 }
1353 }
1354
1355 function openNextUnreadFeed() {
1356 try {
1357 var is_cat = activeFeedIsCat();
1358 var nuf = getNextUnreadFeed(getActiveFeedId(), is_cat);
1359 if (nuf) viewfeed(nuf, '', is_cat);
1360 } catch (e) {
1361 exception_error("openNextUnreadFeed", e);
1362 }
1363 }
1364
1365 function catchupBatchedArticles() {
1366 try {
1367 if (catchup_id_batch.length > 0 && !_infscroll_request_sent && !_catchup_request_sent) {
1368
1369 // make a copy of the array
1370 var batch = catchup_id_batch.slice();
1371 var query = "?op=rpc&method=catchupSelected" +
1372 "&cmode=0&ids=" + param_escape(batch.toString());
1373
1374 console.log(query);
1375
1376 _catchup_request_sent = true;
1377
1378 new Ajax.Request("backend.php", {
1379 parameters: query,
1380 onComplete: function(transport) {
1381 handle_rpc_json(transport);
1382
1383 _catchup_request_sent = false;
1384
1385 reply = JSON.parse(transport.responseText);
1386 var batch = reply.ids;
1387
1388 batch.each(function(id) {
1389 console.log(id);
1390 var elem = $("RROW-" + id);
1391 if (elem) elem.removeClassName("Unread");
1392 catchup_id_batch.remove(id);
1393 });
1394
1395 updateFloatingTitle(true);
1396
1397 } });
1398 }
1399
1400 } catch (e) {
1401 exception_error("catchupBatchedArticles", e);
1402 }
1403 }
1404
1405 function catchupRelativeToArticle(below, id) {
1406
1407 try {
1408
1409 if (!id) id = getActiveArticleId();
1410
1411 if (!id) {
1412 alert(__("No article is selected."));
1413 return;
1414 }
1415
1416 var visible_ids = getVisibleArticleIds();
1417
1418 var ids_to_mark = new Array();
1419
1420 if (!below) {
1421 for (var i = 0; i < visible_ids.length; i++) {
1422 if (visible_ids[i] != id) {
1423 var e = $("RROW-" + visible_ids[i]);
1424
1425 if (e && e.hasClassName("Unread")) {
1426 ids_to_mark.push(visible_ids[i]);
1427 }
1428 } else {
1429 break;
1430 }
1431 }
1432 } else {
1433 for (var i = visible_ids.length-1; i >= 0; i--) {
1434 if (visible_ids[i] != id) {
1435 var e = $("RROW-" + visible_ids[i]);
1436
1437 if (e && e.hasClassName("Unread")) {
1438 ids_to_mark.push(visible_ids[i]);
1439 }
1440 } else {
1441 break;
1442 }
1443 }
1444 }
1445
1446 if (ids_to_mark.length == 0) {
1447 alert(__("No articles found to mark"));
1448 } else {
1449 var msg = ngettext("Mark %d article as read?", "Mark %d articles as read?", ids_to_mark.length).replace("%d", ids_to_mark.length);
1450
1451 if (getInitParam("confirm_feed_catchup") != 1 || confirm(msg)) {
1452
1453 for (var i = 0; i < ids_to_mark.length; i++) {
1454 var e = $("RROW-" + ids_to_mark[i]);
1455 e.removeClassName("Unread");
1456 }
1457
1458 var query = "?op=rpc&method=catchupSelected" +
1459 "&cmode=0" + "&ids=" + param_escape(ids_to_mark.toString());
1460
1461 new Ajax.Request("backend.php", {
1462 parameters: query,
1463 onComplete: function(transport) {
1464 handle_rpc_json(transport);
1465 } });
1466
1467 }
1468 }
1469
1470 } catch (e) {
1471 exception_error("catchupRelativeToArticle", e);
1472 }
1473 }
1474
1475 function cdmCollapseArticle(event, id, unmark) {
1476 try {
1477 if (unmark == undefined) unmark = true;
1478
1479 var row = $("RROW-" + id);
1480 var elem = $("CICD-" + id);
1481
1482 if (elem && row) {
1483 var collapse = $$("div#RROW-" + id +
1484 " span[class='collapseBtn']")[0];
1485
1486 Element.hide(elem);
1487 Element.show("CEXC-" + id);
1488 Element.hide(collapse);
1489
1490 if (unmark) {
1491 row.removeClassName("active");
1492
1493 markHeadline(id, false);
1494
1495 if (id == getActiveArticleId()) {
1496 setActiveArticleId(0);
1497 }
1498
1499 updateSelectedPrompt();
1500 }
1501
1502 if (event) Event.stop(event);
1503
1504 PluginHost.run(PluginHost.HOOK_ARTICLE_COLLAPSED, id);
1505
1506 if (row.offsetTop < $("headlines-frame").scrollTop)
1507 scrollToRowId(row.id);
1508
1509 $("floatingTitle").style.visibility = "hidden";
1510 $("floatingTitle").setAttribute("rowid", false);
1511 }
1512
1513 } catch (e) {
1514 exception_error("cdmCollapseArticle", e);
1515 }
1516 }
1517
1518 function cdmExpandArticle(id, noexpand) {
1519 try {
1520 if (getInitParam("cdm_expanded")) return;
1521
1522 console.log("cdmExpandArticle " + id);
1523
1524 if (!$("RROW-" + id)) return false;
1525
1526 var oldrow = $("RROW-" + getActiveArticleId());
1527
1528 var elem = $("CICD-" + getActiveArticleId());
1529
1530 if (id == getActiveArticleId() && Element.visible(elem))
1531 return true;
1532
1533 selectArticles("none");
1534
1535 var old_offset = $("RROW-" + id).offsetTop;
1536
1537 if (getActiveArticleId() && elem && !getInitParam("cdm_expanded")) {
1538 var collapse = $$("div#RROW-" + getActiveArticleId() +
1539 " span[class='collapseBtn']")[0];
1540
1541 Element.hide(elem);
1542 Element.show("CEXC-" + getActiveArticleId());
1543 Element.hide(collapse);
1544 }
1545
1546 if (oldrow) oldrow.removeClassName("active");
1547
1548 setActiveArticleId(id);
1549
1550 elem = $("CICD-" + id);
1551
1552 var collapse = $$("div#RROW-" + id +
1553 " span[class='collapseBtn']")[0];
1554
1555 var cencw = $("CENCW-" + id);
1556
1557 if (!Element.visible(elem) && !noexpand) {
1558 if (cencw) {
1559 cencw.innerHTML = htmlspecialchars_decode(cencw.innerHTML);
1560 cencw.setAttribute('id', '');
1561 Element.show(cencw);
1562 }
1563
1564 Element.show(elem);
1565 Element.hide("CEXC-" + id);
1566 Element.show(collapse);
1567 }
1568
1569 var new_offset = $("RROW-" + id).offsetTop;
1570
1571 if (old_offset > new_offset)
1572 $("headlines-frame").scrollTop -= (old_offset-new_offset);
1573
1574 if (!noexpand)
1575 toggleUnread(id, 0, true);
1576
1577 toggleSelected(id);
1578 $("RROW-" + id).addClassName("active");
1579
1580 PluginHost.run(PluginHost.HOOK_ARTICLE_EXPANDED, id);
1581
1582 } catch (e) {
1583 exception_error("cdmExpandArticle", e);
1584 }
1585
1586 return false;
1587 }
1588
1589 function getArticleUnderPointer() {
1590 return post_under_pointer;
1591 }
1592
1593 function scrollArticle(offset) {
1594 try {
1595 if (!isCdmMode()) {
1596 var ci = $("content-insert");
1597 if (ci) {
1598 ci.scrollTop += offset;
1599 }
1600 } else {
1601 var hi = $("headlines-frame");
1602 if (hi) {
1603 hi.scrollTop += offset;
1604 }
1605
1606 }
1607 } catch (e) {
1608 exception_error("scrollArticle", e);
1609 }
1610 }
1611
1612 function show_labels_in_headlines(transport) {
1613 try {
1614 var data = JSON.parse(transport.responseText);
1615
1616 if (data) {
1617 data['info-for-headlines'].each(function(elem) {
1618 $$(".HLLCTR-" + elem.id).each(function(ctr) {
1619 ctr.innerHTML = elem.labels;
1620 });
1621 });
1622 }
1623 } catch (e) {
1624 exception_error("show_labels_in_headlines", e);
1625 }
1626 }
1627
1628 function dismissArticle(id) {
1629 try {
1630 var elem = $("RROW-" + id);
1631
1632 if (!elem) return;
1633
1634 toggleUnread(id, 0, true);
1635
1636 new Effect.Fade(elem, {duration : 0.5});
1637
1638 // Remove the content, too
1639 var elem_content = $("CICD-" + id);
1640 if (elem_content) {
1641 Element.remove(elem_content);
1642 }
1643
1644 if (id == getActiveArticleId()) {
1645 setActiveArticleId(0);
1646 }
1647
1648 } catch (e) {
1649 exception_error("dismissArticle", e);
1650 }
1651 }
1652
1653 function dismissSelectedArticles() {
1654 try {
1655
1656 var ids = getVisibleArticleIds();
1657 var tmp = [];
1658 var sel = [];
1659
1660 for (var i = 0; i < ids.length; i++) {
1661 var elem = $("RROW-" + ids[i]);
1662
1663 if (elem.className && elem.hasClassName("Selected") &&
1664 ids[i] != getActiveArticleId()) {
1665 new Effect.Fade(elem, {duration : 0.5});
1666 sel.push(ids[i]);
1667
1668 // Remove the content, too
1669 var elem_content = $("CICD-" + ids[i]);
1670 if (elem_content) {
1671 Element.remove(elem_content);
1672 }
1673 } else {
1674 tmp.push(ids[i]);
1675 }
1676 }
1677
1678 if (sel.length > 0)
1679 selectionToggleUnread(false);
1680
1681
1682 } catch (e) {
1683 exception_error("dismissSelectedArticles", e);
1684 }
1685 }
1686
1687 function dismissReadArticles() {
1688 try {
1689
1690 var ids = getVisibleArticleIds();
1691 var tmp = [];
1692
1693 for (var i = 0; i < ids.length; i++) {
1694 var elem = $("RROW-" + ids[i]);
1695
1696 if (elem.className && !elem.hasClassName("Unread") &&
1697 !elem.hasClassName("Selected")) {
1698
1699 new Effect.Fade(elem, {duration : 0.5});
1700
1701 // Remove the content, too
1702 var elem_content = $("CICD-" + ids[i]);
1703 if (elem_content) {
1704 Element.remove(elem_content);
1705 }
1706 } else {
1707 tmp.push(ids[i]);
1708 }
1709 }
1710
1711 } catch (e) {
1712 exception_error("dismissReadArticles", e);
1713 }
1714 }
1715
1716 // we don't really hide rows anymore
1717 function getVisibleArticleIds() {
1718 return getLoadedArticleIds();
1719
1720 /*var ids = [];
1721
1722 try {
1723
1724 getLoadedArticleIds().each(function(id) {
1725 var elem = $("RROW-" + id);
1726 if (elem && Element.visible(elem))
1727 ids.push(id);
1728 });
1729
1730 } catch (e) {
1731 exception_error("getVisibleArticleIds", e);
1732 }
1733
1734 return ids; */
1735 }
1736
1737 function cdmClicked(event, id) {
1738 try {
1739 //var shift_key = event.shiftKey;
1740
1741 if (!event.ctrlKey) {
1742
1743 if (!getInitParam("cdm_expanded")) {
1744 return cdmExpandArticle(id);
1745 } else {
1746
1747 var elem = $("RROW-" + getActiveArticleId());
1748
1749 if (elem) elem.removeClassName("active");
1750
1751 selectArticles("none");
1752 toggleSelected(id);
1753
1754 var elem = $("RROW-" + id);
1755 var article_is_unread = elem.hasClassName("Unread");
1756
1757 elem.removeClassName("Unread");
1758 elem.addClassName("active");
1759
1760 setActiveArticleId(id);
1761
1762 if (article_is_unread) {
1763 decrementFeedCounter(getActiveFeedId(), activeFeedIsCat());
1764 updateFloatingTitle(true);
1765 }
1766
1767 var query = "?op=rpc&method=catchupSelected" +
1768 "&cmode=0&ids=" + param_escape(id);
1769
1770 new Ajax.Request("backend.php", {
1771 parameters: query,
1772 onComplete: function(transport) {
1773 handle_rpc_json(transport);
1774 } });
1775
1776 return !event.shiftKey;
1777 }
1778
1779 } else if (event.target.parents(".cdmHeader").length > 0) {
1780
1781 toggleSelected(id, true);
1782
1783 var elem = $("RROW-" + id);
1784 var article_is_unread = elem.hasClassName("Unread");
1785
1786 if (article_is_unread) {
1787 decrementFeedCounter(getActiveFeedId(), activeFeedIsCat());
1788 }
1789
1790 toggleUnread(id, 0, false);
1791
1792 openArticleInNewWindow(id);
1793 }
1794
1795 var unread_in_buffer = $$("#headlines-frame > div[id*=RROW][class*=Unread]").length
1796 request_counters(unread_in_buffer == 0);
1797
1798 } catch (e) {
1799 exception_error("cdmClicked");
1800 }
1801
1802 return false;
1803 }
1804
1805 function hlClicked(event, id) {
1806 try {
1807 if (event.which == 2) {
1808 view(id);
1809 return true;
1810 } else if (event.ctrlKey) {
1811 toggleSelected(id, true);
1812 toggleUnread(id, 0, false);
1813 openArticleInNewWindow(id);
1814 return false;
1815 } else {
1816 view(id);
1817 return false;
1818 }
1819
1820 } catch (e) {
1821 exception_error("hlClicked");
1822 }
1823 }
1824
1825 function getFirstVisibleHeadlineId() {
1826 var rows = getVisibleArticleIds();
1827 return rows[0];
1828
1829 }
1830
1831 function getLastVisibleHeadlineId() {
1832 var rows = getVisibleArticleIds();
1833 return rows[rows.length-1];
1834 }
1835
1836 function openArticleInNewWindow(id) {
1837 toggleUnread(id, 0, false);
1838 window.open("backend.php?op=article&method=redirect&id=" + id);
1839 }
1840
1841 function isCdmMode() {
1842 return getInitParam("combined_display_mode");
1843 }
1844
1845 function markHeadline(id, marked) {
1846 if (marked == undefined) marked = true;
1847
1848 var row = $("RROW-" + id);
1849 if (row) {
1850 var check = dijit.getEnclosingWidget(
1851 row.getElementsByClassName("rchk")[0]);
1852
1853 if (check) {
1854 check.attr("checked", marked);
1855 }
1856
1857 if (marked)
1858 row.addClassName("Selected");
1859 else
1860 row.removeClassName("Selected");
1861 }
1862 }
1863
1864 function getRelativePostIds(id, limit) {
1865
1866 var tmp = [];
1867
1868 try {
1869
1870 if (!limit) limit = 6; //3
1871
1872 var ids = getVisibleArticleIds();
1873
1874 for (var i = 0; i < ids.length; i++) {
1875 if (ids[i] == id) {
1876 for (var k = 1; k <= limit; k++) {
1877 //if (i > k-1) tmp.push(ids[i-k]);
1878 if (i < ids.length-k) tmp.push(ids[i+k]);
1879 }
1880 break;
1881 }
1882 }
1883
1884 } catch (e) {
1885 exception_error("getRelativePostIds", e);
1886 }
1887
1888 return tmp;
1889 }
1890
1891 function correctHeadlinesOffset(id) {
1892
1893 try {
1894
1895 var container = $("headlines-frame");
1896 var row = $("RROW-" + id);
1897
1898 if (!container || !row) return;
1899
1900 var viewport = container.offsetHeight;
1901
1902 var rel_offset_top = row.offsetTop - container.scrollTop;
1903 var rel_offset_bottom = row.offsetTop + row.offsetHeight - container.scrollTop;
1904
1905 //console.log("Rtop: " + rel_offset_top + " Rbtm: " + rel_offset_bottom);
1906 //console.log("Vport: " + viewport);
1907
1908 if (rel_offset_top <= 0 || rel_offset_top > viewport) {
1909 container.scrollTop = row.offsetTop;
1910 } else if (rel_offset_bottom > viewport) {
1911
1912 /* doesn't properly work with Opera in some cases because
1913 Opera fucks up element scrolling */
1914
1915 container.scrollTop = row.offsetTop + row.offsetHeight - viewport;
1916 }
1917
1918 } catch (e) {
1919 exception_error("correctHeadlinesOffset", e);
1920 }
1921
1922 }
1923
1924 function headlineActionsChange(elem) {
1925 try {
1926 eval(elem.value);
1927 elem.attr('value', 'false');
1928 } catch (e) {
1929 exception_error("headlineActionsChange", e);
1930 }
1931 }
1932
1933 function closeArticlePanel() {
1934
1935 if (dijit.byId("content-insert"))
1936 dijit.byId("headlines-wrap-inner").removeChild(
1937 dijit.byId("content-insert"));
1938 }
1939
1940 function initFloatingMenu() {
1941 try {
1942 if (dijit.byId("floatingMenu"))
1943 dijit.byId("floatingMenu").destroyRecursive();
1944
1945 var menu = new dijit.Menu({
1946 id: "floatingMenu",
1947 targetNodeIds: ["floatingTitle"]
1948 });
1949
1950 var id = $("floatingTitle").getAttribute("rowid").replace("RROW-", "");
1951
1952 headlinesMenuCommon(menu, id);
1953
1954 menu.startup();
1955 } catch (e) {
1956 exception_error("initFloatingMenu", e);
1957 }
1958 }
1959
1960 function headlinesMenuCommon(menu, base_id) {
1961 try {
1962
1963 menu.addChild(new dijit.MenuItem({
1964 label: __("Open original article"),
1965 onClick: function(event) {
1966 openArticleInNewWindow(base_id ? base_id : this.getParent().callerRowId);
1967 }}));
1968
1969 menu.addChild(new dijit.MenuItem({
1970 label: __("Display article URL"),
1971 onClick: function(event) {
1972 displayArticleUrl(base_id ? base_id : this.getParent().callerRowId);
1973 }}));
1974
1975 menu.addChild(new dijit.MenuSeparator());
1976
1977 menu.addChild(new dijit.MenuItem({
1978 label: __("Toggle unread"),
1979 onClick: function(event) {
1980 var ids = getSelectedArticleIds2();
1981 // cast to string
1982 var id = (base_id ? base_id : this.getParent().callerRowId) + "";
1983 ids = ids.size() != 0 && ids.indexOf(id) != -1 ? ids : [id];
1984
1985 selectionToggleUnread(undefined, false, true, ids);
1986 }}));
1987
1988 menu.addChild(new dijit.MenuItem({
1989 label: __("Toggle starred"),
1990 onClick: function(event) {
1991 var ids = getSelectedArticleIds2();
1992 // cast to string
1993 var id = (base_id ? base_id : this.getParent().callerRowId) + "";
1994 ids = ids.size() != 0 && ids.indexOf(id) != -1 ? ids : [id];
1995
1996 selectionToggleMarked(undefined, false, true, ids);
1997 }}));
1998
1999 menu.addChild(new dijit.MenuItem({
2000 label: __("Toggle published"),
2001 onClick: function(event) {
2002 var ids = getSelectedArticleIds2();
2003 // cast to string
2004 var id = (base_id ? base_id : this.getParent().callerRowId) + "";
2005 ids = ids.size() != 0 && ids.indexOf(id) != -1 ? ids : [id];
2006
2007 selectionTogglePublished(undefined, false, true, ids);
2008 }}));
2009
2010 menu.addChild(new dijit.MenuSeparator());
2011
2012 menu.addChild(new dijit.MenuItem({
2013 label: __("Mark above as read"),
2014 onClick: function(event) {
2015 catchupRelativeToArticle(0, base_id ? base_id : this.getParent().callerRowId);
2016 }}));
2017
2018 menu.addChild(new dijit.MenuItem({
2019 label: __("Mark below as read"),
2020 onClick: function(event) {
2021 catchupRelativeToArticle(1, base_id ? base_id : this.getParent().callerRowId);
2022 }}));
2023
2024
2025 var labels = dijit.byId("feedTree").model.getItemsInCategory(-2);
2026
2027 if (labels) {
2028
2029 menu.addChild(new dijit.MenuSeparator());
2030
2031 var labelAddMenu = new dijit.Menu({ownerMenu: menu});
2032 var labelDelMenu = new dijit.Menu({ownerMenu: menu});
2033
2034 labels.each(function(label) {
2035 var id = label.id[0];
2036 var bare_id = id.substr(id.indexOf(":")+1);
2037 var name = label.name[0];
2038
2039 bare_id = feed_to_label_id(bare_id);
2040
2041 labelAddMenu.addChild(new dijit.MenuItem({
2042 label: name,
2043 labelId: bare_id,
2044 onClick: function(event) {
2045 var ids = getSelectedArticleIds2();
2046 // cast to string
2047 var id = (base_id ? base_id : this.getParent().ownerMenu.callerRowId) + "";
2048
2049 ids = ids.size() != 0 && ids.indexOf(id) != -1 ? ids : [id];
2050
2051 selectionAssignLabel(this.labelId, ids);
2052 }}));
2053
2054 labelDelMenu.addChild(new dijit.MenuItem({
2055 label: name,
2056 labelId: bare_id,
2057 onClick: function(event) {
2058 var ids = getSelectedArticleIds2();
2059 // cast to string
2060 var id = (base_id ? base_id : this.getParent().ownerMenu.callerRowId) + "";
2061
2062 ids = ids.size() != 0 && ids.indexOf(id) != -1 ? ids : [id];
2063
2064 selectionRemoveLabel(this.labelId, ids);
2065 }}));
2066
2067 });
2068
2069 menu.addChild(new dijit.PopupMenuItem({
2070 label: __("Assign label"),
2071 popup: labelAddMenu
2072 }));
2073
2074 menu.addChild(new dijit.PopupMenuItem({
2075 label: __("Remove label"),
2076 popup: labelDelMenu
2077 }));
2078
2079 }
2080
2081
2082 } catch (e) {
2083 exception_error("headlinesMenuCommon", e);
2084 }
2085 }
2086
2087 function initHeadlinesMenu() {
2088 try {
2089 if (dijit.byId("headlinesMenu"))
2090 dijit.byId("headlinesMenu").destroyRecursive();
2091
2092 var ids = [];
2093
2094 if (!isCdmMode()) {
2095 nodes = $$("#headlines-frame > div[id*=RROW]");
2096 } else {
2097 nodes = $$("#headlines-frame span[id*=RTITLE]");
2098 }
2099
2100 nodes.each(function(node) {
2101 ids.push(node.id);
2102 });
2103
2104 var menu = new dijit.Menu({
2105 id: "headlinesMenu",
2106 targetNodeIds: ids
2107 });
2108
2109 var tmph = dojo.connect(menu, '_openMyself', function (event) {
2110 var callerNode = event.target, match = null, tries = 0;
2111
2112 while (match == null && callerNode && tries <= 3) {
2113 match = callerNode.id.match("^[A-Z]+[-]([0-9]+)$");
2114 callerNode = callerNode.parentNode;
2115 ++tries;
2116 }
2117
2118 if (match) this.callerRowId = parseInt(match[1]);
2119
2120 });
2121
2122 headlinesMenuCommon(menu, false);
2123
2124 menu.startup();
2125
2126 /* vgroup feed title menu */
2127
2128 var nodes = $$("#headlines-frame > div[class='cdmFeedTitle']");
2129 var ids = [];
2130
2131 nodes.each(function(node) {
2132 ids.push(node.id);
2133 });
2134
2135 if (ids.length > 0) {
2136 if (dijit.byId("headlinesFeedTitleMenu"))
2137 dijit.byId("headlinesFeedTitleMenu").destroyRecursive();
2138
2139 var menu = new dijit.Menu({
2140 id: "headlinesFeedTitleMenu",
2141 targetNodeIds: ids
2142 });
2143
2144 var tmph = dojo.connect(menu, '_openMyself', function (event) {
2145 var callerNode = event.target, match = null, tries = 0;
2146
2147 while (match == null && callerNode && tries <= 3) {
2148 console.log(callerNode.id);
2149
2150 match = callerNode.id.match("^[A-Z]+[-]([0-9]+)$");
2151 callerNode = callerNode.parentNode;
2152 ++tries;
2153
2154 console.log(match[1]);
2155 }
2156
2157 if (match) this.callerRowId = parseInt(match[1]);
2158
2159 });
2160
2161 menu.addChild(new dijit.MenuItem({
2162 label: __("Select articles in group"),
2163 onClick: function(event) {
2164 selectArticles("all",
2165 "#headlines-frame > div[id*=RROW]"+
2166 "[orig-feed-id='"+menu.callerRowId+"']");
2167
2168 }}));
2169
2170 menu.addChild(new dijit.MenuItem({
2171 label: __("Mark group as read"),
2172 onClick: function(event) {
2173 selectArticles("none");
2174 selectArticles("all",
2175 "#headlines-frame > div[id*=RROW]"+
2176 "[orig-feed-id='"+menu.callerRowId+"']");
2177
2178 catchupSelection();
2179 }}));
2180
2181
2182 menu.addChild(new dijit.MenuItem({
2183 label: __("Mark feed as read"),
2184 onClick: function(event) {
2185 catchupFeedInGroup(menu.callerRowId);
2186 }}));
2187
2188 menu.startup();
2189
2190 }
2191
2192 } catch (e) {
2193 exception_error("initHeadlinesMenu", e);
2194 }
2195 }
2196
2197 function cache_set(id, obj) {
2198 //console.log("cache_set: " + id);
2199 if (has_storage)
2200 try {
2201 sessionStorage[id] = obj;
2202 } catch (e) {
2203 sessionStorage.clear();
2204 }
2205 }
2206
2207 function cache_get(id) {
2208 if (has_storage)
2209 return sessionStorage[id];
2210 }
2211
2212 function cache_clear() {
2213 if (has_storage)
2214 sessionStorage.clear();
2215 }
2216
2217 function cache_delete(id) {
2218 if (has_storage)
2219 sessionStorage.removeItem(id);
2220 }
2221
2222 function cancelSearch() {
2223 try {
2224 _search_query = "";
2225 viewCurrentFeed();
2226 } catch (e) {
2227 exception_error("cancelSearch", e);
2228 }
2229 }
2230
2231 function setSelectionScore() {
2232 try {
2233 var ids = getSelectedArticleIds2();
2234
2235 if (ids.length > 0) {
2236 console.log(ids);
2237
2238 var score = prompt(__("Please enter new score for selected articles:"), score);
2239
2240 if (score != undefined) {
2241 var query = "op=article&method=setScore&id=" + param_escape(ids.toString()) +
2242 "&score=" + param_escape(score);
2243
2244 new Ajax.Request("backend.php", {
2245 parameters: query,
2246 onComplete: function(transport) {
2247 var reply = JSON.parse(transport.responseText);
2248 if (reply) {
2249 console.log(ids);
2250
2251 ids.each(function(id) {
2252 var row = $("RROW-" + id);
2253
2254 if (row) {
2255 var pic = row.getElementsByClassName("hlScorePic")[0];
2256
2257 if (pic) {
2258 pic.src = pic.src.replace(/score_.*?\.png/,
2259 reply["score_pic"]);
2260 pic.setAttribute("score", score);
2261 }
2262 }
2263 });
2264 }
2265 } });
2266 }
2267
2268 } else {
2269 alert(__("No articles are selected."));
2270 }
2271 } catch (e) {
2272 exception_error("setSelectionScore", e);
2273 }
2274 }
2275
2276 function updateScore(id) {
2277 try {
2278 var pic = $$("#RROW-" + id + " .hlScorePic")[0];
2279
2280 if (pic) {
2281
2282 var query = "op=article&method=getScore&id=" + param_escape(id);
2283
2284 new Ajax.Request("backend.php", {
2285 parameters: query,
2286 onComplete: function(transport) {
2287 console.log(transport.responseText);
2288
2289 var reply = JSON.parse(transport.responseText);
2290
2291 if (reply) {
2292 pic.src = pic.src.replace(/score_.*?\.png/, reply["score_pic"]);
2293 pic.setAttribute("score", reply["score"]);
2294 pic.setAttribute("title", reply["score"]);
2295 }
2296 } });
2297 }
2298
2299 } catch (e) {
2300 exception_error("updateScore", e);
2301 }
2302 }
2303
2304 function changeScore(id, pic) {
2305 try {
2306 var score = pic.getAttribute("score");
2307
2308 var new_score = prompt(__("Please enter new score for this article:"), score);
2309
2310 if (new_score != undefined) {
2311
2312 var query = "op=article&method=setScore&id=" + param_escape(id) +
2313 "&score=" + param_escape(new_score);
2314
2315 new Ajax.Request("backend.php", {
2316 parameters: query,
2317 onComplete: function(transport) {
2318 var reply = JSON.parse(transport.responseText);
2319
2320 if (reply) {
2321 pic.src = pic.src.replace(/score_.*?\.png/, reply["score_pic"]);
2322 pic.setAttribute("score", new_score);
2323 pic.setAttribute("title", new_score);
2324 }
2325 } });
2326 }
2327 } catch (e) {
2328 exception_error("changeScore", e);
2329 }
2330 }
2331
2332 function displayArticleUrl(id) {
2333 try {
2334 var query = "op=rpc&method=getlinktitlebyid&id=" + param_escape(id);
2335
2336 new Ajax.Request("backend.php", {
2337 parameters: query,
2338 onComplete: function(transport) {
2339 var reply = JSON.parse(transport.responseText);
2340
2341 if (reply && reply.link) {
2342 prompt(__("Article URL:"), reply.link);
2343 }
2344 } });
2345 } catch (e) {
2346 exception_error("changeScore", e);
2347 }
2348 }
2349
2350 function openSelectedAttachment(elem) {
2351 try {
2352 var url = elem[elem.selectedIndex].value;
2353
2354 if (url) {
2355 window.open(url);
2356 elem.selectedIndex = 0;
2357 }
2358
2359 } catch (e) {
2360 exception_error("openSelectedAttachment", e);
2361 }
2362 }
2363
2364 function scrollToRowId(id) {
2365 try {
2366 var row = $(id);
2367
2368 if (row)
2369 $("headlines-frame").scrollTop = row.offsetTop - 4;
2370
2371 } catch (e) {
2372 exception_error("scrollToRowId", e);
2373 }
2374 }
2375
2376 function updateFloatingTitle(unread_only) {
2377 try {
2378 if (!isCdmMode()) return;
2379
2380 var hf = $("headlines-frame");
2381
2382 var elems = $$("#headlines-frame > div[id*=RROW]");
2383
2384 for (var i = 0; i < elems.length; i++) {
2385
2386 var child = elems[i];
2387
2388 if (child && child.offsetTop + child.offsetHeight > hf.scrollTop) {
2389
2390 var header = child.getElementsByClassName("cdmHeader")[0];
2391
2392 if (unread_only || child.id != $("floatingTitle").getAttribute("rowid")) {
2393 if (child.id != $("floatingTitle").getAttribute("rowid")) {
2394 $("floatingTitle").setAttribute("rowid", child.id);
2395 $("floatingTitle").innerHTML = header.innerHTML;
2396 $("floatingTitle").firstChild.innerHTML = "<img class='anchor markedPic' src='images/page_white_go.png' onclick=\"scrollToRowId('"+child.id+"')\">" + $("floatingTitle").firstChild.innerHTML;
2397
2398 initFloatingMenu();
2399
2400 var cb = $$("#floatingTitle .dijitCheckBox")[0];
2401
2402 if (cb)
2403 cb.parentNode.removeChild(cb);
2404 }
2405
2406 if (child.hasClassName("Unread"))
2407 $("floatingTitle").addClassName("Unread");
2408 else
2409 $("floatingTitle").removeClassName("Unread");
2410
2411 PluginHost.run(PluginHost.HOOK_FLOATING_TITLE, child);
2412 }
2413
2414 $("floatingTitle").style.marginRight = hf.offsetWidth - child.offsetWidth + "px";
2415 if (header.offsetTop + header.offsetHeight < hf.scrollTop + $("floatingTitle").offsetHeight - 5 &&
2416 child.offsetTop + child.offsetHeight >= hf.scrollTop + $("floatingTitle").offsetHeight - 5)
2417 $("floatingTitle").style.visibility = "visible";
2418 else
2419 $("floatingTitle").style.visibility = "hidden";
2420
2421 return;
2422
2423 }
2424 }
2425
2426 } catch (e) {
2427 exception_error("updateFloatingTitle", e);
2428 }
2429 }
2430
2431 function cdmFooterClick(event) {
2432 event.stopPropagation();
2433 }