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