]>
git.wh0rd.org - tt-rss.git/blob - feedlist.js
05867d7b3f874f4ef74dd201e149ebe8d1a6665d
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
.isValidRow()) {
50 unread
= rs_u
.field(0);
51 if (!unread
) unread
= 0;
57 feed_icon
= "<img id='FIMG-"+id
+"' src='" + "icons/" + id
+ ".ico'>";
59 feed_icon
= "<img id='FIMG-"+id
+"' src='images/blank_icon.gif'>";
62 var row_class
= "feed";
65 row_class
+= "Unread";
66 fctr_class
= "feedCtrHasUnread";
68 fctr_class
= "feedCtrNoUnread";
71 var link
= "<a title=\"FIXME\" id=\"FEEDL-"+id
+"\""+
72 "href=\"javascript:viewfeed('"+id
+"', '', false, '', false, 0);\">"+
75 tmp
+= "<li id='FEEDR-"+id
+"' class="+row_class
+">" + feed_icon
+
76 "<span id=\"FEEDN-"+id
+"\">" + link
+ "</span>";
78 tmp
+= " <span class='"+fctr_class
+"' id=\"FEEDCTR-"+id
+"\">" +
79 "(<span id=\"FEEDU-"+id
+"\">"+unread
+"</span>)</span>";
90 exception_error("render_offline_feedlist", e
);
94 function render_feedlist(data
) {
97 var f
= document
.getElementById("feeds-frame");
99 cache_invalidate("FEEDLIST");
100 cache_inject("FEEDLIST", data
, getInitParam("num_feeds"));
104 exception_error("render_feedlist", e
);
108 function feedlist_callback2(transport
) {
110 debug("feedlist_callback2");
111 if (!transport_error_check(transport
)) return;
112 render_feedlist(transport
.responseText
);
114 exception_error("feedlist_callback2", e
);
118 function viewNextFeedPage() {
120 //if (!getActiveFeedId()) return;
122 debug("viewNextFeedPage: calling viewfeed(), p: " + _feed_cur_page
+1);
124 viewfeed(getActiveFeedId(), undefined, activeFeedIsCat(), undefined,
125 undefined, _feed_cur_page
+1);
128 exception_error("viewNextFeedPage", e
);
132 function viewfeed_offline(feed_id
, subop
, is_cat
, subop_param
, skip_history
, offset
) {
136 loading_set_progress(100);
138 clean_feed_selections();
140 setActiveFeedId(feed_id
, is_cat
);
143 var feedr
= document
.getElementById("FEEDR-" + feed_id
);
144 if (feedr
&& !feedr
.className
.match("Selected")) {
145 feedr
.className
= feedr
.className
+ "Selected";
148 var feedr
= document
.getElementById("FCAT-" + feed_id
);
149 if (feedr
&& !feedr
.className
.match("Selected")) {
150 feedr
.className
= feedr
.className
+ "Selected";
154 var f
= document
.getElementById("headlines-frame");
156 if (reply
.offset
== 0) {
157 debug("resetting headlines scrollTop");
163 var container
= document
.getElementById("headlines-frame");
167 rs
= db
.execute("SELECT title FROM feeds WHERE id = ?", [feed_id
]);
169 if (rs
.isValidRow()) {
171 var feed_title
= rs
.field(0);
174 tmp
+= "<div id=\"headlinesContainer\">";
176 tmp
+= "<div class=\"headlinesSubToolbar\">";
177 tmp
+= "<div id=\"subtoolbar_ftitle\">";
181 var sel_all_link
= "javascript:selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', true, '', true)";
182 var sel_unread_link
= "javascript:selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', true, 'Unread', true)";
183 var sel_none_link
= "javascript:selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', false)";
184 var sel_inv_link
= "javascript:invertHeadlineSelection()";
186 tmp
+= __('Select:')+
187 " <a href=\""+sel_all_link
+"\">"+__('All')+"</a>, "+
188 "<a href=\""+sel_unread_link
+"\">"+__('Unread')+"</a>, "+
189 "<a href=\""+sel_inv_link
+"\">"+__('Invert')+"</a>, "+
190 "<a href=\""+sel_none_link
+"\">"+__('None')+"</a>";
192 tmp
+= " ";
196 tmp
+= "<div id=\"headlinesInnerContainer\" onscroll=\"headlines_scroll_handler()\">";
198 tmp
+= "<table class=\"headlinesList\" id=\"headlinesList\" cellspacing=\"0\">";
202 var rs
= db
.execute("SELECT * FROM articles WHERE feed_id = ?", [feed_id
]);
206 while (rs
.isValidRow()) {
208 var id
= rs
.fieldByName("id");
209 var feed_id
= rs
.fieldByName("feed_id");
213 var row_class
= (line_num
% 2) ? "even" : "odd";
215 if (rs
.fieldByName("unread") == "1") {
216 row_class
+= "Unread";
219 if (rs
.fieldByName("marked") == "1") {
220 marked_pic
= "<img id=\"FMPIC-"+id
+"\" "+
221 "src=\"images/mark_set.png\" class=\"markedPic\""+
222 "alt=\"Unstar article\" onclick='javascript:tMark("+id
+")'>";
224 marked_pic
= "<img id=\"FMPIC-"+id
+"\" "+
225 "src=\"images/mark_unset.png\" class=\"markedPic\""+
226 "alt=\"Star article\" onclick='javascript:tMark("+id
+")'>";
229 var mouseover_attrs
= "onmouseover='postMouseIn($id)' "+
230 "onmouseout='postMouseOut($id)'";
232 tmp
+= "<tr class='"+row_class
+"' id='RROW-"+id
+"' "+mouseover_attrs
+">";
234 tmp
+= "<td class='hlUpdPic'> </td>";
236 tmp
+= "<td class='hlSelectRow'>"+
237 "<input type=\"checkbox\" onclick=\"tSR(this)\" id=\"RCHK-"+id
+"\"></td>";
239 tmp
+= "<td class='hlMarkedPic'>"+marked_pic
+"</td>";
241 tmp
+= "<td onclick='view("+id
+","+feed_id
+")' "+
242 "class='hlContent' valign='middle'>";
244 tmp
+= "<a id=\"RTITLE-$id\" href=\"" +
245 param_escape(rs
.fieldByName("link")) + "\"" +
246 "onclick=\"return view("+id
+","+feed_id
+");\">"+
247 rs
.fieldByName("title");
249 var content_preview
= truncate_string(strip_tags(rs
.fieldByName("content")),
252 tmp
+= "<span class=\"contentPreview\"> - "+content_preview
+"</span>";
258 tmp
+= "<td class=\"hlUpdated\" onclick='view("+id
+","+feed_id
+")'>"+
259 "<nobr>"+rs
.fieldByName("updated").substring(0,16)+"</nobr></td>";
269 tmp
+= "</div></div>";
273 container
.innerHTML
= tmp
;
275 var ids
= getSelectedArticleIds2();
277 //container.innerHTML = container.innerHTML + tmp;
279 for (var i
= 0; i
< ids
.length
; i
++) {
280 markHeadline(ids
[i
]);
289 exception_error("viewfeed_offline", e
);
293 function viewfeed(feed
, subop
, is_cat
, subop_param
, skip_history
, offset
) {
296 if (offline_mode
) return viewfeed_offline(feed
, subop
, is_cat
, subop_param
,
297 skip_history
, offset
);
299 // if (!offset) page_offset = 0;
301 last_requested_article
= 0;
302 //counters_last_request = 0;
304 if (feed
== getActiveFeedId()) {
305 cache_invalidate("F:" + feed
);
308 /* if (getInitParam("theme") == "" || getInitParam("theme") == "compact") {
309 if (getInitParam("hide_feedlist") == 1) {
310 Element.hide("feeds-holder");
314 var force_nocache
= false;
319 page_offset
= offset
;
323 _infscroll_disable
= 0;
326 if (getActiveFeedId() != feed
) {
329 _infscroll_disable
= 0;
332 if (page_offset
!= 0 && !subop
) {
333 var date
= new Date();
334 var timestamp
= Math
.round(date
.getTime() / 1000);
336 debug("<b>" + _infscroll_request_sent
+ " : " + timestamp
+ "</b>");
338 if (_infscroll_request_sent
&& _infscroll_request_sent
+ 30 > timestamp
) {
339 debug("infscroll request in progress, aborting");
343 _infscroll_request_sent
= timestamp
;
350 Form
.enable("main_toolbar_form");
352 var toolbar_form
= document
.forms
["main_toolbar_form"];
353 var toolbar_query
= Form
.serialize("main_toolbar_form");
355 if (toolbar_form
.query
) {
356 if (toolbar_form
.query
.value
!= "") {
357 force_nocache
= true;
359 toolbar_form
.query
.value
= "";
362 var query
= "backend.php?op=viewfeed&feed=" + feed
+ "&" +
363 toolbar_query
+ "&subop=" + param_escape(subop
);
365 if (document
.getElementById("search_form")) {
366 var search_query
= Form
.serialize("search_form");
367 query
= query
+ "&" + search_query
;
369 force_nocache
= true;
372 // debug("IS_CAT_STORED: " + activeFeedIsCat() + ", IS_CAT: " + is_cat);
374 if (subop
== "MarkAllRead") {
376 var show_next_feed
= getInitParam("on_catchup_show_next_feed") == "1";
378 if (show_next_feed
) {
380 if (!activeFeedIsCat()) {
382 var feedlist
= document
.getElementById('feedList');
384 var next_unread_feed
= getRelativeFeedId(feedlist
,
387 if (!next_unread_feed
) {
388 next_unread_feed
= getRelativeFeedId(feedlist
,
392 if (next_unread_feed
) {
393 query
= query
+ "&nuf=" + param_escape(next_unread_feed
);
394 //setActiveFeedId(next_unread_feed);
395 feed
= next_unread_feed
;
399 var next_unread_feed
= getNextUnreadCat(feed
);
401 /* we don't need to specify that our next feed is actually
402 a category, because we're in the is_cat mode by definition
405 if (next_unread_feed
&& show_next_feed
) {
406 query
= query
+ "&nuf=" + param_escape(next_unread_feed
);
407 feed
= next_unread_feed
;
415 query
= query
+ "&cat=1";
418 if (page_offset
!= 0) {
419 query
= query
+ "&skip=" + page_offset
;
421 // to prevent duplicate feed titles when showing grouped vfeeds
422 if (vgroup_last_feed
) {
423 query
= query
+ "&vgrlf=" + param_escape(vgroup_last_feed
);
427 var date
= new Date();
428 var timestamp
= Math
.round(date
.getTime() / 1000);
429 query
= query
+ "&ts=" + timestamp
431 disableContainerChildren("headlinesToolbar", false);
432 Form
.enable("main_toolbar_form");
434 // for piggybacked counters
436 if (tagsAreDisplayed()) {
437 query
= query
+ "&omode=lt";
439 query
= query
+ "&omode=flc";
442 if (!async_counters_work
) {
443 query
= query
+ "&csync=true";
448 var container
= document
.getElementById("headlinesInnerContainer");
450 /* if (container && page_offset == 0 && !isCdmMode()) {
451 new Effect.Fade(container, {duration: 1, to: 0.01,
452 queue: { position:'end', scope: 'FEEDL-' + feed, limit: 1 } } );
457 if (!is_cat
) unread_ctr
= get_feed_unread(feed
);
459 var cache_check
= false;
461 if (unread_ctr
!= -1 && !page_offset
&& !force_nocache
&& !subop
) {
463 var cache_prefix
= "";
471 cache_check
= cache_check_param(cache_prefix
+ feed
, unread_ctr
);
472 debug("headline cache check: " + cache_check
);
476 var f
= document
.getElementById("headlines-frame");
478 clean_feed_selections();
480 setActiveFeedId(feed
, is_cat
);
483 var feedr
= document
.getElementById("FEEDR-" + feed
);
484 if (feedr
&& !feedr
.className
.match("Selected")) {
485 feedr
.className
= feedr
.className
+ "Selected";
488 var feedr
= document
.getElementById("FCAT-" + feed_id
);
489 if (feedr
&& !feedr
.className
.match("Selected")) {
490 feedr
.className
= feedr
.className
+ "Selected";
494 f
.innerHTML
= cache_find_param(cache_prefix
+ feed
, unread_ctr
);
502 notify_progress("Loading, please wait...", true);
505 new Ajax
.Request(query
, {
506 onComplete: function(transport
) {
507 headlines_callback2(transport
, page_offset
);
512 exception_error("viewfeed", e
);
516 function toggleCollapseCat_af(effect
) {
517 //var caption = elem.id.replace("FCATLIST-", "");
521 var elem
= effect
.element
;
522 var cat
= elem
.id
.replace("FCATLIST-", "");
523 var cap
= document
.getElementById("FCAP-" + cat
);
525 if (Element
.visible(elem
)) {
526 cap
.innerHTML
= cap
.innerHTML
.replace("…", "");
528 if (cap
.innerHTML
.lastIndexOf("…") != cap
.innerHTML
.length
-3) {
529 cap
.innerHTML
= cap
.innerHTML
+ "…";
534 exception_error("toggleCollapseCat_af", e
);
538 function toggleCollapseCat(cat
) {
541 var cat_elem
= document
.getElementById("FCAT-" + cat
);
542 var cat_list
= document
.getElementById("FCATLIST-" + cat
).parentNode
;
543 var caption
= document
.getElementById("FCAP-" + cat
);
545 /* if (cat_list.className.match("invisible")) {
546 cat_list.className = "";
547 caption.innerHTML = caption.innerHTML.replace("...", "");
549 setCookie("ttrss_vf_uclps", "0");
552 cat_list.className = "invisible";
553 caption.innerHTML = caption.innerHTML + "...";
555 setCookie("ttrss_vf_uclps", "1");
561 if (Element
.visible("FCATLIST-" + cat
)) {
562 setCookie("ttrss_vf_uclps", "1");
564 setCookie("ttrss_vf_uclps", "0");
569 if (Element
.visible("FCATLIST-" + cat
)) {
570 setCookie("ttrss_vf_lclps", "1");
572 setCookie("ttrss_vf_lclps", "0");
577 if (Element
.visible("FCATLIST-" + cat
)) {
578 setCookie("ttrss_vf_vclps", "1");
580 setCookie("ttrss_vf_vclps", "0");
584 Effect
.toggle('FCATLIST-' + cat
, 'blind', { duration
: 0.5,
585 afterFinish
: toggleCollapseCat_af
});
587 new Ajax
.Request("backend.php?op=feeds&subop=collapse&cid=" +
591 exception_error("toggleCollapseCat", e
);
595 function feedlist_dragsorted(ctr
) {
597 var elem
= document
.getElementById("feedList");
599 var cats
= elem
.getElementsByTagName("LI");
600 var ordered_cats
= new Array();
602 for (var i
= 0; i
< cats
.length
; i
++) {
603 if (cats
[i
].id
&& cats
[i
].id
.match("FCAT-")) {
604 ordered_cats
.push(cats
[i
].id
.replace("FCAT-", ""));
608 if (ordered_cats
.length
> 0) {
610 var query
= "backend.php?op=feeds&subop=catsort&corder=" +
611 param_escape(ordered_cats
.toString());
615 new Ajax
.Request(query
);
619 exception_error("feedlist_dragsorted", e
);
623 function feedlist_init() {
625 // if (arguments.callee.done) return;
626 // arguments.callee.done = true;
628 loading_set_progress(90);
630 debug("in feedlist init");
632 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
633 document
.onkeydown
= hotkey_handler
;
634 document
.onmousemove
= mouse_move_handler
;
635 document
.onmousedown
= mouse_down_handler
;
636 document
.onmouseup
= mouse_up_handler
;
638 setTimeout("timeout()", 1);
640 if (typeof correctPNG
!= 'undefined') {
644 if (getActiveFeedId()) {
645 //debug("some feed is open on feedlist refresh, reloading");
646 //setTimeout("viewCurrentFeed()", 100);
648 if (getInitParam("cdm_auto_catchup") != 1 && get_feed_unread(-3) > 0) {
649 notify_silent_next();
650 setTimeout("viewfeed(-3)", 100);
656 if (getInitParam("theme") == "") {
657 setTimeout("hide_footer()", 5000);
660 init_collapsable_feedlist(getInitParam("theme"));
662 toggle_sortable_feedlist(isFeedlistSortable());
665 exception_error("feedlist/init", e
);
669 function hide_footer_af(effect
) {
671 var c
= document
.getElementById("content-frame");
674 c
.style
.bottom
= "0px";
676 var ioa
= document
.getElementById("inline_orig_article");
679 ioa
.height
= c
.offsetHeight
;
683 var h
= document
.getElementById("headlines-frame");
686 h
.style
.bottom
= "0px";
691 exception_error("hide_footer_af", e
);
695 function hide_footer() {
697 if (Element
.visible("footer")) {
698 new Effect
.Fade("footer", { afterFinish
: hide_footer_af
});
701 exception_error("hide_footer", e
);
706 function init_hidden_feedlist(theme) {
708 debug("init_hidden_feedlist");
710 if (theme != "" && theme != "compact") return;
712 var fl = document.getElementById("feeds-holder");
713 var fh = document.getElementById("headlines-frame");
714 var fc = document.getElementById("content-frame");
715 var ft = document.getElementById("toolbar");
716 var ff = document.getElementById("footer");
717 var fhdr = document.getElementById("header");
719 var fbtn = document.getElementById("toggle_feeds_btn");
721 if (fbtn) Element.show(fbtn);
723 fl.style.top = fh.offsetTop + "px";
724 fl.style.backgroundColor = "white"; //FIXME
728 fh.style.left = "0px";
729 ft.style.left = "0px";
730 if (fc) fc.style.left = "0px";
731 if (ff) ff.style.left = "0px";
733 if (theme == "compact") {
734 fhdr.style.left = "10px";
735 fl.style.top = (fh.offsetTop + 1) + "px";
739 exception_error("init_hidden_feedlist", e);
743 function init_collapsable_feedlist(theme
) {
745 debug("init_collapsable_feedlist");
747 if (theme
!= "" && theme
!= "compact" && theme
!= "graycube" &&
748 theme
!= "compat") return;
750 var fbtn
= document
.getElementById("collapse_feeds_btn");
752 if (fbtn
) Element
.show(fbtn
);
754 if (getCookie("ttrss_vf_fclps") == 1) {
759 exception_error("init_hidden_feedlist", e
);
764 function mouse_move_handler(e
) {
770 client_y
= window
.event
.clientY
;
771 client_x
= window
.event
.clientX
;
773 client_x
= e
.screenX
;
774 client_y
= e
.screenY
;
779 if (mouse_y
== 0) mouse_y
= client_y
;
780 if (mouse_x
== 0) mouse_x
= client_x
;
782 resize_headlines(mouse_x
- client_x
, mouse_y
- client_y
);
791 exception_error("mouse_move_handler", e
);
795 function enable_selection(b
) {
796 selection_disabled
= !b
;
799 function enable_resize(b
) {
803 function mouse_down_handler(e
) {
806 /* do not prevent right click */
807 if (e
.button
&& e
.button
== 2) return;
809 if (resize_enabled
) {
810 mouse_is_down
= true;
813 document
.onselectstart = function() { return false; };
817 if (selection_disabled
) {
818 document
.onselectstart = function() { return false; };
823 exception_error("mouse_move_handler", e
);
827 function mouse_up_handler(e
) {
829 mouse_is_down
= false;
831 if (!selection_disabled
) {
832 document
.onselectstart
= null;
833 var e
= document
.getElementById("headlineActionsBody");
834 if (e
) Element
.hide(e
);
838 exception_error("mouse_move_handler", e
);
842 function request_counters_real() {
846 if (offline_mode
) return;
848 debug("requesting counters...");
850 var query
= "backend.php?op=rpc&subop=getAllCounters";
852 if (tagsAreDisplayed()) {
853 query
= query
+ "&omode=tl";
855 query
= query
+ "&omode=flc";
858 new Ajax
.Request(query
, {
859 onComplete: function(transport
) {
861 all_counters_callback2(transport
, true);
863 exception_error("viewfeed/getcounters", e
);
868 exception_error("request_counters_real", e
);
873 function request_counters() {
877 if (getInitParam("bw_limit") == "1") return;
879 var date
= new Date();
880 var timestamp
= Math
.round(date
.getTime() / 1000);
882 // if (getInitParam("sync_counters") == "1" ||
883 // timestamp - counters_last_request > 10) {
885 if (timestamp
- counters_last_request
> 15) {
886 debug("scheduling request of counters...");
887 window
.setTimeout("request_counters_real()", 1000);
888 counters_last_request
= timestamp
;
890 debug("request_counters: rate limit reached: " + (timestamp
- counters_last_request
));
894 exception_error("request_counters", e
);