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 hotkey_prefix = false;
7 var hotkey_prefix_pressed = false;
9 var _force_scheduled_update = false;
10 var last_scheduled_update = false;
23 function activeFeedIsCat() {
24 return _active_feed_is_cat;
27 function getActiveFeedId() {
29 //console.log("gAFID: " + _active_feed_id);
30 return _active_feed_id;
32 exception_error("getActiveFeedId", e);
36 function setActiveFeedId(id, is_cat) {
40 if (is_cat != undefined) {
41 _active_feed_is_cat = is_cat;
44 selectFeed(id, is_cat);
47 exception_error("setActiveFeedId", e);
52 function updateFeedList() {
55 // $("feeds-holder").innerHTML = "<div id=\"feedlistLoading\">" +
56 // __("Loading, please wait...") + "</div>";
58 Element.show("feedlistLoading");
60 if (dijit.byId("feedTree")) {
61 dijit.byId("feedTree").destroyRecursive();
64 var store = new dojo.data.ItemFileWriteStore({
65 url: "backend.php?op=feeds"});
67 var treeModel = new fox.FeedStoreModel({
74 childrenAttrs: ["items"]
77 var tree = new fox.FeedTree({
80 onOpen: function (item, node) {
81 var id = String(item.id);
82 var cat_id = id.substr(id.indexOf(":")+1);
84 new Ajax.Request("backend.php",
85 { parameters: "backend.php?op=feeds&subop=collapse&cid=" +
86 param_escape(cat_id) + "&mode=0" } );
88 onClose: function (item, node) {
89 var id = String(item.id);
90 var cat_id = id.substr(id.indexOf(":")+1);
92 new Ajax.Request("backend.php",
93 { parameters: "backend.php?op=feeds&subop=collapse&cid=" +
94 param_escape(cat_id) + "&mode=1" } );
97 onClick: function (item, node) {
98 var id = String(item.id);
99 var is_cat = id.match("^CAT:");
100 var feed = id.substr(id.indexOf(":")+1);
101 viewfeed(feed, '', is_cat);
109 /* var menu = new dijit.Menu({id: 'feedMenu'});
111 menu.addChild(new dijit.MenuItem({
112 label: "Simple menu item"
115 // menu.bindDomNode(tree.domNode); */
117 var tmph = dojo.connect(dijit.byId('feedMenu'), '_openMyself', function (event) {
118 console.log(dijit.getEnclosingWidget(event.target));
119 dojo.disconnect(tmph);
122 $("feeds-holder").appendChild(tree.domNode);
124 var tmph = dojo.connect(tree, 'onLoad', function() {
125 dojo.disconnect(tmph);
126 Element.hide("feedlistLoading");
128 tree.collapseHiddenCats();
132 // var node = dijit.byId("feedTree")._itemNodesMap['FEED:-2'][0].domNode
133 // menu.bindDomNode(node);
135 loading_set_progress(25);
141 exception_error("updateFeedList", e);
145 function catchupAllFeeds() {
147 var str = __("Mark all articles as read?");
149 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
151 var query_str = "backend.php?op=feeds&subop=catchupAll";
153 notify_progress("Marking all feeds as read...");
155 //console.log("catchupAllFeeds Q=" + query_str);
157 new Ajax.Request("backend.php", {
158 parameters: query_str,
159 onComplete: function(transport) {
160 feedlist_callback2(transport);
168 function viewCurrentFeed(subop) {
170 if (getActiveFeedId() != undefined) {
171 viewfeed(getActiveFeedId(), subop, activeFeedIsCat());
173 return false; // block unneeded form submits
177 if (getInitParam("bw_limit") == "1") return;
180 var date = new Date();
181 var ts = Math.round(date.getTime() / 1000);
183 if (ts - last_scheduled_update > 10 || _force_scheduled_update) {
185 //console.log("timeout()");
187 window.clearTimeout(counter_timeout_id);
189 var query_str = "?op=rpc&subop=getAllCounters&seq=" + next_seq();
193 if (firsttime_update && !navigator.userAgent.match("Opera")) {
194 firsttime_update = false;
200 query_str = query_str + "&omode=" + omode;
202 if (!_force_scheduled_update)
203 query_str = query_str + "&last_article_id=" + getInitParam("last_article_id");
205 //console.log("[timeout]" + query_str);
207 new Ajax.Request("backend.php", {
208 parameters: query_str,
209 onComplete: function(transport) {
210 handle_rpc_json(transport, !_force_scheduled_update);
211 _force_scheduled_update = false;
214 last_scheduled_update = ts;
218 exception_error("timeout", e);
221 setTimeout("timeout()", 3000);
225 var query = "backend.php?op=dlg&id=search¶m=" +
226 param_escape(getActiveFeedId() + ":" + activeFeedIsCat());
228 if (dijit.byId("searchDlg"))
229 dijit.byId("searchDlg").destroyRecursive();
231 dialog = new dijit.Dialog({
234 style: "width: 600px",
235 execute: function() {
236 if (this.validate()) {
237 _search_query = dojo.objectToQuery(this.attr('value'));
247 function updateTitle() {
248 var tmp = "Tiny Tiny RSS";
250 if (global_unread > 0) {
251 tmp = tmp + " (" + global_unread + ")";
255 if (global_unread > 0) {
256 window.fluid.dockBadge = global_unread;
258 window.fluid.dockBadge = "";
262 document.title = tmp;
265 function genericSanityCheck() {
266 setCookie("ttrss_test", "TEST");
268 if (getCookie("ttrss_test") != "TEST") {
277 //Form.disable("main_toolbar_form");
279 dojo.require("dijit.layout.BorderContainer");
280 dojo.require("dijit.layout.TabContainer");
281 dojo.require("dijit.layout.ContentPane");
282 dojo.require("dijit.Dialog");
283 dojo.require("dijit.form.Button");
284 dojo.require("dijit.Menu");
285 dojo.require("dojo.data.ItemFileWriteStore");
286 dojo.require("dijit.Tree");
287 dojo.require("dijit.form.Select");
288 dojo.require("dijit.form.TextBox");
289 dojo.require("dijit.form.ValidationTextBox");
290 dojo.require("dijit.form.FilteringSelect");
291 dojo.require("dijit.form.CheckBox");
292 dojo.require("dijit.form.SimpleTextarea");
293 dojo.require("dijit.Toolbar");
294 dojo.require("dijit.ProgressBar");
295 dojo.require("dijit.Menu");
296 dojo.require("dojo.parser");
298 dojo.registerModulePath("fox", "../..");
300 dojo.require("fox.FeedTree");
302 if (typeof themeBeforeLayout == 'function') {
306 dojo.addOnLoad(function() {
310 if (typeof themeAfterLayout == 'function') {
316 if (!genericSanityCheck())
319 loading_set_progress(20);
321 var hasAudio = !!((myAudioTag = document.createElement('audio')).canPlayType);
323 new Ajax.Request("backend.php", {
324 parameters: {op: "rpc", subop: "sanityCheck", hasAudio: hasAudio},
325 onComplete: function(transport) {
326 backend_sanity_check_callback(transport);
330 exception_error("init", e);
334 function init_second_stage() {
338 delCookie("ttrss_test");
340 var toolbar = document.forms["main_toolbar_form"];
342 dijit.getEnclosingWidget(toolbar.view_mode).attr('value',
343 getInitParam("default_view_mode"));
345 dijit.getEnclosingWidget(toolbar.order_by).attr('value',
346 getInitParam("default_view_order_by"));
348 feeds_sort_by_unread = getInitParam("feeds_sort_by_unread") == 1;
350 loading_set_progress(30);
352 if (has_local_storage())
353 sessionStorage.clear();
355 console.log("second stage ok");
358 exception_error("init_second_stage", e);
362 function quickMenuGo(opid) {
364 if (opid == "qmcPrefs") {
368 if (opid == "qmcTagCloud") {
369 displayDlg("printTagCloud");
372 if (opid == "qmcSearch") {
377 if (opid == "qmcAddFeed") {
382 if (opid == "qmcDigest") {
383 window.location.href = "digest.php";
387 if (opid == "qmcEditFeed") {
388 if (activeFeedIsCat())
389 alert(__("You can't edit this kind of feed."));
391 editFeed(getActiveFeedId());
395 if (opid == "qmcRemoveFeed") {
396 var actid = getActiveFeedId();
398 if (activeFeedIsCat()) {
399 alert(__("You can't unsubscribe from the category."));
404 alert(__("Please select some feed first."));
408 var fn = getFeedName(actid);
410 var pr = __("Unsubscribe from %s?").replace("%s", fn);
413 unsubscribeFeed(actid);
419 if (opid == "qmcCatchupAll") {
424 if (opid == "qmcShowOnlyUnread") {
429 if (opid == "qmcAddFilter") {
434 if (opid == "qmcAddLabel") {
439 if (opid == "qmcRescoreFeed") {
440 rescoreCurrentFeed();
444 if (opid == "qmcHKhelp") {
445 //Element.show("hotkey_help_overlay");
446 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
450 exception_error("quickMenuGo", e);
454 function toggleDispRead() {
457 var hide = !(getInitParam("hide_read_feeds") == "1");
459 hideOrShowFeeds(hide);
461 var query = "?op=rpc&subop=setpref&key=HIDE_READ_FEEDS&value=" +
464 setInitParam("hide_read_feeds", hide);
466 new Ajax.Request("backend.php", {
468 onComplete: function(transport) {
472 exception_error("toggleDispRead", e);
476 function parse_runtime_info(data) {
478 //console.log("parsing runtime info...");
483 // console.log("RI: " + k + " => " + v);
485 if (k == "new_version_available") {
486 var icon = $("newVersionIcon");
489 icon.style.display = "inline";
491 icon.style.display = "none";
499 if (k == "daemon_is_running" && v != 1) {
500 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
504 if (k == "daemon_stamp_ok" && v != 1) {
505 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
509 if (k == "max_feed_id" || k == "num_feeds") {
510 if (init_params[k] != v) {
511 console.log("feed count changed, need to reload feedlist.");
521 function catchupCurrentFeed() {
523 var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
525 var str = __("Mark all articles in %s as read?").replace("%s", fn);
527 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
528 return viewCurrentFeed('MarkAllRead')
532 function catchupFeedInGroup(id) {
536 var title = getFeedName(id);
538 var str = __("Mark all articles in %s as read?").replace("%s", title);
540 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
541 return viewCurrentFeed('MarkAllReadGR:' + id)
545 exception_error("catchupFeedInGroup", e);
549 function collapse_feedlist() {
552 if (!Element.visible('feeds-holder')) {
553 Element.show('feeds-holder');
554 Element.show('feeds-holder_splitter');
555 $("collapse_feeds_btn").innerHTML = "<<";
557 Element.hide('feeds-holder');
558 Element.hide('feeds-holder_splitter');
559 $("collapse_feeds_btn").innerHTML = ">>";
562 dijit.byId("main").resize();
564 query = "?op=rpc&subop=setpref&key=_COLLAPSED_FEEDLIST&value=true";
565 new Ajax.Request("backend.php", { parameters: query });
568 exception_error("collapse_feedlist", e);
572 function viewModeChanged() {
574 return viewCurrentFeed('')
577 function viewLimitChanged() {
579 return viewCurrentFeed('')
582 /* function adjustArticleScore(id, score) {
585 var pr = prompt(__("Assign score to article:"), score);
587 if (pr != undefined) {
588 var query = "?op=rpc&subop=setScore&id=" + id + "&score=" + pr;
590 new Ajax.Request("backend.php", {
592 onComplete: function(transport) {
598 exception_error("adjustArticleScore", e);
602 function rescoreCurrentFeed() {
604 var actid = getActiveFeedId();
606 if (activeFeedIsCat() || actid < 0) {
607 alert(__("You can't rescore this kind of feed."));
612 alert(__("Please select some feed first."));
616 var fn = getFeedName(actid);
617 var pr = __("Rescore articles in %s?").replace("%s", fn);
620 notify_progress("Rescoring articles...");
622 var query = "?op=pref-feeds&subop=rescore&quiet=1&ids=" + actid;
624 new Ajax.Request("backend.php", {
626 onComplete: function(transport) {
632 function hotkey_handler(e) {
635 if (e.target.nodeName == "INPUT" || e.target.nodeName == "TEXTAREA") return;
638 var shift_key = false;
640 var cmdline = $('cmdline');
643 shift_key = e.shiftKey;
649 keycode = window.event.keyCode;
654 var keychar = String.fromCharCode(keycode);
656 if (keycode == 27) { // escape
657 if (Element.visible("hotkey_help_overlay")) {
658 Element.hide("hotkey_help_overlay");
660 hotkey_prefix = false;
663 if (keycode == 16) return; // ignore lone shift
664 if (keycode == 17) return; // ignore lone ctrl
666 if ((keycode == 70 || keycode == 67 || keycode == 71)
669 var date = new Date();
670 var ts = Math.round(date.getTime() / 1000);
672 hotkey_prefix = keycode;
673 hotkey_prefix_pressed = ts;
675 cmdline.innerHTML = keychar;
676 Element.show(cmdline);
678 console.log("KP: PREFIX=" + keycode + " CHAR=" + keychar + " TS=" + ts);
682 if (Element.visible("hotkey_help_overlay")) {
683 Element.hide("hotkey_help_overlay");
688 Element.hide(cmdline);
690 if (!hotkey_prefix) {
692 if ((keycode == 191 || keychar == '?') && shift_key) { // ?
693 if (!Element.visible("hotkey_help_overlay")) {
694 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
696 Element.hide("hotkey_help_overlay");
701 if (keycode == 191 || keychar == '/') { // /
706 if (keycode == 74) { // j
707 var rv = dijit.byId("feedTree").model.getPreviousFeed(
708 getActiveFeedId(), activeFeedIsCat());
710 if (rv) viewfeed(rv[0], '', rv[1]);
715 if (keycode == 75) { // k
716 var rv = dijit.byId("feedTree").model.getNextFeed(
717 getActiveFeedId(), activeFeedIsCat());
719 if (rv) viewfeed(rv[0], '', rv[1]);
724 if (shift_key && keycode == 40) { // shift-down
725 catchupRelativeToArticle(1);
729 if (shift_key && keycode == 38) { // shift-up
730 catchupRelativeToArticle(0);
734 if (shift_key && keycode == 78) { // N
739 if (shift_key && keycode == 80) { // P
744 if (keycode == 68 && shift_key) { // shift-D
745 dismissSelectedArticles();
748 if (keycode == 88 && shift_key) { // shift-X
749 dismissReadArticles();
752 if (keycode == 78 || keycode == 40) { // n, down
753 if (typeof moveToPost != 'undefined') {
759 if (keycode == 80 || keycode == 38) { // p, up
760 if (typeof moveToPost != 'undefined') {
766 if (keycode == 83 && shift_key) { // S
767 selectionTogglePublished(undefined, false, true);
771 if (keycode == 83) { // s
772 selectionToggleMarked(undefined, false, true);
777 if (keycode == 85) { // u
778 selectionToggleUnread(undefined, false, true)
782 if (keycode == 84 && shift_key) { // T
783 var id = getActiveArticleId();
785 editArticleTags(id, getActiveFeedId(), isCdmMode());
790 if (keycode == 9) { // tab
791 var id = getArticleUnderPointer();
793 var cb = $("RCHK-" + id);
796 cb.checked = !cb.checked;
797 toggleSelectRowById(cb, "RROW-" + id);
803 if (keycode == 79) { // o
804 if (getActiveArticleId()) {
805 openArticleInNewWindow(getActiveArticleId());
810 if (keycode == 81 && shift_key) { // Q
811 if (typeof catchupAllFeeds != 'undefined') {
817 if (keycode == 88) { // x
818 if (activeFeedIsCat()) {
819 dijit.byId("feedTree").collapseCat(getActiveFeedId());
826 if (hotkey_prefix == 70) { // f
828 hotkey_prefix = false;
830 if (keycode == 81) { // q
831 if (getActiveFeedId()) {
832 catchupCurrentFeed();
837 if (keycode == 82) { // r
838 if (getActiveFeedId()) {
839 viewfeed(getActiveFeedId(), '', activeFeedIsCat());
844 if (keycode == 65) { // a
849 if (keycode == 85) { // u
850 if (getActiveFeedId()) {
851 viewfeed(getActiveFeedId(), '');
856 if (keycode == 69) { // e
858 if (activeFeedIsCat())
859 alert(__("You can't edit this kind of feed."));
861 editFeed(getActiveFeedId());
867 if (keycode == 83) { // s
872 if (keycode == 67 && shift_key) { // C
873 if (typeof catchupAllFeeds != 'undefined') {
879 if (keycode == 67) { // c
880 if (getActiveFeedId()) {
881 catchupCurrentFeed();
886 if (keycode == 87) { // w
887 feeds_sort_by_unread = !feeds_sort_by_unread;
888 return resort_feedlist();
891 if (keycode == 88) { // x
892 reverseHeadlineOrder();
899 if (hotkey_prefix == 67) { // c
900 hotkey_prefix = false;
902 if (keycode == 70) { // f
907 if (keycode == 76) { // l
912 if (keycode == 83) { // s
913 if (typeof collapse_feedlist != 'undefined') {
919 if (keycode == 77) { // m
920 // TODO: sortable feedlist
924 if (keycode == 78) { // n
925 catchupRelativeToArticle(1);
929 if (keycode == 80) { // p
930 catchupRelativeToArticle(0);
939 if (hotkey_prefix == 71) { // g
941 hotkey_prefix = false;
944 if (keycode == 65) { // a
949 if (keycode == 83) { // s
954 if (keycode == 80 && shift_key) { // P
959 if (keycode == 80) { // p
964 if (keycode == 70) { // f
969 if (keycode == 84 && shift_key) { // T
977 if (hotkey_prefix == 224 || hotkey_prefix == 91) { // f
978 hotkey_prefix = false;
983 console.log("KP: PREFIX=" + hotkey_prefix + " CODE=" + keycode + " CHAR=" + keychar);
985 console.log("KP: CODE=" + keycode + " CHAR=" + keychar);
990 exception_error("hotkey_handler", e);
994 function inPreferences() {
998 function reverseHeadlineOrder() {
1001 var query_str = "?op=rpc&subop=togglepref&key=REVERSE_HEADLINES";
1003 new Ajax.Request("backend.php", {
1004 parameters: query_str,
1005 onComplete: function(transport) {
1010 exception_error("reverseHeadlineOrder", e);
1014 function showFeedsWithErrors() {
1015 displayDlg('feedUpdateErrors');
1018 function handle_rpc_reply(transport, scheduled_call) {
1020 if (transport.responseXML) {
1022 if (!transport_error_check(transport)) return false;
1024 var seq = transport.responseXML.getElementsByTagName("seq")[0];
1027 seq = seq.firstChild.nodeValue;
1029 if (get_seq() != seq) {
1030 //console.log("[handle_rpc_reply] sequence mismatch: " + seq);
1035 var message = transport.responseXML.getElementsByTagName("message")[0];
1038 message = message.firstChild.nodeValue;
1040 if (message == "UPDATE_COUNTERS") {
1041 console.log("need to refresh counters...");
1042 setInitParam("last_article_id", -1);
1043 _force_scheduled_update = true;
1047 var counters = transport.responseXML.getElementsByTagName("counters")[0];
1050 parse_counters(JSON.parse(counters.firstChild.nodeValue), scheduled_call);
1052 var runtime_info = transport.responseXML.getElementsByTagName("runtime-info")[0];
1055 parse_runtime_info(JSON.parse(runtime_info.firstChild.nodeValue));
1057 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1060 notify_error("Error communicating with server.");
1064 exception_error("handle_rpc_reply", e, transport);
1070 function scheduleFeedUpdate(id, is_cat) {
1073 id = getActiveFeedId();
1074 is_cat = activeFeedIsCat();
1078 alert(__("Please select some feed first."));
1082 var query = "?op=rpc&subop=scheduleFeedUpdate&id=" +
1084 "&is_cat=" + param_escape(is_cat);
1088 new Ajax.Request("backend.php", {
1090 onComplete: function(transport) {
1091 handle_rpc_json(transport);
1093 var reply = JSON.parse(transport.responseText);
1094 var message = reply['message'];
1097 notify_info(message);
1105 exception_error("scheduleFeedUpdate", e);
1109 function newVersionDlg() {
1111 var query = "backend.php?op=dlg&id=newVersion";
1113 if (dijit.byId("newVersionDlg"))
1114 dijit.byId("newVersionDlg").destroyRecursive();
1116 dialog = new dijit.Dialog({
1117 id: "newVersionDlg",
1118 title: __("New version available!"),
1119 style: "width: 600px",
1126 exception_error("newVersionDlg", e);
1130 function handle_rpc_json(transport, scheduled_call) {
1132 var reply = JSON.parse(transport.responseText);
1136 var error = reply['error'];
1139 var code = error['code'];
1140 var msg = error['msg'];
1142 fatalError(code, msg);
1147 var seq = reply['seq'];
1150 if (get_seq() != seq) {
1151 console.log("[handle_rpc_json] sequence mismatch: " + seq +
1152 " (want: " + get_seq() + ")");
1157 var message = reply['message'];
1160 if (message == "UPDATE_COUNTERS") {
1161 console.log("need to refresh counters...");
1162 setInitParam("last_article_id", -1);
1163 _force_scheduled_update = true;
1167 var counters = reply['counters'];
1170 parse_counters(counters, scheduled_call);
1172 var runtime_info = reply['runtime-info'];;
1175 parse_runtime_info(runtime_info);
1177 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1180 notify_error("Error communicating with server.");
1184 notify_error("Error communicating with server.");
1186 //exception_error("handle_rpc_json", e, transport);