]>
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 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;
24 function activeFeedIsCat() {
25 return _active_feed_is_cat
;
28 function getActiveFeedId() {
30 //console.log("gAFID: " + _active_feed_id);
31 return _active_feed_id
;
33 exception_error("getActiveFeedId", e
);
37 function setActiveFeedId(id
, is_cat
) {
41 if (is_cat
!= undefined) {
42 _active_feed_is_cat
= is_cat
;
45 selectFeed(id
, is_cat
);
48 exception_error("setActiveFeedId", e
);
53 function dlg_frefresh_callback(transport
, deleted_feed
) {
54 if (getActiveFeedId() == deleted_feed
) {
55 setTimeout("viewfeed(-5)", 100);
58 setTimeout('updateFeedList()', 50);
62 function updateFeedList() {
65 // $("feeds-holder").innerHTML = "<div id=\"feedlistLoading\">" +
66 // __("Loading, please wait...") + "</div>";
68 Element
.show("feedlistLoading");
70 if (dijit
.byId("feedTree")) {
71 dijit
.byId("feedTree").destroyRecursive();
74 var store
= new dojo
.data
.ItemFileWriteStore({
75 url
: "backend.php?op=feeds"});
77 var treeModel
= new fox
.FeedStoreModel({
84 childrenAttrs
: ["items"]
87 var tree
= new fox
.FeedTree({
89 onOpen: function (item
, node
) {
90 var id
= String(item
.id
);
91 var cat_id
= id
.substr(id
.indexOf(":")+1);
93 new Ajax
.Request("backend.php",
94 { parameters
: "backend.php?op=feeds&subop=collapse&cid=" +
95 param_escape(cat_id
) + "&mode=0" } );
97 onClose: function (item
, node
) {
98 var id
= String(item
.id
);
99 var cat_id
= id
.substr(id
.indexOf(":")+1);
101 new Ajax
.Request("backend.php",
102 { parameters
: "backend.php?op=feeds&subop=collapse&cid=" +
103 param_escape(cat_id
) + "&mode=1" } );
106 onClick: function (item
, node
) {
107 var id
= String(item
.id
);
108 var is_cat
= id
.match("^CAT:");
109 var feed
= id
.substr(id
.indexOf(":")+1);
110 viewfeed(feed
, '', is_cat
);
118 /* var menu = new dijit.Menu({id: 'feedMenu'});
120 menu.addChild(new dijit.MenuItem({
121 label: "Simple menu item"
124 // menu.bindDomNode(tree.domNode); */
126 var tmph
= dojo
.connect(dijit
.byId('feedMenu'), '_openMyself', function (event
) {
127 console
.log(dijit
.getEnclosingWidget(event
.target
));
128 dojo
.disconnect(tmph
);
131 $("feeds-holder").appendChild(tree
.domNode
);
133 var tmph
= dojo
.connect(tree
, 'onLoad', function() {
134 dojo
.disconnect(tmph
);
135 Element
.hide("feedlistLoading");
137 tree
.collapseHiddenCats();
141 // var node = dijit.byId("feedTree")._itemNodesMap['FEED:-2'][0].domNode
142 // menu.bindDomNode(node);
144 loading_set_progress(25);
150 exception_error("updateFeedList", e
);
154 function catchupAllFeeds() {
156 var str
= __("Mark all articles as read?");
158 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
160 var query_str
= "backend.php?op=feeds&subop=catchupAll";
162 notify_progress("Marking all feeds as read...");
164 //console.log("catchupAllFeeds Q=" + query_str);
166 new Ajax
.Request("backend.php", {
167 parameters
: query_str
,
168 onComplete: function(transport
) {
169 feedlist_callback2(transport
);
177 function viewCurrentFeed(subop
) {
179 if (getActiveFeedId() != undefined) {
180 viewfeed(getActiveFeedId(), subop
, activeFeedIsCat());
182 return false; // block unneeded form submits
186 if (getInitParam("bw_limit") == "1") return;
189 var date
= new Date();
190 var ts
= Math
.round(date
.getTime() / 1000);
192 if (ts
- last_scheduled_update
> 10 || _force_scheduled_update
) {
194 //console.log("timeout()");
196 window
.clearTimeout(counter_timeout_id
);
198 var query_str
= "?op=rpc&subop=getAllCounters&seq=" + next_seq();
202 if (firsttime_update
&& !navigator
.userAgent
.match("Opera")) {
203 firsttime_update
= false;
209 query_str
= query_str
+ "&omode=" + omode
;
211 if (!_force_scheduled_update
)
212 query_str
= query_str
+ "&last_article_id=" + getInitParam("last_article_id");
214 //console.log("[timeout]" + query_str);
216 new Ajax
.Request("backend.php", {
217 parameters
: query_str
,
218 onComplete: function(transport
) {
219 handle_rpc_reply(transport
, !_force_scheduled_update
);
220 _force_scheduled_update
= false;
223 last_scheduled_update
= ts
;
227 exception_error("timeout", e
);
230 setTimeout("timeout()", 3000);
234 var query
= "backend.php?op=dlg&id=search¶m=" +
235 param_escape(getActiveFeedId() + ":" + activeFeedIsCat());
237 if (dijit
.byId("searchDlg"))
238 dijit
.byId("searchDlg").destroyRecursive();
240 dialog
= new dijit
.Dialog({
243 style
: "width: 600px",
244 execute: function() {
245 if (this.validate()) {
246 _search_query
= dojo
.objectToQuery(this.attr('value'));
256 function updateTitle() {
257 var tmp
= "Tiny Tiny RSS";
259 if (global_unread
> 0) {
260 tmp
= tmp
+ " (" + global_unread
+ ")";
264 if (global_unread
> 0) {
265 window
.fluid
.dockBadge
= global_unread
;
267 window
.fluid
.dockBadge
= "";
271 document
.title
= tmp
;
274 function genericSanityCheck() {
275 setCookie("ttrss_test", "TEST");
277 if (getCookie("ttrss_test") != "TEST") {
286 //Form.disable("main_toolbar_form");
288 dojo
.require("dijit.layout.BorderContainer");
289 dojo
.require("dijit.layout.TabContainer");
290 dojo
.require("dijit.layout.ContentPane");
291 dojo
.require("dijit.Dialog");
292 dojo
.require("dijit.form.Button");
293 dojo
.require("dijit.Menu");
294 dojo
.require("dojo.data.ItemFileWriteStore");
295 dojo
.require("dijit.Tree");
296 dojo
.require("dijit.form.Select");
297 dojo
.require("dijit.form.TextBox");
298 dojo
.require("dijit.form.ValidationTextBox");
299 dojo
.require("dijit.form.FilteringSelect");
300 dojo
.require("dijit.form.CheckBox");
301 dojo
.require("dijit.form.SimpleTextarea");
302 dojo
.require("dijit.Toolbar");
303 dojo
.require("dijit.ProgressBar");
304 dojo
.require("dijit.Menu");
305 dojo
.require("dojo.parser");
307 dojo
.registerModulePath("fox", "../..");
309 dojo
.require("fox.FeedTree");
311 if (typeof themeBeforeLayout
== 'function') {
315 dojo
.addOnLoad(function() {
319 if (typeof themeAfterLayout
== 'function') {
325 if (!genericSanityCheck())
328 loading_set_progress(20);
330 new Ajax
.Request("backend.php", {
331 parameters
: {op
: "rpc", subop
: "sanityCheck"},
332 onComplete: function(transport
) {
333 backend_sanity_check_callback(transport
);
337 exception_error("init", e
);
341 function init_second_stage() {
345 delCookie("ttrss_test");
347 var toolbar
= document
.forms
["main_toolbar_form"];
349 dijit
.getEnclosingWidget(toolbar
.view_mode
).attr('value',
350 getInitParam("default_view_mode"));
352 dijit
.getEnclosingWidget(toolbar
.order_by
).attr('value',
353 getInitParam("default_view_order_by"));
355 feeds_sort_by_unread
= getInitParam("feeds_sort_by_unread") == 1;
357 loading_set_progress(30);
359 if (has_local_storage())
360 localStorage
.clear();
362 console
.log("second stage ok");
365 exception_error("init_second_stage", e
);
369 function quickMenuChange(elem
) {
370 quickMenuGo(elem
.value
);
371 elem
.attr('value', 'qmcDefault');
374 function quickMenuGo(opid
) {
377 if (opid
== "qmcPrefs") {
381 if (opid
== "qmcTagCloud") {
382 displayDlg("printTagCloud");
385 if (opid
== "qmcSearch") {
390 if (opid
== "qmcAddFeed") {
395 if (opid
== "qmcEditFeed") {
396 if (activeFeedIsCat())
397 alert(__("You can't edit this kind of feed."));
399 editFeed(getActiveFeedId());
403 if (opid
== "qmcRemoveFeed") {
404 var actid
= getActiveFeedId();
406 if (activeFeedIsCat()) {
407 alert(__("You can't unsubscribe from the category."));
412 alert(__("Please select some feed first."));
416 var fn
= getFeedName(actid
);
418 var pr
= __("Unsubscribe from %s?").replace("%s", fn
);
421 unsubscribeFeed(actid
);
427 if (opid
== "qmcCatchupAll") {
432 if (opid
== "qmcShowOnlyUnread") {
437 if (opid
== "qmcAddFilter") {
442 if (opid
== "qmcAddLabel") {
447 if (opid
== "qmcRescoreFeed") {
448 rescoreCurrentFeed();
452 if (opid
== "qmcHKhelp") {
453 //Element.show("hotkey_help_overlay");
454 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
458 exception_error("quickMenuGo", e
);
462 function toggleDispRead() {
465 var hide
= !(getInitParam("hide_read_feeds") == "1");
467 hideOrShowFeeds(hide
);
469 var query
= "?op=rpc&subop=setpref&key=HIDE_READ_FEEDS&value=" +
472 setInitParam("hide_read_feeds", hide
);
474 new Ajax
.Request("backend.php", {
476 onComplete: function(transport
) {
480 exception_error("toggleDispRead", e
);
484 function parse_runtime_info(elem
) {
486 if (!elem
|| !elem
.firstChild
) {
487 console
.warn("parse_runtime_info: invalid node passed");
491 var data
= JSON
.parse(elem
.firstChild
.nodeValue
);
493 //console.log("parsing runtime info...");
498 // console.log("RI: " + k + " => " + v);
500 if (k
== "new_version_available") {
501 var icon
= $("newVersionIcon");
504 icon
.style
.display
= "inline";
506 icon
.style
.display
= "none";
514 if (k
== "daemon_is_running" && v
!= 1) {
515 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
519 if (k
== "daemon_stamp_ok" && v
!= 1) {
520 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
529 function catchupCurrentFeed() {
531 var fn
= getFeedName(getActiveFeedId(), activeFeedIsCat());
533 var str
= __("Mark all articles in %s as read?").replace("%s", fn
);
535 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
536 return viewCurrentFeed('MarkAllRead')
540 function catchupFeedInGroup(id
) {
544 var title
= getFeedName(id
);
546 var str
= __("Mark all articles in %s as read?").replace("%s", title
);
548 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
549 return viewCurrentFeed('MarkAllReadGR:' + id
)
553 exception_error("catchupFeedInGroup", e
);
557 function collapse_feedlist() {
560 if (!Element
.visible('feeds-holder')) {
561 Element
.show('feeds-holder');
562 $("collapse_feeds_btn").innerHTML
= "<<";
564 Element
.hide('feeds-holder');
565 $("collapse_feeds_btn").innerHTML
= ">>";
568 dijit
.byId("main").resize();
570 query
= "?op=rpc&subop=setpref&key=_COLLAPSED_FEEDLIST&value=true";
571 new Ajax
.Request("backend.php", { parameters
: query
});
574 exception_error("collapse_feedlist", e
);
578 function viewModeChanged() {
580 return viewCurrentFeed('')
583 function viewLimitChanged() {
585 return viewCurrentFeed('')
588 /* function adjustArticleScore(id, score) {
591 var pr = prompt(__("Assign score to article:"), score);
593 if (pr != undefined) {
594 var query = "?op=rpc&subop=setScore&id=" + id + "&score=" + pr;
596 new Ajax.Request("backend.php", {
598 onComplete: function(transport) {
604 exception_error("adjustArticleScore", e);
608 function rescoreCurrentFeed() {
610 var actid
= getActiveFeedId();
612 if (activeFeedIsCat() || actid
< 0) {
613 alert(__("You can't rescore this kind of feed."));
618 alert(__("Please select some feed first."));
622 var fn
= getFeedName(actid
);
623 var pr
= __("Rescore articles in %s?").replace("%s", fn
);
626 notify_progress("Rescoring articles...");
628 var query
= "?op=pref-feeds&subop=rescore&quiet=1&ids=" + actid
;
630 new Ajax
.Request("backend.php", {
632 onComplete: function(transport
) {
638 function hotkey_handler(e
) {
642 var widget
= dijit
.getEnclosingWidget(e
.target
);
643 if (widget
&& Element
.visible(widget
.domNode
)) return;
646 var shift_key
= false;
648 var cmdline
= $('cmdline');
651 shift_key
= e
.shiftKey
;
657 keycode
= window
.event
.keyCode
;
662 var keychar
= String
.fromCharCode(keycode
);
664 if (keycode
== 27) { // escape
665 if (Element
.visible("hotkey_help_overlay")) {
666 Element
.hide("hotkey_help_overlay");
668 hotkey_prefix
= false;
672 var dialog
= dijit
.byId("infoBox");
673 var dialog_visible
= false;
676 dialog_visible
= Element
.visible(dialog
.domNode
);
678 if (dialog_visible
|| !hotkeys_enabled
) {
679 console
.log("hotkeys disabled");
683 if (keycode
== 16) return; // ignore lone shift
684 if (keycode
== 17) return; // ignore lone ctrl
686 if ((keycode
== 70 || keycode
== 67 || keycode
== 71)
689 var date
= new Date();
690 var ts
= Math
.round(date
.getTime() / 1000);
692 hotkey_prefix
= keycode
;
693 hotkey_prefix_pressed
= ts
;
695 cmdline
.innerHTML
= keychar
;
696 Element
.show(cmdline
);
698 console
.log("KP: PREFIX=" + keycode
+ " CHAR=" + keychar
+ " TS=" + ts
);
702 if (Element
.visible("hotkey_help_overlay")) {
703 Element
.hide("hotkey_help_overlay");
708 Element
.hide(cmdline
);
710 if (!hotkey_prefix
) {
712 if ((keycode
== 191 || keychar
== '?') && shift_key
) { // ?
713 if (!Element
.visible("hotkey_help_overlay")) {
714 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
716 Element
.hide("hotkey_help_overlay");
721 if (keycode
== 191 || keychar
== '/') { // /
726 if (keycode
== 74) { // j
727 // TODO: move to previous feed
731 if (keycode
== 75) { // k
732 // TODO: move to next feed
736 if (shift_key
&& keycode
== 40) { // shift-down
737 catchupRelativeToArticle(1);
741 if (shift_key
&& keycode
== 38) { // shift-up
742 catchupRelativeToArticle(0);
746 if (shift_key
&& keycode
== 78) { // N
751 if (shift_key
&& keycode
== 80) { // P
756 if (keycode
== 68 && shift_key
) { // shift-D
757 dismissSelectedArticles();
760 if (keycode
== 88 && shift_key
) { // shift-X
761 dismissReadArticles();
764 if (keycode
== 78 || keycode
== 40) { // n, down
765 if (typeof moveToPost
!= 'undefined') {
771 if (keycode
== 80 || keycode
== 38) { // p, up
772 if (typeof moveToPost
!= 'undefined') {
778 if (keycode
== 83 && shift_key
) { // S
779 selectionTogglePublished(undefined, false, true);
783 if (keycode
== 83) { // s
784 selectionToggleMarked(undefined, false, true);
789 if (keycode
== 85) { // u
790 selectionToggleUnread(undefined, false, true)
794 if (keycode
== 84 && shift_key
) { // T
795 var id
= getActiveArticleId();
797 editArticleTags(id
, getActiveFeedId(), isCdmMode());
802 if (keycode
== 9) { // tab
803 var id
= getArticleUnderPointer();
805 var cb
= $("RCHK-" + id
);
808 cb
.checked
= !cb
.checked
;
809 toggleSelectRowById(cb
, "RROW-" + id
);
815 if (keycode
== 79) { // o
816 if (getActiveArticleId()) {
817 openArticleInNewWindow(getActiveArticleId());
822 if (keycode
== 81 && shift_key
) { // Q
823 if (typeof catchupAllFeeds
!= 'undefined') {
829 if (keycode
== 88) { // x
830 if (activeFeedIsCat()) {
831 toggleCollapseCat(getActiveFeedId());
838 if (hotkey_prefix
== 70) { // f
840 hotkey_prefix
= false;
842 if (keycode
== 81) { // q
843 if (getActiveFeedId()) {
844 catchupCurrentFeed();
849 if (keycode
== 82) { // r
850 if (getActiveFeedId()) {
851 viewfeed(getActiveFeedId(), "ForceUpdate", activeFeedIsCat());
856 if (keycode
== 65) { // a
861 if (keycode
== 85) { // u
862 if (getActiveFeedId()) {
863 viewfeed(getActiveFeedId(), "ForceUpdate");
868 if (keycode
== 69) { // e
870 if (activeFeedIsCat())
871 alert(__("You can't edit this kind of feed."));
873 editFeed(getActiveFeedId());
879 if (keycode
== 83) { // s
884 if (keycode
== 67 && shift_key
) { // C
885 if (typeof catchupAllFeeds
!= 'undefined') {
891 if (keycode
== 67) { // c
892 if (getActiveFeedId()) {
893 catchupCurrentFeed();
898 if (keycode
== 87) { // w
899 feeds_sort_by_unread
= !feeds_sort_by_unread
;
900 return resort_feedlist();
903 if (keycode
== 88) { // x
904 reverseHeadlineOrder();
911 if (hotkey_prefix
== 67) { // c
912 hotkey_prefix
= false;
914 if (keycode
== 70) { // f
919 if (keycode
== 76) { // l
924 if (keycode
== 83) { // s
925 if (typeof collapse_feedlist
!= 'undefined') {
931 if (keycode
== 77) { // m
932 // TODO: sortable feedlist
936 if (keycode
== 78) { // n
937 catchupRelativeToArticle(1);
941 if (keycode
== 80) { // p
942 catchupRelativeToArticle(0);
951 if (hotkey_prefix
== 71) { // g
953 hotkey_prefix
= false;
956 if (keycode
== 65) { // a
961 if (keycode
== 83) { // s
966 if (keycode
== 80 && shift_key
) { // P
971 if (keycode
== 80) { // p
976 if (keycode
== 70) { // f
981 if (keycode
== 84 && shift_key
) { // T
989 if (hotkey_prefix
== 224 || hotkey_prefix
== 91) { // f
990 hotkey_prefix
= false;
995 console
.log("KP: PREFIX=" + hotkey_prefix
+ " CODE=" + keycode
+ " CHAR=" + keychar
);
997 console
.log("KP: CODE=" + keycode
+ " CHAR=" + keychar
);
1002 exception_error("hotkey_handler", e
);
1006 function inPreferences() {
1010 function reverseHeadlineOrder() {
1013 var query_str
= "?op=rpc&subop=togglepref&key=REVERSE_HEADLINES";
1015 new Ajax
.Request("backend.php", {
1016 parameters
: query_str
,
1017 onComplete: function(transport
) {
1022 exception_error("reverseHeadlineOrder", e
);
1026 function showFeedsWithErrors() {
1027 displayDlg('feedUpdateErrors');
1030 function handle_rpc_reply(transport
, scheduled_call
) {
1032 if (transport
.responseXML
) {
1034 if (!transport_error_check(transport
)) return false;
1036 var seq
= transport
.responseXML
.getElementsByTagName("seq")[0];
1039 seq
= seq
.firstChild
.nodeValue
;
1041 if (get_seq() != seq
) {
1042 //console.log("[handle_rpc_reply] sequence mismatch: " + seq);
1047 var message
= transport
.responseXML
.getElementsByTagName("message")[0];
1050 message
= message
.firstChild
.nodeValue
;
1052 if (message
== "UPDATE_COUNTERS") {
1053 console
.log("need to refresh counters...");
1054 setInitParam("last_article_id", -1);
1055 _force_scheduled_update
= true;
1059 var counters
= transport
.responseXML
.getElementsByTagName("counters")[0];
1062 parse_counters(counters
, scheduled_call
);
1064 var runtime_info
= transport
.responseXML
.getElementsByTagName("runtime-info")[0];
1067 parse_runtime_info(runtime_info
);
1069 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1072 notify_error("Error communicating with server.");
1076 exception_error("handle_rpc_reply", e
, transport
);
1082 function scheduleFeedUpdate(id
, is_cat
) {
1085 id
= getActiveFeedId();
1086 is_cat
= activeFeedIsCat();
1090 alert(__("Please select some feed first."));
1094 var query
= "?op=rpc&subop=scheduleFeedUpdate&id=" +
1096 "&is_cat=" + param_escape(is_cat
);
1100 new Ajax
.Request("backend.php", {
1102 onComplete: function(transport
) {
1104 if (transport
.responseXML
) {
1105 var message
= transport
.responseXML
.getElementsByTagName("message")[0];
1108 notify_info(message
.firstChild
.nodeValue
);
1113 notify_error("Error communicating with server.");
1119 exception_error("scheduleFeedUpdate", e
);