]>
git.wh0rd.org - tt-rss.git/blob - 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;
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") {
269 return fatalError(2);
277 //Form.disable("main_toolbar_form");
279 // Our layer takes care of Dojo dependencies.
281 /* dojo.require("dijit.layout.BorderContainer");
282 dojo.require("dijit.layout.TabContainer");
283 dojo.require("dijit.layout.ContentPane");
284 dojo.require("dijit.Dialog");
285 dojo.require("dijit.form.Button");
286 dojo.require("dijit.Menu");
287 dojo.require("dojo.data.ItemFileWriteStore");
288 dojo.require("dijit.Tree");
289 dojo.require("dijit.form.Select");
290 dojo.require("dijit.form.TextBox");
291 dojo.require("dijit.form.ValidationTextBox");
292 dojo.require("dijit.form.FilteringSelect");
293 dojo.require("dijit.form.CheckBox");
294 dojo.require("dijit.form.SimpleTextarea");
295 dojo.require("dijit.Toolbar");
296 dojo.require("dijit.ProgressBar");
297 dojo.require("dijit.Menu");
298 dojo.require("dojo.parser"); */
300 dojo
.registerModulePath("fox", "../..");
302 dojo
.require("fox.FeedTree");
306 if (typeof themeBeforeLayout
== 'function') {
310 dojo
.addOnLoad(function() {
314 if (typeof themeAfterLayout
== 'function') {
320 if (!genericSanityCheck())
323 loading_set_progress(20);
325 var hasAudio
= !!((myAudioTag
= document
.createElement('audio')).canPlayType
);
327 new Ajax
.Request("backend.php", {
328 parameters
: {op
: "rpc", subop
: "sanityCheck", hasAudio
: hasAudio
},
329 onComplete: function(transport
) {
330 backend_sanity_check_callback(transport
);
334 exception_error("init", e
);
338 function init_second_stage() {
342 delCookie("ttrss_test");
344 var toolbar
= document
.forms
["main_toolbar_form"];
346 dijit
.getEnclosingWidget(toolbar
.view_mode
).attr('value',
347 getInitParam("default_view_mode"));
349 dijit
.getEnclosingWidget(toolbar
.order_by
).attr('value',
350 getInitParam("default_view_order_by"));
352 feeds_sort_by_unread
= getInitParam("feeds_sort_by_unread") == 1;
354 loading_set_progress(30);
356 if (has_local_storage())
357 sessionStorage
.clear();
359 console
.log("second stage ok");
362 exception_error("init_second_stage", e
);
366 function quickMenuGo(opid
) {
368 if (opid
== "qmcPrefs") {
372 if (opid
== "qmcTagCloud") {
373 displayDlg("printTagCloud");
376 if (opid
== "qmcSearch") {
381 if (opid
== "qmcAddFeed") {
386 if (opid
== "qmcDigest") {
387 window
.location
.href
= "digest.php";
391 if (opid
== "qmcEditFeed") {
392 if (activeFeedIsCat())
393 alert(__("You can't edit this kind of feed."));
395 editFeed(getActiveFeedId());
399 if (opid
== "qmcRemoveFeed") {
400 var actid
= getActiveFeedId();
402 if (activeFeedIsCat()) {
403 alert(__("You can't unsubscribe from the category."));
408 alert(__("Please select some feed first."));
412 var fn
= getFeedName(actid
);
414 var pr
= __("Unsubscribe from %s?").replace("%s", fn
);
417 unsubscribeFeed(actid
);
423 if (opid
== "qmcCatchupAll") {
428 if (opid
== "qmcShowOnlyUnread") {
433 if (opid
== "qmcAddFilter") {
438 if (opid
== "qmcAddLabel") {
443 if (opid
== "qmcRescoreFeed") {
444 rescoreCurrentFeed();
448 if (opid
== "qmcHKhelp") {
449 //Element.show("hotkey_help_overlay");
450 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
454 exception_error("quickMenuGo", e
);
458 function toggleDispRead() {
461 var hide
= !(getInitParam("hide_read_feeds") == "1");
463 hideOrShowFeeds(hide
);
465 var query
= "?op=rpc&subop=setpref&key=HIDE_READ_FEEDS&value=" +
468 setInitParam("hide_read_feeds", hide
);
470 new Ajax
.Request("backend.php", {
472 onComplete: function(transport
) {
476 exception_error("toggleDispRead", e
);
480 function parse_runtime_info(data
) {
482 //console.log("parsing runtime info...");
487 // console.log("RI: " + k + " => " + v);
489 if (k
== "new_version_available") {
490 var icon
= $("newVersionIcon");
493 icon
.style
.display
= "inline";
495 icon
.style
.display
= "none";
503 if (k
== "daemon_is_running" && v
!= 1) {
504 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
508 if (k
== "daemon_stamp_ok" && v
!= 1) {
509 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
513 if (k
== "max_feed_id" || k
== "num_feeds") {
514 if (init_params
[k
] != v
) {
515 console
.log("feed count changed, need to reload feedlist.");
525 function catchupCurrentFeed() {
527 var fn
= getFeedName(getActiveFeedId(), activeFeedIsCat());
529 var str
= __("Mark all articles in %s as read?").replace("%s", fn
);
531 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
532 return viewCurrentFeed('MarkAllRead')
536 function catchupFeedInGroup(id
) {
540 var title
= getFeedName(id
);
542 var str
= __("Mark all articles in %s as read?").replace("%s", title
);
544 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
545 return viewCurrentFeed('MarkAllReadGR:' + id
)
549 exception_error("catchupFeedInGroup", e
);
553 function collapse_feedlist() {
556 if (!Element
.visible('feeds-holder')) {
557 Element
.show('feeds-holder');
558 Element
.show('feeds-holder_splitter');
559 $("collapse_feeds_btn").innerHTML
= "<<";
561 Element
.hide('feeds-holder');
562 Element
.hide('feeds-holder_splitter');
563 $("collapse_feeds_btn").innerHTML
= ">>";
566 dijit
.byId("main").resize();
568 query
= "?op=rpc&subop=setpref&key=_COLLAPSED_FEEDLIST&value=true";
569 new Ajax
.Request("backend.php", { parameters
: query
});
572 exception_error("collapse_feedlist", e
);
576 function viewModeChanged() {
578 return viewCurrentFeed('')
581 function viewLimitChanged() {
583 return viewCurrentFeed('')
586 /* function adjustArticleScore(id, score) {
589 var pr = prompt(__("Assign score to article:"), score);
591 if (pr != undefined) {
592 var query = "?op=rpc&subop=setScore&id=" + id + "&score=" + pr;
594 new Ajax.Request("backend.php", {
596 onComplete: function(transport) {
602 exception_error("adjustArticleScore", e);
606 function rescoreCurrentFeed() {
608 var actid
= getActiveFeedId();
610 if (activeFeedIsCat() || actid
< 0) {
611 alert(__("You can't rescore this kind of feed."));
616 alert(__("Please select some feed first."));
620 var fn
= getFeedName(actid
);
621 var pr
= __("Rescore articles in %s?").replace("%s", fn
);
624 notify_progress("Rescoring articles...");
626 var query
= "?op=pref-feeds&subop=rescore&quiet=1&ids=" + actid
;
628 new Ajax
.Request("backend.php", {
630 onComplete: function(transport
) {
636 function hotkey_handler(e
) {
639 if (e
.target
.nodeName
== "INPUT" || e
.target
.nodeName
== "TEXTAREA") return;
642 var shift_key
= false;
644 var cmdline
= $('cmdline');
647 shift_key
= e
.shiftKey
;
653 keycode
= window
.event
.keyCode
;
658 var keychar
= String
.fromCharCode(keycode
);
660 if (keycode
== 27) { // escape
661 if (Element
.visible("hotkey_help_overlay")) {
662 Element
.hide("hotkey_help_overlay");
664 hotkey_prefix
= false;
667 if (keycode
== 16) return; // ignore lone shift
668 if (keycode
== 17) return; // ignore lone ctrl
670 if ((keycode
== 70 || keycode
== 67 || keycode
== 71)
673 var date
= new Date();
674 var ts
= Math
.round(date
.getTime() / 1000);
676 hotkey_prefix
= keycode
;
677 hotkey_prefix_pressed
= ts
;
679 cmdline
.innerHTML
= keychar
;
680 Element
.show(cmdline
);
682 console
.log("KP: PREFIX=" + keycode
+ " CHAR=" + keychar
+ " TS=" + ts
);
686 if (Element
.visible("hotkey_help_overlay")) {
687 Element
.hide("hotkey_help_overlay");
692 Element
.hide(cmdline
);
694 if (!hotkey_prefix
) {
696 if ((keycode
== 191 || keychar
== '?') && shift_key
) { // ?
697 if (!Element
.visible("hotkey_help_overlay")) {
698 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
700 Element
.hide("hotkey_help_overlay");
705 if (keycode
== 191 || keychar
== '/') { // /
710 if (keycode
== 74) { // j
711 var rv
= dijit
.byId("feedTree").getPreviousFeed(
712 getActiveFeedId(), activeFeedIsCat());
714 if (rv
) viewfeed(rv
[0], '', rv
[1]);
719 if (keycode
== 75) { // k
720 var rv
= dijit
.byId("feedTree").getNextFeed(
721 getActiveFeedId(), activeFeedIsCat());
723 if (rv
) viewfeed(rv
[0], '', rv
[1]);
728 if (shift_key
&& keycode
== 40) { // shift-down
729 catchupRelativeToArticle(1);
733 if (shift_key
&& keycode
== 38) { // shift-up
734 catchupRelativeToArticle(0);
738 if (shift_key
&& keycode
== 78) { // N
743 if (shift_key
&& keycode
== 80) { // P
748 if (keycode
== 68 && shift_key
) { // shift-D
749 dismissSelectedArticles();
753 if (keycode
== 88 && shift_key
) { // shift-X
754 dismissReadArticles();
758 if (keycode
== 78 || keycode
== 40) { // n, down
759 if (typeof moveToPost
!= 'undefined') {
765 if (keycode
== 80 || keycode
== 38) { // p, up
766 if (typeof moveToPost
!= 'undefined') {
772 if (keycode
== 83 && shift_key
) { // S
773 selectionTogglePublished(undefined, false, true);
777 if (keycode
== 83) { // s
778 selectionToggleMarked(undefined, false, true);
783 if (keycode
== 85) { // u
784 selectionToggleUnread(undefined, false, true)
788 if (keycode
== 84 && shift_key
) { // T
789 var id
= getActiveArticleId();
791 editArticleTags(id
, getActiveFeedId(), isCdmMode());
796 if (keycode
== 9) { // tab
797 var id
= getArticleUnderPointer();
799 var cb
= $("RCHK-" + id
);
802 cb
.checked
= !cb
.checked
;
803 toggleSelectRowById(cb
, "RROW-" + id
);
809 if (keycode
== 79) { // o
810 if (getActiveArticleId()) {
811 openArticleInNewWindow(getActiveArticleId());
816 if (keycode
== 81 && shift_key
) { // Q
817 if (typeof catchupAllFeeds
!= 'undefined') {
823 if (keycode
== 88 && !shift_key
) { // x
824 if (activeFeedIsCat()) {
825 dijit
.byId("feedTree").collapseCat(getActiveFeedId());
833 if (hotkey_prefix
== 70) { // f
835 hotkey_prefix
= false;
837 if (keycode
== 81) { // q
838 if (getActiveFeedId()) {
839 catchupCurrentFeed();
844 if (keycode
== 82) { // r
845 if (getActiveFeedId()) {
846 viewfeed(getActiveFeedId(), '', activeFeedIsCat());
851 if (keycode
== 65) { // a
856 if (keycode
== 85) { // u
857 if (getActiveFeedId()) {
858 viewfeed(getActiveFeedId(), '');
863 if (keycode
== 69) { // e
865 if (activeFeedIsCat())
866 alert(__("You can't edit this kind of feed."));
868 editFeed(getActiveFeedId());
874 if (keycode
== 83) { // s
879 if (keycode
== 67 && shift_key
) { // C
880 if (typeof catchupAllFeeds
!= 'undefined') {
886 if (keycode
== 67) { // c
887 if (getActiveFeedId()) {
888 catchupCurrentFeed();
893 if (keycode
== 87) { // w
894 feeds_sort_by_unread
= !feeds_sort_by_unread
;
895 return resort_feedlist();
898 if (keycode
== 88) { // x
899 reverseHeadlineOrder();
906 if (hotkey_prefix
== 67) { // c
907 hotkey_prefix
= false;
909 if (keycode
== 70) { // f
914 if (keycode
== 76) { // l
919 if (keycode
== 83) { // s
920 if (typeof collapse_feedlist
!= 'undefined') {
926 if (keycode
== 77) { // m
927 // TODO: sortable feedlist
931 if (keycode
== 78) { // n
932 catchupRelativeToArticle(1);
936 if (keycode
== 80) { // p
937 catchupRelativeToArticle(0);
946 if (hotkey_prefix
== 71) { // g
948 hotkey_prefix
= false;
951 if (keycode
== 65) { // a
956 if (keycode
== 83) { // s
961 if (keycode
== 80 && shift_key
) { // P
966 if (keycode
== 80) { // p
971 if (keycode
== 70) { // f
976 if (keycode
== 84 && shift_key
) { // T
984 if (hotkey_prefix
== 224 || hotkey_prefix
== 91) { // f
985 hotkey_prefix
= false;
990 console
.log("KP: PREFIX=" + hotkey_prefix
+ " CODE=" + keycode
+ " CHAR=" + keychar
);
992 console
.log("KP: CODE=" + keycode
+ " CHAR=" + keychar
);
997 exception_error("hotkey_handler", e
);
1001 function inPreferences() {
1005 function reverseHeadlineOrder() {
1008 var query_str
= "?op=rpc&subop=togglepref&key=REVERSE_HEADLINES";
1010 new Ajax
.Request("backend.php", {
1011 parameters
: query_str
,
1012 onComplete: function(transport
) {
1017 exception_error("reverseHeadlineOrder", e
);
1021 function showFeedsWithErrors() {
1022 displayDlg('feedUpdateErrors');
1025 function handle_rpc_reply(transport
, scheduled_call
) {
1027 if (transport
.responseXML
) {
1029 if (!transport_error_check(transport
)) return false;
1031 var seq
= transport
.responseXML
.getElementsByTagName("seq")[0];
1034 seq
= seq
.firstChild
.nodeValue
;
1036 if (get_seq() != seq
) {
1037 //console.log("[handle_rpc_reply] sequence mismatch: " + seq);
1042 var message
= transport
.responseXML
.getElementsByTagName("message")[0];
1045 message
= message
.firstChild
.nodeValue
;
1047 if (message
== "UPDATE_COUNTERS") {
1048 console
.log("need to refresh counters...");
1049 setInitParam("last_article_id", -1);
1050 _force_scheduled_update
= true;
1054 var counters
= transport
.responseXML
.getElementsByTagName("counters")[0];
1057 parse_counters(JSON
.parse(counters
.firstChild
.nodeValue
), scheduled_call
);
1059 var runtime_info
= transport
.responseXML
.getElementsByTagName("runtime-info")[0];
1062 parse_runtime_info(JSON
.parse(runtime_info
.firstChild
.nodeValue
));
1064 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1067 notify_error("Error communicating with server.");
1071 exception_error("handle_rpc_reply", e
, transport
);
1077 function scheduleFeedUpdate(id
, is_cat
) {
1080 id
= getActiveFeedId();
1081 is_cat
= activeFeedIsCat();
1085 alert(__("Please select some feed first."));
1089 var query
= "?op=rpc&subop=scheduleFeedUpdate&id=" +
1091 "&is_cat=" + param_escape(is_cat
);
1095 new Ajax
.Request("backend.php", {
1097 onComplete: function(transport
) {
1098 handle_rpc_json(transport
);
1100 var reply
= JSON
.parse(transport
.responseText
);
1101 var message
= reply
['message'];
1104 notify_info(message
);
1112 exception_error("scheduleFeedUpdate", e
);
1116 function newVersionDlg() {
1118 var query
= "backend.php?op=dlg&id=newVersion";
1120 if (dijit
.byId("newVersionDlg"))
1121 dijit
.byId("newVersionDlg").destroyRecursive();
1123 dialog
= new dijit
.Dialog({
1124 id
: "newVersionDlg",
1125 title
: __("New version available!"),
1126 style
: "width: 600px",
1133 exception_error("newVersionDlg", e
);
1137 function handle_rpc_json(transport
, scheduled_call
) {
1139 var reply
= JSON
.parse(transport
.responseText
);
1143 var error
= reply
['error'];
1146 var code
= error
['code'];
1147 var msg
= error
['msg'];
1149 console
.warn("[handle_rpc_json] received fatal error " + code
+ "/" + msg
);
1152 fatalError(code
, msg
);
1157 var seq
= reply
['seq'];
1160 if (get_seq() != seq
) {
1161 console
.log("[handle_rpc_json] sequence mismatch: " + seq
+
1162 " (want: " + get_seq() + ")");
1167 var message
= reply
['message'];
1170 if (message
== "UPDATE_COUNTERS") {
1171 console
.log("need to refresh counters...");
1172 setInitParam("last_article_id", -1);
1173 _force_scheduled_update
= true;
1177 var counters
= reply
['counters'];
1180 parse_counters(counters
, scheduled_call
);
1182 var runtime_info
= reply
['runtime-info'];;
1185 parse_runtime_info(runtime_info
);
1187 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1190 notify_error("Error communicating with server.");
1194 notify_error("Error communicating with server.");
1196 //exception_error("handle_rpc_json", e, transport);