3 var display_tags
= false;
4 var global_unread
= -1;
5 var active_title_text
= "";
6 var current_subtitle
= "";
7 var daemon_enabled
= false;
8 var daemon_refresh_only
= false;
9 //var _qfd_deleted_feed = 0;
10 var firsttime_update
= true;
11 var _active_feed_id
= 0;
12 var _active_feed_is_cat
= false;
13 var number_of_feeds
= 0;
14 var sanity_check_done
= false;
15 var _hfd_scrolltop
= 0;
16 var hotkey_prefix
= false;
17 var init_params
= new Object();
20 var feeds_sort_by_unread
= false;
21 var feedlist_sortable_enabled
= false;
22 var offline_mode
= false;
24 var localServer
= false;
26 var download_progress_last
= 0;
28 function activeFeedIsCat() {
29 return _active_feed_is_cat
;
32 function getActiveFeedId() {
33 // return getCookie("ttrss_vf_actfeed");
35 debug("gAFID: " + _active_feed_id
);
36 return _active_feed_id
;
38 exception_error("getActiveFeedId", e
);
42 function setActiveFeedId(id
, is_cat
) {
43 // return setCookie("ttrss_vf_actfeed", id);
45 debug("sAFID(" + id
+ ", " + is_cat
+ ")");
48 if (is_cat
!= undefined) {
49 _active_feed_is_cat
= is_cat
;
53 exception_error("setActiveFeedId", e
);
58 function isFeedlistSortable() {
59 return feedlist_sortable_enabled
;
62 function tagsAreDisplayed() {
66 function toggleTags(show_all
) {
70 debug("toggleTags: " + show_all
+ "; " + display_tags
);
72 var p
= document
.getElementById("dispSwitchPrompt");
74 if (!show_all
&& !display_tags
) {
75 displayDlg("printTagCloud");
76 } else if (show_all
) {
79 p
.innerHTML
= __("display feeds");
80 notify_progress("Loading, please wait...", true);
82 } else if (display_tags
) {
84 p
.innerHTML
= __("tag cloud");
85 notify_progress("Loading, please wait...", true);
90 exception_error("toggleTags", e
);
94 function dlg_frefresh_callback(transport
, deleted_feed
) {
95 if (getActiveFeedId() == deleted_feed
) {
96 var h
= document
.getElementById("headlines-frame");
98 h
.innerHTML
= "<div class='whiteBox'>" + __('No feed selected.') + "</div>";
102 setTimeout('updateFeedList(false, false)', 50);
106 function refetch_callback2(transport
) {
109 var date
= new Date();
111 parse_counters_reply(transport
, true);
113 debug("refetch_callback2: done");
115 /* if (!daemon_enabled && !daemon_refresh_only) {
116 notify_info("All feeds updated.");
122 exception_error("refetch_callback", e
);
127 function backend_sanity_check_callback(transport
) {
131 if (sanity_check_done
) {
132 fatalError(11, "Sanity check request received twice. This can indicate "+
133 "presence of Firebug or some other disrupting extension. "+
134 "Please disable it and try again.");
138 if (!transport
.responseXML
) {
139 if (!window
.google
&& !google
.gears
) {
140 fatalError(3, "Sanity check: Received reply is not XML", transport
.responseText
);
147 if (getURLParam("offline")) {
148 return init_offline();
151 var reply
= transport
.responseXML
.firstChild
.firstChild
;
154 fatalError(3, "Sanity check: invalid RPC reply", transport
.responseText
);
158 var error_code
= reply
.getAttribute("error-code");
160 if (error_code
&& error_code
!= 0) {
161 return fatalError(error_code
, reply
.getAttribute("error-msg"));
164 debug("sanity check ok");
166 var params
= reply
.nextSibling
;
169 debug('reading init-params...');
170 var param
= params
.firstChild
;
173 var k
= param
.getAttribute("key");
174 var v
= param
.getAttribute("value");
175 debug(k
+ " => " + v
);
177 param
= param
.nextSibling
;
181 sanity_check_done
= true;
186 exception_error("backend_sanity_check_callback", e
, transport
);
190 function scheduleFeedUpdate(force
) {
192 debug("in scheduleFeedUpdate");
194 /* if (!daemon_enabled && !daemon_refresh_only) {
195 notify_progress("Updating feeds...", true);
198 var query_str
= "backend.php?op=rpc&subop=";
201 query_str
= query_str
+ "forceUpdateAllFeeds";
203 query_str
= query_str
+ "updateAllFeeds";
208 if (firsttime_update
&& !navigator
.userAgent
.match("Opera")) {
209 firsttime_update
= false;
219 query_str
= query_str
+ "&omode=" + omode
;
220 query_str
= query_str
+ "&uctr=" + global_unread
;
222 var date
= new Date();
223 var timestamp
= Math
.round(date
.getTime() / 1000);
224 query_str
= query_str
+ "&ts=" + timestamp
226 debug("REFETCH query: " + query_str
);
228 new Ajax
.Request(query_str
, {
229 onComplete: function(transport
) {
230 refetch_callback2(transport
);
234 function updateFeedList(silent
, fetch
) {
236 // if (silent != true) {
237 // notify("Loading feed list...");
240 debug("<b>updateFeedList</b>");
242 var query_str
= "backend.php?op=feeds";
245 query_str
= query_str
+ "&tags=1";
248 if (getActiveFeedId() && !activeFeedIsCat()) {
249 query_str
= query_str
+ "&actid=" + getActiveFeedId();
252 var date
= new Date();
253 var timestamp
= Math
.round(date
.getTime() / 1000);
254 query_str
= query_str
+ "&ts=" + timestamp
256 if (fetch
) query_str
= query_str
+ "&fetch=yes";
258 // var feeds_frame = document.getElementById("feeds-frame");
259 // feeds_frame.src = query_str;
261 debug("updateFeedList Q=" + query_str
);
263 new Ajax
.Request(query_str
, {
264 onComplete: function(transport
) {
265 feedlist_callback2(transport
);
270 function catchupAllFeeds() {
272 var str
= __("Mark all articles as read?");
274 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
276 var query_str
= "backend.php?op=feeds&subop=catchupAll";
278 notify_progress("Marking all feeds as read...");
280 debug("catchupAllFeeds Q=" + query_str
);
282 new Ajax
.Request(query_str
, {
283 onComplete: function(transport
) {
284 feedlist_callback2(transport
);
292 function viewCurrentFeed(subop
) {
294 // if (getActiveFeedId()) {
295 if (getActiveFeedId() != undefined) {
296 viewfeed(getActiveFeedId(), subop
, activeFeedIsCat());
298 disableContainerChildren("headlinesToolbar", false, document
);
299 // viewfeed(-1, subop); // FIXME
301 return false; // block unneeded form submits
304 function viewfeed(feed
, subop
) {
305 var f
= window
.frames
["feeds-frame"];
306 f
.viewfeed(feed
, subop
);
310 if (getInitParam("bw_limit") == "1") return;
312 scheduleFeedUpdate(false);
314 var refresh_time
= getInitParam("feeds_frame_refresh");
316 if (!refresh_time
) refresh_time
= 600;
318 setTimeout("timeout()", refresh_time
*1000);
321 function resetSearch() {
322 var searchbox
= document
.getElementById("searchbox")
324 if (searchbox
.value
!= "" && getActiveFeedId()) {
325 searchbox
.value
= "";
326 viewfeed(getActiveFeedId(), "");
330 function searchCancel() {
336 viewCurrentFeed(0, "");
339 // if argument is undefined, current subtitle is not updated
340 // use blank string to clear subtitle
341 function updateTitle(s
) {
342 var tmp
= "Tiny Tiny RSS";
344 if (s
!= undefined) {
345 current_subtitle
= s
;
348 if (global_unread
> 0) {
349 tmp
= tmp
+ " (" + global_unread
+ ")";
352 if (current_subtitle
) {
353 tmp
= tmp
+ " - " + current_subtitle
;
356 if (active_title_text
.length
> 0) {
357 tmp
= tmp
+ " > " + active_title_text
;
360 document
.title
= tmp
;
363 function genericSanityCheck() {
365 // if (!Ajax.getTransport()) fatalError(1);
367 setCookie("ttrss_vf_test", "TEST");
369 if (getCookie("ttrss_vf_test") != "TEST") {
380 // this whole shebang is based on http://www.birnamdesigns.com/misc/busted2.html
382 if (arguments
.callee
.done
) return;
383 arguments
.callee
.done
= true;
387 disableContainerChildren("headlinesToolbar", true);
389 Form
.disable("main_toolbar_form");
391 if (!genericSanityCheck())
394 if (getURLParam('debug')) {
395 Element
.show("debug_output");
396 debug('debug mode activated');
399 var params
= "&ua=" + param_escape(navigator
.userAgent
);
401 loading_set_progress(30);
403 new Ajax
.Request("backend.php?op=rpc&subop=sanityCheck" + params
, {
404 onComplete: function(transport
) {
405 backend_sanity_check_callback(transport
);
409 exception_error("init", e
);
413 function resize_headlines(delta_x
, delta_y
) {
417 debug("resize_headlines: " + delta_x
+ ":" + delta_y
);
419 var h_frame
= document
.getElementById("headlines-frame");
420 var c_frame
= document
.getElementById("content-frame");
421 var f_frame
= document
.getElementById("footer");
422 var feeds_frame
= document
.getElementById("feeds-holder");
423 var resize_grab
= document
.getElementById("resize-grabber");
424 var resize_handle
= document
.getElementById("resize-handle");
426 if (!c_frame
|| !h_frame
) return;
428 if (feeds_frame
&& getInitParam("theme") == "compat") {
429 feeds_frame
.style
.bottom
= f_frame
.offsetHeight
+ "px";
432 if (getInitParam("theme") == "3pane") {
434 if (delta_x
!= undefined) {
435 if (c_frame
.offsetLeft
- delta_x
> feeds_frame
.offsetWidth
+ feeds_frame
.offsetLeft
+ 100 && c_frame
.offsetWidth
+ delta_x
> 100) {
436 hor_offset
= hor_offset
+ delta_x
;
440 debug("resize_headlines: HOR-mode: " + hor_offset
);
442 c_frame
.style
.width
= (400 + hor_offset
) + "px";
443 h_frame
.style
.right
= c_frame
.offsetWidth
- 1 + "px";
445 resize_grab
.style
.top
= (h_frame
.offsetTop
+ h_frame
.offsetHeight
- 60) + "px";
446 resize_grab
.style
.left
= (h_frame
.offsetLeft
+ h_frame
.offsetWidth
-
448 resize_grab
.style
.display
= "block";
450 resize_handle
.src
= "themes/3pane/images/resize_handle_vert.png";
451 resize_handle
.style
.paddingTop
= (resize_grab
.offsetHeight
/ 2 - 7) + "px";
455 if (delta_y
!= undefined) {
456 if (c_frame
.offsetHeight
+ delta_y
> 100 && h_frame
.offsetHeight
- delta_y
> 100) {
457 ver_offset
= ver_offset
+ delta_y
;
461 debug("resize_headlines: VER-mode: " + ver_offset
);
463 h_frame
.style
.height
= (300 - ver_offset
) + "px";
465 c_frame
.style
.top
= (h_frame
.offsetTop
+ h_frame
.offsetHeight
+ 0) + "px";
466 h_frame
.style
.height
= h_frame
.offsetHeight
+ "px";
470 if (getInitParam("theme") == "graycube") {
474 if (getInitParam("theme") == "graycube" || getInitParam("theme") == "compat") {
475 resize_handle
.src
= "themes/graycube/images/resize_handle_horiz.png";
478 /* resize_grab.style.top = (h_frame.offsetTop + h_frame.offsetHeight -
480 resize_grab.style.display = "block"; */
484 if (getInitParam("cookie_lifetime") != 0) {
485 setCookie("ttrss_offset_ver", ver_offset
,
486 getInitParam("cookie_lifetime"));
487 setCookie("ttrss_offset_hor", hor_offset
,
488 getInitParam("cookie_lifetime"));
490 setCookie("ttrss_offset_ver", ver_offset
);
491 setCookie("ttrss_offset_hor", hor_offset
);
495 exception_error("resize_headlines", e
);
500 function init_second_stage() {
504 delCookie("ttrss_vf_test");
506 // document.onresize = resize_headlines;
508 var toolbar
= document
.forms
["main_toolbar_form"];
510 dropboxSelect(toolbar
.view_mode
, getInitParam("default_view_mode"));
511 dropboxSelect(toolbar
.limit
, getInitParam("default_view_limit"));
512 dropboxSelect(toolbar
.order_by
, getInitParam("default_view_order_by"));
514 daemon_enabled
= getInitParam("daemon_enabled") == 1;
515 daemon_refresh_only
= getInitParam("daemon_refresh_only") == 1;
516 feeds_sort_by_unread
= getInitParam("feeds_sort_by_unread") == 1;
518 var fl
= cache_find_param("FEEDLIST", getInitParam("num_feeds"));
522 if (document
.getElementById("feedList")) {
525 setTimeout('updateFeedList(false, false)', 50);
528 setTimeout('updateFeedList(false, false)', 50);
531 debug("second stage ok");
533 loading_set_progress(60);
535 ver_offset
= parseInt(getCookie("ttrss_offset_ver"));
536 hor_offset
= parseInt(getCookie("ttrss_offset_hor"));
538 debug("got offsets from cookies: ver " + ver_offset
+ " hor " + hor_offset
);
542 if (isNaN(hor_offset
)) hor_offset
= 0;
543 if (isNaN(ver_offset
)) ver_offset
= 0;
545 debug("offsets from cookies [x:y]: " + hor_offset
+ ":" + ver_offset
);
550 exception_error("init_second_stage", e
);
554 function quickMenuChange() {
555 var chooser
= document
.getElementById("quickMenuChooser");
556 var opid
= chooser
[chooser
.selectedIndex
].value
;
558 chooser
.selectedIndex
= 0;
562 function quickMenuGo(opid
) {
565 if (opid
== "qmcPrefs") {
569 if (opid
== "qmcSearch") {
570 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat());
574 if (opid
== "qmcAddFeed") {
575 displayDlg("quickAddFeed");
579 if (opid
== "qmcEditFeed") {
580 editFeedDlg(getActiveFeedId());
583 if (opid
== "qmcRemoveFeed") {
584 var actid
= getActiveFeedId();
586 if (activeFeedIsCat()) {
587 alert(__("You can't unsubscribe from the category."));
592 alert(__("Please select some feed first."));
596 var fn
= getFeedName(actid
);
598 var pr
= __("Unsubscribe from %s?").replace("%s", fn
);
601 unsubscribeFeed(actid
);
607 if (opid
== "qmcClearFeed") {
608 var actid
= getActiveFeedId();
611 alert(__("Please select some feed first."));
615 if (activeFeedIsCat() || actid
< 0) {
616 alert(__("You can't clear this type of feed."));
620 var fn
= getFeedName(actid
);
622 var pr
= __("Erase all non-starred articles in %s?").replace("%s", fn
);
625 clearFeedArticles(actid
);
632 if (opid
== "qmcUpdateFeeds") {
633 scheduleFeedUpdate(true);
637 if (opid
== "qmcCatchupAll") {
642 if (opid
== "qmcShowOnlyUnread") {
647 if (opid
== "qmcAddFilter") {
648 displayDlg("quickAddFilter", getActiveFeedId());
651 if (opid
== "qmcAddLabel") {
655 if (opid
== "qmcRescoreFeed") {
656 rescoreCurrentFeed();
659 if (opid
== "qmcHKhelp") {
660 //Element.show("hotkey_help_overlay");
661 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
664 if (opid
== "qmcResetUI") {
670 if (opid
== "qmcDownload") {
671 displayDlg("offlineDownload");
675 if (opid
== "qmcResetCats") {
677 if (confirm(__("Reset category order?"))) {
679 var query
= "backend.php?op=feeds&subop=catsortreset";
681 notify_progress("Loading, please wait...", true);
683 new Ajax
.Request(query
, {
684 onComplete: function(transport
) {
685 window
.setTimeout('updateFeedList(false, false)', 50);
691 exception_error("quickMenuGo", e
);
695 function unsubscribeFeed(feed_id
, title
) {
698 var msg
= __("Unsubscribe from %s?").replace("%s", title
);
700 if (title
== undefined || confirm(msg
)) {
701 notify_progress("Removing feed...");
703 var query
= "backend.php?op=pref-feeds&quiet=1&subop=remove&ids=" + feed_id
;
705 new Ajax
.Request(query
, {
706 onComplete: function(transport
) {
707 dlg_frefresh_callback(transport
, feed_id
);
715 function updateFeedTitle(t
) {
716 active_title_text
= t
;
720 function toggleDispRead() {
723 var hide_read_feeds
= (getInitParam("hide_read_feeds") == "1");
725 hide_read_feeds
= !hide_read_feeds
;
727 debug("toggle_disp_read => " + hide_read_feeds
);
729 hideOrShowFeeds(hide_read_feeds
);
731 storeInitParam("hide_read_feeds", hide_read_feeds
, true);
734 exception_error("toggleDispRead", e
);
738 function parse_runtime_info(elem
) {
740 debug("parse_runtime_info: elem is null, aborting");
744 var param
= elem
.firstChild
;
746 debug("parse_runtime_info: " + param
);
749 var k
= param
.getAttribute("key");
750 var v
= param
.getAttribute("value");
752 debug("RI: " + k
+ " => " + v
);
754 if (k
== "num_feeds") {
758 if (k
== "new_version_available") {
759 var icon
= document
.getElementById("newVersionIcon");
762 icon
.style
.display
= "inline";
764 icon
.style
.display
= "none";
771 if (k
== "daemon_is_running" && v
!= 1) {
772 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
776 if (k
== "daemon_stamp_ok" && v
!= 1) {
777 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
785 /* var w = document.getElementById("noDaemonWarning");
788 if (k == "daemon_is_running" && v != 1) {
789 w.style.display = "block";
791 w.style.display = "none";
794 param
= param
.nextSibling
;
798 function catchupCurrentFeed() {
800 var fn
= getFeedName(getActiveFeedId(), activeFeedIsCat());
802 var str
= __("Mark all articles in %s as read?").replace("%s", fn
);
804 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
805 return viewCurrentFeed('MarkAllRead')
809 function catchupFeedInGroup(id
) {
813 var title
= getFeedName(id
);
815 var str
= __("Mark all articles in %s as read?").replace("%s", title
);
817 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
818 return viewCurrentFeed('MarkAllReadGR:' + id
)
822 exception_error("catchupFeedInGroup", e
);
826 function editFeedDlg(feed
) {
830 alert(__("Please select some feed first."));
834 if ((feed
<= 0) || activeFeedIsCat() || tagsAreDisplayed()) {
835 alert(__("You can't edit this kind of feed."));
842 query
= "backend.php?op=pref-feeds&subop=editfeed&id=" + param_escape(feed
);
844 query
= "backend.php?op=pref-labels&subop=edit&id=" + param_escape(-feed
-11);
849 new Ajax
.Request(query
, {
850 onComplete: function(transport
) {
851 infobox_callback2(transport
);
855 exception_error("editFeedDlg", e
);
859 /* this functions duplicate those of prefs.js feed editor, with
860 some differences because there is no feedlist */
862 function feedEditCancel() {
867 function feedEditSave() {
871 // FIXME: add parameter validation
873 var query
= Form
.serialize("edit_feed_form");
875 notify_progress("Saving feed...");
877 new Ajax
.Request("backend.php", {
879 onComplete: function(transport
) {
880 dlg_frefresh_callback(transport
);
889 exception_error("feedEditSave (main)", e
);
893 function clearFeedArticles(feed_id
) {
895 notify_progress("Clearing feed...");
897 var query
= "backend.php?op=pref-feeds&quiet=1&subop=clear&id=" + feed_id
;
899 new Ajax
.Request(query
, {
900 onComplete: function(transport
) {
901 dlg_frefresh_callback(transport
, feed_id
);
907 function collapse_feedlist() {
909 debug("toggle_feedlist");
911 var theme
= getInitParam("theme");
912 if (theme
!= "" && theme
!= "compact" && theme
!= "graycube" &&
913 theme
!= "compat") return;
915 var fl
= document
.getElementById("feeds-holder");
916 var fh
= document
.getElementById("headlines-frame");
917 var fc
= document
.getElementById("content-frame");
918 var ft
= document
.getElementById("toolbar");
919 var ff
= document
.getElementById("footer");
920 var fhdr
= document
.getElementById("header");
921 var fbtn
= document
.getElementById("collapse_feeds_btn");
923 if (!Element
.visible(fl
)) {
927 if (theme
!= "graycube") {
929 fh
.style
.left
= fl
.offsetWidth
+ "px";
930 ft
.style
.left
= fl
.offsetWidth
+ "px";
931 if (fc
) fc
.style
.left
= fl
.offsetWidth
+ "px";
932 if (ff
&& theme
!= "compat") ff
.style
.left
= (fl
.offsetWidth
-1) + "px";
934 if (theme
== "compact") fhdr
.style
.left
= (fl
.offsetWidth
+ 10) + "px";
936 fh
.style
.left
= fl
.offsetWidth
+ 40 + "px";
937 ft
.style
.left
= fl
.offsetWidth
+ 40 +"px";
938 if (fc
) fc
.style
.left
= fl
.offsetWidth
+ 40 + "px";
941 setCookie("ttrss_vf_fclps", "0");
947 if (theme
!= "graycube") {
949 fh
.style
.left
= "0px";
950 ft
.style
.left
= "0px";
951 if (fc
) fc
.style
.left
= "0px";
952 if (ff
) ff
.style
.left
= "0px";
954 if (theme
== "compact") fhdr
.style
.left
= "10px";
957 fh
.style
.left
= "20px";
958 ft
.style
.left
= "20px";
959 if (fc
) fc
.style
.left
= "20px";
963 setCookie("ttrss_vf_fclps", "1");
966 exception_error("toggle_feedlist", e
);
970 function viewModeChanged() {
972 return viewCurrentFeed(0, '')
975 function viewLimitChanged() {
977 return viewCurrentFeed(0, '')
980 /* function adjustArticleScore(id, score) {
983 var pr = prompt(__("Assign score to article:"), score);
985 if (pr != undefined) {
986 var query = "backend.php?op=rpc&subop=setScore&id=" + id + "&score=" + pr;
988 new Ajax.Request(query, {
989 onComplete: function(transport) {
995 exception_error("adjustArticleScore", e);
999 function rescoreCurrentFeed() {
1001 var actid
= getActiveFeedId();
1003 if (activeFeedIsCat() || actid
< 0 || tagsAreDisplayed()) {
1004 alert(__("You can't rescore this kind of feed."));
1009 alert(__("Please select some feed first."));
1013 var fn
= getFeedName(actid
);
1014 var pr
= __("Rescore articles in %s?").replace("%s", fn
);
1017 notify_progress("Rescoring articles...");
1019 var query
= "backend.php?op=pref-feeds&subop=rescore&quiet=1&ids=" + actid
;
1021 new Ajax
.Request(query
, {
1022 onComplete: function(transport
) {
1028 function hotkey_handler(e
) {
1033 var shift_key
= false;
1035 var feedlist
= document
.getElementById('feedList');
1038 shift_key
= e
.shiftKey
;
1044 keycode
= window
.event
.keyCode
;
1049 var keychar
= String
.fromCharCode(keycode
);
1051 if (keycode
== 27) { // escape
1052 if (Element
.visible("hotkey_help_overlay")) {
1053 Element
.hide("hotkey_help_overlay");
1055 hotkey_prefix
= false;
1059 if (!hotkeys_enabled
) {
1060 debug("hotkeys disabled");
1064 if (keycode
== 16) return; // ignore lone shift
1066 if ((keycode
== 70 || keycode
== 67 || keycode
== 71)
1067 && !hotkey_prefix
) {
1069 hotkey_prefix
= keycode
;
1070 debug("KP: PREFIX=" + keycode
+ " CHAR=" + keychar
);
1074 if (Element
.visible("hotkey_help_overlay")) {
1075 Element
.hide("hotkey_help_overlay");
1078 /* Global hotkeys */
1080 if (!hotkey_prefix
) {
1082 if (keycode
== 68 && shift_key
) { // d
1083 if (!Element
.visible("debug_output")) {
1084 Element
.show("debug_output");
1085 debug('debug mode activated');
1087 Element
.hide("debug_output");
1093 if ((keycode
== 191 || keychar
== '?') && shift_key
) { // ?
1094 if (!Element
.visible("hotkey_help_overlay")) {
1095 //Element.show("hotkey_help_overlay");
1096 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
1098 Element
.hide("hotkey_help_overlay");
1103 if (keycode
== 191 || keychar
== '/') { // /
1104 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat());
1108 if (keycode
== 82 && shift_key
) { // R
1109 scheduleFeedUpdate(true);
1113 if (keycode
== 74) { // j
1114 var feed
= getActiveFeedId();
1115 var new_feed
= getRelativeFeedId2(feed
, activeFeedIsCat(), 'prev');
1116 // alert(feed + " IC: " + activeFeedIsCat() + " => " + new_feed);
1118 var is_cat
= new_feed
.match("CAT:");
1120 new_feed
= new_feed
.replace("CAT:", "");
1121 viewCategory(new_feed
);
1123 viewfeed(new_feed
, '', false);
1129 if (keycode
== 75) { // k
1130 var feed
= getActiveFeedId();
1131 var new_feed
= getRelativeFeedId2(feed
, activeFeedIsCat(), 'next');
1132 // alert(feed + " IC: " + activeFeedIsCat() + " => " + new_feed);
1134 var is_cat
= new_feed
.match("CAT:");
1135 if (is_cat
== "CAT:") {
1136 new_feed
= new_feed
.replace("CAT:", "");
1137 viewCategory(new_feed
);
1139 viewfeed(new_feed
, '', false);
1145 if (shift_key
&& keycode
== 40) { // shift-down
1146 catchupRelativeToArticle(1);
1150 if (shift_key
&& keycode
== 38) { // shift-up
1151 catchupRelativeToArticle(0);
1155 if (shift_key
&& keycode
== 78) { // N
1160 if (shift_key
&& keycode
== 80) { // P
1166 if (keycode
== 78 || keycode
== 40) { // n, down
1167 if (typeof moveToPost
!= 'undefined') {
1173 if (keycode
== 80 || keycode
== 38) { // p, up
1174 if (typeof moveToPost
!= 'undefined') {
1180 if (keycode
== 83 && shift_key
) { // S
1181 var id
= getActiveArticleId();
1188 if (keycode
== 83) { // s
1189 var id
= getActiveArticleId();
1197 if (keycode
== 85) { // u
1198 var id
= getActiveArticleId();
1205 if (keycode
== 84 && shift_key
) { // T
1206 var id
= getActiveArticleId();
1208 editArticleTags(id
, getActiveFeedId(), isCdmMode());
1213 if (keycode
== 9) { // tab
1214 var id
= getArticleUnderPointer();
1216 var cb
= document
.getElementById("RCHK-" + id
);
1219 cb
.checked
= !cb
.checked
;
1220 toggleSelectRowById(cb
, "RROW-" + id
);
1226 if (keycode
== 79) { // o
1227 if (getActiveArticleId()) {
1228 openArticleInNewWindow(getActiveArticleId());
1233 if (keycode
== 81 && shift_key
) { // Q
1234 if (typeof catchupAllFeeds
!= 'undefined') {
1240 if (keycode
== 88) { // x
1241 if (activeFeedIsCat()) {
1242 toggleCollapseCat(getActiveFeedId());
1249 if (hotkey_prefix
== 70) { // f
1251 hotkey_prefix
= false;
1253 if (keycode
== 81) { // q
1254 if (getActiveFeedId()) {
1255 catchupCurrentFeed();
1260 if (keycode
== 82) { // r
1261 if (getActiveFeedId()) {
1262 viewfeed(getActiveFeedId(), "ForceUpdate", activeFeedIsCat());
1267 if (keycode
== 65) { // a
1272 if (keycode
== 85 && shift_key
) { // U
1273 scheduleFeedUpdate(true);
1277 if (keycode
== 85) { // u
1278 if (getActiveFeedId()) {
1279 viewfeed(getActiveFeedId(), "ForceUpdate");
1284 if (keycode
== 69) { // e
1285 editFeedDlg(getActiveFeedId());
1289 if (keycode
== 83) { // s
1290 displayDlg("quickAddFeed");
1294 if (keycode
== 67 && shift_key
) { // C
1295 if (typeof catchupAllFeeds
!= 'undefined') {
1301 if (keycode
== 67) { // c
1302 if (getActiveFeedId()) {
1303 catchupCurrentFeed();
1308 if (keycode
== 68 && shift_key
) { // D
1309 initiate_offline_download();
1313 if (keycode
== 68) { // d
1314 displayDlg("offlineDownload");
1318 if (keycode
== 87) { // w
1319 feeds_sort_by_unread
= !feeds_sort_by_unread
;
1320 return resort_feedlist();
1323 if (keycode
== 72) { // h
1324 hideReadHeadlines();
1332 if (hotkey_prefix
== 67) { // c
1333 hotkey_prefix
= false;
1335 if (keycode
== 70) { // f
1336 displayDlg("quickAddFilter", getActiveFeedId());
1340 if (keycode
== 76) { // l
1345 if (keycode
== 83) { // s
1346 if (typeof collapse_feedlist
!= 'undefined') {
1347 collapse_feedlist();
1352 if (keycode
== 77) { // m
1353 feedlist_sortable_enabled
= !feedlist_sortable_enabled
;
1354 if (feedlist_sortable_enabled
) {
1355 notify_info("Category reordering enabled");
1356 toggle_sortable_feedlist(true);
1358 notify_info("Category reordering disabled");
1359 toggle_sortable_feedlist(false);
1363 if (keycode
== 78) { // n
1364 catchupRelativeToArticle(1);
1368 if (keycode
== 80) { // p
1369 catchupRelativeToArticle(0);
1378 if (hotkey_prefix
== 71) { // g
1380 hotkey_prefix
= false;
1383 if (keycode
== 65) { // a
1388 if (keycode
== 83) { // s
1393 if (keycode
== 80 && shift_key
) { // P
1398 if (keycode
== 80) { // p
1403 if (keycode
== 70) { // f
1408 if (keycode
== 84 && shift_key
) { // T
1416 if (hotkey_prefix
== 224 || hotkey_prefix
== 91) { // f
1417 hotkey_prefix
= false;
1421 if (hotkey_prefix
) {
1422 debug("KP: PREFIX=" + hotkey_prefix
+ " CODE=" + keycode
+ " CHAR=" + keychar
);
1424 debug("KP: CODE=" + keycode
+ " CHAR=" + keychar
);
1429 exception_error("hotkey_handler", e
);
1433 function feedsSortByUnread() {
1434 return feeds_sort_by_unread
;
1437 function addLabel() {
1441 var caption
= prompt(__("Please enter label caption:"), "");
1443 if (caption
!= undefined) {
1445 if (caption
== "") {
1446 alert(__("Can't create label: missing caption."));
1450 var query
= "backend.php?op=pref-labels&subop=add&caption=" +
1451 param_escape(caption
);
1453 notify_progress("Loading, please wait...", true);
1455 new Ajax
.Request(query
, {
1456 onComplete: function(transport
) {
1463 exception_error("addLabel", e
);
1467 function visitOfficialSite() {
1468 window
.open("http://tt-rss.org/");
1472 function feedBrowserSubscribe() {
1475 var selected
= getSelectedFeedsFromBrowser();
1477 if (selected
.length
> 0) {
1480 notify_progress("Loading, please wait...", true);
1482 var query
= "backend.php?op=pref-feeds&subop=massSubscribe&ids="+
1483 param_escape(selected
.toString());
1485 new Ajax
.Request(query
, {
1486 onComplete: function(transport
) {
1491 alert(__("No feeds are selected."));
1495 exception_error("feedBrowserSubscribe", e
);
1499 function init_gears() {
1502 if (window
.google
&& google
.gears
) {
1503 localServer
= google
.gears
.factory
.create("beta.localserver");
1504 store
= localServer
.createManagedStore("tt-rss");
1505 db
= google
.gears
.factory
.create('beta.database');
1508 db
.execute("CREATE TABLE IF NOT EXISTS cache (id text, article text, param text, added text)");
1510 db
.execute("CREATE TABLE if not exists feeds (id integer, title text, has_icon integer)");
1512 db
.execute("CREATE TABLE if not exists articles (id integer, feed_id integer, title text, link text, guid text, updated text, content text, tags text, unread text, marked text)");
1514 var qmcDownload
= document
.getElementById("qmcDownload");
1515 if (qmcDownload
) Element
.show(qmcDownload
);
1522 exception_error("init_gears", e
);
1526 function init_offline() {
1528 offline_mode
= true;
1530 render_offline_feedlist();
1534 exception_error("init_offline", e
);
1538 function offline_download_parse(stage
, transport
) {
1540 if (transport
.responseXML
) {
1544 var feeds
= transport
.responseXML
.getElementsByTagName("feed");
1546 if (feeds
.length
> 0) {
1547 db
.execute("DELETE FROM feeds");
1550 for (var i
= 0; i
< feeds
.length
; i
++) {
1551 var id
= feeds
[i
].getAttribute("id");
1552 var has_icon
= feeds
[i
].getAttribute("has_icon");
1553 var title
= feeds
[i
].firstChild
.nodeValue
;
1555 db
.execute("INSERT INTO feeds (id,title,has_icon)"+
1557 [id
, title
, has_icon
]);
1560 window
.setTimeout("initiate_offline_download("+(stage
+1)+")", 50);
1563 var articles
= transport
.responseXML
.getElementsByTagName("article");
1565 var articles_found
= 0;
1567 for (var i
= 0; i
< articles
.length
; i
++) {
1568 var a
= eval("("+articles
[i
].firstChild
.nodeValue
+")");
1571 db
.execute("DELETE FROM articles WHERE id = ?", [a
.id
]);
1572 db
.execute("INSERT INTO articles "+
1573 "(id, feed_id, title, link, guid, updated, content, unread, marked) "+
1574 "VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)",
1575 [a
.id
, a
.feed_id
, a
.title
, a
.link
, a
.guid
, a
.updated
,
1576 a
.content
, a
.unread
, a
.marked
]);
1581 if (articles_found
> 0) {
1582 window
.setTimeout("initiate_offline_download("+(stage
+1)+")", 50);
1586 notify_info("All done.");
1590 exception_error("offline_download_parse", e
);
1594 function initiate_offline_download(stage
) {
1597 if (!stage
) stage
= 0;
1599 notify_progress("Loading, please wait... (" + stage
+")", true);
1601 var query
= "backend.php?op=rpc&subop=download&stage=" + stage
;
1603 var rs
= db
.execute("SELECT MAX(id) FROM articles");
1604 if (rs
.isValidRow()) {
1605 query
= query
+ "&cid=" + rs
.field(0);
1608 if (document
.getElementById("download_ops_form")) {
1609 query
= query
+ "&" + Form
.serialize("download_ops_form");
1612 new Ajax
.Request(query
, {
1613 onComplete: function(transport
) {
1614 offline_download_parse(stage
, transport
);
1618 exception_error("initiate_offline_download", e
);