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 number_of_feeds = 0;
7 var hotkey_prefix = false;
8 var hotkey_prefix_pressed = false;
10 var _force_scheduled_update = false;
11 var last_scheduled_update = false;
24 function activeFeedIsCat() {
25 return _active_feed_is_cat;
28 function getActiveFeedId() {
30 //console.log("gAFID: " + _active_feed_id);
31 return _active_feed_id;
33 exception_error("getActiveFeedId", e);
37 function setActiveFeedId(id, is_cat) {
41 if (is_cat != undefined) {
42 _active_feed_is_cat = is_cat;
45 selectFeed(id, is_cat);
48 exception_error("setActiveFeedId", e);
53 function dlg_frefresh_callback(transport, deleted_feed) {
54 if (getActiveFeedId() == deleted_feed) {
55 setTimeout("viewfeed(-5)", 100);
58 setTimeout('updateFeedList()', 50);
62 function updateFeedList() {
65 // $("feeds-holder").innerHTML = "<div id=\"feedlistLoading\">" +
66 // __("Loading, please wait...") + "</div>";
68 Element.show("feedlistLoading");
70 if (dijit.byId("feedTree")) {
71 dijit.byId("feedTree").destroyRecursive();
74 var store = new dojo.data.ItemFileWriteStore({
75 url: "backend.php?op=feeds"});
77 var treeModel = new fox.FeedStoreModel({
84 childrenAttrs: ["items"]
87 var tree = new fox.FeedTree({
89 onOpen: function (item, node) {
90 var id = String(item.id);
91 var cat_id = id.substr(id.indexOf(":")+1);
93 new Ajax.Request("backend.php",
94 { parameters: "backend.php?op=feeds&subop=collapse&cid=" +
95 param_escape(cat_id) + "&mode=1" } );
97 onClose: function (item, node) {
98 var id = String(item.id);
99 var cat_id = id.substr(id.indexOf(":")+1);
101 new Ajax.Request("backend.php",
102 { parameters: "backend.php?op=feeds&subop=collapse&cid=" +
103 param_escape(cat_id) + "&mode=0" } );
106 onClick: function (item, node) {
107 var id = String(item.id);
108 var is_cat = id.match("^CAT:");
109 var feed = id.substr(id.indexOf(":")+1);
110 viewfeed(feed, '', is_cat);
118 /* var menu = new dijit.Menu({id: 'feedMenu'});
120 menu.addChild(new dijit.MenuItem({
121 label: "Simple menu item"
124 // menu.bindDomNode(tree.domNode); */
126 var tmph = dojo.connect(dijit.byId('feedMenu'), '_openMyself', function (event) {
127 console.log(dijit.getEnclosingWidget(event.target));
128 dojo.disconnect(tmph);
131 $("feeds-holder").appendChild(tree.domNode);
133 var tmph = dojo.connect(tree, 'onLoad', function() {
134 dojo.disconnect(tmph);
135 Element.hide("feedlistLoading");
138 // var node = dijit.byId("feedTree")._itemNodesMap['FEED:-2'][0].domNode
139 // menu.bindDomNode(node);
141 loading_set_progress(25);
147 exception_error("updateFeedList", e);
151 function catchupAllFeeds() {
153 var str = __("Mark all articles as read?");
155 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
157 var query_str = "backend.php?op=feeds&subop=catchupAll";
159 notify_progress("Marking all feeds as read...");
161 //console.log("catchupAllFeeds Q=" + query_str);
163 new Ajax.Request("backend.php", {
164 parameters: query_str,
165 onComplete: function(transport) {
166 feedlist_callback2(transport);
174 function viewCurrentFeed(subop) {
176 if (getActiveFeedId() != undefined) {
177 viewfeed(getActiveFeedId(), subop, activeFeedIsCat());
179 return false; // block unneeded form submits
183 if (getInitParam("bw_limit") == "1") return;
186 var date = new Date();
187 var ts = Math.round(date.getTime() / 1000);
189 if (ts - last_scheduled_update > 10 || _force_scheduled_update) {
191 //console.log("timeout()");
193 window.clearTimeout(counter_timeout_id);
195 var query_str = "?op=rpc&subop=getAllCounters&seq=" + next_seq();
199 if (firsttime_update && !navigator.userAgent.match("Opera")) {
200 firsttime_update = false;
206 query_str = query_str + "&omode=" + omode;
208 if (!_force_scheduled_update)
209 query_str = query_str + "&last_article_id=" + getInitParam("last_article_id");
211 //console.log("[timeout]" + query_str);
213 new Ajax.Request("backend.php", {
214 parameters: query_str,
215 onComplete: function(transport) {
216 handle_rpc_reply(transport, !_force_scheduled_update);
217 _force_scheduled_update = false;
220 last_scheduled_update = ts;
224 exception_error("timeout", e);
227 setTimeout("timeout()", 3000);
235 function updateTitle() {
236 var tmp = "Tiny Tiny RSS";
238 if (global_unread > 0) {
239 tmp = tmp + " (" + global_unread + ")";
243 if (global_unread > 0) {
244 window.fluid.dockBadge = global_unread;
246 window.fluid.dockBadge = "";
250 document.title = tmp;
253 function genericSanityCheck() {
254 setCookie("ttrss_test", "TEST");
256 if (getCookie("ttrss_test") != "TEST") {
265 Form.disable("main_toolbar_form");
267 dojo.require("dijit.layout.BorderContainer");
268 dojo.require("dijit.layout.TabContainer");
269 dojo.require("dijit.layout.ContentPane");
270 dojo.require("dijit.Dialog");
271 dojo.require("dijit.form.Button");
272 dojo.require("dijit.Menu");
273 dojo.require("dojo.data.ItemFileWriteStore");
274 dojo.require("dijit.Tree");
275 dojo.require("dijit.form.Select");
276 dojo.require("dijit.Toolbar");
277 dojo.require("dijit.ProgressBar");
278 dojo.require("dijit.Menu");
280 dojo.registerModulePath("fox", "../..");
282 dojo.require("fox.FeedTree");
284 if (typeof themeBeforeLayout == 'function') {
288 dojo.addOnLoad(function() {
292 if (typeof themeAfterLayout == 'function') {
298 if (!genericSanityCheck())
301 loading_set_progress(20);
303 new Ajax.Request("backend.php", {
304 parameters: {op: "rpc", subop: "sanityCheck"},
305 onComplete: function(transport) {
306 backend_sanity_check_callback(transport);
310 exception_error("init", e);
314 function init_second_stage() {
318 delCookie("ttrss_test");
320 var toolbar = document.forms["main_toolbar_form"];
322 dropboxSelect(toolbar.view_mode, getInitParam("default_view_mode"));
323 dropboxSelect(toolbar.order_by, getInitParam("default_view_order_by"));
325 feeds_sort_by_unread = getInitParam("feeds_sort_by_unread") == 1;
327 loading_set_progress(30);
329 if (has_local_storage())
330 localStorage.clear();
332 console.log("second stage ok");
335 exception_error("init_second_stage", e);
339 function quickMenuChange() {
340 var chooser = $("quickMenuChooser");
341 var opid = chooser[chooser.selectedIndex].value;
343 chooser.selectedIndex = 0;
347 function quickMenuGo(opid) {
350 if (opid == "qmcPrefs") {
354 if (opid == "qmcTagCloud") {
355 displayDlg("printTagCloud");
358 if (opid == "qmcSearch") {
359 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat(),
361 document.forms['search_form'].query.focus();
366 if (opid == "qmcAddFeed") {
371 if (opid == "qmcEditFeed") {
372 editFeedDlg(getActiveFeedId());
375 if (opid == "qmcRemoveFeed") {
376 var actid = getActiveFeedId();
378 if (activeFeedIsCat()) {
379 alert(__("You can't unsubscribe from the category."));
384 alert(__("Please select some feed first."));
388 var fn = getFeedName(actid);
390 var pr = __("Unsubscribe from %s?").replace("%s", fn);
393 unsubscribeFeed(actid);
399 if (opid == "qmcCatchupAll") {
404 if (opid == "qmcShowOnlyUnread") {
409 if (opid == "qmcAddFilter") {
410 displayDlg('quickAddFilter', '',
411 function () {document.forms['filter_add_form'].reg_exp.focus();});
414 if (opid == "qmcAddLabel") {
418 if (opid == "qmcRescoreFeed") {
419 rescoreCurrentFeed();
422 if (opid == "qmcHKhelp") {
423 //Element.show("hotkey_help_overlay");
424 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
428 exception_error("quickMenuGo", e);
432 function toggleDispRead() {
435 var hide = !(getInitParam("hide_read_feeds") == "1");
437 hideOrShowFeeds(hide);
439 var query = "?op=rpc&subop=setpref&key=HIDE_READ_FEEDS&value=" +
442 setInitParam("hide_read_feeds", hide);
444 new Ajax.Request("backend.php", {
446 onComplete: function(transport) {
450 exception_error("toggleDispRead", e);
454 function parse_runtime_info(elem) {
456 if (!elem || !elem.firstChild) {
457 console.warn("parse_runtime_info: invalid node passed");
461 var data = JSON.parse(elem.firstChild.nodeValue);
463 //console.log("parsing runtime info...");
468 // console.log("RI: " + k + " => " + v);
470 if (k == "new_version_available") {
471 var icon = $("newVersionIcon");
474 icon.style.display = "inline";
476 icon.style.display = "none";
484 if (k == "daemon_is_running" && v != 1) {
485 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
489 if (k == "daemon_stamp_ok" && v != 1) {
490 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
499 function catchupCurrentFeed() {
501 var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
503 var str = __("Mark all articles in %s as read?").replace("%s", fn);
505 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
506 return viewCurrentFeed('MarkAllRead')
510 function catchupFeedInGroup(id) {
514 var title = getFeedName(id);
516 var str = __("Mark all articles in %s as read?").replace("%s", title);
518 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
519 return viewCurrentFeed('MarkAllReadGR:' + id)
523 exception_error("catchupFeedInGroup", e);
527 function editFeedDlg(feed) {
531 alert(__("Please select some feed first."));
535 if ((feed <= 0) || activeFeedIsCat()) {
536 alert(__("You can't edit this kind of feed."));
543 query = "?op=pref-feeds&subop=editfeed&id=" + param_escape(feed);
545 query = "?op=pref-labels&subop=edit&id=" + param_escape(-feed-11);
550 notify_progress("Loading, please wait...", true);
552 new Ajax.Request("backend.php", {
554 onComplete: function(transport) {
555 infobox_callback2(transport);
556 document.forms["edit_feed_form"].title.focus();
560 exception_error("editFeedDlg", e);
564 /* this functions duplicate those of prefs.js feed editor, with
565 some differences because there is no feedlist */
567 function feedEditCancel() {
572 function feedEditSave() {
576 // FIXME: add parameter validation
578 var query = Form.serialize("edit_feed_form");
580 notify_progress("Saving feed...");
582 new Ajax.Request("backend.php", {
584 onComplete: function(transport) {
585 dlg_frefresh_callback(transport);
594 exception_error("feedEditSave (main)", e);
598 function collapse_feedlist() {
601 if (!Element.visible('feeds-holder')) {
602 Element.show('feeds-holder');
603 $("collapse_feeds_btn").innerHTML = "<<";
605 Element.hide('feeds-holder');
606 $("collapse_feeds_btn").innerHTML = ">>";
609 dijit.byId("main").resize();
611 query = "?op=rpc&subop=setpref&key=_COLLAPSED_FEEDLIST&value=true";
612 new Ajax.Request("backend.php", { parameters: query });
615 exception_error("collapse_feedlist", e);
619 function viewModeChanged() {
621 return viewCurrentFeed('')
624 function viewLimitChanged() {
626 return viewCurrentFeed('')
629 /* function adjustArticleScore(id, score) {
632 var pr = prompt(__("Assign score to article:"), score);
634 if (pr != undefined) {
635 var query = "?op=rpc&subop=setScore&id=" + id + "&score=" + pr;
637 new Ajax.Request("backend.php", {
639 onComplete: function(transport) {
645 exception_error("adjustArticleScore", e);
649 function rescoreCurrentFeed() {
651 var actid = getActiveFeedId();
653 if (activeFeedIsCat() || actid < 0) {
654 alert(__("You can't rescore this kind of feed."));
659 alert(__("Please select some feed first."));
663 var fn = getFeedName(actid);
664 var pr = __("Rescore articles in %s?").replace("%s", fn);
667 notify_progress("Rescoring articles...");
669 var query = "?op=pref-feeds&subop=rescore&quiet=1&ids=" + actid;
671 new Ajax.Request("backend.php", {
673 onComplete: function(transport) {
679 function hotkey_handler(e) {
684 var shift_key = false;
686 var cmdline = $('cmdline');
689 shift_key = e.shiftKey;
695 keycode = window.event.keyCode;
700 var keychar = String.fromCharCode(keycode);
702 if (keycode == 27) { // escape
703 if (Element.visible("hotkey_help_overlay")) {
704 Element.hide("hotkey_help_overlay");
706 hotkey_prefix = false;
710 var dialog = dijit.byId("infoBox");
711 var dialog_visible = false;
714 dialog_visible = Element.visible(dialog.domNode);
716 if (dialog_visible || !hotkeys_enabled) {
717 console.log("hotkeys disabled");
721 if (keycode == 16) return; // ignore lone shift
722 if (keycode == 17) return; // ignore lone ctrl
724 if ((keycode == 70 || keycode == 67 || keycode == 71)
727 var date = new Date();
728 var ts = Math.round(date.getTime() / 1000);
730 hotkey_prefix = keycode;
731 hotkey_prefix_pressed = ts;
733 cmdline.innerHTML = keychar;
734 Element.show(cmdline);
736 console.log("KP: PREFIX=" + keycode + " CHAR=" + keychar + " TS=" + ts);
740 if (Element.visible("hotkey_help_overlay")) {
741 Element.hide("hotkey_help_overlay");
746 Element.hide(cmdline);
748 if (!hotkey_prefix) {
750 if ((keycode == 191 || keychar == '?') && shift_key) { // ?
751 if (!Element.visible("hotkey_help_overlay")) {
752 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
754 Element.hide("hotkey_help_overlay");
759 if (keycode == 191 || keychar == '/') { // /
760 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat(),
762 document.forms['search_form'].query.focus();
767 if (keycode == 74) { // j
768 // TODO: move to previous feed
772 if (keycode == 75) { // k
773 // TODO: move to next feed
777 if (shift_key && keycode == 40) { // shift-down
778 catchupRelativeToArticle(1);
782 if (shift_key && keycode == 38) { // shift-up
783 catchupRelativeToArticle(0);
787 if (shift_key && keycode == 78) { // N
792 if (shift_key && keycode == 80) { // P
797 if (keycode == 68 && shift_key) { // shift-D
798 dismissSelectedArticles();
801 if (keycode == 88 && shift_key) { // shift-X
802 dismissReadArticles();
805 if (keycode == 78 || keycode == 40) { // n, down
806 if (typeof moveToPost != 'undefined') {
812 if (keycode == 80 || keycode == 38) { // p, up
813 if (typeof moveToPost != 'undefined') {
819 if (keycode == 83 && shift_key) { // S
820 selectionTogglePublished(undefined, false, true);
824 if (keycode == 83) { // s
825 selectionToggleMarked(undefined, false, true);
830 if (keycode == 85) { // u
831 selectionToggleUnread(undefined, false, true)
835 if (keycode == 84 && shift_key) { // T
836 var id = getActiveArticleId();
838 editArticleTags(id, getActiveFeedId(), isCdmMode());
843 if (keycode == 9) { // tab
844 var id = getArticleUnderPointer();
846 var cb = $("RCHK-" + id);
849 cb.checked = !cb.checked;
850 toggleSelectRowById(cb, "RROW-" + id);
856 if (keycode == 79) { // o
857 if (getActiveArticleId()) {
858 openArticleInNewWindow(getActiveArticleId());
863 if (keycode == 81 && shift_key) { // Q
864 if (typeof catchupAllFeeds != 'undefined') {
870 if (keycode == 88) { // x
871 if (activeFeedIsCat()) {
872 toggleCollapseCat(getActiveFeedId());
879 if (hotkey_prefix == 70) { // f
881 hotkey_prefix = false;
883 if (keycode == 81) { // q
884 if (getActiveFeedId()) {
885 catchupCurrentFeed();
890 if (keycode == 82) { // r
891 if (getActiveFeedId()) {
892 viewfeed(getActiveFeedId(), "ForceUpdate", activeFeedIsCat());
897 if (keycode == 65) { // a
902 if (keycode == 85) { // u
903 if (getActiveFeedId()) {
904 viewfeed(getActiveFeedId(), "ForceUpdate");
909 if (keycode == 69) { // e
910 editFeedDlg(getActiveFeedId());
914 if (keycode == 83) { // s
919 if (keycode == 67 && shift_key) { // C
920 if (typeof catchupAllFeeds != 'undefined') {
926 if (keycode == 67) { // c
927 if (getActiveFeedId()) {
928 catchupCurrentFeed();
933 if (keycode == 87) { // w
934 feeds_sort_by_unread = !feeds_sort_by_unread;
935 return resort_feedlist();
938 if (keycode == 88) { // x
939 reverseHeadlineOrder();
946 if (hotkey_prefix == 67) { // c
947 hotkey_prefix = false;
949 if (keycode == 70) { // f
950 displayDlg('quickAddFilter', '',
951 function () {document.forms['filter_add_form'].reg_exp.focus();});
955 if (keycode == 76) { // l
960 if (keycode == 83) { // s
961 if (typeof collapse_feedlist != 'undefined') {
967 if (keycode == 77) { // m
968 // TODO: sortable feedlist
972 if (keycode == 78) { // n
973 catchupRelativeToArticle(1);
977 if (keycode == 80) { // p
978 catchupRelativeToArticle(0);
987 if (hotkey_prefix == 71) { // g
989 hotkey_prefix = false;
992 if (keycode == 65) { // a
997 if (keycode == 83) { // s
1002 if (keycode == 80 && shift_key) { // P
1007 if (keycode == 80) { // p
1012 if (keycode == 70) { // f
1017 if (keycode == 84 && shift_key) { // T
1025 if (hotkey_prefix == 224 || hotkey_prefix == 91) { // f
1026 hotkey_prefix = false;
1030 if (hotkey_prefix) {
1031 console.log("KP: PREFIX=" + hotkey_prefix + " CODE=" + keycode + " CHAR=" + keychar);
1033 console.log("KP: CODE=" + keycode + " CHAR=" + keychar);
1038 exception_error("hotkey_handler", e);
1042 function inPreferences() {
1046 function reverseHeadlineOrder() {
1049 var query_str = "?op=rpc&subop=togglepref&key=REVERSE_HEADLINES";
1051 new Ajax.Request("backend.php", {
1052 parameters: query_str,
1053 onComplete: function(transport) {
1058 exception_error("reverseHeadlineOrder", e);
1062 function showFeedsWithErrors() {
1063 displayDlg('feedUpdateErrors');
1066 function handle_rpc_reply(transport, scheduled_call) {
1068 if (transport.responseXML) {
1070 if (!transport_error_check(transport)) return false;
1072 var seq = transport.responseXML.getElementsByTagName("seq")[0];
1075 seq = seq.firstChild.nodeValue;
1077 if (get_seq() != seq) {
1078 //console.log("[handle_rpc_reply] sequence mismatch: " + seq);
1083 var message = transport.responseXML.getElementsByTagName("message")[0];
1086 message = message.firstChild.nodeValue;
1088 if (message == "UPDATE_COUNTERS") {
1089 console.log("need to refresh counters...");
1090 setInitParam("last_article_id", -1);
1091 _force_scheduled_update = true;
1095 var counters = transport.responseXML.getElementsByTagName("counters")[0];
1098 parse_counters(counters, scheduled_call);
1100 var runtime_info = transport.responseXML.getElementsByTagName("runtime-info")[0];
1103 parse_runtime_info(runtime_info);
1105 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1108 notify_error("Error communicating with server.");
1112 exception_error("handle_rpc_reply", e, transport);
1118 function scheduleFeedUpdate(id, is_cat) {
1121 id = getActiveFeedId();
1122 is_cat = activeFeedIsCat();
1126 alert(__("Please select some feed first."));
1130 var query = "?op=rpc&subop=scheduleFeedUpdate&id=" +
1132 "&is_cat=" + param_escape(is_cat);
1136 new Ajax.Request("backend.php", {
1138 onComplete: function(transport) {
1140 if (transport.responseXML) {
1141 var message = transport.responseXML.getElementsByTagName("message")[0];
1144 notify_info(message.firstChild.nodeValue);
1149 notify_error("Error communicating with server.");
1155 exception_error("scheduleFeedUpdate", e);