]>
git.wh0rd.org - tt-rss.git/blob - functions.js
1 var hotkeys_enabled
= true;
2 var debug_mode_enabled
= false;
3 var xmlhttp_rpc
= Ajax
.getTransport();
5 /* add method to remove element from array */
7 Array
.prototype.remove = function(s
) {
8 for (var i
=0; i
< this.length
; i
++) {
9 if (s
== this[i
]) this.splice(i
, 1);
13 function browser_has_opacity() {
14 return navigator
.userAgent
.match("Gecko") != null ||
15 navigator
.userAgent
.match("Opera") != null;
19 return navigator
.userAgent
.match("MSIE");
23 return navigator
.userAgent
.match("Opera");
27 return navigator
.userAgent
.match("KHTML");
30 function is_safari() {
31 return navigator
.userAgent
.match("Safari");
34 function exception_error(location
, e
, silent
) {
38 var base_fname
= e
.fileName
.substring(e
.fileName
.lastIndexOf("/") + 1);
40 msg
= "Exception: " + e
.name
+ ", " + e
.message
+
41 "\nFunction: " + location
+ "()" +
42 "\nLocation: " + base_fname
+ ":" + e
.lineNumber
;
45 msg
= "Exception: " + e
+ "\nFunction: " + location
+ "()";
48 debug("<b>EXCEPTION: " + msg
+ "</b>");
55 function disableHotkeys() {
56 hotkeys_enabled
= false;
59 function enableHotkeys() {
60 hotkeys_enabled
= true;
63 function xmlhttp_ready(obj
) {
64 return obj
.readyState
== 4 || obj
.readyState
== 0 || !obj
.readyState
;
67 function open_article_callback() {
68 if (xmlhttp_rpc
.readyState
== 4) {
71 if (xmlhttp_rpc
.responseXML
) {
72 var link
= xmlhttp_rpc
.responseXML
.getElementsByTagName("link")[0];
73 var id
= xmlhttp_rpc
.responseXML
.getElementsByTagName("id")[0];
76 window
.open(link
.firstChild
.nodeValue
, "_blank");
79 id
= id
.firstChild
.nodeValue
;
80 if (!document
.getElementById("headlinesList")) {
81 window
.setTimeout("toggleUnread(" + id
+ ", 0)", 100);
88 exception_error("open_article_callback", e
);
93 function logout_callback() {
94 var container
= document
.getElementById('notify');
95 if (xmlhttp
.readyState
== 4) {
97 var date
= new Date();
98 var timestamp
= Math
.round(date
.getTime() / 1000);
99 window
.location
.href
= "tt-rss.php";
101 exception_error("logout_callback", e
);
106 function notify_callback() {
107 var container
= document
.getElementById('notify');
108 if (xmlhttp
.readyState
== 4) {
109 container
.innerHTML
=xmlhttp
.responseText
;
113 function rpc_notify_callback() {
114 var container
= document
.getElementById('notify');
115 if (xmlhttp_rpc
.readyState
== 4) {
116 container
.innerHTML
=xmlhttp_rpc
.responseText
;
120 function param_escape(arg
) {
121 if (typeof encodeURIComponent
!= 'undefined')
122 return encodeURIComponent(arg
);
127 function param_unescape(arg
) {
128 if (typeof decodeURIComponent
!= 'undefined')
129 return decodeURIComponent(arg
);
131 return unescape(arg
);
134 function delay(gap
) {
136 then
=new Date().getTime();
138 while((now
-then
)<gap
) {
139 now
=new Date().getTime();
143 var notify_hide_timerid
= false;
145 function hide_notify() {
146 var n
= document
.getElementById("notify");
148 n
.style
.display
= "none";
152 function notify_real(msg
, no_hide
, n_type
) {
154 var n
= document
.getElementById("notify");
155 var nb
= document
.getElementById("notify_body");
157 if (!n
|| !nb
) return;
159 if (notify_hide_timerid
) {
160 window
.clearTimeout(notify_hide_timerid
);
164 if (n
.style
.display
== "block") {
165 notify_hide_timerid
= window
.setTimeout("hide_notify()", 0);
169 n
.style
.display
= "block";
181 if (typeof __
!= 'undefined') {
186 n
.className
= "notify";
187 } else if (n_type
== 2) {
188 n
.className
= "notifyProgress";
189 msg
= "<img src='images/indicator_white.gif'> " + msg
;
190 } else if (n_type
== 3) {
191 n
.className
= "notifyError";
192 msg
= "<img src='images/sign_excl.png'> " + msg
;
193 } else if (n_type
== 4) {
194 n
.className
= "notifyInfo";
195 msg
= "<img src='images/sign_info.png'> " + msg
;
198 // msg = "<img src='images/live_com_loading.gif'> " + msg;
203 notify_hide_timerid
= window
.setTimeout("hide_notify()", 3000);
207 function notify(msg
, no_hide
) {
208 notify_real(msg
, no_hide
, 1);
211 function notify_progress(msg
, no_hide
) {
212 notify_real(msg
, no_hide
, 2);
215 function notify_error(msg
, no_hide
) {
216 notify_real(msg
, no_hide
, 3);
220 function notify_info(msg
, no_hide
) {
221 notify_real(msg
, no_hide
, 4);
224 function printLockingError() {
225 notify_info("Please wait until operation finishes.");
228 function hotkey_handler(e
) {
233 var shift_key
= false;
236 shift_key
= e
.shiftKey
;
241 if (!hotkeys_enabled
) return;
244 keycode
= window
.event
.keyCode
;
249 if (keycode
== 82) { // r
250 return scheduleFeedUpdate(true);
253 if (keycode
== 83) { // s
254 return displayDlg("search", getActiveFeedId());
257 if (keycode
== 85) { // u
258 if (getActiveFeedId()) {
259 return viewfeed(getActiveFeedId(), "ForceUpdate");
263 if (keycode
== 65) { // a
264 return toggleDispRead();
267 var feedlist
= document
.getElementById('feedList');
269 if (keycode
== 74) { // j
270 var feed
= getActiveFeedId();
271 var new_feed
= getRelativeFeedId(feedlist
, feed
, 'prev');
272 if (new_feed
) viewfeed(new_feed
, '');
275 if (keycode
== 75) { // k
276 var feed
= getActiveFeedId();
277 var new_feed
= getRelativeFeedId(feedlist
, feed
, 'next');
278 if (new_feed
) viewfeed(new_feed
, '');
281 if (keycode
== 78 || keycode
== 40) { // n, down
282 if (typeof moveToPost
!= 'undefined') {
283 return moveToPost('next');
287 if (keycode
== 80 || keycode
== 38) { // p, up
288 if (typeof moveToPost
!= 'undefined') {
289 return moveToPost('prev');
293 if (keycode
== 68 && shift_key
) { // d
294 if (!debug_mode_enabled
) {
295 document
.getElementById('debug_output').style
.display
= 'block';
296 debug('debug mode activated');
298 document
.getElementById('debug_output').style
.display
= 'none';
301 debug_mode_enabled
= !debug_mode_enabled
;
304 if (keycode
== 190 && shift_key
) { // >
308 if (keycode
== 188 && shift_key
) { // <
312 if (keycode
== 191 && shift_key
) { // ?
316 if (keycode
== 69 && shift_key
) { // e
317 return editFeedDlg(getActiveFeedId());
320 if (keycode
== 70 && shift_key
) { // f
321 if (getActiveFeedId()) {
322 return catchupCurrentFeed();
326 if (keycode
== 80 && shift_key
) { // p
327 if (getActiveFeedId()) {
328 return catchupPage();
332 if (keycode
== 86) { // v
333 if (getActiveArticleId()) {
334 openArticleInNewWindow(getActiveArticleId());
338 if (keycode
== 84) { // t
340 var id
= getActiveArticleId();
344 var cb
= document
.getElementById("RCHK-" + id
);
347 cb
.checked
= !cb
.checked
;
348 toggleSelectRowById(cb
, "RROW-" + id
);
353 if (keycode
== 67) { // c
354 var id
= getActiveArticleId();
361 if (typeof localHotkeyHandler
!= 'undefined') {
363 return localHotkeyHandler(e
);
365 exception_error("hotkey_handler, local:", e
);
369 debug("KP=" + keycode
);
371 exception_error("hotkey_handler", e
);
375 function cleanSelectedList(element
) {
376 var content
= document
.getElementById(element
);
378 if (!document
.getElementById("feedCatHolder")) {
379 for (i
= 0; i
< content
.childNodes
.length
; i
++) {
380 var child
= content
.childNodes
[i
];
382 child
.className
= child
.className
.replace("Selected", "");
388 for (i
= 0; i
< content
.childNodes
.length
; i
++) {
389 var child
= content
.childNodes
[i
];
390 if (child
.id
== "feedCatHolder") {
392 var fcat
= child
.lastChild
;
393 for (j
= 0; j
< fcat
.childNodes
.length
; j
++) {
394 var feed
= fcat
.childNodes
[j
];
395 feed
.className
= feed
.className
.replace("Selected", "");
403 function cleanSelected(element
) {
404 var content
= document
.getElementById(element
);
406 for (i
= 0; i
< content
.rows
.length
; i
++) {
407 content
.rows
[i
].className
= content
.rows
[i
].className
.replace("Selected", "");
411 function getVisibleUnreadHeadlines() {
412 var content
= document
.getElementById("headlinesList");
414 var rows
= new Array();
416 for (i
= 0; i
< content
.rows
.length
; i
++) {
417 var row_id
= content
.rows
[i
].id
.replace("RROW-", "");
418 if (row_id
.length
> 0 && content
.rows
[i
].className
.match("Unread")) {
425 function getVisibleHeadlineIds() {
427 var content
= document
.getElementById("headlinesList");
429 var rows
= new Array();
431 for (i
= 0; i
< content
.rows
.length
; i
++) {
432 var row_id
= content
.rows
[i
].id
.replace("RROW-", "");
433 if (row_id
.length
> 0) {
440 function getFirstVisibleHeadlineId() {
441 var rows
= getVisibleHeadlineIds();
445 function getLastVisibleHeadlineId() {
446 var rows
= getVisibleHeadlineIds();
447 return rows
[rows
.length
-1];
450 function markHeadline(id
) {
451 var row
= document
.getElementById("RROW-" + id
);
453 var is_active
= false;
455 if (row
.className
.match("Active")) {
458 row
.className
= row
.className
.replace("Selected", "");
459 row
.className
= row
.className
.replace("Active", "");
460 row
.className
= row
.className
.replace("Insensitive", "");
463 row
.className
= row
.className
= "Active";
466 var check
= document
.getElementById("RCHK-" + id
);
469 check
.checked
= true;
472 row
.className
= row
.className
+ "Selected";
477 function getFeedIds() {
478 var content
= document
.getElementById("feedsList");
480 var rows
= new Array();
482 for (i
= 0; i
< content
.rows
.length
; i
++) {
483 var id
= content
.rows
[i
].id
.replace("FEEDR-", "");
492 function setCookie(name
, value
, lifetime
, path
, domain
, secure
) {
498 d
.setTime(d
.getTime() + (lifetime
* 1000));
501 debug("setCookie: " + name
+ " => " + value
+ ": " + d
);
503 int_setCookie(name
, value
, d
, path
, domain
, secure
);
507 function int_setCookie(name
, value
, expires
, path
, domain
, secure
) {
508 document
.cookie
= name
+ "=" + escape(value
) +
509 ((expires
) ? "; expires=" + expires
.toGMTString() : "") +
510 ((path
) ? "; path=" + path
: "") +
511 ((domain
) ? "; domain=" + domain
: "") +
512 ((secure
) ? "; secure" : "");
515 function delCookie(name
, path
, domain
) {
516 if (getCookie(name
)) {
517 document
.cookie
= name
+ "=" +
518 ((path
) ? ";path=" + path
: "") +
519 ((domain
) ? ";domain=" + domain
: "" ) +
520 ";expires=Thu, 01-Jan-1970 00:00:01 GMT";
525 function getCookie(name
) {
527 var dc
= document
.cookie
;
528 var prefix
= name
+ "=";
529 var begin
= dc
.indexOf("; " + prefix
);
531 begin
= dc
.indexOf(prefix
);
532 if (begin
!= 0) return null;
537 var end
= document
.cookie
.indexOf(";", begin
);
541 return unescape(dc
.substring(begin
+ prefix
.length
, end
));
544 function disableContainerChildren(id
, disable
, doc
) {
546 if (!doc
) doc
= document
;
548 var container
= doc
.getElementById(id
);
551 //alert("disableContainerChildren: element " + id + " not found");
555 for (var i
= 0; i
< container
.childNodes
.length
; i
++) {
556 var child
= container
.childNodes
[i
];
559 child
.disabled
= disable
;
565 if (child
.className
&& child
.className
.match("button")) {
566 child
.className
= "disabledButton";
569 if (child
.className
&& child
.className
.match("disabledButton")) {
570 child
.className
= "button";
577 function gotoPreferences() {
578 document
.location
.href
= "prefs.php";
581 function gotoMain() {
582 document
.location
.href
= "tt-rss.php";
585 function gotoExportOpml() {
586 document
.location
.href
= "opml.php?op=Export";
589 function getActiveFeedId() {
590 // return getCookie("ttrss_vf_actfeed");
592 debug("gAFID: " + active_feed_id
);
593 return active_feed_id
;
595 exception_error("getActiveFeedId", e
);
599 function activeFeedIsCat() {
600 return active_feed_is_cat
;
603 function setActiveFeedId(id
) {
604 // return setCookie("ttrss_vf_actfeed", id);
606 debug("sAFID(" + id
+ ")");
609 exception_error("setActiveFeedId", e
);
613 function parse_counters(reply
, scheduled_call
) {
618 if (reply
.firstChild
&& reply
.firstChild
.firstChild
) {
619 debug("<b>wrong element passed to parse_counters, adjusting.</b>");
620 reply
= reply
.firstChild
;
623 for (var l
= 0; l
< reply
.childNodes
.length
; l
++) {
624 if (!reply
.childNodes
[l
] ||
625 typeof(reply
.childNodes
[l
].getAttribute
) == "undefined") {
626 // where did this come from?
630 var id
= reply
.childNodes
[l
].getAttribute("id");
631 var t
= reply
.childNodes
[l
].getAttribute("type");
632 var ctr
= reply
.childNodes
[l
].getAttribute("counter");
633 var error
= reply
.childNodes
[l
].getAttribute("error");
634 var has_img
= reply
.childNodes
[l
].getAttribute("hi");
635 var updated
= reply
.childNodes
[l
].getAttribute("updated");
637 if (id
== "global-unread") {
643 if (id
== "subscribed-feeds") {
648 if (t
== "category") {
649 var catctr
= document
.getElementById("FCATCTR-" + id
);
651 catctr
.innerHTML
= "(" + ctr
+ ")";
653 catctr
.className
= "catCtrHasUnread";
655 catctr
.className
= "catCtrNoUnread";
661 var feedctr
= document
.getElementById("FEEDCTR-" + id
);
662 var feedu
= document
.getElementById("FEEDU-" + id
);
663 var feedr
= document
.getElementById("FEEDR-" + id
);
664 var feed_img
= document
.getElementById("FIMG-" + id
);
665 var feedlink
= document
.getElementById("FEEDL-" + id
);
666 var feedupd
= document
.getElementById("FLUPD-" + id
);
668 if (updated
&& feedlink
) {
670 feedlink
.title
= "Error: " + error
+ " (" + updated
+ ")";
672 feedlink
.title
= "Updated: " + updated
;
676 if (updated
&& feedupd
) {
678 feedupd
.innerHTML
= updated
+ " (Error)";
680 feedupd
.innerHTML
= updated
;
684 if (feedctr
&& feedu
&& feedr
) {
686 if (feedu
.innerHTML
!= ctr
&& id
== getActiveFeedId() && scheduled_call
) {
690 var row_needs_hl
= (ctr
> 0 && ctr
> parseInt(feedu
.innerHTML
));
692 feedu
.innerHTML
= ctr
;
695 feedr
.className
= feedr
.className
.replace("feed", "error");
697 feedr
.className
= feedr
.className
.replace("error", "feed");
701 feedctr
.className
= "odd";
702 if (!feedr
.className
.match("Unread")) {
703 var is_selected
= feedr
.className
.match("Selected");
705 feedr
.className
= feedr
.className
.replace("Selected", "");
706 feedr
.className
= feedr
.className
.replace("Unread", "");
708 feedr
.className
= feedr
.className
+ "Unread";
711 feedr
.className
= feedr
.className
+ "Selected";
717 new Effect
.Highlight(feedr
, {duration
: 1, startcolor
: "#fff7d5"});
720 feedctr
.className
= "invisible";
721 feedr
.className
= feedr
.className
.replace("Unread", "");
726 hideOrShowFeeds(document
, getInitParam("hide_read_feeds") == 1);
728 var feeds_stored
= number_of_feeds
;
730 debug("Feed counters, C: " + feeds_found
+ ", S:" + feeds_stored
);
732 if (feeds_stored
!= feeds_found
) {
733 number_of_feeds
= feeds_found
;
735 if (feeds_stored
!= 0) {
736 debug("Subscribed feed number changed, refreshing feedlist");
737 setTimeout('updateFeedList(false, false)', 50);
742 exception_error("parse_counters", e
);
746 function parse_counters_reply(xmlhttp
, scheduled_call
) {
748 if (!xmlhttp
.responseXML
) {
749 notify_error("Backend did not return valid XML", true);
753 var reply
= xmlhttp
.responseXML
.firstChild
;
756 notify_error("Backend did not return expected XML object", true);
761 var error_code
= false;
762 var error_msg
= false;
764 if (reply
.firstChild
) {
765 error_code
= reply
.firstChild
.getAttribute("error-code");
766 error_msg
= reply
.firstChild
.getAttribute("error-msg");
770 error_code
= reply
.getAttribute("error-code");
771 error_msg
= reply
.getAttribute("error-msg");
774 if (error_code
&& error_code
!= 0) {
775 debug("refetch_callback: got error code " + error_code
);
776 return fatalError(error_code
, error_msg
);
779 var counters
= reply
.firstChild
;
781 parse_counters(counters
, scheduled_call
);
783 var runtime_info
= counters
.nextSibling
;
785 parse_runtime_info(runtime_info
);
787 if (getInitParam("feeds_sort_by_unread") == 1) {
791 hideOrShowFeeds(document
, getInitParam("hide_read_feeds") == 1);
795 function all_counters_callback() {
796 if (xmlhttp_rpc
.readyState
== 4) {
798 /* if (!xmlhttp_rpc.responseXML || !xmlhttp_rpc.responseXML.firstChild) {
799 debug("[all_counters_callback] backend did not return valid XML");
803 debug("in all_counters_callback : " + xmlhttp_rpc.responseXML);
805 var reply = xmlhttp_rpc.responseXML.firstChild;
807 var counters = reply.firstChild;
809 parse_counters(counters);
811 var runtime = counters.nextSibling;
814 parse_runtime_info(runtime);
817 if (getInitParam("feeds_sort_by_unread") == 1) {
821 hideOrShowFeeds(document, getInitParam("hide_read_feeds") == 1); */
823 debug("in all_counters_callback");
825 parse_counters_reply(xmlhttp_rpc
);
828 exception_error("all_counters_callback", e
);
833 function get_feed_entry_unread(doc
, elem
) {
835 var id
= elem
.id
.replace("FEEDR-", "");
842 return parseInt(doc
.getElementById("FEEDU-" + id
).innerHTML
);
848 function resort_category(doc
, node
) {
849 debug("resort_category: " + node
);
851 if (node
.hasChildNodes() && node
.firstChild
.nextSibling
!= false) {
852 for (i
= 0; i
< node
.childNodes
.length
; i
++) {
853 if (node
.childNodes
[i
].nodeName
!= "LI") { continue; }
855 if (get_feed_entry_unread(doc
, node
.childNodes
[i
]) < 0) {
859 for (j
= i
+1; j
< node
.childNodes
.length
; j
++) {
860 if (node
.childNodes
[j
].nodeName
!= "LI") { continue; }
862 var tmp_val
= get_feed_entry_unread(doc
, node
.childNodes
[i
]);
863 var cur_val
= get_feed_entry_unread(doc
, node
.childNodes
[j
]);
865 if (cur_val
> tmp_val
) {
866 tempnode_i
= node
.childNodes
[i
].cloneNode(true);
867 tempnode_j
= node
.childNodes
[j
].cloneNode(true);
868 node
.replaceChild(tempnode_i
, node
.childNodes
[j
]);
869 node
.replaceChild(tempnode_j
, node
.childNodes
[i
]);
878 function resort_feedlist() {
879 debug("resort_feedlist");
883 if (fd
.getElementById("feedCatHolder")) {
885 var feeds
= fd
.getElementById("feedList");
886 var child
= feeds
.firstChild
;
890 if (child
.id
== "feedCatHolder") {
891 resort_category(fd
, child
.firstChild
);
894 child
= child
.nextSibling
;
898 resort_category(fd
, fd
.getElementById("feedList"));
902 function update_all_counters(feed
) {
903 if (xmlhttp_ready(xmlhttp_rpc
)) {
904 var query
= "backend.php?op=rpc&subop=getAllCounters";
907 query
= query
+ "&aid=" + feed
;
910 if (tagsAreDisplayed()) {
911 query
= query
+ "&omode=lt";
913 query
= query
+ "&omode=flc";
916 debug("update_all_counters QUERY: " + query
);
918 var date
= new Date();
919 var timestamp
= Math
.round(date
.getTime() / 1000);
920 query
= query
+ "&ts=" + timestamp
922 xmlhttp_rpc
.open("GET", query
, true);
923 xmlhttp_rpc
.onreadystatechange
=all_counters_callback
;
924 xmlhttp_rpc
.send(null);
928 function popupHelp(tid
) {
929 var w
= window
.open("backend.php?op=help&tid=" + tid
,
931 "menubar=no,location=no,resizable=yes,scrollbars=yes,status=no");
934 /** * @(#)isNumeric.js * * Copyright (c) 2000 by Sundar Dorai-Raj
935 * * @author Sundar Dorai-Raj
936 * * Email: sdoraira@vt.edu
937 * * This program is free software; you can redistribute it and/or
938 * * modify it under the terms of the GNU General Public License
939 * * as published by the Free Software Foundation; either version 2
940 * * of the License, or (at your option) any later version,
941 * * provided that any use properly credits the author.
942 * * This program is distributed in the hope that it will be useful,
943 * * but WITHOUT ANY WARRANTY; without even the implied warranty of
944 * * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
945 * * GNU General Public License for more details at http://www.gnu.org * * */
947 var numbers
=".0123456789";
948 function isNumeric(x
) {
949 // is x a String or a character?
951 // remove negative sign
953 for(j
=0;j
<x
.length
;j
++) {
954 // call isNumeric recursively for each character
955 number
=isNumeric(x
.substring(j
,j
+1));
956 if(!number
) return number
;
961 // if x is number return true
962 if(numbers
.indexOf(x
)>=0) return true;
968 function hideOrShowFeeds(doc
, hide
) {
970 debug("hideOrShowFeeds: " + doc
+ ", " + hide
);
974 var list
= fd
.getElementById("feedList");
976 if (fd
.getElementById("feedCatHolder")) {
978 var feeds
= fd
.getElementById("feedList");
979 var child
= feeds
.firstChild
;
983 if (child
.id
== "feedCatHolder") {
984 hideOrShowFeedsCategory(fd
, child
.firstChild
, hide
, child
.previousSibling
);
987 child
= child
.nextSibling
;
991 hideOrShowFeedsCategory(fd
, fd
.getElementById("feedList"), hide
);
995 function hideOrShowFeedsCategory(doc
, node
, hide
, cat_node
) {
997 // debug("hideOrShowFeedsCategory: " + node + " (" + hide + ")");
1002 debug("hideOrShowFeeds: passed node is null, aborting");
1006 if (node
.hasChildNodes() && node
.firstChild
.nextSibling
!= false) {
1007 for (i
= 0; i
< node
.childNodes
.length
; i
++) {
1008 if (node
.childNodes
[i
].nodeName
!= "LI") { continue; }
1010 if (node
.childNodes
[i
].style
!= undefined) {
1012 var has_unread
= (node
.childNodes
[i
].className
!= "feed" &&
1013 node
.childNodes
[i
].className
!= "label" &&
1014 node
.childNodes
[i
].className
!= "tag");
1016 // debug(node.childNodes[i].id + " --> " + has_unread);
1018 if (hide
&& !has_unread
) {
1019 //node.childNodes[i].style.display = "none";
1020 Effect
.Fade(node
.childNodes
[i
], {duration
: 0.3});
1024 node
.childNodes
[i
].style
.display
= "list-item";
1025 //Effect.Appear(node.childNodes[i], {duration : 0.3});
1029 node
.childNodes
[i
].style
.display
= "list-item";
1031 //Effect.Appear(node.childNodes[i], {duration : 0.3});
1032 //Effect.Highlight(node.childNodes[i]);
1038 if (cat_unread
== 0) {
1039 if (cat_node
.style
== undefined) {
1040 debug("ERROR: supplied cat_node " + cat_node
+
1041 " has no styles. WTF?");
1045 //cat_node.style.display = "none";
1046 Effect
.Fade(cat_node
, {duration
: 0.3});
1048 cat_node
.style
.display
= "list-item";
1052 cat_node
.style
.display
= "list-item";
1058 // debug("unread for category: " + cat_unread);
1061 function selectTableRow(r
, do_select
) {
1062 r
.className
= r
.className
.replace("Selected", "");
1065 r
.className
= r
.className
+ "Selected";
1069 function selectTableRowById(elem_id
, check_id
, do_select
) {
1073 var row
= document
.getElementById(elem_id
);
1076 selectTableRow(row
, do_select
);
1079 var check
= document
.getElementById(check_id
);
1082 check
.checked
= do_select
;
1085 exception_error("selectTableRowById", e
);
1089 function selectTableRowsByIdPrefix(content_id
, prefix
, check_prefix
, do_select
,
1090 classcheck
, reset_others
) {
1092 var content
= document
.getElementById(content_id
);
1095 alert("[selectTableRows] Element " + content_id
+ " not found.");
1099 for (i
= 0; i
< content
.rows
.length
; i
++) {
1100 if (!classcheck
|| content
.rows
[i
].className
.match(classcheck
)) {
1102 if (content
.rows
[i
].id
.match(prefix
)) {
1103 selectTableRow(content
.rows
[i
], do_select
);
1105 var row_id
= content
.rows
[i
].id
.replace(prefix
, "");
1106 var check
= document
.getElementById(check_prefix
+ row_id
);
1109 check
.checked
= do_select
;
1111 } else if (reset_others
) {
1112 selectTableRow(content
.rows
[i
], false);
1114 var row_id
= content
.rows
[i
].id
.replace(prefix
, "");
1115 var check
= document
.getElementById(check_prefix
+ row_id
);
1118 check
.checked
= false;
1122 } else if (reset_others
) {
1123 selectTableRow(content
.rows
[i
], false);
1125 var row_id
= content
.rows
[i
].id
.replace(prefix
, "");
1126 var check
= document
.getElementById(check_prefix
+ row_id
);
1129 check
.checked
= false;
1136 function getSelectedTableRowIds(content_id
, prefix
) {
1138 var content
= document
.getElementById(content_id
);
1141 alert("[getSelectedTableRowIds] Element " + content_id
+ " not found.");
1145 var sel_rows
= new Array();
1147 for (i
= 0; i
< content
.rows
.length
; i
++) {
1148 if (content
.rows
[i
].id
.match(prefix
) &&
1149 content
.rows
[i
].className
.match("Selected")) {
1151 var row_id
= content
.rows
[i
].id
.replace(prefix
+ "-", "");
1152 sel_rows
.push(row_id
);
1160 function toggleSelectRowById(sender
, id
) {
1161 var row
= document
.getElementById(id
);
1163 if (sender
.checked
) {
1164 if (!row
.className
.match("Selected")) {
1165 row
.className
= row
.className
+ "Selected";
1168 if (row
.className
.match("Selected")) {
1169 row
.className
= row
.className
.replace("Selected", "");
1174 function toggleSelectListRow(sender
) {
1175 var parent_row
= sender
.parentNode
;
1177 if (sender
.checked
) {
1178 if (!parent_row
.className
.match("Selected")) {
1179 parent_row
.className
= parent_row
.className
+ "Selected";
1182 if (parent_row
.className
.match("Selected")) {
1183 parent_row
.className
= parent_row
.className
.replace("Selected", "");
1188 function tSR(sender
) {
1189 return toggleSelectRow(sender
);
1192 function toggleSelectRow(sender
) {
1193 var parent_row
= sender
.parentNode
.parentNode
;
1195 if (sender
.checked
) {
1196 if (!parent_row
.className
.match("Selected")) {
1197 parent_row
.className
= parent_row
.className
+ "Selected";
1200 if (parent_row
.className
.match("Selected")) {
1201 parent_row
.className
= parent_row
.className
.replace("Selected", "");
1206 function openExternalUrl(url
) {
1207 var w
= window
.open(url
);
1210 function getRelativeFeedId(list
, id
, direction
, unread_only
) {
1211 var rows
= list
.getElementsByTagName("LI");
1212 var feeds
= new Array();
1214 for (var i
= 0; i
< rows
.length
; i
++) {
1215 if (rows
[i
].id
.match("FEEDR-")) {
1217 if (rows
[i
].id
== "FEEDR-" + id
|| (Element
.visible(rows
[i
]) && Element
.visible(rows
[i
].parentNode
))) {
1220 (rows
[i
].className
.match("Unread") || rows
[i
].id
== "FEEDR-" + id
)) {
1221 feeds
.push(rows
[i
].id
.replace("FEEDR-", ""));
1228 if (direction
== "next") {
1229 return feeds
.shift();
1234 if (direction
== "next") {
1235 var idx
= feeds
.indexOf(id
);
1236 if (idx
!= -1 && idx
< feeds
.length
) {
1237 return feeds
[idx
+1];
1239 return getRelativeFeedId(list
, false, direction
, unread_only
);
1242 var idx
= feeds
.indexOf(id
);
1244 return feeds
[idx
-1];
1246 return getRelativeFeedId(list
, false, direction
, unread_only
);
1253 if (direction == "next") {
1254 for (i = 0; i < list.childNodes.length; i++) {
1255 var child = list.childNodes[i];
1256 if (child.id && child.id == "feedCatHolder") {
1257 if (child.lastChild) {
1258 var cr = getRelativeFeedId(child.firstChild, id, direction, unread_only);
1261 } else if (child.id && child.id.match("FEEDR-")) {
1262 return child.id.replace('FEEDR-', '');
1267 // FIXME select last feed doesn't work when only unread feeds are visible
1269 if (direction == "prev") {
1270 for (i = list.childNodes.length-1; i >= 0; i--) {
1271 var child = list.childNodes[i];
1272 if (child.id == "feedCatHolder" && Element.visible(child)) {
1273 if (child.firstChild) {
1274 var cr = getRelativeFeedId(child.firstChild, id, direction);
1277 } else if (child.id.match("FEEDR-")) {
1279 if (getInitParam("hide_read_feeds") == 1) {
1280 if (child.className != "feed") {
1281 // alert(child.className);
1282 return child.id.replace('FEEDR-', '');
1285 return child.id.replace('FEEDR-', '');
1292 var feed = list.ownerDocument.getElementById("FEEDR-" + id);
1294 if (getInitParam("hide_read_feeds") == 1) {
1298 if (direction == "next") {
1304 if (e.nextSibling) {
1308 } else if (e.parentNode.parentNode.nextSibling) {
1310 var this_cat = e.parentNode.parentNode;
1314 if (this_cat && this_cat.nextSibling) {
1315 while (!e && this_cat.nextSibling) {
1316 this_cat = this_cat.nextSibling;
1317 if (this_cat.id == "feedCatHolder") {
1318 e = this_cat.firstChild.firstChild;
1328 if (!unread_only || (unread_only && e.className != "feed" &&
1329 e.className.match("feed"))) {
1330 if (e.parentNode.parentNode &&
1331 Element.visible(e.parentNode.parentNode)) {
1332 return e.id.replace("FEEDR-", "");
1338 } else if (direction == "prev") {
1344 if (e.previousSibling) {
1346 e = e.previousSibling;
1348 } else if (e.parentNode.parentNode.previousSibling) {
1350 var this_cat = e.parentNode.parentNode;
1354 if (this_cat && this_cat.previousSibling) {
1355 while (!e && this_cat.previousSibling) {
1356 this_cat = this_cat.previousSibling;
1357 if (this_cat.id == "feedCatHolder") {
1358 e = this_cat.firstChild.lastChild;
1368 if (!unread_only || (unread_only && e.className != "feed" &&
1369 e.className.match("feed"))) {
1370 if (e.parentNode.parentNode &&
1371 Element.visible(e.parentNode.parentNode)) {
1372 return e.id.replace("FEEDR-", "");
1381 function showBlockElement(id
, h_id
) {
1382 var elem
= document
.getElementById(id
);
1385 elem
.style
.display
= "block";
1388 elem
= document
.getElementById(h_id
);
1390 elem
.style
.display
= "none";
1394 alert("[showBlockElement] can't find element with id " + id
);
1398 function hideParentElement(e
) {
1399 e
.parentNode
.style
.display
= "none";
1402 function dropboxSelect(e
, v
) {
1403 for (i
= 0; i
< e
.length
; i
++) {
1404 if (e
[i
].value
== v
) {
1405 e
.selectedIndex
= i
;
1411 // originally stolen from http://www.11tmr.com/11tmr.nsf/d6plinks/MWHE-695L9Z
1412 // bugfixed just a little bit :-)
1413 function getURLParam(strParamName
){
1415 var strHref
= window
.location
.href
;
1417 if (strHref
.indexOf("#") == strHref
.length
-1) {
1418 strHref
= strHref
.substring(0, strHref
.length
-1);
1421 if ( strHref
.indexOf("?") > -1 ){
1422 var strQueryString
= strHref
.substr(strHref
.indexOf("?"));
1423 var aQueryString
= strQueryString
.split("&");
1424 for ( var iParam
= 0; iParam
< aQueryString
.length
; iParam
++ ){
1425 if (aQueryString
[iParam
].indexOf(strParamName
+ "=") > -1 ){
1426 var aParam
= aQueryString
[iParam
].split("=");
1427 strReturn
= aParam
[1];
1435 function leading_zero(p
) {
1437 if (s
.length
== 1) s
= "0" + s
;
1441 function closeInfoBox(cleanup
) {
1443 if (!is_msie() && !getInitParam("infobox_disable_overlay")) {
1444 var overlay
= document
.getElementById("dialog_overlay");
1446 overlay
.style
.display
= "none";
1450 var box
= document
.getElementById('infoBox');
1451 var shadow
= document
.getElementById('infoBoxShadow');
1454 shadow
.style
.display
= "none";
1456 box
.style
.display
= "none";
1459 if (cleanup
) box
.innerHTML
= " ";
1467 function displayDlg(id
, param
) {
1469 if (!xmlhttp_ready(xmlhttp
)) {
1470 printLockingError();
1474 notify_progress("Loading, please wait...", true);
1476 xmlhttp
.open("GET", "backend.php?op=dlg&id=" +
1477 param_escape(id
) + "¶m=" + param_escape(param
), true);
1478 xmlhttp
.onreadystatechange
=infobox_callback
;
1486 function infobox_submit_callback() {
1487 if (xmlhttp
.readyState
== 4) {
1491 // called from prefs, reload tab
1493 selectTab(active_tab
, false);
1497 if (xmlhttp
.responseText
) {
1498 notify_info(xmlhttp
.responseText
);
1504 function infobox_callback() {
1505 if (xmlhttp
.readyState
== 4) {
1509 if (!is_msie() && !getInitParam("infobox_disable_overlay")) {
1510 var overlay
= document
.getElementById("dialog_overlay");
1512 overlay
.style
.display
= "block";
1516 var box
= document
.getElementById('infoBox');
1517 var shadow
= document
.getElementById('infoBoxShadow');
1520 new Draggable(shadow
);
1522 box
.innerHTML
=xmlhttp
.responseText
;
1524 shadow
.style
.display
= "block";
1526 box
.style
.display
= "block";
1530 /* FIXME this needs to be moved out somewhere */
1532 if (document
.getElementById("tags_choices")) {
1533 new Ajax
.Autocompleter('tags_str', 'tags_choices',
1534 "backend.php?op=rpc&subop=completeTags",
1535 { tokens
: ',', paramName
: "search" });
1540 exception_error("infobox_callback", e
);
1545 function helpbox_callback() {
1546 if (xmlhttp
.readyState
== 4) {
1547 var box
= document
.getElementById('helpBox');
1548 var shadow
= document
.getElementById('helpBoxShadow');
1550 box
.innerHTML
=xmlhttp
.responseText
;
1552 shadow
.style
.display
= "block";
1554 box
.style
.display
= "block";
1561 function addFilter() {
1563 if (!xmlhttp_ready(xmlhttp
)) {
1564 printLockingError();
1568 var form
= document
.forms
['filter_add_form'];
1569 var reg_exp
= form
.reg_exp
.value
;
1571 if (reg_exp
== "") {
1572 alert(__("Can't add filter: nothing to match on."));
1576 var query
= Form
.serialize("filter_add_form");
1578 xmlhttp
.open("GET", "backend.php?" + query
, true);
1579 xmlhttp
.onreadystatechange
=infobox_submit_callback
;
1585 function toggleSubmitNotEmpty(e
, submit_id
) {
1587 document
.getElementById(submit_id
).disabled
= (e
.value
== "")
1589 exception_error("toggleSubmitNotEmpty", e
);
1593 function isValidURL(s
) {
1594 return s
.match("http://") != null || s
.match("https://") != null || s
.match("feed://") != null;
1597 function qaddFeed() {
1599 if (!xmlhttp_ready(xmlhttp
)) {
1600 printLockingError();
1604 var form
= document
.forms
['feed_add_form'];
1605 var feed_url
= form
.feed_url
.value
;
1607 if (feed_url
== "") {
1608 alert(__("Can't subscribe: no feed URL given."));
1612 notify_progress(__("Subscribing to feed..."), true);
1616 var feeds_doc
= document
;
1618 // feeds_doc.location.href = "backend.php?op=error&msg=Loading,%20please wait...";
1620 var query
= Form
.serialize("feed_add_form");
1622 debug("subscribe q: " + query
);
1624 /* xmlhttp.open("GET", "backend.php?" + query, true);
1625 xmlhttp.onreadystatechange=dlg_frefresh_callback;
1626 xmlhttp.send(null); */
1628 xmlhttp
.open("POST", "backend.php", true);
1629 xmlhttp
.onreadystatechange
=dlg_frefresh_callback
;
1630 xmlhttp
.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
1631 xmlhttp
.send(query
);
1636 function filterCR(e
, f
)
1641 key
= window
.event
.keyCode
; //IE
1643 key
= e
.which
; //firefox
1646 if (typeof f
!= 'undefined') {
1657 function getMainContext() {
1661 function getFeedsContext() {
1665 function getContentContext() {
1669 function getHeadlinesContext() {
1673 var debug_last_class
= "even";
1675 function debug(msg
) {
1677 if (debug_last_class
== "even") {
1678 debug_last_class
= "odd";
1680 debug_last_class
= "even";
1683 var c
= document
.getElementById('debug_output');
1684 if (c
&& c
.style
.display
== "block") {
1685 while (c
.lastChild
!= 'undefined' && c
.childNodes
.length
> 100) {
1686 c
.removeChild(c
.lastChild
);
1690 var ts
= leading_zero(d
.getHours()) + ":" + leading_zero(d
.getMinutes()) +
1691 ":" + leading_zero(d
.getSeconds());
1692 c
.innerHTML
= "<li class=\"" + debug_last_class
+ "\"><span class=\"debugTS\">[" + ts
+ "]</span> " +
1693 msg
+ "</li>" + c
.innerHTML
;
1697 function getInitParam(key
) {
1698 return init_params
[key
];
1701 function storeInitParam(key
, value
) {
1702 debug("<b>storeInitParam is OBSOLETE: " + key
+ " => " + value
+ "</b>");
1703 init_params
[key
] = value
;
1706 function fatalError(code
, message
) {
1710 window
.location
.href
= "tt-rss.php";
1711 } else if (code
== 5) {
1712 window
.location
.href
= "update.php";
1714 var fe
= document
.getElementById("fatal_error");
1715 var fc
= document
.getElementById("fatal_error_msg");
1717 if (message
== "") message
= "Unknown error";
1719 fc
.innerHTML
= "<img src='images/sign_excl.png'> " + message
+ " (Code " + code
+ ")";
1721 fe
.style
.display
= "block";
1725 exception_error("fatalError", e
);
1729 function getFeedName(id
, is_cat
) {
1730 var d
= getFeedsContext().document
;
1735 e
= d
.getElementById("FCATN-" + id
);
1737 e
= d
.getElementById("FEEDN-" + id
);
1740 return e
.innerHTML
.stripTags();
1746 function viewContentUrl(url
) {
1747 getContentContext().location
= url
;
1750 function filterDlgCheckAction(sender
) {
1754 var action
= sender
[sender
.selectedIndex
].value
;
1756 var form
= document
.forms
["filter_add_form"];
1759 form
= document
.forms
["filter_edit_form"];
1763 debug("filterDlgCheckAction: can't find form!");
1767 var action_param
= form
.action_param
;
1769 if (!action_param
) {
1770 debug("filterDlgCheckAction: can't find action param!");
1774 // if selected action supports parameters, enable params field
1776 action_param
.disabled
= false;
1778 action_param
.disabled
= true;
1782 exception_error(e
, "filterDlgCheckAction");
1787 function explainError(code
) {
1788 return displayDlg("explainError", code
);
1791 function logoutUser() {
1793 if (xmlhttp_ready(xmlhttp_rpc
)) {
1795 notify_progress("Logging out, please wait...", true);
1797 xmlhttp_rpc
.open("GET", "backend.php?op=rpc&subop=logout", true);
1798 xmlhttp_rpc
.onreadystatechange
=logout_callback
;
1799 xmlhttp_rpc
.send(null);
1801 printLockingError();
1804 exception_error("logoutUser", e
);
1808 // this only searches loaded headlines list, not in CDM
1809 function getRelativePostIds(id
) {
1811 debug("getRelativePostIds: " + id
);
1813 var ids
= new Array();
1814 var container
= document
.getElementById("headlinesList");
1817 var rows
= container
.rows
;
1819 for (var i
= 0; i
< rows
.length
; i
++) {
1820 var r_id
= rows
[i
].id
.replace("RROW-", "");
1823 if (i
> 0) ids
.push(rows
[i
-1].id
.replace("RROW-", ""));
1824 if (i
> 1) ids
.push(rows
[i
-2].id
.replace("RROW-", ""));
1825 if (i
> 2) ids
.push(rows
[i
-3].id
.replace("RROW-", ""));
1827 if (i
< rows
.length
-1) ids
.push(rows
[i
+1].id
.replace("RROW-", ""));
1828 if (i
< rows
.length
-2) ids
.push(rows
[i
+2].id
.replace("RROW-", ""));
1829 if (i
< rows
.length
-3) ids
.push(rows
[i
+3].id
.replace("RROW-", ""));
1839 function openArticleInNewWindow(id
) {
1842 if (!xmlhttp_ready(xmlhttp_rpc
)) {
1843 printLockingError();
1847 debug("openArticleInNewWindow: " + id
);
1849 var query
= "backend.php?op=rpc&subop=getArticleLink&id=" + id
;
1853 xmlhttp_rpc
.open("GET", query
, true);
1854 xmlhttp_rpc
.onreadystatechange
=open_article_callback
;
1855 xmlhttp_rpc
.send(null);
1858 exception_error("openArticleInNewWindow", e
);