2 var display_tags
= false;
3 var global_unread
= -1;
4 var firsttime_update
= true;
5 var _active_feed_id
= 0;
6 var _active_feed_is_cat
= false;
7 var number_of_feeds
= 0;
8 var hotkey_prefix
= false;
9 var hotkey_prefix_pressed
= false;
11 var _force_scheduled_update
= false;
12 var last_scheduled_update
= false;
26 function activeFeedIsCat() {
27 return _active_feed_is_cat
;
30 function getActiveFeedId() {
32 //console.log("gAFID: " + _active_feed_id);
33 return _active_feed_id
;
35 exception_error("getActiveFeedId", e
);
39 function setActiveFeedId(id
, is_cat
) {
43 if (is_cat
!= undefined) {
44 _active_feed_is_cat
= is_cat
;
47 selectFeed(id
, is_cat
);
50 exception_error("setActiveFeedId", e
);
55 function isFeedlistSortable() {
56 return feedlist_sortable_enabled
;
59 function tagsAreDisplayed() {
63 function toggleTags(show_all
) {
67 console
.log("toggleTags: " + show_all
+ "; " + display_tags
);
69 var p
= $("dispSwitchPrompt");
71 if (!show_all
&& !display_tags
) {
72 displayDlg("printTagCloud");
73 } else if (show_all
) {
76 p
.innerHTML
= __("display feeds");
77 notify_progress("Loading, please wait...", true);
79 } else if (display_tags
) {
81 p
.innerHTML
= __("tag cloud");
82 notify_progress("Loading, please wait...", true);
87 exception_error("toggleTags", e
);
91 function dlg_frefresh_callback(transport
, deleted_feed
) {
92 if (getActiveFeedId() == deleted_feed
) {
93 setTimeout("viewfeed(-5)", 100);
96 setTimeout('updateFeedList()', 50);
100 function updateFeedList() {
102 console
.warn("updateFeedList: function not implemented");
104 /* var query_str = "backend.php?op=feeds";
107 query_str = query_str + "&tags=1";
110 if (getActiveFeedId() && !activeFeedIsCat()) {
111 query_str = query_str + "&actid=" + getActiveFeedId();
114 new Ajax.Request("backend.php", {
115 parameters: query_str,
116 onComplete: function(transport) {
117 render_feedlist(transport.responseText);
121 exception_error("updateFeedList", e
);
125 function catchupAllFeeds() {
127 var str
= __("Mark all articles as read?");
129 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
131 var query_str
= "backend.php?op=feeds&subop=catchupAll";
133 notify_progress("Marking all feeds as read...");
135 //console.log("catchupAllFeeds Q=" + query_str);
137 new Ajax
.Request("backend.php", {
138 parameters
: query_str
,
139 onComplete: function(transport
) {
140 feedlist_callback2(transport
);
148 function viewCurrentFeed(subop
) {
150 if (getActiveFeedId() != undefined) {
151 viewfeed(getActiveFeedId(), subop
, activeFeedIsCat());
153 return false; // block unneeded form submits
157 if (getInitParam("bw_limit") == "1") return;
160 var date
= new Date();
161 var ts
= Math
.round(date
.getTime() / 1000);
163 if (ts
- last_scheduled_update
> 10 || _force_scheduled_update
) {
165 //console.log("timeout()");
167 window
.clearTimeout(counter_timeout_id
);
169 var query_str
= "?op=rpc&subop=getAllCounters&seq=" + next_seq();
173 if (firsttime_update
&& !navigator
.userAgent
.match("Opera")) {
174 firsttime_update
= false;
184 query_str
= query_str
+ "&omode=" + omode
;
186 if (!_force_scheduled_update
)
187 query_str
= query_str
+ "&last_article_id=" + getInitParam("last_article_id");
189 //console.log("[timeout]" + query_str);
191 new Ajax
.Request("backend.php", {
192 parameters
: query_str
,
193 onComplete: function(transport
) {
194 handle_rpc_reply(transport
, !_force_scheduled_update
);
195 _force_scheduled_update
= false;
198 last_scheduled_update
= ts
;
202 exception_error("timeout", e
);
205 setTimeout("timeout()", 3000);
213 function updateTitle() {
214 var tmp
= "Tiny Tiny RSS";
216 if (global_unread
> 0) {
217 tmp
= tmp
+ " (" + global_unread
+ ")";
221 if (global_unread
> 0) {
222 window
.fluid
.dockBadge
= global_unread
;
224 window
.fluid
.dockBadge
= "";
228 document
.title
= tmp
;
231 function genericSanityCheck() {
232 setCookie("ttrss_test", "TEST");
234 if (getCookie("ttrss_test") != "TEST") {
243 Form
.disable("main_toolbar_form");
245 dojo
.require("dijit.layout.BorderContainer");
246 dojo
.require("dijit.layout.ContentPane");
247 dojo
.require("dijit.Dialog");
248 dojo
.require("dijit.form.Button");
249 dojo
.require("dojo.data.ItemFileWriteStore");
250 dojo
.require("dijit.Tree");
251 dojo
.require("dijit.form.Select");
252 dojo
.require("dojo.parser");
254 dojo
.addOnLoad(function() {
256 var store
= new dojo
.data
.ItemFileWriteStore({
257 url
: "backend.php?op=feeds"});
259 treeModel
= new dijit
.tree
.ForestStoreModel({
266 childrenAttrs
: ["items"]
269 var tree
= new dijit
.Tree({
271 _createTreeNode: function(args
) {
272 var tnode
= new dijit
._TreeNode(args
);
273 tnode
.labelNode
.innerHTML
= args
.label
;
276 getLabelClass: function (item
, opened
) {
277 return (item
.unread
== 0) ? "dijitTreeLabel" : "dijitTreeLabel Unread";
279 getRowClass: function (item
, opened
) {
280 return (!item
.error
|| item
.error
== '') ? "dijitTreeRow" :
281 "dijitTreeRow Error";
283 getLabel: function(item
) {
284 if (item
.unread
> 0) {
285 return item
.name
+ " (" + item
.unread
+ ")";
290 onOpen: function (item
, node
) {
291 var id
= String(item
.id
);
292 var cat_id
= id
.substr(id
.indexOf(":")+1);
294 new Ajax
.Request("backend.php",
295 { parameters
: "backend.php?op=feeds&subop=collapse&cid=" +
296 param_escape(cat_id
) + "&mode=1" } );
298 onClose: function (item
, node
) {
299 var id
= String(item
.id
);
300 var cat_id
= id
.substr(id
.indexOf(":")+1);
302 new Ajax
.Request("backend.php",
303 { parameters
: "backend.php?op=feeds&subop=collapse&cid=" +
304 param_escape(cat_id
) + "&mode=0" } );
307 onClick: function (item
, node
) {
308 var id
= String(item
.id
);
309 var is_cat
= id
.match("^CAT:");
310 var feed
= id
.substr(id
.indexOf(":")+1);
311 viewfeed(feed
, '', is_cat
);
318 if (!genericSanityCheck())
321 var params
= "&ua=" + param_escape(navigator
.userAgent
);
323 loading_set_progress(30);
325 new Ajax
.Request("backend.php", {
326 parameters
: "backend.php?op=rpc&subop=sanityCheck" + params
,
327 onComplete: function(transport
) {
328 backend_sanity_check_callback(transport
);
332 exception_error("init", e
);
336 function init_second_stage() {
340 delCookie("ttrss_test");
342 var toolbar
= document
.forms
["main_toolbar_form"];
344 dropboxSelect(toolbar
.view_mode
, getInitParam("default_view_mode"));
345 dropboxSelect(toolbar
.order_by
, getInitParam("default_view_order_by"));
347 feeds_sort_by_unread
= getInitParam("feeds_sort_by_unread") == 1;
349 loading_set_progress(60);
351 if (has_local_storage())
352 localStorage
.clear();
354 console
.log("second stage ok");
359 exception_error("init_second_stage", e
);
363 function quickMenuChange() {
364 var chooser
= $("quickMenuChooser");
365 var opid
= chooser
[chooser
.selectedIndex
].value
;
367 chooser
.selectedIndex
= 0;
371 function quickMenuGo(opid
) {
374 if (opid
== "qmcPrefs") {
378 if (opid
== "qmcSearch") {
379 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat(),
381 document
.forms
['search_form'].query
.focus();
386 if (opid
== "qmcAddFeed") {
391 if (opid
== "qmcEditFeed") {
392 editFeedDlg(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") {
430 displayDlg('quickAddFilter', '',
431 function () {document
.forms
['filter_add_form'].reg_exp
.focus();});
434 if (opid
== "qmcAddLabel") {
438 if (opid
== "qmcRescoreFeed") {
439 rescoreCurrentFeed();
442 if (opid
== "qmcHKhelp") {
443 //Element.show("hotkey_help_overlay");
444 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
447 if (opid
== "qmcResetUI") {
448 alert("Function not implemented");
451 if (opid
== "qmcToggleReorder") {
452 feedlist_sortable_enabled
= !feedlist_sortable_enabled
;
454 if (feedlist_sortable_enabled
) {
455 notify_info("Category reordering enabled");
456 toggle_sortable_feedlist(true);
458 notify_info("Category reordering disabled");
459 toggle_sortable_feedlist(false);
463 if (opid
== "qmcResetCats") {
465 if (confirm(__("Reset category order?"))) {
467 var query
= "?op=feeds&subop=catsortreset";
469 notify_progress("Loading, please wait...", true);
471 new Ajax
.Request("backend.php", {
473 onComplete: function(transport
) {
474 window
.setTimeout('updateFeedList(false, false)', 50);
480 exception_error("quickMenuGo", e
);
484 function toggleDispRead() {
487 var hide
= !(getInitParam("hide_read_feeds") == "1");
489 hideOrShowFeeds(hide
);
491 var query
= "?op=rpc&subop=setpref&key=HIDE_READ_FEEDS&value=" +
494 new Ajax
.Request("backend.php", {
496 onComplete: function(transport
) {
497 setInitParam("hide_read_feeds", hide
);
501 exception_error("toggleDispRead", e
);
505 function parse_runtime_info(elem
) {
507 if (!elem
|| !elem
.firstChild
) {
508 console
.warn("parse_runtime_info: invalid node passed");
512 var data
= JSON
.parse(elem
.firstChild
.nodeValue
);
514 //console.log("parsing runtime info...");
519 // console.log("RI: " + k + " => " + v);
521 if (k
== "new_version_available") {
522 var icon
= $("newVersionIcon");
525 icon
.style
.display
= "inline";
527 icon
.style
.display
= "none";
535 if (k
== "daemon_is_running" && v
!= 1) {
536 notify_error("<span onclick=\"javascript:explainError(1)\">Update daemon is not running.</span>", true);
540 if (k
== "daemon_stamp_ok" && v
!= 1) {
541 notify_error("<span onclick=\"javascript:explainError(3)\">Update daemon is not updating feeds.</span>", true);
550 function catchupCurrentFeed() {
552 var fn
= getFeedName(getActiveFeedId(), activeFeedIsCat());
554 var str
= __("Mark all articles in %s as read?").replace("%s", fn
);
556 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
557 return viewCurrentFeed('MarkAllRead')
561 function catchupFeedInGroup(id
) {
565 var title
= getFeedName(id
);
567 var str
= __("Mark all articles in %s as read?").replace("%s", title
);
569 if (getInitParam("confirm_feed_catchup") != 1 || confirm(str
)) {
570 return viewCurrentFeed('MarkAllReadGR:' + id
)
574 exception_error("catchupFeedInGroup", e
);
578 function editFeedDlg(feed
) {
582 alert(__("Please select some feed first."));
586 if ((feed
<= 0) || activeFeedIsCat() || tagsAreDisplayed()) {
587 alert(__("You can't edit this kind of feed."));
594 query
= "?op=pref-feeds&subop=editfeed&id=" + param_escape(feed
);
596 query
= "?op=pref-labels&subop=edit&id=" + param_escape(-feed
-11);
601 notify_progress("Loading, please wait...", true);
603 new Ajax
.Request("backend.php", {
605 onComplete: function(transport
) {
606 infobox_callback2(transport
);
607 document
.forms
["edit_feed_form"].title
.focus();
611 exception_error("editFeedDlg", e
);
615 /* this functions duplicate those of prefs.js feed editor, with
616 some differences because there is no feedlist */
618 function feedEditCancel() {
623 function feedEditSave() {
627 // FIXME: add parameter validation
629 var query
= Form
.serialize("edit_feed_form");
631 notify_progress("Saving feed...");
633 new Ajax
.Request("backend.php", {
635 onComplete: function(transport
) {
636 dlg_frefresh_callback(transport
);
645 exception_error("feedEditSave (main)", e
);
649 function collapse_feedlist() {
651 console
.warn("collapse_feedlist: function not implemented");
653 query
= "?op=rpc&subop=setpref&key=_COLLAPSED_FEEDLIST&value=true";
654 new Ajax
.Request("backend.php", { parameters
: query
});
657 exception_error("collapse_feedlist", e
);
661 function viewModeChanged() {
663 return viewCurrentFeed('')
666 function viewLimitChanged() {
668 return viewCurrentFeed('')
671 /* function adjustArticleScore(id, score) {
674 var pr = prompt(__("Assign score to article:"), score);
676 if (pr != undefined) {
677 var query = "?op=rpc&subop=setScore&id=" + id + "&score=" + pr;
679 new Ajax.Request("backend.php", {
681 onComplete: function(transport) {
687 exception_error("adjustArticleScore", e);
691 function rescoreCurrentFeed() {
693 var actid
= getActiveFeedId();
695 if (activeFeedIsCat() || actid
< 0 || tagsAreDisplayed()) {
696 alert(__("You can't rescore this kind of feed."));
701 alert(__("Please select some feed first."));
705 var fn
= getFeedName(actid
);
706 var pr
= __("Rescore articles in %s?").replace("%s", fn
);
709 notify_progress("Rescoring articles...");
711 var query
= "?op=pref-feeds&subop=rescore&quiet=1&ids=" + actid
;
713 new Ajax
.Request("backend.php", {
715 onComplete: function(transport
) {
721 function hotkey_handler(e
) {
726 var shift_key
= false;
728 var cmdline
= $('cmdline');
731 shift_key
= e
.shiftKey
;
737 keycode
= window
.event
.keyCode
;
742 var keychar
= String
.fromCharCode(keycode
);
744 if (keycode
== 27) { // escape
745 if (Element
.visible("hotkey_help_overlay")) {
746 Element
.hide("hotkey_help_overlay");
748 hotkey_prefix
= false;
752 if (dialogs
.length
> 0 || !hotkeys_enabled
) {
753 console
.log("hotkeys disabled");
757 if (keycode
== 16) return; // ignore lone shift
758 if (keycode
== 17) return; // ignore lone ctrl
760 if ((keycode
== 70 || keycode
== 67 || keycode
== 71)
763 var date
= new Date();
764 var ts
= Math
.round(date
.getTime() / 1000);
766 hotkey_prefix
= keycode
;
767 hotkey_prefix_pressed
= ts
;
769 cmdline
.innerHTML
= keychar
;
770 Element
.show(cmdline
);
772 console
.log("KP: PREFIX=" + keycode
+ " CHAR=" + keychar
+ " TS=" + ts
);
776 if (Element
.visible("hotkey_help_overlay")) {
777 Element
.hide("hotkey_help_overlay");
782 Element
.hide(cmdline
);
784 if (!hotkey_prefix
) {
786 if ((keycode
== 191 || keychar
== '?') && shift_key
) { // ?
787 if (!Element
.visible("hotkey_help_overlay")) {
788 //Element.show("hotkey_help_overlay");
789 Effect
.Appear("hotkey_help_overlay", {duration
: 0.3});
791 Element
.hide("hotkey_help_overlay");
796 if (keycode
== 191 || keychar
== '/') { // /
797 displayDlg("search", getActiveFeedId() + ":" + activeFeedIsCat(),
799 document
.forms
['search_form'].query
.focus();
804 /* if (keycode == 82 && shift_key) { // R
805 scheduleFeedUpdate(true);
809 if (keycode
== 74) { // j
810 var feed
= getActiveFeedId();
811 var new_feed
= getRelativeFeedId2(feed
, activeFeedIsCat(), 'prev');
812 // alert(feed + " IC: " + activeFeedIsCat() + " => " + new_feed);
814 var is_cat
= new_feed
.match("CAT:");
816 new_feed
= new_feed
.replace("CAT:", "");
817 viewCategory(new_feed
);
819 viewfeed(new_feed
, '', false);
825 if (keycode
== 75) { // k
826 var feed
= getActiveFeedId();
827 var new_feed
= getRelativeFeedId2(feed
, activeFeedIsCat(), 'next');
828 // alert(feed + " IC: " + activeFeedIsCat() + " => " + new_feed);
830 var is_cat
= new_feed
.match("CAT:");
831 if (is_cat
== "CAT:") {
832 new_feed
= new_feed
.replace("CAT:", "");
833 viewCategory(new_feed
);
835 viewfeed(new_feed
, '', false);
841 if (shift_key
&& keycode
== 40) { // shift-down
842 catchupRelativeToArticle(1);
846 if (shift_key
&& keycode
== 38) { // shift-up
847 catchupRelativeToArticle(0);
851 if (shift_key
&& keycode
== 78) { // N
856 if (shift_key
&& keycode
== 80) { // P
861 if (keycode
== 68 && shift_key
) { // shift-D
862 dismissSelectedArticles();
865 if (keycode
== 88 && shift_key
) { // shift-X
866 dismissReadArticles();
869 if (keycode
== 78 || keycode
== 40) { // n, down
870 if (typeof moveToPost
!= 'undefined') {
876 if (keycode
== 80 || keycode
== 38) { // p, up
877 if (typeof moveToPost
!= 'undefined') {
883 if (keycode
== 83 && shift_key
) { // S
884 selectionTogglePublished(undefined, false, true);
888 if (keycode
== 83) { // s
889 selectionToggleMarked(undefined, false, true);
894 if (keycode
== 85) { // u
895 selectionToggleUnread(undefined, false, true)
899 if (keycode
== 84 && shift_key
) { // T
900 var id
= getActiveArticleId();
902 editArticleTags(id
, getActiveFeedId(), isCdmMode());
907 if (keycode
== 9) { // tab
908 var id
= getArticleUnderPointer();
910 var cb
= $("RCHK-" + id
);
913 cb
.checked
= !cb
.checked
;
914 toggleSelectRowById(cb
, "RROW-" + id
);
920 if (keycode
== 79) { // o
921 if (getActiveArticleId()) {
922 openArticleInNewWindow(getActiveArticleId());
927 if (keycode
== 81 && shift_key
) { // Q
928 if (typeof catchupAllFeeds
!= 'undefined') {
934 if (keycode
== 88) { // x
935 if (activeFeedIsCat()) {
936 toggleCollapseCat(getActiveFeedId());
943 if (hotkey_prefix
== 70) { // f
945 hotkey_prefix
= false;
947 if (keycode
== 81) { // q
948 if (getActiveFeedId()) {
949 catchupCurrentFeed();
954 if (keycode
== 82) { // r
955 if (getActiveFeedId()) {
956 viewfeed(getActiveFeedId(), "ForceUpdate", activeFeedIsCat());
961 if (keycode
== 65) { // a
966 /* if (keycode == 85 && shift_key) { // U
967 scheduleFeedUpdate(true);
971 if (keycode
== 85) { // u
972 if (getActiveFeedId()) {
973 viewfeed(getActiveFeedId(), "ForceUpdate");
978 if (keycode
== 69) { // e
979 editFeedDlg(getActiveFeedId());
983 if (keycode
== 83) { // s
988 if (keycode
== 67 && shift_key
) { // C
989 if (typeof catchupAllFeeds
!= 'undefined') {
995 if (keycode
== 67) { // c
996 if (getActiveFeedId()) {
997 catchupCurrentFeed();
1002 if (keycode
== 87) { // w
1003 feeds_sort_by_unread
= !feeds_sort_by_unread
;
1004 return resort_feedlist();
1007 if (keycode
== 88) { // x
1008 reverseHeadlineOrder();
1015 if (hotkey_prefix
== 67) { // c
1016 hotkey_prefix
= false;
1018 if (keycode
== 70) { // f
1019 displayDlg('quickAddFilter', '',
1020 function () {document
.forms
['filter_add_form'].reg_exp
.focus();});
1024 if (keycode
== 76) { // l
1029 if (keycode
== 83) { // s
1030 if (typeof collapse_feedlist
!= 'undefined') {
1031 collapse_feedlist();
1036 if (keycode
== 77) { // m
1037 feedlist_sortable_enabled
= !feedlist_sortable_enabled
;
1038 if (feedlist_sortable_enabled
) {
1039 notify_info("Category reordering enabled");
1040 toggle_sortable_feedlist(true);
1042 notify_info("Category reordering disabled");
1043 toggle_sortable_feedlist(false);
1047 if (keycode
== 78) { // n
1048 catchupRelativeToArticle(1);
1052 if (keycode
== 80) { // p
1053 catchupRelativeToArticle(0);
1062 if (hotkey_prefix
== 71) { // g
1064 hotkey_prefix
= false;
1067 if (keycode
== 65) { // a
1072 if (keycode
== 83) { // s
1077 if (keycode
== 80 && shift_key
) { // P
1082 if (keycode
== 80) { // p
1087 if (keycode
== 70) { // f
1092 if (keycode
== 84 && shift_key
) { // T
1100 if (hotkey_prefix
== 224 || hotkey_prefix
== 91) { // f
1101 hotkey_prefix
= false;
1105 if (hotkey_prefix
) {
1106 console
.log("KP: PREFIX=" + hotkey_prefix
+ " CODE=" + keycode
+ " CHAR=" + keychar
);
1108 console
.log("KP: CODE=" + keycode
+ " CHAR=" + keychar
);
1113 exception_error("hotkey_handler", e
);
1117 function inPreferences() {
1121 function reverseHeadlineOrder() {
1124 var query_str
= "?op=rpc&subop=togglepref&key=REVERSE_HEADLINES";
1126 new Ajax
.Request("backend.php", {
1127 parameters
: query_str
,
1128 onComplete: function(transport
) {
1133 exception_error("reverseHeadlineOrder", e
);
1137 function showFeedsWithErrors() {
1138 displayDlg('feedUpdateErrors');
1141 function handle_rpc_reply(transport
, scheduled_call
) {
1143 if (transport
.responseXML
) {
1145 if (!transport_error_check(transport
)) return false;
1147 var seq
= transport
.responseXML
.getElementsByTagName("seq")[0];
1150 seq
= seq
.firstChild
.nodeValue
;
1152 if (get_seq() != seq
) {
1153 //console.log("[handle_rpc_reply] sequence mismatch: " + seq);
1158 var message
= transport
.responseXML
.getElementsByTagName("message")[0];
1161 message
= message
.firstChild
.nodeValue
;
1163 if (message
== "UPDATE_COUNTERS") {
1164 console
.log("need to refresh counters...");
1165 setInitParam("last_article_id", -1);
1166 _force_scheduled_update
= true;
1170 var counters
= transport
.responseXML
.getElementsByTagName("counters")[0];
1173 parse_counters(counters
, scheduled_call
);
1175 var runtime_info
= transport
.responseXML
.getElementsByTagName("runtime-info")[0];
1178 parse_runtime_info(runtime_info
);
1180 if (feedsSortByUnread())
1183 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
1186 notify_error("Error communicating with server.");
1190 exception_error("handle_rpc_reply", e
, transport
);
1196 function scheduleFeedUpdate() {
1199 if (!getActiveFeedId()) {
1200 alert(__("Please select some feed first."));
1204 var query
= "?op=rpc&subop=scheduleFeedUpdate&id=" +
1205 param_escape(getActiveFeedId()) +
1206 "&is_cat=" + param_escape(activeFeedIsCat());
1210 new Ajax
.Request("backend.php", {
1212 onComplete: function(transport
) {
1214 if (transport
.responseXML
) {
1215 var message
= transport
.responseXML
.getElementsByTagName("message")[0];
1218 notify_info(message
.firstChild
.nodeValue
);
1223 notify_error("Error communicating with server.");
1229 exception_error("scheduleFeedUpdate", e
);