]> git.wh0rd.org - tt-rss.git/blobdiff - viewfeed.js
use multiple classes to mark statuses, JS code cleanup
[tt-rss.git] / viewfeed.js
index 3c9b306417a401f65ab4062d7a304284fb896784..597ae53021bc6fa863b5df794827202d58e6420d 100644 (file)
@@ -15,11 +15,16 @@ var post_under_pointer = false;
 
 var last_requested_article = false;
 
+var preload_id_batch = [];
+var preload_timeout_id = false;
+
+var cache_added = [];
+
 function catchup_callback2(transport, callback) {
        try {
-               debug("catchup_callback2 " + transport + ", " + callback);
+               console.log("catchup_callback2 " + transport + ", " + callback);
                notify("");                     
-               all_counters_callback2(transport);
+               handle_rpc_reply(transport);
                if (callback) {
                        setTimeout(callback, 10);       
                }
@@ -28,29 +33,14 @@ function catchup_callback2(transport, callback) {
        }
 }
 
-function clean_feed_selections() {
-       try {
-               var feeds = document.getElementById("feedList").getElementsByTagName("LI");
-
-               for (var i = 0; i < feeds.length; i++) {
-                       if (feeds[i].id && feeds[i].id.match("FEEDR-")) {
-                               feeds[i].className = feeds[i].className.replace("Selected", "");
-                       }                       
-                       if (feeds[i].id && feeds[i].id.match("FCAT-")) {
-                               feeds[i].className = feeds[i].className.replace("Selected", "");
-                       }
-               }
-       } catch (e) {
-               exception_error("clean_feed_selections", e);
-       }
-}
-
 function headlines_callback2(transport, feed_cur_page) {
        try {
 
+               if (!handle_rpc_reply(transport)) return;
+
                loading_set_progress(100);
 
-               debug("headlines_callback2 [page=" + feed_cur_page + "]");
+               console.log("headlines_callback2 [page=" + feed_cur_page + "]");
 
                if (!transport_error_check(transport)) return;
 
@@ -68,49 +58,69 @@ function headlines_callback2(transport, feed_cur_page) {
                        }
                }
 
+               var update_btn = document.forms["main_toolbar_form"].update;
+
+               update_btn.disabled = !(feed_id >= 0 && !is_cat);
+
+               var ll = $('FLL-' + feed_id);
+
+               if (ll && ll.parentNode) 
+                       ll.parentNode.removeChild(ll);
+
                if (!is_cat) {
-                       var feedr = document.getElementById("FEEDR-" + feed_id);
-                       if (feedr && !feedr.className.match("Selected")) {      
-                               feedr.className = feedr.className + "Selected";
+                       var feedr = $("FEEDR-" + feed_id);
+                       if (feedr) {    
+                               feedr.addClassName("Selected");
                        } 
                } else {
-                       var feedr = document.getElementById("FCAT-" + feed_id);
-                       if (feedr && !feedr.className.match("Selected")) {      
-                               feedr.className = feedr.className + "Selected";
+                       var feedr = $("FCAT-" + feed_id);
+                       if (feedr) {    
+                               feedr.addClassName("Selected");
                        } 
                }
-       
-               var f = document.getElementById("headlines-frame");
+
+               var img = $('FIMG-' + feed_id);
+
+               if (img && !is_cat) {
+                       img.src = img.alt;
+               }
+
+               var f = $("headlines-frame");
                try {
                        if (feed_cur_page == 0) { 
-                               debug("resetting headlines scrollTop");
+                               //console.log("resetting headlines scrollTop");
                                f.scrollTop = 0; 
                        }
                } catch (e) { };
        
                if (transport.responseXML) {
-                       var headlines = transport.responseXML.getElementsByTagName("headlines")[0];
-                       var headlines_count_obj = transport.responseXML.getElementsByTagName("headlines-count")[0];
-                       var headlines_unread_obj = transport.responseXML.getElementsByTagName("headlines-unread")[0];
-                       var disable_cache_obj = transport.responseXML.getElementsByTagName("disable-cache")[0];
+                       var response = transport.responseXML;
 
-                       var vgroup_last_feed_obj =  transport.responseXML.getElementsByTagName("vgroup-last-feed")[0];
+                       var headlines = response.getElementsByTagName("headlines")[0];
+                       var headlines_info = response.getElementsByTagName("headlines-info")[0];
 
-                       var headlines_count = headlines_count_obj.getAttribute("value");
-                       var headlines_unread = headlines_unread_obj.getAttribute("value");
-                       var disable_cache = disable_cache_obj.getAttribute("value") != "0";
+                       if (headlines_info)
+                               headlines_info = JSON.parse(headlines_info.firstChild.nodeValue);
+                       else {
+                               console.error("didn't find headlines-info object in response");
+                               return;
+                       }
 
-                       vgroup_last_feed = vgroup_last_feed_obj.getAttribute("value");
+                       var headlines_count = headlines_info.count;
+                       var headlines_unread = headlines_info.unread;
+                       var disable_cache = headlines_info.disable_cache;
+                       
+                       vgroup_last_feed = headlines_info.vgroup_last_feed;
 
-                       if (headlines_count == 0) {
+                       if (parseInt(headlines_count) < getInitParam("default_article_limit")) {
                                _infscroll_disable = 1;
                        } else {
                                _infscroll_disable = 0;
                        }
 
-                       var counters = transport.responseXML.getElementsByTagName("counters")[0];
-                       var articles = transport.responseXML.getElementsByTagName("article");
-                       var runtime_info = transport.responseXML.getElementsByTagName("runtime-info");
+                       var counters = response.getElementsByTagName("counters")[0];
+                       var articles = response.getElementsByTagName("article");
+                       var runtime_info = response.getElementsByTagName("runtime-info");
        
                        if (feed_cur_page == 0) {
                                if (headlines) {
@@ -132,36 +142,32 @@ function headlines_callback2(transport, feed_cur_page) {
                                        }
 
                                } else {
-                                       debug("headlines_callback: returned no data");
+                                       console.warn("headlines_callback: returned no data");
                                f.innerHTML = "<div class='whiteBox'>" + __('Could not update headlines (missing XML data)') + "</div>";
        
                                }
                        } else {
                                if (headlines) {
                                        if (headlines_count > 0) {
-                                               debug("adding some more headlines...");
+                                               console.log("adding some more headlines...");
        
-                                               var c = document.getElementById("headlinesList");
-               
-                                               if (!c) {
-                                                       c = document.getElementById("headlinesInnerContainer");
-                                               }
+                                               c = $("headlinesInnerContainer");
 
                                                var ids = getSelectedArticleIds2();
        
                                                c.innerHTML = c.innerHTML + headlines.firstChild.nodeValue;
 
-                                               debug("restore selected ids: " + ids);
+                                               console.log("restore selected ids: " + ids);
 
                                                for (var i = 0; i < ids.length; i++) {
                                                        markHeadline(ids[i]);
                                                }
 
                                        } else {
-                                               debug("no new headlines received");
+                                               console.log("no new headlines received");
                                        }
                                } else {
-                                       debug("headlines_callback: returned no data");
+                                       console.warn("headlines_callback: returned no data");
                                        notify_error("Error while trying to load more headlines");      
                                }
 
@@ -170,47 +176,38 @@ function headlines_callback2(transport, feed_cur_page) {
                        if (articles) {
                                for (var i = 0; i < articles.length; i++) {
                                        var a_id = articles[i].getAttribute("id");
-                                       debug("found id: " + a_id);
+                                       //console.log("found id: " + a_id);
                                        cache_inject(a_id, articles[i].firstChild.nodeValue);
                                }
                        } else {
-                               debug("no cached articles received");
+                               console.log("no cached articles received");
                        }
-       
-                       if (counters) {
-                               debug("parsing piggybacked counters: " + counters);
-                               parse_counters(counters, false);
-                       } else {
-                               debug("counters container not found in reply, requesting...");
+
+                       if (counters)
+                               parse_counters(counters);
+                       else
                                request_counters();
-                       }
-       
+
                        if (runtime_info) {
-                               debug("parsing runtime info: " + runtime_info[0]);
                                parse_runtime_info(runtime_info[0]);
-                       } else {
-                               debug("counters container not found in reply");
-                       }
+                       } 
        
                } else {
-                       debug("headlines_callback: returned no XML object");
+                       console.warn("headlines_callback: returned no XML object");
                        f.innerHTML = "<div class='whiteBox'>" + __('Could not update headlines (missing XML object)') + "</div>";
                }
        
-               if (typeof correctPNG != 'undefined') {
-                       correctPNG();
-               }
-       
+
                if (_cdm_wd_timeout) window.clearTimeout(_cdm_wd_timeout);
-       
-               if (!document.getElementById("headlinesList") && 
+
+               if (isCdmMode() && 
                                getActiveFeedId() != -3 &&
                                getInitParam("cdm_auto_catchup") == 1) {
-                       debug("starting CDM watchdog");
+                       console.log("starting CDM watchdog");
                        _cdm_wd_timeout = window.setTimeout("cdmWatchdog()", 5000);
                        _cdm_wd_vishist = new Array();
                } else {
-                       debug("not in CDM mode or watchdog disabled");
+                       console.log("not in CDM mode or watchdog disabled");
                }
        
                _feed_cur_page = feed_cur_page;
@@ -227,18 +224,20 @@ function headlines_callback2(transport, feed_cur_page) {
 
 function render_article(article) {
        try {
-               var f = document.getElementById("content-frame");
+               var f = $("content-frame");
                try {
                        f.scrollTop = 0;
                } catch (e) { };
 
-               var fi = document.getElementById("content-insert");
+               var fi = $("content-insert");
 
                try {
                        fi.scrollTop = 0;
                } catch (e) { };
-
+               
                fi.innerHTML = article;
+               
+//             article.evalScripts();          
 
        } catch (e) {
                exception_error("render_article", e);
@@ -249,19 +248,19 @@ function showArticleInHeadlines(id) {
 
        try {
 
-               cleanSelected("headlinesList");
-       
-               var crow = document.getElementById("RROW-" + id);
+               selectArticles("none");
+
+               var crow = $("RROW-" + id);
 
                if (!crow) return;
 
-               var article_is_unread = crow.className.match("Unread");
+               var article_is_unread = crow.hasClassName("Unread");
                        
-               crow.className = crow.className.replace("Unread", "");
+               crow.removeClassName("Unread");
 
-               selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', false);
-       
-               var upd_img_pic = document.getElementById("FUPDPIC-" + id);
+               selectArticles('none');
+
+               var upd_img_pic = $("FUPDPIC-" + id);
 
                var cache_prefix = "";
                                
@@ -280,13 +279,15 @@ function showArticleInHeadlines(id) {
                        //
                }
 
-               if (upd_img_pic && upd_img_pic.src.match("updated.png")) {
+               if (upd_img_pic && (upd_img_pic.src.match("updated.png") || 
+                                       upd_img_pic.src.match("fresh_sign.png"))) {
+
                        upd_img_pic.src = "images/blank_icon.gif";
 
                        cache_invalidate(cache_prefix + getActiveFeedId());
 
                        cache_inject(cache_prefix + getActiveFeedId(),
-                               document.getElementById("headlines-frame").innerHTML,
+                               $("headlines-frame").innerHTML,
                                get_feed_unread(getActiveFeedId()));
 
                } else if (article_is_unread && view_mode == "all_articles") {
@@ -294,7 +295,7 @@ function showArticleInHeadlines(id) {
                        cache_invalidate(cache_prefix + getActiveFeedId());
 
                        cache_inject(cache_prefix + getActiveFeedId(),
-                               document.getElementById("headlines-frame").innerHTML,
+                               $("headlines-frame").innerHTML,
                                get_feed_unread(getActiveFeedId())-1);
 
                } else if (article_is_unread) {
@@ -303,45 +304,61 @@ function showArticleInHeadlines(id) {
 
                markHeadline(id);
 
+               if (article_is_unread)
+                       _force_scheduled_update = true;
+
        } catch (e) {
                exception_error("showArticleInHeadlines", e);
        }
 }
 
-function article_callback2(transport, id, feed_id) {
+function article_callback2(transport, id) {
        try {
-               debug("article_callback2 " + id);
+               console.log("article_callback2 " + id);
+
+               if (!handle_rpc_reply(transport)) return;
 
                if (transport.responseXML) {
 
                        if (!transport_error_check(transport)) return;
 
-                       debug("looking for articles to cache...");
+/*                     var ll = $('LL-' + id);
+                       var content = $('HLC-' + id);
+
+                       if (ll && content) content.removeChild(ll); */
+                       
+                       var upic = $('FUPDPIC-' + id);
+
+                       if (upic) {
+                               upic.src = 'images/blank_icon.gif';
+                       }
+
+                       if (id != last_requested_article) {
+                               console.log("requested article id is out of sequence, aborting");
+                               return;
+                       }
+
+//                     active_post_id = id; 
+
+                       //console.log("looking for articles to cache...");
 
                        var articles = transport.responseXML.getElementsByTagName("article");
 
                        for (var i = 0; i < articles.length; i++) {
                                var a_id = articles[i].getAttribute("id");
 
-                               debug("found id: " + a_id);
+                               //console.log("found id: " + a_id);
 
                                if (a_id == active_post_id) {
-                                       debug("active article, rendering...");                                  
+                                       //console.log("active article, rendering...");                                  
                                        render_article(articles[i].firstChild.nodeValue);
                                }
 
                                cache_inject(a_id, articles[i].firstChild.nodeValue);
                        }
 
-                       if (id != last_requested_article) {
-                               debug("requested article id is out of sequence, aborting");
-                               return;
-                       }
-
-                       active_real_feed_id = feed_id;
-                       active_post_id = id; 
 
-                       showArticleInHeadlines(id);     
+//                     showArticleInHeadlines(id);     
 
                        if (db) {
                                db.execute("UPDATE articles SET unread = 0 WHERE id = ?", [id]);
@@ -350,33 +367,19 @@ function article_callback2(transport, id, feed_id) {
                        var reply = transport.responseXML.firstChild.firstChild;
                
                } else {
-                       debug("article_callback: returned no XML object");
-                       //var f = document.getElementById("content-frame");
+                       console.warn("article_callback: returned no XML object");
+                       //var f = $("content-frame");
                        //f.innerHTML = "<div class='whiteBox'>" + __('Could not display article (missing XML object)') + "</div>";
                }
 
                var date = new Date();
                last_article_view = date.getTime() / 1000;
 
-               if (typeof correctPNG != 'undefined') {
-                       correctPNG();
-               }
-
                if (_reload_feedlist_after_view) {
                        setTimeout('updateFeedList(false, false)', 50);                 
                        _reload_feedlist_after_view = false;
                } else {
-                       if (transport.responseXML) {
-                               var counters = transport.responseXML.getElementsByTagName("counters")[0];
-
-                               if (counters) {
-                                       debug("parsing piggybacked counters: " + counters);
-                                       parse_counters(counters, false);
-                               } else {
-                                       debug("counters container not found in reply, requesting...");
-                                       request_counters();
-                               }
-                       }
+                       request_counters();
                }
 
                notify("");
@@ -385,45 +388,20 @@ function article_callback2(transport, id, feed_id) {
        }
 }
 
-function view_offline(id, feed_id) {
-       try {
-
-               enableHotkeys();
-               showArticleInHeadlines(id);
-
-               if (db) {
-                       db.execute("UPDATE articles SET unread = 0 WHERE id = ?", [id]);
-               }
-
-               render_article("FIXME");
-               update_feedlist_counters();
-
-               return false;
-
-       } catch (e) {
-               exception_error("view_offline", e);
-       }
-}
-
-function view(id, feed_id, skip_history) {
-       
+function view(id) {
        try {
-               debug("loading article: " + id + "/" + feed_id);
+               console.log("loading article: " + id);
 
-               if (offline_mode) return view_offline(id, feed_id);
+               if (offline_mode) return view_offline(id);
 
                var cached_article = cache_find(id);
 
-               debug("cache check result: " + (cached_article != false));
+               console.log("cache check result: " + (cached_article != false));
        
                enableHotkeys();
-       
-               //setActiveFeedId(feed_id);
-
-               var query = "backend.php?op=view&id=" + param_escape(id) +
-                       "&feed=" + param_escape(feed_id);
+               hideAuxDlg();
 
-               var date = new Date();
+               var query = "?op=view&id=" + param_escape(id);
 
                var neighbor_ids = getRelativePostIds(active_post_id);
 
@@ -437,34 +415,23 @@ function view(id, feed_id, skip_history) {
                        }
                }
 
-               debug("additional ids: " + cids_to_request.toString());                 
-
-               /* additional info for piggyback counters */
-
-               if (tagsAreDisplayed()) {
-                       query = query + "&omode=lt";
-               } else {
-                       query = query + "&omode=flc";
-               }
-
-               var date = new Date();
-               var timestamp = Math.round(date.getTime() / 1000);
-               query = query + "&ts=" + timestamp;
-
+               console.log("additional ids: " + cids_to_request.toString());                   
+       
                query = query + "&cids=" + cids_to_request.toString();
 
-               var crow = document.getElementById("RROW-" + id);
-               var article_is_unread = crow.className.match("Unread");
-
-               if (!async_counters_work) {
-                       query = query + "&csync=true";
-               }
+               var crow = $("RROW-" + id);
+               var article_is_unread = crow.hasClassName("Unread");
 
+               active_post_id = id;
                showArticleInHeadlines(id);
 
                if (!cached_article) {
 
-                       notify_progress("Loading, please wait...", true);
+                       var upic = $('FUPDPIC-' + id);
+
+                       if (upic) {     
+                               upic.src = getInitParam("sign_progress");
+                       }
 
                } else if (cached_article && article_is_unread) {
 
@@ -483,9 +450,10 @@ function view(id, feed_id, skip_history) {
 
                last_requested_article = id;
 
-               new Ajax.Request(query, {
+               new Ajax.Request("backend.php", {
+                       parameters: query,
                        onComplete: function(transport) { 
-                               article_callback2(transport, id, feed_id); 
+                               article_callback2(transport, id); 
                        } });
 
                return false;
@@ -507,7 +475,7 @@ function tMark_afh_off(effect) {
        try {
                var elem = effect.effects[0].element;
 
-               debug("tMark_afh_off : " + elem.id);
+               //console.log("tMark_afh_off : " + elem.id);
 
                if (elem) {
                        elem.src = elem.src.replace("mark_set", "mark_unset");
@@ -524,7 +492,7 @@ function tPub_afh_off(effect) {
        try {
                var elem = effect.effects[0].element;
 
-               debug("tPub_afh_off : " + elem.id);
+               //console.log("tPub_afh_off : " + elem.id);
 
                if (elem) {
                        elem.src = elem.src.replace("pub_set", "pub_unset");
@@ -537,30 +505,17 @@ function tPub_afh_off(effect) {
        }
 }
 
-function toggleMark(id, client_only, no_effects) {
-
+function toggleMark(id, client_only) {
        try {
-
-               var query = "backend.php?op=rpc&id=" + id + "&subop=mark";
-       
-               query = query + "&afid=" + getActiveFeedId();
+               var query = "?op=rpc&id=" + id + "&subop=mark";
        
-               if (tagsAreDisplayed()) {
-                       query = query + "&omode=tl";
-               } else {
-                       query = query + "&omode=flc";
-               }
-       
-               var mark_img = document.getElementById("FMPIC-" + id);
-
-               if (!mark_img) return;
+               var img = $("FMPIC-" + id);
 
-               var vfeedu = document.getElementById("FEEDU--1");
-               var crow = document.getElementById("RROW-" + id);
+               if (!img) return;
        
-               if (mark_img.src.match("mark_unset")) {
-                       mark_img.src = mark_img.src.replace("mark_unset", "mark_set");
-                       mark_img.alt = __("Unstar article");
+               if (img.src.match("mark_unset")) {
+                       img.src = img.src.replace("mark_unset", "mark_set");
+                       img.alt = __("Unstar article");
                        query = query + "&mark=1";
 
                        if (db) {
@@ -568,31 +523,23 @@ function toggleMark(id, client_only, no_effects) {
                        }
 
                } else {
-                       //mark_img.src = "images/mark_unset.png";
-                       mark_img.alt = __("Please wait...");
+                       img.src = img.src.replace("mark_set", "mark_unset");
+                       img.alt = __("Star article");
                        query = query + "&mark=0";
-       
-                       if (document.getElementById("headlinesList") && !no_effects) {
-                               Effect.Puff(mark_img, {duration : 0.25, afterFinish: tMark_afh_off});
-                       } else { 
-                               mark_img.src = mark_img.src.replace("mark_set", "mark_unset");
-                               mark_img.alt = __("Star article");
-                       }
 
                        if (db) {
                                db.execute("UPDATE articles SET marked = 0 WHERE id = ?", [id]);
                        }
-
                }
 
-               if (!client_only) {
-                       debug(query);
+               update_local_feedlist_counters();
 
-                       new Ajax.Request(query, {
+               if (!client_only) {
+                       new Ajax.Request("backend.php", {
+                               parameters: query,
                                onComplete: function(transport) { 
-                                       all_counters_callback2(transport); 
+                                       handle_rpc_reply(transport); 
                                } });
-
                }
 
        } catch (e) {
@@ -600,107 +547,77 @@ function toggleMark(id, client_only, no_effects) {
        }
 }
 
-function togglePub(id, client_only, no_effects) {
-
+function togglePub(id, client_only, no_effects, note) {
        try {
-
-               var query = "backend.php?op=rpc&id=" + id + "&subop=publ";
+               var query = "?op=rpc&id=" + id + "&subop=publ";
        
-               query = query + "&afid=" + getActiveFeedId();
-       
-               if (tagsAreDisplayed()) {
-                       query = query + "&omode=tl";
+               if (note != undefined) {
+                       query = query + "&note=" + param_escape(note);
                } else {
-                       query = query + "&omode=flc";
+                       query = query + "&note=undefined";
                }
-       
-               var mark_img = document.getElementById("FPPIC-" + id);
 
-               if (!mark_img) return;
+               var img = $("FPPIC-" + id);
 
-               var vfeedu = document.getElementById("FEEDU--2");
-               var crow = document.getElementById("RROW-" + id);
+               if (!img) return;
        
-               if (mark_img.src.match("pub_unset")) {
-                       mark_img.src = mark_img.src.replace("pub_unset", "pub_set");
-                       mark_img.alt = __("Unpublish article");
+               if (img.src.match("pub_unset") || note != undefined) {
+                       img.src = img.src.replace("pub_unset", "pub_set");
+                       img.alt = __("Unpublish article");
                        query = query + "&pub=1";
-       
 
                } else {
-                       //mark_img.src = "images/pub_unset.png";
-                       mark_img.alt = __("Please wait...");
+                       img.src = img.src.replace("pub_set", "pub_unset");
+                       img.alt = __("Publish article");
+
                        query = query + "&pub=0";
-       
-                       if (document.getElementById("headlinesList") && !no_effects) {
-                               Effect.Puff(mark_img, {duration : 0.25, afterFinish: tPub_afh_off});
-                       } else { 
-                               mark_img.src = mark_img.src.replace("pub_set", "pub_unset");
-                               mark_img.alt = __("Publish article");
-                       }
                }
 
                if (!client_only) {
-                       new Ajax.Request(query, {
+                       new Ajax.Request("backend.php", {
+                               parameters: query,
                                onComplete: function(transport) { 
-                                       all_counters_callback2(transport); 
+                                       handle_rpc_reply(transport);
+               
+                                       var note = transport.responseXML.getElementsByTagName("note")[0];
+               
+                                       if (note) {
+                                               var note_id = note.getAttribute("id");
+                                               var note_size = note.getAttribute("size");
+                                               var note_content = note.firstChild.nodeValue;
+               
+                                               var container = $('POSTNOTE-' + note_id);
+               
+                                               cache_invalidate(note_id);
+               
+                                               if (container) {
+                                                       if (note_size == "0") {
+                                                               Element.hide(container);
+                                                       } else {
+                                                               container.innerHTML = note_content;
+                                                               Element.show(container);
+                                                       }
+                                               }
+                                       }       
+
                                } });
                }
 
        } catch (e) {
-
                exception_error("togglePub", e);
        }
 }
 
-function correctHeadlinesOffset(id) {
-       
-       try {
-
-               var hlist = document.getElementById("headlinesList");
-               var container = document.getElementById("headlinesInnerContainer");
-               var row = document.getElementById("RROW-" + id);
-       
-               var viewport = container.offsetHeight;
-       
-               var rel_offset_top = row.offsetTop - container.scrollTop;
-               var rel_offset_bottom = row.offsetTop + row.offsetHeight - container.scrollTop;
-       
-               debug("Rtop: " + rel_offset_top + " Rbtm: " + rel_offset_bottom);
-               debug("Vport: " + viewport);
-
-               if (rel_offset_top <= 0 || rel_offset_top > viewport) {
-                       container.scrollTop = row.offsetTop;
-               } else if (rel_offset_bottom > viewport) {
-
-                       /* doesn't properly work with Opera in some cases because
-                               Opera fucks up element scrolling */
-
-                       container.scrollTop = row.offsetTop + row.offsetHeight - viewport;              
-               } 
-
-       } catch (e) {
-               exception_error("correctHeadlinesOffset", e);
-       }
-
-}
-
 function moveToPost(mode) {
 
        try {
 
-               var rows;
+               var rows = getVisibleArticleIds();
 
-               if (isCdmMode()) {
-                       rows = cdmGetVisibleArticles();
-               } else {
-                       rows = getVisibleHeadlineIds();
-               }
-               
                var prev_id = false;
                var next_id = false;
                
-               if (!document.getElementById('RROW-' + active_post_id)) {
+               if (!$('RROW-' + active_post_id)) {
                        active_post_id = false;
                }
                
@@ -720,12 +637,8 @@ function moveToPost(mode) {
                        if (next_id) {
                                if (isCdmMode()) {
        
-                                       if (!cdmArticleIsActuallyVisible(next_id)) {
-                                               cdmScrollToArticleId(next_id);
-                                       }
-                                       cdmSelectArticles("none");
-                                       toggleUnread(next_id, 0, true);
-                                       toggleSelected(next_id);
+                                       cdmExpandArticle(next_id);
+                                       cdmScrollToArticleId(next_id);
 
                                } else {
                                        correctHeadlinesOffset(next_id);
@@ -737,10 +650,8 @@ function moveToPost(mode) {
                if (mode == "prev") {
                        if (prev_id) {
                                if (isCdmMode()) {
+                                       cdmExpandArticle(prev_id);
                                        cdmScrollToArticleId(prev_id);
-                                       cdmSelectArticles("none");
-                                       toggleUnread(prev_id, 0, true);
-                                       toggleSelected(prev_id);
                                } else {
                                        correctHeadlinesOffset(prev_id);
                                        view(prev_id, getActiveFeedId());
@@ -756,29 +667,17 @@ function moveToPost(mode) {
 function toggleSelected(id) {
        try {
        
-               var cb = document.getElementById("RCHK-" + id);
+               var cb = $("RCHK-" + id);
+               var row = $("RROW-" + id);
 
-               var row = document.getElementById("RROW-" + id);
                if (row) {
-                       var nc = row.className;
-                       
-                       if (!nc.match("Selected")) {
-                               nc = nc + "Selected";
-                               if (cb) {
-                                       cb.checked = true;
-                               }
-
-                               // In CDM basically last selected article == active article
-                               if (isCdmMode()) active_post_id = id;
+                       if (row.hasClassName('Selected')) {
+                               row.removeClassName('Selected');
+                               if (cb) cb.checked = false;
                        } else {
-                               nc = nc.replace("Selected", "");
-                               if (cb) {
-                                       cb.checked = false;
-                               }
-
+                               row.addClassName('Selected');
+                               if (cb) cb.checked = true;
                        }
-
-                       row.className = nc;
                }
        } catch (e) {
                exception_error("toggleSelected", e);
@@ -799,26 +698,11 @@ function toggleUnread_afh(effect) {
 function toggleUnread(id, cmode, effect) {
        try {
        
-               var row = document.getElementById("RROW-" + id);
+               var row = $("RROW-" + id);
                if (row) {
-                       var nc = row.className;
-                       var is_selected = row.className.match("Selected");
-                       nc = nc.replace("Unread", "");
-                       nc = nc.replace("Selected", "");
-
-                       // since we are removing selection from the object, uncheck
-                       // corresponding checkbox
-
-                       var cb = document.getElementById("RCHK-" + id);
-                       if (cb) {
-                               cb.checked = false;
-                       }
-
-                       // NOTE: I'm not sure that resetting selection here is a feature -fox
-
                        if (cmode == undefined || cmode == 2) {
-                               if (row.className.match("Unread")) {
-                                       row.className = nc;
+                               if (row.hasClassName("Unread")) {
+                                       row.removeClassName("Unread");
 
                                        if (effect) {
                                                new Effect.Highlight(row, {duration: 1, startcolor: "#fff7d5",
@@ -827,38 +711,56 @@ function toggleUnread(id, cmode, effect) {
                                        } 
 
                                } else {
-                                       row.className = nc + "Unread";
+                                       row.addClassName("Unread");
+                               }
+
+                               if (db) {
+                                       db.execute("UPDATE articles SET unread = not unread "+
+                                               "WHERE id = ?", [id]);
                                }
+
                        } else if (cmode == 0) {
-                               row.className = nc;
+
+                               row.removeClassName("Unread");
 
                                if (effect) {
                                        new Effect.Highlight(row, {duration: 1, startcolor: "#fff7d5",
                                                afterFinish: toggleUnread_afh,
                                                queue: { position:'end', scope: 'TMRQ-' + id, limit: 1 } } );
                                } 
+
+                               if (db) {
+                                       db.execute("UPDATE articles SET unread = 0 "+
+                                               "WHERE id = ?", [id]);
+                               }
+
                        } else if (cmode == 1) {
-                               row.className = nc + "Unread";
+                               row.addClassName("Unread");
+
+                               if (db) {
+                                       db.execute("UPDATE articles SET unread = 1 "+
+                                               "WHERE id = ?", [id]);
+                               }
+
                        }
 
-                       // Disable unmarking as selected for the time being (16.05.08) -fox
-                       if (is_selected) row.className = row.className + "Selected";
+                       update_local_feedlist_counters();
 
                        if (cmode == undefined) cmode = 2;
 
-                       var query = "backend.php?op=rpc&subop=catchupSelected&ids=" +
-                               param_escape(id) + "&cmode=" + param_escape(cmode);
+                       var query = "?op=rpc&subop=catchupSelected" +
+                               "&cmode=" + param_escape(cmode) + "&ids=" + param_escape(id);
 
 //                     notify_progress("Loading, please wait...");
 
-                       new Ajax.Request(query, {
+                       new Ajax.Request("backend.php", {
+                               parameters: query,
                                onComplete: function(transport) { 
-                                       all_counters_callback2(transport); 
+                                       handle_rpc_reply(transport); 
                                } });
 
                }
 
-
        } catch (e) {
                exception_error("toggleUnread", e);
        }
@@ -878,17 +780,20 @@ function selectionRemoveLabel(id) {
 
 //             if (ok) {
 
-                       var query = "backend.php?op=rpc&subop=removeFromLabel&ids=" +
+                       var query = "?op=rpc&subop=removeFromLabel&ids=" +
                                param_escape(ids.toString()) + "&lid=" + param_escape(id);
 
+                       console.log(query);
+
 //                     notify_progress("Loading, please wait...");
 
                        cache_invalidate("F:" + (-11 - id));
 
-                       new Ajax.Request(query, {
+                       new Ajax.Request("backend.php", {
+                               parameters: query,
                                onComplete: function(transport) { 
                                        show_labels_in_headlines(transport);
-                                       all_counters_callback2(transport);
+                                       handle_rpc_reply(transport);
                                } });
 
 //             }
@@ -915,15 +820,18 @@ function selectionAssignLabel(id) {
 
                        cache_invalidate("F:" + (-11 - id));
 
-                       var query = "backend.php?op=rpc&subop=assignToLabel&ids=" +
+                       var query = "?op=rpc&subop=assignToLabel&ids=" +
                                param_escape(ids.toString()) + "&lid=" + param_escape(id);
 
+                       console.log(query);
+
 //                     notify_progress("Loading, please wait...");
 
-                       new Ajax.Request(query, {
+                       new Ajax.Request("backend.php", {
+                               parameters: query,
                                onComplete: function(transport) { 
                                        show_labels_in_headlines(transport);
-                                       all_counters_callback2(transport);
+                                       handle_rpc_reply(transport);
                                } });
 
 //             }
@@ -934,15 +842,9 @@ function selectionAssignLabel(id) {
        }
 }
 
-function selectionToggleUnread(cdm_mode, set_state, callback_func, no_error) {
+function selectionToggleUnread(set_state, callback_func, no_error) {
        try {
-               var rows;
-
-               if (cdm_mode) {
-                       rows = cdmGetSelectedArticles();
-               } else {        
-                       rows = getSelectedTableRowIds("headlinesList", "RROW", "RCHK");
-               }
+               var rows = getSelectedArticleIds2();
 
                if (rows.length == 0 && !no_error) {
                        alert(__("No articles are selected."));
@@ -950,32 +852,44 @@ function selectionToggleUnread(cdm_mode, set_state, callback_func, no_error) {
                }
 
                for (i = 0; i < rows.length; i++) {
-                       var row = document.getElementById("RROW-" + rows[i]);
+                       var row = $("RROW-" + rows[i]);
                        if (row) {
-                               var nc = row.className;
-                               nc = nc.replace("Unread", "");
-                               nc = nc.replace("Selected", "");
-
                                if (set_state == undefined) {
-                                       if (row.className.match("Unread")) {
-                                               row.className = nc + "Selected";
+                                       if (row.hasClassName("Unread")) {
+                                               row.removeClassName("Unread");
                                        } else {
-                                               row.className = nc + "UnreadSelected";
+                                               row.addClassName("Unread");
+                                       }
+                                       if (db) {
+                                               db.execute("UPDATE articles SET unread = NOT unread WHERE id = ?", 
+                                                       [rows[i]]);
                                        }
                                }
 
                                if (set_state == false) {
-                                       row.className = nc + "Selected";
+                                       row.removeClassName("Unread");
+
+                                       if (db) {
+                                               db.execute("UPDATE articles SET unread = 0 WHERE id = ?", 
+                                                       [rows[i]]);
+                                       }
                                }
 
                                if (set_state == true) {
-                                       row.className = nc + "UnreadSelected";
+                                       row.addClassName("Unread");
+
+                                       if (db) {
+                                               db.execute("UPDATE articles SET unread = 1 WHERE id = ?", 
+                                                       [rows[i]]);
+                                       }
                                }
                        }
                }
 
                if (rows.length > 0) {
 
+                       update_local_feedlist_counters();
+
                        var cmode = "";
 
                        if (set_state == undefined) {
@@ -986,12 +900,13 @@ function selectionToggleUnread(cdm_mode, set_state, callback_func, no_error) {
                                cmode = "0";
                        }
 
-                       var query = "backend.php?op=rpc&subop=catchupSelected&ids=" +
-                               param_escape(rows.toString()) + "&cmode=" + cmode;
+                       var query = "?op=rpc&subop=catchupSelected" +
+                               "&cmode=" + cmode + "&ids=" + param_escape(rows.toString()); 
 
                        notify_progress("Loading, please wait...");
 
-                       new Ajax.Request(query, {
+                       new Ajax.Request("backend.php", {
+                               parameters: query,
                                onComplete: function(transport) { 
                                        catchup_callback2(transport, callback_func); 
                                } });
@@ -1003,17 +918,11 @@ function selectionToggleUnread(cdm_mode, set_state, callback_func, no_error) {
        }
 }
 
-function selectionToggleMarked(cdm_mode) {
+function selectionToggleMarked() {
        try {
        
-               var rows;
+               var rows = getSelectedArticleIds2();
                
-               if (cdm_mode) {
-                       rows = cdmGetSelectedArticles();
-               } else {        
-                       rows = getSelectedTableRowIds("headlinesList", "RROW", "RCHK");
-               }       
-
                if (rows.length == 0) {
                        alert(__("No articles are selected."));
                        return;
@@ -1023,24 +932,17 @@ function selectionToggleMarked(cdm_mode) {
                        toggleMark(rows[i], true, true);
                }
 
+               update_local_feedlist_counters();
+
                if (rows.length > 0) {
 
-                       var query = "backend.php?op=rpc&subop=markSelected&ids=" +
+                       var query = "?op=rpc&subop=markSelected&ids=" +
                                param_escape(rows.toString()) + "&cmode=2";
 
-                       query = query + "&afid=" + getActiveFeedId();
-
-/*                     if (tagsAreDisplayed()) {
-                               query = query + "&omode=tl";
-                       } else {
-                               query = query + "&omode=flc";
-                       } */
-
-                       query = query + "&omode=lc";
-
-                       new Ajax.Request(query, {
+                       new Ajax.Request("backend.php", {
+                               parameters: query,
                                onComplete: function(transport) { 
-                                       all_counters_callback2(transport); 
+                                       handle_rpc_reply(transport); 
                                } });
 
                }
@@ -1050,16 +952,10 @@ function selectionToggleMarked(cdm_mode) {
        }
 }
 
-function selectionTogglePublished(cdm_mode) {
+function selectionTogglePublished() {
        try {
        
-               var rows;
-               
-               if (cdm_mode) {
-                       rows = cdmGetSelectedArticles();
-               } else {        
-                       rows = getSelectedTableRowIds("headlinesList", "RROW", "RCHK");
-               }       
+               var rows = getSelectedArticleIds2();
 
                if (rows.length == 0) {
                        alert(__("No articles are selected."));
@@ -1072,22 +968,13 @@ function selectionTogglePublished(cdm_mode) {
 
                if (rows.length > 0) {
 
-                       var query = "backend.php?op=rpc&subop=publishSelected&ids=" +
+                       var query = "?op=rpc&subop=publishSelected&ids=" +
                                param_escape(rows.toString()) + "&cmode=2";
 
-                       query = query + "&afid=" + getActiveFeedId();
-
-/*                     if (tagsAreDisplayed()) {
-                               query = query + "&omode=tl";
-                       } else {
-                               query = query + "&omode=flc";
-                       } */
-
-                       query = query + "&omode=lc";
-
-                       new Ajax.Request(query, {
+                       new Ajax.Request("backend.php", {
+                               parameters: query,
                                onComplete: function(transport) { 
-                                       all_counters_callback2(transport); 
+                                       handle_rpc_reply(transport); 
                                } });
 
                }
@@ -1097,84 +984,69 @@ function selectionTogglePublished(cdm_mode) {
        }
 }
 
-function cdmGetSelectedArticles() {
-       var sel_articles = new Array();
-       var container = document.getElementById("headlinesInnerContainer");
+function getSelectedArticleIds2() {
 
-       for (i = 0; i < container.childNodes.length; i++) {
-               var child = container.childNodes[i];
+       var rv = [];
 
-               if (child.id && child.id.match("RROW-") && child.className.match("Selected")) {
-                       var c_id = child.id.replace("RROW-", "");
-                       sel_articles.push(c_id);
-               }
-       }
+       $$("#headlinesInnerContainer > div[id*=RROW][class*=Selected]").each(
+               function(child) {
+                       rv.push(child.id.replace("RROW-", ""));
+               });
 
-       return sel_articles;
+       return rv;
 }
 
-function cdmGetVisibleArticles() {
-       var sel_articles = new Array();
-       var container = document.getElementById("headlinesInnerContainer");
-
-       if (!container) return sel_articles;
-
-       for (i = 0; i < container.childNodes.length; i++) {
-               var child = container.childNodes[i];
-
-               if (child.id && child.id.match("RROW-")) {
-                       var c_id = child.id.replace("RROW-", "");
-                       sel_articles.push(c_id);
-               }
-       }
-
-       return sel_articles;
-}
+function getLoadedArticleIds() {
+       var rv = [];
 
-function cdmGetUnreadArticles() {
-       var sel_articles = new Array();
-       var container = document.getElementById("headlinesInnerContainer");
+       var children = $$("#headlinesInnerContainer > div[id*=RROW-]");
 
-       for (i = 0; i < container.childNodes.length; i++) {
-               var child = container.childNodes[i];
+       children.each(function(child) {
+                       rv.push(child.id.replace("RROW-", ""));
+               });
 
-               if (child.id && child.id.match("RROW-") && child.className.match("Unread")) {
-                       var c_id = child.id.replace("RROW-", "");
-                       sel_articles.push(c_id);
-               }
-       }
+       return rv;
 
-       return sel_articles;
 }
 
+// mode = all,none,unread,invert
+function selectArticles(mode) {
+       try {
 
-// mode = all,none,unread
-function cdmSelectArticles(mode) {
-       var container = document.getElementById("headlinesInnerContainer");
-
-       for (i = 0; i < container.childNodes.length; i++) {
-               var child = container.childNodes[i];
-
-               if (child.id && child.id.match("RROW-")) {
-                       var aid = child.id.replace("RROW-", "");
+               var children = $$("#headlinesInnerContainer > div[id*=RROW]");
 
-                       var cb = document.getElementById("RCHK-" + aid);
+               children.each(function(child) {
+                       var id = child.id.replace("RROW-", "");
+                       var cb = $("RCHK-" + id);
 
                        if (mode == "all") {
-                               if (!child.className.match("Selected")) {
-                                       child.className = child.className + "Selected";
+                               child.addClassName("Selected");
+                               cb.checked = true;
+                       } else if (mode == "unread") {
+                               if (child.hasClassName("Unread")) {
+                                       child.addClassName("Selected");
                                        cb.checked = true;
+                               } else {
+                                       child.removeClassName("Selected");
+                                       cb.checked = false;
                                }
-                       } else if (mode == "unread") {
-                               if (child.className.match("Unread") && !child.className.match("Selected")) {
-                                       child.className = child.className + "Selected";
+                       } else if (mode == "invert") {
+                               if (child.hasClassName("Selected")) {
+                                       child.removeClassName("Selected");
+                                       cb.checked = false;
+                               } else {
+                                       child.addClassName("Selected");
                                        cb.checked = true;
                                }
+
                        } else {
-                               child.className = child.className.replace("Selected", "");
+                               child.removeClassName("Selected");
                                cb.checked = false;
                        }
-               }               
+               });
+
+       } catch (e) {
+               exception_error("selectArticles", e);
        }
 }
 
@@ -1190,38 +1062,31 @@ function catchupPage() {
                return;
        }
 
-       if (document.getElementById("headlinesList")) {
-               selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', true, 'Unread', true);
-               selectionToggleUnread(false, false, 'viewCurrentFeed()', true);
-               selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', false);
-       } else {
-               cdmSelectArticles('all');
-               selectionToggleUnread(true, false, 'viewCurrentFeed()', true)
-               cdmSelectArticles('none');
-       }
+       selectArticles('all');
+       selectionToggleUnread(false, 'viewCurrentFeed()', true)
+       selectArticles('none');
 }
 
-function catchupSelection() {
+function deleteSelection() {
 
        try {
-
-               var rows;
-       
-               if (document.getElementById("headlinesList")) {
-                       rows = getSelectedTableRowIds("headlinesList", "RROW", "RCHK");
-               } else {        
-                       rows = cdmGetSelectedArticles();
-               }
        
+               var rows = getSelectedArticleIds2();
+
                if (rows.length == 0) {
                        alert(__("No articles are selected."));
                        return;
                }
        
-       
                var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
-               
-               var str = __("Mark %d selected articles in %s as read?");
+               var str;
+               var op;
+       
+               if (getActiveFeedId() != 0) {
+                       str = __("Delete %d selected articles in %s?");
+               } else {
+                       str = __("Delete %d selected articles?");
+               }
        
                str = str.replace("%d", rows.length);
                str = str.replace("%s", fn);
@@ -1229,57 +1094,111 @@ function catchupSelection() {
                if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
                        return;
                }
-       
-               if (document.getElementById("headlinesList")) {
-                       selectionToggleUnread(false, false, 'viewCurrentFeed()', true);
-       //              selectTableRowsByIdPrefix('headlinesList', 'RROW-', 'RCHK-', false);
-               } else {
-                       selectionToggleUnread(true, false, 'viewCurrentFeed()', true)
-       //              cdmSelectArticles('none');
-               }
 
-       } catch (e) {
-               exception_error("catchupSelection", e);
-       }
-}
+               query = "?op=rpc&subop=delete&ids=" + param_escape(rows);
 
-function editArticleTags(id, feed_id, cdm_enabled) {
-       displayDlg('editArticleTags', id);
+               console.log(query);
+
+               new Ajax.Request("backend.php", {
+                       parameters: query,
+                       onComplete: function(transport) {
+                                       viewCurrentFeed();
+                               } });
+
+       } catch (e) {
+               exception_error("deleteSelection", e);
+       }
 }
 
+function archiveSelection() {
 
-function tag_saved_callback(transport) {
        try {
-               debug("in tag_saved_callback");
 
-               closeInfoBox();
-               notify("");
+               var rows = getSelectedArticleIds2();
 
-               if (tagsAreDisplayed()) {
-                       _reload_feedlist_after_view = true;
+               if (rows.length == 0) {
+                       alert(__("No articles are selected."));
+                       return;
+               }
+       
+               var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
+               var str;
+               var op;
+       
+               if (getActiveFeedId() != 0) {
+                       str = __("Archive %d selected articles in %s?");
+                       op = "archive";
+               } else {
+                       str = __("Move %d archived articles back?");
+                       op = "unarchive";
+               }
+       
+               str = str.replace("%d", rows.length);
+               str = str.replace("%s", fn);
+       
+               if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
+                       return;
                }
 
+               query = "?op=rpc&subop="+op+"&ids=" + param_escape(rows);
 
-               if (transport.responseXML) {
-                       var tags_str = transport.responseXML.getElementsByTagName("tags-str")[0];
-                       
-                       if (tags_str) {
-                               var id = tags_str.getAttribute("id");
+               console.log(query);
 
-                               if (id) {
-                                       var tags = document.getElementById("ATSTR-" + id);
-                                       if (tags) {
-                                               tags.innerHTML = tags_str.firstChild.nodeValue;
-                                       }
-                               }
-                       }
+               for (var i = 0; i < rows.length; i++) {
+                       cache_invalidate(rows[i]);
+               }
+
+               new Ajax.Request("backend.php", {
+                       parameters: query,
+                       onComplete: function(transport) {
+                                       viewCurrentFeed();
+                               } });
+
+       } catch (e) {
+               exception_error("archiveSelection", e);
+       }
+}
+
+function catchupSelection() {
+
+       try {
+
+               var rows = getSelectedArticleIds2();
+
+               if (rows.length == 0) {
+                       alert(__("No articles are selected."));
+                       return;
+               }
+       
+               var fn = getFeedName(getActiveFeedId(), activeFeedIsCat());
+               
+               var str = __("Mark %d selected articles in %s as read?");
+       
+               str = str.replace("%d", rows.length);
+               str = str.replace("%s", fn);
+       
+               if (getInitParam("confirm_feed_catchup") == 1 && !confirm(str)) {
+                       return;
                }
+       
+               selectionToggleUnread(false, 'viewCurrentFeed()', true)
 
        } catch (e) {
-               exception_error("tag_saved_callback", e);
+               exception_error("catchupSelection", e);
        }
 }
 
+function editArticleTags(id, feed_id, cdm_enabled) {
+       displayDlg('editArticleTags', id,
+                          function () {
+                                       $("tags_str").focus();
+
+                                       new Ajax.Autocompleter('tags_str', 'tags_choices',
+                                          "backend.php?op=rpc&subop=completeTags",
+                                          { tokens: ',', paramName: "search" });
+                          });
+}
+
 function editTagsSave() {
 
        notify_progress("Saving article tags...");
@@ -1288,15 +1207,44 @@ function editTagsSave() {
 
        var query = Form.serialize("tag_edit_form");
 
-       query = "backend.php?op=rpc&subop=setArticleTags&" + query;
+       query = "?op=rpc&subop=setArticleTags&" + query;
 
-       debug(query);
+       //console.log(query);
 
-       new Ajax.Request(query, {
+       new Ajax.Request("backend.php", {
+               parameters: query,
                onComplete: function(transport) {
-                               tag_saved_callback(transport);
+                               try {
+                                       //console.log("tags saved...");
+                       
+                                       closeInfoBox();
+                                       notify("");
+                       
+                                       if (tagsAreDisplayed()) {
+                                               _reload_feedlist_after_view = true;
+                                       }                       
+                       
+                                       if (transport.responseXML) {
+                                               var tags_str = transport.responseXML.getElementsByTagName("tags-str")[0];
+                                               
+                                               if (tags_str) {
+                                                       var id = tags_str.getAttribute("id");
+                       
+                                                       if (id) {
+                                                               var tags = $("ATSTR-" + id);
+                                                               if (tags) {
+                                                                       tags.innerHTML = tags_str.firstChild.nodeValue;
+                                                               }
+
+                                                               cache_invalidate(id);
+                                                       }
+                                               }
+                                       }
+                       
+                               } catch (e) {
+                                       exception_error("editTagsSave", e);
+                               }
                        } });
-
 }
 
 function editTagsInsert() {
@@ -1324,64 +1272,10 @@ function editTagsInsert() {
        }
 }
 
-function cdmScrollViewport(where) {
-       debug("cdmScrollViewport: " + where);
-
-       var ctr = document.getElementById("headlinesInnerContainer");
-
-       if (!ctr) return;
-
-       if (where == "bottom") {
-               ctr.scrollTop = ctr.scrollHeight;
-       } else {
-               ctr.scrollTop = where;
-       }
-}
-
-function cdmArticleIsBelowViewport(id) {
-       try {
-               var ctr = document.getElementById("headlinesInnerContainer");
-               var e = document.getElementById("RROW-" + id);
-
-               if (!e || !ctr) return;
-
-               // article starts below viewport
-
-               if (ctr.scrollTop < e.offsetTop) {
-                       return true;
-               } else {        
-                       return false;
-               }
-
-       } catch (e) {
-               exception_error("cdmArticleIsVisible", e);
-       }
-}
-
-function cdmArticleIsAboveViewport(id) {
-       try {
-               var ctr = document.getElementById("headlinesInnerContainer");
-               var e = document.getElementById("RROW-" + id);
-
-               if (!e || !ctr) return;
-
-               // article starts above viewport
-
-               if (ctr.scrollTop > e.offsetTop + e.offsetHeight) {
-                       return true;
-               } else {        
-                       return false;
-               }
-
-       } catch (e) {
-               exception_error("cdmArticleIsVisible", e);
-       }
-}
-
 function cdmScrollToArticleId(id) {
        try {
-               var ctr = document.getElementById("headlinesInnerContainer");
-               var e = document.getElementById("RROW-" + id);
+               var ctr = $("headlinesInnerContainer");
+               var e = $("RROW-" + id);
 
                if (!e || !ctr) return;
 
@@ -1392,41 +1286,11 @@ function cdmScrollToArticleId(id) {
        }
 }
 
-function cdmArticleIsActuallyVisible(id) {
-       try {
-               var ctr = document.getElementById("headlinesInnerContainer");
-               var e = document.getElementById("RROW-" + id);
-
-               if (!e || !ctr) return;
-
-               // article fits in viewport OR article is longer than viewport and
-               // its bottom is visible
-
-               if (ctr.scrollTop <= e.offsetTop && e.offsetTop + e.offsetHeight <=
-                               ctr.scrollTop + ctr.offsetHeight) {
-
-                       return true;
-               
-               } else if (e.offsetHeight > ctr.offsetHeight &&
-                               e.offsetTop + e.offsetHeight >= ctr.scrollTop &&
-                               e.offsetTop + e.offsetHeight <= ctr.scrollTop + ctr.offsetHeight) {
-
-                       return true;
-
-               }
-
-               return false;
-
-       } catch (e) {
-               exception_error("cdmArticleIsVisible", e);
-       }
-}
-
 function cdmWatchdog() {
 
        try {
 
-               var ctr = document.getElementById("headlinesInnerContainer");
+               var ctr = $("headlinesInnerContainer");
 
                if (!ctr) return;
 
@@ -1435,7 +1299,7 @@ function cdmWatchdog() {
                var e = ctr.firstChild;
 
                while (e) {
-                       if (e.className && e.className == "cdmArticleUnread" && e.id &&
+                       if (e.className && e.hasClassName("Unread") && e.id &&
                                        e.id.match("RROW-")) {
 
                                // article fits in viewport OR article is longer than viewport and
@@ -1444,7 +1308,7 @@ function cdmWatchdog() {
                                if (ctr.scrollTop <= e.offsetTop && e.offsetTop + e.offsetHeight <=
                                                ctr.scrollTop + ctr.offsetHeight) {
 
-//                                     debug(e.id + " is visible " + e.offsetTop + "." + 
+//                                     console.log(e.id + " is visible " + e.offsetTop + "." + 
 //                                             (e.offsetTop + e.offsetHeight) + " vs " + ctr.scrollTop + "." +
 //                                             (ctr.scrollTop + ctr.offsetHeight));
 
@@ -1472,23 +1336,24 @@ function cdmWatchdog() {
                        e = e.nextSibling;
                }
 
-               debug("cdmWatchdog, ids= " + ids.toString());
+               console.log("cdmWatchdog, ids= " + ids.toString());
 
                if (ids.length > 0) {
 
                        for (var i = 0; i < ids.length; i++) {
-                               var e = document.getElementById("RROW-" + ids[i]);
+                               var e = $("RROW-" + ids[i]);
                                if (e) {
-                                       e.className = e.className.replace("Unread", "");
+                                       e.removeClassName("Unread");
                                }
                        }
 
-                       var query = "backend.php?op=rpc&subop=catchupSelected&ids=" +
-                               param_escape(ids.toString()) + "&cmode=0";
+                       var query = "?op=rpc&subop=catchupSelected" +
+                               "&cmode=0" + "&ids=" + param_escape(ids.toString());
 
-                       new Ajax.Request(query, {
+                       new Ajax.Request("backend.php", {
+                               parameters: query,
                                onComplete: function(transport) { 
-                                       all_counters_callback2(transport); 
+                                       handle_rpc_reply(transport); 
                                } });
 
                }
@@ -1503,31 +1368,38 @@ function cdmWatchdog() {
 
 
 function cache_inject(id, article, param) {
+
        try {
                if (!cache_check_param(id, param)) {
-                       debug("cache_article: miss: " + id + " [p=" + param + "]");
-       
-       
-                       if (db) {
+                       //console.log("cache_article: miss: " + id + " [p=" + param + "]");
+
+                  var date = new Date();
+             var ts = Math.round(date.getTime() / 1000);
 
-                          var date = new Date();
-                     var ts = Math.round(date.getTime() / 1000);
+                       if (db) {
 
                                db.execute("INSERT INTO cache (id, article, param, added) VALUES (?, ?, ?, ?)",
                                        [id, article, param, ts]);                              
                        } else {
        
-                               var cache_obj = new Array();
+                               var cache_obj = {};
        
                                cache_obj["id"] = id;
                                cache_obj["data"] = article;
                                cache_obj["param"] = param;
-       
-                               article_cache.push(cache_obj);
+
+                               if (param) id = id + ":" + param;
+
+                               cache_added["TS:" + id] = ts;
+
+                               if (has_local_storage()) 
+                                       localStorage.setItem(id, JSON.stringify(cache_obj));
+                               else
+                                       article_cache.push(cache_obj);
                        }
        
                } else {
-                       debug("cache_article: hit: " + id + " [p=" + param + "]");
+                       //console.log("cache_article: hit: " + id + " [p=" + param + "]");
                }
        } catch (e) {   
                exception_error("cache_inject", e);
@@ -1538,15 +1410,33 @@ function cache_find(id) {
 
        if (db) {
                var rs = db.execute("SELECT article FROM cache WHERE id = ?", [id]);
+               var a = false;
 
                if (rs.isValidRow()) {
-                       return rs.field(0);
+                       var a = rs.field(0);                    
                }
 
+               rs.close();
+
+               return a;
+
        } else {
-               for (var i = 0; i < article_cache.length; i++) {
-                       if (article_cache[i]["id"] == id) {
-                               return article_cache[i]["data"];
+
+               if (has_local_storage()) {
+                       var cache_obj = localStorage.getItem(id);
+       
+                       if (cache_obj) {
+                               cache_obj = JSON.parse(cache_obj);
+
+                               if (cache_obj)
+                                       return cache_obj['data'];
+                       }
+
+               } else {
+                       for (var i = 0; i < article_cache.length; i++) {
+                               if (article_cache[i]["id"] == id) {
+                                       return article_cache[i]["data"];
+                               }
                        }
                }
        }
@@ -1558,15 +1448,36 @@ function cache_find_param(id, param) {
        if (db) {
                var rs = db.execute("SELECT article FROM cache WHERE id = ? AND param = ?",
                        [id, param]);
+               var a = false;
 
                if (rs.isValidRow()) {
-                       return rs.field(0);
+                       a = rs.field(0);
                }
 
+               rs.close();
+
+               return a;
+
        } else {
-               for (var i = 0; i < article_cache.length; i++) {
-                       if (article_cache[i]["id"] == id && article_cache[i]["param"] == param) {
-                               return article_cache[i]["data"];
+
+               if (has_local_storage()) {
+
+                       if (param) id = id + ":" + param;
+
+                       var cache_obj = localStorage.getItem(id);
+
+                       if (cache_obj) {
+                               cache_obj = JSON.parse(cache_obj);
+
+                               if (cache_obj)
+                                       return cache_obj['data'];
+                       }
+
+               } else {
+                       for (var i = 0; i < article_cache.length; i++) {
+                               if (article_cache[i]["id"] == id && article_cache[i]["param"] == param) {
+                                       return article_cache[i]["data"];
+                               }
                        }
                }
        }
@@ -1578,15 +1489,25 @@ function cache_check(id) {
        if (db) {
                var rs = db.execute("SELECT COUNT(*) AS c FROM cache WHERE id = ?",
                        [id]);
+               var a = false;
 
                if (rs.isValidRow()) {
-                       return rs.field(0) != "0";
+                        a = rs.field(0) != "0";
                }
 
+               rs.close();
+
+               return a;
+
        } else {
-               for (var i = 0; i < article_cache.length; i++) {
-                       if (article_cache[i]["id"] == id) {
+               if (has_local_storage()) {
+                       if (localStorage.getItem(id))
                                return true;
+               } else {
+                       for (var i = 0; i < article_cache.length; i++) {
+                               if (article_cache[i]["id"] == id) {
+                                       return true;
+                               }
                        }
                }
        }
@@ -1598,15 +1519,30 @@ function cache_check_param(id, param) {
        if (db) {
                var rs = db.execute("SELECT COUNT(*) AS c FROM cache WHERE id = ? AND param = ?",
                        [id, param]);
+               var a = false;
 
                if (rs.isValidRow()) {
-                       return rs.field(0) != "0";
+                       a = rs.field(0) != "0";
                }
 
+               rs.close();
+
+               return a;
+
        } else {
-               for (var i = 0; i < article_cache.length; i++) {
-                       if (article_cache[i]["id"] == id && article_cache[i]["param"] == param) {
+
+               if (has_local_storage()) {
+
+                       if (param) id = id + ":" + param;
+
+                       if (localStorage.getItem(id))
                                return true;
+
+               } else {
+                       for (var i = 0; i < article_cache.length; i++) {
+                               if (article_cache[i]["id"] == id && article_cache[i]["param"] == param) {
+                                       return true;
+                               }
                        }
                }
        }
@@ -1624,14 +1560,36 @@ function cache_expire() {
 
 
        } else {
-               while (article_cache.length > 25) {
-                       article_cache.shift();
+               if (has_local_storage()) {
+
+                       var date = new Date();
+                       var timestamp = Math.round(date.getTime() / 1000);
+
+                       for (var i = 0; i < localStorage.length; i++) {
+
+                               var id = localStorage.key(i);
+
+                               if (timestamp - cache_added["TS:" + id] > 180) {
+                                       localStorage.removeItem(id);
+                               }
+                       }
+
+               } else {
+                       while (article_cache.length > 25) {
+                               article_cache.shift();
+                       }
                }
        }
 }
 
-function cache_empty() {
-       article_cache = new Array();
+function cache_flush() {
+       if (db) {
+               db.execute("DELETE FROM cache");
+       } else if (has_local_storage()) {
+               localStorage.clear();
+       } else {
+               article_cache = new Array();
+       }
 }
 
 function cache_invalidate(id) {
@@ -1642,19 +1600,39 @@ function cache_invalidate(id) {
                        return rs.rowsAffected != 0;
                } else {
 
-                       var i = 0
+                       if (has_local_storage()) {
 
-                       while (i < article_cache.length) {
-                               if (article_cache[i]["id"] == id) {
-                                       debug("cache_invalidate: removed id " + id);
-                                       article_cache.splice(i, 1);
-                                       return true;
+                               var found = false;
+
+                               for (var i = 0; i < localStorage.length; i++) {
+                                       var key = localStorage.key(i);
+
+//                                     console.warn("cache_invalidate: " + key_id + " cmp " + id);
+
+                                       if (key == id || key.indexOf(id + ":") == 0) {
+                                               localStorage.removeItem(key);
+                                               found = true;
+                                               break;
+                                       }
+                               }
+
+                               return found;
+
+                       } else {
+                               var i = 0
+
+                               while (i < article_cache.length) {
+                                       if (article_cache[i]["id"] == id) {
+                                               //console.log("cache_invalidate: removed id " + id);
+                                               article_cache.splice(i, 1);
+                                               return true;
+                                       }
+                                       i++;
                                }
-                               i++;
                        }
                }
 
-               debug("cache_invalidate: id not found: " + id);
+               //console.log("cache_invalidate: id not found: " + id);
                return false;
        } catch (e) {
                exception_error("cache_invalidate", e);
@@ -1665,39 +1643,31 @@ function getActiveArticleId() {
        return active_post_id;
 }
 
-function cdmClicked(id) {
+function preloadBatchedArticles() {
        try {
-               var elem = document.getElementById("RROW-" + id);
 
-               if (elem) {
-                       var id = elem.id.replace("RROW-", "");
-                       active_post_id = id;
+               var query = "?op=rpc&subop=getArticles&ids=" + 
+                       preload_id_batch.toString();
 
-                       cdmSelectArticles("none");
-                       toggleUnread(id, 0, true);
-                       toggleSelected(id);
+               new Ajax.Request("backend.php", {
+                       parameters: query,
+                       onComplete: function(transport) { 
 
-               }
-       } catch (e) {
-               exception_error("cdmClicked", e);
-       } 
-}
+                               preload_id_batch = [];
 
-function preload_article_callback(transport) {
-       try {
-               if (transport.responseXML) {
-                       var articles = transport.responseXML.getElementsByTagName("article");
+                               var articles = transport.responseXML.getElementsByTagName("article");
 
-                       for (var i = 0; i < articles.length; i++) {
-                               var id = articles[i].getAttribute("id");
-                               if (!cache_check(id)) {
-                                       cache_inject(id, articles[i].firstChild.nodeValue);                             
-                                       debug("preloaded article: " + id);
+                               for (var i = 0; i < articles.length; i++) {
+                                       var id = articles[i].getAttribute("id");
+                                       if (!cache_check(id)) {
+                                               cache_inject(id, articles[i].firstChild.nodeValue);                             
+                                               console.log("preloaded article: " + id);
+                                       }
                                }
-                       }
-               }
+               } }); 
+
        } catch (e) {
-               exception_error("preload_article_callback", e);
+               exception_error("preloadBatchedArticles", e);
        }
 }
 
@@ -1707,29 +1677,28 @@ function preloadArticleUnderPointer(id) {
 
                if (post_under_pointer == id && !cache_check(id)) {
 
-                       debug("trying to preload article " + id);
+                       console.log("trying to preload article " + id);
 
                        var neighbor_ids = getRelativePostIds(id, 1);
 
                        /* only request uncached articles */
 
-                       var cids_to_request = Array();
-
-                       for (var i = 0; i < neighbor_ids.length; i++) {
-                               if (!cache_check(neighbor_ids[i])) {
-                                       cids_to_request.push(neighbor_ids[i]);
+                       if (preload_id_batch.indexOf(id) == -1) {
+                               for (var i = 0; i < neighbor_ids.length; i++) {
+                                       if (!cache_check(neighbor_ids[i])) {
+                                               preload_id_batch.push(neighbor_ids[i]);
+                                       }
                                }
                        }
-                       debug("additional ids: " + cids_to_request.toString());
 
-                       cids_to_request.push(id);
+                       if (preload_id_batch.indexOf(id) == -1)
+                               preload_id_batch.push(id);
+
+                       //console.log("preload ids batch: " + preload_id_batch.toString());
+
+                       window.clearTimeout(preload_timeout_id);
+                       preload_batch_timeout_id = window.setTimeout('preloadBatchedArticles()', 1000);
 
-                       var query = "backend.php?op=rpc&subop=getArticles&ids=" + 
-                               cids_to_request.toString();
-                       new Ajax.Request(query, {
-                               onComplete: function(transport) { 
-                                       preload_article_callback(transport);
-                       } });
                }
        } catch (e) {
                exception_error("preloadArticleUnderPointer", e);
@@ -1761,23 +1730,16 @@ function postMouseOut(id) {
 function headlines_scroll_handler() {
        try {
 
-               var e = document.getElementById("headlinesInnerContainer");
-
-               // don't do infinite scrolling when Limit == All
+               var e = $("headlinesInnerContainer");
 
                var toolbar_form = document.forms["main_toolbar_form"];
 
-               var limit = toolbar_form.limit[toolbar_form.limit.selectedIndex];
-               if (limit.value != 0) {
-               
-                       debug((e.scrollTop + e.offsetHeight) + " vs " + e.scrollHeight + " dis? " +
-                               _infscroll_disable);
+//             console.log((e.scrollTop + e.offsetHeight) + " vs " + e.scrollHeight + " dis? " +
+//                     _infscroll_disable);
 
-                       if (e.scrollTop + e.offsetHeight > e.scrollHeight - 100) {
-                               if (!_infscroll_disable) {
-                                       debug("more cowbell!");
-                                       viewNextFeedPage();
-                               }
+               if (e.scrollTop + e.offsetHeight > e.scrollHeight - 100) {
+                       if (!_infscroll_disable) {
+                               viewNextFeedPage();
                        }
                }
 
@@ -1796,22 +1758,16 @@ function catchupRelativeToArticle(below) {
                        return;
                }
 
-               var visible_ids;
-
-               if (document.getElementById("headlinesList")) {
-                       visible_ids = getVisibleHeadlineIds();
-               } else {
-                       visible_ids = cdmGetVisibleArticles();
-               }
+               var visible_ids = getVisibleArticleIds();
 
                var ids_to_mark = new Array();
 
                if (!below) {
                        for (var i = 0; i < visible_ids.length; i++) {
                                if (visible_ids[i] != getActiveArticleId()) {
-                                       var e = document.getElementById("RROW-" + visible_ids[i]);
+                                       var e = $("RROW-" + visible_ids[i]);
 
-                                       if (e && e.className.match("Unread")) {
+                                       if (e && e.hasClassName("Unread")) {
                                                ids_to_mark.push(visible_ids[i]);
                                        }
                                } else {
@@ -1821,9 +1777,9 @@ function catchupRelativeToArticle(below) {
                } else {
                        for (var i = visible_ids.length-1; i >= 0; i--) {
                                if (visible_ids[i] != getActiveArticleId()) {
-                                       var e = document.getElementById("RROW-" + visible_ids[i]);
+                                       var e = $("RROW-" + visible_ids[i]);
 
-                                       if (e && e.className.match("Unread")) {
+                                       if (e && e.hasClassName("Unread")) {
                                                ids_to_mark.push(visible_ids[i]);
                                        }
                                } else {
@@ -1837,17 +1793,18 @@ function catchupRelativeToArticle(below) {
                } else {
                        var msg = __("Mark %d article(s) as read?").replace("%d", ids_to_mark.length);
 
-                       if (confirm(msg)) {
+                       if (getInitParam("confirm_feed_catchup") != 1 || confirm(msg)) {
 
                                for (var i = 0; i < ids_to_mark.length; i++) {
-                                       var e = document.getElementById("RROW-" + ids_to_mark[i]);
-                                       e.className = e.className.replace("Unread", "");
+                                       var e = $("RROW-" + ids_to_mark[i]);
+                                       e.removeClassName("Unread");
                                }
 
-                               var query = "backend.php?op=rpc&subop=catchupSelected&ids=" +
-                                       param_escape(ids_to_mark.toString()) + "&cmode=0";
+                               var query = "?op=rpc&subop=catchupSelected" +
+                                       "&cmode=0" + "&ids=" + param_escape(ids_to_mark.toString()); 
 
-                               new Ajax.Request(query, {
+                               new Ajax.Request("backend.php", {
+                                       parameters: query,
                                        onComplete: function(transport) { 
                                                catchup_callback2(transport); 
                                        } });
@@ -1860,126 +1817,107 @@ function catchupRelativeToArticle(below) {
        }
 }
 
-function cdmExpandArticle(a_id) {
+function cdmExpandArticle(id) {
        try {
-               var id = 'CICD-' + a_id;
-
-               try {
-                       Element.hide("CEXC-" + a_id);
-               } catch (e) { } 
-
-               Effect.Appear(id, {duration : 0.5, 
-                       beforeStart: function(effect) { 
-                               var h_id = 'CICH-' + a_id;
-                               var h_elem = document.getElementById(h_id);
-                               if (h_elem) { h_elem.style.display = "none"; }
 
-                               toggleUnread(a_id, 0);
-                       }});
+               hideAuxDlg();
 
+               var elem = $("CICD-" + active_post_id);
 
-       } catch (e) {
-               exception_error("appearBlockElementF", e);
-       }
-
-}
+               var upd_img_pic = $("FUPDPIC-" + id);
 
-function fixHeadlinesOrder(ids) {
-       try {
-               for (var i = 0; i < ids.length; i++) {
-                       var e = document.getElementById("RROW-" + ids[i]);
+               if (upd_img_pic && (upd_img_pic.src.match("updated.png") || 
+                               upd_img_pic.src.match("fresh_sign.png"))) {
 
-                       if (e) {
-                               if (i % 2 == 0) {
-                                       e.className = e.className.replace("even", "odd");
-                               } else {
-                                       e.className = e.className.replace("odd", "even");
-                               }
-                       }
+                       upd_img_pic.src = "images/blank_icon.gif";
                }
-       } catch (e) {
-               exception_error("fixHeadlinesOrder", e);
-       }
-}
 
-function hideReadHeadlines() {
-       try {
+               if (id == active_post_id && Element.visible(elem))
+                       return true;
 
-               var ids = false;
-               var vis_ids = new Array();
+               selectArticles("none");
 
-               if (document.getElementById("headlinesList")) {
-                       ids = getVisibleHeadlineIds();
-               } else {
-                       ids = cdmGetVisibleArticles();
+               var old_offset = $("RROW-" + id).offsetTop;
+
+               if (active_post_id && elem && !getInitParam("cdm_expanded")) {
+                       Element.hide(elem);
+                       Element.show("CEXC-" + active_post_id);
                }
 
-               var read_headlines_visible = true;
+               active_post_id = id;
 
-               for (var i = 0; i < ids.length; i++) {
-                       var row = document.getElementById("RROW-" + ids[i]);
+               elem = $("CICD-" + id);
 
-                       if (row && row.className) {
-                               if (read_headlines_visible) {
-                                       if (row.className.match("Unread") || row.className.match("Selected")) {
-                                               Element.show(row);
-                                               vis_ids.push(ids[i]);
-                                       } else {
-                                               //Effect.Fade(row, {duration : 0.3});
-                                               Element.hide(row);
-                                       }
-                               } else {
-                                       Element.show(row);
-                                       vis_ids.push(ids[i]);
-                               }
-                       }
-               }
-               
-               fixHeadlinesOrder(vis_ids);
+               if (!Element.visible(elem)) {
+                       Element.show(elem);
+                       Element.hide("CEXC-" + id);
 
-               read_headlines_visible = !read_headlines_visible;
+                       if ($("CWRAP-" + id).innerHTML == "") {
 
-       } catch (e) {
-               exception_error("hideReadHeadlines", e);
-       } 
-}
-
-function invertHeadlineSelection() {
-       try {
-               var rows = new Array();
-               var r = false;
-               
-               if (!isCdmMode()) {             
-                       r = document.getElementsByTagName("TR");
-               } else {
-                       r = document.getElementsByTagName("DIV");
-               }
+                               $("FUPDPIC-" + id).src = "images/indicator_tiny.gif";
 
-               for (var i = 0; i < r.length; i++) {
-                       if (r[i].id && r[i].id.match("RROW-")) {
-                               rows.push(r[i]);
+                               $("CWRAP-" + id).innerHTML = "<div class=\"insensitive\">" + 
+                                       __("Loading, please wait...") + "</div>";
+       
+                               var query = "?op=rpc&subop=cdmGetArticle&id=" + param_escape(id);
+       
+                               //console.log(query);
+       
+                               new Ajax.Request("backend.php", {
+                                       parameters: query,
+                                       onComplete: function(transport) { 
+                                               $("FUPDPIC-" + id).src = 'images/blank_icon.gif';
+       
+                                               if (transport.responseXML) {
+                                                       var article = transport.responseXML.getElementsByTagName("article")[0];
+                                                       var recv_id = article.getAttribute("id");
+       
+                                                       if (recv_id == id)
+                                                               $("CWRAP-" + id).innerHTML = article.firstChild.nodeValue;
+       
+                                               } else {
+                                                       $("CWRAP-" + id).innerHTML = __("Unable to load article.");
+       
+                                               }
+                               }});
+       
                        }
                }
-               
-               for (var i = 0; i < rows.length; i++) {
-                       var nc = rows[i].className;
-                       var id = rows[i].id.replace("RROW-", "");
-                       var cb = document.getElementById("RCHK-" + id);
 
-                       if (!rows[i].className.match("Selected")) {
-                               nc = nc + "Selected";
-                               cb.checked = true;
-                       } else {
-                               nc = nc.replace("Selected", "");
-                               cb.checked = false;
-                       }
+               var new_offset = $("RROW-" + id).offsetTop;
 
-                       rows[i].className = nc;
+               $("headlinesInnerContainer").scrollTop += (new_offset-old_offset);
 
-               }
+               if ($("RROW-" + id).offsetTop != old_offset) 
+                       $("headlinesInnerContainer").scrollTop = new_offset;
+
+               toggleUnread(id, 0, true);
+               toggleSelected(id);
 
        } catch (e) {
-               exception_error("invertHeadlineSelection", e);
+               exception_error("cdmExpandArticle", e);
+       }
+
+       return false;
+}
+
+function fixHeadlinesOrder(ids) {
+       try {
+               for (var i = 0; i < ids.length; i++) {
+                       var e = $("RROW-" + ids[i]);
+
+                       if (e) {
+                               if (i % 2 == 0) {
+                                       e.removeClassName("even");
+                                       e.addClassName("odd");
+                               } else {
+                                       e.removeClassName("odd");
+                                       e.addClassName("even");
+                               }
+                       }
+               }
+       } catch (e) {
+               exception_error("fixHeadlinesOrder", e);
        }
 }
 
@@ -1998,51 +1936,15 @@ function zoomToArticle(id) {
        }
 }
 
-function showOriginalArticleInline(id) {
-       try {
-
-               var query = "backend.php?op=rpc&subop=getArticleLink&id=" + id;
-
-               notify_progress("Loading, please wait...", true);
-
-               new Ajax.Request(query, {
-                       onComplete: function(transport) { 
-
-                               if (transport.responseXML) {
-                       
-                                       var link = transport.responseXML.getElementsByTagName("link")[0];
-                                       var id = transport.responseXML.getElementsByTagName("id")[0];
-
-                                       notify("");
-
-                                       if (link && id) {
-                                               link = link.firstChild.nodeValue;
-
-                                               var ci = document.getElementById("content-insert");
-
-                                               var tmp = "<iframe id=\"inline_orig_article\" width=\""+ci.offsetWidth+"\" height=\""+ci.offsetHeight+"\" style=\"border-width : 0px;\" src=\""+link+"\"></iframe>";
-
-                                               render_article(tmp);
-
-                                       }
-                               }
-                       } });
-
-       } catch (e) {
-               exception_error("showOriginalArticleInline", e);
-       }
-}
-
-
 function scrollArticle(offset) {
        try {
                if (!isCdmMode()) {
-                       var ci = document.getElementById("content-insert");
+                       var ci = $("content-insert");
                        if (ci) {
                                ci.scrollTop += offset;
                        }
                } else {
-                       var hi = document.getElementById("headlinesInnerContainer");
+                       var hi = $("headlinesInnerContainer");
                        if (hi) {
                                hi.scrollTop += offset;
                        }
@@ -2065,7 +1967,7 @@ function show_labels_in_headlines(transport) {
 
                                if (e_id) {
 
-                                       var ctr = document.getElementById("HLLCTR-" + e_id);
+                                       var ctr = $("HLLCTR-" + e_id);
 
                                        if (ctr) {
                                                ctr.innerHTML = elems[l].firstChild.nodeValue;
@@ -2083,8 +1985,8 @@ function show_labels_in_headlines(transport) {
 
 function toggleHeadlineActions() {
        try {
-               var e = document.getElementById("headlineActionsBody");
-               var p = document.getElementById("headlineActionsDrop");
+               var e = $("headlineActionsBody");
+               var p = $("headlineActionsDrop");
 
                if (!Element.visible(e)) {
                        Element.show(e);
@@ -2094,9 +1996,380 @@ function toggleHeadlineActions() {
 
                e.scrollTop = 0;
                e.style.left = (p.offsetLeft + 1) + "px";
-//             e.style.top = (p.offsetTop + p.offsetHeight - 1) + "px";
+               e.style.top = (p.offsetTop + p.offsetHeight + 2) + "px";
 
        } catch (e) {
                exception_error("toggleHeadlineActions", e);
        }
 }
+
+function publishWithNote(id, def_note) {
+       try {
+               if (!def_note) def_note = '';
+
+               var note = prompt(__("Please enter a note for this article:"), def_note);
+
+               if (note != undefined) {
+                       togglePub(id, false, false, note);
+               }
+
+       } catch (e) {
+               exception_error("publishWithNote", e);
+       }
+}
+
+function emailArticle(id) {
+       try {
+               if (!id) {
+                       var ids = getSelectedArticleIds2();
+
+                       if (ids.length == 0) {
+                               alert(__("No articles are selected."));
+                               return;
+                       }
+
+                       id = ids.toString();
+               }
+
+               displayDlg('emailArticle', id, 
+                  function () {                                
+                               document.forms['article_email_form'].destination.focus();
+
+                          new Ajax.Autocompleter('destination', 'destination_choices',
+                                  "backend.php?op=rpc&subop=completeEmails",
+                                  { tokens: '', paramName: "search" });
+
+                       });
+
+       } catch (e) {
+               exception_error("emailArticle", e);
+       }
+}
+
+function emailArticleDo() {
+       try {
+               var f = document.forms['article_email_form'];
+
+               if (f.destination.value == "") {
+                       alert("Please fill in the destination email.");
+                       return;
+               }
+
+               if (f.subject.value == "") {
+                       alert("Please fill in the subject.");
+                       return;
+               }
+
+               var query = Form.serialize("article_email_form");
+
+//             console.log(query);
+
+               new Ajax.Request("backend.php", {
+                       parameters: query,
+                       onComplete: function(transport) { 
+                               try {
+
+                                       var error = transport.responseXML.getElementsByTagName('error')[0];
+
+                                       if (error) {
+                                               alert(__('Error sending email:') + ' ' + error.firstChild.nodeValue);
+                                       } else {
+                                               notify_info('Your message has been sent.');
+                                               closeInfoBox();
+                                       }
+
+                               } catch (e) {
+                                       exception_error("sendEmailDo", e);
+                               }
+
+                       } });
+
+       } catch (e) {
+               exception_error("emailArticleDo", e);
+       }
+}
+
+function dismissArticle(id) {
+       try {
+               var elem = $("RROW-" + id);
+
+               toggleUnread(id, 0, true);
+
+               new Effect.Fade(elem, {duration : 0.5});
+
+               active_post_id = false;
+
+       } catch (e) {
+               exception_error("dismissArticle", e);
+       }
+}
+
+function dismissSelectedArticles() {
+       try {
+
+               var ids = getVisibleArticleIds();
+               var tmp = [];
+               var sel = [];
+
+               for (var i = 0; i < ids.length; i++) {
+                       var elem = $("RROW-" + ids[i]);
+
+                       if (elem.className && elem.hasClassName("Selected") && 
+                                       ids[i] != active_post_id) {
+                               new Effect.Fade(elem, {duration : 0.5});
+                               sel.push(ids[i]);
+                       } else {
+                               tmp.push(ids[i]);
+                       }
+               }
+
+               if (sel.length > 0)
+                       selectionToggleUnread(false);
+
+               fixHeadlinesOrder(tmp);
+
+       } catch (e) {
+               exception_error("dismissSelectedArticles", e);
+       }
+}
+
+function dismissReadArticles() {
+       try {
+
+               var ids = getVisibleArticleIds();
+               var tmp = [];
+
+               for (var i = 0; i < ids.length; i++) {
+                       var elem = $("RROW-" + ids[i]);
+
+                       if (elem.className && !elem.hasClassName("Unread") && 
+                                       !elem.hasClassName("Selected")) {
+                       
+                               new Effect.Fade(elem, {duration : 0.5});
+                       } else {
+                               tmp.push(ids[i]);
+                       }
+               }
+
+               fixHeadlinesOrder(tmp);
+
+       } catch (e) {
+               exception_error("dismissSelectedArticles", e);
+       }
+}
+
+function getVisibleArticleIds() {
+       var ids = [];
+
+       try {
+               
+               getLoadedArticleIds().each(function(id) {
+                       var elem = $("RROW-" + id);
+                       if (elem && Element.visible(elem))
+                               ids.push(id);
+                       });
+
+       } catch (e) {
+               exception_error("getVisibleArticleIds", e);
+       }
+
+       return ids;
+}
+
+function cdmClicked(event, id) {
+       try {
+               var shift_key = event.shiftKey;
+
+               hideAuxDlg();
+
+               if (!event.ctrlKey) {
+                       selectArticles("none");
+                       toggleSelected(id);
+
+                       var elem = $("RROW-" + id);
+
+                       if (elem)
+                               elem.removeClassName("Unread");
+
+                       var upd_img_pic = $("FUPDPIC-" + id);
+
+                       if (upd_img_pic && (upd_img_pic.src.match("updated.png") || 
+                                       upd_img_pic.src.match("fresh_sign.png"))) {
+
+                               upd_img_pic.src = "images/blank_icon.gif";
+                       }
+
+                       active_post_id = id;
+
+                       var query = "?op=rpc&subop=catchupSelected" +
+                               "&cmode=0&ids=" + param_escape(id);
+
+                       new Ajax.Request("backend.php", {
+                               parameters: query,
+                               onComplete: function(transport) { 
+                                       handle_rpc_reply(transport); 
+                               } });
+
+                       return true;
+               } else {
+                       toggleSelected(id);
+               }
+
+       } catch (e) {
+               exception_error("cdmClicked");
+       }
+
+       return false;
+}
+
+function hlClicked(event, id) {
+       try {
+
+               if (!event.ctrlKey) {
+                       view(id);
+                       return true;
+               } else {
+                       toggleSelected(id);
+                       return false;
+               }
+
+       } catch (e) {
+               exception_error("hlClicked");
+       }
+
+       return false;
+}
+
+function getFirstVisibleHeadlineId() {
+       var rows = getVisibleArticleIds();
+       return rows[0];
+       
+}
+
+function getLastVisibleHeadlineId() {
+       var rows = getVisibleArticleIds();
+       return rows[rows.length-1];
+}
+
+function openArticleInNewWindow(id) {
+       try {
+               console.log("openArticleInNewWindow: " + id);
+
+               var query = "?op=rpc&subop=getArticleLink&id=" + id;
+               var wname = "ttrss_article_" + id;
+
+               console.log(query + " " + wname);
+
+               var w = window.open("", wname);
+
+               if (!w) notify_error("Failed to open window for the article");
+
+               new Ajax.Request("backend.php", {
+                       parameters: query,
+                       onComplete: function(transport) { 
+
+                                       var link = transport.responseXML.getElementsByTagName("link")[0];
+                                       var id = transport.responseXML.getElementsByTagName("id")[0];
+               
+                                       console.log("open_article received link: " + link);
+               
+                                       if (link && id) {
+               
+                                               var wname = "ttrss_article_" + id.firstChild.nodeValue;
+               
+                                               console.log("link url: " + link.firstChild.nodeValue + ", wname " + wname);
+               
+                                               var w = window.open(link.firstChild.nodeValue, wname);
+               
+                                               if (!w) { notify_error("Failed to load article in new window"); }
+               
+                                               if (id) {
+                                                       id = id.firstChild.nodeValue;
+                                                       window.setTimeout("toggleUnread(" + id + ", 0)", 100);
+                                               }
+                                       } else {
+                                               notify_error("Can't open article: received invalid article link");
+                                       }
+                               } });
+
+       } catch (e) {
+               exception_error("openArticleInNewWindow", e);
+       }
+}
+
+function isCdmMode() {
+       return getInitParam("combined_display_mode");
+}
+
+function markHeadline(id) {
+       var row = $("RROW-" + id);
+       if (row) {
+               var check = $("RCHK-" + id);
+
+               if (check) {
+                       check.checked = true;
+               }
+
+               row.addClassName("Selected");
+       }
+}
+
+function getRelativePostIds(id, limit) {
+
+       var tmp = [];
+
+       try {
+
+               if (!limit) limit = 3;
+       
+               var ids = getVisibleArticleIds();
+       
+               for (var i = 0; i < ids.length; i++) {
+                       if (ids[i] == id) {
+                               for (var k = 1; k <= limit; k++) {
+                                       if (i > k-1) tmp.push(ids[i-k]);
+                                       if (i < ids.length-k) tmp.push(ids[i+k]);
+                               }
+                               break;
+                       }
+               }
+
+       } catch (e) {
+               exception_error("getRelativePostIds", e);
+       }
+
+       return tmp;
+}
+
+function correctHeadlinesOffset(id) {
+       
+       try {
+
+               var container = $("headlinesInnerContainer");
+               var row = $("RROW-" + id);
+       
+               var viewport = container.offsetHeight;
+       
+               var rel_offset_top = row.offsetTop - container.scrollTop;
+               var rel_offset_bottom = row.offsetTop + row.offsetHeight - container.scrollTop;
+       
+               //console.log("Rtop: " + rel_offset_top + " Rbtm: " + rel_offset_bottom);
+               //console.log("Vport: " + viewport);
+
+               if (rel_offset_top <= 0 || rel_offset_top > viewport) {
+                       container.scrollTop = row.offsetTop;
+               } else if (rel_offset_bottom > viewport) {
+
+                       /* doesn't properly work with Opera in some cases because
+                               Opera fucks up element scrolling */
+
+                       container.scrollTop = row.offsetTop + row.offsetHeight - viewport;              
+               } 
+
+       } catch (e) {
+               exception_error("correctHeadlinesOffset", e);
+       }
+
+}
+
+