2 var display_tags = false;
3 var global_unread = -1;
4 var firsttime_update = true;
5 var _active_feed_id = 0;
6 var _active_feed_is_cat = false;
7 var number_of_feeds = 0;
8 var hotkey_prefix = false;
9 var hotkey_prefix_pressed = false;
11 var _force_scheduled_update = false;
12 var last_scheduled_update = false;
26 function activeFeedIsCat() {
27 return _active_feed_is_cat;
30 function getActiveFeedId() {
32 //console.log("gAFID: " + _active_feed_id);
33 return _active_feed_id;
35 exception_error("getActiveFeedId", e);
39 function setActiveFeedId(id, is_cat) {
43 if (is_cat != undefined) {
44 _active_feed_is_cat = is_cat;
47 selectFeed(id, is_cat);
50 exception_error("setActiveFeedId", e);
55 function isFeedlistSortable() {
56 return feedlist_sortable_enabled;
59 function tagsAreDisplayed() {
63 function toggleTags(show_all) {
67 console.log("toggleTags: " + show_all + "; " + display_tags);
69 var p = $("dispSwitchPrompt");
71 if (!show_all && !display_tags) {
72 displayDlg("printTagCloud");
73 } else if (show_all) {
76 p.innerHTML = __("display feeds");
77 notify_progress("Loading, please wait...", true);
79 } else if (display_tags) {
81 p.innerHTML = __("tag cloud");
82 notify_progress("Loading, please wait...", true);
87 exception_error("toggleTags", e);
91 function dlg_frefresh_callback(transport, deleted_feed) {
92 if (getActiveFeedId() == deleted_feed) {
93 setTimeout("viewfeed(-5)", 100);
96 setTimeout('updateFeedList()', 50);
100 function updateFeedList() {
102 console.warn("updateFeedList: function not implemented");
104 /* var query_str = "backend.php?op=feeds";
107 query_str = query_str + "&tags=1";
110 if (getActiveFeedId() && !activeFeedIsCat()) {
111 query_str = query_str + "&actid=" + getActiveFeedId();
114 new Ajax.Request("backend.php", {
115 parameters: query_str,
116 onComplete: function(transport) {
117 render_feedlist(transport.responseText);
121 exception_error("updateFeedList", e);
125 function catchupAllFeeds() {
127 var str = __("Mark all articles as read?");
129 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
131 var query_str = "backend.php?op=feeds&subop=catchupAll";
133 notify_progress("Marking all feeds as read...");
135 //console.log("catchupAllFeeds Q=" + query_str);
137 new Ajax.Request("backend.php", {
138 parameters: query_str,
139 onComplete: function(transport) {
140 feedlist_callback2(transport);
148 function viewCurrentFeed(subop) {
150 if (getActiveFeedId() != undefined) {
151 viewfeed(getActiveFeedId(), subop, activeFeedIsCat());
153 return false; // block unneeded form submits
157 if (getInitParam("bw_limit") == "1") return;
160 var date = new Date();
161 var ts = Math.round(date.getTime() / 1000);
163 if (ts - last_scheduled_update > 10 || _force_scheduled_update) {
165 //console.log("timeout()");
167 window.clearTimeout(counter_timeout_id);
169 var query_str = "?op=rpc&subop=getAllCounters&seq=" + next_seq();
173 if (firsttime_update && !navigator.userAgent.match("Opera")) {
174 firsttime_update = false;
184 query_str = query_str + "&omode=" + omode;
186 if (!_force_scheduled_update)
187 query_str = query_str + "&last_article_id=" + getInitParam("last_article_id");
189 //console.log("[timeout]" + query_str);
191 new Ajax.Request("backend.php", {
192 parameters: query_str,
193 onComplete: function(transport) {
194 handle_rpc_reply(transport, !_force_scheduled_update);
195 _force_scheduled_update = false;
198 last_scheduled_update = ts;
202 exception_error("timeout", e);
205 setTimeout("timeout()", 3000);
213 function updateTitle() {
214 var tmp = "Tiny Tiny RSS";
216 if (global_unread > 0) {
217 tmp = tmp + " (" + global_unread + ")";
221 if (global_unread > 0) {
222 window.fluid.dockBadge = global_unread;
224 window.fluid.dockBadge = "";
228 document.title = tmp;
231 function genericSanityCheck() {
232 setCookie("ttrss_test", "TEST");
234 if (getCookie("ttrss_test") != "TEST") {
243 Form.disable("main_toolbar_form");
245 dojo.require("dijit.layout.BorderContainer");
246 dojo.require("dijit.layout.ContentPane");
247 dojo.require("dijit.Dialog");
248 dojo.require("dijit.form.Button");
249 dojo.require("dojo.data.ItemFileWriteStore");
250 dojo.require("dijit.Tree");
251 dojo.require("dijit.form.Select");
252 dojo.require("dojo.parser");
254 dojo.addOnLoad(function() {
256 var store = new dojo.data.ItemFileWriteStore({
257 url: "backend.php?op=feeds"});
259 treeModel = new dijit.tree.ForestStoreModel({
266 childrenAttrs: ["items"]
269 var tree = new dijit.Tree({
271 _createTreeNode: function(args) {
272 var tnode = new dijit._TreeNode(args);
273 tnode.labelNode.innerHTML = args.label;
276 getLabelClass: function (item, opened) {
277 return (item.unread == 0) ? "dijitTreeLabel" : "dijitTreeLabel Unread";
279 getRowClass: function (item, opened) {
280 return (!item.error || item.error == '') ? "dijitTreeRow" :
281 "dijitTreeRow Error";
283 getLabel: function(item) {
284 if (item.unread > 0) {
285 return item.name + " (" + item.unread + ")";
290 onOpen: function (item, node) {
291 var id = String(item.id);
292 var cat_id = id.substr(id.indexOf(":")+1);
294 new Ajax.Request("backend.php",
295 { parameters: "backend.php?op=feeds&subop=collapse&cid=" +
296 param_escape(cat_id) + "&mode=1" } );
298 onClose: function (item, node) {
299 var id = String(item.id);
300 var cat_id = id.substr(id.indexOf(":")+1);
302 new Ajax.Request("backend.php",
303 { parameters: "backend.php?op=feeds&subop=collapse&cid=" +
304 param_escape(cat_id) + "&mode=0" } );
307 onClick: function (item, node) {
308 var id = String(item.id);
309 var is_cat = id.match("^CAT:");
310 var feed = id.substr(id.indexOf(":")+1);
311 viewfeed(feed, '', is_cat);
318 if (!genericSanityCheck())
321 var params = "&ua=" + param_escape(navigator.userAgent);
323 loading_set_progress(30);
325 new Ajax.Request("backend.php", {
326 parameters: "backend.php?op=rpc&subop=sanityCheck" + params,
327 onComplete: function(transport) {
328 backend_sanity_check_callback(transport);
332 exception_error("init", e);
336 function init_second_stage() {
340 delCookie("ttrss_test");
342 var toolbar = document.forms["main_toolbar_form"];
344 dropboxSelect(toolbar.view_mode, getInitParam("default_view_mode"));
345 dropboxSelect(toolbar.order_by, getInitParam("default_view_order_by"));
347 feeds_sort_by_unread = getInitParam("feeds_sort_by_unread") == 1;
349 loading_set_progress(60);
351 if (has_local_storage())
352 localStorage.clear();
355 setTimeout("timeout()", 3000);
357 console.log("second stage ok");
360 exception_error("init_second_stage", e);
364 function quickMenuChange() {
365 var chooser = $("quickMenuChooser");
366 var opid = chooser[chooser.selectedIndex].value;
368 chooser.selectedIndex = 0;
372 function quickMenuGo(opid) {
375 if (opid == "qmcPrefs") {
379 if (opid == "qmcSearch") {
380 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat(),
382 document.forms['search_form'].query.focus();
387 if (opid == "qmcAddFeed") {
392 if (opid == "qmcEditFeed") {
393 editFeedDlg(getActiveFeedId());
396 if (opid == "qmcRemoveFeed") {
397 var actid = getActiveFeedId();
399 if (activeFeedIsCat()) {
400 alert(__("You can't unsubscribe from the category."));
405 alert(__("Please select some feed first."));
409 var fn = getFeedName(actid);
411 var pr = __("Unsubscribe from %s?").replace("%s", fn);
414 unsubscribeFeed(actid);
420 if (opid == "qmcCatchupAll") {
425 if (opid == "qmcShowOnlyUnread") {
430 if (opid == "qmcAddFilter") {
431 displayDlg('quickAddFilter', '',
432 function () {document.forms['filter_add_form'].reg_exp.focus();});
435 if (opid == "qmcAddLabel") {
439 if (opid == "qmcRescoreFeed") {
440 rescoreCurrentFeed();
443 if (opid == "qmcHKhelp") {
444 //Element.show("hotkey_help_overlay");
445 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
448 if (opid == "qmcResetUI") {
449 alert("Function not implemented");
452 if (opid == "qmcToggleReorder") {
453 feedlist_sortable_enabled = !feedlist_sortable_enabled;
455 if (feedlist_sortable_enabled) {
456 notify_info("Category reordering enabled");
457 toggle_sortable_feedlist(true);
459 notify_info("Category reordering disabled");
460 toggle_sortable_feedlist(false);
464 if (opid == "qmcResetCats") {
466 if (confirm(__("Reset category order?"))) {
468 var query = "?op=feeds&subop=catsortreset";
470 notify_progress("Loading, please wait...", true);
472 new Ajax.Request("backend.php", {
474 onComplete: function(transport) {
475 window.setTimeout('updateFeedList(false, false)', 50);
481 exception_error("quickMenuGo", e);
485 function toggleDispRead() {
488 var hide = !(getInitParam("hide_read_feeds") == "1");
490 hideOrShowFeeds(hide);
492 var query = "?op=rpc&subop=setpref&key=HIDE_READ_FEEDS&value=" +
495 new Ajax.Request("backend.php", {
497 onComplete: function(transport) {
498 setInitParam("hide_read_feeds", hide);
502 exception_error("toggleDispRead", e);
506 function parse_runtime_info(elem) {
508 if (!elem || !elem.firstChild) {
509 console.warn("parse_runtime_info: invalid node passed");
513 var data = JSON.parse(elem.firstChild.nodeValue);
515 //console.log("parsing runtime info...");
520 // console.log("RI: " + k + " => " + v);
522 if (k == "new_version_available") {
523 var icon = $("newVersionIcon");
526 icon.style.display = "inline";
528 icon.style.display = "none";
536 if (k == "daemon_is_running" && v != 1) {
537 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
541 if (k == "daemon_stamp_ok" && v != 1) {
542 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
551 function catchupCurrentFeed() {
553 var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
555 var str = __("Mark all articles in %s as read?").replace("%s", fn);
557 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
558 return viewCurrentFeed('MarkAllRead')
562 function catchupFeedInGroup(id) {
566 var title = getFeedName(id);
568 var str = __("Mark all articles in %s as read?").replace("%s", title);
570 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
571 return viewCurrentFeed('MarkAllReadGR:' + id)
575 exception_error("catchupFeedInGroup", e);
579 function editFeedDlg(feed) {
583 alert(__("Please select some feed first."));
587 if ((feed <= 0) || activeFeedIsCat() || tagsAreDisplayed()) {
588 alert(__("You can't edit this kind of feed."));
595 query = "?op=pref-feeds&subop=editfeed&id=" + param_escape(feed);
597 query = "?op=pref-labels&subop=edit&id=" + param_escape(-feed-11);
602 notify_progress("Loading, please wait...", true);
604 new Ajax.Request("backend.php", {
606 onComplete: function(transport) {
607 infobox_callback2(transport);
608 document.forms["edit_feed_form"].title.focus();
612 exception_error("editFeedDlg", e);
616 /* this functions duplicate those of prefs.js feed editor, with
617 some differences because there is no feedlist */
619 function feedEditCancel() {
624 function feedEditSave() {
628 // FIXME: add parameter validation
630 var query = Form.serialize("edit_feed_form");
632 notify_progress("Saving feed...");
634 new Ajax.Request("backend.php", {
636 onComplete: function(transport) {
637 dlg_frefresh_callback(transport);
646 exception_error("feedEditSave (main)", e);
650 function collapse_feedlist() {
653 if (!Element.visible('feeds-holder')) {
654 Element.show('feeds-holder');
655 $("collapse_feeds_btn").innerHTML = "<<";
657 Element.hide('feeds-holder');
658 $("collapse_feeds_btn").innerHTML = ">>";
661 dijit.byId("main").resize();
663 query = "?op=rpc&subop=setpref&key=_COLLAPSED_FEEDLIST&value=true";
664 new Ajax.Request("backend.php", { parameters: query });
667 exception_error("collapse_feedlist", e);
671 function viewModeChanged() {
673 return viewCurrentFeed('')
676 function viewLimitChanged() {
678 return viewCurrentFeed('')
681 /* function adjustArticleScore(id, score) {
684 var pr = prompt(__("Assign score to article:"), score);
686 if (pr != undefined) {
687 var query = "?op=rpc&subop=setScore&id=" + id + "&score=" + pr;
689 new Ajax.Request("backend.php", {
691 onComplete: function(transport) {
697 exception_error("adjustArticleScore", e);
701 function rescoreCurrentFeed() {
703 var actid = getActiveFeedId();
705 if (activeFeedIsCat() || actid < 0 || tagsAreDisplayed()) {
706 alert(__("You can't rescore this kind of feed."));
711 alert(__("Please select some feed first."));
715 var fn = getFeedName(actid);
716 var pr = __("Rescore articles in %s?").replace("%s", fn);
719 notify_progress("Rescoring articles...");
721 var query = "?op=pref-feeds&subop=rescore&quiet=1&ids=" + actid;
723 new Ajax.Request("backend.php", {
725 onComplete: function(transport) {
731 function hotkey_handler(e) {
736 var shift_key = false;
738 var cmdline = $('cmdline');
741 shift_key = e.shiftKey;
747 keycode = window.event.keyCode;
752 var keychar = String.fromCharCode(keycode);
754 if (keycode == 27) { // escape
755 if (Element.visible("hotkey_help_overlay")) {
756 Element.hide("hotkey_help_overlay");
758 hotkey_prefix = false;
762 if (dialogs.length > 0 || !hotkeys_enabled) {
763 console.log("hotkeys disabled");
767 if (keycode == 16) return; // ignore lone shift
768 if (keycode == 17) return; // ignore lone ctrl
770 if ((keycode == 70 || keycode == 67 || keycode == 71)
773 var date = new Date();
774 var ts = Math.round(date.getTime() / 1000);
776 hotkey_prefix = keycode;
777 hotkey_prefix_pressed = ts;
779 cmdline.innerHTML = keychar;
780 Element.show(cmdline);
782 console.log("KP: PREFIX=" + keycode + " CHAR=" + keychar + " TS=" + ts);
786 if (Element.visible("hotkey_help_overlay")) {
787 Element.hide("hotkey_help_overlay");
792 Element.hide(cmdline);
794 if (!hotkey_prefix) {
796 if ((keycode == 191 || keychar == '?') && shift_key) { // ?
797 if (!Element.visible("hotkey_help_overlay")) {
798 //Element.show("hotkey_help_overlay");
799 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
801 Element.hide("hotkey_help_overlay");
806 if (keycode == 191 || keychar == '/') { // /
807 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat(),
809 document.forms['search_form'].query.focus();
814 /* if (keycode == 82 && shift_key) { // R
815 scheduleFeedUpdate(true);
819 if (keycode == 74) { // j
820 var feed = getActiveFeedId();
821 var new_feed = getRelativeFeedId2(feed, activeFeedIsCat(), 'prev');
822 // alert(feed + " IC: " + activeFeedIsCat() + " => " + new_feed);
824 var is_cat = new_feed.match("CAT:");
826 new_feed = new_feed.replace("CAT:", "");
827 viewCategory(new_feed);
829 viewfeed(new_feed, '', false);
835 if (keycode == 75) { // k
836 var feed = getActiveFeedId();
837 var new_feed = getRelativeFeedId2(feed, activeFeedIsCat(), 'next');
838 // alert(feed + " IC: " + activeFeedIsCat() + " => " + new_feed);
840 var is_cat = new_feed.match("CAT:");
841 if (is_cat == "CAT:") {
842 new_feed = new_feed.replace("CAT:", "");
843 viewCategory(new_feed);
845 viewfeed(new_feed, '', false);
851 if (shift_key && keycode == 40) { // shift-down
852 catchupRelativeToArticle(1);
856 if (shift_key && keycode == 38) { // shift-up
857 catchupRelativeToArticle(0);
861 if (shift_key && keycode == 78) { // N
866 if (shift_key && keycode == 80) { // P
871 if (keycode == 68 && shift_key) { // shift-D
872 dismissSelectedArticles();
875 if (keycode == 88 && shift_key) { // shift-X
876 dismissReadArticles();
879 if (keycode == 78 || keycode == 40) { // n, down
880 if (typeof moveToPost != 'undefined') {
886 if (keycode == 80 || keycode == 38) { // p, up
887 if (typeof moveToPost != 'undefined') {
893 if (keycode == 83 && shift_key) { // S
894 selectionTogglePublished(undefined, false, true);
898 if (keycode == 83) { // s
899 selectionToggleMarked(undefined, false, true);
904 if (keycode == 85) { // u
905 selectionToggleUnread(undefined, false, true)
909 if (keycode == 84 && shift_key) { // T
910 var id = getActiveArticleId();
912 editArticleTags(id, getActiveFeedId(), isCdmMode());
917 if (keycode == 9) { // tab
918 var id = getArticleUnderPointer();
920 var cb = $("RCHK-" + id);
923 cb.checked = !cb.checked;
924 toggleSelectRowById(cb, "RROW-" + id);
930 if (keycode == 79) { // o
931 if (getActiveArticleId()) {
932 openArticleInNewWindow(getActiveArticleId());
937 if (keycode == 81 && shift_key) { // Q
938 if (typeof catchupAllFeeds != 'undefined') {
944 if (keycode == 88) { // x
945 if (activeFeedIsCat()) {
946 toggleCollapseCat(getActiveFeedId());
953 if (hotkey_prefix == 70) { // f
955 hotkey_prefix = false;
957 if (keycode == 81) { // q
958 if (getActiveFeedId()) {
959 catchupCurrentFeed();
964 if (keycode == 82) { // r
965 if (getActiveFeedId()) {
966 viewfeed(getActiveFeedId(), "ForceUpdate", activeFeedIsCat());
971 if (keycode == 65) { // a
976 /* if (keycode == 85 && shift_key) { // U
977 scheduleFeedUpdate(true);
981 if (keycode == 85) { // u
982 if (getActiveFeedId()) {
983 viewfeed(getActiveFeedId(), "ForceUpdate");
988 if (keycode == 69) { // e
989 editFeedDlg(getActiveFeedId());
993 if (keycode == 83) { // s
998 if (keycode == 67 && shift_key) { // C
999 if (typeof catchupAllFeeds != 'undefined') {
1005 if (keycode == 67) { // c
1006 if (getActiveFeedId()) {
1007 catchupCurrentFeed();
1012 if (keycode == 87) { // w
1013 feeds_sort_by_unread = !feeds_sort_by_unread;
1014 return resort_feedlist();
1017 if (keycode == 88) { // x
1018 reverseHeadlineOrder();
1025 if (hotkey_prefix == 67) { // c
1026 hotkey_prefix = false;
1028 if (keycode == 70) { // f
1029 displayDlg('quickAddFilter', '',
1030 function () {document.forms['filter_add_form'].reg_exp.focus();});
1034 if (keycode == 76) { // l
1039 if (keycode == 83) { // s
1040 if (typeof collapse_feedlist != 'undefined') {
1041 collapse_feedlist();
1046 if (keycode == 77) { // m
1047 feedlist_sortable_enabled = !feedlist_sortable_enabled;
1048 if (feedlist_sortable_enabled) {
1049 notify_info("Category reordering enabled");
1050 toggle_sortable_feedlist(true);
1052 notify_info("Category reordering disabled");
1053 toggle_sortable_feedlist(false);
1057 if (keycode == 78) { // n
1058 catchupRelativeToArticle(1);
1062 if (keycode == 80) { // p
1063 catchupRelativeToArticle(0);
1072 if (hotkey_prefix == 71) { // g
1074 hotkey_prefix = false;
1077 if (keycode == 65) { // a
1082 if (keycode == 83) { // s
1087 if (keycode == 80 && shift_key) { // P
1092 if (keycode == 80) { // p
1097 if (keycode == 70) { // f
1102 if (keycode == 84 && shift_key) { // T
1110 if (hotkey_prefix == 224 || hotkey_prefix == 91) { // f
1111 hotkey_prefix = false;
1115 if (hotkey_prefix) {
1116 console.log("KP: PREFIX=" + hotkey_prefix + " CODE=" + keycode + " CHAR=" + keychar);
1118 console.log("KP: CODE=" + keycode + " CHAR=" + keychar);
1123 exception_error("hotkey_handler", e);
1127 function inPreferences() {
1131 function reverseHeadlineOrder() {
1134 var query_str = "?op=rpc&subop=togglepref&key=REVERSE_HEADLINES";
1136 new Ajax.Request("backend.php", {
1137 parameters: query_str,
1138 onComplete: function(transport) {
1143 exception_error("reverseHeadlineOrder", e);
1147 function showFeedsWithErrors() {
1148 displayDlg('feedUpdateErrors');
1151 function handle_rpc_reply(transport, scheduled_call) {
1153 if (transport.responseXML) {
1155 if (!transport_error_check(transport)) return false;
1157 var seq = transport.responseXML.getElementsByTagName("seq")[0];
1160 seq = seq.firstChild.nodeValue;
1162 if (get_seq() != seq) {
1163 //console.log("[handle_rpc_reply] sequence mismatch: " + seq);
1168 var message = transport.responseXML.getElementsByTagName("message")[0];
1171 message = message.firstChild.nodeValue;
1173 if (message == "UPDATE_COUNTERS") {
1174 console.log("need to refresh counters...");
1175 setInitParam("last_article_id", -1);
1176 _force_scheduled_update = true;
1180 var counters = transport.responseXML.getElementsByTagName("counters")[0];
1183 parse_counters(counters, scheduled_call);
1185 var runtime_info = transport.responseXML.getElementsByTagName("runtime-info")[0];
1188 parse_runtime_info(runtime_info);
1190 if (feedsSortByUnread())
1193 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1196 notify_error("Error communicating with server.");
1200 exception_error("handle_rpc_reply", e, transport);
1206 function scheduleFeedUpdate() {
1209 if (!getActiveFeedId()) {
1210 alert(__("Please select some feed first."));
1214 var query = "?op=rpc&subop=scheduleFeedUpdate&id=" +
1215 param_escape(getActiveFeedId()) +
1216 "&is_cat=" + param_escape(activeFeedIsCat());
1220 new Ajax.Request("backend.php", {
1222 onComplete: function(transport) {
1224 if (transport.responseXML) {
1225 var message = transport.responseXML.getElementsByTagName("message")[0];
1228 notify_info(message.firstChild.nodeValue);
1233 notify_error("Error communicating with server.");
1239 exception_error("scheduleFeedUpdate", e);