]> git.wh0rd.org Git - tt-rss.git/blob - js/tt-rss.js
Merge pull request #19 from plaidfluff/7e454e815dd67f0ac3804e213599be9bbbc2755e
[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                 if (!genericSanityCheck())
287                         return false;
288
289                 loading_set_progress(20);
290
291                 var hasAudio = !!((myAudioTag = document.createElement('audio')).canPlayType);
292
293                 new Ajax.Request("backend.php", {
294                         parameters: {op: "rpc", method: "sanityCheck", hasAudio: hasAudio},
295                         onComplete: function(transport) {
296                                         backend_sanity_check_callback(transport);
297                                 } });
298
299         } catch (e) {
300                 exception_error("init", e);
301         }
302 }
303
304 function init_second_stage() {
305
306         try {
307                 dojo.addOnLoad(function() {
308                         updateFeedList();
309                         closeArticlePanel();
310
311                         if (typeof themeAfterLayout == 'function') {
312                                 themeAfterLayout();
313                         }
314
315                 });
316
317                 delCookie("ttrss_test");
318
319                 var toolbar = document.forms["main_toolbar_form"];
320
321                 dijit.getEnclosingWidget(toolbar.view_mode).attr('value',
322                         getInitParam("default_view_mode"));
323
324                 dijit.getEnclosingWidget(toolbar.order_by).attr('value',
325                         getInitParam("default_view_order_by"));
326
327                 feeds_sort_by_unread = getInitParam("feeds_sort_by_unread") == 1;
328
329                 loading_set_progress(30);
330
331                 // can't use cache_clear() here because viewfeed might not have initialized yet
332                 if ('sessionStorage' in window && window['sessionStorage'] !== null)
333                         sessionStorage.clear();
334
335                 console.log("second stage ok");
336
337         } catch (e) {
338                 exception_error("init_second_stage", e);
339         }
340 }
341
342 function quickMenuGo(opid) {
343         try {
344                 if (opid == "qmcPrefs") {
345                         gotoPreferences();
346                 }
347
348                 if (opid == "qmcTagCloud") {
349                         displayDlg("printTagCloud");
350                 }
351
352                 if (opid == "qmcTagSelect") {
353                         displayDlg("printTagSelect");
354                 }
355
356                 if (opid == "qmcSearch") {
357                         search();
358                         return;
359                 }
360
361                 if (opid == "qmcAddFeed") {
362                         quickAddFeed();
363                         return;
364                 }
365
366                 if (opid == "qmcDigest") {
367                         window.location.href = "digest.php";
368                         return;
369                 }
370
371                 if (opid == "qmcEditFeed") {
372                         if (activeFeedIsCat())
373                                 alert(__("You can't edit this kind of feed."));
374                         else
375                                 editFeed(getActiveFeedId());
376                         return;
377                 }
378
379                 if (opid == "qmcRemoveFeed") {
380                         var actid = getActiveFeedId();
381
382                         if (activeFeedIsCat()) {
383                                 alert(__("You can't unsubscribe from the category."));
384                                 return;
385                         }
386
387                         if (!actid) {
388                                 alert(__("Please select some feed first."));
389                                 return;
390                         }
391
392                         var fn = getFeedName(actid);
393
394                         var pr = __("Unsubscribe from %s?").replace("%s", fn);
395
396                         if (confirm(pr)) {
397                                 unsubscribeFeed(actid);
398                         }
399
400                         return;
401                 }
402
403                 if (opid == "qmcCatchupAll") {
404                         catchupAllFeeds();
405                         return;
406                 }
407
408                 if (opid == "qmcShowOnlyUnread") {
409                         toggleDispRead();
410                         return;
411                 }
412
413                 if (opid == "qmcAddFilter") {
414                         quickAddFilter();
415                         return;
416                 }
417
418                 if (opid == "qmcAddLabel") {
419                         addLabel();
420                         return;
421                 }
422
423                 if (opid == "qmcRescoreFeed") {
424                         rescoreCurrentFeed();
425                         return;
426                 }
427
428                 if (opid == "qmcHKhelp") {
429                         new Ajax.Request("backend.php", {
430                                 parameters: "?op=backend&method=help&topic=main",
431                                 onComplete: function(transport) {
432                                         $("hotkey_help_overlay").innerHTML = transport.responseText;
433                                         Effect.Appear("hotkey_help_overlay", {duration : 0.3});
434                                 } });
435                 }
436
437         } catch (e) {
438                 exception_error("quickMenuGo", e);
439         }
440 }
441
442 function toggleDispRead() {
443         try {
444
445                 var hide = !(getInitParam("hide_read_feeds") == "1");
446
447                 hideOrShowFeeds(hide);
448
449                 var query = "?op=rpc&method=setpref&key=HIDE_READ_FEEDS&value=" +
450                         param_escape(hide);
451
452                 setInitParam("hide_read_feeds", hide);
453
454                 new Ajax.Request("backend.php", {
455                         parameters: query,
456                         onComplete: function(transport) {
457                         } });
458
459         } catch (e) {
460                 exception_error("toggleDispRead", e);
461         }
462 }
463
464 function parse_runtime_info(data) {
465
466         //console.log("parsing runtime info...");
467
468         for (k in data) {
469                 var v = data[k];
470
471 //              console.log("RI: " + k + " => " + v);
472
473                 if (k == "new_version_available") {
474                         var icon = $("newVersionIcon");
475                         if (icon) {
476                                 if (v == "1") {
477                                         icon.style.display = "inline";
478                                 } else {
479                                         icon.style.display = "none";
480                                 }
481                         }
482                         return;
483                 }
484
485                 if (k == "daemon_is_running" && v != 1) {
486                         notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
487                         return;
488                 }
489
490                 if (k == "daemon_stamp_ok" && v != 1) {
491                         notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
492                         return;
493                 }
494
495                 if (k == "max_feed_id" || k == "num_feeds") {
496                         if (init_params[k] != v) {
497                                 console.log("feed count changed, need to reload feedlist.");
498                                 updateFeedList();
499                         }
500                 }
501
502                 init_params[k] = v;
503                 notify('');
504         }
505 }
506
507 function collapse_feedlist() {
508         try {
509
510                 if (!Element.visible('feeds-holder')) {
511                         Element.show('feeds-holder');
512                         Element.show('feeds-holder_splitter');
513                         $("collapse_feeds_btn").innerHTML = "&lt;&lt;";
514                 } else {
515                         Element.hide('feeds-holder');
516                         Element.hide('feeds-holder_splitter');
517                         $("collapse_feeds_btn").innerHTML = "&gt;&gt;";
518                 }
519
520                 dijit.byId("main").resize();
521
522                 query = "?op=rpc&method=setpref&key=_COLLAPSED_FEEDLIST&value=true";
523                 new Ajax.Request("backend.php", { parameters: query });
524
525         } catch (e) {
526                 exception_error("collapse_feedlist", e);
527         }
528 }
529
530 function viewModeChanged() {
531         return viewCurrentFeed('');
532 }
533
534 function viewLimitChanged() {
535         return viewCurrentFeed('');
536 }
537
538 /* function adjustArticleScore(id, score) {
539         try {
540
541                 var pr = prompt(__("Assign score to article:"), score);
542
543                 if (pr != undefined) {
544                         var query = "?op=rpc&method=setScore&id=" + id + "&score=" + pr;
545
546                         new Ajax.Request("backend.php", {
547                         parameters: query,
548                         onComplete: function(transport) {
549                                         viewCurrentFeed();
550                                 } });
551
552                 }
553         } catch (e) {
554                 exception_error("adjustArticleScore", e);
555         }
556 } */
557
558 function rescoreCurrentFeed() {
559
560         var actid = getActiveFeedId();
561
562         if (activeFeedIsCat() || actid < 0) {
563                 alert(__("You can't rescore this kind of feed."));
564                 return;
565         }
566
567         if (!actid) {
568                 alert(__("Please select some feed first."));
569                 return;
570         }
571
572         var fn = getFeedName(actid);
573         var pr = __("Rescore articles in %s?").replace("%s", fn);
574
575         if (confirm(pr)) {
576                 notify_progress("Rescoring articles...");
577
578                 var query = "?op=pref-feeds&method=rescore&quiet=1&ids=" + actid;
579
580                 new Ajax.Request("backend.php", {
581                         parameters: query,
582                         onComplete: function(transport) {
583                                 viewCurrentFeed();
584                         } });
585         }
586 }
587
588 function hotkey_handler(e) {
589         try {
590
591                 if (e.target.nodeName == "INPUT" || e.target.nodeName == "TEXTAREA") return;
592
593                 var keycode = false;
594                 var shift_key = false;
595
596                 var cmdline = $('cmdline');
597
598                 try {
599                         shift_key = e.shiftKey;
600                 } catch (e) {
601
602                 }
603
604                 if (window.event) {
605                         keycode = window.event.keyCode;
606                 } else if (e) {
607                         keycode = e.which;
608                 }
609
610                 var keychar = String.fromCharCode(keycode);
611
612                 if (keycode == 27) { // escape
613                         if (Element.visible("hotkey_help_overlay")) {
614                                 Element.hide("hotkey_help_overlay");
615                         }
616                         hotkey_prefix = false;
617                 }
618
619                 if (keycode == 16) return; // ignore lone shift
620                 if (keycode == 17) return; // ignore lone ctrl
621
622                 if ((keycode == 70 || keycode == 67 || keycode == 71 || keycode == 65)
623                                 && !hotkey_prefix) {
624
625                         var date = new Date();
626                         var ts = Math.round(date.getTime() / 1000);
627
628                         hotkey_prefix = keycode;
629                         hotkey_prefix_pressed = ts;
630
631                         cmdline.innerHTML = keychar;
632                         Element.show(cmdline);
633
634                         console.log("KP: PREFIX=" + keycode + " CHAR=" + keychar + " TS=" + ts);
635                         return true;
636                 }
637
638                 if (Element.visible("hotkey_help_overlay")) {
639                         Element.hide("hotkey_help_overlay");
640                 }
641
642                 /* Global hotkeys */
643
644                 Element.hide(cmdline);
645
646                 if (!hotkey_prefix) {
647
648                         if (keycode == 27) { // escape
649                                 closeArticlePanel();
650                                 return;
651                         }
652
653                         if (keycode == 69) { // e
654                                 var id = getActiveArticleId();
655                                 emailArticle(id);
656                         }
657
658                         if ((keycode == 191 || keychar == '?') && shift_key) { // ?
659
660                                 new Ajax.Request("backend.php", {
661                                         parameters: "?op=backend&method=help&topic=main",
662                                         onComplete: function(transport) {
663                                                 $("hotkey_help_overlay").innerHTML = transport.responseText;
664                                                 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
665                                         } });
666                                 return false;
667                         }
668
669                         if (keycode == 191 || keychar == '/') { // /
670                                 search();
671                                 return false;
672                         }
673
674                         if (keycode == 74 && !shift_key) { // j
675                                 var rv = dijit.byId("feedTree").getPreviousFeed(
676                                                 getActiveFeedId(), activeFeedIsCat());
677
678                                 if (rv) viewfeed(rv[0], '', rv[1]);
679
680                                 return;
681                         }
682
683                         if (keycode == 75) { // k
684                                 var rv = dijit.byId("feedTree").getNextFeed(
685                                                 getActiveFeedId(), activeFeedIsCat());
686
687                                 if (rv) viewfeed(rv[0], '', rv[1]);
688
689                                 return;
690                         }
691
692                         if (shift_key && keycode == 40) { // shift-down
693                                 catchupRelativeToArticle(1);
694                                 return;
695                         }
696
697                         if (shift_key && keycode == 38) { // shift-up
698                                 catchupRelativeToArticle(0);
699                                 return;
700                         }
701
702                         if (shift_key && keycode == 78) { // N
703                                 scrollArticle(50);
704                                 return;
705                         }
706
707                         if (shift_key && keycode == 80) { // P
708                                 scrollArticle(-50);
709                                 return;
710                         }
711
712                         if (keycode == 68 && shift_key) { // shift-D
713                                 dismissSelectedArticles();
714                                 return;
715                         }
716
717                         if (keycode == 88 && shift_key) { // shift-X
718                                 dismissReadArticles();
719                                 return;
720                         }
721
722                         if (keycode == 78 || keycode == 40) { // n, down
723                                 if (typeof moveToPost != 'undefined') {
724                                         moveToPost('next');
725                                         return false;
726                                 }
727                         }
728
729                         if (keycode == 80 || keycode == 38) { // p, up
730                                 if (typeof moveToPost != 'undefined') {
731                                         moveToPost('prev');
732                                         return false;
733                                 }
734                         }
735
736                         if (keycode == 83 && shift_key) { // S
737                                 selectionTogglePublished(undefined, false, true);
738                                 return;
739                         }
740
741                         if (keycode == 83) { // s
742                                 selectionToggleMarked(undefined, false, true);
743                                 return;
744                         }
745
746                         if (keycode == 85) { // u
747                                 selectionToggleUnread(undefined, false, true);
748                                 return;
749                         }
750
751                         if (keycode == 84 && shift_key) { // T
752                                 var id = getActiveArticleId();
753                                 if (id) {
754                                         editArticleTags(id, getActiveFeedId(), isCdmMode());
755                                         return;
756                                 }
757                         }
758
759                         if (keycode == 9) { // tab
760                                 var id = getArticleUnderPointer();
761                                 if (id) {
762                                         var cb = $("RCHK-" + id);
763
764                                         if (cb) {
765                                                 cb.checked = !cb.checked;
766                                                 toggleSelectRowById(cb, "RROW-" + id);
767                                                 return false;
768                                         }
769                                 }
770                         }
771
772                         if (keycode == 79) { // o
773                                 if (getActiveArticleId()) {
774                                         openArticleInNewWindow(getActiveArticleId());
775                                         return;
776                                 }
777                         }
778
779                         if (keycode == 81 && shift_key) { // Q
780                                 if (typeof catchupAllFeeds != 'undefined') {
781                                         catchupAllFeeds();
782                                         return;
783                                 }
784                         }
785
786                         if (keycode == 88 && !shift_key) { // x
787                                 if (activeFeedIsCat()) {
788                                         dijit.byId("feedTree").collapseCat(getActiveFeedId());
789                                         return;
790                                 }
791                         }
792                 }
793
794                 /* Prefix a */
795
796                 if (hotkey_prefix == 65) { // a
797                         hotkey_prefix = false;
798
799                         if (keycode == 65) { // a
800                                 selectArticles('all');
801                                 return;
802                         }
803
804                         if (keycode == 85) { // u
805                                 selectArticles('unread');
806                                 return;
807                         }
808
809                         if (keycode == 73) { // i
810                                 selectArticles('invert');
811                                 return;
812                         }
813
814                         if (keycode == 78) { // n
815                                 selectArticles('none');
816                                 return;
817                         }
818
819                 }
820
821                 /* Prefix f */
822
823                 if (hotkey_prefix == 70) { // f
824
825                         hotkey_prefix = false;
826
827                         if (keycode == 81) { // q
828                                 if (getActiveFeedId()) {
829                                         catchupCurrentFeed();
830                                         return;
831                                 }
832                         }
833
834                         if (keycode == 82) { // r
835                                 if (getActiveFeedId()) {
836                                         viewfeed(getActiveFeedId(), '', activeFeedIsCat());
837                                         return;
838                                 }
839                         }
840
841                         if (keycode == 65) { // a
842                                 toggleDispRead();
843                                 return false;
844                         }
845
846                         if (keycode == 85) { // u
847                                 if (getActiveFeedId()) {
848                                         viewfeed(getActiveFeedId(), '');
849                                         return false;
850                                 }
851                         }
852
853                         if (keycode == 69) { // e
854
855                                 if (activeFeedIsCat())
856                                         alert(__("You can't edit this kind of feed."));
857                                 else
858                                         editFeed(getActiveFeedId());
859                                 return;
860
861                                 return false;
862                         }
863
864                         if (keycode == 83) { // s
865                                 quickAddFeed();
866                                 return false;
867                         }
868
869                         if (keycode == 67 && shift_key) { // C
870                                 if (typeof catchupAllFeeds != 'undefined') {
871                                         catchupAllFeeds();
872                                         return false;
873                                 }
874                         }
875
876                         if (keycode == 67) { // c
877                                 if (getActiveFeedId()) {
878                                         catchupCurrentFeed();
879                                         return false;
880                                 }
881                         }
882
883                         if (keycode == 88) { // x
884                                 reverseHeadlineOrder();
885                                 return;
886                         }
887                 }
888
889                 /* Prefix c */
890
891                 if (hotkey_prefix == 67) { // c
892                         hotkey_prefix = false;
893
894                         if (keycode == 70) { // f
895                                 quickAddFilter();
896                                 return false;
897                         }
898
899                         if (keycode == 76) { // l
900                                 addLabel();
901                                 return false;
902                         }
903
904                         if (keycode == 83) { // s
905                                 if (typeof collapse_feedlist != 'undefined') {
906                                         collapse_feedlist();
907                                         return false;
908                                 }
909                         }
910
911                         if (keycode == 77) { // m
912                                 // TODO: sortable feedlist
913                                 return;
914                         }
915
916                         if (keycode == 78) { // n
917                                 catchupRelativeToArticle(1);
918                                 return;
919                         }
920
921                         if (keycode == 80) { // p
922                                 catchupRelativeToArticle(0);
923                                 return;
924                         }
925
926
927                 }
928
929                 /* Prefix g */
930
931                 if (hotkey_prefix == 71) { // g
932
933                         hotkey_prefix = false;
934
935
936                         if (keycode == 65) { // a
937                                 viewfeed(-4);
938                                 return false;
939                         }
940
941                         if (keycode == 83) { // s
942                                 viewfeed(-1);
943                                 return false;
944                         }
945
946                         if (keycode == 80 && shift_key) { // P
947                                 gotoPreferences();
948                                 return false;
949                         }
950
951                         if (keycode == 80) { // p
952                                 viewfeed(-2);
953                                 return false;
954                         }
955
956                         if (keycode == 70) { // f
957                                 viewfeed(-3);
958                                 return false;
959                         }
960
961                         if (keycode == 84) { // t
962                                 displayDlg("printTagCloud");
963                                 return false;
964                         }
965                 }
966
967                 /* Cmd */
968
969                 if (hotkey_prefix == 224 || hotkey_prefix == 91) { // f
970                         hotkey_prefix = false;
971                         return;
972                 }
973
974                 if (hotkey_prefix) {
975                         console.log("KP: PREFIX=" + hotkey_prefix + " CODE=" + keycode + " CHAR=" + keychar);
976                 } else {
977                         console.log("KP: CODE=" + keycode + " CHAR=" + keychar);
978                 }
979
980
981         } catch (e) {
982                 exception_error("hotkey_handler", e);
983         }
984 }
985
986 function inPreferences() {
987         return false;
988 }
989
990 function reverseHeadlineOrder() {
991         try {
992
993                 var query_str = "?op=rpc&method=togglepref&key=REVERSE_HEADLINES";
994
995                 new Ajax.Request("backend.php", {
996                         parameters: query_str,
997                         onComplete: function(transport) {
998                                         viewCurrentFeed();
999                                 } });
1000
1001         } catch (e) {
1002                 exception_error("reverseHeadlineOrder", e);
1003         }
1004 }
1005
1006 function scheduleFeedUpdate(id, is_cat) {
1007         try {
1008                 if (!id) {
1009                         id = getActiveFeedId();
1010                         is_cat = activeFeedIsCat();
1011                 }
1012
1013                 if (!id) {
1014                         alert(__("Please select some feed first."));
1015                         return;
1016                 }
1017
1018                 var query = "?op=rpc&method=scheduleFeedUpdate&id=" +
1019                         param_escape(id) +
1020                         "&is_cat=" + param_escape(is_cat);
1021
1022                 console.log(query);
1023
1024                 new Ajax.Request("backend.php", {
1025                         parameters: query,
1026                         onComplete: function(transport) {
1027                                 handle_rpc_json(transport);
1028
1029                                 var reply = JSON.parse(transport.responseText);
1030                                 var message = reply['message'];
1031
1032                                 if (message) {
1033                                         notify_info(message);
1034                                         return;
1035                                 }
1036
1037                         } });
1038
1039
1040         } catch (e) {
1041                 exception_error("scheduleFeedUpdate", e);
1042         }
1043 }
1044
1045 function newVersionDlg() {
1046         try {
1047                 var query = "backend.php?op=dlg&method=newVersion";
1048
1049                 if (dijit.byId("newVersionDlg"))
1050                         dijit.byId("newVersionDlg").destroyRecursive();
1051
1052                 dialog = new dijit.Dialog({
1053                         id: "newVersionDlg",
1054                         title: __("New version available!"),
1055                         style: "width: 600px",
1056                         href: query,
1057                 });
1058
1059                 dialog.show();
1060
1061         } catch (e) {
1062                 exception_error("newVersionDlg", e);
1063         }
1064 }
1065
1066 function handle_rpc_json(transport, scheduled_call) {
1067         try {
1068                 var reply = JSON.parse(transport.responseText);
1069
1070                 if (reply) {
1071
1072                         var error = reply['error'];
1073
1074                         if (error) {
1075                                 var code = error['code'];
1076                                 var msg = error['msg'];
1077
1078                                 console.warn("[handle_rpc_json] received fatal error " + code + "/" + msg);
1079
1080                                 if (code != 0) {
1081                                         fatalError(code, msg);
1082                                         return false;
1083                                 }
1084                         }
1085
1086                         var seq = reply['seq'];
1087
1088                         if (seq) {
1089                                 if (get_seq() != seq) {
1090                                         console.log("[handle_rpc_json] sequence mismatch: " + seq +
1091                                                 " (want: " + get_seq() + ")");
1092                                         return true;
1093                                 }
1094                         }
1095
1096                         var message = reply['message'];
1097
1098                         if (message) {
1099                                 if (message == "UPDATE_COUNTERS") {
1100                                         console.log("need to refresh counters...");
1101                                         setInitParam("last_article_id", -1);
1102                                         _force_scheduled_update = true;
1103                                 }
1104                         }
1105
1106                         var counters = reply['counters'];
1107
1108                         if (counters)
1109                                 parse_counters(counters, scheduled_call);
1110
1111                         var runtime_info = reply['runtime-info'];;
1112
1113                         if (runtime_info)
1114                                 parse_runtime_info(runtime_info);
1115
1116                         hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1117
1118                 } else {
1119                         notify_error("Error communicating with server.");
1120                 }
1121
1122         } catch (e) {
1123                 notify_error("Error communicating with server.");
1124                 console.log(e);
1125                 //exception_error("handle_rpc_json", e, transport);
1126         }
1127
1128         return true;
1129 }
1130