]>
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 SUM(unread) FROM articles WHERE marked = 1");
41 if (rs
.isValidRow()) {
46 var title
= __("Starred articles");
47 var row_class
= "feed";
50 row_class
+= "Unread";
51 fctr_class
= "feedCtrHasUnread";
53 fctr_class
= "feedCtrNoUnread";
56 var link
= "<a title=\"FIXME\" id=\"FEEDL-"+id
+"\""+
57 "href=\"javascript:viewfeed('"+id
+"', '', false, '', false, 0);\">"+
60 feed_icon
= "<img id='FIMG-"+id
+"' src='images/mark_set.png'>";
62 tmp
+= "<li id='FEEDR-"+id
+"' class="+row_class
+">" + feed_icon
+
63 "<span id=\"FEEDN-"+id
+"\">" + link
+ "</span>";
65 tmp
+= " <span class='"+fctr_class
+"' id=\"FEEDCTR-"+id
+"\">" +
66 "(<span id=\"FEEDU-"+id
+"\">"+unread
+"</span>)</span>";
70 tmp
+= "<li><hr/></li>";
72 var rs
= db
.execute("SELECT id,title,has_icon FROM feeds ORDER BY title");
74 while (rs
.isValidRow()) {
77 var title
= rs
.field(1);
78 var has_icon
= rs
.field(2);
80 var rs_u
= db
.execute("SELECT SUM(unread) FROM articles WHERE feed_id = ?",
84 if (rs_u
.isValidRow()) {
85 unread
= rs_u
.field(0);
86 if (!unread
) unread
= 0;
93 feed_icon
= "<img id='FIMG-"+id
+"' src='" + "icons/" + id
+ ".ico'>";
95 feed_icon
= "<img id='FIMG-"+id
+"' src='images/blank_icon.gif'>";
98 var row_class
= "feed";
101 row_class
+= "Unread";
102 fctr_class
= "feedCtrHasUnread";
104 fctr_class
= "feedCtrNoUnread";
107 var link
= "<a title=\"FIXME\" id=\"FEEDL-"+id
+"\""+
108 "href=\"javascript:viewfeed('"+id
+"', '', false, '', false, 0);\">"+
111 tmp
+= "<li id='FEEDR-"+id
+"' class="+row_class
+">" + feed_icon
+
112 "<span id=\"FEEDN-"+id
+"\">" + link
+ "</span>";
114 tmp
+= " <span class='"+fctr_class
+"' id=\"FEEDCTR-"+id
+"\">" +
115 "(<span id=\"FEEDU-"+id
+"\">"+unread
+"</span>)</span>";
124 render_feedlist(tmp
);
126 exception_error("render_offline_feedlist", e
);
130 function render_feedlist(data
) {
133 var f
= document
.getElementById("feeds-frame");
135 cache_invalidate("FEEDLIST");
136 cache_inject("FEEDLIST", data
, getInitParam("num_feeds"));
140 exception_error("render_feedlist", e
);
144 function feedlist_callback2(transport
) {
146 debug("feedlist_callback2");
147 if (!transport_error_check(transport
)) return;
148 render_feedlist(transport
.responseText
);
150 exception_error("feedlist_callback2", e
);
154 function viewNextFeedPage() {
156 //if (!getActiveFeedId()) return;
158 debug("viewNextFeedPage: calling viewfeed(), p: " + _feed_cur_page
+1);
160 viewfeed(getActiveFeedId(), undefined, activeFeedIsCat(), undefined,
161 undefined, _feed_cur_page
+1);
164 exception_error("viewNextFeedPage", e
);
168 function viewfeed_offline(feed_id
, subop
, is_cat
, subop_param
, skip_history
, offset
) {
172 loading_set_progress(100);
174 clean_feed_selections();
176 setActiveFeedId(feed_id
, is_cat
);
179 var feedr
= document
.getElementById("FEEDR-" + feed_id
);
180 if (feedr
&& !feedr
.className
.match("Selected")) {
181 feedr
.className
= feedr
.className
+ "Selected";
184 var feedr
= document
.getElementById("FCAT-" + feed_id
);
185 if (feedr
&& !feedr
.className
.match("Selected")) {
186 feedr
.className
= feedr
.className
+ "Selected";
190 disableContainerChildren("headlinesToolbar", false);
191 Form
.enable("main_toolbar_form");
193 var f
= document
.getElementById("headlines-frame");
195 if (reply
.offset
== 0) {
196 debug("resetting headlines scrollTop");
202 var container
= document
.getElementById("headlines-frame");
206 rs
= db
.execute("SELECT title FROM feeds WHERE id = ?", [feed_id
]);
208 if (rs
.isValidRow() || feed_id
== -1) {
210 feed_title
= rs
.field(0);
213 feed_title
= __("Starred articles");
217 tmp
+= "<div id=\"headlinesContainer\">";
219 tmp
+= "<div class=\"headlinesSubToolbar\">";
220 tmp
+= "<div id=\"subtoolbar_ftitle\">";
224 var sel_all_link
= "javascript:selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', true, '', true)";
225 var sel_unread_link
= "javascript:selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', true, 'Unread', true)";
226 var sel_none_link
= "javascript:selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', false)";
227 var sel_inv_link
= "javascript:invertHeadlineSelection()";
229 tmp
+= __('Select:')+
230 " <a href=\""+sel_all_link
+"\">"+__('All')+"</a>, "+
231 "<a href=\""+sel_unread_link
+"\">"+__('Unread')+"</a>, "+
232 "<a href=\""+sel_inv_link
+"\">"+__('Invert')+"</a>, "+
233 "<a href=\""+sel_none_link
+"\">"+__('None')+"</a>";
235 tmp
+= " ";
239 tmp
+= "<div id=\"headlinesInnerContainer\" onscroll=\"headlines_scroll_handler()\">";
241 tmp
+= "<table class=\"headlinesList\" id=\"headlinesList\" cellspacing=\"0\">";
248 rs
= db
.execute("SELECT * FROM articles WHERE feed_id = ? "+
249 "ORDER BY updated DESC", [feed_id
]);
250 } else if (feed_id
= -1) {
251 rs
= db
.execute("SELECT * FROM articles WHERE marked = 1 "+
252 "ORDER BY updated DESC");
257 while (rs
.isValidRow()) {
259 var id
= rs
.fieldByName("id");
260 var feed_id
= rs
.fieldByName("feed_id");
264 var row_class
= (line_num
% 2) ? "even" : "odd";
266 if (rs
.fieldByName("unread") == "1") {
267 row_class
+= "Unread";
270 if (rs
.fieldByName("marked") == "1") {
271 marked_pic
= "<img id=\"FMPIC-"+id
+"\" "+
272 "src=\"images/mark_set.png\" class=\"markedPic\""+
273 "alt=\"Unstar article\" onclick='javascript:tMark("+id
+")'>";
275 marked_pic
= "<img id=\"FMPIC-"+id
+"\" "+
276 "src=\"images/mark_unset.png\" class=\"markedPic\""+
277 "alt=\"Star article\" onclick='javascript:tMark("+id
+")'>";
280 var mouseover_attrs
= "onmouseover='postMouseIn($id)' "+
281 "onmouseout='postMouseOut($id)'";
283 tmp
+= "<tr class='"+row_class
+"' id='RROW-"+id
+"' "+mouseover_attrs
+">";
285 tmp
+= "<td class='hlUpdPic'> </td>";
287 tmp
+= "<td class='hlSelectRow'>"+
288 "<input type=\"checkbox\" onclick=\"tSR(this)\" id=\"RCHK-"+id
+"\"></td>";
290 tmp
+= "<td class='hlMarkedPic'>"+marked_pic
+"</td>";
292 tmp
+= "<td onclick='view("+id
+","+feed_id
+")' "+
293 "class='hlContent' valign='middle'>";
295 tmp
+= "<a target=\"_blank\" id=\"RTITLE-$id\" href=\"" +
296 rs
.fieldByName("link") + "\"" +
297 "onclick=\"return view("+id
+","+feed_id
+");\">"+
298 rs
.fieldByName("title");
300 var content_preview
= truncate_string(strip_tags(rs
.fieldByName("content")),
303 tmp
+= "<span class=\"contentPreview\"> - "+content_preview
+"</span>";
309 tmp
+= "<td class=\"hlUpdated\" onclick='view("+id
+","+feed_id
+")'>"+
310 "<nobr>"+rs
.fieldByName("updated").substring(0,16)+"</nobr></td>";
320 tmp
+= "</div></div>";
324 container
.innerHTML
= tmp
;
326 var ids
= getSelectedArticleIds2();
328 //container.innerHTML = container.innerHTML + tmp;
330 for (var i
= 0; i
< ids
.length
; i
++) {
331 markHeadline(ids
[i
]);
340 exception_error("viewfeed_offline", e
);
344 function viewfeed(feed
, subop
, is_cat
, subop_param
, skip_history
, offset
) {
347 if (offline_mode
) return viewfeed_offline(feed
, subop
, is_cat
, subop_param
,
348 skip_history
, offset
);
350 // if (!offset) page_offset = 0;
352 last_requested_article
= 0;
353 //counters_last_request = 0;
355 if (feed
== getActiveFeedId()) {
356 cache_invalidate("F:" + feed
);
359 /* if (getInitParam("theme") == "" || getInitParam("theme") == "compact") {
360 if (getInitParam("hide_feedlist") == 1) {
361 Element.hide("feeds-holder");
365 var force_nocache
= false;
370 page_offset
= offset
;
374 _infscroll_disable
= 0;
377 if (getActiveFeedId() != feed
) {
380 _infscroll_disable
= 0;
383 if (page_offset
!= 0 && !subop
) {
384 var date
= new Date();
385 var timestamp
= Math
.round(date
.getTime() / 1000);
387 debug("<b>" + _infscroll_request_sent
+ " : " + timestamp
+ "</b>");
389 if (_infscroll_request_sent
&& _infscroll_request_sent
+ 30 > timestamp
) {
390 debug("infscroll request in progress, aborting");
394 _infscroll_request_sent
= timestamp
;
401 Form
.enable("main_toolbar_form");
403 var toolbar_form
= document
.forms
["main_toolbar_form"];
404 var toolbar_query
= Form
.serialize("main_toolbar_form");
406 if (toolbar_form
.query
) {
407 if (toolbar_form
.query
.value
!= "") {
408 force_nocache
= true;
410 toolbar_form
.query
.value
= "";
413 var query
= "backend.php?op=viewfeed&feed=" + feed
+ "&" +
414 toolbar_query
+ "&subop=" + param_escape(subop
);
416 if (document
.getElementById("search_form")) {
417 var search_query
= Form
.serialize("search_form");
418 query
= query
+ "&" + search_query
;
420 force_nocache
= true;
423 // debug("IS_CAT_STORED: " + activeFeedIsCat() + ", IS_CAT: " + is_cat);
425 if (subop
== "MarkAllRead") {
427 var show_next_feed
= getInitParam("on_catchup_show_next_feed") == "1";
429 if (show_next_feed
) {
431 if (!activeFeedIsCat()) {
433 var feedlist
= document
.getElementById('feedList');
435 var next_unread_feed
= getRelativeFeedId(feedlist
,
438 if (!next_unread_feed
) {
439 next_unread_feed
= getRelativeFeedId(feedlist
,
443 if (next_unread_feed
) {
444 query
= query
+ "&nuf=" + param_escape(next_unread_feed
);
445 //setActiveFeedId(next_unread_feed);
446 feed
= next_unread_feed
;
450 var next_unread_feed
= getNextUnreadCat(feed
);
452 /* we don't need to specify that our next feed is actually
453 a category, because we're in the is_cat mode by definition
456 if (next_unread_feed
&& show_next_feed
) {
457 query
= query
+ "&nuf=" + param_escape(next_unread_feed
);
458 feed
= next_unread_feed
;
466 query
= query
+ "&cat=1";
469 if (page_offset
!= 0) {
470 query
= query
+ "&skip=" + page_offset
;
472 // to prevent duplicate feed titles when showing grouped vfeeds
473 if (vgroup_last_feed
) {
474 query
= query
+ "&vgrlf=" + param_escape(vgroup_last_feed
);
478 var date
= new Date();
479 var timestamp
= Math
.round(date
.getTime() / 1000);
480 query
= query
+ "&ts=" + timestamp
482 disableContainerChildren("headlinesToolbar", false);
483 Form
.enable("main_toolbar_form");
485 // for piggybacked counters
487 if (tagsAreDisplayed()) {
488 query
= query
+ "&omode=lt";
490 query
= query
+ "&omode=flc";
493 if (!async_counters_work
) {
494 query
= query
+ "&csync=true";
499 var container
= document
.getElementById("headlinesInnerContainer");
501 /* if (container && page_offset == 0 && !isCdmMode()) {
502 new Effect.Fade(container, {duration: 1, to: 0.01,
503 queue: { position:'end', scope: 'FEEDL-' + feed, limit: 1 } } );
508 if (!is_cat
) unread_ctr
= get_feed_unread(feed
);
510 var cache_check
= false;
512 if (unread_ctr
!= -1 && !page_offset
&& !force_nocache
&& !subop
) {
514 var cache_prefix
= "";
522 cache_check
= cache_check_param(cache_prefix
+ feed
, unread_ctr
);
523 debug("headline cache check: " + cache_check
);
527 var f
= document
.getElementById("headlines-frame");
529 clean_feed_selections();
531 setActiveFeedId(feed
, is_cat
);
534 var feedr
= document
.getElementById("FEEDR-" + feed
);
535 if (feedr
&& !feedr
.className
.match("Selected")) {
536 feedr
.className
= feedr
.className
+ "Selected";
539 var feedr
= document
.getElementById("FCAT-" + feed_id
);
540 if (feedr
&& !feedr
.className
.match("Selected")) {
541 feedr
.className
= feedr
.className
+ "Selected";
545 f
.innerHTML
= cache_find_param(cache_prefix
+ feed
, unread_ctr
);
553 notify_progress("Loading, please wait...", true);
556 new Ajax
.Request(query
, {
557 onComplete: function(transport
) {
558 headlines_callback2(transport
, page_offset
);
563 exception_error("viewfeed", e
);
567 function toggleCollapseCat_af(effect
) {
568 //var caption = elem.id.replace("FCATLIST-", "");
572 var elem
= effect
.element
;
573 var cat
= elem
.id
.replace("FCATLIST-", "");
574 var cap
= document
.getElementById("FCAP-" + cat
);
576 if (Element
.visible(elem
)) {
577 cap
.innerHTML
= cap
.innerHTML
.replace("…", "");
579 if (cap
.innerHTML
.lastIndexOf("…") != cap
.innerHTML
.length
-3) {
580 cap
.innerHTML
= cap
.innerHTML
+ "…";
585 exception_error("toggleCollapseCat_af", e
);
589 function toggleCollapseCat(cat
) {
592 var cat_elem
= document
.getElementById("FCAT-" + cat
);
593 var cat_list
= document
.getElementById("FCATLIST-" + cat
).parentNode
;
594 var caption
= document
.getElementById("FCAP-" + cat
);
596 /* if (cat_list.className.match("invisible")) {
597 cat_list.className = "";
598 caption.innerHTML = caption.innerHTML.replace("...", "");
600 setCookie("ttrss_vf_uclps", "0");
603 cat_list.className = "invisible";
604 caption.innerHTML = caption.innerHTML + "...";
606 setCookie("ttrss_vf_uclps", "1");
612 if (Element
.visible("FCATLIST-" + cat
)) {
613 setCookie("ttrss_vf_uclps", "1");
615 setCookie("ttrss_vf_uclps", "0");
620 if (Element
.visible("FCATLIST-" + cat
)) {
621 setCookie("ttrss_vf_lclps", "1");
623 setCookie("ttrss_vf_lclps", "0");
628 if (Element
.visible("FCATLIST-" + cat
)) {
629 setCookie("ttrss_vf_vclps", "1");
631 setCookie("ttrss_vf_vclps", "0");
635 Effect
.toggle('FCATLIST-' + cat
, 'blind', { duration
: 0.5,
636 afterFinish
: toggleCollapseCat_af
});
638 new Ajax
.Request("backend.php?op=feeds&subop=collapse&cid=" +
642 exception_error("toggleCollapseCat", e
);
646 function feedlist_dragsorted(ctr
) {
648 var elem
= document
.getElementById("feedList");
650 var cats
= elem
.getElementsByTagName("LI");
651 var ordered_cats
= new Array();
653 for (var i
= 0; i
< cats
.length
; i
++) {
654 if (cats
[i
].id
&& cats
[i
].id
.match("FCAT-")) {
655 ordered_cats
.push(cats
[i
].id
.replace("FCAT-", ""));
659 if (ordered_cats
.length
> 0) {
661 var query
= "backend.php?op=feeds&subop=catsort&corder=" +
662 param_escape(ordered_cats
.toString());
666 new Ajax
.Request(query
);
670 exception_error("feedlist_dragsorted", e
);
674 function feedlist_init() {
676 // if (arguments.callee.done) return;
677 // arguments.callee.done = true;
679 loading_set_progress(90);
681 debug("in feedlist init");
683 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
684 document
.onkeydown
= hotkey_handler
;
685 document
.onmousemove
= mouse_move_handler
;
686 document
.onmousedown
= mouse_down_handler
;
687 document
.onmouseup
= mouse_up_handler
;
689 setTimeout("timeout()", 1);
691 if (typeof correctPNG
!= 'undefined') {
695 if (getActiveFeedId()) {
696 //debug("some feed is open on feedlist refresh, reloading");
697 //setTimeout("viewCurrentFeed()", 100);
699 if (getInitParam("cdm_auto_catchup") != 1 && get_feed_unread(-3) > 0) {
700 notify_silent_next();
701 setTimeout("viewfeed(-3)", 100);
707 if (getInitParam("theme") == "") {
708 setTimeout("hide_footer()", 5000);
711 init_collapsable_feedlist(getInitParam("theme"));
713 toggle_sortable_feedlist(isFeedlistSortable());
716 exception_error("feedlist/init", e
);
720 function hide_footer_af(effect
) {
722 var c
= document
.getElementById("content-frame");
725 c
.style
.bottom
= "0px";
727 var ioa
= document
.getElementById("inline_orig_article");
730 ioa
.height
= c
.offsetHeight
;
734 var h
= document
.getElementById("headlines-frame");
737 h
.style
.bottom
= "0px";
742 exception_error("hide_footer_af", e
);
746 function hide_footer() {
748 if (Element
.visible("footer")) {
749 new Effect
.Fade("footer", { afterFinish
: hide_footer_af
});
752 exception_error("hide_footer", e
);
757 function init_hidden_feedlist(theme) {
759 debug("init_hidden_feedlist");
761 if (theme != "" && theme != "compact") return;
763 var fl = document.getElementById("feeds-holder");
764 var fh = document.getElementById("headlines-frame");
765 var fc = document.getElementById("content-frame");
766 var ft = document.getElementById("toolbar");
767 var ff = document.getElementById("footer");
768 var fhdr = document.getElementById("header");
770 var fbtn = document.getElementById("toggle_feeds_btn");
772 if (fbtn) Element.show(fbtn);
774 fl.style.top = fh.offsetTop + "px";
775 fl.style.backgroundColor = "white"; //FIXME
779 fh.style.left = "0px";
780 ft.style.left = "0px";
781 if (fc) fc.style.left = "0px";
782 if (ff) ff.style.left = "0px";
784 if (theme == "compact") {
785 fhdr.style.left = "10px";
786 fl.style.top = (fh.offsetTop + 1) + "px";
790 exception_error("init_hidden_feedlist", e);
794 function init_collapsable_feedlist(theme
) {
796 debug("init_collapsable_feedlist");
798 if (theme
!= "" && theme
!= "compact" && theme
!= "graycube" &&
799 theme
!= "compat") return;
801 var fbtn
= document
.getElementById("collapse_feeds_btn");
803 if (fbtn
) Element
.show(fbtn
);
805 if (getCookie("ttrss_vf_fclps") == 1) {
810 exception_error("init_hidden_feedlist", e
);
815 function mouse_move_handler(e
) {
821 client_y
= window
.event
.clientY
;
822 client_x
= window
.event
.clientX
;
824 client_x
= e
.screenX
;
825 client_y
= e
.screenY
;
830 if (mouse_y
== 0) mouse_y
= client_y
;
831 if (mouse_x
== 0) mouse_x
= client_x
;
833 resize_headlines(mouse_x
- client_x
, mouse_y
- client_y
);
842 exception_error("mouse_move_handler", e
);
846 function enable_selection(b
) {
847 selection_disabled
= !b
;
850 function enable_resize(b
) {
854 function mouse_down_handler(e
) {
857 /* do not prevent right click */
858 if (e
.button
&& e
.button
== 2) return;
860 if (resize_enabled
) {
861 mouse_is_down
= true;
864 document
.onselectstart = function() { return false; };
868 if (selection_disabled
) {
869 document
.onselectstart = function() { return false; };
874 exception_error("mouse_move_handler", e
);
878 function mouse_up_handler(e
) {
880 mouse_is_down
= false;
882 if (!selection_disabled
) {
883 document
.onselectstart
= null;
884 var e
= document
.getElementById("headlineActionsBody");
885 if (e
) Element
.hide(e
);
889 exception_error("mouse_move_handler", e
);
893 function request_counters_real() {
897 if (offline_mode
) return;
899 debug("requesting counters...");
901 var query
= "backend.php?op=rpc&subop=getAllCounters";
903 if (tagsAreDisplayed()) {
904 query
= query
+ "&omode=tl";
906 query
= query
+ "&omode=flc";
909 new Ajax
.Request(query
, {
910 onComplete: function(transport
) {
912 all_counters_callback2(transport
, true);
914 exception_error("viewfeed/getcounters", e
);
919 exception_error("request_counters_real", e
);
924 function request_counters() {
928 if (getInitParam("bw_limit") == "1") return;
930 var date
= new Date();
931 var timestamp
= Math
.round(date
.getTime() / 1000);
933 // if (getInitParam("sync_counters") == "1" ||
934 // timestamp - counters_last_request > 10) {
936 if (timestamp
- counters_last_request
> 15) {
937 debug("scheduling request of counters...");
938 window
.setTimeout("request_counters_real()", 1000);
939 counters_last_request
= timestamp
;
941 debug("request_counters: rate limit reached: " + (timestamp
- counters_last_request
));
945 exception_error("request_counters", e
);