]>
git.wh0rd.org - tt-rss.git/blob - js/tt-rss.js
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=pref_feeds&method=getfeedtree&mode=2"});
66 var treeModel
= new fox
.FeedStoreModel({
69 "type": init_params
['enable_feed_cats'] == 1 ? "category" : "feed"
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 _force_scheduled_update
= true;
110 /* var menu = new dijit.Menu({id: 'feedMenu'});
112 menu.addChild(new dijit.MenuItem({
113 label: "Simple menu item"
116 // menu.bindDomNode(tree.domNode); */
118 var tmph
= dojo
.connect(dijit
.byId('feedMenu'), '_openMyself', function (event
) {
119 console
.log(dijit
.getEnclosingWidget(event
.target
));
120 dojo
.disconnect(tmph
);
123 $("feeds-holder").appendChild(tree
.domNode
);
125 var tmph
= dojo
.connect(tree
, 'onLoad', function() {
126 dojo
.disconnect(tmph
);
127 Element
.hide("feedlistLoading");
129 tree
.collapseHiddenCats();
133 // var node = dijit.byId("feedTree")._itemNodesMap['FEED:-2'][0].domNode
134 // menu.bindDomNode(node);
136 loading_set_progress(25);
142 exception_error("updateFeedList", e
);
146 function catchupAllFeeds() {
148 var str
= __("Mark all articles as read?");
150 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
152 var query_str
= "backend.php?op=feeds&method=catchupAll";
154 notify_progress("Marking all feeds as read...");
156 //console.log("catchupAllFeeds Q=" + query_str);
158 new Ajax
.Request("backend.php", {
159 parameters
: query_str
,
160 onComplete: function(transport
) {
161 feedlist_callback2(transport
);
169 function viewCurrentFeed(method
) {
171 if (getActiveFeedId() != undefined) {
172 viewfeed(getActiveFeedId(), method
, activeFeedIsCat());
174 return false; // block unneeded form submits
178 if (getInitParam("bw_limit") == "1") return;
181 var date
= new Date();
182 var ts
= Math
.round(date
.getTime() / 1000);
184 if (ts
- last_scheduled_update
> 10 || _force_scheduled_update
) {
186 //console.log("timeout()");
188 window
.clearTimeout(counter_timeout_id
);
190 var query_str
= "?op=rpc&method=getAllCounters&seq=" + next_seq();
194 if (firsttime_update
&& !navigator
.userAgent
.match("Opera")) {
195 firsttime_update
= false;
201 query_str
= query_str
+ "&omode=" + omode
;
203 if (!_force_scheduled_update
)
204 query_str
= query_str
+ "&last_article_id=" + getInitParam("last_article_id");
206 //console.log("[timeout]" + query_str);
208 new Ajax
.Request("backend.php", {
209 parameters
: query_str
,
210 onComplete: function(transport
) {
211 handle_rpc_json(transport
, !_force_scheduled_update
);
212 _force_scheduled_update
= false;
215 last_scheduled_update
= ts
;
219 exception_error("timeout", e
);
222 setTimeout("timeout()", 3000);
226 var query
= "backend.php?op=dlg&method=search¶m=" +
227 param_escape(getActiveFeedId() + ":" + activeFeedIsCat());
229 if (dijit
.byId("searchDlg"))
230 dijit
.byId("searchDlg").destroyRecursive();
232 dialog
= new dijit
.Dialog({
235 style
: "width: 600px",
236 execute: function() {
237 if (this.validate()) {
238 _search_query
= dojo
.objectToQuery(this.attr('value'));
248 function updateTitle() {
249 var tmp
= "Tiny Tiny RSS";
251 if (global_unread
> 0) {
252 tmp
= tmp
+ " (" + global_unread
+ ")";
256 if (global_unread
> 0) {
257 window
.fluid
.dockBadge
= global_unread
;
259 window
.fluid
.dockBadge
= "";
263 document
.title
= tmp
;
266 function genericSanityCheck() {
267 setCookie("ttrss_test", "TEST");
269 if (getCookie("ttrss_test") != "TEST") {
270 return fatalError(2);
278 //dojo.registerModulePath("fox", "../../js/");
280 dojo
.require("fox.FeedTree");
282 if (typeof themeBeforeLayout
== 'function') {
288 if (!genericSanityCheck())
291 loading_set_progress(20);
293 var hasAudio
= !!((myAudioTag
= document
.createElement('audio')).canPlayType
);
295 new Ajax
.Request("backend.php", {
296 parameters
: {op
: "rpc", method
: "sanityCheck", hasAudio
: hasAudio
},
297 onComplete: function(transport
) {
298 backend_sanity_check_callback(transport
);
302 exception_error("init", e
);
306 function init_second_stage() {
309 dojo
.addOnLoad(function() {
313 if (typeof themeAfterLayout
== 'function') {
319 delCookie("ttrss_test");
321 var toolbar
= document
.forms
["main_toolbar_form"];
323 dijit
.getEnclosingWidget(toolbar
.view_mode
).attr('value',
324 getInitParam("default_view_mode"));
326 dijit
.getEnclosingWidget(toolbar
.order_by
).attr('value',
327 getInitParam("default_view_order_by"));
329 feeds_sort_by_unread
= getInitParam("feeds_sort_by_unread") == 1;
331 loading_set_progress(30);
333 // can't use cache_clear() here because viewfeed might not have initialized yet
334 if ('sessionStorage' in window
&& window
['sessionStorage'] !== null)
335 sessionStorage
.clear();
337 console
.log("second stage ok");
340 exception_error("init_second_stage", e
);
344 function quickMenuGo(opid
) {
346 if (opid
== "qmcPrefs") {
350 if (opid
== "qmcTagCloud") {
351 displayDlg("printTagCloud");
354 if (opid
== "qmcTagSelect") {
355 displayDlg("printTagSelect");
358 if (opid
== "qmcSearch") {
363 if (opid
== "qmcAddFeed") {
368 if (opid
== "qmcDigest") {
369 window
.location
.href
= "digest.php";
373 if (opid
== "qmcEditFeed") {
374 if (activeFeedIsCat())
375 alert(__("You can't edit this kind of feed."));
377 editFeed(getActiveFeedId());
381 if (opid
== "qmcRemoveFeed") {
382 var actid
= getActiveFeedId();
384 if (activeFeedIsCat()) {
385 alert(__("You can't unsubscribe from the category."));
390 alert(__("Please select some feed first."));
394 var fn
= getFeedName(actid
);
396 var pr
= __("Unsubscribe from %s?").replace("%s", fn
);
399 unsubscribeFeed(actid
);
405 if (opid
== "qmcCatchupAll") {
410 if (opid
== "qmcShowOnlyUnread") {
415 if (opid
== "qmcAddFilter") {
420 if (opid
== "qmcAddLabel") {
425 if (opid
== "qmcRescoreFeed") {
426 rescoreCurrentFeed();
430 if (opid
== "qmcHKhelp") {
431 new Ajax
.Request("backend.php", {
432 parameters
: "?op=backend&method=help&topic=main",
433 onComplete: function(transport
) {
434 $("hotkey_help_overlay").innerHTML
= transport
.responseText
;
435 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
440 exception_error("quickMenuGo", e
);
444 function toggleDispRead() {
447 var hide
= !(getInitParam("hide_read_feeds") == "1");
449 hideOrShowFeeds(hide
);
451 var query
= "?op=rpc&method=setpref&key=HIDE_READ_FEEDS&value=" +
454 setInitParam("hide_read_feeds", hide
);
456 new Ajax
.Request("backend.php", {
458 onComplete: function(transport
) {
462 exception_error("toggleDispRead", e
);
466 function parse_runtime_info(data
) {
468 //console.log("parsing runtime info...");
473 // console.log("RI: " + k + " => " + v);
475 if (k
== "new_version_available") {
476 var icon
= $("newVersionIcon");
479 icon
.style
.display
= "inline";
481 icon
.style
.display
= "none";
487 if (k
== "daemon_is_running" && v
!= 1) {
488 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
492 if (k
== "daemon_stamp_ok" && v
!= 1) {
493 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
497 if (k
== "max_feed_id" || k
== "num_feeds") {
498 if (init_params
[k
] != v
) {
499 console
.log("feed count changed, need to reload feedlist.");
509 function collapse_feedlist() {
512 if (!Element
.visible('feeds-holder')) {
513 Element
.show('feeds-holder');
514 Element
.show('feeds-holder_splitter');
515 $("collapse_feeds_btn").innerHTML
= "<<";
517 Element
.hide('feeds-holder');
518 Element
.hide('feeds-holder_splitter');
519 $("collapse_feeds_btn").innerHTML
= ">>";
522 dijit
.byId("main").resize();
524 query
= "?op=rpc&method=setpref&key=_COLLAPSED_FEEDLIST&value=true";
525 new Ajax
.Request("backend.php", { parameters
: query
});
528 exception_error("collapse_feedlist", e
);
532 function viewModeChanged() {
533 return viewCurrentFeed('');
536 function viewLimitChanged() {
537 return viewCurrentFeed('');
540 /* function adjustArticleScore(id, score) {
543 var pr = prompt(__("Assign score to article:"), score);
545 if (pr != undefined) {
546 var query = "?op=rpc&method=setScore&id=" + id + "&score=" + pr;
548 new Ajax.Request("backend.php", {
550 onComplete: function(transport) {
556 exception_error("adjustArticleScore", e);
560 function rescoreCurrentFeed() {
562 var actid
= getActiveFeedId();
564 if (activeFeedIsCat() || actid
< 0) {
565 alert(__("You can't rescore this kind of feed."));
570 alert(__("Please select some feed first."));
574 var fn
= getFeedName(actid
);
575 var pr
= __("Rescore articles in %s?").replace("%s", fn
);
578 notify_progress("Rescoring articles...");
580 var query
= "?op=pref-feeds&method=rescore&quiet=1&ids=" + actid
;
582 new Ajax
.Request("backend.php", {
584 onComplete: function(transport
) {
590 function hotkey_handler(e
) {
593 if (e
.target
.nodeName
== "INPUT" || e
.target
.nodeName
== "TEXTAREA") return;
596 var shift_key
= false;
598 var cmdline
= $('cmdline');
601 shift_key
= e
.shiftKey
;
607 keycode
= window
.event
.keyCode
;
612 var keychar
= String
.fromCharCode(keycode
);
614 if (keycode
== 27) { // escape
615 if (Element
.visible("hotkey_help_overlay")) {
616 Element
.hide("hotkey_help_overlay");
618 hotkey_prefix
= false;
621 if (keycode
== 16) return; // ignore lone shift
622 if (keycode
== 17) return; // ignore lone ctrl
624 if ((keycode
== 70 || keycode
== 67 || keycode
== 71 || keycode
== 65)
627 var date
= new Date();
628 var ts
= Math
.round(date
.getTime() / 1000);
630 hotkey_prefix
= keycode
;
631 hotkey_prefix_pressed
= ts
;
633 cmdline
.innerHTML
= keychar
;
634 Element
.show(cmdline
);
636 console
.log("KP: PREFIX=" + keycode
+ " CHAR=" + keychar
+ " TS=" + ts
);
640 if (Element
.visible("hotkey_help_overlay")) {
641 Element
.hide("hotkey_help_overlay");
646 Element
.hide(cmdline
);
648 if (!hotkey_prefix
) {
650 if (keycode
== 27) { // escape
655 if (keycode
== 69) { // e
656 var id
= getActiveArticleId();
660 if ((keycode
== 191 || keychar
== '?') && shift_key
) { // ?
662 new Ajax
.Request("backend.php", {
663 parameters
: "?op=backend&method=help&topic=main",
664 onComplete: function(transport
) {
665 $("hotkey_help_overlay").innerHTML
= transport
.responseText
;
666 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
671 if (keycode
== 191 || keychar
== '/') { // /
676 if (keycode
== 74 && !shift_key
) { // j
677 var rv
= dijit
.byId("feedTree").getPreviousFeed(
678 getActiveFeedId(), activeFeedIsCat());
680 if (rv
) viewfeed(rv
[0], '', rv
[1]);
685 if (keycode
== 75) { // k
686 var rv
= dijit
.byId("feedTree").getNextFeed(
687 getActiveFeedId(), activeFeedIsCat());
689 if (rv
) viewfeed(rv
[0], '', rv
[1]);
694 if (shift_key
&& keycode
== 40) { // shift-down
695 catchupRelativeToArticle(1);
699 if (shift_key
&& keycode
== 38) { // shift-up
700 catchupRelativeToArticle(0);
704 if (shift_key
&& keycode
== 78) { // N
709 if (shift_key
&& keycode
== 80) { // P
714 if (keycode
== 68 && shift_key
) { // shift-D
715 dismissSelectedArticles();
719 if (keycode
== 88 && shift_key
) { // shift-X
720 dismissReadArticles();
724 if (keycode
== 78 || keycode
== 40) { // n, down
725 if (typeof moveToPost
!= 'undefined') {
731 if (keycode
== 80 || keycode
== 38) { // p, up
732 if (typeof moveToPost
!= 'undefined') {
738 if (keycode
== 83 && shift_key
) { // S
739 selectionTogglePublished(undefined, false, true);
743 if (keycode
== 83) { // s
744 selectionToggleMarked(undefined, false, true);
748 if (keycode
== 85) { // u
749 selectionToggleUnread(undefined, false, true);
753 if (keycode
== 84 && shift_key
) { // T
754 var id
= getActiveArticleId();
756 editArticleTags(id
, getActiveFeedId(), isCdmMode());
761 if (keycode
== 9) { // tab
762 var id
= getArticleUnderPointer();
764 var cb
= $("RCHK-" + id
);
767 cb
.checked
= !cb
.checked
;
768 toggleSelectRowById(cb
, "RROW-" + id
);
774 if (keycode
== 79) { // o
775 if (getActiveArticleId()) {
776 openArticleInNewWindow(getActiveArticleId());
781 if (keycode
== 81 && shift_key
) { // Q
782 if (typeof catchupAllFeeds
!= 'undefined') {
788 if (keycode
== 88 && !shift_key
) { // x
789 if (activeFeedIsCat()) {
790 dijit
.byId("feedTree").collapseCat(getActiveFeedId());
798 if (hotkey_prefix
== 65) { // a
799 hotkey_prefix
= false;
801 if (keycode
== 65) { // a
802 selectArticles('all');
806 if (keycode
== 85) { // u
807 selectArticles('unread');
811 if (keycode
== 73) { // i
812 selectArticles('invert');
816 if (keycode
== 78) { // n
817 selectArticles('none');
825 if (hotkey_prefix
== 70) { // f
827 hotkey_prefix
= false;
829 if (keycode
== 81) { // q
830 if (getActiveFeedId()) {
831 catchupCurrentFeed();
836 if (keycode
== 82) { // r
837 if (getActiveFeedId()) {
838 viewfeed(getActiveFeedId(), '', activeFeedIsCat());
843 if (keycode
== 65) { // a
848 if (keycode
== 85) { // u
849 if (getActiveFeedId()) {
850 viewfeed(getActiveFeedId(), '');
855 if (keycode
== 69) { // e
857 if (activeFeedIsCat())
858 alert(__("You can't edit this kind of feed."));
860 editFeed(getActiveFeedId());
866 if (keycode
== 83) { // s
871 if (keycode
== 67 && shift_key
) { // C
872 if (typeof catchupAllFeeds
!= 'undefined') {
878 if (keycode
== 67) { // c
879 if (getActiveFeedId()) {
880 catchupCurrentFeed();
885 if (keycode
== 88) { // x
886 reverseHeadlineOrder();
893 if (hotkey_prefix
== 67) { // c
894 hotkey_prefix
= false;
896 if (keycode
== 70) { // f
901 if (keycode
== 76) { // l
906 if (keycode
== 83) { // s
907 if (typeof collapse_feedlist
!= 'undefined') {
913 if (keycode
== 77) { // m
914 // TODO: sortable feedlist
918 if (keycode
== 78) { // n
919 catchupRelativeToArticle(1);
923 if (keycode
== 80) { // p
924 catchupRelativeToArticle(0);
933 if (hotkey_prefix
== 71) { // g
935 hotkey_prefix
= false;
938 if (keycode
== 65) { // a
943 if (keycode
== 83) { // s
948 if (keycode
== 80 && shift_key
) { // P
953 if (keycode
== 80) { // p
958 if (keycode
== 70) { // f
963 if (keycode
== 84) { // t
964 displayDlg("printTagCloud");
971 if (hotkey_prefix
== 224 || hotkey_prefix
== 91) { // f
972 hotkey_prefix
= false;
977 console
.log("KP: PREFIX=" + hotkey_prefix
+ " CODE=" + keycode
+ " CHAR=" + keychar
);
979 console
.log("KP: CODE=" + keycode
+ " CHAR=" + keychar
);
984 exception_error("hotkey_handler", e
);
988 function inPreferences() {
992 function reverseHeadlineOrder() {
995 var query_str
= "?op=rpc&method=togglepref&key=REVERSE_HEADLINES";
997 new Ajax
.Request("backend.php", {
998 parameters
: query_str
,
999 onComplete: function(transport
) {
1004 exception_error("reverseHeadlineOrder", e
);
1008 function scheduleFeedUpdate(id
, is_cat
) {
1011 id
= getActiveFeedId();
1012 is_cat
= activeFeedIsCat();
1016 alert(__("Please select some feed first."));
1020 var query
= "?op=rpc&method=scheduleFeedUpdate&id=" +
1022 "&is_cat=" + param_escape(is_cat
);
1026 new Ajax
.Request("backend.php", {
1028 onComplete: function(transport
) {
1029 handle_rpc_json(transport
);
1031 var reply
= JSON
.parse(transport
.responseText
);
1032 var message
= reply
['message'];
1035 notify_info(message
);
1043 exception_error("scheduleFeedUpdate", e
);
1047 function newVersionDlg() {
1049 var query
= "backend.php?op=dlg&method=newVersion";
1051 if (dijit
.byId("newVersionDlg"))
1052 dijit
.byId("newVersionDlg").destroyRecursive();
1054 dialog
= new dijit
.Dialog({
1055 id
: "newVersionDlg",
1056 title
: __("New version available!"),
1057 style
: "width: 600px",
1064 exception_error("newVersionDlg", e
);
1068 function handle_rpc_json(transport
, scheduled_call
) {
1070 var reply
= JSON
.parse(transport
.responseText
);
1074 var error
= reply
['error'];
1077 var code
= error
['code'];
1078 var msg
= error
['msg'];
1080 console
.warn("[handle_rpc_json] received fatal error " + code
+ "/" + msg
);
1083 fatalError(code
, msg
);
1088 var seq
= reply
['seq'];
1091 if (get_seq() != seq
) {
1092 console
.log("[handle_rpc_json] sequence mismatch: " + seq
+
1093 " (want: " + get_seq() + ")");
1098 var message
= reply
['message'];
1101 if (message
== "UPDATE_COUNTERS") {
1102 console
.log("need to refresh counters...");
1103 setInitParam("last_article_id", -1);
1104 _force_scheduled_update
= true;
1108 var counters
= reply
['counters'];
1111 parse_counters(counters
, scheduled_call
);
1113 var runtime_info
= reply
['runtime-info'];;
1116 parse_runtime_info(runtime_info
);
1118 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1121 notify_error("Error communicating with server.");
1125 notify_error("Error communicating with server.");
1127 //exception_error("handle_rpc_json", e, transport);