4 var display_tags = false;
5 var global_unread = -1;
6 var active_title_text = "";
7 var current_subtitle = "";
8 var daemon_enabled = false;
9 var daemon_refresh_only = false;
10 //var _qfd_deleted_feed = 0;
11 var firsttime_update = true;
12 var _active_feed_id = 0;
13 var _active_feed_is_cat = false;
14 var number_of_feeds = 0;
15 var sanity_check_done = false;
16 var _hfd_scrolltop = 0;
17 var hotkey_prefix = false;
18 var hotkey_prefix_pressed = false;
19 var init_params = new Object();
22 var feeds_sort_by_unread = false;
23 var feedlist_sortable_enabled = false;
25 function activeFeedIsCat() {
26 return _active_feed_is_cat;
29 function getActiveFeedId() {
31 debug("gAFID: " + _active_feed_id);
32 return _active_feed_id;
34 exception_error("getActiveFeedId", e);
38 function setActiveFeedId(id, is_cat) {
40 debug("sAFID(" + id + ", " + is_cat + ")");
43 if (is_cat != undefined) {
44 _active_feed_is_cat = is_cat;
48 exception_error("setActiveFeedId", e);
53 function isFeedlistSortable() {
54 return feedlist_sortable_enabled;
57 function tagsAreDisplayed() {
61 function toggleTags(show_all) {
65 debug("toggleTags: " + show_all + "; " + display_tags);
67 var p = $("dispSwitchPrompt");
69 if (!show_all && !display_tags) {
70 displayDlg("printTagCloud");
71 } else if (show_all) {
74 p.innerHTML = __("display feeds");
75 notify_progress("Loading, please wait...", true);
77 } else if (display_tags) {
79 p.innerHTML = __("tag cloud");
80 notify_progress("Loading, please wait...", true);
85 exception_error("toggleTags", e);
89 function dlg_frefresh_callback(transport, deleted_feed) {
90 if (getActiveFeedId() == deleted_feed) {
91 var h = $("headlines-frame");
93 h.innerHTML = "<div class='whiteBox'>" + __('No feed selected.') + "</div>";
97 setTimeout('updateFeedList(false, false)', 50);
101 function backend_sanity_check_callback(transport) {
105 if (sanity_check_done) {
106 fatalError(11, "Sanity check request received twice. This can indicate "+
107 "presence of Firebug or some other disrupting extension. "+
108 "Please disable it and try again.");
112 if (!transport.responseXML) {
114 fatalError(3, "Sanity check: Received reply is not XML",
115 transport.responseText);
123 if (getURLParam("offline")) {
124 return init_offline();
127 var reply = transport.responseXML.firstChild.firstChild;
130 fatalError(3, "Sanity check: invalid RPC reply", transport.responseText);
134 var error_code = reply.getAttribute("error-code");
136 if (error_code && error_code != 0) {
137 return fatalError(error_code, reply.getAttribute("error-msg"));
140 debug("sanity check ok");
142 var params = reply.nextSibling;
145 debug('reading init-params...');
146 var param = params.firstChild;
149 var k = param.getAttribute("key");
150 var v = param.getAttribute("value");
151 debug(k + " => " + v);
155 db.execute("DELETE FROM init_params WHERE key = ?", [k]);
156 db.execute("INSERT INTO init_params (key,value) VALUES (?, ?)",
160 param = param.nextSibling;
164 sanity_check_done = true;
169 exception_error("backend_sanity_check_callback", e, transport);
173 function scheduleFeedUpdate(force) {
175 debug("in scheduleFeedUpdate");
177 /* if (!daemon_enabled && !daemon_refresh_only) {
178 notify_progress("Updating feeds...", true);
181 var query_str = "backend.php?op=rpc&subop=";
184 query_str = query_str + "forceUpdateAllFeeds";
186 query_str = query_str + "updateAllFeeds";
191 if (firsttime_update && !navigator.userAgent.match("Opera")) {
192 firsttime_update = false;
202 query_str = query_str + "&omode=" + omode;
203 query_str = query_str + "&uctr=" + global_unread;
205 var date = new Date();
206 var timestamp = Math.round(date.getTime() / 1000);
207 query_str = query_str + "&ts=" + timestamp
209 debug("REFETCH query: " + query_str);
211 new Ajax.Request("backend.php", {
212 parameters: query_str,
213 onComplete: function(transport) {
214 parse_counters_reply(transport, true);
218 function updateFeedList(silent, fetch) {
220 // if (silent != true) {
221 // notify("Loading feed list...");
224 debug("<b>updateFeedList</b>");
226 if (offline_mode) return render_offline_feedlist();
228 var query_str = "backend.php?op=feeds";
231 query_str = query_str + "&tags=1";
234 if (getActiveFeedId() && !activeFeedIsCat()) {
235 query_str = query_str + "&actid=" + getActiveFeedId();
238 var date = new Date();
239 var timestamp = Math.round(date.getTime() / 1000);
240 query_str = query_str + "&ts=" + timestamp
242 if (fetch) query_str = query_str + "&fetch=yes";
244 // var feeds_frame = $("feeds-frame");
245 // feeds_frame.src = query_str;
247 debug("updateFeedList Q=" + query_str);
249 new Ajax.Request("backend.php", {
250 parameters: query_str,
251 onComplete: function(transport) {
252 feedlist_callback2(transport);
257 function catchupAllFeeds() {
259 var str = __("Mark all articles as read?");
261 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
263 var query_str = "backend.php?op=feeds&subop=catchupAll";
265 notify_progress("Marking all feeds as read...");
267 debug("catchupAllFeeds Q=" + query_str);
269 new Ajax.Request("backend.php", {
270 parameters: query_str,
271 onComplete: function(transport) {
272 feedlist_callback2(transport);
280 function viewCurrentFeed(subop) {
282 // if (getActiveFeedId()) {
283 if (getActiveFeedId() != undefined) {
284 viewfeed(getActiveFeedId(), subop, activeFeedIsCat());
286 disableContainerChildren("headlinesToolbar", false, document);
287 // viewfeed(-1, subop); // FIXME
289 return false; // block unneeded form submits
292 function viewfeed(feed, subop) {
293 var f = window.frames["feeds-frame"];
294 f.viewfeed(feed, subop);
298 if (getInitParam("bw_limit") == "1") return;
300 scheduleFeedUpdate(false);
302 var refresh_time = getInitParam("feeds_frame_refresh");
304 if (!refresh_time) refresh_time = 600;
306 setTimeout("timeout()", refresh_time*1000);
309 function resetSearch() {
310 var searchbox = $("searchbox")
312 if (searchbox.value != "" && getActiveFeedId()) {
313 searchbox.value = "";
314 viewfeed(getActiveFeedId(), "");
318 function searchCancel() {
324 viewCurrentFeed(0, "");
327 // if argument is undefined, current subtitle is not updated
328 // use blank string to clear subtitle
329 function updateTitle(s) {
330 var tmp = "Tiny Tiny RSS";
332 if (s != undefined) {
333 current_subtitle = s;
336 if (global_unread > 0) {
337 tmp = tmp + " (" + global_unread + ")";
340 if (current_subtitle) {
341 tmp = tmp + " - " + current_subtitle;
344 if (active_title_text.length > 0) {
345 tmp = tmp + " > " + active_title_text;
348 document.title = tmp;
351 function genericSanityCheck() {
353 // if (!Ajax.getTransport()) fatalError(1);
355 setCookie("ttrss_test", "TEST");
357 if (getCookie("ttrss_test") != "TEST") {
370 disableContainerChildren("headlinesToolbar", true);
372 Form.disable("main_toolbar_form");
374 if (!genericSanityCheck())
377 if (getURLParam('debug')) {
378 Element.show("debug_output");
379 debug('debug mode activated');
382 var params = "&ua=" + param_escape(navigator.userAgent);
384 loading_set_progress(30);
386 new Ajax.Request("backend.php", {
387 parameters: "backend.php?op=rpc&subop=sanityCheck" + params,
388 onComplete: function(transport) {
389 backend_sanity_check_callback(transport);
393 exception_error("init", e);
397 function resize_headlines(delta_x, delta_y) {
401 debug("resize_headlines: " + delta_x + ":" + delta_y);
403 var h_frame = $("headlines-frame");
404 var c_frame = $("content-frame");
405 var f_frame = $("footer");
406 var feeds_frame = $("feeds-holder");
407 var resize_grab = $("resize-grabber");
408 var resize_handle = $("resize-handle");
410 if (!c_frame || !h_frame) return;
412 if (feeds_frame && getInitParam("theme") == "compat") {
413 feeds_frame.style.bottom = f_frame.offsetHeight + "px";
416 if (getInitParam("theme") == "3pane") {
418 if (delta_x != undefined) {
419 if (c_frame.offsetLeft - delta_x > feeds_frame.offsetWidth + feeds_frame.offsetLeft + 100 && c_frame.offsetWidth + delta_x > 100) {
420 hor_offset = hor_offset + delta_x;
424 debug("resize_headlines: HOR-mode: " + hor_offset);
426 c_frame.style.width = (400 + hor_offset) + "px";
427 h_frame.style.right = c_frame.offsetWidth - 1 + "px";
429 resize_grab.style.top = (h_frame.offsetTop + h_frame.offsetHeight - 60) + "px";
430 resize_grab.style.left = (h_frame.offsetLeft + h_frame.offsetWidth -
432 resize_grab.style.display = "block";
434 resize_handle.src = "themes/3pane/images/resize_handle_vert.png";
435 resize_handle.style.paddingTop = (resize_grab.offsetHeight / 2 - 7) + "px";
439 if (delta_y != undefined) {
440 if (c_frame.offsetHeight + delta_y > 100 && h_frame.offsetHeight - delta_y > 100) {
441 ver_offset = ver_offset + delta_y;
445 debug("resize_headlines: VER-mode: " + ver_offset);
447 h_frame.style.height = (300 - ver_offset) + "px";
449 c_frame.style.top = (h_frame.offsetTop + h_frame.offsetHeight + 0) + "px";
450 h_frame.style.height = h_frame.offsetHeight + "px";
454 if (getInitParam("theme") == "graycube") {
458 if (getInitParam("theme") == "graycube" || getInitParam("theme") == "compat") {
459 resize_handle.src = "themes/graycube/images/resize_handle_horiz.png";
462 /* resize_grab.style.top = (h_frame.offsetTop + h_frame.offsetHeight -
464 resize_grab.style.display = "block"; */
468 if (getInitParam("cookie_lifetime") != 0) {
469 setCookie("ttrss_offset_ver", ver_offset,
470 getInitParam("cookie_lifetime"));
471 setCookie("ttrss_offset_hor", hor_offset,
472 getInitParam("cookie_lifetime"));
474 setCookie("ttrss_offset_ver", ver_offset);
475 setCookie("ttrss_offset_hor", hor_offset);
479 exception_error("resize_headlines", e);
484 function init_second_stage() {
488 delCookie("ttrss_test");
490 // document.onresize = resize_headlines;
491 window.onresize=resize_headlines;
493 var toolbar = document.forms["main_toolbar_form"];
495 dropboxSelect(toolbar.view_mode, getInitParam("default_view_mode"));
496 dropboxSelect(toolbar.order_by, getInitParam("default_view_order_by"));
498 daemon_enabled = getInitParam("daemon_enabled") == 1;
499 daemon_refresh_only = getInitParam("daemon_refresh_only") == 1;
500 feeds_sort_by_unread = getInitParam("feeds_sort_by_unread") == 1;
502 /* var fl = cache_find_param("FEEDLIST", getInitParam("num_feeds"));
509 setTimeout('updateFeedList(false, false)', 50);
512 setTimeout('updateFeedList(false, false)', 50);
515 setTimeout('updateFeedList(false, false)', 50);
517 debug("second stage ok");
519 loading_set_progress(60);
521 ver_offset = parseInt(getCookie("ttrss_offset_ver"));
522 hor_offset = parseInt(getCookie("ttrss_offset_hor"));
524 debug("got offsets from cookies: ver " + ver_offset + " hor " + hor_offset);
528 if (isNaN(hor_offset)) hor_offset = 0;
529 if (isNaN(ver_offset)) ver_offset = 0;
531 debug("offsets from cookies [x:y]: " + hor_offset + ":" + ver_offset);
535 enable_offline_reading();
538 exception_error("init_second_stage", e);
542 function quickMenuChange() {
543 var chooser = $("quickMenuChooser");
544 var opid = chooser[chooser.selectedIndex].value;
546 chooser.selectedIndex = 0;
550 function quickMenuGo(opid) {
553 if (opid == "qmcPrefs") {
557 if (opid == "qmcSearch") {
558 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat());
562 if (opid == "qmcAddFeed") {
563 displayDlg('quickAddFeed', '',
564 function () {$('feed_url').focus();});
568 if (opid == "qmcEditFeed") {
569 editFeedDlg(getActiveFeedId());
572 if (opid == "qmcRemoveFeed") {
573 var actid = getActiveFeedId();
575 if (activeFeedIsCat()) {
576 alert(__("You can't unsubscribe from the category."));
581 alert(__("Please select some feed first."));
585 var fn = getFeedName(actid);
587 var pr = __("Unsubscribe from %s?").replace("%s", fn);
590 unsubscribeFeed(actid);
596 if (opid == "qmcClearFeed") {
597 var actid = getActiveFeedId();
600 alert(__("Please select some feed first."));
604 if (activeFeedIsCat() || actid < 0) {
605 alert(__("You can't clear this type of feed."));
609 var fn = getFeedName(actid);
611 var pr = __("Erase all non-starred articles in %s?").replace("%s", fn);
614 clearFeedArticles(actid);
621 if (opid == "qmcUpdateFeeds") {
622 scheduleFeedUpdate(true);
626 if (opid == "qmcCatchupAll") {
631 if (opid == "qmcShowOnlyUnread") {
636 if (opid == "qmcAddFilter") {
637 displayDlg("quickAddFilter", getActiveFeedId());
640 if (opid == "qmcAddLabel") {
644 if (opid == "qmcRescoreFeed") {
645 rescoreCurrentFeed();
648 if (opid == "qmcHKhelp") {
649 //Element.show("hotkey_help_overlay");
650 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
653 if (opid == "qmcResetUI") {
659 if (opid == "qmcResetCats") {
661 if (confirm(__("Reset category order?"))) {
663 var query = "?op=feeds&subop=catsortreset";
665 notify_progress("Loading, please wait...", true);
667 new Ajax.Request("backend.php", {
669 onComplete: function(transport) {
670 window.setTimeout('updateFeedList(false, false)', 50);
676 exception_error("quickMenuGo", e);
680 function unsubscribeFeed(feed_id, title) {
683 var msg = __("Unsubscribe from %s?").replace("%s", title);
685 if (title == undefined || confirm(msg)) {
686 notify_progress("Removing feed...");
688 var query = "?op=pref-feeds&quiet=1&subop=remove&ids=" + feed_id;
690 new Ajax.Request("backend.php", {
692 onComplete: function(transport) {
693 dlg_frefresh_callback(transport, feed_id);
701 function updateFeedTitle(t) {
702 active_title_text = t;
706 function toggleDispRead() {
709 var hide = !(getInitParam("hide_read_feeds") == "1");
711 hideOrShowFeeds(hide);
713 var query = "?op=rpc&subop=setpref&key=HIDE_READ_FEEDS&value=" +
716 new Ajax.Request("backend.php", {
718 onComplete: function(transport) {
719 setInitParam("hide_read_feeds", hide);
723 exception_error("toggleDispRead", e);
727 function parse_runtime_info(elem) {
729 debug("parse_runtime_info: elem is null, aborting");
733 var param = elem.firstChild;
735 debug("parse_runtime_info: " + param);
738 var k = param.getAttribute("key");
739 var v = param.getAttribute("value");
741 debug("RI: " + k + " => " + v);
743 if (k == "num_feeds") {
747 if (k == "new_version_available") {
748 var icon = $("newVersionIcon");
751 icon.style.display = "inline";
753 icon.style.display = "none";
760 if (k == "daemon_is_running" && v != 1) {
761 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
765 if (k == "daemon_stamp_ok" && v != 1) {
766 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
774 /* var w = $("noDaemonWarning");
777 if (k == "daemon_is_running" && v != 1) {
778 w.style.display = "block";
780 w.style.display = "none";
783 param = param.nextSibling;
787 function catchupCurrentFeed() {
789 var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
791 var str = __("Mark all articles in %s as read?").replace("%s", fn);
793 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
794 return viewCurrentFeed('MarkAllRead')
798 function catchupFeedInGroup(id) {
802 var title = getFeedName(id);
804 var str = __("Mark all articles in %s as read?").replace("%s", title);
806 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
807 return viewCurrentFeed('MarkAllReadGR:' + id)
811 exception_error("catchupFeedInGroup", e);
815 function editFeedDlg(feed) {
819 alert(__("Please select some feed first."));
823 if ((feed <= 0) || activeFeedIsCat() || tagsAreDisplayed()) {
824 alert(__("You can't edit this kind of feed."));
831 query = "?op=pref-feeds&subop=editfeed&id=" + param_escape(feed);
833 query = "?op=pref-labels&subop=edit&id=" + param_escape(-feed-11);
838 notify_progress("Loading, please wait...", true);
840 new Ajax.Request("backend.php", {
842 onComplete: function(transport) {
843 infobox_callback2(transport);
847 exception_error("editFeedDlg", e);
851 /* this functions duplicate those of prefs.js feed editor, with
852 some differences because there is no feedlist */
854 function feedEditCancel() {
859 function feedEditSave() {
863 // FIXME: add parameter validation
865 var query = Form.serialize("edit_feed_form");
867 notify_progress("Saving feed...");
869 new Ajax.Request("backend.php", {
871 onComplete: function(transport) {
872 dlg_frefresh_callback(transport);
881 exception_error("feedEditSave (main)", e);
885 function clearFeedArticles(feed_id) {
887 notify_progress("Clearing feed...");
889 var query = "?op=pref-feeds&quiet=1&subop=clear&id=" + feed_id;
891 new Ajax.Request("backend.php", {
893 onComplete: function(transport) {
894 dlg_frefresh_callback(transport, feed_id);
900 function collapse_feedlist() {
902 debug("collapse_feedlist");
904 var theme = getInitParam("theme");
905 if (theme != "" && theme != "compact" && theme != "graycube" &&
906 theme != "compat") return;
908 var fl = $("feeds-holder");
909 var fh = $("headlines-frame");
910 var fc = $("content-frame");
911 var ft = $("toolbar");
912 var ff = $("footer");
913 var fhdr = $("header");
914 var fbtn = $("collapse_feeds_btn");
916 if (!Element.visible(fl)) {
918 fbtn.innerHTML = "<<";
920 if (theme != "graycube") {
922 fh.style.left = fl.offsetWidth + "px";
923 ft.style.left = fl.offsetWidth + "px";
924 if (fc) fc.style.left = fl.offsetWidth + "px";
925 if (ff && theme != "compat") ff.style.left = (fl.offsetWidth-1) + "px";
927 if (theme == "compact") fhdr.style.left = (fl.offsetWidth + 10) + "px";
929 fh.style.left = fl.offsetWidth + 40 + "px";
930 ft.style.left = fl.offsetWidth + 40 +"px";
931 if (fc) fc.style.left = fl.offsetWidth + 40 + "px";
934 query = "?op=rpc&subop=setpref&key=_COLLAPSED_FEEDLIST&value=false";
936 new Ajax.Request("backend.php", { parameters: query });
940 fbtn.innerHTML = ">>";
942 if (theme != "graycube") {
944 fh.style.left = "0px";
945 ft.style.left = "0px";
946 if (fc) fc.style.left = "0px";
947 if (ff) ff.style.left = "0px";
949 if (theme == "compact") fhdr.style.left = "10px";
952 fh.style.left = "20px";
953 ft.style.left = "20px";
954 if (fc) fc.style.left = "20px";
958 query = "?op=rpc&subop=setpref&key=_COLLAPSED_FEEDLIST&value=true";
960 new Ajax.Request("backend.php", { parameters: query });
964 exception_error("collapse_feedlist", e);
968 function viewModeChanged() {
970 return viewCurrentFeed(0, '')
973 function viewLimitChanged() {
975 return viewCurrentFeed(0, '')
978 /* function adjustArticleScore(id, score) {
981 var pr = prompt(__("Assign score to article:"), score);
983 if (pr != undefined) {
984 var query = "?op=rpc&subop=setScore&id=" + id + "&score=" + pr;
986 new Ajax.Request("backend.php", {
988 onComplete: function(transport) {
994 exception_error("adjustArticleScore", e);
998 function rescoreCurrentFeed() {
1000 var actid = getActiveFeedId();
1002 if (activeFeedIsCat() || actid < 0 || tagsAreDisplayed()) {
1003 alert(__("You can't rescore this kind of feed."));
1008 alert(__("Please select some feed first."));
1012 var fn = getFeedName(actid);
1013 var pr = __("Rescore articles in %s?").replace("%s", fn);
1016 notify_progress("Rescoring articles...");
1018 var query = "?op=pref-feeds&subop=rescore&quiet=1&ids=" + actid;
1020 new Ajax.Request("backend.php", {
1022 onComplete: function(transport) {
1028 function hotkey_handler(e) {
1033 var shift_key = false;
1035 var cmdline = $('cmdline');
1036 var feedlist = $('feedList');
1039 shift_key = e.shiftKey;
1045 keycode = window.event.keyCode;
1050 var keychar = String.fromCharCode(keycode);
1052 if (keycode == 27) { // escape
1053 if (Element.visible("hotkey_help_overlay")) {
1054 Element.hide("hotkey_help_overlay");
1056 hotkey_prefix = false;
1060 if (!hotkeys_enabled) {
1061 debug("hotkeys disabled");
1065 if (keycode == 16) return; // ignore lone shift
1067 if ((keycode == 70 || keycode == 67 || keycode == 71)
1068 && !hotkey_prefix) {
1070 var date = new Date();
1071 var ts = Math.round(date.getTime() / 1000);
1073 hotkey_prefix = keycode;
1074 hotkey_prefix_pressed = ts;
1076 cmdline.innerHTML = keychar;
1077 Element.show(cmdline);
1079 debug("KP: PREFIX=" + keycode + " CHAR=" + keychar + " TS=" + ts);
1083 if (Element.visible("hotkey_help_overlay")) {
1084 Element.hide("hotkey_help_overlay");
1087 /* Global hotkeys */
1089 Element.hide(cmdline);
1091 if (!hotkey_prefix) {
1093 if (keycode == 68 && shift_key) { // d
1094 if (!Element.visible("debug_output")) {
1095 Element.show("debug_output");
1096 debug('debug mode activated');
1098 Element.hide("debug_output");
1104 if ((keycode == 191 || keychar == '?') && shift_key) { // ?
1105 if (!Element.visible("hotkey_help_overlay")) {
1106 //Element.show("hotkey_help_overlay");
1107 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
1109 Element.hide("hotkey_help_overlay");
1114 if (keycode == 191 || keychar == '/') { // /
1115 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat());
1119 if (keycode == 82 && shift_key) { // R
1120 scheduleFeedUpdate(true);
1124 if (keycode == 74) { // j
1125 var feed = getActiveFeedId();
1126 var new_feed = getRelativeFeedId2(feed, activeFeedIsCat(), 'prev');
1127 // alert(feed + " IC: " + activeFeedIsCat() + " => " + new_feed);
1129 var is_cat = new_feed.match("CAT:");
1131 new_feed = new_feed.replace("CAT:", "");
1132 viewCategory(new_feed);
1134 viewfeed(new_feed, '', false);
1140 if (keycode == 75) { // k
1141 var feed = getActiveFeedId();
1142 var new_feed = getRelativeFeedId2(feed, activeFeedIsCat(), 'next');
1143 // alert(feed + " IC: " + activeFeedIsCat() + " => " + new_feed);
1145 var is_cat = new_feed.match("CAT:");
1146 if (is_cat == "CAT:") {
1147 new_feed = new_feed.replace("CAT:", "");
1148 viewCategory(new_feed);
1150 viewfeed(new_feed, '', false);
1156 if (shift_key && keycode == 40) { // shift-down
1157 catchupRelativeToArticle(1);
1161 if (shift_key && keycode == 38) { // shift-up
1162 catchupRelativeToArticle(0);
1166 if (shift_key && keycode == 78) { // N
1171 if (shift_key && keycode == 80) { // P
1177 if (keycode == 78 || keycode == 40) { // n, down
1178 if (typeof moveToPost != 'undefined') {
1184 if (keycode == 80 || keycode == 38) { // p, up
1185 if (typeof moveToPost != 'undefined') {
1191 if (keycode == 83 && shift_key) { // S
1192 var id = getActiveArticleId();
1199 if (keycode == 83) { // s
1200 var id = getActiveArticleId();
1208 if (keycode == 85) { // u
1209 var id = getActiveArticleId();
1216 if (keycode == 84 && shift_key) { // T
1217 var id = getActiveArticleId();
1219 editArticleTags(id, getActiveFeedId(), isCdmMode());
1224 if (keycode == 9) { // tab
1225 var id = getArticleUnderPointer();
1227 var cb = $("RCHK-" + id);
1230 cb.checked = !cb.checked;
1231 toggleSelectRowById(cb, "RROW-" + id);
1237 if (keycode == 79) { // o
1238 if (getActiveArticleId()) {
1239 openArticleInNewWindow(getActiveArticleId());
1244 if (keycode == 81 && shift_key) { // Q
1245 if (typeof catchupAllFeeds != 'undefined') {
1251 if (keycode == 88) { // x
1252 if (activeFeedIsCat()) {
1253 toggleCollapseCat(getActiveFeedId());
1260 if (hotkey_prefix == 70) { // f
1262 hotkey_prefix = false;
1264 if (keycode == 81) { // q
1265 if (getActiveFeedId()) {
1266 catchupCurrentFeed();
1271 if (keycode == 82) { // r
1272 if (getActiveFeedId()) {
1273 viewfeed(getActiveFeedId(), "ForceUpdate", activeFeedIsCat());
1278 if (keycode == 65) { // a
1283 if (keycode == 85 && shift_key) { // U
1284 scheduleFeedUpdate(true);
1288 if (keycode == 85) { // u
1289 if (getActiveFeedId()) {
1290 viewfeed(getActiveFeedId(), "ForceUpdate");
1295 if (keycode == 69) { // e
1296 editFeedDlg(getActiveFeedId());
1300 if (keycode == 83) { // s
1301 displayDlg("quickAddFeed");
1305 if (keycode == 67 && shift_key) { // C
1306 if (typeof catchupAllFeeds != 'undefined') {
1312 if (keycode == 67) { // c
1313 if (getActiveFeedId()) {
1314 catchupCurrentFeed();
1319 if (keycode == 87) { // w
1320 feeds_sort_by_unread = !feeds_sort_by_unread;
1321 return resort_feedlist();
1324 if (keycode == 72) { // h
1325 hideReadHeadlines();
1333 if (hotkey_prefix == 67) { // c
1334 hotkey_prefix = false;
1336 if (keycode == 70) { // f
1337 displayDlg("quickAddFilter", getActiveFeedId());
1341 if (keycode == 76) { // l
1346 if (keycode == 83) { // s
1347 if (typeof collapse_feedlist != 'undefined') {
1348 collapse_feedlist();
1353 if (keycode == 77) { // m
1354 feedlist_sortable_enabled = !feedlist_sortable_enabled;
1355 if (feedlist_sortable_enabled) {
1356 notify_info("Category reordering enabled");
1357 toggle_sortable_feedlist(true);
1359 notify_info("Category reordering disabled");
1360 toggle_sortable_feedlist(false);
1364 if (keycode == 78) { // n
1365 catchupRelativeToArticle(1);
1369 if (keycode == 80) { // p
1370 catchupRelativeToArticle(0);
1379 if (hotkey_prefix == 71) { // g
1381 hotkey_prefix = false;
1384 if (keycode == 65) { // a
1389 if (keycode == 83) { // s
1394 if (keycode == 80 && shift_key) { // P
1399 if (keycode == 80) { // p
1404 if (keycode == 70) { // f
1409 if (keycode == 84 && shift_key) { // T
1417 if (hotkey_prefix == 224 || hotkey_prefix == 91) { // f
1418 hotkey_prefix = false;
1422 if (hotkey_prefix) {
1423 debug("KP: PREFIX=" + hotkey_prefix + " CODE=" + keycode + " CHAR=" + keychar);
1425 debug("KP: CODE=" + keycode + " CHAR=" + keychar);
1430 exception_error("hotkey_handler", e);
1434 function feedsSortByUnread() {
1435 return feeds_sort_by_unread;
1438 function addLabel() {
1442 var caption = prompt(__("Please enter label caption:"), "");
1444 if (caption != undefined) {
1446 if (caption == "") {
1447 alert(__("Can't create label: missing caption."));
1451 var query = "?op=pref-labels&subop=add&caption=" +
1452 param_escape(caption);
1454 notify_progress("Loading, please wait...", true);
1456 new Ajax.Request("backend.php", {
1458 onComplete: function(transport) {
1465 exception_error("addLabel", e);
1469 function visitOfficialSite() {
1470 window.open("http://tt-rss.org/");
1473 function inPreferences() {