1 var SCHEMA_VERSION
= 1;
5 var display_tags
= false;
6 var global_unread
= -1;
7 var active_title_text
= "";
8 var current_subtitle
= "";
9 var daemon_enabled
= false;
10 var daemon_refresh_only
= false;
11 //var _qfd_deleted_feed = 0;
12 var firsttime_update
= true;
13 var _active_feed_id
= 0;
14 var _active_feed_is_cat
= false;
15 var number_of_feeds
= 0;
16 var sanity_check_done
= false;
17 var _hfd_scrolltop
= 0;
18 var hotkey_prefix
= false;
19 var init_params
= new Object();
22 var feeds_sort_by_unread
= false;
23 var feedlist_sortable_enabled
= false;
24 var offline_mode
= false;
26 var localServer
= false;
28 var download_progress_last
= 0;
29 var offline_dl_max_id
= 0;
31 function activeFeedIsCat() {
32 return _active_feed_is_cat
;
35 function getActiveFeedId() {
36 // return getCookie("ttrss_vf_actfeed");
38 debug("gAFID: " + _active_feed_id
);
39 return _active_feed_id
;
41 exception_error("getActiveFeedId", e
);
45 function setActiveFeedId(id
, is_cat
) {
46 // return setCookie("ttrss_vf_actfeed", id);
48 debug("sAFID(" + id
+ ", " + is_cat
+ ")");
51 if (is_cat
!= undefined) {
52 _active_feed_is_cat
= is_cat
;
56 exception_error("setActiveFeedId", e
);
61 function isFeedlistSortable() {
62 return feedlist_sortable_enabled
;
65 function tagsAreDisplayed() {
69 function toggleTags(show_all
) {
73 debug("toggleTags: " + show_all
+ "; " + display_tags
);
75 var p
= document
.getElementById("dispSwitchPrompt");
77 if (!show_all
&& !display_tags
) {
78 displayDlg("printTagCloud");
79 } else if (show_all
) {
82 p
.innerHTML
= __("display feeds");
83 notify_progress("Loading, please wait...", true);
85 } else if (display_tags
) {
87 p
.innerHTML
= __("tag cloud");
88 notify_progress("Loading, please wait...", true);
93 exception_error("toggleTags", e
);
97 function dlg_frefresh_callback(transport
, deleted_feed
) {
98 if (getActiveFeedId() == deleted_feed
) {
99 var h
= document
.getElementById("headlines-frame");
101 h
.innerHTML
= "<div class='whiteBox'>" + __('No feed selected.') + "</div>";
105 setTimeout('updateFeedList(false, false)', 50);
109 function refetch_callback2(transport
) {
112 var date
= new Date();
114 parse_counters_reply(transport
, true);
116 debug("refetch_callback2: done");
118 /* if (!daemon_enabled && !daemon_refresh_only) {
119 notify_info("All feeds updated.");
125 exception_error("refetch_callback", e
);
130 function backend_sanity_check_callback(transport
) {
134 if (sanity_check_done
) {
135 fatalError(11, "Sanity check request received twice. This can indicate "+
136 "presence of Firebug or some other disrupting extension. "+
137 "Please disable it and try again.");
141 if (!transport
.responseXML
) {
142 if (!window
.google
&& !google
.gears
) {
143 fatalError(3, "Sanity check: Received reply is not XML", transport
.responseText
);
150 if (getURLParam("offline")) {
151 return init_offline();
154 var reply
= transport
.responseXML
.firstChild
.firstChild
;
157 fatalError(3, "Sanity check: invalid RPC reply", transport
.responseText
);
161 var error_code
= reply
.getAttribute("error-code");
163 if (error_code
&& error_code
!= 0) {
164 return fatalError(error_code
, reply
.getAttribute("error-msg"));
167 debug("sanity check ok");
169 var params
= reply
.nextSibling
;
172 debug('reading init-params...');
173 var param
= params
.firstChild
;
176 var k
= param
.getAttribute("key");
177 var v
= param
.getAttribute("value");
178 debug(k
+ " => " + v
);
180 param
= param
.nextSibling
;
184 sanity_check_done
= true;
189 exception_error("backend_sanity_check_callback", e
, transport
);
193 function scheduleFeedUpdate(force
) {
195 debug("in scheduleFeedUpdate");
197 /* if (!daemon_enabled && !daemon_refresh_only) {
198 notify_progress("Updating feeds...", true);
201 var query_str
= "backend.php?op=rpc&subop=";
204 query_str
= query_str
+ "forceUpdateAllFeeds";
206 query_str
= query_str
+ "updateAllFeeds";
211 if (firsttime_update
&& !navigator
.userAgent
.match("Opera")) {
212 firsttime_update
= false;
222 query_str
= query_str
+ "&omode=" + omode
;
223 query_str
= query_str
+ "&uctr=" + global_unread
;
225 var date
= new Date();
226 var timestamp
= Math
.round(date
.getTime() / 1000);
227 query_str
= query_str
+ "&ts=" + timestamp
229 debug("REFETCH query: " + query_str
);
231 new Ajax
.Request(query_str
, {
232 onComplete: function(transport
) {
233 refetch_callback2(transport
);
237 function updateFeedList(silent
, fetch
) {
239 // if (silent != true) {
240 // notify("Loading feed list...");
243 debug("<b>updateFeedList</b>");
245 var query_str
= "backend.php?op=feeds";
248 query_str
= query_str
+ "&tags=1";
251 if (getActiveFeedId() && !activeFeedIsCat()) {
252 query_str
= query_str
+ "&actid=" + getActiveFeedId();
255 var date
= new Date();
256 var timestamp
= Math
.round(date
.getTime() / 1000);
257 query_str
= query_str
+ "&ts=" + timestamp
259 if (fetch
) query_str
= query_str
+ "&fetch=yes";
261 // var feeds_frame = document.getElementById("feeds-frame");
262 // feeds_frame.src = query_str;
264 debug("updateFeedList Q=" + query_str
);
266 new Ajax
.Request(query_str
, {
267 onComplete: function(transport
) {
268 feedlist_callback2(transport
);
273 function catchupAllFeeds() {
275 var str
= __("Mark all articles as read?");
277 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
279 var query_str
= "backend.php?op=feeds&subop=catchupAll";
281 notify_progress("Marking all feeds as read...");
283 debug("catchupAllFeeds Q=" + query_str
);
285 new Ajax
.Request(query_str
, {
286 onComplete: function(transport
) {
287 feedlist_callback2(transport
);
295 function viewCurrentFeed(subop
) {
297 // if (getActiveFeedId()) {
298 if (getActiveFeedId() != undefined) {
299 viewfeed(getActiveFeedId(), subop
, activeFeedIsCat());
301 disableContainerChildren("headlinesToolbar", false, document
);
302 // viewfeed(-1, subop); // FIXME
304 return false; // block unneeded form submits
307 function viewfeed(feed
, subop
) {
308 var f
= window
.frames
["feeds-frame"];
309 f
.viewfeed(feed
, subop
);
313 if (getInitParam("bw_limit") == "1") return;
315 scheduleFeedUpdate(false);
317 var refresh_time
= getInitParam("feeds_frame_refresh");
319 if (!refresh_time
) refresh_time
= 600;
321 setTimeout("timeout()", refresh_time
*1000);
324 function resetSearch() {
325 var searchbox
= document
.getElementById("searchbox")
327 if (searchbox
.value
!= "" && getActiveFeedId()) {
328 searchbox
.value
= "";
329 viewfeed(getActiveFeedId(), "");
333 function searchCancel() {
339 viewCurrentFeed(0, "");
342 // if argument is undefined, current subtitle is not updated
343 // use blank string to clear subtitle
344 function updateTitle(s
) {
345 var tmp
= "Tiny Tiny RSS";
347 if (s
!= undefined) {
348 current_subtitle
= s
;
351 if (global_unread
> 0) {
352 tmp
= tmp
+ " (" + global_unread
+ ")";
355 if (current_subtitle
) {
356 tmp
= tmp
+ " - " + current_subtitle
;
359 if (active_title_text
.length
> 0) {
360 tmp
= tmp
+ " > " + active_title_text
;
363 document
.title
= tmp
;
366 function genericSanityCheck() {
368 // if (!Ajax.getTransport()) fatalError(1);
370 setCookie("ttrss_vf_test", "TEST");
372 if (getCookie("ttrss_vf_test") != "TEST") {
383 // this whole shebang is based on http://www.birnamdesigns.com/misc/busted2.html
385 if (arguments
.callee
.done
) return;
386 arguments
.callee
.done
= true;
390 disableContainerChildren("headlinesToolbar", true);
392 Form
.disable("main_toolbar_form");
394 if (!genericSanityCheck())
397 if (getURLParam('debug')) {
398 Element
.show("debug_output");
399 debug('debug mode activated');
402 var params
= "&ua=" + param_escape(navigator
.userAgent
);
404 loading_set_progress(30);
406 new Ajax
.Request("backend.php?op=rpc&subop=sanityCheck" + params
, {
407 onComplete: function(transport
) {
408 backend_sanity_check_callback(transport
);
412 exception_error("init", e
);
416 function resize_headlines(delta_x
, delta_y
) {
420 debug("resize_headlines: " + delta_x
+ ":" + delta_y
);
422 var h_frame
= document
.getElementById("headlines-frame");
423 var c_frame
= document
.getElementById("content-frame");
424 var f_frame
= document
.getElementById("footer");
425 var feeds_frame
= document
.getElementById("feeds-holder");
426 var resize_grab
= document
.getElementById("resize-grabber");
427 var resize_handle
= document
.getElementById("resize-handle");
429 if (!c_frame
|| !h_frame
) return;
431 if (feeds_frame
&& getInitParam("theme") == "compat") {
432 feeds_frame
.style
.bottom
= f_frame
.offsetHeight
+ "px";
435 if (getInitParam("theme") == "3pane") {
437 if (delta_x
!= undefined) {
438 if (c_frame
.offsetLeft
- delta_x
> feeds_frame
.offsetWidth
+ feeds_frame
.offsetLeft
+ 100 && c_frame
.offsetWidth
+ delta_x
> 100) {
439 hor_offset
= hor_offset
+ delta_x
;
443 debug("resize_headlines: HOR-mode: " + hor_offset
);
445 c_frame
.style
.width
= (400 + hor_offset
) + "px";
446 h_frame
.style
.right
= c_frame
.offsetWidth
- 1 + "px";
448 resize_grab
.style
.top
= (h_frame
.offsetTop
+ h_frame
.offsetHeight
- 60) + "px";
449 resize_grab
.style
.left
= (h_frame
.offsetLeft
+ h_frame
.offsetWidth
-
451 resize_grab
.style
.display
= "block";
453 resize_handle
.src
= "themes/3pane/images/resize_handle_vert.png";
454 resize_handle
.style
.paddingTop
= (resize_grab
.offsetHeight
/ 2 - 7) + "px";
458 if (delta_y
!= undefined) {
459 if (c_frame
.offsetHeight
+ delta_y
> 100 && h_frame
.offsetHeight
- delta_y
> 100) {
460 ver_offset
= ver_offset
+ delta_y
;
464 debug("resize_headlines: VER-mode: " + ver_offset
);
466 h_frame
.style
.height
= (300 - ver_offset
) + "px";
468 c_frame
.style
.top
= (h_frame
.offsetTop
+ h_frame
.offsetHeight
+ 0) + "px";
469 h_frame
.style
.height
= h_frame
.offsetHeight
+ "px";
473 if (getInitParam("theme") == "graycube") {
477 if (getInitParam("theme") == "graycube" || getInitParam("theme") == "compat") {
478 resize_handle
.src
= "themes/graycube/images/resize_handle_horiz.png";
481 /* resize_grab.style.top = (h_frame.offsetTop + h_frame.offsetHeight -
483 resize_grab.style.display = "block"; */
487 if (getInitParam("cookie_lifetime") != 0) {
488 setCookie("ttrss_offset_ver", ver_offset
,
489 getInitParam("cookie_lifetime"));
490 setCookie("ttrss_offset_hor", hor_offset
,
491 getInitParam("cookie_lifetime"));
493 setCookie("ttrss_offset_ver", ver_offset
);
494 setCookie("ttrss_offset_hor", hor_offset
);
498 exception_error("resize_headlines", e
);
503 function init_second_stage() {
507 delCookie("ttrss_vf_test");
509 // document.onresize = resize_headlines;
511 var toolbar
= document
.forms
["main_toolbar_form"];
513 dropboxSelect(toolbar
.view_mode
, getInitParam("default_view_mode"));
514 dropboxSelect(toolbar
.limit
, getInitParam("default_view_limit"));
515 dropboxSelect(toolbar
.order_by
, getInitParam("default_view_order_by"));
517 daemon_enabled
= getInitParam("daemon_enabled") == 1;
518 daemon_refresh_only
= getInitParam("daemon_refresh_only") == 1;
519 feeds_sort_by_unread
= getInitParam("feeds_sort_by_unread") == 1;
521 var fl
= cache_find_param("FEEDLIST", getInitParam("num_feeds"));
525 if (document
.getElementById("feedList")) {
528 setTimeout('updateFeedList(false, false)', 50);
531 setTimeout('updateFeedList(false, false)', 50);
534 debug("second stage ok");
536 loading_set_progress(60);
538 ver_offset
= parseInt(getCookie("ttrss_offset_ver"));
539 hor_offset
= parseInt(getCookie("ttrss_offset_hor"));
541 debug("got offsets from cookies: ver " + ver_offset
+ " hor " + hor_offset
);
545 if (isNaN(hor_offset
)) hor_offset
= 0;
546 if (isNaN(ver_offset
)) ver_offset
= 0;
548 debug("offsets from cookies [x:y]: " + hor_offset
+ ":" + ver_offset
);
553 exception_error("init_second_stage", e
);
557 function quickMenuChange() {
558 var chooser
= document
.getElementById("quickMenuChooser");
559 var opid
= chooser
[chooser
.selectedIndex
].value
;
561 chooser
.selectedIndex
= 0;
565 function quickMenuGo(opid
) {
568 if (opid
== "qmcPrefs") {
572 if (opid
== "qmcSearch") {
573 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat());
577 if (opid
== "qmcAddFeed") {
578 displayDlg("quickAddFeed");
582 if (opid
== "qmcEditFeed") {
583 editFeedDlg(getActiveFeedId());
586 if (opid
== "qmcRemoveFeed") {
587 var actid
= getActiveFeedId();
589 if (activeFeedIsCat()) {
590 alert(__("You can't unsubscribe from the category."));
595 alert(__("Please select some feed first."));
599 var fn
= getFeedName(actid
);
601 var pr
= __("Unsubscribe from %s?").replace("%s", fn
);
604 unsubscribeFeed(actid
);
610 if (opid
== "qmcClearFeed") {
611 var actid
= getActiveFeedId();
614 alert(__("Please select some feed first."));
618 if (activeFeedIsCat() || actid
< 0) {
619 alert(__("You can't clear this type of feed."));
623 var fn
= getFeedName(actid
);
625 var pr
= __("Erase all non-starred articles in %s?").replace("%s", fn
);
628 clearFeedArticles(actid
);
635 if (opid
== "qmcUpdateFeeds") {
636 scheduleFeedUpdate(true);
640 if (opid
== "qmcCatchupAll") {
645 if (opid
== "qmcShowOnlyUnread") {
650 if (opid
== "qmcAddFilter") {
651 displayDlg("quickAddFilter", getActiveFeedId());
654 if (opid
== "qmcAddLabel") {
658 if (opid
== "qmcRescoreFeed") {
659 rescoreCurrentFeed();
662 if (opid
== "qmcHKhelp") {
663 //Element.show("hotkey_help_overlay");
664 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
667 if (opid
== "qmcResetUI") {
673 if (opid
== "qmcDownload") {
674 displayDlg("offlineDownload");
678 if (opid
== "qmcResetCats") {
680 if (confirm(__("Reset category order?"))) {
682 var query
= "backend.php?op=feeds&subop=catsortreset";
684 notify_progress("Loading, please wait...", true);
686 new Ajax
.Request(query
, {
687 onComplete: function(transport
) {
688 window
.setTimeout('updateFeedList(false, false)', 50);
694 exception_error("quickMenuGo", e
);
698 function unsubscribeFeed(feed_id
, title
) {
701 var msg
= __("Unsubscribe from %s?").replace("%s", title
);
703 if (title
== undefined || confirm(msg
)) {
704 notify_progress("Removing feed...");
706 var query
= "backend.php?op=pref-feeds&quiet=1&subop=remove&ids=" + feed_id
;
708 new Ajax
.Request(query
, {
709 onComplete: function(transport
) {
710 dlg_frefresh_callback(transport
, feed_id
);
718 function updateFeedTitle(t
) {
719 active_title_text
= t
;
723 function toggleDispRead() {
726 var hide_read_feeds
= (getInitParam("hide_read_feeds") == "1");
728 hide_read_feeds
= !hide_read_feeds
;
730 debug("toggle_disp_read => " + hide_read_feeds
);
732 hideOrShowFeeds(hide_read_feeds
);
734 storeInitParam("hide_read_feeds", hide_read_feeds
, true);
737 exception_error("toggleDispRead", e
);
741 function parse_runtime_info(elem
) {
743 debug("parse_runtime_info: elem is null, aborting");
747 var param
= elem
.firstChild
;
749 debug("parse_runtime_info: " + param
);
752 var k
= param
.getAttribute("key");
753 var v
= param
.getAttribute("value");
755 debug("RI: " + k
+ " => " + v
);
757 if (k
== "num_feeds") {
761 if (k
== "new_version_available") {
762 var icon
= document
.getElementById("newVersionIcon");
765 icon
.style
.display
= "inline";
767 icon
.style
.display
= "none";
774 if (k
== "daemon_is_running" && v
!= 1) {
775 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
779 if (k
== "daemon_stamp_ok" && v
!= 1) {
780 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
788 /* var w = document.getElementById("noDaemonWarning");
791 if (k == "daemon_is_running" && v != 1) {
792 w.style.display = "block";
794 w.style.display = "none";
797 param
= param
.nextSibling
;
801 function catchupCurrentFeed() {
803 var fn
= getFeedName(getActiveFeedId(), activeFeedIsCat());
805 var str
= __("Mark all articles in %s as read?").replace("%s", fn
);
807 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
808 return viewCurrentFeed('MarkAllRead')
812 function catchupFeedInGroup(id
) {
816 var title
= getFeedName(id
);
818 var str
= __("Mark all articles in %s as read?").replace("%s", title
);
820 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
821 return viewCurrentFeed('MarkAllReadGR:' + id
)
825 exception_error("catchupFeedInGroup", e
);
829 function editFeedDlg(feed
) {
833 alert(__("Please select some feed first."));
837 if ((feed
<= 0) || activeFeedIsCat() || tagsAreDisplayed()) {
838 alert(__("You can't edit this kind of feed."));
845 query
= "backend.php?op=pref-feeds&subop=editfeed&id=" + param_escape(feed
);
847 query
= "backend.php?op=pref-labels&subop=edit&id=" + param_escape(-feed
-11);
852 new Ajax
.Request(query
, {
853 onComplete: function(transport
) {
854 infobox_callback2(transport
);
858 exception_error("editFeedDlg", e
);
862 /* this functions duplicate those of prefs.js feed editor, with
863 some differences because there is no feedlist */
865 function feedEditCancel() {
870 function feedEditSave() {
874 // FIXME: add parameter validation
876 var query
= Form
.serialize("edit_feed_form");
878 notify_progress("Saving feed...");
880 new Ajax
.Request("backend.php", {
882 onComplete: function(transport
) {
883 dlg_frefresh_callback(transport
);
892 exception_error("feedEditSave (main)", e
);
896 function clearFeedArticles(feed_id
) {
898 notify_progress("Clearing feed...");
900 var query
= "backend.php?op=pref-feeds&quiet=1&subop=clear&id=" + feed_id
;
902 new Ajax
.Request(query
, {
903 onComplete: function(transport
) {
904 dlg_frefresh_callback(transport
, feed_id
);
910 function collapse_feedlist() {
912 debug("toggle_feedlist");
914 var theme
= getInitParam("theme");
915 if (theme
!= "" && theme
!= "compact" && theme
!= "graycube" &&
916 theme
!= "compat") return;
918 var fl
= document
.getElementById("feeds-holder");
919 var fh
= document
.getElementById("headlines-frame");
920 var fc
= document
.getElementById("content-frame");
921 var ft
= document
.getElementById("toolbar");
922 var ff
= document
.getElementById("footer");
923 var fhdr
= document
.getElementById("header");
924 var fbtn
= document
.getElementById("collapse_feeds_btn");
926 if (!Element
.visible(fl
)) {
930 if (theme
!= "graycube") {
932 fh
.style
.left
= fl
.offsetWidth
+ "px";
933 ft
.style
.left
= fl
.offsetWidth
+ "px";
934 if (fc
) fc
.style
.left
= fl
.offsetWidth
+ "px";
935 if (ff
&& theme
!= "compat") ff
.style
.left
= (fl
.offsetWidth
-1) + "px";
937 if (theme
== "compact") fhdr
.style
.left
= (fl
.offsetWidth
+ 10) + "px";
939 fh
.style
.left
= fl
.offsetWidth
+ 40 + "px";
940 ft
.style
.left
= fl
.offsetWidth
+ 40 +"px";
941 if (fc
) fc
.style
.left
= fl
.offsetWidth
+ 40 + "px";
944 setCookie("ttrss_vf_fclps", "0");
950 if (theme
!= "graycube") {
952 fh
.style
.left
= "0px";
953 ft
.style
.left
= "0px";
954 if (fc
) fc
.style
.left
= "0px";
955 if (ff
) ff
.style
.left
= "0px";
957 if (theme
== "compact") fhdr
.style
.left
= "10px";
960 fh
.style
.left
= "20px";
961 ft
.style
.left
= "20px";
962 if (fc
) fc
.style
.left
= "20px";
966 setCookie("ttrss_vf_fclps", "1");
969 exception_error("toggle_feedlist", e
);
973 function viewModeChanged() {
975 return viewCurrentFeed(0, '')
978 function viewLimitChanged() {
980 return viewCurrentFeed(0, '')
983 /* function adjustArticleScore(id, score) {
986 var pr = prompt(__("Assign score to article:"), score);
988 if (pr != undefined) {
989 var query = "backend.php?op=rpc&subop=setScore&id=" + id + "&score=" + pr;
991 new Ajax.Request(query, {
992 onComplete: function(transport) {
998 exception_error("adjustArticleScore", e);
1002 function rescoreCurrentFeed() {
1004 var actid
= getActiveFeedId();
1006 if (activeFeedIsCat() || actid
< 0 || tagsAreDisplayed()) {
1007 alert(__("You can't rescore this kind of feed."));
1012 alert(__("Please select some feed first."));
1016 var fn
= getFeedName(actid
);
1017 var pr
= __("Rescore articles in %s?").replace("%s", fn
);
1020 notify_progress("Rescoring articles...");
1022 var query
= "backend.php?op=pref-feeds&subop=rescore&quiet=1&ids=" + actid
;
1024 new Ajax
.Request(query
, {
1025 onComplete: function(transport
) {
1031 function hotkey_handler(e
) {
1036 var shift_key
= false;
1038 var feedlist
= document
.getElementById('feedList');
1041 shift_key
= e
.shiftKey
;
1047 keycode
= window
.event
.keyCode
;
1052 var keychar
= String
.fromCharCode(keycode
);
1054 if (keycode
== 27) { // escape
1055 if (Element
.visible("hotkey_help_overlay")) {
1056 Element
.hide("hotkey_help_overlay");
1058 hotkey_prefix
= false;
1062 if (!hotkeys_enabled
) {
1063 debug("hotkeys disabled");
1067 if (keycode
== 16) return; // ignore lone shift
1069 if ((keycode
== 70 || keycode
== 67 || keycode
== 71)
1070 && !hotkey_prefix
) {
1072 hotkey_prefix
= keycode
;
1073 debug("KP: PREFIX=" + keycode
+ " CHAR=" + keychar
);
1077 if (Element
.visible("hotkey_help_overlay")) {
1078 Element
.hide("hotkey_help_overlay");
1081 /* Global hotkeys */
1083 if (!hotkey_prefix
) {
1085 if (keycode
== 68 && shift_key
) { // d
1086 if (!Element
.visible("debug_output")) {
1087 Element
.show("debug_output");
1088 debug('debug mode activated');
1090 Element
.hide("debug_output");
1096 if ((keycode
== 191 || keychar
== '?') && shift_key
) { // ?
1097 if (!Element
.visible("hotkey_help_overlay")) {
1098 //Element.show("hotkey_help_overlay");
1099 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
1101 Element
.hide("hotkey_help_overlay");
1106 if (keycode
== 191 || keychar
== '/') { // /
1107 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat());
1111 if (keycode
== 82 && shift_key
) { // R
1112 scheduleFeedUpdate(true);
1116 if (keycode
== 74) { // j
1117 var feed
= getActiveFeedId();
1118 var new_feed
= getRelativeFeedId2(feed
, activeFeedIsCat(), 'prev');
1119 // alert(feed + " IC: " + activeFeedIsCat() + " => " + new_feed);
1121 var is_cat
= new_feed
.match("CAT:");
1123 new_feed
= new_feed
.replace("CAT:", "");
1124 viewCategory(new_feed
);
1126 viewfeed(new_feed
, '', false);
1132 if (keycode
== 75) { // k
1133 var feed
= getActiveFeedId();
1134 var new_feed
= getRelativeFeedId2(feed
, activeFeedIsCat(), 'next');
1135 // alert(feed + " IC: " + activeFeedIsCat() + " => " + new_feed);
1137 var is_cat
= new_feed
.match("CAT:");
1138 if (is_cat
== "CAT:") {
1139 new_feed
= new_feed
.replace("CAT:", "");
1140 viewCategory(new_feed
);
1142 viewfeed(new_feed
, '', false);
1148 if (shift_key
&& keycode
== 40) { // shift-down
1149 catchupRelativeToArticle(1);
1153 if (shift_key
&& keycode
== 38) { // shift-up
1154 catchupRelativeToArticle(0);
1158 if (shift_key
&& keycode
== 78) { // N
1163 if (shift_key
&& keycode
== 80) { // P
1169 if (keycode
== 78 || keycode
== 40) { // n, down
1170 if (typeof moveToPost
!= 'undefined') {
1176 if (keycode
== 80 || keycode
== 38) { // p, up
1177 if (typeof moveToPost
!= 'undefined') {
1183 if (keycode
== 83 && shift_key
) { // S
1184 var id
= getActiveArticleId();
1191 if (keycode
== 83) { // s
1192 var id
= getActiveArticleId();
1200 if (keycode
== 85) { // u
1201 var id
= getActiveArticleId();
1208 if (keycode
== 84 && shift_key
) { // T
1209 var id
= getActiveArticleId();
1211 editArticleTags(id
, getActiveFeedId(), isCdmMode());
1216 if (keycode
== 9) { // tab
1217 var id
= getArticleUnderPointer();
1219 var cb
= document
.getElementById("RCHK-" + id
);
1222 cb
.checked
= !cb
.checked
;
1223 toggleSelectRowById(cb
, "RROW-" + id
);
1229 if (keycode
== 79) { // o
1230 if (getActiveArticleId()) {
1231 openArticleInNewWindow(getActiveArticleId());
1236 if (keycode
== 81 && shift_key
) { // Q
1237 if (typeof catchupAllFeeds
!= 'undefined') {
1243 if (keycode
== 88) { // x
1244 if (activeFeedIsCat()) {
1245 toggleCollapseCat(getActiveFeedId());
1252 if (hotkey_prefix
== 70) { // f
1254 hotkey_prefix
= false;
1256 if (keycode
== 81) { // q
1257 if (getActiveFeedId()) {
1258 catchupCurrentFeed();
1263 if (keycode
== 82) { // r
1264 if (getActiveFeedId()) {
1265 viewfeed(getActiveFeedId(), "ForceUpdate", activeFeedIsCat());
1270 if (keycode
== 65) { // a
1275 if (keycode
== 85 && shift_key
) { // U
1276 scheduleFeedUpdate(true);
1280 if (keycode
== 85) { // u
1281 if (getActiveFeedId()) {
1282 viewfeed(getActiveFeedId(), "ForceUpdate");
1287 if (keycode
== 69) { // e
1288 editFeedDlg(getActiveFeedId());
1292 if (keycode
== 83) { // s
1293 displayDlg("quickAddFeed");
1297 if (keycode
== 67 && shift_key
) { // C
1298 if (typeof catchupAllFeeds
!= 'undefined') {
1304 if (keycode
== 67) { // c
1305 if (getActiveFeedId()) {
1306 catchupCurrentFeed();
1311 if (keycode
== 68 && shift_key
) { // D
1312 initiate_offline_download();
1316 if (keycode
== 68) { // d
1317 displayDlg("offlineDownload");
1321 if (keycode
== 87) { // w
1322 feeds_sort_by_unread
= !feeds_sort_by_unread
;
1323 return resort_feedlist();
1326 if (keycode
== 72) { // h
1327 hideReadHeadlines();
1335 if (hotkey_prefix
== 67) { // c
1336 hotkey_prefix
= false;
1338 if (keycode
== 70) { // f
1339 displayDlg("quickAddFilter", getActiveFeedId());
1343 if (keycode
== 76) { // l
1348 if (keycode
== 83) { // s
1349 if (typeof collapse_feedlist
!= 'undefined') {
1350 collapse_feedlist();
1355 if (keycode
== 77) { // m
1356 feedlist_sortable_enabled
= !feedlist_sortable_enabled
;
1357 if (feedlist_sortable_enabled
) {
1358 notify_info("Category reordering enabled");
1359 toggle_sortable_feedlist(true);
1361 notify_info("Category reordering disabled");
1362 toggle_sortable_feedlist(false);
1366 if (keycode
== 78) { // n
1367 catchupRelativeToArticle(1);
1371 if (keycode
== 80) { // p
1372 catchupRelativeToArticle(0);
1381 if (hotkey_prefix
== 71) { // g
1383 hotkey_prefix
= false;
1386 if (keycode
== 65) { // a
1391 if (keycode
== 83) { // s
1396 if (keycode
== 80 && shift_key
) { // P
1401 if (keycode
== 80) { // p
1406 if (keycode
== 70) { // f
1411 if (keycode
== 84 && shift_key
) { // T
1419 if (hotkey_prefix
== 224 || hotkey_prefix
== 91) { // f
1420 hotkey_prefix
= false;
1424 if (hotkey_prefix
) {
1425 debug("KP: PREFIX=" + hotkey_prefix
+ " CODE=" + keycode
+ " CHAR=" + keychar
);
1427 debug("KP: CODE=" + keycode
+ " CHAR=" + keychar
);
1432 exception_error("hotkey_handler", e
);
1436 function feedsSortByUnread() {
1437 return feeds_sort_by_unread
;
1440 function addLabel() {
1444 var caption
= prompt(__("Please enter label caption:"), "");
1446 if (caption
!= undefined) {
1448 if (caption
== "") {
1449 alert(__("Can't create label: missing caption."));
1453 var query
= "backend.php?op=pref-labels&subop=add&caption=" +
1454 param_escape(caption
);
1456 notify_progress("Loading, please wait...", true);
1458 new Ajax
.Request(query
, {
1459 onComplete: function(transport
) {
1466 exception_error("addLabel", e
);
1470 function visitOfficialSite() {
1471 window
.open("http://tt-rss.org/");
1475 function feedBrowserSubscribe() {
1478 var selected
= getSelectedFeedsFromBrowser();
1480 if (selected
.length
> 0) {
1483 notify_progress("Loading, please wait...", true);
1485 var query
= "backend.php?op=pref-feeds&subop=massSubscribe&ids="+
1486 param_escape(selected
.toString());
1488 new Ajax
.Request(query
, {
1489 onComplete: function(transport
) {
1494 alert(__("No feeds are selected."));
1498 exception_error("feedBrowserSubscribe", e
);
1502 function init_gears() {
1505 if (window
.google
&& google
.gears
) {
1506 localServer
= google
.gears
.factory
.create("beta.localserver");
1507 store
= localServer
.createManagedStore("tt-rss");
1508 db
= google
.gears
.factory
.create('beta.database');
1511 db
.execute("CREATE TABLE IF NOT EXISTS version (schema_version text)");
1513 var rs
= db
.execute("SELECT schema_version FROM version");
1517 if (rs
.isValidRow()) {
1518 version
= rs
.field(0);
1521 if (version
!= SCHEMA_VERSION
) {
1522 db
.execute("DROP TABLE cache");
1523 db
.execute("DROP TABLE feeds");
1524 db
.execute("DROP TABLE articles");
1525 db
.execute("INSERT INTO version (schema_version) VALUES (?)",
1529 db
.execute("CREATE TABLE IF NOT EXISTS cache (id text, article text, param text, added text)");
1531 db
.execute("CREATE TABLE if not exists feeds (id integer, title text, has_icon integer)");
1533 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, added text)");
1535 var qmcDownload
= document
.getElementById("qmcDownload");
1536 if (qmcDownload
) Element
.show(qmcDownload
);
1543 exception_error("init_gears", e
);
1547 function init_offline() {
1549 offline_mode
= true;
1551 Element
.hide("dispSwitchPrompt");
1552 Element
.hide("feedBrowserPrompt");
1553 Element
.hide("quickMenuChooser");
1555 init_params
["theme"] = "";
1557 render_offline_feedlist();
1560 exception_error("init_offline", e
);
1564 function offline_download_parse(stage
, transport
) {
1566 if (transport
.responseXML
) {
1570 var feeds
= transport
.responseXML
.getElementsByTagName("feed");
1572 if (feeds
.length
> 0) {
1573 db
.execute("DELETE FROM feeds");
1576 for (var i
= 0; i
< feeds
.length
; i
++) {
1577 var id
= feeds
[i
].getAttribute("id");
1578 var has_icon
= feeds
[i
].getAttribute("has_icon");
1579 var title
= feeds
[i
].firstChild
.nodeValue
;
1581 db
.execute("INSERT INTO feeds (id,title,has_icon)"+
1583 [id
, title
, has_icon
]);
1586 window
.setTimeout("initiate_offline_download("+(stage
+1)+")", 50);
1589 var articles
= transport
.responseXML
.getElementsByTagName("article");
1591 var articles_found
= 0;
1593 for (var i
= 0; i
< articles
.length
; i
++) {
1594 var a
= eval("("+articles
[i
].firstChild
.nodeValue
+")");
1598 var date
= new Date();
1599 var ts
= Math
.round(date
.getTime() / 1000);
1601 db
.execute("DELETE FROM articles WHERE id = ?", [a
.id
]);
1602 db
.execute("INSERT INTO articles "+
1603 "(id, feed_id, title, link, guid, updated, content, "+
1604 "unread, marked, tags, added) "+
1605 "VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)",
1606 [a
.id
, a
.feed_id
, a
.title
, a
.link
, a
.guid
, a
.updated
,
1607 a
.content
, a
.unread
, a
.marked
, a
.tags
, ts
]);
1612 if (articles_found
> 0) {
1613 window
.setTimeout("initiate_offline_download("+(stage
+1)+")", 50);
1615 notify_info("All done.");
1622 exception_error("offline_download_parse", e
);
1626 function initiate_offline_download(stage
, caller
) {
1629 if (!stage
) stage
= 0;
1630 if (caller
) caller
.disabled
= true;
1632 notify_progress("Loading, please wait... (" + stage
+")", true);
1634 var query
= "backend.php?op=rpc&subop=download&stage=" + stage
;
1637 var rs
= db
.execute("SELECT MAX(id) FROM articles");
1638 if (rs
.isValidRow() && rs
.field(0)) {
1639 offline_dl_max_id
= rs
.field(0);
1643 if (offline_dl_max_id
) {
1644 query
= query
+ "&cid=" + offline_dl_max_id
;
1647 if (document
.getElementById("download_ops_form")) {
1648 query
= query
+ "&" + Form
.serialize("download_ops_form");
1651 new Ajax
.Request(query
, {
1652 onComplete: function(transport
) {
1653 offline_download_parse(stage
, transport
);
1657 exception_error("initiate_offline_download", e
);