]> git.wh0rd.org - tt-rss.git/blob - js/tt-rss.js
experimental CSRF protection
[tt-rss.git] / js / tt-rss.js
1 var total_unread = 0;
2 var global_unread = -1;
3 var firsttime_update = true;
4 var _active_feed_id = 0;
5 var _active_feed_is_cat = false;
6 var hotkey_prefix = false;
7 var hotkey_prefix_pressed = false;
8 var _force_scheduled_update = false;
9 var last_scheduled_update = false;
10
11 var _rpc_seq = 0;
12
13 function next_seq() {
14 _rpc_seq += 1;
15 return _rpc_seq;
16 }
17
18 function get_seq() {
19 return _rpc_seq;
20 }
21
22 function activeFeedIsCat() {
23 return _active_feed_is_cat;
24 }
25
26 function getActiveFeedId() {
27 try {
28 //console.log("gAFID: " + _active_feed_id);
29 return _active_feed_id;
30 } catch (e) {
31 exception_error("getActiveFeedId", e);
32 }
33 }
34
35 function setActiveFeedId(id, is_cat) {
36 try {
37 _active_feed_id = id;
38
39 if (is_cat != undefined) {
40 _active_feed_is_cat = is_cat;
41 }
42
43 selectFeed(id, is_cat);
44
45 } catch (e) {
46 exception_error("setActiveFeedId", e);
47 }
48 }
49
50
51 function updateFeedList() {
52 try {
53
54 // $("feeds-holder").innerHTML = "<div id=\"feedlistLoading\">" +
55 // __("Loading, please wait...") + "</div>";
56
57 Element.show("feedlistLoading");
58
59 if (dijit.byId("feedTree")) {
60 dijit.byId("feedTree").destroyRecursive();
61 }
62
63 var store = new dojo.data.ItemFileWriteStore({
64 url: "backend.php?op=feeds"});
65
66 var treeModel = new fox.FeedStoreModel({
67 store: store,
68 query: {
69 "type": "feed"
70 },
71 rootId: "root",
72 rootLabel: "Feeds",
73 childrenAttrs: ["items"]
74 });
75
76 var tree = new fox.FeedTree({
77 persist: false,
78 model: treeModel,
79 onOpen: function (item, node) {
80 var id = String(item.id);
81 var cat_id = id.substr(id.indexOf(":")+1);
82
83 new Ajax.Request("backend.php",
84 { parameters: "backend.php?op=feeds&method=collapse&cid=" +
85 param_escape(cat_id) + "&mode=0" } );
86 },
87 onClose: function (item, node) {
88 var id = String(item.id);
89 var cat_id = id.substr(id.indexOf(":")+1);
90
91 new Ajax.Request("backend.php",
92 { parameters: "backend.php?op=feeds&method=collapse&cid=" +
93 param_escape(cat_id) + "&mode=1" } );
94
95 },
96 onClick: function (item, node) {
97 var id = String(item.id);
98 var is_cat = id.match("^CAT:");
99 var feed = id.substr(id.indexOf(":")+1);
100 viewfeed(feed, '', is_cat);
101 return false;
102 },
103 openOnClick: false,
104 showRoot: false,
105 id: "feedTree",
106 }, "feedTree");
107
108 /* var menu = new dijit.Menu({id: 'feedMenu'});
109
110 menu.addChild(new dijit.MenuItem({
111 label: "Simple menu item"
112 }));
113
114 // menu.bindDomNode(tree.domNode); */
115
116 var tmph = dojo.connect(dijit.byId('feedMenu'), '_openMyself', function (event) {
117 console.log(dijit.getEnclosingWidget(event.target));
118 dojo.disconnect(tmph);
119 });
120
121 $("feeds-holder").appendChild(tree.domNode);
122
123 var tmph = dojo.connect(tree, 'onLoad', function() {
124 dojo.disconnect(tmph);
125 Element.hide("feedlistLoading");
126
127 tree.collapseHiddenCats();
128
129 feedlist_init();
130
131 // var node = dijit.byId("feedTree")._itemNodesMap['FEED:-2'][0].domNode
132 // menu.bindDomNode(node);
133
134 loading_set_progress(25);
135 });
136
137 tree.startup();
138
139 } catch (e) {
140 exception_error("updateFeedList", e);
141 }
142 }
143
144 function catchupAllFeeds() {
145
146 var str = __("Mark all articles as read?");
147
148 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
149
150 var query_str = "backend.php?op=feeds&method=catchupAll";
151
152 notify_progress("Marking all feeds as read...");
153
154 //console.log("catchupAllFeeds Q=" + query_str);
155
156 new Ajax.Request("backend.php", {
157 parameters: query_str,
158 onComplete: function(transport) {
159 feedlist_callback2(transport);
160 } });
161
162 global_unread = 0;
163 updateTitle("");
164 }
165 }
166
167 function viewCurrentFeed(method) {
168
169 if (getActiveFeedId() != undefined) {
170 viewfeed(getActiveFeedId(), method, activeFeedIsCat());
171 }
172 return false; // block unneeded form submits
173 }
174
175 function timeout() {
176 if (getInitParam("bw_limit") == "1") return;
177
178 try {
179 var date = new Date();
180 var ts = Math.round(date.getTime() / 1000);
181
182 if (ts - last_scheduled_update > 10 || _force_scheduled_update) {
183
184 //console.log("timeout()");
185
186 window.clearTimeout(counter_timeout_id);
187
188 var query_str = "?op=rpc&method=getAllCounters&seq=" + next_seq();
189
190 var omode;
191
192 if (firsttime_update && !navigator.userAgent.match("Opera")) {
193 firsttime_update = false;
194 omode = "T";
195 } else {
196 omode = "flc";
197 }
198
199 query_str = query_str + "&omode=" + omode;
200
201 if (!_force_scheduled_update)
202 query_str = query_str + "&last_article_id=" + getInitParam("last_article_id");
203
204 //console.log("[timeout]" + query_str);
205
206 new Ajax.Request("backend.php", {
207 parameters: query_str,
208 onComplete: function(transport) {
209 handle_rpc_json(transport, !_force_scheduled_update);
210 _force_scheduled_update = false;
211 } });
212
213 last_scheduled_update = ts;
214 }
215
216 } catch (e) {
217 exception_error("timeout", e);
218 }
219
220 setTimeout("timeout()", 3000);
221 }
222
223 function search() {
224 var query = "backend.php?op=dlg&method=search&param=" +
225 param_escape(getActiveFeedId() + ":" + activeFeedIsCat());
226
227 if (dijit.byId("searchDlg"))
228 dijit.byId("searchDlg").destroyRecursive();
229
230 dialog = new dijit.Dialog({
231 id: "searchDlg",
232 title: __("Search"),
233 style: "width: 600px",
234 execute: function() {
235 if (this.validate()) {
236 _search_query = dojo.objectToQuery(this.attr('value'));
237 this.hide();
238 viewCurrentFeed();
239 }
240 },
241 href: query});
242
243 dialog.show();
244 }
245
246 function updateTitle() {
247 var tmp = "Tiny Tiny RSS";
248
249 if (global_unread > 0) {
250 tmp = tmp + " (" + global_unread + ")";
251 }
252
253 if (window.fluid) {
254 if (global_unread > 0) {
255 window.fluid.dockBadge = global_unread;
256 } else {
257 window.fluid.dockBadge = "";
258 }
259 }
260
261 document.title = tmp;
262 }
263
264 function genericSanityCheck() {
265 setCookie("ttrss_test", "TEST");
266
267 if (getCookie("ttrss_test") != "TEST") {
268 return fatalError(2);
269 }
270
271 return true;
272 }
273
274 function init() {
275 try {
276 //dojo.registerModulePath("fox", "../../js/");
277
278 dojo.require("fox.FeedTree");
279
280 if (typeof themeBeforeLayout == 'function') {
281 themeBeforeLayout();
282 }
283
284 dojo.parser.parse();
285
286 dojo.addOnLoad(function() {
287 updateFeedList();
288 closeArticlePanel();
289
290 if (typeof themeAfterLayout == 'function') {
291 themeAfterLayout();
292 }
293
294 });
295
296 if (!genericSanityCheck())
297 return false;
298
299 loading_set_progress(20);
300
301 var hasAudio = !!((myAudioTag = document.createElement('audio')).canPlayType);
302
303 new Ajax.Request("backend.php", {
304 parameters: {op: "rpc", method: "sanityCheck", hasAudio: hasAudio},
305 onComplete: function(transport) {
306 backend_sanity_check_callback(transport);
307 } });
308
309 } catch (e) {
310 exception_error("init", e);
311 }
312 }
313
314 function init_second_stage() {
315
316 try {
317
318 delCookie("ttrss_test");
319
320 var toolbar = document.forms["main_toolbar_form"];
321
322 dijit.getEnclosingWidget(toolbar.view_mode).attr('value',
323 getInitParam("default_view_mode"));
324
325 dijit.getEnclosingWidget(toolbar.order_by).attr('value',
326 getInitParam("default_view_order_by"));
327
328 feeds_sort_by_unread = getInitParam("feeds_sort_by_unread") == 1;
329
330 loading_set_progress(30);
331
332 // can't use cache_clear() here because viewfeed might not have initialized yet
333 if ('sessionStorage' in window && window['sessionStorage'] !== null)
334 sessionStorage.clear();
335
336 console.log("second stage ok");
337
338 } catch (e) {
339 exception_error("init_second_stage", e);
340 }
341 }
342
343 function quickMenuGo(opid) {
344 try {
345 if (opid == "qmcPrefs") {
346 gotoPreferences();
347 }
348
349 if (opid == "qmcTagCloud") {
350 displayDlg("printTagCloud");
351 }
352
353 if (opid == "qmcTagSelect") {
354 displayDlg("printTagSelect");
355 }
356
357 if (opid == "qmcSearch") {
358 search();
359 return;
360 }
361
362 if (opid == "qmcAddFeed") {
363 quickAddFeed();
364 return;
365 }
366
367 if (opid == "qmcDigest") {
368 window.location.href = "digest.php";
369 return;
370 }
371
372 if (opid == "qmcEditFeed") {
373 if (activeFeedIsCat())
374 alert(__("You can't edit this kind of feed."));
375 else
376 editFeed(getActiveFeedId());
377 return;
378 }
379
380 if (opid == "qmcRemoveFeed") {
381 var actid = getActiveFeedId();
382
383 if (activeFeedIsCat()) {
384 alert(__("You can't unsubscribe from the category."));
385 return;
386 }
387
388 if (!actid) {
389 alert(__("Please select some feed first."));
390 return;
391 }
392
393 var fn = getFeedName(actid);
394
395 var pr = __("Unsubscribe from %s?").replace("%s", fn);
396
397 if (confirm(pr)) {
398 unsubscribeFeed(actid);
399 }
400
401 return;
402 }
403
404 if (opid == "qmcCatchupAll") {
405 catchupAllFeeds();
406 return;
407 }
408
409 if (opid == "qmcShowOnlyUnread") {
410 toggleDispRead();
411 return;
412 }
413
414 if (opid == "qmcAddFilter") {
415 quickAddFilter();
416 return;
417 }
418
419 if (opid == "qmcAddLabel") {
420 addLabel();
421 return;
422 }
423
424 if (opid == "qmcRescoreFeed") {
425 rescoreCurrentFeed();
426 return;
427 }
428
429 if (opid == "qmcHKhelp") {
430 new Ajax.Request("backend.php", {
431 parameters: "?op=backend&method=help&topic=main",
432 onComplete: function(transport) {
433 $("hotkey_help_overlay").innerHTML = transport.responseText;
434 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
435 } });
436 }
437
438 } catch (e) {
439 exception_error("quickMenuGo", e);
440 }
441 }
442
443 function toggleDispRead() {
444 try {
445
446 var hide = !(getInitParam("hide_read_feeds") == "1");
447
448 hideOrShowFeeds(hide);
449
450 var query = "?op=rpc&method=setpref&key=HIDE_READ_FEEDS&value=" +
451 param_escape(hide);
452
453 setInitParam("hide_read_feeds", hide);
454
455 new Ajax.Request("backend.php", {
456 parameters: query,
457 onComplete: function(transport) {
458 } });
459
460 } catch (e) {
461 exception_error("toggleDispRead", e);
462 }
463 }
464
465 function parse_runtime_info(data) {
466
467 //console.log("parsing runtime info...");
468
469 for (k in data) {
470 var v = data[k];
471
472 // console.log("RI: " + k + " => " + v);
473
474 if (k == "new_version_available") {
475 var icon = $("newVersionIcon");
476 if (icon) {
477 if (v == "1") {
478 icon.style.display = "inline";
479 } else {
480 icon.style.display = "none";
481 }
482 }
483 return;
484 }
485
486 if (k == "daemon_is_running" && v != 1) {
487 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
488 return;
489 }
490
491 if (k == "daemon_stamp_ok" && v != 1) {
492 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
493 return;
494 }
495
496 if (k == "max_feed_id" || k == "num_feeds") {
497 if (init_params[k] != v) {
498 console.log("feed count changed, need to reload feedlist.");
499 updateFeedList();
500 }
501 }
502
503 init_params[k] = v;
504 notify('');
505 }
506 }
507
508 function catchupCurrentFeed() {
509
510 var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
511
512 var str = __("Mark all articles in %s as read?").replace("%s", fn);
513
514 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
515 return viewCurrentFeed('MarkAllRead');
516 }
517 }
518
519 function catchupFeedInGroup(id) {
520
521 try {
522
523 var title = getFeedName(id);
524
525 var str = __("Mark all articles in %s as read?").replace("%s", title);
526
527 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
528 return viewCurrentFeed('MarkAllReadGR:' + id);
529 }
530
531 } catch (e) {
532 exception_error("catchupFeedInGroup", e);
533 }
534 }
535
536 function collapse_feedlist() {
537 try {
538
539 if (!Element.visible('feeds-holder')) {
540 Element.show('feeds-holder');
541 Element.show('feeds-holder_splitter');
542 $("collapse_feeds_btn").innerHTML = "&lt;&lt;";
543 } else {
544 Element.hide('feeds-holder');
545 Element.hide('feeds-holder_splitter');
546 $("collapse_feeds_btn").innerHTML = "&gt;&gt;";
547 }
548
549 dijit.byId("main").resize();
550
551 query = "?op=rpc&method=setpref&key=_COLLAPSED_FEEDLIST&value=true";
552 new Ajax.Request("backend.php", { parameters: query });
553
554 } catch (e) {
555 exception_error("collapse_feedlist", e);
556 }
557 }
558
559 function viewModeChanged() {
560 return viewCurrentFeed('');
561 }
562
563 function viewLimitChanged() {
564 return viewCurrentFeed('');
565 }
566
567 /* function adjustArticleScore(id, score) {
568 try {
569
570 var pr = prompt(__("Assign score to article:"), score);
571
572 if (pr != undefined) {
573 var query = "?op=rpc&method=setScore&id=" + id + "&score=" + pr;
574
575 new Ajax.Request("backend.php", {
576 parameters: query,
577 onComplete: function(transport) {
578 viewCurrentFeed();
579 } });
580
581 }
582 } catch (e) {
583 exception_error("adjustArticleScore", e);
584 }
585 } */
586
587 function rescoreCurrentFeed() {
588
589 var actid = getActiveFeedId();
590
591 if (activeFeedIsCat() || actid < 0) {
592 alert(__("You can't rescore this kind of feed."));
593 return;
594 }
595
596 if (!actid) {
597 alert(__("Please select some feed first."));
598 return;
599 }
600
601 var fn = getFeedName(actid);
602 var pr = __("Rescore articles in %s?").replace("%s", fn);
603
604 if (confirm(pr)) {
605 notify_progress("Rescoring articles...");
606
607 var query = "?op=pref-feeds&method=rescore&quiet=1&ids=" + actid;
608
609 new Ajax.Request("backend.php", {
610 parameters: query,
611 onComplete: function(transport) {
612 viewCurrentFeed();
613 } });
614 }
615 }
616
617 function hotkey_handler(e) {
618 try {
619
620 if (e.target.nodeName == "INPUT" || e.target.nodeName == "TEXTAREA") return;
621
622 var keycode = false;
623 var shift_key = false;
624
625 var cmdline = $('cmdline');
626
627 try {
628 shift_key = e.shiftKey;
629 } catch (e) {
630
631 }
632
633 if (window.event) {
634 keycode = window.event.keyCode;
635 } else if (e) {
636 keycode = e.which;
637 }
638
639 var keychar = String.fromCharCode(keycode);
640
641 if (keycode == 27) { // escape
642 if (Element.visible("hotkey_help_overlay")) {
643 Element.hide("hotkey_help_overlay");
644 }
645 hotkey_prefix = false;
646 }
647
648 if (keycode == 16) return; // ignore lone shift
649 if (keycode == 17) return; // ignore lone ctrl
650
651 if ((keycode == 70 || keycode == 67 || keycode == 71 || keycode == 65)
652 && !hotkey_prefix) {
653
654 var date = new Date();
655 var ts = Math.round(date.getTime() / 1000);
656
657 hotkey_prefix = keycode;
658 hotkey_prefix_pressed = ts;
659
660 cmdline.innerHTML = keychar;
661 Element.show(cmdline);
662
663 console.log("KP: PREFIX=" + keycode + " CHAR=" + keychar + " TS=" + ts);
664 return true;
665 }
666
667 if (Element.visible("hotkey_help_overlay")) {
668 Element.hide("hotkey_help_overlay");
669 }
670
671 /* Global hotkeys */
672
673 Element.hide(cmdline);
674
675 if (!hotkey_prefix) {
676
677 if (keycode == 27) { // escape
678 closeArticlePanel();
679 return;
680 }
681
682 if (keycode == 69) { // e
683 var id = getActiveArticleId();
684 emailArticle(id);
685 }
686
687 if ((keycode == 191 || keychar == '?') && shift_key) { // ?
688 if (!Element.visible("hotkey_help_overlay")) {
689 Effect.Appear("hotkey_help_overlay", {duration : 0.3, to : 0.9});
690 } else {
691 Element.hide("hotkey_help_overlay");
692 }
693 return false;
694 }
695
696 if (keycode == 191 || keychar == '/') { // /
697 search();
698 return false;
699 }
700
701 if (keycode == 74 && !shift_key) { // j
702 var rv = dijit.byId("feedTree").getPreviousFeed(
703 getActiveFeedId(), activeFeedIsCat());
704
705 if (rv) viewfeed(rv[0], '', rv[1]);
706
707 return;
708 }
709
710 if (keycode == 75) { // k
711 var rv = dijit.byId("feedTree").getNextFeed(
712 getActiveFeedId(), activeFeedIsCat());
713
714 if (rv) viewfeed(rv[0], '', rv[1]);
715
716 return;
717 }
718
719 if (shift_key && keycode == 40) { // shift-down
720 catchupRelativeToArticle(1);
721 return;
722 }
723
724 if (shift_key && keycode == 38) { // shift-up
725 catchupRelativeToArticle(0);
726 return;
727 }
728
729 if (shift_key && keycode == 78) { // N
730 scrollArticle(50);
731 return;
732 }
733
734 if (shift_key && keycode == 80) { // P
735 scrollArticle(-50);
736 return;
737 }
738
739 if (keycode == 68 && shift_key) { // shift-D
740 dismissSelectedArticles();
741 return;
742 }
743
744 if (keycode == 88 && shift_key) { // shift-X
745 dismissReadArticles();
746 return;
747 }
748
749 if (keycode == 78 || keycode == 40) { // n, down
750 if (typeof moveToPost != 'undefined') {
751 moveToPost('next');
752 return false;
753 }
754 }
755
756 if (keycode == 80 || keycode == 38) { // p, up
757 if (typeof moveToPost != 'undefined') {
758 moveToPost('prev');
759 return false;
760 }
761 }
762
763 if (keycode == 83 && shift_key) { // S
764 selectionTogglePublished(undefined, false, true);
765 return;
766 }
767
768 if (keycode == 83) { // s
769 selectionToggleMarked(undefined, false, true);
770 return;
771 }
772
773 if (keycode == 85) { // u
774 selectionToggleUnread(undefined, false, true);
775 return;
776 }
777
778 if (keycode == 84 && shift_key) { // T
779 var id = getActiveArticleId();
780 if (id) {
781 editArticleTags(id, getActiveFeedId(), isCdmMode());
782 return;
783 }
784 }
785
786 if (keycode == 9) { // tab
787 var id = getArticleUnderPointer();
788 if (id) {
789 var cb = $("RCHK-" + id);
790
791 if (cb) {
792 cb.checked = !cb.checked;
793 toggleSelectRowById(cb, "RROW-" + id);
794 return false;
795 }
796 }
797 }
798
799 if (keycode == 79) { // o
800 if (getActiveArticleId()) {
801 openArticleInNewWindow(getActiveArticleId());
802 return;
803 }
804 }
805
806 if (keycode == 81 && shift_key) { // Q
807 if (typeof catchupAllFeeds != 'undefined') {
808 catchupAllFeeds();
809 return;
810 }
811 }
812
813 if (keycode == 88 && !shift_key) { // x
814 if (activeFeedIsCat()) {
815 dijit.byId("feedTree").collapseCat(getActiveFeedId());
816 return;
817 }
818 }
819 }
820
821 /* Prefix a */
822
823 if (hotkey_prefix == 65) { // a
824 hotkey_prefix = false;
825
826 if (keycode == 65) { // a
827 selectArticles('all');
828 return;
829 }
830
831 if (keycode == 85) { // u
832 selectArticles('unread');
833 return;
834 }
835
836 if (keycode == 73) { // i
837 selectArticles('invert');
838 return;
839 }
840
841 if (keycode == 78) { // n
842 selectArticles('none');
843 return;
844 }
845
846 }
847
848 /* Prefix f */
849
850 if (hotkey_prefix == 70) { // f
851
852 hotkey_prefix = false;
853
854 if (keycode == 81) { // q
855 if (getActiveFeedId()) {
856 catchupCurrentFeed();
857 return;
858 }
859 }
860
861 if (keycode == 82) { // r
862 if (getActiveFeedId()) {
863 viewfeed(getActiveFeedId(), '', activeFeedIsCat());
864 return;
865 }
866 }
867
868 if (keycode == 65) { // a
869 toggleDispRead();
870 return false;
871 }
872
873 if (keycode == 85) { // u
874 if (getActiveFeedId()) {
875 viewfeed(getActiveFeedId(), '');
876 return false;
877 }
878 }
879
880 if (keycode == 69) { // e
881
882 if (activeFeedIsCat())
883 alert(__("You can't edit this kind of feed."));
884 else
885 editFeed(getActiveFeedId());
886 return;
887
888 return false;
889 }
890
891 if (keycode == 83) { // s
892 quickAddFeed();
893 return false;
894 }
895
896 if (keycode == 67 && shift_key) { // C
897 if (typeof catchupAllFeeds != 'undefined') {
898 catchupAllFeeds();
899 return false;
900 }
901 }
902
903 if (keycode == 67) { // c
904 if (getActiveFeedId()) {
905 catchupCurrentFeed();
906 return false;
907 }
908 }
909
910 if (keycode == 88) { // x
911 reverseHeadlineOrder();
912 return;
913 }
914 }
915
916 /* Prefix c */
917
918 if (hotkey_prefix == 67) { // c
919 hotkey_prefix = false;
920
921 if (keycode == 70) { // f
922 quickAddFilter();
923 return false;
924 }
925
926 if (keycode == 76) { // l
927 addLabel();
928 return false;
929 }
930
931 if (keycode == 83) { // s
932 if (typeof collapse_feedlist != 'undefined') {
933 collapse_feedlist();
934 return false;
935 }
936 }
937
938 if (keycode == 77) { // m
939 // TODO: sortable feedlist
940 return;
941 }
942
943 if (keycode == 78) { // n
944 catchupRelativeToArticle(1);
945 return;
946 }
947
948 if (keycode == 80) { // p
949 catchupRelativeToArticle(0);
950 return;
951 }
952
953
954 }
955
956 /* Prefix g */
957
958 if (hotkey_prefix == 71) { // g
959
960 hotkey_prefix = false;
961
962
963 if (keycode == 65) { // a
964 viewfeed(-4);
965 return false;
966 }
967
968 if (keycode == 83) { // s
969 viewfeed(-1);
970 return false;
971 }
972
973 if (keycode == 80 && shift_key) { // P
974 gotoPreferences();
975 return false;
976 }
977
978 if (keycode == 80) { // p
979 viewfeed(-2);
980 return false;
981 }
982
983 if (keycode == 70) { // f
984 viewfeed(-3);
985 return false;
986 }
987
988 if (keycode == 84) { // t
989 displayDlg("printTagCloud");
990 return false;
991 }
992 }
993
994 /* Cmd */
995
996 if (hotkey_prefix == 224 || hotkey_prefix == 91) { // f
997 hotkey_prefix = false;
998 return;
999 }
1000
1001 if (hotkey_prefix) {
1002 console.log("KP: PREFIX=" + hotkey_prefix + " CODE=" + keycode + " CHAR=" + keychar);
1003 } else {
1004 console.log("KP: CODE=" + keycode + " CHAR=" + keychar);
1005 }
1006
1007
1008 } catch (e) {
1009 exception_error("hotkey_handler", e);
1010 }
1011 }
1012
1013 function inPreferences() {
1014 return false;
1015 }
1016
1017 function reverseHeadlineOrder() {
1018 try {
1019
1020 var query_str = "?op=rpc&method=togglepref&key=REVERSE_HEADLINES";
1021
1022 new Ajax.Request("backend.php", {
1023 parameters: query_str,
1024 onComplete: function(transport) {
1025 viewCurrentFeed();
1026 } });
1027
1028 } catch (e) {
1029 exception_error("reverseHeadlineOrder", e);
1030 }
1031 }
1032
1033 function scheduleFeedUpdate(id, is_cat) {
1034 try {
1035 if (!id) {
1036 id = getActiveFeedId();
1037 is_cat = activeFeedIsCat();
1038 }
1039
1040 if (!id) {
1041 alert(__("Please select some feed first."));
1042 return;
1043 }
1044
1045 var query = "?op=rpc&method=scheduleFeedUpdate&id=" +
1046 param_escape(id) +
1047 "&is_cat=" + param_escape(is_cat);
1048
1049 console.log(query);
1050
1051 new Ajax.Request("backend.php", {
1052 parameters: query,
1053 onComplete: function(transport) {
1054 handle_rpc_json(transport);
1055
1056 var reply = JSON.parse(transport.responseText);
1057 var message = reply['message'];
1058
1059 if (message) {
1060 notify_info(message);
1061 return;
1062 }
1063
1064 } });
1065
1066
1067 } catch (e) {
1068 exception_error("scheduleFeedUpdate", e);
1069 }
1070 }
1071
1072 function newVersionDlg() {
1073 try {
1074 var query = "backend.php?op=dlg&method=newVersion";
1075
1076 if (dijit.byId("newVersionDlg"))
1077 dijit.byId("newVersionDlg").destroyRecursive();
1078
1079 dialog = new dijit.Dialog({
1080 id: "newVersionDlg",
1081 title: __("New version available!"),
1082 style: "width: 600px",
1083 href: query,
1084 });
1085
1086 dialog.show();
1087
1088 } catch (e) {
1089 exception_error("newVersionDlg", e);
1090 }
1091 }
1092
1093 function handle_rpc_json(transport, scheduled_call) {
1094 try {
1095 var reply = JSON.parse(transport.responseText);
1096
1097 if (reply) {
1098
1099 var error = reply['error'];
1100
1101 if (error) {
1102 var code = error['code'];
1103 var msg = error['msg'];
1104
1105 console.warn("[handle_rpc_json] received fatal error " + code + "/" + msg);
1106
1107 if (code != 0) {
1108 fatalError(code, msg);
1109 return false;
1110 }
1111 }
1112
1113 var seq = reply['seq'];
1114
1115 if (seq) {
1116 if (get_seq() != seq) {
1117 console.log("[handle_rpc_json] sequence mismatch: " + seq +
1118 " (want: " + get_seq() + ")");
1119 return true;
1120 }
1121 }
1122
1123 var message = reply['message'];
1124
1125 if (message) {
1126 if (message == "UPDATE_COUNTERS") {
1127 console.log("need to refresh counters...");
1128 setInitParam("last_article_id", -1);
1129 _force_scheduled_update = true;
1130 }
1131 }
1132
1133 var counters = reply['counters'];
1134
1135 if (counters)
1136 parse_counters(counters, scheduled_call);
1137
1138 var runtime_info = reply['runtime-info'];;
1139
1140 if (runtime_info)
1141 parse_runtime_info(runtime_info);
1142
1143 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1144
1145 } else {
1146 notify_error("Error communicating with server.");
1147 }
1148
1149 } catch (e) {
1150 notify_error("Error communicating with server.");
1151 console.log(e);
1152 //exception_error("handle_rpc_json", e, transport);
1153 }
1154
1155 return true;
1156 }
1157