]> git.wh0rd.org - tt-rss.git/blame - digest.js
support zoom for blank excerpts
[tt-rss.git] / digest.js
CommitLineData
c01f40f4
AD
1var last_feeds = [];
2
1ca8997b
AD
3var _active_feed_id = false;
4var _active_feed_offset = false;
118e9399 5var _update_timeout = false;
e4c530dc 6var _view_update_timeout = false;
c1b5cd23 7var _feedlist_expanded = false;
e4c530dc 8var _update_seq = 1;
c1b5cd23 9
41de9581
AD
10function article_appear(article_id) {
11 try {
12 new Effect.Appear('A-' + article_id);
13 } catch (e) {
14 exception_error("article_appear", e);
15 }
16}
17
c1b5cd23
AD
18function catchup_feed(feed_id, callback) {
19 try {
20
21 var fn = find_feed(last_feeds, feed_id).title;
22
23 if (confirm(__("Mark all articles in %s as read?").replace("%s", fn))) {
24
25 var is_cat = "";
26
9ed133e7 27 if (feed_id < 0) is_cat = "true"; // KLUDGE
c1b5cd23
AD
28
29 var query = "?op=rpc&subop=catchupFeed&feed_id=" +
30 feed_id + "&is_cat=" + is_cat;
31
32 new Ajax.Request("backend.php", {
33 parameters: query,
34 onComplete: function(transport) {
35 if (callback) callback(transport);
36
37 update();
38 } });
39 }
40
41 } catch (e) {
42 exception_error("catchup_article", e);
43 }
44}
45
46360a96
AD
46function catchup_visible_articles(callback) {
47 try {
46360a96 48
eb4f33ec 49 if (confirm(__("Mark all displayed articles as read?"))) {
46360a96 50
eb4f33ec
AD
51 var elems = $("headlines-content").getElementsByTagName("LI");
52 var ids = [];
53
54 for (var i = 0; i < elems.length; i++) {
55 if (elems[i].id && elems[i].id.match("A-")) {
56 ids.push(elems[i].id.replace("A-", ""));
57 }
58 }
59
60 var query = "?op=rpc&subop=catchupSelected" +
61 "&cmode=0&ids=" + param_escape(ids);
62
63 new Ajax.Request("backend.php", {
64 parameters: query,
65 onComplete: function(transport) {
66 if (callback) callback(transport);
67
68 viewfeed(_active_feed_id, 0);
69 } });
70
71 }
46360a96
AD
72
73 } catch (e) {
74 exception_error("catchup_visible_articles", e);
75 }
76}
1ca8997b 77
d5d56329
AD
78function catchup_article(article_id, callback) {
79 try {
80 var query = "?op=rpc&subop=catchupSelected" +
81 "&cmode=0&ids=" + article_id;
82
83 new Ajax.Request("backend.php", {
84 parameters: query,
85 onComplete: function(transport) {
86 if (callback) callback(transport);
87 } });
88
89 } catch (e) {
90 exception_error("catchup_article", e);
91 }
92}
93
94function set_selected_feed(feed_id) {
11a7a966
AD
95 try {
96 var feeds = $("feeds-content").getElementsByTagName("LI");
97
98 for (var i = 0; i < feeds.length; i++) {
99 if (feeds[i].id == "F-" + feed_id)
100 feeds[i].className = "selected";
101 else
102 feeds[i].className = "";
103 }
104
d5d56329
AD
105 _active_feed_id = feed_id;
106
11a7a966
AD
107 } catch (e) {
108 exception_error("mark_selected_feed", e);
109 }
110}
111
d8ea9902 112function zoom(elem, article_id) {
22933e5e 113 try {
d8ea9902 114 //alert(elem + "/" + article_id);
22933e5e 115
d8ea9902
AD
116 elem.innerHTML = "<img src='images/indicator_tiny.gif'> " +
117 __("Loading, please wait...");
118
119 new Ajax.Request("backend.php", {
120 parameters: "?op=rpc&subop=digest-get-contents&article_id=" +
121 article_id,
122 onComplete: function(transport) {
123 fatal_error_check(transport);
124
125 if (transport.responseXML) {
126 var article = transport.responseXML.getElementsByTagName('article')[0];
127 elem.innerHTML = article.firstChild.nodeValue;
85629f6c
AD
128
129 new Effect.BlindDown(elem, {duration : 0.5});
130
131 elem.onclick = false;
132 elem.style.cursor = "auto";
d8ea9902
AD
133 } else {
134 elem.innerHTML = __("Error: unable to load article.");
135 }
136
137 } });
118e9399 138
118e9399
AD
139
140 } catch (e) {
141 exception_error("zoom", e);
142 }
143}
144
145function load_more() {
146 try {
118e9399
AD
147 viewfeed(_active_feed_id, _active_feed_offset + 10);
148 } catch (e) {
149 exception_error("load_more", e);
150 }
151}
152
153function update() {
154 try {
d5d56329
AD
155 console.log('updating feeds...');
156
157 window.clearTimeout(_update_timeout);
158
159 new Ajax.Request("backend.php", {
160 parameters: "?op=rpc&subop=digest-init",
161 onComplete: function(transport) {
e0cebf2a 162 fatal_error_check(transport);
d5d56329
AD
163 parse_feeds(transport);
164 set_selected_feed(_active_feed_id);
165 } });
166
167 _update_timeout = window.setTimeout('update()', 5*1000);
118e9399
AD
168 } catch (e) {
169 exception_error("update", e);
170 }
171}
172
d5d56329 173function remove_headline_entry(article_id) {
118e9399
AD
174 try {
175 var elem = $('A-' + article_id);
176
d5d56329
AD
177 if (elem) {
178 elem.parentNode.removeChild(elem);
179 }
118e9399 180
d5d56329
AD
181 } catch (e) {
182 exception_error("remove_headline_entry", e);
183 }
184}
118e9399 185
e4c530dc
AD
186function view_update() {
187 try {
188 viewfeed(_active_feed_id, _active_feed_offset, false, true);
189 update();
190 } catch (e) {
191 exception_error("view_update", e);
192 }
193}
194
d5d56329
AD
195function view(article_id, dismiss_only) {
196 try {
197 remove_headline_entry(article_id);
118e9399 198
d5d56329 199 catchup_article(article_id,
1a434472 200 function() {
e4c530dc
AD
201 window.clearTimeout(_view_update_timeout);
202 _view_update_timeout = window.setTimeout("view_update()", 1000);
1a434472 203 });
118e9399
AD
204
205 return dismiss_only != true;
22933e5e
AD
206 } catch (e) {
207 exception_error("view", e);
208 }
209}
210
41de9581 211function viewfeed(feed_id, offset, replace, no_effects) {
b41c2549
AD
212 try {
213
1ca8997b
AD
214 if (!feed_id) feed_id = _active_feed_id;
215
d5d56329 216 if (!offset) {
1ca8997b 217 offset = 0;
d5d56329 218 } else {
1ca8997b 219 offset = _active_feed_offset + offset;
d5d56329 220 }
1ca8997b 221
41de9581
AD
222 if (replace == undefined) replace = (offset == 0);
223
e4c530dc
AD
224 _update_seq = _update_seq + 1;
225
78ac6caf 226 var query = "backend.php?op=rpc&subop=digest-update&feed_id=" +
e4c530dc
AD
227 param_escape(feed_id) + "&offset=" + offset +
228 "&seq=" + _update_seq;
78ac6caf
AD
229
230 console.log(query);
1ca8997b 231
d3f13000
AD
232 var img = $("F-" + feed_id).getElementsByTagName("IMG")[0];
233
234 img.setAttribute("orig_src", img.src);
235 img.src = 'images/indicator_tiny.gif';
236
b41c2549 237 new Ajax.Request("backend.php", {
1ca8997b 238 parameters: query,
b41c2549 239 onComplete: function(transport) {
e0cebf2a 240 fatal_error_check(transport);
41de9581 241 parse_headlines(transport, replace, no_effects);
d5d56329 242 set_selected_feed(feed_id);
1ca8997b 243 _active_feed_offset = offset;
d3f13000 244 img.src = img.getAttribute("orig_src");
11a7a966 245 } });
b41c2549
AD
246
247 } catch (e) {
248 exception_error("view", e);
249 }
250}
251
118e9399
AD
252function find_article(articles, article_id) {
253 try {
254 for (var i = 0; i < articles.length; i++) {
255 if (articles[i].id == article_id)
256 return articles[i];
257 }
258
259 return false;
260
261 } catch (e) {
262 exception_error("find_article", e);
263 }
264}
265
c01f40f4
AD
266function find_feed(feeds, feed_id) {
267 try {
268 for (var i = 0; i < feeds.length; i++) {
269 if (feeds[i].id == feed_id)
270 return feeds[i];
271 }
272
273 return false;
274
275 } catch (e) {
276 exception_error("find_feed", e);
277 }
278}
279
b41c2549
AD
280function get_feed_icon(feed) {
281 try {
282 if (feed.has_icon)
283 return 'icons/' + feed.id + '.ico';
284
285 if (feed.id == -1)
286 return 'images/mark_set.png';
287
288 if (feed.id == -2)
289 return 'images/pub_set.png';
290
291 if (feed.id == -3)
292 return 'images/fresh.png';
293
294 if (feed.id == -4)
295 return 'images/tag.png';
296
297 if (feed.id < -10)
298 return 'images/label.png';
299
d5d56329
AD
300 return 'images/blank_icon.gif';
301
b41c2549
AD
302 } catch (e) {
303 exception_error("get_feed_icon", e);
304 }
305}
306
c01f40f4
AD
307function add_feed_entry(feed) {
308 try {
309 var icon_part = "";
310
b41c2549 311 icon_part = "<img src='" + get_feed_icon(feed) + "'/>";
c01f40f4 312
c1b5cd23
AD
313 var tmp_html = "<li id=\"F-"+feed.id+"\" " +
314 "onmouseover=\"feed_mi(this)\" onmouseout=\"feed_mo(this)\">" +
c01f40f4 315 icon_part +
c1b5cd23
AD
316 "<a href=\"#\" onclick=\"viewfeed("+feed.id+")\">" + feed.title + "</a>" +
317 "<div class='unread-ctr'>" +
46360a96
AD
318 "<img onclick=\"catchup_feed("+feed.id+")\" title=\"" +
319 __("Mark as read") +
320 "\" class=\"dismiss\" style='display : none' src=\"images/digest_checkbox.png\">" +
c1b5cd23
AD
321 "<span class=\"unread\">" + feed.unread + "</span>" +
322 "</div>" +
c01f40f4
AD
323 "</li>";
324
325 $("feeds-content").innerHTML += tmp_html;
326
327 } catch (e) {
328 exception_error("add_feed_entry", e);
329 }
330}
331
41de9581 332function add_headline_entry(article, feed, no_effects) {
c01f40f4
AD
333 try {
334
335 var icon_part = "";
336
d5d56329 337 icon_part = "<img class='icon' src='" + get_feed_icon(feed) + "'/>";
c01f40f4 338
9ed133e7
AD
339 var mark_part = "";
340 var publ_part = "";
341
78ac6caf
AD
342 var tags_part = "";
343
344 if (article.tags.length > 0) {
345
346 tags_part = " " + __("in") + " ";
347
348 for (var i = 0; i < Math.min(5, article.tags.length); i++) {
349 tags_part += "<a href=\"#\" onclick=\"viewfeed('" +
350 article.tags[i] + "')\">" +
351 article.tags[i] + "</a>, ";
352 }
353
354 tags_part = tags_part.replace(/, $/, "");
355 tags_part = "<span class=\"tags\">" + tags_part + "</span>";
356 }
357
9ed133e7
AD
358 if (article.marked)
359 mark_part = "<img title='"+ __("Unstar article")+"' onclick=\"toggle_mark(this, "+article.id+")\" src='images/mark_set.png'>";
360 else
361 mark_part = "<img title='"+__("Star article")+"' onclick=\"toggle_mark(this, "+article.id+")\" src='images/mark_unset.png'>";
362
363 if (article.published)
364 publ_part = "<img title='"+__("Unpublish article")+"' onclick=\"toggle_pub(this, "+article.id+")\" src='images/pub_set.png'>";
365 else
366 publ_part = "<img title='"+__("Publish article")+"' onclick=\"toggle_pub(this, "+article.id+")\" src='images/pub_unset.png'>";
367
41de9581 368 var style = "";
9ed133e7 369
41de9581
AD
370 if (!no_effects) style = "style=\"display : none\"";
371
ed6c208d
AD
372 if (article.excerpt.trim() == "")
373 article.excerpt = __("Click to expand article.");
374
41de9581 375 var tmp_html = "<li id=\"A-"+article.id+"\" "+style+">" +
c01f40f4 376 icon_part +
85629f6c 377
6361fd20 378 "<div class='digest-check'>" +
85629f6c
AD
379 mark_part +
380 publ_part +
381 "<img title='" + __("Mark as read") + "' onclick=\"view("+article.id+", true)\" src='images/digest_checkbox.png'>" +
6361fd20 382 "</div>" +
22933e5e
AD
383 "<a target=\"_blank\" href=\""+article.link+"\""+
384 "onclick=\"return view("+article.id+")\" class='title'>" +
385 article.title + "</a>" +
118e9399 386 "<div class='body'>" +
d8ea9902 387 "<div title=\""+__("Click to expand article")+"\" onclick=\"zoom(this, "+article.id+")\" class='excerpt'>" +
118e9399 388 article.excerpt + "</div>" +
c524d7e6 389 "<div class='info'><a href=\#\" onclick=\"viewfeed("+feed.id+")\">" +
78ac6caf 390 feed.title + "</a> " + tags_part + " @ " +
b41c2549 391 new Date(article.updated * 1000) + "</div>" +
1ca8997b 392 "</div></li>";
c01f40f4
AD
393
394 $("headlines-content").innerHTML += tmp_html;
395
41de9581
AD
396 if (!no_effects)
397 window.setTimeout('article_appear(' + article.id + ')', 100);
398
c01f40f4
AD
399 } catch (e) {
400 exception_error("add_headline_entry", e);
401 }
402}
403
c1b5cd23
AD
404function expand_feeds() {
405 try {
406 _feedlist_expanded = true;
407
408 redraw_feedlist(last_feeds);
409
410 } catch (e) {
411 exception_error("expand_feeds", e);
412 }
413}
414
415function redraw_feedlist(feeds) {
416 try {
417
418 $('feeds-content').innerHTML = "";
419
420 var limit = 10;
421
422 if (_feedlist_expanded) limit = feeds.length;
423
424 for (var i = 0; i < Math.min(limit, feeds.length); i++) {
425 add_feed_entry(feeds[i]);
426 }
427
428 if (feeds.length > limit) {
429 $('feeds-content').innerHTML += "<li id='F-MORE-PROMPT'>" +
430 "<img src='images/blank_icon.gif'>" +
431 "<a href=\"#\" onclick=\"expand_feeds()\">" +
432 __("%d more...").replace("%d", feeds.length-10) +
433 "</a>" + "</li>";
434 }
435
436 } catch (e) {
437 exception_error("redraw_feedlist", e);
438 }
439}
440
d5d56329
AD
441function parse_feeds(transport) {
442 try {
443
c1b5cd23
AD
444 if (!transport.responseXML) return;
445
d5d56329
AD
446 var feeds = transport.responseXML.getElementsByTagName('feeds')[0];
447
448 if (feeds) {
449 feeds = eval("(" + feeds.firstChild.nodeValue + ")");
450
e0cebf2a
AD
451 feeds.sort( function (a,b)
452 {
453 if (b.unread != a.unread)
454 return (b.unread - a.unread)
455 else
456 if (a.title > b.title)
457 return 1;
458 else if (a.title < b.title)
459 return -1;
460 else
461 return 0;
462 });
463
4311cc7e
AD
464 var all_articles = find_feed(feeds, -4);
465
466 update_title(all_articles.unread);
467
d5d56329
AD
468 last_feeds = feeds;
469
c1b5cd23 470 redraw_feedlist(feeds);
d5d56329
AD
471 }
472
473 } catch (e) {
474 exception_error("parse_feeds", e);
475 }
476}
477
41de9581 478function parse_headlines(transport, replace, no_effects) {
d5d56329 479 try {
c1b5cd23
AD
480 if (!transport.responseXML) return;
481
e4c530dc
AD
482 var seq = transport.responseXML.getElementsByTagName('seq')[0];
483
484 if (seq) {
485 seq = seq.firstChild.nodeValue;
486 if (seq != _update_seq) {
487 console.log("parse_headlines: wrong sequence received.");
488 return;
489 }
490 } else {
491 return;
492 }
493
d5d56329 494 var headlines = transport.responseXML.getElementsByTagName('headlines')[0];
78ac6caf 495 var headlines_title = transport.responseXML.getElementsByTagName('headlines-title')[0];
d5d56329 496
78ac6caf 497 if (headlines && headlines_title) {
d5d56329
AD
498 headlines = eval("(" + headlines.firstChild.nodeValue + ")");
499
78ac6caf
AD
500 var title = headlines_title.firstChild.nodeValue;
501
502 $("headlines-title").innerHTML = title;
503
6eed9e80
AD
504 if (replace) {
505 $('headlines-content').innerHTML = '';
506 Element.hide('headlines-content');
507 }
d5d56329 508
c1b5cd23 509 var pr = $('H-MORE-PROMPT');
d5d56329
AD
510
511 if (pr) pr.parentNode.removeChild(pr);
512
6eed9e80
AD
513 var inserted = false;
514
d5d56329
AD
515 for (var i = 0; i < headlines.length; i++) {
516
517 if (!$('A-' + headlines[i].id)) {
518 add_headline_entry(headlines[i],
41de9581 519 find_feed(last_feeds, headlines[i].feed_id), !no_effects);
6eed9e80
AD
520
521 inserted = $("A-" + headlines[i].id);
d5d56329
AD
522 }
523 }
524
525 if (pr) {
526 $('headlines-content').appendChild(pr);
41de9581 527 if (!no_effects) new Effect.ScrollTo(inserted);
d5d56329 528 } else {
c1b5cd23 529 $('headlines-content').innerHTML += "<li id='H-MORE-PROMPT'>" +
46360a96
AD
530 "<div class='body'>" +
531 "<a href=\"javascript:catchup_visible_articles()\">" +
532 __("Mark as read") + "</a> | " +
533 "<a href=\"javascript:load_more()\">" +
534 __("Load more...") + "</a>" +
eb4f33ec
AD
535 "<img style=\"display : none\" "+
536 "id=\"H-LOADING-IMG\" src='images/indicator_tiny.gif'>" +
46360a96 537 "</div></li>";
d5d56329
AD
538 }
539
41de9581
AD
540 if (replace && !no_effects)
541 new Effect.Appear('headlines-content', {duration : 0.3});
6eed9e80
AD
542
543 //new Effect.Appear('headlines-content');
d5d56329
AD
544 }
545
546 } catch (e) {
547 exception_error("parse_headlines", e);
548 }
549}
550
d5d56329 551function init() {
c01f40f4
AD
552 try {
553
554 new Ajax.Request("backend.php", {
555 parameters: "backend.php?op=rpc&subop=digest-init",
556 onComplete: function(transport) {
d5d56329 557 parse_feeds(transport);
1ca8997b 558 window.setTimeout('viewfeed(-4)', 100);
d5d56329 559 _update_timeout = window.setTimeout('update()', 5*1000);
c01f40f4
AD
560 } });
561
562 } catch (e) {
563 exception_error("digest_init", e);
564 }
565}
6361fd20 566
9ed133e7 567function toggle_mark(mark_img, id) {
6361fd20
AD
568
569 try {
570
571 var query = "?op=rpc&id=" + id + "&subop=mark";
572
573 query = query + "&afid=" + _active_feed_id;
574 query = query + "&omode=c";
575
576 if (!mark_img) return;
577
6361fd20
AD
578 if (mark_img.src.match("mark_unset")) {
579 mark_img.src = mark_img.src.replace("mark_unset", "mark_set");
580 mark_img.alt = __("Unstar article");
581 query = query + "&mark=1";
582 } else {
583 mark_img.alt = __("Please wait...");
584 query = query + "&mark=0";
585
586 mark_img.src = mark_img.src.replace("mark_set", "mark_unset");
587 mark_img.alt = __("Star article");
588 }
589
590 new Ajax.Request("backend.php", {
591 parameters: query,
592 onComplete: function(transport) {
9ed133e7 593 update();
6361fd20
AD
594 } });
595
596 } catch (e) {
9ed133e7 597 exception_error("toggle_mark", e);
6361fd20
AD
598 }
599}
600
9ed133e7 601function toggle_pub(mark_img, id, note) {
6361fd20
AD
602
603 try {
604
605 var query = "?op=rpc&id=" + id + "&subop=publ";
606
607 query = query + "&afid=" + _active_feed_id;
608
609 if (note != undefined) {
610 query = query + "&note=" + param_escape(note);
611 } else {
612 query = query + "&note=undefined";
613 }
614
615 query = query + "&omode=c";
616
617 if (!mark_img) return;
618
619 if (mark_img.src.match("pub_unset") || note != undefined) {
620 mark_img.src = mark_img.src.replace("pub_unset", "pub_set");
621 mark_img.alt = __("Unpublish article");
622 query = query + "&pub=1";
623
624 } else {
625 mark_img.alt = __("Please wait...");
626 query = query + "&pub=0";
627
628 mark_img.src = mark_img.src.replace("pub_set", "pub_unset");
629 mark_img.alt = __("Publish article");
630 }
631
632 new Ajax.Request("backend.php", {
633 parameters: query,
634 onComplete: function(transport) {
9ed133e7 635 update();
6361fd20
AD
636 } });
637
638 } catch (e) {
9ed133e7 639 exception_error("toggle_pub", e);
6361fd20
AD
640 }
641}
642
e0cebf2a
AD
643function fatal_error(code, msg) {
644 try {
645
646 if (code == 6) {
647 window.location.href = "digest.php";
648 } else if (code == 5) {
649 window.location.href = "update.php";
650 } else {
651
652 if (msg == "") msg = "Unknown error";
653
654 console.error("Fatal error: " + code + "\n" +
655 msg);
656
657 }
658
659 } catch (e) {
660 exception_error("fatalError", e);
661 }
662}
663
664function fatal_error_check(transport) {
665 try {
666 if (transport.responseXML) {
667 var error = transport.responseXML.getElementsByTagName("error")[0];
668
669 if (error) {
670 var code = error.getAttribute("error-code");
671 var msg = error.getAttribute("error-msg");
672 if (code != 0) {
673 fatal_error(code, msg);
674 return false;
675 }
676 }
677 }
678 } catch (e) {
679 exception_error("fatal_error_check", e);
680 }
681 return true;
682}
683
c1b5cd23
AD
684function feed_mi(elem) {
685 try {
686 var imgs = elem.getElementsByTagName('IMG');
687 var spans = elem.getElementsByTagName('SPAN');
688
689 for (var i = 0; i < imgs.length; i++) {
690 if (imgs[i].className == "dismiss")
691 Element.show(imgs[i]);
692 }
693
694 for (var i = 0; i < spans.length; i++) {
695 if (spans[i].className == "unread")
696 Element.hide(spans[i]);
697 }
698
699
700 } catch (e) {
701 exception_error("feed_mi", e);
702 }
703}
704
705function feed_mo(elem) {
706 try {
707 var imgs = elem.getElementsByTagName('IMG');
708 var spans = elem.getElementsByTagName('SPAN');
709
710 for (var i = 0; i < imgs.length; i++) {
711 if (imgs[i].className == "dismiss")
712 Element.hide(imgs[i]);
713 }
714
715 for (var i = 0; i < spans.length; i++) {
716 if (spans[i].className == "unread")
717 Element.show(spans[i]);
718 }
e0cebf2a 719
c1b5cd23
AD
720 } catch (e) {
721 exception_error("feed_mo", e);
722 }
723}
4311cc7e
AD
724
725function update_title(unread) {
726 try {
727 document.title = "Tiny Tiny RSS";
728
729 if (unread > 0)
730 document.title += " (" + unread + ")";
731
732 } catch (e) {
733 exception_error("update_title", e);
734 }
735}
736