]>
git.wh0rd.org - tt-rss.git/blob - functions.js
1 var hotkeys_enabled
= true;
2 var xmlhttp_rpc
= Ajax
.getTransport();
3 var notify_silent
= false;
4 var last_progress_point
= 0;
5 var async_counters_work
= false;
7 /* add method to remove element from array */
9 Array
.prototype.remove = function(s
) {
10 for (var i
=0; i
< this.length
; i
++) {
11 if (s
== this[i
]) this.splice(i
, 1);
19 function exception_error(location
, e
, ext_info
) {
20 var msg
= format_exception_error(location
, e
);
26 var ebc
= document
.getElementById("xebContent");
30 Element
.show("dialog_overlay");
31 Element
.show("errorBoxShadow");
34 if (ext_info
.responseText
) {
35 ext_info
= ext_info
.responseText
;
40 "<div><b>Error message:</b></div>" +
41 "<pre>" + msg
+ "</pre>" +
42 "<div><b>Additional information:</b></div>" +
43 "<textarea readonly=\"1\">" + ext_info
+ "</textarea>";
56 function format_exception_error(location
, e
) {
60 var base_fname
= e
.fileName
.substring(e
.fileName
.lastIndexOf("/") + 1);
62 msg
= "Exception: " + e
.name
+ ", " + e
.message
+
63 "\nFunction: " + location
+ "()" +
64 "\nLocation: " + base_fname
+ ":" + e
.lineNumber
;
66 } else if (e
.description
) {
67 msg
= "Exception: " + e
.description
+ "\nFunction: " + location
+ "()";
69 msg
= "Exception: " + e
+ "\nFunction: " + location
+ "()";
72 debug("<b>EXCEPTION: " + msg
+ "</b>");
78 function disableHotkeys() {
79 hotkeys_enabled
= false;
82 function enableHotkeys() {
83 hotkeys_enabled
= true;
86 function xmlhttp_ready(obj
) {
87 return obj
.readyState
== 4 || obj
.readyState
== 0 || !obj
.readyState
;
90 function open_article_callback(transport
) {
93 if (transport
.responseXML
) {
95 var link
= transport
.responseXML
.getElementsByTagName("link")[0];
96 var id
= transport
.responseXML
.getElementsByTagName("id")[0];
98 debug("open_article_callback, received link: " + link
);
102 var wname
= "ttrss_article_" + id
.firstChild
.nodeValue
;
104 debug("link url: " + link
.firstChild
.nodeValue
+ ", wname " + wname
);
106 var w
= window
.open(link
.firstChild
.nodeValue
, wname
);
108 if (!w
) { notify_error("Failed to load article in new window"); }
111 id
= id
.firstChild
.nodeValue
;
112 if (!document
.getElementById("headlinesList")) {
113 window
.setTimeout("toggleUnread(" + id
+ ", 0)", 100);
117 notify_error("Can't open article: received invalid article link");
120 notify_error("Can't open article: received invalid XML");
124 exception_error("open_article_callback", e
);
128 function param_escape(arg
) {
129 if (typeof encodeURIComponent
!= 'undefined')
130 return encodeURIComponent(arg
);
135 function param_unescape(arg
) {
136 if (typeof decodeURIComponent
!= 'undefined')
137 return decodeURIComponent(arg
);
139 return unescape(arg
);
142 function delay(gap
) {
144 then
=new Date().getTime();
146 while((now
-then
)<gap
) {
147 now
=new Date().getTime();
151 var notify_hide_timerid
= false;
153 function hide_notify() {
154 var n
= document
.getElementById("notify");
156 n
.style
.display
= "none";
160 function notify_silent_next() {
161 notify_silent
= true;
164 function notify_real(msg
, no_hide
, n_type
) {
167 notify_silent
= false;
171 var n
= document
.getElementById("notify");
172 var nb
= document
.getElementById("notify_body");
174 if (!n
|| !nb
) return;
176 if (notify_hide_timerid
) {
177 window
.clearTimeout(notify_hide_timerid
);
181 if (n
.style
.display
== "block") {
182 notify_hide_timerid
= window
.setTimeout("hide_notify()", 0);
186 n
.style
.display
= "block";
198 if (typeof __
!= 'undefined') {
203 n
.className
= "notify";
204 } else if (n_type
== 2) {
205 n
.className
= "notifyProgress";
206 msg
= "<img src='images/indicator_white.gif'> " + msg
;
207 } else if (n_type
== 3) {
208 n
.className
= "notifyError";
209 msg
= "<img src='images/sign_excl.gif'> " + msg
;
210 } else if (n_type
== 4) {
211 n
.className
= "notifyInfo";
212 msg
= "<img src='images/sign_info.gif'> " + msg
;
215 // msg = "<img src='images/live_com_loading.gif'> " + msg;
220 notify_hide_timerid
= window
.setTimeout("hide_notify()", 3000);
224 function notify(msg
, no_hide
) {
225 notify_real(msg
, no_hide
, 1);
228 function notify_progress(msg
, no_hide
) {
229 notify_real(msg
, no_hide
, 2);
232 function notify_error(msg
, no_hide
) {
233 notify_real(msg
, no_hide
, 3);
237 function notify_info(msg
, no_hide
) {
238 notify_real(msg
, no_hide
, 4);
241 function printLockingError() {
242 notify_info("Please wait until operation finishes.");
245 function cleanSelected(element
) {
246 var content
= document
.getElementById(element
);
248 for (i
= 0; i
< content
.rows
.length
; i
++) {
249 content
.rows
[i
].className
= content
.rows
[i
].className
.replace("Selected", "");
253 function getVisibleUnreadHeadlines() {
254 var content
= document
.getElementById("headlinesList");
256 var rows
= new Array();
258 if (!content
) return rows
;
260 for (i
= 0; i
< content
.rows
.length
; i
++) {
261 var row_id
= content
.rows
[i
].id
.replace("RROW-", "");
262 if (row_id
.length
> 0 && content
.rows
[i
].className
.match("Unread")) {
269 function getVisibleHeadlineIds() {
271 var content
= document
.getElementById("headlinesList");
273 var rows
= new Array();
275 if (!content
) return rows
;
277 for (i
= 0; i
< content
.rows
.length
; i
++) {
278 var row_id
= content
.rows
[i
].id
.replace("RROW-", "");
279 if (row_id
.length
> 0) {
286 function getFirstVisibleHeadlineId() {
288 var rows
= cdmGetVisibleArticles();
291 var rows
= getVisibleHeadlineIds();
296 function getLastVisibleHeadlineId() {
298 var rows
= cdmGetVisibleArticles();
299 return rows
[rows
.length
-1];
301 var rows
= getVisibleHeadlineIds();
302 return rows
[rows
.length
-1];
306 function markHeadline(id
) {
307 var row
= document
.getElementById("RROW-" + id
);
309 var is_active
= false;
311 if (row
.className
.match("Active")) {
314 row
.className
= row
.className
.replace("Selected", "");
315 row
.className
= row
.className
.replace("Active", "");
316 row
.className
= row
.className
.replace("Insensitive", "");
319 row
.className
= row
.className
= "Active";
322 var check
= document
.getElementById("RCHK-" + id
);
325 check
.checked
= true;
328 row
.className
= row
.className
+ "Selected";
333 function getFeedIds() {
334 var content
= document
.getElementById("feedsList");
336 var rows
= new Array();
338 for (i
= 0; i
< content
.rows
.length
; i
++) {
339 var id
= content
.rows
[i
].id
.replace("FEEDR-", "");
348 function setCookie(name
, value
, lifetime
, path
, domain
, secure
) {
354 d
.setTime(d
.getTime() + (lifetime
* 1000));
357 debug("setCookie: " + name
+ " => " + value
+ ": " + d
);
359 int_setCookie(name
, value
, d
, path
, domain
, secure
);
363 function int_setCookie(name
, value
, expires
, path
, domain
, secure
) {
364 document
.cookie
= name
+ "=" + escape(value
) +
365 ((expires
) ? "; expires=" + expires
.toGMTString() : "") +
366 ((path
) ? "; path=" + path
: "") +
367 ((domain
) ? "; domain=" + domain
: "") +
368 ((secure
) ? "; secure" : "");
371 function delCookie(name
, path
, domain
) {
372 if (getCookie(name
)) {
373 document
.cookie
= name
+ "=" +
374 ((path
) ? ";path=" + path
: "") +
375 ((domain
) ? ";domain=" + domain
: "" ) +
376 ";expires=Thu, 01-Jan-1970 00:00:01 GMT";
381 function getCookie(name
) {
383 var dc
= document
.cookie
;
384 var prefix
= name
+ "=";
385 var begin
= dc
.indexOf("; " + prefix
);
387 begin
= dc
.indexOf(prefix
);
388 if (begin
!= 0) return null;
393 var end
= document
.cookie
.indexOf(";", begin
);
397 return unescape(dc
.substring(begin
+ prefix
.length
, end
));
400 function disableContainerChildren(id
, disable
, doc
) {
402 if (!doc
) doc
= document
;
404 var container
= doc
.getElementById(id
);
407 //alert("disableContainerChildren: element " + id + " not found");
411 for (var i
= 0; i
< container
.childNodes
.length
; i
++) {
412 var child
= container
.childNodes
[i
];
415 child
.disabled
= disable
;
421 if (child
.className
&& child
.className
.match("button")) {
422 child
.className
= "disabledButton";
425 if (child
.className
&& child
.className
.match("disabledButton")) {
426 child
.className
= "button";
433 function gotoPreferences() {
434 document
.location
.href
= "prefs.php";
437 function gotoMain() {
438 document
.location
.href
= "tt-rss.php";
441 function gotoExportOpml() {
442 document
.location
.href
= "opml.php?op=Export";
445 function parse_counters(reply
, scheduled_call
) {
450 var elems
= reply
.getElementsByTagName("counter");
452 for (var l
= 0; l
< elems
.length
; l
++) {
454 var id
= elems
[l
].getAttribute("id");
455 var t
= elems
[l
].getAttribute("type");
456 var ctr
= elems
[l
].getAttribute("counter");
457 var error
= elems
[l
].getAttribute("error");
458 var has_img
= elems
[l
].getAttribute("hi");
459 var updated
= elems
[l
].getAttribute("updated");
460 var title
= elems
[l
].getAttribute("title");
461 var xmsg
= elems
[l
].getAttribute("xmsg");
463 if (id
== "global-unread") {
469 if (id
== "subscribed-feeds") {
474 if (t
== "category") {
475 var catctr
= document
.getElementById("FCATCTR-" + id
);
477 catctr
.innerHTML
= "(" + ctr
+ ")";
479 catctr
.className
= "catCtrHasUnread";
481 catctr
.className
= "catCtrNoUnread";
487 var feedctr
= document
.getElementById("FEEDCTR-" + id
);
488 var feedu
= document
.getElementById("FEEDU-" + id
);
489 var feedr
= document
.getElementById("FEEDR-" + id
);
490 var feed_img
= document
.getElementById("FIMG-" + id
);
491 var feedlink
= document
.getElementById("FEEDL-" + id
);
492 var feedupd
= document
.getElementById("FLUPD-" + id
);
494 if (updated
&& feedlink
) {
496 feedlink
.title
= "Error: " + error
+ " (" + updated
+ ")";
498 feedlink
.title
= "Updated: " + updated
;
503 if (!updated
) updated
= "";
507 feedupd
.innerHTML
= updated
+ " " + xmsg
+ " (Error)";
509 feedupd
.innerHTML
= updated
+ " (Error)";
513 feedupd
.innerHTML
= updated
+ " " + xmsg
;
515 feedupd
.innerHTML
= updated
;
520 if (has_img
&& feed_img
) {
521 if (!feed_img
.src
.match(id
+ ".ico")) {
522 feed_img
.src
= getInitParam("icons_location") + "/" + id
+ ".ico";
526 if (feedlink
&& title
) {
527 feedlink
.innerHTML
= title
;
530 if (feedctr
&& feedu
&& feedr
) {
532 if (feedu
.innerHTML
!= ctr
&& id
== getActiveFeedId() && scheduled_call
) {
536 var row_needs_hl
= (ctr
> 0 && ctr
> parseInt(feedu
.innerHTML
));
538 feedu
.innerHTML
= ctr
;
541 feedr
.className
= feedr
.className
.replace("feed", "error");
543 feedr
.className
= feedr
.className
.replace("error", "feed");
547 feedctr
.className
= "feedCtrHasUnread";
548 if (!feedr
.className
.match("Unread")) {
549 var is_selected
= feedr
.className
.match("Selected");
551 feedr
.className
= feedr
.className
.replace("Selected", "");
552 feedr
.className
= feedr
.className
.replace("Unread", "");
554 feedr
.className
= feedr
.className
+ "Unread";
557 feedr
.className
= feedr
.className
+ "Selected";
563 new Effect
.Highlight(feedr
, {duration
: 1, startcolor
: "#fff7d5",
564 queue
: { position
:'end', scope
: 'EFQ-' + id
, limit
: 1 } } );
567 feedctr
.className
= "feedCtrNoUnread";
568 feedr
.className
= feedr
.className
.replace("Unread", "");
573 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
575 var feeds_stored
= number_of_feeds
;
577 debug("Feed counters, C: " + feeds_found
+ ", S:" + feeds_stored
);
579 if (feeds_stored
!= feeds_found
) {
580 number_of_feeds
= feeds_found
;
582 if (feeds_stored
!= 0 && feeds_found
!= 0) {
583 debug("Subscribed feed number changed, refreshing feedlist");
584 setTimeout('updateFeedList(false, false)', 50);
589 exception_error("parse_counters", e
);
593 function parse_counters_reply(transport
, scheduled_call
) {
595 if (!transport
.responseXML
) {
596 notify_error("Backend did not return valid XML", true);
600 var reply
= transport
.responseXML
.firstChild
;
603 notify_error("Backend did not return expected XML object", true);
608 var error_code
= false;
609 var error_msg
= false;
611 if (reply
.firstChild
) {
612 error_code
= reply
.firstChild
.getAttribute("error-code");
613 error_msg
= reply
.firstChild
.getAttribute("error-msg");
617 error_code
= reply
.getAttribute("error-code");
618 error_msg
= reply
.getAttribute("error-msg");
621 if (error_code
&& error_code
!= 0) {
622 debug("refetch_callback: got error code " + error_code
);
623 return fatalError(error_code
, error_msg
);
626 var counters
= reply
.getElementsByTagName("counters")[0];
628 parse_counters(counters
, scheduled_call
);
630 var runtime_info
= reply
.getElementsByTagName("runtime-info")[0];
632 parse_runtime_info(runtime_info
);
634 if (feedsSortByUnread()) {
638 hideOrShowFeeds(getInitParam("hide_read_feeds") == 1);
642 function all_counters_callback2(transport
, async_call
) {
644 if (async_call
) async_counters_work
= true;
646 debug("<b>all_counters_callback2 IN: " + transport
+ "</b>");
647 parse_counters_reply(transport
);
648 debug("<b>all_counters_callback2 OUT: " + transport
+ "</b>");
651 exception_error("all_counters_callback2", e
, transport
);
655 function get_feed_unread(id
) {
657 return parseInt(document
.getElementById("FEEDU-" + id
).innerHTML
);
663 function get_cat_unread(id
) {
665 var ctr
= document
.getElementById("FCATCTR-" + id
).innerHTML
;
666 ctr
= ctr
.replace("(", "");
667 ctr
= ctr
.replace(")", "");
668 return parseInt(ctr
);
674 function get_feed_entry_unread(elem
) {
676 var id
= elem
.id
.replace("FEEDR-", "");
683 return parseInt(document
.getElementById("FEEDU-" + id
).innerHTML
);
689 function get_feed_entry_name(elem
) {
690 var id
= elem
.id
.replace("FEEDR-", "");
691 return getFeedName(id
);
695 function resort_category(node
) {
699 debug("resort_category: " + node
);
701 var by_unread
= feedsSortByUnread();
703 var list
= node
.getElementsByTagName("LI");
705 for (i
= 0; i
< list
.length
; i
++) {
707 for (j
= i
+1; j
< list
.length
; j
++) {
709 var tmp_val
= get_feed_entry_unread(list
[i
]);
710 var cur_val
= get_feed_entry_unread(list
[j
]);
712 var tmp_name
= get_feed_entry_name(list
[i
]);
713 var cur_name
= get_feed_entry_name(list
[j
]);
715 if ((by_unread
&& (cur_val
> tmp_val
)) || (!by_unread
&& (cur_name
< tmp_name
))) {
716 tempnode_i
= list
[i
].cloneNode(true);
717 tempnode_j
= list
[j
].cloneNode(true);
718 node
.replaceChild(tempnode_i
, list
[j
]);
719 node
.replaceChild(tempnode_j
, list
[i
]);
725 exception_error("resort_category", e
);
730 function resort_feedlist() {
731 debug("resort_feedlist");
733 if (document
.getElementById("FCATLIST--1")) {
735 var lists
= document
.getElementsByTagName("UL");
737 for (var i
= 0; i
< lists
.length
; i
++) {
738 if (lists
[i
].id
&& lists
[i
].id
.match("FCATLIST-")) {
739 resort_category(lists
[i
]);
744 resort_category(document
.getElementById("feedList"));
748 /** * @(#)isNumeric.js * * Copyright (c) 2000 by Sundar Dorai-Raj
749 * * @author Sundar Dorai-Raj
750 * * Email: sdoraira@vt.edu
751 * * This program is free software; you can redistribute it and/or
752 * * modify it under the terms of the GNU General Public License
753 * * as published by the Free Software Foundation; either version 2
754 * * of the License, or (at your option) any later version,
755 * * provided that any use properly credits the author.
756 * * This program is distributed in the hope that it will be useful,
757 * * but WITHOUT ANY WARRANTY; without even the implied warranty of
758 * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
759 * * GNU General Public License for more details at http://www.gnu.org * * */
761 var numbers
=".0123456789";
762 function isNumeric(x
) {
763 // is x a String or a character?
765 // remove negative sign
767 for(j
=0;j
<x
.length
;j
++) {
768 // call isNumeric recursively for each character
769 number
=isNumeric(x
.substring(j
,j
+1));
770 if(!number
) return number
;
775 // if x is number return true
776 if(numbers
.indexOf(x
)>=0) return true;
782 function hideOrShowFeeds(hide
) {
786 debug("hideOrShowFeeds: " + hide
);
788 if (document
.getElementById("FCATLIST--1")) {
790 var lists
= document
.getElementsByTagName("UL");
792 for (var i
= 0; i
< lists
.length
; i
++) {
793 if (lists
[i
].id
&& lists
[i
].id
.match("FCATLIST-")) {
795 var id
= lists
[i
].id
.replace("FCATLIST-", "");
796 hideOrShowFeedsCategory(id
, hide
);
801 hideOrShowFeedsCategory(null, hide
);
805 exception_error("hideOrShowFeeds", e
);
809 function hideOrShowFeedsCategory(id
, hide
) {
817 node
= document
.getElementById("FCATLIST-" + id
);
818 cat_node
= document
.getElementById("FCAT-" + id
);
820 node
= document
.getElementById("feedList"); // no categories
823 // debug("hideOrShowFeedsCategory: " + node + " (" + hide + ")");
828 debug("hideOrShowFeeds: passed node is null, aborting");
832 // debug("cat: " + node.id);
834 if (node
.hasChildNodes() && node
.firstChild
.nextSibling
!= false) {
835 for (i
= 0; i
< node
.childNodes
.length
; i
++) {
836 if (node
.childNodes
[i
].nodeName
!= "LI") { continue; }
838 if (node
.childNodes
[i
].style
!= undefined) {
840 var has_unread
= (node
.childNodes
[i
].className
!= "feed" &&
841 node
.childNodes
[i
].className
!= "label" &&
842 !(!getInitParam("hide_read_shows_special") &&
843 node
.childNodes
[i
].className
== "virt") &&
844 node
.childNodes
[i
].className
!= "error" &&
845 node
.childNodes
[i
].className
!= "tag");
847 // debug(node.childNodes[i].id + " --> " + has_unread);
849 if (hide
&& !has_unread
) {
850 //node.childNodes[i].style.display = "none";
851 var id
= node
.childNodes
[i
].id
;
852 Effect
.Fade(node
.childNodes
[i
], {duration
: 0.3,
853 queue
: { position
: 'end', scope
: 'FFADE-' + id
, limit
: 1 }});
857 node
.childNodes
[i
].style
.display
= "list-item";
858 //Effect.Appear(node.childNodes[i], {duration : 0.3});
862 node
.childNodes
[i
].style
.display
= "list-item";
864 //Effect.Appear(node.childNodes[i], {duration : 0.3});
865 //Effect.Highlight(node.childNodes[i]);
871 // debug("end cat: " + node.id + " unread " + cat_unread);
873 if (cat_unread
== 0) {
874 if (cat_node
.style
== undefined) {
875 debug("ERROR: supplied cat_node " + cat_node
+
876 " has no styles. WTF?");
880 //cat_node.style.display = "none";
881 Effect
.Fade(cat_node
, {duration
: 0.3,
882 queue
: { position
: 'end', scope
: 'CFADE-' + node
.id
, limit
: 1 }});
884 cat_node
.style
.display
= "list-item";
888 cat_node
.style
.display
= "list-item";
894 // debug("unread for category: " + cat_unread);
897 exception_error("hideOrShowFeedsCategory", e
);
901 function selectTableRow(r
, do_select
) {
902 r
.className
= r
.className
.replace("Selected", "");
905 r
.className
= r
.className
+ "Selected";
909 function selectTableRowById(elem_id
, check_id
, do_select
) {
913 var row
= document
.getElementById(elem_id
);
916 selectTableRow(row
, do_select
);
919 var check
= document
.getElementById(check_id
);
922 check
.checked
= do_select
;
925 exception_error("selectTableRowById", e
);
929 function selectTableRowsByIdPrefix(content_id
, prefix
, check_prefix
, do_select
,
930 classcheck
, reset_others
) {
932 var content
= document
.getElementById(content_id
);
935 alert("[selectTableRows] Element " + content_id
+ " not found.");
939 for (i
= 0; i
< content
.rows
.length
; i
++) {
940 if (Element
.visible(content
.rows
[i
])) {
941 if (!classcheck
|| content
.rows
[i
].className
.match(classcheck
)) {
943 if (content
.rows
[i
].id
.match(prefix
)) {
944 selectTableRow(content
.rows
[i
], do_select
);
946 var row_id
= content
.rows
[i
].id
.replace(prefix
, "");
947 var check
= document
.getElementById(check_prefix
+ row_id
);
950 check
.checked
= do_select
;
952 } else if (reset_others
) {
953 selectTableRow(content
.rows
[i
], false);
955 var row_id
= content
.rows
[i
].id
.replace(prefix
, "");
956 var check
= document
.getElementById(check_prefix
+ row_id
);
959 check
.checked
= false;
963 } else if (reset_others
) {
964 selectTableRow(content
.rows
[i
], false);
966 var row_id
= content
.rows
[i
].id
.replace(prefix
, "");
967 var check
= document
.getElementById(check_prefix
+ row_id
);
970 check
.checked
= false;
978 function getSelectedTableRowIds(content_id
, prefix
) {
980 var content
= document
.getElementById(content_id
);
983 alert("[getSelectedTableRowIds] Element " + content_id
+ " not found.");
987 var sel_rows
= new Array();
989 for (i
= 0; i
< content
.rows
.length
; i
++) {
990 if (content
.rows
[i
].id
.match(prefix
) &&
991 content
.rows
[i
].className
.match("Selected")) {
993 var row_id
= content
.rows
[i
].id
.replace(prefix
+ "-", "");
994 sel_rows
.push(row_id
);
1002 function toggleSelectRowById(sender
, id
) {
1003 var row
= document
.getElementById(id
);
1005 if (sender
.checked
) {
1006 if (!row
.className
.match("Selected")) {
1007 row
.className
= row
.className
+ "Selected";
1010 if (row
.className
.match("Selected")) {
1011 row
.className
= row
.className
.replace("Selected", "");
1016 function toggleSelectListRow(sender
) {
1017 var parent_row
= sender
.parentNode
;
1019 if (sender
.checked
) {
1020 if (!parent_row
.className
.match("Selected")) {
1021 parent_row
.className
= parent_row
.className
+ "Selected";
1024 if (parent_row
.className
.match("Selected")) {
1025 parent_row
.className
= parent_row
.className
.replace("Selected", "");
1030 function tSR(sender
) {
1031 return toggleSelectRow(sender
);
1034 function toggleSelectRow(sender
) {
1035 var parent_row
= sender
.parentNode
.parentNode
;
1037 if (sender
.checked
) {
1038 if (!parent_row
.className
.match("Selected")) {
1039 parent_row
.className
= parent_row
.className
+ "Selected";
1042 if (parent_row
.className
.match("Selected")) {
1043 parent_row
.className
= parent_row
.className
.replace("Selected", "");
1048 function getNextUnreadCat(id
) {
1050 var rows
= document
.getElementById("feedList").getElementsByTagName("LI");
1051 var feeds
= new Array();
1053 var unread_only
= true;
1056 for (var i
= 0; i
< rows
.length
; i
++) {
1057 if (rows
[i
].id
.match("FCAT-")) {
1058 if (rows
[i
].id
== "FCAT-" + id
&& is_cat
|| (Element
.visible(rows
[i
]) && Element
.visible(rows
[i
].parentNode
))) {
1060 var cat_id
= parseInt(rows
[i
].id
.replace("FCAT-", ""));
1063 if (!unread_only
|| get_cat_unread(cat_id
) > 0) {
1071 var idx
= feeds
.indexOf(id
);
1072 if (idx
!= -1 && idx
< feeds
.length
) {
1073 return feeds
[idx
+1];
1075 return feeds
.shift();
1079 exception_error("getNextUnreadCat", e
);
1083 function getRelativeFeedId2(id
, is_cat
, direction
, unread_only
) {
1086 // alert(id + " IC: " + is_cat + " D: " + direction + " U: " + unread_only);
1088 var rows
= document
.getElementById("feedList").getElementsByTagName("LI");
1089 var feeds
= new Array();
1091 for (var i
= 0; i
< rows
.length
; i
++) {
1092 if (rows
[i
].id
.match("FEEDR-")) {
1094 if (rows
[i
].id
== "FEEDR-" + id
&& !is_cat
|| (Element
.visible(rows
[i
]) && Element
.visible(rows
[i
].parentNode
))) {
1097 (rows
[i
].className
.match("Unread") || rows
[i
].id
== "FEEDR-" + id
)) {
1098 feeds
.push(rows
[i
].id
.replace("FEEDR-", ""));
1103 if (rows
[i
].id
.match("FCAT-")) {
1104 if (rows
[i
].id
== "FCAT-" + id
&& is_cat
|| (Element
.visible(rows
[i
]) && Element
.visible(rows
[i
].parentNode
))) {
1106 var cat_id
= parseInt(rows
[i
].id
.replace("FCAT-", ""));
1109 if (!unread_only
|| get_cat_unread(cat_id
) > 0) {
1110 feeds
.push("CAT:"+cat_id
);
1117 // alert(feeds.toString());
1120 if (direction
== "next") {
1121 return feeds
.shift();
1126 if (direction
== "next") {
1127 if (is_cat
) id
= "CAT:" + id
;
1128 var idx
= feeds
.indexOf(id
);
1129 if (idx
!= -1 && idx
< feeds
.length
) {
1130 return feeds
[idx
+1];
1132 return getRelativeFeedId2(false, is_cat
, direction
, unread_only
);
1135 if (is_cat
) id
= "CAT:" + id
;
1136 var idx
= feeds
.indexOf(id
);
1138 return feeds
[idx
-1];
1140 return getRelativeFeedId2(false, is_cat
, direction
, unread_only
);
1147 exception_error("getRelativeFeedId2", e
);
1152 function getRelativeFeedId(list
, id
, direction
, unread_only
) {
1153 var rows
= list
.getElementsByTagName("LI");
1154 var feeds
= new Array();
1156 for (var i
= 0; i
< rows
.length
; i
++) {
1157 if (rows
[i
].id
.match("FEEDR-")) {
1159 if (rows
[i
].id
== "FEEDR-" + id
|| (Element
.visible(rows
[i
]) && Element
.visible(rows
[i
].parentNode
))) {
1162 (rows
[i
].className
.match("Unread") || rows
[i
].id
== "FEEDR-" + id
)) {
1163 feeds
.push(rows
[i
].id
.replace("FEEDR-", ""));
1170 if (direction
== "next") {
1171 return feeds
.shift();
1176 if (direction
== "next") {
1177 var idx
= feeds
.indexOf(id
);
1178 if (idx
!= -1 && idx
< feeds
.length
) {
1179 return feeds
[idx
+1];
1181 return getRelativeFeedId(list
, false, direction
, unread_only
);
1184 var idx
= feeds
.indexOf(id
);
1186 return feeds
[idx
-1];
1188 return getRelativeFeedId(list
, false, direction
, unread_only
);
1195 function showBlockElement(id
, h_id
) {
1196 var elem
= document
.getElementById(id
);
1199 elem
.style
.display
= "block";
1202 elem
= document
.getElementById(h_id
);
1204 elem
.style
.display
= "none";
1208 alert("[showBlockElement] can't find element with id " + id
);
1212 function appearBlockElement_afh(effect
) {
1216 function checkboxToggleElement(elem
, id
) {
1218 Effect
.Appear(id
, {duration
: 0.5});
1220 Effect
.Fade(id
, {duration
: 0.5});
1224 function appearBlockElement(id
, h_id
) {
1230 Effect
.SlideDown(id
, {duration
: 1.0, afterFinish
: appearBlockElement_afh
});
1232 exception_error("appearBlockElement", e
);
1237 function hideParentElement(e
) {
1238 e
.parentNode
.style
.display
= "none";
1241 function dropboxSelect(e
, v
) {
1242 for (i
= 0; i
< e
.length
; i
++) {
1243 if (e
[i
].value
== v
) {
1244 e
.selectedIndex
= i
;
1250 // originally stolen from http://www.11tmr.com/11tmr.nsf/d6plinks/MWHE-695L9Z
1251 // bugfixed just a little bit :-)
1252 function getURLParam(strParamName
){
1254 var strHref
= window
.location
.href
;
1256 if (strHref
.indexOf("#") == strHref
.length
-1) {
1257 strHref
= strHref
.substring(0, strHref
.length
-1);
1260 if ( strHref
.indexOf("?") > -1 ){
1261 var strQueryString
= strHref
.substr(strHref
.indexOf("?"));
1262 var aQueryString
= strQueryString
.split("&");
1263 for ( var iParam
= 0; iParam
< aQueryString
.length
; iParam
++ ){
1264 if (aQueryString
[iParam
].indexOf(strParamName
+ "=") > -1 ){
1265 var aParam
= aQueryString
[iParam
].split("=");
1266 strReturn
= aParam
[1];
1274 function leading_zero(p
) {
1276 if (s
.length
== 1) s
= "0" + s
;
1280 function closeErrorBox() {
1282 if (Element
.visible("errorBoxShadow")) {
1283 Element
.hide("dialog_overlay");
1284 Element
.hide("errorBoxShadow");
1292 function closeInfoBox(cleanup
) {
1294 if (Element
.visible("infoBoxShadow")) {
1295 Element
.hide("dialog_overlay");
1297 var shadow
= document
.getElementById('infoBoxShadow');
1298 var box
= document
.getElementById('infoBoxShadow');
1300 Element
.hide(shadow
);
1302 if (cleanup
) box
.innerHTML
= " ";
1311 function displayDlg(id
, param
) {
1313 notify_progress("Loading, please wait...", true);
1317 var query
= "backend.php?op=dlg&id=" +
1318 param_escape(id
) + "¶m=" + param_escape(param
);
1320 new Ajax
.Request(query
, {
1321 onComplete: function (transport
) {
1322 infobox_callback2(transport
);
1328 function infobox_submit_callback2(transport
) {
1332 // called from prefs, reload tab
1333 if (typeof active_tab
!= 'undefined' && active_tab
) {
1334 selectTab(active_tab
, false);
1338 if (transport
.responseText
) {
1339 notify_info(transport
.responseText
);
1343 function infobox_callback2(transport
) {
1346 debug("infobox_callback2");
1348 if (!getInitParam("infobox_disable_overlay")) {
1349 Element
.show("dialog_overlay");
1352 var box
= document
.getElementById('infoBox');
1353 var shadow
= document
.getElementById('infoBoxShadow');
1356 box
.innerHTML
=transport
.responseText
;
1358 shadow
.style
.display
= "block";
1360 box
.style
.display
= "block";
1364 /* FIXME this needs to be moved out somewhere */
1366 if (document
.getElementById("tags_choices")) {
1367 new Ajax
.Autocompleter('tags_str', 'tags_choices',
1368 "backend.php?op=rpc&subop=completeTags",
1369 { tokens
: ',', paramName
: "search" });
1376 exception_error("infobox_callback2", e
);
1380 function createFilter() {
1384 var form
= document
.forms
['filter_add_form'];
1385 var reg_exp
= form
.reg_exp
.value
;
1387 if (reg_exp
== "") {
1388 alert(__("Can't add filter: nothing to match on."));
1392 var query
= Form
.serialize("filter_add_form");
1394 // we can be called from some other tab in Prefs
1395 if (typeof active_tab
!= 'undefined' && active_tab
) {
1396 active_tab
= "filterConfig";
1399 new Ajax
.Request("backend.php?" + query
, {
1400 onComplete: function (transport
) {
1401 infobox_submit_callback2(transport
);
1407 exception_error("createFilter", e
);
1411 function toggleSubmitNotEmpty(e
, submit_id
) {
1413 document
.getElementById(submit_id
).disabled
= (e
.value
== "")
1415 exception_error("toggleSubmitNotEmpty", e
);
1419 function isValidURL(s
) {
1420 return s
.match("http://") != null || s
.match("https://") != null || s
.match("feed://") != null;
1423 function subscribeToFeed() {
1425 var form
= document
.forms
['feed_add_form'];
1426 var feed_url
= form
.feed_url
.value
;
1428 if (feed_url
== "") {
1429 alert(__("Can't subscribe: no feed URL given."));
1433 notify_progress(__("Subscribing to feed..."), true);
1437 var feeds_doc
= document
;
1439 // feeds_doc.location.href = "backend.php?op=error&msg=Loading,%20please wait...";
1441 var query
= Form
.serialize("feed_add_form");
1443 debug("subscribe q: " + query
);
1445 new Ajax
.Request("backend.php", {
1447 onComplete: function(transport
) {
1448 dlg_frefresh_callback(transport
);
1454 function filterCR(e
, f
)
1459 key
= window
.event
.keyCode
; //IE
1461 key
= e
.which
; //firefox
1464 if (typeof f
!= 'undefined') {
1475 var debug_last_class
= "even";
1477 function debug(msg
) {
1479 if (debug_last_class
== "even") {
1480 debug_last_class
= "odd";
1482 debug_last_class
= "even";
1485 var c
= document
.getElementById('debug_output');
1486 if (c
&& Element
.visible(c
)) {
1487 while (c
.lastChild
!= 'undefined' && c
.childNodes
.length
> 100) {
1488 c
.removeChild(c
.lastChild
);
1492 var ts
= leading_zero(d
.getHours()) + ":" + leading_zero(d
.getMinutes()) +
1493 ":" + leading_zero(d
.getSeconds());
1494 c
.innerHTML
= "<li class=\"" + debug_last_class
+ "\"><span class=\"debugTS\">[" + ts
+ "]</span> " +
1495 msg
+ "</li>" + c
.innerHTML
;
1499 function getInitParam(key
) {
1500 return init_params
[key
];
1503 function storeInitParam(key
, value
) {
1504 debug("<b>storeInitParam is OBSOLETE: " + key
+ " => " + value
+ "</b>");
1505 init_params
[key
] = value
;
1508 function fatalError(code
, msg
, ext_info
) {
1512 window
.location
.href
= "tt-rss.php";
1513 } else if (code
== 5) {
1514 window
.location
.href
= "update.php";
1517 if (msg
== "") msg
= "Unknown error";
1519 var ebc
= document
.getElementById("xebContent");
1523 Element
.show("dialog_overlay");
1524 Element
.show("errorBoxShadow");
1525 Element
.hide("xebBtn");
1528 if (ext_info
.responseText
) {
1529 ext_info
= ext_info
.responseText
;
1534 "<div><b>Error message:</b></div>" +
1535 "<pre>" + msg
+ "</pre>" +
1536 "<div><b>Additional information:</b></div>" +
1537 "<textarea readonly=\"1\">" + ext_info
+ "</textarea>";
1542 exception_error("fatalError", e
);
1546 function getFeedName(id
, is_cat
) {
1550 e
= document
.getElementById("FCATN-" + id
);
1552 e
= document
.getElementById("FEEDN-" + id
);
1555 return e
.innerHTML
.stripTags();
1561 function filterDlgCheckType(sender
) {
1565 var ftype
= sender
[sender
.selectedIndex
].value
;
1567 var form
= document
.forms
["filter_add_form"];
1570 form
= document
.forms
["filter_edit_form"];
1574 debug("filterDlgCheckType: can't find form!");
1578 // if selected filter type is 5 (Date) enable the modifier dropbox
1580 Element
.show("filter_dlg_date_mod_box");
1581 Element
.show("filter_dlg_date_chk_box");
1583 Element
.hide("filter_dlg_date_mod_box");
1584 Element
.hide("filter_dlg_date_chk_box");
1589 exception_error("filterDlgCheckType", e
);
1594 function filterDlgCheckAction(sender
) {
1598 var action
= sender
[sender
.selectedIndex
].value
;
1600 var form
= document
.forms
["filter_add_form"];
1603 form
= document
.forms
["filter_edit_form"];
1607 debug("filterDlgCheckAction: can't find form!");
1611 var action_param
= document
.getElementById("filter_dlg_param_box");
1613 if (!action_param
) {
1614 debug("filterDlgCheckAction: can't find action param box!");
1618 // if selected action supports parameters, enable params field
1619 if (action
== 4 || action
== 6 || action
== 7) {
1620 Element
.show(action_param
);
1622 Element
.show(form
.action_param
);
1623 Element
.hide(form
.action_param_label
);
1625 Element
.show(form
.action_param_label
);
1626 Element
.hide(form
.action_param
);
1629 Element
.hide(action_param
);
1633 exception_error("filterDlgCheckAction", e
);
1638 function filterDlgCheckDate() {
1640 var form
= document
.forms
["filter_add_form"];
1643 form
= document
.forms
["filter_edit_form"];
1647 debug("filterDlgCheckAction: can't find form!");
1651 var reg_exp
= form
.reg_exp
.value
;
1653 var query
= "backend.php?op=rpc&subop=checkDate&date=" + reg_exp
;
1655 new Ajax
.Request(query
, {
1656 onComplete: function(transport
) {
1658 var form
= document
.forms
["filter_add_form"];
1661 form
= document
.forms
["filter_edit_form"];
1664 if (transport
.responseXML
) {
1665 var result
= transport
.responseXML
.getElementsByTagName("result")[0];
1667 if (result
&& result
.firstChild
) {
1668 if (result
.firstChild
.nodeValue
== "1") {
1670 new Effect
.Highlight(form
.reg_exp
, {startcolor
: '#00ff00'});
1677 new Effect
.Highlight(form
.reg_exp
, {startcolor
: '#ff0000'});
1683 exception_error("filterDlgCheckDate", e
);
1687 function explainError(code
) {
1688 return displayDlg("explainError", code
);
1691 // this only searches loaded headlines list, not in CDM
1692 function getRelativePostIds(id
, limit
) {
1694 if (!limit
) limit
= 3;
1696 debug("getRelativePostIds: " + id
+ " limit=" + limit
);
1698 var ids
= new Array();
1699 var container
= document
.getElementById("headlinesList");
1702 var rows
= container
.rows
;
1704 for (var i
= 0; i
< rows
.length
; i
++) {
1705 var r_id
= rows
[i
].id
.replace("RROW-", "");
1708 for (var k
= 1; k
<= limit
; k
++) {
1711 if (i
> k
-1) var nid
= rows
[i
-k
].id
.replace("RROW-", "");
1712 if (nid
) ids
.push(nid
);
1714 if (i
< rows
.length
-k
) nid
= rows
[i
+k
].id
.replace("RROW-", "");
1715 if (nid
) ids
.push(nid
);
1726 function openArticleInNewWindow(id
) {
1728 debug("openArticleInNewWindow: " + id
);
1730 var query
= "backend.php?op=rpc&subop=getArticleLink&id=" + id
;
1731 var wname
= "ttrss_article_" + id
;
1733 debug(query
+ " " + wname
);
1735 var w
= window
.open("", wname
);
1737 if (!w
) notify_error("Failed to open window for the article");
1739 new Ajax
.Request(query
, {
1740 onComplete: function(transport
) {
1741 open_article_callback(transport
);
1746 exception_error("openArticleInNewWindow", e
);
1750 /* http://textsnippets.com/posts/show/835 */
1752 Position
.GetWindowSize = function(w
) {
1754 var width
= w
.innerWidth
|| (w
.document
.documentElement
.clientWidth
|| w
.document
.body
.clientWidth
);
1755 var height
= w
.innerHeight
|| (w
.document
.documentElement
.clientHeight
|| w
.document
.body
.clientHeight
);
1756 return [width
, height
]
1759 /* http://textsnippets.com/posts/show/836 */
1761 Position
.Center = function(element
, parent
) {
1763 var d
= Element
.getDimensions(element
);
1768 var ws
= Position
.GetWindowSize();
1772 pw
= parent
.offsetWidth
;
1773 ph
= parent
.offsetHeight
;
1775 element
.style
.top
= (ph
/2) - (h/2) - Position
.deltaY
+ "px";
1776 element
.style
.left
= (pw
/2) - (w/2) - Position
.deltaX
+ "px";
1780 function labeltest_callback(transport
) {
1782 var container
= document
.getElementById('label_test_result');
1784 container
.innerHTML
= transport
.responseText
;
1785 if (!Element
.visible(container
)) {
1786 Effect
.SlideDown(container
, { duration
: 0.5 });
1791 exception_error("labeltest_callback", e
);
1795 function labelTest() {
1798 var container
= document
.getElementById('label_test_result');
1800 var form
= document
.forms
['label_edit_form'];
1802 var sql_exp
= form
.sql_exp
.value
;
1803 var description
= form
.description
.value
;
1805 notify_progress("Loading, please wait...");
1807 var query
= "backend.php?op=pref-labels&subop=test&expr=" +
1808 param_escape(sql_exp
) + "&descr=" + param_escape(description
);
1810 new Ajax
.Request(query
, {
1811 onComplete: function (transport
) {
1812 labeltest_callback(transport
);
1818 exception_error("labelTest", e
);
1822 function isCdmMode() {
1823 return !document
.getElementById("headlinesList");
1826 function getSelectedArticleIds2() {
1827 var rows
= new Array();
1828 var cdm_mode
= isCdmMode();
1831 rows
= cdmGetSelectedArticles();
1833 rows
= getSelectedTableRowIds("headlinesList", "RROW", "RCHK");
1836 var ids
= new Array();
1838 for (var i
= 0; i
< rows
.length
; i
++) {
1839 var chk
= document
.getElementById("RCHK-" + rows
[i
]);
1840 if (chk
&& chk
.checked
) {
1848 function displayHelpInfobox(topic_id
) {
1850 var url
= "backend.php?op=help&tid=" + param_escape(topic_id
);
1852 var w
= window
.open(url
, "ttrss_help",
1853 "status=0,toolbar=0,location=0,width=450,height=500,scrollbars=1,menubar=0");
1857 function focus_element(id
) {
1859 var e
= document
.getElementById(id
);
1862 exception_error("focus_element", e
);
1867 function loading_set_progress(p
) {
1869 if (p
< last_progress_point
|| !Element
.visible("overlay")) return;
1871 debug("<b>loading_set_progress : " + p
+ " (" + last_progress_point
+ ")</b>");
1873 var o
= document
.getElementById("l_progress_i");
1875 // o.style.width = (p * 2) + "px";
1877 new Effect
.Scale(o
, p
, {
1879 scaleFrom
: last_progress_point
,
1880 scaleMode
: { originalWidth
: 200 },
1881 queue
: { position
: 'end', scope
: 'LSP-Q', limit
: 3 } });
1883 last_progress_point
= p
;
1886 exception_error("loading_set_progress", e
);
1890 function remove_splash() {
1891 if (Element
.visible("overlay")) {
1892 debug("about to remove splash, OMG!");
1893 Element
.hide("overlay");
1894 debug("removed splash!");
1898 function addLabelExample() {
1900 var form
= document
.forms
["label_edit_form"];
1902 var text
= form
.sql_exp
;
1903 var op
= form
.label_fields
[form
.label_fields
.selectedIndex
];
1904 var p
= form
.label_fields_param
;
1911 if (text
.value
!= "") {
1912 if (text
.value
.substring(text
.value
.length
-3, 3).toUpperCase() != "AND") {
1919 if (op
== "unread") {
1920 tmp
= tmp
+ "unread = true";
1923 if (op
== "updated") {
1924 tmp
= tmp
+ "last_read is null and unread = false";
1927 if (op
== "kw_title") {
1928 if (p
.value
== "") {
1929 alert("This action requires a parameter.");
1932 tmp
= tmp
+ "ttrss_entries.title like '%"+p
.value
+"%'";
1935 if (op
== "kw_content") {
1936 if (p
.value
== "") {
1937 alert("This action requires a parameter.");
1941 tmp
= tmp
+ "ttrss_entries.content like '%"+p
.value
+"%'";
1944 if (op
== "scoreE") {
1945 if (isNaN(parseInt(p
.value
))) {
1946 alert("This action expects numeric parameter.");
1949 tmp
= tmp
+ "score = " + p
.value
;
1952 if (op
== "scoreG") {
1953 if (isNaN(parseInt(p
.value
))) {
1954 alert("This action expects numeric parameter.");
1957 tmp
= tmp
+ "score > " + p
.value
;
1960 if (op
== "scoreL") {
1961 if (isNaN(parseInt(p
.value
))) {
1962 alert("This action expects numeric parameter.");
1965 tmp
= tmp
+ "score < " + p
.value
;
1968 if (op
== "newerD") {
1969 if (isNaN(parseInt(p
.value
))) {
1970 alert("This action expects numeric parameter.");
1973 tmp
= tmp
+ "updated > NOW() - INTERVAL '"+parseInt(p
.value
)+" days'";
1976 if (op
== "newerH") {
1977 if (isNaN(parseInt(p
.value
))) {
1978 alert("This action expects numeric parameter.");
1982 tmp
= tmp
+ "updated > NOW() - INTERVAL '"+parseInt(p
.value
)+" hours'";
1985 text
.value
= text
.value
+ tmp
;
1992 exception_error("addLabelExample", e
);
1998 function labelFieldsCheck(elem
) {
2000 var op
= elem
[elem
.selectedIndex
].value
;
2002 var p
= document
.forms
["label_edit_form"].label_fields_param
;
2004 if (op
== "kw_title" || op
== "kw_content" || op
== "scoreL" ||
2005 op
== "scoreG" || op
== "scoreE" || op
== "newerD" ||
2013 exception_error("labelFieldsCheck", e
);
2018 function getSelectedFeedsFromBrowser() {
2020 var list
= document
.getElementById("browseFeedList");
2021 if (!list
) list
= document
.getElementById("browseBigFeedList");
2023 var selected
= new Array();
2025 for (i
= 0; i
< list
.childNodes
.length
; i
++) {
2026 var child
= list
.childNodes
[i
];
2027 if (child
.id
&& child
.id
.match("FBROW-")) {
2028 var id
= child
.id
.replace("FBROW-", "");
2030 var cb
= document
.getElementById("FBCHK-" + id
);
2041 function updateFeedBrowser() {
2044 var query
= "backend.php?op=rpc&subop=feedBrowser";
2046 var search
= document
.getElementById("feed_browser_search");
2047 var limit
= document
.getElementById("feed_browser_limit");
2050 query
= query
+ "&limit=" + limit
[limit
.selectedIndex
].value
;
2054 query
= query
+ "&search=" + param_escape(search
.value
);
2057 notify_progress("Loading, please wait...", true);
2059 new Ajax
.Request(query
, {
2060 onComplete: function(transport
) {
2063 var c
= document
.getElementById("browseFeedList");
2064 var r
= transport
.responseXML
.getElementsByTagName("content")[0];
2065 var nr
= transport
.responseXML
.getElementsByTagName("num-results")[0];
2066 var sb
= document
.getElementById("feed_browser_subscribe");
2069 c
.innerHTML
= r
.firstChild
.nodeValue
;
2073 if (nr
.getAttribute("value") > 0) {
2074 sb
.disabled
= false;
2084 exception_error("updateFeedBrowser", e
);
2088 function browseFeeds(limit
) {
2092 var query
= "backend.php?op=pref-feeds&subop=browse";
2094 notify_progress("Loading, please wait...", true);
2096 new Ajax
.Request(query
, {
2097 onComplete: function(transport
) {
2098 infobox_callback2(transport
);
2103 exception_error("browseFeeds", e
);