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;
25 function activeFeedIsCat() {
26 return _active_feed_is_cat;
29 function getActiveFeedId() {
31 //console.log("gAFID: " + _active_feed_id);
32 return _active_feed_id;
34 exception_error("getActiveFeedId", e);
38 function setActiveFeedId(id, is_cat) {
42 if (is_cat != undefined) {
43 _active_feed_is_cat = is_cat;
46 selectFeed(id, is_cat);
49 exception_error("setActiveFeedId", e);
54 function dlg_frefresh_callback(transport, deleted_feed) {
55 if (getActiveFeedId() == deleted_feed) {
56 setTimeout("viewfeed(-5)", 100);
59 setTimeout('updateFeedList()', 50);
63 function updateFeedList() {
66 // $("feeds-holder").innerHTML = "<div id=\"feedlistLoading\">" +
67 // __("Loading, please wait...") + "</div>";
69 Element.show("feedlistLoading");
71 if (dijit.byId("feedTree")) {
72 dijit.byId("feedTree").destroyRecursive();
75 var store = new dojo.data.ItemFileWriteStore({
76 url: "backend.php?op=feeds"});
78 treeModel = new dijit.tree.ForestStoreModel({
85 childrenAttrs: ["items"]
88 var tree = new dijit.Tree({
90 _createTreeNode: function(args) {
91 var tnode = new dijit._TreeNode(args);
94 tnode.iconNode.src = args.item.icon[0];
96 //tnode.labelNode.innerHTML = args.label;
99 getIconClass: function (item, opened) {
100 return (!item || this.model.mayHaveChildren(item)) ? (opened ? "dijitFolderOpened" : "dijitFolderClosed") : "feedIcon";
102 getLabelClass: function (item, opened) {
103 return (item.unread == 0) ? "dijitTreeLabel" : "dijitTreeLabel Unread";
105 getRowClass: function (item, opened) {
106 return (!item.error || item.error == '') ? "dijitTreeRow" :
107 "dijitTreeRow Error";
109 getLabel: function(item) {
110 if (item.unread > 0) {
111 return item.name + " (" + item.unread + ")";
116 onOpen: function (item, node) {
117 var id = String(item.id);
118 var cat_id = id.substr(id.indexOf(":")+1);
120 new Ajax.Request("backend.php",
121 { parameters: "backend.php?op=feeds&subop=collapse&cid=" +
122 param_escape(cat_id) + "&mode=1" } );
124 onClose: function (item, node) {
125 var id = String(item.id);
126 var cat_id = id.substr(id.indexOf(":")+1);
128 new Ajax.Request("backend.php",
129 { parameters: "backend.php?op=feeds&subop=collapse&cid=" +
130 param_escape(cat_id) + "&mode=0" } );
133 onClick: function (item, node) {
134 var id = String(item.id);
135 var is_cat = id.match("^CAT:");
136 var feed = id.substr(id.indexOf(":")+1);
137 viewfeed(feed, '', is_cat);
145 $("feeds-holder").appendChild(tree.domNode);
147 var tmph = dojo.connect(tree, 'onLoad', function() {
148 dojo.disconnect(tmph);
149 Element.hide("feedlistLoading");
155 exception_error("updateFeedList", e);
159 function catchupAllFeeds() {
161 var str = __("Mark all articles as read?");
163 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
165 var query_str = "backend.php?op=feeds&subop=catchupAll";
167 notify_progress("Marking all feeds as read...");
169 //console.log("catchupAllFeeds Q=" + query_str);
171 new Ajax.Request("backend.php", {
172 parameters: query_str,
173 onComplete: function(transport) {
174 feedlist_callback2(transport);
182 function viewCurrentFeed(subop) {
184 if (getActiveFeedId() != undefined) {
185 viewfeed(getActiveFeedId(), subop, activeFeedIsCat());
187 return false; // block unneeded form submits
191 if (getInitParam("bw_limit") == "1") return;
194 var date = new Date();
195 var ts = Math.round(date.getTime() / 1000);
197 if (ts - last_scheduled_update > 10 || _force_scheduled_update) {
199 //console.log("timeout()");
201 window.clearTimeout(counter_timeout_id);
203 var query_str = "?op=rpc&subop=getAllCounters&seq=" + next_seq();
207 if (firsttime_update && !navigator.userAgent.match("Opera")) {
208 firsttime_update = false;
214 query_str = query_str + "&omode=" + omode;
216 if (!_force_scheduled_update)
217 query_str = query_str + "&last_article_id=" + getInitParam("last_article_id");
219 //console.log("[timeout]" + query_str);
221 new Ajax.Request("backend.php", {
222 parameters: query_str,
223 onComplete: function(transport) {
224 handle_rpc_reply(transport, !_force_scheduled_update);
225 _force_scheduled_update = false;
228 last_scheduled_update = ts;
232 exception_error("timeout", e);
235 setTimeout("timeout()", 3000);
243 function updateTitle() {
244 var tmp = "Tiny Tiny RSS";
246 if (global_unread > 0) {
247 tmp = tmp + " (" + global_unread + ")";
251 if (global_unread > 0) {
252 window.fluid.dockBadge = global_unread;
254 window.fluid.dockBadge = "";
258 document.title = tmp;
261 function genericSanityCheck() {
262 setCookie("ttrss_test", "TEST");
264 if (getCookie("ttrss_test") != "TEST") {
273 Form.disable("main_toolbar_form");
275 dojo.require("dijit.layout.BorderContainer");
276 dojo.require("dijit.layout.TabContainer");
277 dojo.require("dijit.layout.ContentPane");
278 dojo.require("dijit.Dialog");
279 dojo.require("dijit.form.Button");
280 dojo.require("dojo.data.ItemFileWriteStore");
281 dojo.require("dijit.Tree");
282 dojo.require("dijit.form.Select");
283 dojo.require("dojo.parser");
285 dojo.addOnLoad(function() {
290 if (!genericSanityCheck())
293 var params = "&ua=" + param_escape(navigator.userAgent);
295 loading_set_progress(30);
297 new Ajax.Request("backend.php", {
298 parameters: "backend.php?op=rpc&subop=sanityCheck" + params,
299 onComplete: function(transport) {
300 backend_sanity_check_callback(transport);
304 exception_error("init", e);
308 function init_second_stage() {
312 delCookie("ttrss_test");
314 var toolbar = document.forms["main_toolbar_form"];
316 dropboxSelect(toolbar.view_mode, getInitParam("default_view_mode"));
317 dropboxSelect(toolbar.order_by, getInitParam("default_view_order_by"));
319 feeds_sort_by_unread = getInitParam("feeds_sort_by_unread") == 1;
321 loading_set_progress(60);
323 if (has_local_storage())
324 localStorage.clear();
327 setTimeout("timeout()", 3000);
329 console.log("second stage ok");
332 exception_error("init_second_stage", e);
336 function quickMenuChange() {
337 var chooser = $("quickMenuChooser");
338 var opid = chooser[chooser.selectedIndex].value;
340 chooser.selectedIndex = 0;
344 function quickMenuGo(opid) {
347 if (opid == "qmcPrefs") {
351 if (opid == "qmcTagCloud") {
352 displayDlg("printTagCloud");
355 if (opid == "qmcSearch") {
356 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat(),
358 document.forms['search_form'].query.focus();
363 if (opid == "qmcAddFeed") {
368 if (opid == "qmcEditFeed") {
369 editFeedDlg(getActiveFeedId());
372 if (opid == "qmcRemoveFeed") {
373 var actid = getActiveFeedId();
375 if (activeFeedIsCat()) {
376 alert(__("You can't unsubscribe from the category."));
381 alert(__("Please select some feed first."));
385 var fn = getFeedName(actid);
387 var pr = __("Unsubscribe from %s?").replace("%s", fn);
390 unsubscribeFeed(actid);
396 if (opid == "qmcCatchupAll") {
401 if (opid == "qmcShowOnlyUnread") {
406 if (opid == "qmcAddFilter") {
407 displayDlg('quickAddFilter', '',
408 function () {document.forms['filter_add_form'].reg_exp.focus();});
411 if (opid == "qmcAddLabel") {
415 if (opid == "qmcRescoreFeed") {
416 rescoreCurrentFeed();
419 if (opid == "qmcHKhelp") {
420 //Element.show("hotkey_help_overlay");
421 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
425 exception_error("quickMenuGo", e);
429 function toggleDispRead() {
432 var hide = !(getInitParam("hide_read_feeds") == "1");
434 hideOrShowFeeds(hide);
436 var query = "?op=rpc&subop=setpref&key=HIDE_READ_FEEDS&value=" +
439 new Ajax.Request("backend.php", {
441 onComplete: function(transport) {
442 setInitParam("hide_read_feeds", hide);
446 exception_error("toggleDispRead", e);
450 function parse_runtime_info(elem) {
452 if (!elem || !elem.firstChild) {
453 console.warn("parse_runtime_info: invalid node passed");
457 var data = JSON.parse(elem.firstChild.nodeValue);
459 //console.log("parsing runtime info...");
464 // console.log("RI: " + k + " => " + v);
466 if (k == "new_version_available") {
467 var icon = $("newVersionIcon");
470 icon.style.display = "inline";
472 icon.style.display = "none";
480 if (k == "daemon_is_running" && v != 1) {
481 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
485 if (k == "daemon_stamp_ok" && v != 1) {
486 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
495 function catchupCurrentFeed() {
497 var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
499 var str = __("Mark all articles in %s as read?").replace("%s", fn);
501 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
502 return viewCurrentFeed('MarkAllRead')
506 function catchupFeedInGroup(id) {
510 var title = getFeedName(id);
512 var str = __("Mark all articles in %s as read?").replace("%s", title);
514 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
515 return viewCurrentFeed('MarkAllReadGR:' + id)
519 exception_error("catchupFeedInGroup", e);
523 function editFeedDlg(feed) {
527 alert(__("Please select some feed first."));
531 if ((feed <= 0) || activeFeedIsCat()) {
532 alert(__("You can't edit this kind of feed."));
539 query = "?op=pref-feeds&subop=editfeed&id=" + param_escape(feed);
541 query = "?op=pref-labels&subop=edit&id=" + param_escape(-feed-11);
546 notify_progress("Loading, please wait...", true);
548 new Ajax.Request("backend.php", {
550 onComplete: function(transport) {
551 infobox_callback2(transport);
552 document.forms["edit_feed_form"].title.focus();
556 exception_error("editFeedDlg", e);
560 /* this functions duplicate those of prefs.js feed editor, with
561 some differences because there is no feedlist */
563 function feedEditCancel() {
568 function feedEditSave() {
572 // FIXME: add parameter validation
574 var query = Form.serialize("edit_feed_form");
576 notify_progress("Saving feed...");
578 new Ajax.Request("backend.php", {
580 onComplete: function(transport) {
581 dlg_frefresh_callback(transport);
590 exception_error("feedEditSave (main)", e);
594 function collapse_feedlist() {
597 if (!Element.visible('feeds-holder')) {
598 Element.show('feeds-holder');
599 $("collapse_feeds_btn").innerHTML = "<<";
601 Element.hide('feeds-holder');
602 $("collapse_feeds_btn").innerHTML = ">>";
605 dijit.byId("main").resize();
607 query = "?op=rpc&subop=setpref&key=_COLLAPSED_FEEDLIST&value=true";
608 new Ajax.Request("backend.php", { parameters: query });
611 exception_error("collapse_feedlist", e);
615 function viewModeChanged() {
617 return viewCurrentFeed('')
620 function viewLimitChanged() {
622 return viewCurrentFeed('')
625 /* function adjustArticleScore(id, score) {
628 var pr = prompt(__("Assign score to article:"), score);
630 if (pr != undefined) {
631 var query = "?op=rpc&subop=setScore&id=" + id + "&score=" + pr;
633 new Ajax.Request("backend.php", {
635 onComplete: function(transport) {
641 exception_error("adjustArticleScore", e);
645 function rescoreCurrentFeed() {
647 var actid = getActiveFeedId();
649 if (activeFeedIsCat() || actid < 0) {
650 alert(__("You can't rescore this kind of feed."));
655 alert(__("Please select some feed first."));
659 var fn = getFeedName(actid);
660 var pr = __("Rescore articles in %s?").replace("%s", fn);
663 notify_progress("Rescoring articles...");
665 var query = "?op=pref-feeds&subop=rescore&quiet=1&ids=" + actid;
667 new Ajax.Request("backend.php", {
669 onComplete: function(transport) {
675 function hotkey_handler(e) {
680 var shift_key = false;
682 var cmdline = $('cmdline');
685 shift_key = e.shiftKey;
691 keycode = window.event.keyCode;
696 var keychar = String.fromCharCode(keycode);
698 if (keycode == 27) { // escape
699 if (Element.visible("hotkey_help_overlay")) {
700 Element.hide("hotkey_help_overlay");
702 hotkey_prefix = false;
706 var dialog = dijit.byId("infoBox");
707 var dialog_visible = false;
710 dialog_visible = Element.visible(dialog.domNode);
712 if (dialog_visible || !hotkeys_enabled) {
713 console.log("hotkeys disabled");
717 if (keycode == 16) return; // ignore lone shift
718 if (keycode == 17) return; // ignore lone ctrl
720 if ((keycode == 70 || keycode == 67 || keycode == 71)
723 var date = new Date();
724 var ts = Math.round(date.getTime() / 1000);
726 hotkey_prefix = keycode;
727 hotkey_prefix_pressed = ts;
729 cmdline.innerHTML = keychar;
730 Element.show(cmdline);
732 console.log("KP: PREFIX=" + keycode + " CHAR=" + keychar + " TS=" + ts);
736 if (Element.visible("hotkey_help_overlay")) {
737 Element.hide("hotkey_help_overlay");
742 Element.hide(cmdline);
744 if (!hotkey_prefix) {
746 if ((keycode == 191 || keychar == '?') && shift_key) { // ?
747 if (!Element.visible("hotkey_help_overlay")) {
748 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
750 Element.hide("hotkey_help_overlay");
755 if (keycode == 191 || keychar == '/') { // /
756 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat(),
758 document.forms['search_form'].query.focus();
763 if (keycode == 74) { // j
764 // TODO: move to previous feed
768 if (keycode == 75) { // k
769 // TODO: move to next feed
773 if (shift_key && keycode == 40) { // shift-down
774 catchupRelativeToArticle(1);
778 if (shift_key && keycode == 38) { // shift-up
779 catchupRelativeToArticle(0);
783 if (shift_key && keycode == 78) { // N
788 if (shift_key && keycode == 80) { // P
793 if (keycode == 68 && shift_key) { // shift-D
794 dismissSelectedArticles();
797 if (keycode == 88 && shift_key) { // shift-X
798 dismissReadArticles();
801 if (keycode == 78 || keycode == 40) { // n, down
802 if (typeof moveToPost != 'undefined') {
808 if (keycode == 80 || keycode == 38) { // p, up
809 if (typeof moveToPost != 'undefined') {
815 if (keycode == 83 && shift_key) { // S
816 selectionTogglePublished(undefined, false, true);
820 if (keycode == 83) { // s
821 selectionToggleMarked(undefined, false, true);
826 if (keycode == 85) { // u
827 selectionToggleUnread(undefined, false, true)
831 if (keycode == 84 && shift_key) { // T
832 var id = getActiveArticleId();
834 editArticleTags(id, getActiveFeedId(), isCdmMode());
839 if (keycode == 9) { // tab
840 var id = getArticleUnderPointer();
842 var cb = $("RCHK-" + id);
845 cb.checked = !cb.checked;
846 toggleSelectRowById(cb, "RROW-" + id);
852 if (keycode == 79) { // o
853 if (getActiveArticleId()) {
854 openArticleInNewWindow(getActiveArticleId());
859 if (keycode == 81 && shift_key) { // Q
860 if (typeof catchupAllFeeds != 'undefined') {
866 if (keycode == 88) { // x
867 if (activeFeedIsCat()) {
868 toggleCollapseCat(getActiveFeedId());
875 if (hotkey_prefix == 70) { // f
877 hotkey_prefix = false;
879 if (keycode == 81) { // q
880 if (getActiveFeedId()) {
881 catchupCurrentFeed();
886 if (keycode == 82) { // r
887 if (getActiveFeedId()) {
888 viewfeed(getActiveFeedId(), "ForceUpdate", activeFeedIsCat());
893 if (keycode == 65) { // a
898 if (keycode == 85) { // u
899 if (getActiveFeedId()) {
900 viewfeed(getActiveFeedId(), "ForceUpdate");
905 if (keycode == 69) { // e
906 editFeedDlg(getActiveFeedId());
910 if (keycode == 83) { // s
915 if (keycode == 67 && shift_key) { // C
916 if (typeof catchupAllFeeds != 'undefined') {
922 if (keycode == 67) { // c
923 if (getActiveFeedId()) {
924 catchupCurrentFeed();
929 if (keycode == 87) { // w
930 feeds_sort_by_unread = !feeds_sort_by_unread;
931 return resort_feedlist();
934 if (keycode == 88) { // x
935 reverseHeadlineOrder();
942 if (hotkey_prefix == 67) { // c
943 hotkey_prefix = false;
945 if (keycode == 70) { // f
946 displayDlg('quickAddFilter', '',
947 function () {document.forms['filter_add_form'].reg_exp.focus();});
951 if (keycode == 76) { // l
956 if (keycode == 83) { // s
957 if (typeof collapse_feedlist != 'undefined') {
963 if (keycode == 77) { // m
964 // TODO: sortable feedlist
968 if (keycode == 78) { // n
969 catchupRelativeToArticle(1);
973 if (keycode == 80) { // p
974 catchupRelativeToArticle(0);
983 if (hotkey_prefix == 71) { // g
985 hotkey_prefix = false;
988 if (keycode == 65) { // a
993 if (keycode == 83) { // s
998 if (keycode == 80 && shift_key) { // P
1003 if (keycode == 80) { // p
1008 if (keycode == 70) { // f
1013 if (keycode == 84 && shift_key) { // T
1021 if (hotkey_prefix == 224 || hotkey_prefix == 91) { // f
1022 hotkey_prefix = false;
1026 if (hotkey_prefix) {
1027 console.log("KP: PREFIX=" + hotkey_prefix + " CODE=" + keycode + " CHAR=" + keychar);
1029 console.log("KP: CODE=" + keycode + " CHAR=" + keychar);
1034 exception_error("hotkey_handler", e);
1038 function inPreferences() {
1042 function reverseHeadlineOrder() {
1045 var query_str = "?op=rpc&subop=togglepref&key=REVERSE_HEADLINES";
1047 new Ajax.Request("backend.php", {
1048 parameters: query_str,
1049 onComplete: function(transport) {
1054 exception_error("reverseHeadlineOrder", e);
1058 function showFeedsWithErrors() {
1059 displayDlg('feedUpdateErrors');
1062 function handle_rpc_reply(transport, scheduled_call) {
1064 if (transport.responseXML) {
1066 if (!transport_error_check(transport)) return false;
1068 var seq = transport.responseXML.getElementsByTagName("seq")[0];
1071 seq = seq.firstChild.nodeValue;
1073 if (get_seq() != seq) {
1074 //console.log("[handle_rpc_reply] sequence mismatch: " + seq);
1079 var message = transport.responseXML.getElementsByTagName("message")[0];
1082 message = message.firstChild.nodeValue;
1084 if (message == "UPDATE_COUNTERS") {
1085 console.log("need to refresh counters...");
1086 setInitParam("last_article_id", -1);
1087 _force_scheduled_update = true;
1091 var counters = transport.responseXML.getElementsByTagName("counters")[0];
1094 parse_counters(counters, scheduled_call);
1096 var runtime_info = transport.responseXML.getElementsByTagName("runtime-info")[0];
1099 parse_runtime_info(runtime_info);
1101 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1104 notify_error("Error communicating with server.");
1108 exception_error("handle_rpc_reply", e, transport);
1114 function scheduleFeedUpdate() {
1117 if (!getActiveFeedId()) {
1118 alert(__("Please select some feed first."));
1122 var query = "?op=rpc&subop=scheduleFeedUpdate&id=" +
1123 param_escape(getActiveFeedId()) +
1124 "&is_cat=" + param_escape(activeFeedIsCat());
1128 new Ajax.Request("backend.php", {
1130 onComplete: function(transport) {
1132 if (transport.responseXML) {
1133 var message = transport.responseXML.getElementsByTagName("message")[0];
1136 notify_info(message.firstChild.nodeValue);
1141 notify_error("Error communicating with server.");
1147 exception_error("scheduleFeedUpdate", e);