]>
git.wh0rd.org - tt-rss.git/blob - digest.js
3 var _active_feed_id
= false;
4 var _active_feed_offset
= false;
5 var _update_timeout
= false;
6 var _view_update_timeout
= false;
7 var _feedlist_expanded
= false;
10 function article_appear(article_id
) {
12 new Effect
.Appear('A-' + article_id
);
14 exception_error("article_appear", e
);
18 function catchup_feed(feed_id
, callback
) {
21 var fn
= find_feed(last_feeds
, feed_id
).title
;
23 if (confirm(__("Mark all articles in %s as read?").replace("%s", fn
))) {
27 if (feed_id
< 0) is_cat
= "true"; // KLUDGE
29 var query
= "?op=rpc&subop=catchupFeed&feed_id=" +
30 feed_id
+ "&is_cat=" + is_cat
;
32 new Ajax
.Request("backend.php", {
34 onComplete: function(transport
) {
35 if (callback
) callback(transport
);
42 exception_error("catchup_article", e
);
46 function catchup_visible_articles(callback
) {
49 var elems
= $("headlines-content").getElementsByTagName("LI");
52 for (var i
= 0; i
< elems
.length
; i
++) {
53 if (elems
[i
].id
&& elems
[i
].id
.match("A-")) {
54 ids
.push(elems
[i
].id
.replace("A-", ""));
58 if (confirm(__("Mark %d displayed articles as read?").replace("%d", ids
.length
))) {
60 var query
= "?op=rpc&subop=catchupSelected" +
61 "&cmode=0&ids=" + param_escape(ids
);
63 new Ajax
.Request("backend.php", {
65 onComplete: function(transport
) {
66 if (callback
) callback(transport
);
68 viewfeed(_active_feed_id
, 0);
74 exception_error("catchup_visible_articles", e
);
78 function catchup_article(article_id
, callback
) {
80 var query
= "?op=rpc&subop=catchupSelected" +
81 "&cmode=0&ids=" + article_id
;
83 new Ajax
.Request("backend.php", {
85 onComplete: function(transport
) {
86 if (callback
) callback(transport
);
90 exception_error("catchup_article", e
);
94 function set_selected_feed(feed_id
) {
96 var feeds
= $("feeds-content").getElementsByTagName("LI");
98 for (var i
= 0; i
< feeds
.length
; i
++) {
99 if (feeds
[i
].id
== "F-" + feed_id
)
100 feeds
[i
].className
= "selected";
102 feeds
[i
].className
= "";
105 _active_feed_id
= feed_id
;
108 exception_error("mark_selected_feed", e
);
112 function zoom(elem
, article_id
) {
114 //alert(elem + "/" + article_id);
116 elem
.innerHTML
= "<img src='images/indicator_tiny.gif'> " +
117 __("Loading, please wait...");
119 new Ajax
.Request("backend.php", {
120 parameters
: "?op=rpc&subop=digest-get-contents&article_id=" +
122 onComplete: function(transport
) {
123 fatal_error_check(transport
);
125 if (transport
.responseXML
) {
126 var article
= transport
.responseXML
.getElementsByTagName('article')[0];
127 elem
.innerHTML
= article
.firstChild
.nodeValue
;
129 new Effect
.BlindDown(elem
, {duration
: 0.5});
131 elem
.onclick
= false;
132 elem
.style
.cursor
= "auto";
134 catchup_article(article_id
,
136 window
.clearTimeout(_view_update_timeout
);
137 _view_update_timeout
= window
.setTimeout("view_update()", 500);
138 $("A-" + article_id
).className
= "read";
143 elem
.innerHTML
= __("Error: unable to load article.");
150 exception_error("zoom", e
);
154 function load_more() {
156 var pr
= $("H-LOADING-IMG");
158 if (pr
) Element
.show(pr
);
160 viewfeed(_active_feed_id
, _active_feed_offset
+ 10, false, false, true,
162 var pr
= $("H-LOADING-IMG");
164 if (pr
) Element
.hide(pr
);
167 exception_error("load_more", e
);
171 function update(callback
) {
173 console
.log('updating feeds...');
175 window
.clearTimeout(_update_timeout
);
177 new Ajax
.Request("backend.php", {
178 parameters
: "?op=rpc&subop=digest-init",
179 onComplete: function(transport
) {
180 fatal_error_check(transport
);
181 parse_feeds(transport
);
182 set_selected_feed(_active_feed_id
);
184 if (callback
) callback(transport
);
187 _update_timeout
= window
.setTimeout('update()', 5*1000);
189 exception_error("update", e
);
193 function remove_headline_entry(article_id
) {
195 var elem
= $('A-' + article_id
);
198 elem
.parentNode
.removeChild(elem
);
202 exception_error("remove_headline_entry", e
);
206 function view_update() {
208 viewfeed(_active_feed_id
, _active_feed_offset
, false, true, true);
211 exception_error("view_update", e
);
215 function view(article_id
, dismiss_only
) {
217 remove_headline_entry(article_id
);
219 catchup_article(article_id
,
221 window
.clearTimeout(_view_update_timeout
);
222 _view_update_timeout
= window
.setTimeout("view_update()", 500);
225 return dismiss_only
!= true;
227 exception_error("view", e
);
231 function viewfeed(feed_id
, offset
, replace
, no_effects
, no_indicator
, callback
) {
234 if (!feed_id
) feed_id
= _active_feed_id
;
239 offset
= _active_feed_offset
+ offset
;
242 if (replace
== undefined) replace
= (offset
== 0);
244 _update_seq
= _update_seq
+ 1;
246 var query
= "backend.php?op=rpc&subop=digest-update&feed_id=" +
247 param_escape(feed_id
) + "&offset=" + offset
+
248 "&seq=" + _update_seq
;
252 var img
= $("F-" + feed_id
).getElementsByTagName("IMG")[0];
254 if (img
&& !no_indicator
) {
255 img
.setAttribute("orig_src", img
.src
);
256 img
.src
= 'images/indicator_tiny.gif';
259 new Ajax
.Request("backend.php", {
261 onComplete: function(transport
) {
262 fatal_error_check(transport
);
263 parse_headlines(transport
, replace
, no_effects
);
264 set_selected_feed(feed_id
);
265 _active_feed_offset
= offset
;
267 if (img
&& !no_indicator
)
268 img
.src
= img
.getAttribute("orig_src");
270 if (callback
) callback(transport
);
275 exception_error("view", e
);
279 function find_article(articles
, article_id
) {
281 for (var i
= 0; i
< articles
.length
; i
++) {
282 if (articles
[i
].id
== article_id
)
289 exception_error("find_article", e
);
293 function find_feed(feeds
, feed_id
) {
295 for (var i
= 0; i
< feeds
.length
; i
++) {
296 if (feeds
[i
].id
== feed_id
)
303 exception_error("find_feed", e
);
307 function get_feed_icon(feed
) {
310 return 'icons/' + feed
.id
+ '.ico';
313 return 'images/mark_set.png';
316 return 'images/pub_set.png';
319 return 'images/fresh.png';
322 return 'images/tag.png';
325 return 'images/label.png';
327 return 'images/blank_icon.gif';
330 exception_error("get_feed_icon", e
);
334 function add_feed_entry(feed
) {
338 icon_part
= "<img src='" + get_feed_icon(feed
) + "'/>";
340 var tmp_html
= "<li id=\"F-"+feed
.id
+"\" " +
341 "onmouseover=\"feed_mi(this)\" onmouseout=\"feed_mo(this)\">" +
343 "<a href=\"#\" onclick=\"viewfeed("+feed
.id
+")\">" + feed
.title
+ "</a>" +
344 "<div class='unread-ctr'>" +
345 "<img onclick=\"catchup_feed("+feed
.id
+")\" title=\"" +
347 "\" class=\"dismiss\" style='display : none' src=\"images/digest_checkbox.png\">" +
348 "<span class=\"unread\">" + feed
.unread
+ "</span>" +
352 $("feeds-content").innerHTML
+= tmp_html
;
355 exception_error("add_feed_entry", e
);
359 function add_headline_entry(article
, feed
, no_effects
) {
364 icon_part
= "<img class='icon' src='" + get_feed_icon(feed
) + "'/>";
371 if (article
.tags
.length
> 0) {
373 tags_part
= " " + __("in") + " ";
375 for (var i
= 0; i
< Math
.min(5, article
.tags
.length
); i
++) {
376 tags_part
+= "<a href=\"#\" onclick=\"viewfeed('" +
377 article
.tags
[i
] + "')\">" +
378 article
.tags
[i
] + "</a>, ";
381 tags_part
= tags_part
.replace(/, $/, "");
382 tags_part
= "<span class=\"tags\">" + tags_part
+ "</span>";
386 mark_part
= "<img title='"+ __("Unstar article")+"' onclick=\"toggle_mark(this, "+article
.id
+")\" src='images/mark_set.png'>";
388 mark_part
= "<img title='"+__("Star article")+"' onclick=\"toggle_mark(this, "+article
.id
+")\" src='images/mark_unset.png'>";
390 if (article
.published
)
391 publ_part
= "<img title='"+__("Unpublish article")+"' onclick=\"toggle_pub(this, "+article
.id
+")\" src='images/pub_set.png'>";
393 publ_part
= "<img title='"+__("Publish article")+"' onclick=\"toggle_pub(this, "+article
.id
+")\" src='images/pub_unset.png'>";
397 if (!no_effects
) style
= "style=\"display : none\"";
399 if (article
.excerpt
.trim() == "")
400 article
.excerpt
= __("Click to expand article.");
402 var tmp_html
= "<li id=\"A-"+article
.id
+"\" "+style
+" class=\"unread\">" +
405 "<div class='digest-check'>" +
408 "<img title='" + __("Mark as read") + "' onclick=\"view("+article
.id
+", true)\" src='images/digest_checkbox.png'>" +
410 "<a target=\"_blank\" href=\""+article
.link
+"\""+
411 "onclick=\"return view("+article
.id
+")\" class='title'>" +
412 article
.title
+ "</a>" +
413 "<div class='body'>" +
414 "<div title=\""+__("Click to expand article")+"\" onclick=\"zoom(this, "+article
.id
+")\" class='excerpt'>" +
415 article
.excerpt
+ "</div>" +
416 "<div class='info'><a href=\#\" onclick=\"viewfeed("+feed
.id
+")\">" +
417 feed
.title
+ "</a> " + tags_part
+ " @ " +
418 new Date(article
.updated
* 1000) + "</div>" +
421 $("headlines-content").innerHTML
+= tmp_html
;
424 window
.setTimeout('article_appear(' + article
.id
+ ')', 100);
427 exception_error("add_headline_entry", e
);
431 function expand_feeds() {
433 _feedlist_expanded
= true;
435 redraw_feedlist(last_feeds
);
438 exception_error("expand_feeds", e
);
442 function redraw_feedlist(feeds
) {
445 $('feeds-content').innerHTML
= "";
449 if (_feedlist_expanded
) limit
= feeds
.length
;
451 for (var i
= 0; i
< Math
.min(limit
, feeds
.length
); i
++) {
452 add_feed_entry(feeds
[i
]);
455 if (feeds
.length
> limit
) {
456 $('feeds-content').innerHTML
+= "<li id='F-MORE-PROMPT'>" +
457 "<img src='images/blank_icon.gif'>" +
458 "<a href=\"#\" onclick=\"expand_feeds()\">" +
459 __("%d more...").replace("%d", feeds
.length
-10) +
464 exception_error("redraw_feedlist", e
);
468 function parse_feeds(transport
) {
471 if (!transport
.responseXML
) return;
473 var feeds
= transport
.responseXML
.getElementsByTagName('feeds')[0];
476 feeds
= eval("(" + feeds
.firstChild
.nodeValue
+ ")");
478 feeds
.sort( function (a
,b
)
480 if (b
.unread
!= a
.unread
)
481 return (b
.unread
- a
.unread
)
483 if (a
.title
> b
.title
)
485 else if (a
.title
< b
.title
)
491 var all_articles
= find_feed(feeds
, -4);
493 update_title(all_articles
.unread
);
497 redraw_feedlist(feeds
);
501 exception_error("parse_feeds", e
);
505 function parse_headlines(transport
, replace
, no_effects
) {
507 if (!transport
.responseXML
) return;
509 var seq
= transport
.responseXML
.getElementsByTagName('seq')[0];
512 seq
= seq
.firstChild
.nodeValue
;
513 if (seq
!= _update_seq
) {
514 console
.log("parse_headlines: wrong sequence received.");
521 var headlines
= transport
.responseXML
.getElementsByTagName('headlines')[0];
522 var headlines_title
= transport
.responseXML
.getElementsByTagName('headlines-title')[0];
524 if (headlines
&& headlines_title
) {
525 headlines
= eval("(" + headlines
.firstChild
.nodeValue
+ ")");
527 var title
= headlines_title
.firstChild
.nodeValue
;
529 $("headlines-title").innerHTML
= title
;
532 $('headlines-content').innerHTML
= '';
533 Element
.hide('headlines-content');
536 var pr
= $('H-MORE-PROMPT');
538 if (pr
) pr
.parentNode
.removeChild(pr
);
540 var inserted
= false;
542 for (var i
= 0; i
< headlines
.length
; i
++) {
544 if (!$('A-' + headlines
[i
].id
)) {
545 add_headline_entry(headlines
[i
],
546 find_feed(last_feeds
, headlines
[i
].feed_id
), !no_effects
);
548 inserted
= $("A-" + headlines
[i
].id
);
553 $('headlines-content').appendChild(pr
);
554 if (!no_effects
) new Effect
.ScrollTo(inserted
);
556 $('headlines-content').innerHTML
+= "<li id='H-MORE-PROMPT'>" +
557 "<div class='body'>" +
558 "<a href=\"javascript:catchup_visible_articles()\">" +
559 __("Mark as read") + "</a> | " +
560 "<a href=\"javascript:load_more()\">" +
561 __("Load more...") + "</a>" +
562 "<img style=\"display : none\" "+
563 "id=\"H-LOADING-IMG\" src='images/indicator_tiny.gif'>" +
567 if (replace
&& !no_effects
)
568 new Effect
.Appear('headlines-content', {duration
: 0.3});
570 //new Effect.Appear('headlines-content');
574 exception_error("parse_headlines", e
);
581 new Ajax
.Request("backend.php", {
582 parameters
: "backend.php?op=rpc&subop=digest-init",
583 onComplete: function(transport
) {
584 parse_feeds(transport
);
585 window
.setTimeout('viewfeed(-4)', 100);
586 _update_timeout
= window
.setTimeout('update()', 5*1000);
590 exception_error("digest_init", e
);
594 function toggle_mark(mark_img
, id
) {
598 var query
= "?op=rpc&id=" + id
+ "&subop=mark";
600 query
= query
+ "&afid=" + _active_feed_id
;
601 query
= query
+ "&omode=c";
603 if (!mark_img
) return;
605 if (mark_img
.src
.match("mark_unset")) {
606 mark_img
.src
= mark_img
.src
.replace("mark_unset", "mark_set");
607 mark_img
.alt
= __("Unstar article");
608 query
= query
+ "&mark=1";
610 mark_img
.alt
= __("Please wait...");
611 query
= query
+ "&mark=0";
613 mark_img
.src
= mark_img
.src
.replace("mark_set", "mark_unset");
614 mark_img
.alt
= __("Star article");
617 new Ajax
.Request("backend.php", {
619 onComplete: function(transport
) {
624 exception_error("toggle_mark", e
);
628 function toggle_pub(mark_img
, id
, note
) {
632 var query
= "?op=rpc&id=" + id
+ "&subop=publ";
634 query
= query
+ "&afid=" + _active_feed_id
;
636 if (note
!= undefined) {
637 query
= query
+ "¬e=" + param_escape(note
);
639 query
= query
+ "¬e=undefined";
642 query
= query
+ "&omode=c";
644 if (!mark_img
) return;
646 if (mark_img
.src
.match("pub_unset") || note
!= undefined) {
647 mark_img
.src
= mark_img
.src
.replace("pub_unset", "pub_set");
648 mark_img
.alt
= __("Unpublish article");
649 query
= query
+ "&pub=1";
652 mark_img
.alt
= __("Please wait...");
653 query
= query
+ "&pub=0";
655 mark_img
.src
= mark_img
.src
.replace("pub_set", "pub_unset");
656 mark_img
.alt
= __("Publish article");
659 new Ajax
.Request("backend.php", {
661 onComplete: function(transport
) {
666 exception_error("toggle_pub", e
);
670 function fatal_error(code
, msg
) {
674 window
.location
.href
= "digest.php";
675 } else if (code
== 5) {
676 window
.location
.href
= "update.php";
679 if (msg
== "") msg
= "Unknown error";
681 console
.error("Fatal error: " + code
+ "\n" +
687 exception_error("fatalError", e
);
691 function fatal_error_check(transport
) {
693 if (transport
.responseXML
) {
694 var error
= transport
.responseXML
.getElementsByTagName("error")[0];
697 var code
= error
.getAttribute("error-code");
698 var msg
= error
.getAttribute("error-msg");
700 fatal_error(code
, msg
);
706 exception_error("fatal_error_check", e
);
711 function feed_mi(elem
) {
713 var imgs
= elem
.getElementsByTagName('IMG');
714 var spans
= elem
.getElementsByTagName('SPAN');
716 for (var i
= 0; i
< imgs
.length
; i
++) {
717 if (imgs
[i
].className
== "dismiss")
718 Element
.show(imgs
[i
]);
721 for (var i
= 0; i
< spans
.length
; i
++) {
722 if (spans
[i
].className
== "unread")
723 Element
.hide(spans
[i
]);
728 exception_error("feed_mi", e
);
732 function feed_mo(elem
) {
734 var imgs
= elem
.getElementsByTagName('IMG');
735 var spans
= elem
.getElementsByTagName('SPAN');
737 for (var i
= 0; i
< imgs
.length
; i
++) {
738 if (imgs
[i
].className
== "dismiss")
739 Element
.hide(imgs
[i
]);
742 for (var i
= 0; i
< spans
.length
; i
++) {
743 if (spans
[i
].className
== "unread")
744 Element
.show(spans
[i
]);
748 exception_error("feed_mo", e
);
752 function update_title(unread
) {
754 document
.title
= "Tiny Tiny RSS";
757 document
.title
+= " (" + unread
+ ")";
760 exception_error("update_title", e
);