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;
8 var _force_scheduled_update = false;
9 var last_scheduled_update = false;
22 function activeFeedIsCat() {
23 return _active_feed_is_cat;
26 function getActiveFeedId() {
28 //console.log("gAFID: " + _active_feed_id);
29 return _active_feed_id;
31 exception_error("getActiveFeedId", e);
35 function setActiveFeedId(id, is_cat) {
39 if (is_cat != undefined) {
40 _active_feed_is_cat = is_cat;
43 selectFeed(id, is_cat);
46 exception_error("setActiveFeedId", e);
51 function updateFeedList() {
54 // $("feeds-holder").innerHTML = "<div id=\"feedlistLoading\">" +
55 // __("Loading, please wait...") + "</div>";
57 Element.show("feedlistLoading");
59 if (dijit.byId("feedTree")) {
60 dijit.byId("feedTree").destroyRecursive();
63 var store = new dojo.data.ItemFileWriteStore({
64 url: "backend.php?op=feeds"});
66 var treeModel = new fox.FeedStoreModel({
73 childrenAttrs: ["items"]
76 var tree = new fox.FeedTree({
79 onOpen: function (item, node) {
80 var id = String(item.id);
81 var cat_id = id.substr(id.indexOf(":")+1);
83 new Ajax.Request("backend.php",
84 { parameters: "backend.php?op=feeds&method=collapse&cid=" +
85 param_escape(cat_id) + "&mode=0" } );
87 onClose: function (item, node) {
88 var id = String(item.id);
89 var cat_id = id.substr(id.indexOf(":")+1);
91 new Ajax.Request("backend.php",
92 { parameters: "backend.php?op=feeds&method=collapse&cid=" +
93 param_escape(cat_id) + "&mode=1" } );
96 onClick: function (item, node) {
97 var id = String(item.id);
98 var is_cat = id.match("^CAT:");
99 var feed = id.substr(id.indexOf(":")+1);
100 viewfeed(feed, '', is_cat);
108 /* var menu = new dijit.Menu({id: 'feedMenu'});
110 menu.addChild(new dijit.MenuItem({
111 label: "Simple menu item"
114 // menu.bindDomNode(tree.domNode); */
116 var tmph = dojo.connect(dijit.byId('feedMenu'), '_openMyself', function (event) {
117 console.log(dijit.getEnclosingWidget(event.target));
118 dojo.disconnect(tmph);
121 $("feeds-holder").appendChild(tree.domNode);
123 var tmph = dojo.connect(tree, 'onLoad', function() {
124 dojo.disconnect(tmph);
125 Element.hide("feedlistLoading");
127 tree.collapseHiddenCats();
131 // var node = dijit.byId("feedTree")._itemNodesMap['FEED:-2'][0].domNode
132 // menu.bindDomNode(node);
134 loading_set_progress(25);
140 exception_error("updateFeedList", e);
144 function catchupAllFeeds() {
146 var str = __("Mark all articles as read?");
148 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str)) {
150 var query_str = "backend.php?op=feeds&method=catchupAll";
152 notify_progress("Marking all feeds as read...");
154 //console.log("catchupAllFeeds Q=" + query_str);
156 new Ajax.Request("backend.php", {
157 parameters: query_str,
158 onComplete: function(transport) {
159 feedlist_callback2(transport);
167 function viewCurrentFeed(method) {
169 if (getActiveFeedId() != undefined) {
170 viewfeed(getActiveFeedId(), method, activeFeedIsCat());
172 return false; // block unneeded form submits
176 if (getInitParam("bw_limit") == "1") return;
179 var date = new Date();
180 var ts = Math.round(date.getTime() / 1000);
182 if (ts - last_scheduled_update > 10 || _force_scheduled_update) {
184 //console.log("timeout()");
186 window.clearTimeout(counter_timeout_id);
188 var query_str = "?op=rpc&method=getAllCounters&seq=" + next_seq();
192 if (firsttime_update && !navigator.userAgent.match("Opera")) {
193 firsttime_update = false;
199 query_str = query_str + "&omode=" + omode;
201 if (!_force_scheduled_update)
202 query_str = query_str + "&last_article_id=" + getInitParam("last_article_id");
204 //console.log("[timeout]" + query_str);
206 new Ajax.Request("backend.php", {
207 parameters: query_str,
208 onComplete: function(transport) {
209 handle_rpc_json(transport, !_force_scheduled_update);
210 _force_scheduled_update = false;
213 last_scheduled_update = ts;
217 exception_error("timeout", e);
220 setTimeout("timeout()", 3000);
224 var query = "backend.php?op=dlg&method=search¶m=" +
225 param_escape(getActiveFeedId() + ":" + activeFeedIsCat());
227 if (dijit.byId("searchDlg"))
228 dijit.byId("searchDlg").destroyRecursive();
230 dialog = new dijit.Dialog({
233 style: "width: 600px",
234 execute: function() {
235 if (this.validate()) {
236 _search_query = dojo.objectToQuery(this.attr('value'));
246 function updateTitle() {
247 var tmp = "Tiny Tiny RSS";
249 if (global_unread > 0) {
250 tmp = tmp + " (" + global_unread + ")";
254 if (global_unread > 0) {
255 window.fluid.dockBadge = global_unread;
257 window.fluid.dockBadge = "";
261 document.title = tmp;
264 function genericSanityCheck() {
265 setCookie("ttrss_test", "TEST");
267 if (getCookie("ttrss_test") != "TEST") {
268 return fatalError(2);
276 //dojo.registerModulePath("fox", "../../js/");
278 dojo.require("fox.FeedTree");
280 if (typeof themeBeforeLayout == 'function') {
286 if (!genericSanityCheck())
289 loading_set_progress(20);
291 var hasAudio = !!((myAudioTag = document.createElement('audio')).canPlayType);
293 new Ajax.Request("backend.php", {
294 parameters: {op: "rpc", method: "sanityCheck", hasAudio: hasAudio},
295 onComplete: function(transport) {
296 backend_sanity_check_callback(transport);
300 exception_error("init", e);
304 function init_second_stage() {
307 dojo.addOnLoad(function() {
311 if (typeof themeAfterLayout == 'function') {
317 delCookie("ttrss_test");
319 var toolbar = document.forms["main_toolbar_form"];
321 dijit.getEnclosingWidget(toolbar.view_mode).attr('value',
322 getInitParam("default_view_mode"));
324 dijit.getEnclosingWidget(toolbar.order_by).attr('value',
325 getInitParam("default_view_order_by"));
327 feeds_sort_by_unread = getInitParam("feeds_sort_by_unread") == 1;
329 loading_set_progress(30);
331 // can't use cache_clear() here because viewfeed might not have initialized yet
332 if ('sessionStorage' in window && window['sessionStorage'] !== null)
333 sessionStorage.clear();
335 console.log("second stage ok");
338 exception_error("init_second_stage", e);
342 function quickMenuGo(opid) {
344 if (opid == "qmcPrefs") {
348 if (opid == "qmcTagCloud") {
349 displayDlg("printTagCloud");
352 if (opid == "qmcTagSelect") {
353 displayDlg("printTagSelect");
356 if (opid == "qmcSearch") {
361 if (opid == "qmcAddFeed") {
366 if (opid == "qmcDigest") {
367 window.location.href = "digest.php";
371 if (opid == "qmcEditFeed") {
372 if (activeFeedIsCat())
373 alert(__("You can't edit this kind of feed."));
375 editFeed(getActiveFeedId());
379 if (opid == "qmcRemoveFeed") {
380 var actid = getActiveFeedId();
382 if (activeFeedIsCat()) {
383 alert(__("You can't unsubscribe from the category."));
388 alert(__("Please select some feed first."));
392 var fn = getFeedName(actid);
394 var pr = __("Unsubscribe from %s?").replace("%s", fn);
397 unsubscribeFeed(actid);
403 if (opid == "qmcCatchupAll") {
408 if (opid == "qmcShowOnlyUnread") {
413 if (opid == "qmcAddFilter") {
418 if (opid == "qmcAddLabel") {
423 if (opid == "qmcRescoreFeed") {
424 rescoreCurrentFeed();
428 if (opid == "qmcHKhelp") {
429 new Ajax.Request("backend.php", {
430 parameters: "?op=backend&method=help&topic=main",
431 onComplete: function(transport) {
432 $("hotkey_help_overlay").innerHTML = transport.responseText;
433 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
438 exception_error("quickMenuGo", e);
442 function toggleDispRead() {
445 var hide = !(getInitParam("hide_read_feeds") == "1");
447 hideOrShowFeeds(hide);
449 var query = "?op=rpc&method=setpref&key=HIDE_READ_FEEDS&value=" +
452 setInitParam("hide_read_feeds", hide);
454 new Ajax.Request("backend.php", {
456 onComplete: function(transport) {
460 exception_error("toggleDispRead", e);
464 function parse_runtime_info(data) {
466 //console.log("parsing runtime info...");
471 // console.log("RI: " + k + " => " + v);
473 if (k == "new_version_available") {
474 var icon = $("newVersionIcon");
477 icon.style.display = "inline";
479 icon.style.display = "none";
485 if (k == "daemon_is_running" && v != 1) {
486 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
490 if (k == "daemon_stamp_ok" && v != 1) {
491 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
495 if (k == "max_feed_id" || k == "num_feeds") {
496 if (init_params[k] != v) {
497 console.log("feed count changed, need to reload feedlist.");
507 function collapse_feedlist() {
510 if (!Element.visible('feeds-holder')) {
511 Element.show('feeds-holder');
512 Element.show('feeds-holder_splitter');
513 $("collapse_feeds_btn").innerHTML = "<<";
515 Element.hide('feeds-holder');
516 Element.hide('feeds-holder_splitter');
517 $("collapse_feeds_btn").innerHTML = ">>";
520 dijit.byId("main").resize();
522 query = "?op=rpc&method=setpref&key=_COLLAPSED_FEEDLIST&value=true";
523 new Ajax.Request("backend.php", { parameters: query });
526 exception_error("collapse_feedlist", e);
530 function viewModeChanged() {
531 return viewCurrentFeed('');
534 function viewLimitChanged() {
535 return viewCurrentFeed('');
538 /* function adjustArticleScore(id, score) {
541 var pr = prompt(__("Assign score to article:"), score);
543 if (pr != undefined) {
544 var query = "?op=rpc&method=setScore&id=" + id + "&score=" + pr;
546 new Ajax.Request("backend.php", {
548 onComplete: function(transport) {
554 exception_error("adjustArticleScore", e);
558 function rescoreCurrentFeed() {
560 var actid = getActiveFeedId();
562 if (activeFeedIsCat() || actid < 0) {
563 alert(__("You can't rescore this kind of feed."));
568 alert(__("Please select some feed first."));
572 var fn = getFeedName(actid);
573 var pr = __("Rescore articles in %s?").replace("%s", fn);
576 notify_progress("Rescoring articles...");
578 var query = "?op=pref-feeds&method=rescore&quiet=1&ids=" + actid;
580 new Ajax.Request("backend.php", {
582 onComplete: function(transport) {
588 function hotkey_handler(e) {
591 if (e.target.nodeName == "INPUT" || e.target.nodeName == "TEXTAREA") return;
594 var shift_key = false;
596 var cmdline = $('cmdline');
599 shift_key = e.shiftKey;
605 keycode = window.event.keyCode;
610 var keychar = String.fromCharCode(keycode);
612 if (keycode == 27) { // escape
613 if (Element.visible("hotkey_help_overlay")) {
614 Element.hide("hotkey_help_overlay");
616 hotkey_prefix = false;
619 if (keycode == 16) return; // ignore lone shift
620 if (keycode == 17) return; // ignore lone ctrl
622 if ((keycode == 70 || keycode == 67 || keycode == 71 || keycode == 65)
625 var date = new Date();
626 var ts = Math.round(date.getTime() / 1000);
628 hotkey_prefix = keycode;
629 hotkey_prefix_pressed = ts;
631 cmdline.innerHTML = keychar;
632 Element.show(cmdline);
634 console.log("KP: PREFIX=" + keycode + " CHAR=" + keychar + " TS=" + ts);
638 if (Element.visible("hotkey_help_overlay")) {
639 Element.hide("hotkey_help_overlay");
644 Element.hide(cmdline);
646 if (!hotkey_prefix) {
648 if (keycode == 27) { // escape
653 if (keycode == 69) { // e
654 var id = getActiveArticleId();
658 if ((keycode == 191 || keychar == '?') && shift_key) { // ?
660 new Ajax.Request("backend.php", {
661 parameters: "?op=backend&method=help&topic=main",
662 onComplete: function(transport) {
663 $("hotkey_help_overlay").innerHTML = transport.responseText;
664 Effect.Appear("hotkey_help_overlay", {duration : 0.3});
669 if (keycode == 191 || keychar == '/') { // /
674 if (keycode == 74 && !shift_key) { // j
675 var rv = dijit.byId("feedTree").getPreviousFeed(
676 getActiveFeedId(), activeFeedIsCat());
678 if (rv) viewfeed(rv[0], '', rv[1]);
683 if (keycode == 75) { // k
684 var rv = dijit.byId("feedTree").getNextFeed(
685 getActiveFeedId(), activeFeedIsCat());
687 if (rv) viewfeed(rv[0], '', rv[1]);
692 if (shift_key && keycode == 40) { // shift-down
693 catchupRelativeToArticle(1);
697 if (shift_key && keycode == 38) { // shift-up
698 catchupRelativeToArticle(0);
702 if (shift_key && keycode == 78) { // N
707 if (shift_key && keycode == 80) { // P
712 if (keycode == 68 && shift_key) { // shift-D
713 dismissSelectedArticles();
717 if (keycode == 88 && shift_key) { // shift-X
718 dismissReadArticles();
722 if (keycode == 78 || keycode == 40) { // n, down
723 if (typeof moveToPost != 'undefined') {
729 if (keycode == 80 || keycode == 38) { // p, up
730 if (typeof moveToPost != 'undefined') {
736 if (keycode == 83 && shift_key) { // S
737 selectionTogglePublished(undefined, false, true);
741 if (keycode == 83) { // s
742 selectionToggleMarked(undefined, false, true);
746 if (keycode == 85) { // u
747 selectionToggleUnread(undefined, false, true);
751 if (keycode == 84 && shift_key) { // T
752 var id = getActiveArticleId();
754 editArticleTags(id, getActiveFeedId(), isCdmMode());
759 if (keycode == 9) { // tab
760 var id = getArticleUnderPointer();
762 var cb = $("RCHK-" + id);
765 cb.checked = !cb.checked;
766 toggleSelectRowById(cb, "RROW-" + id);
772 if (keycode == 79) { // o
773 if (getActiveArticleId()) {
774 openArticleInNewWindow(getActiveArticleId());
779 if (keycode == 81 && shift_key) { // Q
780 if (typeof catchupAllFeeds != 'undefined') {
786 if (keycode == 88 && !shift_key) { // x
787 if (activeFeedIsCat()) {
788 dijit.byId("feedTree").collapseCat(getActiveFeedId());
796 if (hotkey_prefix == 65) { // a
797 hotkey_prefix = false;
799 if (keycode == 65) { // a
800 selectArticles('all');
804 if (keycode == 85) { // u
805 selectArticles('unread');
809 if (keycode == 73) { // i
810 selectArticles('invert');
814 if (keycode == 78) { // n
815 selectArticles('none');
823 if (hotkey_prefix == 70) { // f
825 hotkey_prefix = false;
827 if (keycode == 81) { // q
828 if (getActiveFeedId()) {
829 catchupCurrentFeed();
834 if (keycode == 82) { // r
835 if (getActiveFeedId()) {
836 viewfeed(getActiveFeedId(), '', activeFeedIsCat());
841 if (keycode == 65) { // a
846 if (keycode == 85) { // u
847 if (getActiveFeedId()) {
848 viewfeed(getActiveFeedId(), '');
853 if (keycode == 69) { // e
855 if (activeFeedIsCat())
856 alert(__("You can't edit this kind of feed."));
858 editFeed(getActiveFeedId());
864 if (keycode == 83) { // s
869 if (keycode == 67 && shift_key) { // C
870 if (typeof catchupAllFeeds != 'undefined') {
876 if (keycode == 67) { // c
877 if (getActiveFeedId()) {
878 catchupCurrentFeed();
883 if (keycode == 88) { // x
884 reverseHeadlineOrder();
891 if (hotkey_prefix == 67) { // c
892 hotkey_prefix = false;
894 if (keycode == 70) { // f
899 if (keycode == 76) { // l
904 if (keycode == 83) { // s
905 if (typeof collapse_feedlist != 'undefined') {
911 if (keycode == 77) { // m
912 // TODO: sortable feedlist
916 if (keycode == 78) { // n
917 catchupRelativeToArticle(1);
921 if (keycode == 80) { // p
922 catchupRelativeToArticle(0);
931 if (hotkey_prefix == 71) { // g
933 hotkey_prefix = false;
936 if (keycode == 65) { // a
941 if (keycode == 83) { // s
946 if (keycode == 80 && shift_key) { // P
951 if (keycode == 80) { // p
956 if (keycode == 70) { // f
961 if (keycode == 84) { // t
962 displayDlg("printTagCloud");
969 if (hotkey_prefix == 224 || hotkey_prefix == 91) { // f
970 hotkey_prefix = false;
975 console.log("KP: PREFIX=" + hotkey_prefix + " CODE=" + keycode + " CHAR=" + keychar);
977 console.log("KP: CODE=" + keycode + " CHAR=" + keychar);
982 exception_error("hotkey_handler", e);
986 function inPreferences() {
990 function reverseHeadlineOrder() {
993 var query_str = "?op=rpc&method=togglepref&key=REVERSE_HEADLINES";
995 new Ajax.Request("backend.php", {
996 parameters: query_str,
997 onComplete: function(transport) {
1002 exception_error("reverseHeadlineOrder", e);
1006 function scheduleFeedUpdate(id, is_cat) {
1009 id = getActiveFeedId();
1010 is_cat = activeFeedIsCat();
1014 alert(__("Please select some feed first."));
1018 var query = "?op=rpc&method=scheduleFeedUpdate&id=" +
1020 "&is_cat=" + param_escape(is_cat);
1024 new Ajax.Request("backend.php", {
1026 onComplete: function(transport) {
1027 handle_rpc_json(transport);
1029 var reply = JSON.parse(transport.responseText);
1030 var message = reply['message'];
1033 notify_info(message);
1041 exception_error("scheduleFeedUpdate", e);
1045 function newVersionDlg() {
1047 var query = "backend.php?op=dlg&method=newVersion";
1049 if (dijit.byId("newVersionDlg"))
1050 dijit.byId("newVersionDlg").destroyRecursive();
1052 dialog = new dijit.Dialog({
1053 id: "newVersionDlg",
1054 title: __("New version available!"),
1055 style: "width: 600px",
1062 exception_error("newVersionDlg", e);
1066 function handle_rpc_json(transport, scheduled_call) {
1068 var reply = JSON.parse(transport.responseText);
1072 var error = reply['error'];
1075 var code = error['code'];
1076 var msg = error['msg'];
1078 console.warn("[handle_rpc_json] received fatal error " + code + "/" + msg);
1081 fatalError(code, msg);
1086 var seq = reply['seq'];
1089 if (get_seq() != seq) {
1090 console.log("[handle_rpc_json] sequence mismatch: " + seq +
1091 " (want: " + get_seq() + ")");
1096 var message = reply['message'];
1099 if (message == "UPDATE_COUNTERS") {
1100 console.log("need to refresh counters...");
1101 setInitParam("last_article_id", -1);
1102 _force_scheduled_update = true;
1106 var counters = reply['counters'];
1109 parse_counters(counters, scheduled_call);
1111 var runtime_info = reply['runtime-info'];;
1114 parse_runtime_info(runtime_info);
1116 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1119 notify_error("Error communicating with server.");
1123 notify_error("Error communicating with server.");
1125 //exception_error("handle_rpc_json", e, transport);