]>
git.wh0rd.org - tt-rss.git/blob - feedlist.js
1 var _feed_cur_page
= 0;
2 var _infscroll_disable
= 0;
3 var _infscroll_request_sent
= 0;
4 var feed_under_pointer
= undefined;
6 var mouse_is_down
= false;
10 var resize_enabled
= false;
11 var selection_disabled
= false;
12 var counters_last_request
= 0;
14 function toggle_sortable_feedlist(enabled
) {
18 Sortable
.create('feedList', {onChange
: feedlist_dragsorted
, only
: "feedCat"});
20 Sortable
.destroy('feedList');
24 exception_error("toggle_sortable_feedlist", e
);
28 function viewCategory(cat
) {
29 viewfeed(cat
, '', true);
33 function render_offline_feedlist() {
35 var tmp
= "<ul class=\"feedList\" id=\"feedList\">";
37 var rs
= db
.execute("SELECT id,title,has_icon FROM feeds ORDER BY title");
39 while (rs
.isValidRow()) {
42 var title
= rs
.field(1);
43 var has_icon
= rs
.field(2);
45 var rs_u
= db
.execute("SELECT SUM(unread) FROM articles WHERE feed_id = ?",
49 if (rs_u
.isValidRow()) {
50 unread
= rs_u
.field(0);
51 if (!unread
) unread
= 0;
58 feed_icon
= "<img id='FIMG-"+id
+"' src='" + "icons/" + id
+ ".ico'>";
60 feed_icon
= "<img id='FIMG-"+id
+"' src='images/blank_icon.gif'>";
63 var row_class
= "feed";
66 row_class
+= "Unread";
67 fctr_class
= "feedCtrHasUnread";
69 fctr_class
= "feedCtrNoUnread";
72 var link
= "<a title=\"FIXME\" id=\"FEEDL-"+id
+"\""+
73 "href=\"javascript:viewfeed('"+id
+"', '', false, '', false, 0);\">"+
76 tmp
+= "<li id='FEEDR-"+id
+"' class="+row_class
+">" + feed_icon
+
77 "<span id=\"FEEDN-"+id
+"\">" + link
+ "</span>";
79 tmp
+= " <span class='"+fctr_class
+"' id=\"FEEDCTR-"+id
+"\">" +
80 "(<span id=\"FEEDU-"+id
+"\">"+unread
+"</span>)</span>";
91 exception_error("render_offline_feedlist", e
);
95 function render_feedlist(data
) {
98 var f
= document
.getElementById("feeds-frame");
100 cache_invalidate("FEEDLIST");
101 cache_inject("FEEDLIST", data
, getInitParam("num_feeds"));
105 exception_error("render_feedlist", e
);
109 function feedlist_callback2(transport
) {
111 debug("feedlist_callback2");
112 if (!transport_error_check(transport
)) return;
113 render_feedlist(transport
.responseText
);
115 exception_error("feedlist_callback2", e
);
119 function viewNextFeedPage() {
121 //if (!getActiveFeedId()) return;
123 debug("viewNextFeedPage: calling viewfeed(), p: " + _feed_cur_page
+1);
125 viewfeed(getActiveFeedId(), undefined, activeFeedIsCat(), undefined,
126 undefined, _feed_cur_page
+1);
129 exception_error("viewNextFeedPage", e
);
133 function viewfeed_offline(feed_id
, subop
, is_cat
, subop_param
, skip_history
, offset
) {
137 loading_set_progress(100);
139 clean_feed_selections();
141 setActiveFeedId(feed_id
, is_cat
);
144 var feedr
= document
.getElementById("FEEDR-" + feed_id
);
145 if (feedr
&& !feedr
.className
.match("Selected")) {
146 feedr
.className
= feedr
.className
+ "Selected";
149 var feedr
= document
.getElementById("FCAT-" + feed_id
);
150 if (feedr
&& !feedr
.className
.match("Selected")) {
151 feedr
.className
= feedr
.className
+ "Selected";
155 disableContainerChildren("headlinesToolbar", false);
156 Form
.enable("main_toolbar_form");
158 var f
= document
.getElementById("headlines-frame");
160 if (reply
.offset
== 0) {
161 debug("resetting headlines scrollTop");
167 var container
= document
.getElementById("headlines-frame");
171 rs
= db
.execute("SELECT title FROM feeds WHERE id = ?", [feed_id
]);
173 if (rs
.isValidRow()) {
175 var feed_title
= rs
.field(0);
178 tmp
+= "<div id=\"headlinesContainer\">";
180 tmp
+= "<div class=\"headlinesSubToolbar\">";
181 tmp
+= "<div id=\"subtoolbar_ftitle\">";
185 var sel_all_link
= "javascript:selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', true, '', true)";
186 var sel_unread_link
= "javascript:selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', true, 'Unread', true)";
187 var sel_none_link
= "javascript:selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', false)";
188 var sel_inv_link
= "javascript:invertHeadlineSelection()";
190 tmp
+= __('Select:')+
191 " <a href=\""+sel_all_link
+"\">"+__('All')+"</a>, "+
192 "<a href=\""+sel_unread_link
+"\">"+__('Unread')+"</a>, "+
193 "<a href=\""+sel_inv_link
+"\">"+__('Invert')+"</a>, "+
194 "<a href=\""+sel_none_link
+"\">"+__('None')+"</a>";
196 tmp
+= " ";
200 tmp
+= "<div id=\"headlinesInnerContainer\" onscroll=\"headlines_scroll_handler()\">";
202 tmp
+= "<table class=\"headlinesList\" id=\"headlinesList\" cellspacing=\"0\">";
206 var rs
= db
.execute("SELECT * FROM articles WHERE feed_id = ?", [feed_id
]);
210 while (rs
.isValidRow()) {
212 var id
= rs
.fieldByName("id");
213 var feed_id
= rs
.fieldByName("feed_id");
217 var row_class
= (line_num
% 2) ? "even" : "odd";
219 if (rs
.fieldByName("unread") == "1") {
220 row_class
+= "Unread";
223 if (rs
.fieldByName("marked") == "1") {
224 marked_pic
= "<img id=\"FMPIC-"+id
+"\" "+
225 "src=\"images/mark_set.png\" class=\"markedPic\""+
226 "alt=\"Unstar article\" onclick='javascript:tMark("+id
+")'>";
228 marked_pic
= "<img id=\"FMPIC-"+id
+"\" "+
229 "src=\"images/mark_unset.png\" class=\"markedPic\""+
230 "alt=\"Star article\" onclick='javascript:tMark("+id
+")'>";
233 var mouseover_attrs
= "onmouseover='postMouseIn($id)' "+
234 "onmouseout='postMouseOut($id)'";
236 tmp
+= "<tr class='"+row_class
+"' id='RROW-"+id
+"' "+mouseover_attrs
+">";
238 tmp
+= "<td class='hlUpdPic'> </td>";
240 tmp
+= "<td class='hlSelectRow'>"+
241 "<input type=\"checkbox\" onclick=\"tSR(this)\" id=\"RCHK-"+id
+"\"></td>";
243 tmp
+= "<td class='hlMarkedPic'>"+marked_pic
+"</td>";
245 tmp
+= "<td onclick='view("+id
+","+feed_id
+")' "+
246 "class='hlContent' valign='middle'>";
248 tmp
+= "<a target=\"_blank\" id=\"RTITLE-$id\" href=\"" +
249 rs
.fieldByName("link") + "\"" +
250 "onclick=\"return view("+id
+","+feed_id
+");\">"+
251 rs
.fieldByName("title");
253 var content_preview
= truncate_string(strip_tags(rs
.fieldByName("content")),
256 tmp
+= "<span class=\"contentPreview\"> - "+content_preview
+"</span>";
262 tmp
+= "<td class=\"hlUpdated\" onclick='view("+id
+","+feed_id
+")'>"+
263 "<nobr>"+rs
.fieldByName("updated").substring(0,16)+"</nobr></td>";
273 tmp
+= "</div></div>";
277 container
.innerHTML
= tmp
;
279 var ids
= getSelectedArticleIds2();
281 //container.innerHTML = container.innerHTML + tmp;
283 for (var i
= 0; i
< ids
.length
; i
++) {
284 markHeadline(ids
[i
]);
293 exception_error("viewfeed_offline", e
);
297 function viewfeed(feed
, subop
, is_cat
, subop_param
, skip_history
, offset
) {
300 if (offline_mode
) return viewfeed_offline(feed
, subop
, is_cat
, subop_param
,
301 skip_history
, offset
);
303 // if (!offset) page_offset = 0;
305 last_requested_article
= 0;
306 //counters_last_request = 0;
308 if (feed
== getActiveFeedId()) {
309 cache_invalidate("F:" + feed
);
312 /* if (getInitParam("theme") == "" || getInitParam("theme") == "compact") {
313 if (getInitParam("hide_feedlist") == 1) {
314 Element.hide("feeds-holder");
318 var force_nocache
= false;
323 page_offset
= offset
;
327 _infscroll_disable
= 0;
330 if (getActiveFeedId() != feed
) {
333 _infscroll_disable
= 0;
336 if (page_offset
!= 0 && !subop
) {
337 var date
= new Date();
338 var timestamp
= Math
.round(date
.getTime() / 1000);
340 debug("<b>" + _infscroll_request_sent
+ " : " + timestamp
+ "</b>");
342 if (_infscroll_request_sent
&& _infscroll_request_sent
+ 30 > timestamp
) {
343 debug("infscroll request in progress, aborting");
347 _infscroll_request_sent
= timestamp
;
354 Form
.enable("main_toolbar_form");
356 var toolbar_form
= document
.forms
["main_toolbar_form"];
357 var toolbar_query
= Form
.serialize("main_toolbar_form");
359 if (toolbar_form
.query
) {
360 if (toolbar_form
.query
.value
!= "") {
361 force_nocache
= true;
363 toolbar_form
.query
.value
= "";
366 var query
= "backend.php?op=viewfeed&feed=" + feed
+ "&" +
367 toolbar_query
+ "&subop=" + param_escape(subop
);
369 if (document
.getElementById("search_form")) {
370 var search_query
= Form
.serialize("search_form");
371 query
= query
+ "&" + search_query
;
373 force_nocache
= true;
376 // debug("IS_CAT_STORED: " + activeFeedIsCat() + ", IS_CAT: " + is_cat);
378 if (subop
== "MarkAllRead") {
380 var show_next_feed
= getInitParam("on_catchup_show_next_feed") == "1";
382 if (show_next_feed
) {
384 if (!activeFeedIsCat()) {
386 var feedlist
= document
.getElementById('feedList');
388 var next_unread_feed
= getRelativeFeedId(feedlist
,
391 if (!next_unread_feed
) {
392 next_unread_feed
= getRelativeFeedId(feedlist
,
396 if (next_unread_feed
) {
397 query
= query
+ "&nuf=" + param_escape(next_unread_feed
);
398 //setActiveFeedId(next_unread_feed);
399 feed
= next_unread_feed
;
403 var next_unread_feed
= getNextUnreadCat(feed
);
405 /* we don't need to specify that our next feed is actually
406 a category, because we're in the is_cat mode by definition
409 if (next_unread_feed
&& show_next_feed
) {
410 query
= query
+ "&nuf=" + param_escape(next_unread_feed
);
411 feed
= next_unread_feed
;
419 query
= query
+ "&cat=1";
422 if (page_offset
!= 0) {
423 query
= query
+ "&skip=" + page_offset
;
425 // to prevent duplicate feed titles when showing grouped vfeeds
426 if (vgroup_last_feed
) {
427 query
= query
+ "&vgrlf=" + param_escape(vgroup_last_feed
);
431 var date
= new Date();
432 var timestamp
= Math
.round(date
.getTime() / 1000);
433 query
= query
+ "&ts=" + timestamp
435 disableContainerChildren("headlinesToolbar", false);
436 Form
.enable("main_toolbar_form");
438 // for piggybacked counters
440 if (tagsAreDisplayed()) {
441 query
= query
+ "&omode=lt";
443 query
= query
+ "&omode=flc";
446 if (!async_counters_work
) {
447 query
= query
+ "&csync=true";
452 var container
= document
.getElementById("headlinesInnerContainer");
454 /* if (container && page_offset == 0 && !isCdmMode()) {
455 new Effect.Fade(container, {duration: 1, to: 0.01,
456 queue: { position:'end', scope: 'FEEDL-' + feed, limit: 1 } } );
461 if (!is_cat
) unread_ctr
= get_feed_unread(feed
);
463 var cache_check
= false;
465 if (unread_ctr
!= -1 && !page_offset
&& !force_nocache
&& !subop
) {
467 var cache_prefix
= "";
475 cache_check
= cache_check_param(cache_prefix
+ feed
, unread_ctr
);
476 debug("headline cache check: " + cache_check
);
480 var f
= document
.getElementById("headlines-frame");
482 clean_feed_selections();
484 setActiveFeedId(feed
, is_cat
);
487 var feedr
= document
.getElementById("FEEDR-" + feed
);
488 if (feedr
&& !feedr
.className
.match("Selected")) {
489 feedr
.className
= feedr
.className
+ "Selected";
492 var feedr
= document
.getElementById("FCAT-" + feed_id
);
493 if (feedr
&& !feedr
.className
.match("Selected")) {
494 feedr
.className
= feedr
.className
+ "Selected";
498 f
.innerHTML
= cache_find_param(cache_prefix
+ feed
, unread_ctr
);
506 notify_progress("Loading, please wait...", true);
509 new Ajax
.Request(query
, {
510 onComplete: function(transport
) {
511 headlines_callback2(transport
, page_offset
);
516 exception_error("viewfeed", e
);
520 function toggleCollapseCat_af(effect
) {
521 //var caption = elem.id.replace("FCATLIST-", "");
525 var elem
= effect
.element
;
526 var cat
= elem
.id
.replace("FCATLIST-", "");
527 var cap
= document
.getElementById("FCAP-" + cat
);
529 if (Element
.visible(elem
)) {
530 cap
.innerHTML
= cap
.innerHTML
.replace("…", "");
532 if (cap
.innerHTML
.lastIndexOf("…") != cap
.innerHTML
.length
-3) {
533 cap
.innerHTML
= cap
.innerHTML
+ "…";
538 exception_error("toggleCollapseCat_af", e
);
542 function toggleCollapseCat(cat
) {
545 var cat_elem
= document
.getElementById("FCAT-" + cat
);
546 var cat_list
= document
.getElementById("FCATLIST-" + cat
).parentNode
;
547 var caption
= document
.getElementById("FCAP-" + cat
);
549 /* if (cat_list.className.match("invisible")) {
550 cat_list.className = "";
551 caption.innerHTML = caption.innerHTML.replace("...", "");
553 setCookie("ttrss_vf_uclps", "0");
556 cat_list.className = "invisible";
557 caption.innerHTML = caption.innerHTML + "...";
559 setCookie("ttrss_vf_uclps", "1");
565 if (Element
.visible("FCATLIST-" + cat
)) {
566 setCookie("ttrss_vf_uclps", "1");
568 setCookie("ttrss_vf_uclps", "0");
573 if (Element
.visible("FCATLIST-" + cat
)) {
574 setCookie("ttrss_vf_lclps", "1");
576 setCookie("ttrss_vf_lclps", "0");
581 if (Element
.visible("FCATLIST-" + cat
)) {
582 setCookie("ttrss_vf_vclps", "1");
584 setCookie("ttrss_vf_vclps", "0");
588 Effect
.toggle('FCATLIST-' + cat
, 'blind', { duration
: 0.5,
589 afterFinish
: toggleCollapseCat_af
});
591 new Ajax
.Request("backend.php?op=feeds&subop=collapse&cid=" +
595 exception_error("toggleCollapseCat", e
);
599 function feedlist_dragsorted(ctr
) {
601 var elem
= document
.getElementById("feedList");
603 var cats
= elem
.getElementsByTagName("LI");
604 var ordered_cats
= new Array();
606 for (var i
= 0; i
< cats
.length
; i
++) {
607 if (cats
[i
].id
&& cats
[i
].id
.match("FCAT-")) {
608 ordered_cats
.push(cats
[i
].id
.replace("FCAT-", ""));
612 if (ordered_cats
.length
> 0) {
614 var query
= "backend.php?op=feeds&subop=catsort&corder=" +
615 param_escape(ordered_cats
.toString());
619 new Ajax
.Request(query
);
623 exception_error("feedlist_dragsorted", e
);
627 function feedlist_init() {
629 // if (arguments.callee.done) return;
630 // arguments.callee.done = true;
632 loading_set_progress(90);
634 debug("in feedlist init");
636 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
637 document
.onkeydown
= hotkey_handler
;
638 document
.onmousemove
= mouse_move_handler
;
639 document
.onmousedown
= mouse_down_handler
;
640 document
.onmouseup
= mouse_up_handler
;
642 setTimeout("timeout()", 1);
644 if (typeof correctPNG
!= 'undefined') {
648 if (getActiveFeedId()) {
649 //debug("some feed is open on feedlist refresh, reloading");
650 //setTimeout("viewCurrentFeed()", 100);
652 if (getInitParam("cdm_auto_catchup") != 1 && get_feed_unread(-3) > 0) {
653 notify_silent_next();
654 setTimeout("viewfeed(-3)", 100);
660 if (getInitParam("theme") == "") {
661 setTimeout("hide_footer()", 5000);
664 init_collapsable_feedlist(getInitParam("theme"));
666 toggle_sortable_feedlist(isFeedlistSortable());
669 exception_error("feedlist/init", e
);
673 function hide_footer_af(effect
) {
675 var c
= document
.getElementById("content-frame");
678 c
.style
.bottom
= "0px";
680 var ioa
= document
.getElementById("inline_orig_article");
683 ioa
.height
= c
.offsetHeight
;
687 var h
= document
.getElementById("headlines-frame");
690 h
.style
.bottom
= "0px";
695 exception_error("hide_footer_af", e
);
699 function hide_footer() {
701 if (Element
.visible("footer")) {
702 new Effect
.Fade("footer", { afterFinish
: hide_footer_af
});
705 exception_error("hide_footer", e
);
710 function init_hidden_feedlist(theme) {
712 debug("init_hidden_feedlist");
714 if (theme != "" && theme != "compact") return;
716 var fl = document.getElementById("feeds-holder");
717 var fh = document.getElementById("headlines-frame");
718 var fc = document.getElementById("content-frame");
719 var ft = document.getElementById("toolbar");
720 var ff = document.getElementById("footer");
721 var fhdr = document.getElementById("header");
723 var fbtn = document.getElementById("toggle_feeds_btn");
725 if (fbtn) Element.show(fbtn);
727 fl.style.top = fh.offsetTop + "px";
728 fl.style.backgroundColor = "white"; //FIXME
732 fh.style.left = "0px";
733 ft.style.left = "0px";
734 if (fc) fc.style.left = "0px";
735 if (ff) ff.style.left = "0px";
737 if (theme == "compact") {
738 fhdr.style.left = "10px";
739 fl.style.top = (fh.offsetTop + 1) + "px";
743 exception_error("init_hidden_feedlist", e);
747 function init_collapsable_feedlist(theme
) {
749 debug("init_collapsable_feedlist");
751 if (theme
!= "" && theme
!= "compact" && theme
!= "graycube" &&
752 theme
!= "compat") return;
754 var fbtn
= document
.getElementById("collapse_feeds_btn");
756 if (fbtn
) Element
.show(fbtn
);
758 if (getCookie("ttrss_vf_fclps") == 1) {
763 exception_error("init_hidden_feedlist", e
);
768 function mouse_move_handler(e
) {
774 client_y
= window
.event
.clientY
;
775 client_x
= window
.event
.clientX
;
777 client_x
= e
.screenX
;
778 client_y
= e
.screenY
;
783 if (mouse_y
== 0) mouse_y
= client_y
;
784 if (mouse_x
== 0) mouse_x
= client_x
;
786 resize_headlines(mouse_x
- client_x
, mouse_y
- client_y
);
795 exception_error("mouse_move_handler", e
);
799 function enable_selection(b
) {
800 selection_disabled
= !b
;
803 function enable_resize(b
) {
807 function mouse_down_handler(e
) {
810 /* do not prevent right click */
811 if (e
.button
&& e
.button
== 2) return;
813 if (resize_enabled
) {
814 mouse_is_down
= true;
817 document
.onselectstart = function() { return false; };
821 if (selection_disabled
) {
822 document
.onselectstart = function() { return false; };
827 exception_error("mouse_move_handler", e
);
831 function mouse_up_handler(e
) {
833 mouse_is_down
= false;
835 if (!selection_disabled
) {
836 document
.onselectstart
= null;
837 var e
= document
.getElementById("headlineActionsBody");
838 if (e
) Element
.hide(e
);
842 exception_error("mouse_move_handler", e
);
846 function request_counters_real() {
850 if (offline_mode
) return;
852 debug("requesting counters...");
854 var query
= "backend.php?op=rpc&subop=getAllCounters";
856 if (tagsAreDisplayed()) {
857 query
= query
+ "&omode=tl";
859 query
= query
+ "&omode=flc";
862 new Ajax
.Request(query
, {
863 onComplete: function(transport
) {
865 all_counters_callback2(transport
, true);
867 exception_error("viewfeed/getcounters", e
);
872 exception_error("request_counters_real", e
);
877 function request_counters() {
881 if (getInitParam("bw_limit") == "1") return;
883 var date
= new Date();
884 var timestamp
= Math
.round(date
.getTime() / 1000);
886 // if (getInitParam("sync_counters") == "1" ||
887 // timestamp - counters_last_request > 10) {
889 if (timestamp
- counters_last_request
> 15) {
890 debug("scheduling request of counters...");
891 window
.setTimeout("request_counters_real()", 1000);
892 counters_last_request
= timestamp
;
894 debug("request_counters: rate limit reached: " + (timestamp
- counters_last_request
));
898 exception_error("request_counters", e
);