Greasy Fork is available in English.

[TS] Schmoogle

Dated but most functions work; Google: Fully Customisable Skins (columns, font colors, bgc etc.) | AutoPaging with Lazy Load Support | Remove Tracking/Redirection | URL Cleaner | Quick Search Filter | URL Blacklisting | SSL Search | Multiple Search Services | Quick Links To Google Services | Ad Removal...

// ==UserScript==
// @name            [TS] Schmoogle
// @namespace       TimidScript
// @version         2.1.43.1
// @description     Dated but most functions work; Google: Fully Customisable Skins (columns, font colors, bgc etc.) | AutoPaging with Lazy Load Support | Remove Tracking/Redirection | URL Cleaner | Quick Search Filter | URL Blacklisting | SSL Search | Multiple Search Services | Quick Links To Google Services | Ad Removal... 
// @icon            https://i.imgur.com/Kvb2Ili.png
// @author          TimidScript
// @homepageURL     https://openuserjs.org/users/TimidScript
// @copyright       © 2013+ TimidScript, Some Rights Reserved.
// @license         https://github.com/TimidScript/UserScripts/blob/master/license.txt
// @include         *//www.google.*
// @include         *//encrypted.google.*
// @require         https://openuserjs.org/src/libs/TimidScript/TSL_-_jsColorGM.js
// @require         https://openuserjs.org/src/libs/TimidScript/TSL_-_Generic.js
// @require         https://openuserjs.org/src/libs/TimidScript/TSL_-_GM_Update.js
// @homeURL         https://openuserjs.org/scripts/TimidScript/[TS]_Schmoogle
// @grant           GM_listValues
// @grant           GM_xmlhttpRequest
// @grant           GM_info
// @grant           GM_getMetadata
// @grant           GM_getValue
// @grant           GM_setValue
// @grant           GM_deleteValue
// @grant           GM_registerMenuCommand
// ==/UserScript==

/* License + Copyright Notice
********************************************************************************************
License can be found at: https://github.com/TimidScript/UserScripts/blob/master/license.txt
Below is a copy of the license the may not be up-to-date.

Copyright © TimidScript, Some Rights Reserved.

Redistribution and use in source and binary forms, with or without modification, are permitted provided that the
following conditions are met:

1) GPL-3 License is met that does not conflict with the rest of the license (http://www.gnu.org/licenses/gpl-3.0.en.html)
2) This notice must be included
3) Due credits and link to original author's homepage (included in this notice).
4) Notify the original author of redistribution
5) Clear clarification of the License and Notice to the end user
6) Do not upload on OpenUserJS.org or any other site that infringes on this license

TimidScript's Homepages:  GitHub:      https://github.com/TimidScript
                          GreasyFork:  https://greasyfork.org/users/1455
*/
/* Information
********************************************************************************************
TODO List
----------------------------------------------
PRIMARY
 - Add Comments
 - Image Search
 - GUI to change Privacy settings
 - GUI to ad removal options
 - Better Skin Tab

SECONDARY
 - Better privacy: Scramble cookies, remove click tracking,
 replaces input box, replace instant search
 - CSS Tab (Add your own and access Schmoogle CSS)
 
BUGS
 - Change Settings Main Tab (Naming) match ContentType bug
 - No longer picking up places content type due to lazy
 loading
 
----------------------------------------------
    Version History
----------------------------------------------
2.1.X (2013/08/07  Complete re-write) 
  Since last release Google has changed a lot of it's html script by changing much of the
layout, removing some features and increase use of Lazy Loading. This has broken the script
in many places so I decided it was about time I rewrite the script to make it more concise 
and remove the broken features. This is a complete rewrite of the script. Please
show your appreciation by becoming a Fan of the script on USO (Userscript.org).
 - Full support of Lazy Loading. Images now work :)
 - Optimised and more powerful script (Concise Coding)
 - Efficient usage of CSS3
 - Better Interface
 - Remove Tracking Functionality
 - URL Cleaner
 - Better formatting according to content
 - More Google customization options and customisable skin
 - Safe Search Option 
 - Much more...
 - FRemoved: Schmoogle Menu (replaced by buttons)
 - FRemoved: GoogleSharing (service has been discontinued)
 - FRemoved: Preview Mode (Google has removed "Google Instant Previews"). 
   Preview locations of places is still available but all handled by Google and 
   not this script.

Image Search
GUI to change Privacy settings
GUI to ad removal options
Maybe better CSS interface
Fix Bug listed in 2.1.40 (Places)
2.1.43 (2014-09-08)
 - Bug Fix: Due to changes to TSL-Generic.
2.1.42 (2014-08-29)
 - Added GM_update
2.1.41
 - Bug Fix: window.scrollMaxY undefined in Google Chrome
 - Bug Fix: Auto-paging in Google Chrome
 - Removal of unwanted grouping header li
 - [d] Bug Fix: Added missing image CSS for auto-page
2.1.40
 - Bug Fix: HTML Styles
 - Bug Fix: Introduced in 2.1.39 a bug by renaming table values. Temporary 
 fix is made. Need to change settings window html accordingly.
 attribute instead of element content in settings window. 
 - Bug: Introduced in 2.1.39. Due to changes in observe, result items with
 lazy loading class names (Places) are not being correctly detected. Will try to
 fix for version 2.1.41.
2.1.39
 - Context Menu Added (Skin, Jump)
 - Major rewrite in the way the first page is added
 - Major changes to the Beautifier object
 - CTinfo has fixed size now.
 - Improved URL cleaner
 - Support for instant search added and also rewrote Observe.callback & Pager.newPage
 - Bug Fix: UK stores navcnt in center_col while COM version does not. There
 is a list of conflicts between Google national services
2.1.38 (Features Added)
 - Pager runs through MutationObserver rather than interval timer
 - Support more topstuff and extra informational content (ongoing endeavour)
 - Minor Changes 
.1.37.BETA (Features Added)
 - Pager Timer Based
 - Minor Changes
2.1.36
 - Removed AutoPager Support (Found GoogleKingKong Lazy Loading Fix)
 - Major functionality changes
2.1.35.ALPHA (Features Added)
 - Major functionality changes
 - URL Cleaner
 - Remove Tracking
2.1.34.ALPHA Complete Rewrite
 - Most features added
 - Capture Changes through timer
 - New Interface
 - CSS Added
 - AutoPager Support

1.1.33
 - Small Changes
 
1.1.32
 - Bug Fix: HashChange pickup
 - Temporary bug fix: live.intervalTB never cleared
 
1.1.31
 - Feature Added: Filter already added it won't add it again.
 - Small Bug Fix: Filter  bug fix for Schmoogle

1.1.30
 - Small Bug Fix: Content of type moreLinks transparent setting was not being set.

1.1.29
 - With the introduction of Font colours, the skin window uses last loaded saved font colours. This
 replaces this behaviour by providing a default font colours if one does not exist. 
(Uses default on GM_getValue).

1.1.28
 - Bug Fix: Bug fixes to version 1.1.27 forget to take into account the way 
 colourizetable works

1.1.27
 - Added ability to change font colours
 - Added ability to change body background colour
 -
1.1.26
 - GUI Fix: InfoTab causes horizontal scrollbar.

1.1.25
 - Bug Fix: Search Filter in Schmoogle sometimes has Google redirection prefix. 
 - Added Feature: You can remove blacklist from the mini-form now. 

1.1.24
 - TBug Fix: When you add the Schmoogle menu too fast Google removes it. I thought I fixed
 the issue in latest release but failed. Try to add the menu twice. Temporary fix of adding
 AddGoogleToolbarButtons at the end of GoogleUpdate
 - Graphic Fix: Change the Preview eye style from relative to absolute and placed it to 
 the right.

1.1.23
 - Bug Fix: Unknown Content settings are not being saved from the "Main Settings" window
 - Bug Fix: If you check remove video content and there are more video links of type 
 "moreLinks" (video) an error use to occur.
 - GoogleSharing now has its own column storage
 - oldFormat checking is removed as only GoogleSharing has old google format
 - Default Values Changes: Unknown Content items take a whole row. 

1.1.22
 - GoogleSharing Bug Fix: Spelling correction appears at the top now 
 - Replaced setInterval with setTimout. LoadWaitPeriod() removed

1.1.21
 - Bug Fix: OnMouseOut changed to OnMouseLeave

1.1.20
 - Few changes to the Schmoogle Menu (Width, Order, Location).
 - Removed Beta Status

0.1.19
 - Bug Fix: Removed GM_info as Scriptish does not support it. 

0.1.18 Initial Early Beta Release
 - Added basic comments
 - Renaming functions and variables and cleaned up the code a little. 
 - Changed some default setting's values.
 - Uses version meta data to check settings. 
 - Bug Fix: GoogleSharing #rhs element is null and not checked

0.1.17 (3/03/2013) Initial Early Beta Release
 - Feature: Hidden RHS: Removed WindowResized. Fixes issues with table width.
      Now RHS panel can be accessed through vertical tab. 
 - Feature: Skins Settings Window
 - Feature: Dynamic Skins Menu
 - Feature: Main Settings Window + Lots of options
 - Feature: Blacklist Settings Window    
 - BUG!!! Still unable to fix autopaging error 

0.1.13 (24/02/2013)
 - Code Structure stabilized.
 - Implemented that coding the enables colour scheme changing with a single call.     
 - Storing most of the settings. Blacklisting is still temporary.
 - Partial functioning menu. CSS for menu has some noticeable cosmetic issues.     
 - AddGoogleToolbarButtons now executes GoogleUpdates if Mutation does not catch
      changes in time.
 - Better support for GoogleSharing. 
 - Blacklisting temporary state enabled. Permanent blacklisting and settings window
      yet to by implemented

0.1.12 
 - Restructured much of the code, using different approach in handling events and
      implementation of new interface using themes. 
 - Added blacklisted buttons.
 - Added "Similar" and "Cached" options

0.0.10 (18/02/2013)
 - Major changes to GUI.
 - Questionable fix for content width bug (useless)          
       
0.0.5
 - Added filtering
 - Auto-paging removed due to incomplete images. Possible fix is simulate a click on Google Next.

0.0.2
 - Fixed most of the sync issues
 - Changed layout
 - Multi-column support added
 - Partial support for GoogleSharing. GoogleSharing removes important ids from result and does
    slight reformatting.
 - Auto-paging added
 
0.0.1 (13/02/2013)
 - Initial release
********************************************************************************************/
console.info("Schmoogling Google");

/********************************************************************************************/
/*
---------------------------------------
 Variable You Can Change (VYCH)
---------------------------------------*/


/*
---------------------------------------
 Do not touch
---------------------------------------*/

var SEARCHTYPES = "unknown webSearch imageSearch videoSearch newsSearch blogSearch discussionSearch placesSearch bookSearch appSearch patentSearch recipeSearch";
var SearchTypes = new (makeStruct(SEARCHTYPES))();
var SearchType = GetSearchType();

var CONTENTTYPES = makeStruct("info images videos news places unknown standard moreLinks");
var ContentTypes = new CONTENTTYPES();

var rsoRemoved = false;
var Counter = 0;
var StartPage = 1;
var PageCount = 0;
var ResultItemCounter = 0;
var ResultItems = document.createElement("div");
/********************************************************************************************/

//#region Global Functions ---------------------------------------------------------------------
//For Google Chrome
//if (typeof window.scrollMaxY !== "number")
//{
//    Object.defineProperties(window, "scrollMaxY",
//            {
//                get: function () { return document.documentElement.scrollHeight - document.documentElement.clientHeight; },
//                //set : function(newValue){ bValue = newValue; },
//                enumerable: true,
//                configurable: true
//            });
//};

String.prototype.lPad = function (padString, length)
{
    var str = this;
    while (str.length < length)
        str = padString + str;
    return str;
}

function FindStringInArray(arr, str)
{
    var s2 = str.toLowerCase();
    for (var i = 0; i < arr.length; i++)
    {
        if (arr[i].toLowerCase() == s2) return i;
    }

    return -1;
}

/*
=================================================================================================
 Possible flag values for extra type of checking:
 2: start match
 4: contained match
 8: end match
 Always checks for exact match
 
=================================================================================================*/
function MatchArrayStringToString(arr, str, flag, prefix)
{
    if (!flag) flag = 2;

    var s1 = str.toLowerCase();


    for (var i = 0; i < arr.length; i++)
    {
        s2 = arr[i].toLowerCase();

        if (s1 == s2) return i;
        if (prefix) s2 = prefix + s2;

        var pos = s1.indexOf(s2);
        if (2 & flag && pos == 0) return i;
        else if (4 & flag && pos >= 0) return i;
        else if (8 & flag && pos > 0 && (pos + s2.length) == s1.length) return i;
    }

    return -1;
}

/*
=================================================================================================
 Tries to get domain name from hostname. Hit and miss with this. But in most cases it's a hit.
=================================================================================================*/
function GetDomain(hostname)
{
    if (hostname.toLowerCase().indexOf("www") == 0) return hostname.substring(4);
    else
    {
        //SLD = [co, ltd, net, org, plc, sch, ac, gov, mod, nhs, police] //Second level domains        
        return hostname.match(/\w+\.\w+$|\w+\.(co|ltd|org|gov|ac|nhs|ac)\.\w+$/i)[0];
    }
}


function makeStruct(names)
{
    var names = names.split(' ');
    var count = names.length;
    function constructor()
    {
        for (var i = 0; i < count; i++)
        {
            this[names[i]] = names[i];
        }
    }
    return constructor;
}


function IsNull(obj)
{
    return (obj === null);
}

function removeNode(node)
{
    if (typeof node == "string") node = document.getElementById(node);
    if (!IsNull(node) && node.parentNode) node.parentNode.removeChild(node);
}

function GetSearchType()
{
    SearchType = SearchTypes.unknown;

    var url = document.URL;
    if (!(url.match(/(&|\?)tbm=/)) && url.match(/(&|\?|#)q=/) && url != "http://www.google.com/blank.html") SearchType = SearchTypes.webSearch;
    else
    {
        //Use match statement as sometimes you have history in URL and thus contain
        //more than one tbm and we try to capture last one only.
        var m = url.match(/(&|\?)tbm=([a-z]+)/i);
        if (!m) return SearchTypes.unknown;
        m = m[m.length - 1].replace(/(&|\?)tbm=/i, "");
        switch (m)
        {
            case "vid":
                SearchType = SearchTypes.videoSearch;
                break;
            case "nws":
                SearchType = SearchTypes.newsSearch;
                break;
            case "blg":
                SearchType = SearchTypes.blogSearch;
                break;
            case "dsc":
                SearchType = SearchTypes.discussionSearch;
                break;
            case "plcs":
                SearchType = SearchTypes.placesSearch;
                break;
            case "bks":
                SearchType = SearchTypes.bookSearch;
                break;
            case "pts":
                SearchType = SearchTypes.patentSearch;
                break;
            case "app":
                SearchType = SearchTypes.appSearch;
                break;
            case "rcp":
                SearchType = SearchTypes.recipeSearch;
                break;
            case "isch": //Image Search
                SearchType = SearchTypes.imageSearch;
                break;
            case "shop": //Shopping                            
                break;
        }
    }

    return SearchType;
};


function AddHTMLElement(htmlCode)
{
    var temp = document.createElement("div");
    temp.innerHTML = htmlCode;
    return temp.firstElementChild;
}


function AddStyle(id, script)
{
    var style = document.createElement("style");
    style.type = "text/css";
    if (id)
    {
        style.id = id;
        removeNode(id);
    }
    style.innerHTML = script;

    document.head.appendChild(style);
}

//#endregion ----------------------------------------

var Settings =
{
    VersionCheck: function (current)
    {
        var saved = GM_getValue("Version", "0");

        if (saved != "0" && saved != current) alert("Some Schmoogle settings will be reset due to major changes in the update.");

        saved = saved.split(".");
        currentA = current.split(".");
        //Remove defunct settings        
        if (saved[0] != currentA[0])
        {
            var names = GM_listValues();
            for (var i = 0; name = names[i], i < names.length; i++)
            {
                var skipNames = ["[SKIN]", "USO-Updater", "BLH", "BLD"];
                var found = false;
                for (var j = 0; j < skipNames.length; j++) found = found || (name.indexOf(skipNames[j]) == 0);
                if (!found) GM_deleteValue(name);
            }
        }

        //Remove skins
        if (saved[1] != currentA[1])
        {
            var names = GM_listValues();
            for (var i = 0; name = names[i], i < names.length; i++)
            {
                var found = false;

                if (name.indexOf("[SKIN]") == 0) GM_deleteValue(name);
            }

            GM_setValue("[SKIN] 0", "Default|B8FF8F,5EFF79,D4ECFF,ADADAD,666666,E3E3E3|122,FFFFFF,21FF3F,1453FF,5E5C5C,FFFF85,FF0000|6|0,000000,BCBCBB,FFA500,FF4500|2,0000FF,800080,000000,008000");
            GM_setValue("[SKIN] 1", "Nature|703700,A85600,D9A773,ADADAD,666666,E3E3E3|126,FFFFFF,D5FFB0,74A7E0,49698C,545454,D1EAFF|2|2,000000,FFFFFF,FF8800,FFFF00|2,944200,45BF11,000000,6DFF24");
            GM_setValue("[SKIN] 2", "Forest|115421,0C9656,733300,ADADAD,666666,E3E3E3|2,FFFFFF,2B2B2B,474747,A6A6A6,545454,ABB3C9|2|2,522801,FFFFFF,FF8800,FFFF00|2,02FF14,361800,4A9400,FF8800");
            GM_setValue("[SKIN] 3", "Going Green|4E6E3F,0B854C,0B730B,ADADAD,666666,E3E3E3|62,FFFFFF,01541D,6EFFA3,000000,FFFF00,ABB3C9|206|2,0B9114,FFFFFF,FF8800,FFFF00|2,02FF14,652D00,000000,FF8800");
            GM_setValue("[SKIN] 4", "Black & Blue|454545,333333,FFFFFF,ADADAD,666666,E3E3E3|30,FFFFFF,000000,00063D,04022E,545454,ABB3C9|154|2,000000,FFFFFF,FF8800,FFFF00|2,12B8FF,382EBF,A3A3A3,FFFF00");
            GM_setValue("[SKIN] 5", "White Orange|FFE561,FFC48A,FFFFFF,ADADAD,666666,E3E3E3|126,FFFFFF,FFAA2B,1453FF,5E5C5C,FFFF85,FF0000|6|2,FFFFFF,8C8C8C,FFA500,FF4500|2,0000FF,800080,000000,008000");
        }


        GM_setValue("Version", current);
    },

    GetPostfix: function ()
    {
        return " " + FindStringInArray(SEARCHTYPES.split(" "), SearchType).toString().lPad("0", 2) + ": ";
    },


    Load: function ()
    {
        Settings.VersionCheck("125.125");

        // ------------ Loading General Settings (Cross all pages)
        var vals = GM_getValue("General", "500,1500|126,6,0,0,2|6").split("|"); //14



        vals[0] = vals[0].split(",");
        Settings.paging = new Object();
        Settings.paging.offset = parseInt(vals[0][0]);
        Settings.paging.interval = parseInt(vals[0][1]);

        vals = vals[1].split(",");
        Settings.result = new Object();
        Settings.result.number = vals[0];
        Settings.result.wholerow = vals[1];
        Settings.result.remove = vals[2];
        Settings.result.group = vals[3];
        Settings.result.groupSize = vals[4];

        var flag = parseInt(vals[2]);
        Settings.removeTracking = (flag & 2);
        Settings.cleanURLs = (flag & 4);
        Settings.cookieScramble = (flag & 8);
        Settings.replaceSearchBox = (flag & 16);

        // ------------ Blacklist Settings
        if (!Settings.blacklist) //Does not get reset on refresh
        {
            Settings.blacklist = new Object();
            Settings.blacklist.sessionH = new Array();
            Settings.blacklist.sessionD = new Array();
        }
        Settings.blacklist.permanentH = new Array();
        Settings.blacklist.permanentD = new Array();

        vals = GM_getValue("BLH", "");
        if (vals) Settings.blacklist.permanentH = vals.split(",");
        vals = GM_getValue("BLD", "");
        if (vals) Settings.blacklist.permanentD = vals.split(",");

        // ------------ Loading Search Type Specific Settings


        vals = GM_getValue("Generic" + Settings.GetPostfix(), "0,1,4").split(",");
        flag = parseInt(vals[2]);
        Settings.columns = parseInt(vals[1]);
        Settings.paging.enabled = (flag & 2);
        Settings.blacklist.enabled = (flag & 4);
        Settings.pacify = (flag & 8);
        Settings.skin = Settings.GetSkin(vals[0]);
    },

    GetSkin: function (id)
    {
        var vals = GM_getValue("[SKIN] " + id, null);

        if (vals) vals = vals.split("|");
        else
        {
            return Settings.GetSkin(0);
        }

        var skin = new Object();
        skin.id = id;
        skin.name = vals[0];
        skin.general = vals[1].split(",");
        skin.border = vals[2].split(",");
        skin.transparency = vals[3].split(",");
        skin.bgc = vals[4].split(",");
        skin.font = vals[5].split(",");

        return skin;
    },

    Save: function ()
    {
        var flag = (Settings.removeTracking ? 2 : 0) + (Settings.cleanURLs ? 4 : 0) +
                (Settings.cookieScramble ? 8 : 0) + (Settings.replaceSearchBox ? 16 : 0);

        GM_setValue("General", Settings.paging.offset + "," + Settings.paging.interval + "|" + [Settings.result.number, Settings.result.wholerow, Settings.result.remove, Settings.result.group, Settings.result.groupSize].toString() + "|" + flag); //14

        flag = ((Settings.paging.enabled) ? 2 : 0) + ((Settings.blacklist.enabled) ? 4 : 0) + ((Settings.pacify) ? 8 : 0);
        GM_setValue("Generic" + Settings.GetPostfix(), Settings.skin.id + "," + Settings.columns + "," + flag);

        GM_setValue("BLH", Settings.blacklist.permanentH.toString());
        GM_setValue("BLD", Settings.blacklist.permanentD.toString());

        Settings.SaveSkin();
    },

    BoolToString: function (val)
    {
        return (val) ? 1 : 0;
    },

    SaveSkin: function ()
    {
        var s = Settings.skin;
        GM_setValue("[SKIN] " + s.id, s.name + "|" + s.general.toString() + "|" + s.border.toString() + "|" + s.transparency + "|" + s.bgc.toString() + "|" + s.font.toString());
    },

    DeleteSkin: function ()
    {
        GM_deleteValue("[SKIN] " + Settings.skin.id);
        Settings.skin = Settings.GetSkin(0); //Loads default skin
    },

    CompareSkins: function (a, b)
    {
        if (a.name == "Default") return -1;
        if (b.name == "Default") return 1;

        if (a.name < b.name) return -1;
        if (b.name < a.name) return 1;
        return 0;
    },

    GetSkinNames: function ()
    {
        var names = GM_listValues();

        var skins = new Array();

        for (var i = 0; name = names[i], i < names.length; i++)
        {
            if (name.indexOf("[SKIN] ") == 0)
            {
                var s = new Object();
                s.id = name.substring(7);
                s.name = Settings.GetSkin(s.id).name;
                skins.push(s);
            }
        }

        skins.sort(Settings.CompareSkins);
        return skins
    }

};


var Observe =
{
    bodyChanges: function ()
    {
        var mo = window.MutationObserver || window.MozMutationObserver || window.WebKitMutationObserver;

        if (mo)
        {
            Observe.observer = new mo(Observe.callback);
            Observe.observer.observe(document.body, { childList: true, subtree: true });
        }
    },

    callback: function (mutations)
    {

        GetSearchType();
        if (SearchType == SearchTypes.unknown)
        {
            clearTimeout(Observe.intervalID);
            clearInterval(Pager.intervalID);
        }
        else if (SearchType == SearchTypes.imageSearch)
        {
            console.warn("Images Search not yet implemented");
        }
        else
        {
            var rso = document.getElementById("rso");
            if (rso && !rso.getAttribute("parsing"))
            {
                rso.setAttribute("parsing", true);
                Observe.URL = document.URL;
                Pager.newPage();
                Pager.paused = true;

                MenuHQ.highlight(); //Safe search can be swapped without reloading full page.
            }

            Privacy.removeTracking(document);


            if (rso && rso.outerHTML.length != Observe.rsoLength)
            {
                Observe.rsoLength = rso.outerHTML.length;

                clearTimeout(Observe.intervalID);
                clearInterval(Pager.intervalID);

                Beautifier.addDocumentContent(document, false);
                Beautifier.removeAdvertNodes();
                InfoTab.AddRHSTab();
                rso.style.display = "none";

                //TODO: No need to this interval deletion but leave it for now.
                Observe.intervalID = setTimeout(function ()
                {
                    Pager.paused = false;
                    removeNode("rso"); //Can't remove it as it is still being used in the UK version. Stores navcnt there
                    if (Settings.paging.enabled && document.getElementById("pnnext")) Pager.intervalID = setInterval(Pager.loadNextPage, 250);

                    InfoTab.AddRHSTab(); //capture RHS if not captures. Issue with instant search
                }, 2000); // 5 second wait

                //document.getElementById("center_col").style.display = "none"; //Used by UK version. navcnt is stored there
            }
        }
    }
};

var Pager =
{
    newPage: function ()
    {
        console.info("New Page");
        Settings.Load();
        Pager.paused = true;
        PageCount = 0;
        Counter = 0;
        ResultItemCounter = 0;

        ResultItems.innerHTML = "";
        removeNode("ResultTable");

        Beautifier.addFixedStyles();
        Beautifier.addDynamicStyles();
        MenuHQ.addQuickMenu();
        if (document.getElementById("navcnt")) Privacy.cleanLinks(document.getElementById("navcnt"));

        var cur = document.evaluate('//div[@id="navcnt"]//td[@class="cur"]', document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
        StartPage = (cur) ? parseInt(cur.textContent) : 1;
    },


    loadNextPage: function ()
    {
        var scrolledLow;
        if (typeof window.scrollMaxY !== "number")
            scrolledLow = (document.documentElement.scrollHeight - document.documentElement.clientHeight - window.scrollY) < Settings.paging.offset
        else scrolledLow = (window.scrollMaxY - window.scrollY) < Settings.paging.offset;

        if (!(Settings.paging.enabled && !Pager.paused && scrolledLow)) return;

        var url = document.URL;
        clearInterval(Pager.intervalID);
        console.log(url);
        Pager.nextPageURL = document.getElementById("pnnext").href;


        var xhr = new XMLHttpRequest();
        xhr.open("GET", Pager.nextPageURL, true);
        xhr.responseType = "document";
        xhr.onload = function (e)
        {
            if (url != document.URL) return; //New document has been loaded. Should avoid adding next page            
            /* Chrome does not like this method of getting document
            if (xhr.readyState == 4 && xhr.status == 200) 
            {
                var xdt = document.implementation.createDocumentType("html", "-//W3C//DTD HTML 4.01 Transitional//EN", "http://www.w3.org/TR/html4/loose.dtd");
                var doc = document.implementation.createDocument("", "", xdt);
                var elHtml = doc.createElement("html");

                elHtml.innerHTML = xhr.responseText;
                doc.appendChild(elHtml);

                Pager.parseDocument(xhr.responseText);
            }
            */
            if (xhr.readyState == 4 && xhr.status == 200) Pager.parseDocument(xhr.response);

            else console.error(xhr.statusText);
        };
        xhr.send();
        //console.log(navigator.userAgent.match(/chrome/i));
    },

    parseDocument: function (doc)
    {


        //Remove existing image ids just in case of duplicates
        var imgs = document.evaluate(".//img[contains(@id,'vidthumb')]|.//img[contains(@id,'apthumb')]", document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
        for (i = 0; img = imgs.snapshotItem(i), i < imgs.snapshotLength; i++) img.removeAttribute('id');

        Privacy.removeTracking(doc);
        Beautifier.addDocumentContent(doc, false); //Add document content

        // ------- Handles Lazy Loading Images. Code is based on GoogleKingKongR script
        var scripts = doc.evaluate(".//div[@id='xfoot']/script", doc, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
        if (!scripts) scripts = doc.evaluate("./script", doc, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);

        for (i = 0; scripts && i < (scripts.snapshotLength) ; i++)
        {
            var scr = scripts.snapshotItem(i) && scripts.snapshotItem(i).innerHTML;
            if (scr.indexOf('thumb') != -1) eval(scr);
        }

        //Replace navigation panel
        var nav = doc.getElementById("navcnt");
        if (nav)
        {
            Privacy.cleanLinks(nav);
            document.getElementById("navcnt").outerHTML = nav.outerHTML;
            setTimeout(function () { if (Settings.paging.enabled) Pager.intervalID = setInterval(Pager.loadNextPage, 250); }, Settings.paging.interval);
        }
        else removeNode("navcnt");
    }
};


var Beautifier =
{
    addFixedStyles: function ()
    {
        AddStyle("SchmoogleContainer", "#schmoogleContainer {margin: 0px 25px 0px 25px;width: auto; }");
        AddStyle("AlterWidth", '.mw{ width:100%; max-width: 100% !important;}');
        AddStyle("SchmoogleFixed", '#QuickMenu{z-index: 1000;position: fixed;top: 150px;left: 2px;}#QuickMenu ul{margin: 0;padding: 0;list-style-type: none;text-align: center;word-wrap: break-word;}#QuickMenu input{font-size: x-small;padding: 0px;margin: 0px;background-color: lightgrey;min-width: 20px;}#QuickMenu input:single-button{font-size: x-small;border-radius: 4px;padding: 0px;margin: 0px;}#QuickMenu hr{margin: 2px 0;}.resultTable{max-width: 100% !important;}.resultTable > tbody > tr > td{border-radius: 3px;padding: 5px;vertical-align: top;}.CTShared *{word-break: break-all;white-space: normal;}.CTShared > li, .CTinfo > li{margin: 0 !important;list-style-type: none;padding: 10px;border-radius: 3px;}.CTinfo > li{background-color: white !important;}.milkyButton{text-decoration: none;color: #555 !important;background-color: #fcfcfc;font-family: sans-serif;font-size: 0.85em;margin: auto 2px;border: 1px solid #ccc;border-radius: 4px;padding: 2px 3px;background-position: center center;background-repeat: no-repeat;}.milkyButton:hover{color: #000;background: #ff8;background-position: center center;background-repeat: no-repeat;}#schmoogleInfoTab{z-index: 1000;position: absolute;top: 250px;right: 0px;border-style: ridge;padding: 5px 0 0 0;background-color: gray;background-color: lightgray;}#schmoogleInfoTab.itSelected{background-color: lime;border-color: yellow;}#schmoogleInfoTab:hover{background-color: yellow;border-color: lime;}.buttonsPanel{padding-top: 5px;}.buttonsPanel > span{text-align: center;vertical-align: middle;display: table-cell;margin: 0px;padding: 0px;}.contextMenu{position: fixed;list-style-type: none;border: solid lightgray;background-color: #F5F1F1;margin: 0;left: 30px;z-index: 500;color: black;}.contextMenu a, .contextMenu a:visited{padding: 8px 15px 8px 5px;display: inline-block;text-decoration: none;color: black;}.contextMenu li:hover{background-color: #D3D3D3;}.contextMenu hr{padding: 0;margin: 0;}');
        AddStyle("MissingImageStyles", ".birrg {font-size: 13px;overflow: hidden;}.rg_ul {white-space: normal;}.img-kc-m, .bi-io {}.bili {display: inline-block;margin: 0 6px 6px 0;overflow: hidden;position: relative;vertical-align: top;}.bicc {line-height: 0;overflow: hidden;position: relative;}.kno-cdta .kno-ibrg._Dl._Pj .img-brk {visibility: inherit;}.kno-cdta .kno-ibrg._Dl._Pj .img-brk {visibility: inherit;}.kno-cdta.xpdclose .kno-ibrg._Dl {max-height: inherit;}.kno-cdta ._pf {display: none;}.kno-mrg .kno-ibrg {float: left;}.kno-mrg-hnm .kno-ibrg {float: inherit;}.kno-mrg-hnm .kno-ibrg {display: block;}#iur .kno-ibrg {display: block;}.img-brk {display: block;overflow: hidden;}.bili .rg_meta {display: none;}");

        //Alteration of Google stlyes
        //TODO: Removed alteration of existing styles. Also removed margin:0 from vk_c as it breaks some items. 
        AddStyle("RHSSTyle", "#rhs {background-color:white;z-index: 20;}");
        AddStyle("AlterGoogle", ".CTinfo .vk_c{margin-left: 0 !important; margin-right: 0 !important;}");
        //AddStyle("AlterGoogle", ".vk_c{color:black;}.vk_c a{color:blue !important;}.vk_c a:visited{color:purple !important;}");                        
    },

    addDynamicStyles: function ()
    {
        removeNode("CSSMain");
        //CSSMain              
        var css = ".resultTable > tbody > tr:nth-child(odd) > td:nth-child(odd){background-color: F2;}.resultTable > tbody > tr:nth-child(odd) > td:nth-child(even){background-color: F4;}.resultTable > tbody > tr:nth-child(even) > td:nth-child(odd){background-color: F4;}.resultTable > tbody > tr:nth-child(even) > td:nth-child(even){background-color: F2;}.resultTable > tbody > tr > td > li{background-color: F8;}.permanentBan, .sessionBan{color: F64;padding: 5px 20px !important;}.permanentBan > div > h3, .permanentBan > div > h3{color: F64;}.sessionBan{background-color: F16;}.permanentBan{background-color: F32;}";
        for (var i = 0; i < Settings.skin.general.length; i++)
        {
            css = css.replace(new RegExp("F" + Math.pow(2, i + 1) + ";", "g"), "#" + Settings.skin.general[i] + ((i > 2) ? " !important;" : ";"));
        }
        AddStyle("CSSMain", css);

        removeNode("CSSBGC");
        removeNode("CSSFont");
        removeNode("CSSSpecific");

        //CSSBGC
        if (Settings.skin.bgc[0] > 0)
        {
            var css = "#main, #topbar, #hdtbSum, #appbar, #hdtbMenus{background-color: F2;}.hdtb_mitem a, #hdtb_more, #hdtb_tls, #appbar div, .hdtb-mn-cont div{color: F4;}.hdtb_mitem a:hover, #hdtb_more:hover, .hdtb_msel:hover, .hdtb-mn-cont div:hover{color: F8;border-color: F8;}#hdtb_tls:hover{color: #000 !important;}#hdtb_more_mn a{color: #808080 !important;}#hdtb_tls.hdtb-tl-sel{color: #000 !important;}.hdtb_msel{color: F16;border-color: F16;}";
            for (var i = 1; i < Settings.skin.bgc.length; i++)
            {
                css = css.replace(new RegExp("F" + Math.pow(2, i) + ";", "g"), "#" + Settings.skin.bgc[i] + " !important;");
            }
            AddStyle("CSSBGC", css);
        }

        //CSSFont
        if (Settings.skin.font[0] > 0)
        {
            var css = ".resultTable, .resultTable div{color: F8;}.resultTable a, .resultTable a.fl {color: F2;}.resultTable a:visited{color: F4;}.resultTable cite{color: F16;}";
            for (var i = 1; i < Settings.skin.font.length; i++)
            {
                css = css.replace(new RegExp("F" + Math.pow(2, i) + ";", "g"), "#" + Settings.skin.font[i] + ";");
            }
            AddStyle("CSSFont", css);
        }

        //CSSSpecific
        css = "";
        var ct = "unused info images videos news places unknown standard".split(" "); //Must imitate ContentTypes
        if (Settings.skin.border[0] > 0)
        {
            for (var i = 1; i < Settings.skin.border.length; i++)
            {
                if (Math.pow(2, i) & Settings.skin.border[0])
                {
                    css += ".CT" + ct[i] + " {background-color:#" + Settings.skin.border[i] + " !important;}";
                }
            }
        }

        if (Settings.skin.transparency > 0)
        {
            //document "skinBorderTran"
            for (var i = 1; i < ct.length; i++)
            {
                if (Math.pow(2, i) & Settings.skin.transparency)
                {
                    css += ".CT" + ct[i] + " > li{background-color:transparent !important;}";
                }
            }
        }
        if (css.length > 0) AddStyle("CSSSpecific", css);
    },

    addCell: function (el, type, clone, fixedSize)
    {
        if (el.clientWidth > 50 && (type == ContentTypes.info || fixedSize))
        {
            el.style.width = el.clientWidth + "px";
            el.style.height = el.clientHeight + "px";
        }

        var cell = document.createElement("td");
        cell.setAttribute("index", ResultItemCounter++);
        ResultItems.appendChild(cell);

        cell.setAttribute("contentType", type);
        cell.className = "CT" + type + ((type != ContentTypes.info) ? " CTShared" : "");

        cell.setAttribute("name", "resultItem");
        cell.setAttribute("PageNumber", PageCount);


        if (el.tagName != "LI")
        {
            li = document.createElement("li");
            cell.appendChild(li);
            if (clone) li.appendChild(el.cloneNode(true))
            else li.appendChild(el);
        }
        if (clone) cell.appendChild(el.cloneNode(true));
        else cell.appendChild(el);

        return cell;
    },


    addDocumentContent: function (doc, clone)
    {
        PageCount++;
        console.log("AddDocuments");

        var el = document.evaluate("//div[@id='res']//div[@id='topstuff']//div[contains(@class,'vk_c')]", document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
        if (el) Beautifier.addCell(document.getElementById("topstuff"), ContentTypes.info, clone, true);

        el = document.evaluate("//div[@class='med']/p[@class='sp_cnt']", document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
        if (el) Beautifier.addCell(el, ContentTypes.info, clone, true, false);

        Beautifier.parseRSOListItems(doc.getElementById("rso"), clone);
        Beautifier.addCellsToTable();
    },

    parseRSOListItems: function (rso, clone)
    {
        var lis = rso.children;
        var i = 0;
        while (lis.length > 0 && i < lis.length)
        {
            var li;
            if (clone) li = lis[i++]
            else li = lis[0];

            var type = Beautifier.getContentType(li);
            if (li.tagName == "LI")
            {
                var unwantedClasses = ["mas-sc-row"];
                for (var j = 0; j < unwantedClasses.length; j++)
                {
                    var els = li.getElementsByClassName("mas-sc-row");
                    for (var n = 0; n < els.length; n++) els[n].className = els[n].className.replace("mas-sc-row", "");
                }

                var type = Beautifier.getContentType(li);
                if (!type) removeNode(li);
                else if (type == ContentTypes.moreLinks && ResultItems.lastElementChild)
                {
                    li.style.marginTop = "2px";
                    li.style.padding = "10px";
                    if (clone) ResultItems.lastElementChild.appendChild(li.cloneNode(true));
                    else ResultItems.lastElementChild.appendChild(li);
                }
                else
                {
                    var cell = Beautifier.addCell(li, type, clone, false);
                    if (type != ContentTypes.info) Beautifier.addButtons(cell);
                }
            }
            else if (li.tagName == "DIV") //Since April (2013) Google has included result in grouping inside a DIV
            {
                Beautifier.parseRSOListItems(li);
                if (!clone)
                {
                    console.warn("Removing grouping div");
                    console.log(li);
                    console.log(li.innerHTML);
                    removeNode(li); //remove it no matter what is left
                }
            }
            else if (!clone)
            {
                console.error("List items not captured! Removing it.");
                console.log(li);
                console.log(li.innerHTML);
                removeNode(li);
            }
            else console.error("Cloned item not captured!");
        }
    },


    typeFilter: function (type, flag, minus)
    {
        if (isNaN(minus)) minus = 0;

        var ct = "info images videos news places unknown standard".split(" ");
        var pos = FindStringInArray(ct, type) - minus + 1;

        return (pos > 0 && (Math.pow(2, pos) & flag));
    },

    addCellNumber: function (cell)
    {
        var type = Beautifier.getContentType(cell);

        if (type != ContentTypes.standard && !Beautifier.typeFilter(type, Settings.result.number)) return;

        var position = cell.getElementsByClassName("ItemPos")[0];
        if (!position)
        {
            var h3 = cell.getElementsByTagName("h3")[0];
            if (h3)
            {
                position = document.createElement("strong");
                position.className = "ItemPos";
                h3.insertBefore(position, h3.firstElementChild);
            }
        }

        if (position) position.textContent = ++Counter + " ";
    },

    createBlacklistCell: function (cell, row, isSessionBan)
    {
        var cellBL = document.createElement("td");

        var evaluator = new XPathEvaluator();
        if (cell.getAttribute("name") == "resultItem")
        {
            cellBL.innerHTML = "<div><h3>" + cell.getElementsByTagName("h3")[0].firstElementChild.innerHTML + "</h3></div>";
        }
        else
        {
            var nodesSnapshot = evaluator.evaluate(".//td[@name='resultItem']//h3/a", cell, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null)
            for (var i = 0; i < nodesSnapshot.snapshotLength; i++)
            {
                cellBL.innerHTML += "<div style='padding: 2px 0;'><h3>" + nodesSnapshot.snapshotItem(i).innerHTML + "</h3></div>";
            }
        }

        cellBL.className = (isSessionBan) ? "sessionBan" : "permanentBan";
        cellBL.colSpan = Settings.columns;

        var t = Beautifier.createResultTable();
        var row2 = t.insertRow(-1);
        row2.appendChild(cell);
        t.style.display = "none";
        cellBL.appendChild(t);

        cellBL.onclick = function (e)
        {
            clearTimeout(this.getAttribute("stID"));
            this.lastElementChild.style.display = (this.lastElementChild.style.display) ? null : "none";
        };

        /* Hide when mouse has left cell
        cellBL.onmouseover = function (e)
        {
            clearTimeout(this.getAttribute("stID"));
        };


        cellBL.onmouseout = function (e)
        {
            clearTimeout(this.getAttribute("stID"));
            var stID = setTimeout(function (el)
            {
                el.lastElementChild.style.display = "none";
            }, 1000, this);

            this.setAttribute("stID", stID);
        };
        */

        return cellBL;
    },

    addCellsToTable: function ()
    {
        var table = Beautifier.getResultTable();

        var row = null;

        //This statement deals with the consequence of having content that takes a whole row.
        if (table.rows.length == 0 || Settings.columns == 1) row = table.insertRow(-1);
        else if (table.rows[table.rows.length - 1].cells.length == 0 || !table.rows[table.rows.length - 1].cells[0].getAttribute("colspan")) row = table.rows[table.rows.length - 1];
        else for (var i = 0; i < table.rows.length; i++)
            {
            if (table.rows[i].cells.length < Settings.columns && !table.rows[i].cells[0].getAttribute("colspan"))
            {
                row = table.rows[i];
                break;
            }
        }
        if (row == null) row = table.insertRow(-1);


        if (Settings.columns == 1) removeNode("TDWidths");
        else if (Settings.columns == 2) AddStyle("TDWidths", "#ResultTable > tbody > tr > td { width: 50%; }");
        else if (Settings.columns == 3) AddStyle("TDWidths", "#ResultTable > tbody > tr > td { width: 33%; }");
        else if (Settings.columns == 4) AddStyle("TDWidths", "#ResultTable > tbody > tr > td { width: 25%; }");

        while (ResultItems.firstElementChild)
        {

            if (row.cells.length == Settings.columns) row = table.insertRow(-1);

            while (ResultItems.firstElementChild && row.cells.length < Settings.columns)
            {
                var cell = ResultItems.firstElementChild;
                var type = Beautifier.getContentType(cell);
                var isWebSearch = (SearchType == SearchTypes.webSearch);

                if (Beautifier.typeFilter(type, Settings.result.remove) && isWebSearch)
                {
                    document.getElementById("schmoogleContainer").appendChild(cell);
                    cell.style.display = "none";
                    continue;
                }

                var group = isWebSearch && Beautifier.typeFilter(type, Settings.result.group, 2) && cell.nextElementSibling && type == Beautifier.getContentType(cell.nextElementSibling);

                if (group)
                {
                    //We are trying to divide the grouping to even out the size.
                    //Example if grouping is 5 and you have 6 videos it will
                    //split them 3/4 instead of 5/1.
                    var countTotal = 0;
                    var BLItems = document.createElement("div");
                    var BLCount = 0;
                    var maxsize = Settings.result.groupSize;
                    do
                    {
                        var next = cell.nextElementSibling;

                        var filter = cell.getElementsByClassName("imgbtn")[0];
                        if (filter) filter = FilterSystem.isBlacklisted(filter.metadata);
                        if (filter > 0)
                        {
                            BLItems.appendChild(cell);
                            BLCount++;
                        }

                        countTotal++;
                        cell = next;
                    } while (cell && Beautifier.getContentType(cell) == type);


                    var onlyBLItems = (BLCount == countTotal);
                    while (BLItems.children.length > 0)
                    {
                        ResultItems.insertBefore(BLItems.firstElementChild, cell);
                    }

                    if (countTotal > maxsize)
                    {
                        d = Math.ceil(countTotal / maxsize);
                        maxsize = Math.ceil(countTotal / d); //Get the new maxsize
                    }


                    var count = 0;
                    var t = Beautifier.createResultTable();
                    while (count < countTotal && ((count < maxsize && (count + BLCount) < countTotal) || onlyBLItems))
                    {
                        if (!onlyBLItems) Beautifier.addCellNumber(ResultItems.firstElementChild);
                        t.insertRow(-1).appendChild(ResultItems.firstElementChild);
                        count++;
                    }

                    cell = document.createElement("td");
                    cell.className = "CT" + type;
                    t.setAttribute("style", "width: 100%;");
                    cell.appendChild(t);
                }

                var filter = cell.getElementsByClassName("imgbtn")[0];
                if (filter)
                {
                    filter = FilterSystem.isBlacklisted(filter.metadata);
                    if (filter > 0)
                    {
                        cell = Beautifier.createBlacklistCell(cell, row, filter == 2);

                        if (row.cells.length > 0)
                        {
                            row2 = table.insertRow(-1);
                            row2.appendChild(cell);
                        }
                        else
                        {
                            row.appendChild(cell);
                            row = table.insertRow(-1);
                        }

                        continue;
                    }
                }


                if (!group) Beautifier.addCellNumber(cell);

                if (isWebSearch && Settings.columns > 1 && Beautifier.typeFilter(type, Settings.result.wholerow))
                {
                    var d = document.createElement("div");
                    d.style.display = "inline-block";
                    var t = Beautifier.createResultTable();
                    var r = t.insertRow(-1);
                    r.appendChild(cell);
                    t.style.minWidth = "520px";
                    d.appendChild(t);

                    cell = document.createElement("td");
                    cell.appendChild(d);

                    cell.colSpan = Settings.columns;
                    cell.style.textAlign = "center";
                    cell.className = "CT" + type;

                    //cell.firstElementChild.style.textAlign = "left";
                    if (row.cells.length > 0)
                    {
                        var row2 = table.insertRow(-1);
                        row2.appendChild(cell);
                    }
                    else
                    {
                        row.appendChild(cell);
                        row = table.insertRow(-1);
                    }
                }
                else
                {
                    row.appendChild(cell);
                }
            }
        }
    },

    refreshTable: function ()
    {
        Pager.paused = true;
        Settings.Load();
        Beautifier.addDynamicStyles();
        var ritems = document.getElementsByName("resultItem");

        while (ritems.length > 0)
        {
            var cell = ritems[0];
            var position = cell.getElementsByClassName("ItemPos")[0]; //Removes numbering
            if (position) removeNode(position);
            cell.removeAttribute("colspan");
            cell.style.textAlign = null;
            cell.style.display = null;

            var inserted = false;

            //Sort the result items to get original position as it may be altered with removal of results
            for (var i = 0; i < ResultItems.children.length; i++)
            {
                if (parseInt(ResultItems.children[i].getAttribute("index")) > parseInt(cell.getAttribute("index")))
                {
                    inserted = true;
                    ResultItems.insertBefore(cell, ResultItems.children[i]);
                    break;
                }
            }

            if (!inserted) ResultItems.appendChild(cell);
        }

        Counter = 0;
        removeNode("ResultTable");
        Beautifier.addCellsToTable();

        Pager.paused = false;
    },

    getContentType: function (li)
    {
        var type = li.getAttribute("contentType");
        if (type) return type;

        type = ContentTypes.unknown;

        if (SearchType == SearchTypes.placesSearch) type = ContentTypes.standard;
        else if (li.className.indexOf("g tpo") != -1) type = ContentTypes.info;
        else if (SearchType == SearchTypes.videoSearch && li.className == "g videobox") type = ContentTypes.standard;
        else if (SearchType == SearchTypes.newsSearch && li.id.indexOf("esc-story") != -1) type = ContentTypes.standard;
        else

            switch (li.id)
            {
                case "newsbox":
                    type = ContentTypes.news;
                    break;
                case "imagebox_bigimages":
                    type = ContentTypes.images;
                    break;
                default:
                    if (li.childElementCount > 0)
                    {
                        //console.warn(li.firstElementChild.className);
                        switch (li.firstElementChild.className)
                        {
                            case "vsc": //Standard result item                            
                                type = ContentTypes.standard;
                                break;
                            case "vsc vslru": //Places result item
                                type = ContentTypes.places;
                                break;
                            case "s": //information                            
                                type = ContentTypes.info;
                                break;
                            case "fl": //Link to more videos. 
                                type = ContentTypes.moreLinks;
                                break;
                            case "dict vk_c vk_bk":
                            case "dict vk_c":
                                //case "sp_cnt": //Spell correction
                                type = ContentTypes.info;
                                break;
                            default: //Do not know what the heck it is                                
                                //More links most likely found in news search
                                if (li.childElementCount == 1 && li.firstElementChild.tagName == "A" && li.textContent.match(/ »$/gi))
                                {
                                    type = ContentTypes.moreLinks;
                                }
                                else if (li.firstElementChild && li.firstElementChild.tagName == "H3" && li.firstElementChild.childElementCount == 0)
                                {
                                    //Most likely text that say "Results for similar searches". Used by Google Grouping result                                    
                                    return null; //TODO: Need a better way of handling grouping
                                }
                                else
                                {
                                    type = (SearchType == SearchTypes.newsSearch || li.firstElementChild.className == "rc") ? ContentTypes.standard : ContentTypes.unknown;

                                    if (SearchType == SearchTypes.webSearch)
                                    {
                                        var imgs = li.getElementsByTagName("img");
                                        if (imgs.length == 1 && imgs[0].id.indexOf("vidthumb") == 0)
                                        {
                                            type = ContentTypes.videos;
                                        }
                                    }
                                    break;
                                }
                        }
                    }
                    break;
            }

        return type;
    },

    createResultTable: function ()
    {
        var table = document.createElement("table");
        table.className = "resultTable";
        return table;
    },

    getResultTable: function ()
    {
        var table = document.getElementById("ResultTable");

        if (IsNull(table))
        {
            var schmoogle = document.getElementById("schmoogleContainer");
            if (!schmoogle)
            {
                var schmoogle = document.createElement("div");
                schmoogle.id = "schmoogleContainer";
                var rcnt = document.getElementById("rcnt")
                rcnt.insertBefore(schmoogle, rcnt.firstElementChild);
            }

            table = Beautifier.createResultTable();
            table.id = "ResultTable";
            schmoogle.appendChild(table);
        }
        return table;
    },


    createButton: function (name, metadata, imgsrc)
    {
        var link = document.createElement("a");
        link.name = name;
        link.metadata = metadata;
        link.href = "#";
        link.className = "imgbtn";

        var img = document.createElement("img");
        img.setAttribute("src", imgsrc);
        link.appendChild(img);

        link.onclick = FilterSystem.buttonPressed;
        return link;
    },

    addButtons: function (cell)
    {
        var cashedLink = null;
        var similarLink = null;
        var div = document.createElement("div");
        div.className = "buttonsPanel";
        cell.lastElementChild.appendChild(div);

        AddStyle("liButtons", "text-align: center; vertical-align: middle; display: table-cell; margin: 0px; padding: 0px;");

        var evaluator = new XPathEvaluator();
        var dropmenu = evaluator.evaluate(".//a[@class='clickable-dropdown-arrow ab_button']", cell, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
        var droppanel = evaluator.evaluate(".//div[@class='action-menu-panel ab_dropdown']", cell, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;

        if (dropmenu && droppanel)
        {
            var links = droppanel.getElementsByTagName("a");

            for (var i = 0; i < links.length ; i++)
            {
                var span = document.createElement("span");
                var a = document.createElement("a");
                //a.setAttribute("role", "button");
                a.className = "milkyButton";
                a.textContent = links[i].textContent;
                a.href = links[i].href;
                span.appendChild(a);
                div.appendChild(span);
            }

            dropmenu.style.display = "none"; //Hide as if I remove it causes errors          
        }

        if (SearchType != SearchTypes.bookSearch && FindStringInArray(["info", "images", "videos"], cell.getAttribute("contenttype")) < 0)
        {
            var evaluator = new XPathEvaluator();
            var node = evaluator.evaluate(".//h3/a", cell, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;

            if (node)
            {
                var hostname = node.hostname;
                var span = document.createElement("span");


                span.appendChild(
                 Beautifier.createButton("Filter", hostname, "")
                 );

                span.appendChild(
                Beautifier.createButton("SBan", hostname, "")
                );

                span.appendChild(
                Beautifier.createButton("PBan", hostname, "")
                );

                span.style.paddingTop = "5px";
                div.appendChild(span);
            }
        }
    },

    removeAdvertNodes: function ()
    {

        removeNode("tads"); //Top Adverts    
        removeNode("bottomads"); //Bottom Adverts
        removeNode("mbEnd"); //Side Adverts
        removeNode("extrares"); //Related Search Words
        removeNode("botstuff"); //Related Search Words#
        removeNode("tvcap");
        //removeNode("res"); Breaks google update using topstuff instead
        //removeNode("topstuff"); //Removes related word search that occurs at top
        //removeNode("taw"); //Removes search word correction
    }
};

var FilterSystem =
{
    buttonPressed: function (e)
    {
        var btn = e.target;

        while (btn.tagName != "A") btn = btn.parentElement;

        switch (btn.name)
        {
            case "SBan":
                FilterSystem.showBlackListWindow(e, btn.metadata, true);
                break;
            case "PBan":
                FilterSystem.showBlackListWindow(e, btn.metadata, false);
                break;
            default:
                var ids = ["gbqfq", "lst-ib", "sbhost"];
                for (var i = 0; el = document.getElementById(ids[i]), i < ids.length; i++)
                {
                    if (el)
                    {
                        //Does not remove spaces for some unknown reason
                        var newFilter = "-site:" + btn.metadata;
                        if (el.value.indexOf(newFilter) == -1) el.value = el.value.replace(/(\s+)?$/, " ") + newFilter;
                        else el.value = el.value.replace(new RegExp("\s*" + newFilter, "i"), "");
                        break;
                    }
                }
                break;
        }
        e.stopPropagation();
        return false;
    },

    showBlackListWindow: function (e, hostname, isSessionBan)
    {
        var blw = document.getElementById("BLWindow");
        if (!blw)
        {
            blw = AddHTMLElement('<div id="BLWindow" style="background-color: white; width: 250px; padding: 5px; border: ridge; border-radius: 5px; position: absolute; left: 100px; top: 100px; z-index: 999;"><p style="text-align: center"><strong id="BLTitle"></strong></p><p><div><input name="shoBL" type="checkbox" value="hostname" /><label>www.hostname.com</label></div><div><input name="shoBL" type="checkbox" value="domain" /><label>domain.com</label></div></p><div style="text-align: center;"><input id="BLBtn" type="submit" value="OK" style="width: 75px" /></div></div>');

            document.body.appendChild(blw);
            document.getElementById("BLBtn").onclick = FilterSystem.blacklistCallback;
        }

        var pos = FilterSystem.GetEventsPagePosition(e);
        blw.style.left = pos.x + "px";
        blw.style.top = pos.y + "px";

        var lbls = blw.getElementsByTagName("label");
        lbls[0].textContent = hostname;
        lbls[1].textContent = GetDomain(hostname);

        els = blw.getElementsByTagName("input"); //checkboxes

        var arr = (isSessionBan) ? Settings.blacklist.sessionH : Settings.blacklist.permanentH;
        var i = FindStringInArray(arr, hostname);
        if (i >= 0) els[0].checked = true;


        arr = (isSessionBan) ? Settings.blacklist.sessionD : Settings.blacklist.permanentD;
        var i = FindStringInArray(arr, lbls[1].textContent);
        if (i >= 0) els[1].checked = true;

        if (isSessionBan)
        {
            document.getElementById("BLTitle").textContent = "Session Blacklist";
            blw.setAttribute("session", "true");
        }
        else
        {
            document.getElementById("BLTitle").textContent = "Permanent Blacklist";
            blw.removeAttribute("session");
        }

    },

    blacklistCallback: function (e)
    {
        var blw = document.getElementById("BLWindow");

        var els = blw.getElementsByTagName("label");
        var hostname = els[0].textContent;
        var domain = els[1].textContent;

        els = blw.getElementsByTagName("input"); //checkboxes

        var isSessionBan = blw.getAttribute("session");

        var changed = false;
        if (els[0].checked) changed = changed || FilterSystem.blacklistAdd(hostname, true, isSessionBan);
        else changed = changed || FilterSystem.blacklistRemove(hostname, true, isSessionBan);

        if (els[1].checked) changed = changed || FilterSystem.blacklistAdd(domain, false, isSessionBan);
        else changed = changed || FilterSystem.blacklistRemove(domain, false, isSessionBan);

        if (!blw.getAttribute("session")) Settings.Save();
        if (changed) Beautifier.refreshTable();
        removeNode(blw);
    },

    blacklistAdd: function (name, isHostname, isSessionBan)
    {
        var arr;
        if (isHostname) arr = (isSessionBan) ? Settings.blacklist.sessionH : Settings.blacklist.permanentH;
        else arr = (isSessionBan) ? Settings.blacklist.sessionD : Settings.blacklist.permanentD;
        var l = arr.length;

        var i = FindStringInArray(arr, name);
        if (i < 0) arr.push(name);
        return (arr.length != l)
    },

    blacklistRemove: function (name, isHostname, isSessionBan)
    {
        var arr;

        if (isHostname) arr = (isSessionBan) ? Settings.blacklist.sessionH : Settings.blacklist.permanentH;
        else arr = (isSessionBan) ? Settings.blacklist.sessionD : Settings.blacklist.permanentD;
        var l = arr.length;

        var i = FindStringInArray(arr, name);
        if (i >= 0) arr.splice(i, 1);

        return (arr.length != l)
    },

    /* Checks if hostname is blackslisted
    --------------------------------------------- 
     0: Not blacklisted
     1: permanent blacklist
     2: session blacklist */
    isBlacklisted: function (name)
    {
        if (!Settings.blacklist.enabled) return 0;
        if (MatchArrayStringToString(Settings.blacklist.permanentH, name, 8) >= 0) return 1;
        if (MatchArrayStringToString(Settings.blacklist.permanentD, name, 8, ".") >= 0) return 1;

        if (MatchArrayStringToString(Settings.blacklist.sessionH, name, 8) >= 0) return 2;
        if (MatchArrayStringToString(Settings.blacklist.sessionD, name, 8, ".") >= 0) return 2;
        return 0;
    },

    GetEventsPagePosition: function (e)
    {
        var posx = 0;
        var posy = 0;
        if (!e) var e = window.event;
        if (e.pageX || e.pageY)
        {
            posx = e.pageX;
            posy = e.pageY;
        }
        else if (e.clientX || e.clientY)
        {
            posx = e.clientX + document.body.scrollLeft
                + document.documentElement.scrollLeft;
            posy = e.clientY + document.body.scrollTop
                + document.documentElement.scrollTop;
        }

        var pos = new Object();
        pos.x = posx;
        pos.y = posy;
        return pos;
    }
}

var InfoTab =
{
    AddRHSTab: function ()
    {
        removeNode("schmoogleInfoTab");

        var rhs = document.getElementById("rhs");
        if (rhs && rhs.innerHTML.length > 300)
        {
            //TODO: Try and fix RHS positioning            
            var w = rhs.clientWidth;

            var tab = AddHTMLElement('<a href="#" id="schmoogleInfoTab"><img alt="" src="" /></a>');
            tab.onclick = InfoTab.TabClick;
            document.body.appendChild(tab);

            rhs.setAttribute("style", "position: absolute; width: " + w + "px; right: 25px; top: 0px; z-index: 500");
            rhs.style.visibility = "hidden";
        }
        else if (rhs) rhs.style.visibility = "hidden";
    },

    TabClick: function (e)
    {
        rhs = document.getElementById("rhs");
        var alink = (e.target.tagName == "A") ? e.target : e.target.parentElement;

        if (rhs.style.visibility == "hidden")
        {
            rhs.style.visibility = null;
            alink.className = "itSelected";
        }
        else
        {
            rhs.style.visibility = "hidden";
            alink.className = null;
        }

        return false;
    }
};

var SettingsWindow =
{
    showWindow: function ()
    {
        if (document.getElementById("Settings") != null) return true;

        if (document.getElementById("schmoogleSW") == null)
            AddStyle("schmoogleSW", '#Settings{width: 450px;background-color: floralwhite;display: table-cell;vertical-align: middle;text-align: center;border: 3px double black;z-index: 1001;position: fixed;right: 10px;top: 10px;}#SettingsMain{background-color: #F2FBFF;font-size: small;}#SettingsMain th{margin: 0;padding: 0;width: 70px;text-align: center;}#SettingsMain td{margin: 0;padding: 0;text-align: center;vertical-align: middle;}#SettingsMain td a{display: inline-block;margin-top: 2px;}#SettingsMain tr:nth-child(odd){background-color: #C0EEFD;}#SettingsMain tr:nth-child(even ){background-color: #D5BBFD;}#SettingsMain th{background-color: lightcoral;}.bgiL, .cellOn, .btOn{background-color: rgb(0,255,0);background-image: linear-gradient(bottom,rgb(200,255,200) 0%, rgb(0,255,0) 50%, rgb(200,255,200) 100%);background-image: -o-linear-gradient(bottom, rgb(200,255,200) 0%, rgb(0,255,0) 50%, rgb(200,255,200) 100%);background-image: -moz-linear-gradient(bottom, rgb(200,255,200) 0%, rgb(0,255,0) 50%, rgb(200,255,200) 100%);background-image: -webkit-linear-gradient(bottom, rgb(200,255,200) 0%, rgb(0,255,0) 50%, rgb(200,255,200) 100%);background-image: -ms-linear-gradient(bottom, rgb(200,255,200) 0%, rgb(0,255,0) 50%, rgb(200,255,200) 100%);background-image: -webkit-gradient( linear, left bottom, left top, color-stop(0, rgb(200,255,200)), color-stop(0.5, rgb(0,255,0) ), color-stop(1, rgb(200,255,200)) );}.bgiG, .cellOff, .btOff{background-color: rgb(156,156,156);background-image: linear-gradient(bottom, rgb(238,238,238) 0%, rgb(156,156,156) 50%, rgb(238,238,238) 100%);background-image: -o-linear-gradient(bottom, rgb(238,238,238) 0%, rgb(156,156,156) 50%, rgb(238,238,238) 100%);background-image: -moz-linear-gradient(bottom, rgb(238,238,238) 0%, rgb(156,156,156) 50%, rgb(238,238,238) 100%);background-image: -webkit-linear-gradient(bottom, rgb(238,238,238) 0%, rgb(156,156,156) 50%, rgb(238,238,238) 100%);background-image: -ms-linear-gradient(bottom, rgb(238,238,238) 0%, rgb(156,156,156) 50%, rgb(238,238,238) 100%);background-image: -webkit-gradient( linear, left bottom, left top, color-stop(0, rgb(238,238,238)), color-stop(0.5, rgb(156,156,156)), color-stop(1, rgb(238,238,238)) );}.cellOn, .cellOff{border-radius: 10px;display: inline-block;height: 15px;width: 50px;}.IBL{display: inline-block;text-align: left;}.TabMenu{width: 100%;border-collapse: collapse;}.TabMenu td{width: 20%;padding: 0;}.TabMenu input{background-color: #EDF0F0;width: 100%;}.ColourTable{font-size: small;background-color: #74A7B8;display: inline-block;}.ColourTable td input{font-size: x-small;text-align: right;width: 45px;}.ColourTable th{text-align: center;background-color: lightblue;}.ColourRow{text-align: center;background-color: #3F89A2;}.ColourRowTD > td{text-align: center;background-color: #094D64;}.btOn, .btOff{display: inline-block;width: 30px;height: 12px;border-radius: 2px;margin: 2px 3px 0 3px;}#BLDiv{height: 300px;overflow-y: auto;padding: 0 5px;}#BLTable{width: 100%;height: 100%;background-color: #FFFFD6;}#BLTable th{background-color: #FFD3DA;height: 25px;}#BLTable td{font-size: small;width: 50%;text-align: left;padding-right: 5px;vertical-align: top;}#BLTable tr td:nth-child(2n){background-color: #D2FFB2;}#BLTable td:nth-child(2n+1){background-color: #CBE9F3;}#BLTable button{font-size: x-small;margin-right: 5px;}');

        var sw = AddHTMLElement('<section id="Settings"><div style="margin-bottom: 10px;"><table class="TabMenu"><tbody><tr><td><input name="MTab" type="button" value="Main" /></td><td><input name="MTab" type="button" value="Skins" /></td><td><input name="MTab" type="button" value="Blacklist" /></td><td><input name="MTab" type="button" value="Other" /></td><td><input name="MTab" type="button" value="About" /></td></tr></tbody></table></div><div id="MainMenu" name="MTabW" class="IBL"><div><input id="PInterval" type="text" value="1500" style="text-align: right; width: 40px; margin: 0 2px;" />Wait period between page fetching</div><table id="SettingsMain"><tbody><tr><th>Content</th><th>Number</th><th>Whole Row</th><th>Remove</th><th>Group</th></tr><tr><th>Info</th><td><a href="#"><div class="cellOn" name="cNumber"></div></a></td><td><a href="#"><div class="cellOff" name="cWholeRow"></div></a></td><td><a href="#"><div class="cellOff" name="cRemove"></div></a></td></tr><tr><th>Images</th><td><a href="#"><div class="cellOn" name="cNumber"></div></a></td><td><a href="#"><div class="cellOff" name="cWholeRow"></div></a></td><td><a href="#"><div class="cellOff" name="cRemove"></div></a></td></tr><tr><th>Videos</th><td><a href="#"><div class="cellOn" name="cNumber"></div></a></td><td><a href="#"><div class="cellOff" name="cWholeRow"></div></a></td><td><a href="#"><div class="cellOff" name="cRemove"></div></a></td><td><a href="#"><div class="cellOff" name="cGroup"></div></a></td></tr><tr><th>News</th><td><a href="#"><div class="cellOn" name="cNumber"></div></a></td><td><a href="#"><div class="cellOff" name="cWholeRow"></div></a></td><td><a href="#"><div class="cellOff" name="cRemove"></div></a></td><td><a href="#"><div class="cellOff" name="cGroup"></div></a></td></tr><tr><th>Places</th><td><a href="#"><div class="cellOn" name="cNumber"></div></a></td><td><a href="#"><div class="cellOff" name="cWholeRow"></div></a></td><td><a href="#"><div class="cellOff" name="cRemove"></div></a></td><td><a href="#"><div class="cellOff" name="cGroup"></div></a></td></tr><tr><th>Unknown</th><td><a href="#"><div class="cellOn" name="cNumber"></div></a></td><td><a href="#"><div class="cellOff" name="cWholeRow"></div></a></td><td><a href="#"><div class="cellOff" name="cRemove"></div></a></td><td><select id="GroupSel"><option selected="selected">2</option><option>3</option><option>4</option><option>5</option><option>6</option><option>7</option><option>8</option><option>9</option></select></td></tr></tbody></table></div><div id="SkinsMenu" name="MTabW" class="IBL" style="text-align: left; margin: 0 5px 0 5px;"><div style="text-align: right; margin-bottom: 5px;"><select id="skinSel" style="width: 100%"></select><input type="button" value="New" /><input type="button" value="Delete" id="deleteSkin" /></div><div class="IBL"><table class="ColourTable"><tbody><tr class="ColourRow"><td colspan="7">General Colours</td></tr><tr><th>Result 1</th><th>Result 2</th><th>Inner BGC</th><th>Session Ban</th><th>Perm Ban</th><th>Ban Font</th></tr><tr id="skinGeneral" class="ColourRowTD"><td><input class="jscolorGM" /></td><td><input class="jscolorGM" /></td><td><input class="jscolorGM" /></td><td><input class="jscolorGM" /></td><td><input class="jscolorGM" /></td><td><input class="jscolorGM" /></td></tr><tr class="ColourRow"><td colspan="7">Specific Border Colours</td></tr><tr><th>Info</th><th>Image</th><th>Videos</th><th>News</th><th>Places</th><th>Unknown</th></tr><tr id="skinBorderColor" class="ColourRowTD"><td><a href="#"><div class="btOn"></div></a><input class="jscolorGM" /></td><td><a href="#"><div class="btOn"></div></a><input class="jscolorGM" /></td><td><a href="#"><div class="btOn"></div></a><input class="jscolorGM" /></td><td><a href="#"><div class="btOn"></div></a><input class="jscolorGM" /></td><td><a href="#"><div class="btOn"></div></a><input class="jscolorGM" /></td><td><a href="#"><div class="btOn"></div></a><input class="jscolorGM" /></td><td style="visibility: collapse; width: 50px;"></td></tr><tr class="ColourRow"><td colspan="7">Transparent Inner Result</td></tr><tr><th>Info</th><th>Images</th><th>Videos</th><th>News</th><th>Places</th><th>Unknown</th><th>Result</th></tr><tr id="skinBorderTrans" class="ColourRowTD"><td><a href="#"><div class="btOn"></div></a></td><td><a href="#"><div class="btOn"></div></a></td><td><a href="#"><div class="btOn"></div></a></td><td><a href="#"><div class="btOn"></div></a></td><td><a href="#"><div class="btOn"></div></a></td><td><a href="#"><div class="btOn"></div></a></td><td><a href="#"><div class="btOn"></div></a></td></tr></tbody></table></div><div class="IBL"><table class="ColourTable"><tbody><tr class="ColourRow"><td colspan="4" style="text-align: left;"><a id="SkinBGCBtn" href="#"><div class="btOff"></div></a>Google Background Colours</td></tr><tr><th>BGC</th><th>Normal Font</th><th>Hover Font</th><th>Selected Font</th></tr><tr id="skinBGC" class="ColourRowTD"><td><input class="jscolorGM" /></td><td><input class="jscolorGM" /></td><td><input class="jscolorGM" /></td><td><input class="jscolorGM" /></td></tr></tbody></table></div><div class="IBL"><table class="ColourTable"><tbody><tr class="ColourRow"><td colspan="4" style="text-align: left;"><a id="SkinFontBtn" href="#"><div class="btOff"></div></a>Font Colours</td></tr><tr><th>Link</th><th>Visited</th><th>Text</th><th>Citation</th></tr><tr id="skinFont" class="ColourRowTD"><td><input class="jscolorGM" /></td><td><input class="jscolorGM" /></td><td><input class="jscolorGM" /></td><td><input class="jscolorGM" /></td></tr></tbody></table></div></div><div name="MTabW"><div id="BLDiv"><table id="BLTable"><tbody><tr><th>Hostname</th><th>Domain</th></tr><tr><td></td><td></td></tr></tbody></table></div></div><div name="MTabW">Unused at the moment</div><div name="MTabW" style="text-align: left; padding: 0 10px 0 10px;"><p style=""><i style="padding-left: 20px;">“We know where you are. We know where you’ve been. We can more or less know what you’re thinking about.”</i><br />- Eric Schmidt, ex-CEO for Google</p><p><img style="float: left; margin: 0 10px 0 0;" src="" />&nbsp;&nbsp;&nbsp;&nbsp;<a href="http://userscripts.org/scripts/show/159750">Schmoogle</a>&nbsp;was created to improve the usage of Google Search by&nbsp;<a href="http://userscripts.org/users/100610/scripts">TimidScript</a>. It is one of the most robust&nbsp;scripts for Google search, capturing most searches with little fault.</p><p>Google is one of the most invasive companies around, so please remember to protect your privacy.&nbsp;Install scripts that remove tracking and also consider using proxies. Currently Schmoogle&nbsp;removes tracking from links and unwanted metadata from URL.</p><p>Assistant required in translations of text to provide script in multiple languages. Please&nbsp;visit <a href="http://userscripts.org/topics/129406">here</a> for list of required&nbsp;translations.</p><p>Positive feedbacks are always welcome. If you enjoy this script, give a positive&nbsp;<a href="http://userscripts.org/scripts/reviews/159750">review</a>&nbsp;and become a<a href="http://userscripts.org/scripts/fans/159750">fan</a>.</p></div><div style="text-align: right; margin: 0 5px 5px 5px;"><input id="SettingsExit" type="button" value="Exit" /></div></section>');

        sw.style.position = "fixed";
        sw.style.top = "10px";
        sw.style.right = "10px";

        document.body.appendChild(sw);

        var btns = sw.getElementsByTagName("input");
        for (var i = 0; i < btns.length; i++)
        {
            if (btns[i].type == "button") btns[i].onclick = SettingsWindow.buttonPressed;
        }

        var btns = sw.getElementsByTagName("a");
        for (var i = 0; i < btns.length; i++)
        {
            if (btns[i].firstElementChild && btns[i].firstElementChild.tagName == "DIV") btns[i].onclick = SettingsWindow.switchPressed;
        }

        document.getElementById("PInterval").value = Settings.paging.interval;
        document.getElementById("GroupSel").selectedIndex = Settings.result.groupSize - 2;

        SettingsWindow.setSwitches(document.getElementsByName("cNumber"), Settings.result.number);
        SettingsWindow.setSwitches(document.getElementsByName("cWholeRow"), Settings.result.wholerow);
        SettingsWindow.setSwitches(document.getElementsByName("cRemove"), Settings.result.remove);
        SettingsWindow.setSwitches(document.getElementsByName("cGroup"), Settings.result.group);

        // -------------- Fill blacklist table
        var tds = document.getElementById("BLTable").getElementsByTagName("td");
        for (var i = 0; i < 2; i++)
        {
            if (i == 0) arr = Settings.blacklist.permanentH;
            else arr = Settings.blacklist.permanentD;
            for (var j = 0; j < arr.length; j++)
            {
                var d = document.createElement("div");
                d.innerHTML = "<button>X</button>" + arr[j];
                d.firstElementChild.onclick = function (e)
                {
                    var btn = e.target;
                    var isHostname = !(btn.parentElement.parentElement.nextElementSibling == null);
                    FilterSystem.blacklistRemove(btn.parentElement.textContent.substring(1), isHostname, false);
                    removeNode(btn.parentElement);
                };

                tds[i].appendChild(d);
            }
        }

        SettingsWindow.displaySkinSettings();

        var tabs = document.getElementsByName("MTab");
        for (var i = 0; i < tabs.length; i++) tabs[i].onclick = SettingsWindow.tabSelected;
        document.getElementsByName("MTab")[0].click();
        jscolor.fixed = true;
        jscolor.bind();
    },

    displaySkinSettings: function ()
    {
        SettingsWindow.monitorSkinChanges(false);

        SettingsWindow.setColours(document.getElementById("skinGeneral").getElementsByTagName("input"), Settings.skin.general);
        SettingsWindow.setColours(document.getElementById("skinBorderColor").getElementsByTagName("input"), Settings.skin.border);
        SettingsWindow.setColours(document.getElementById("skinBGC").getElementsByTagName("input"), Settings.skin.bgc);
        SettingsWindow.setColours(document.getElementById("skinFont").getElementsByTagName("input"), Settings.skin.font);

        SettingsWindow.setSwitches(document.getElementById("skinBorderColor").getElementsByTagName("div"), Settings.skin.border[0]);
        SettingsWindow.setSwitches(document.getElementById("skinBorderTrans").getElementsByTagName("div"), Settings.skin.transparency);
        SettingsWindow.setSwitches(document.getElementById("skinBorderColor").getElementsByTagName("div"), Settings.skin.border[0]);
        SettingsWindow.setSwitches(document.getElementById("SkinBGCBtn").getElementsByTagName("div"), Settings.skin.bgc[0]);
        SettingsWindow.setSwitches(document.getElementById("SkinFontBtn").getElementsByTagName("div"), Settings.skin.font[0]);


        var sel = document.getElementById("skinSel");
        sel.innerHTML = "";

        var skins = Settings.GetSkinNames();
        for (var i = 0; i < skins.length; i++)
        {
            var opt = document.createElement("option");
            opt.textContent = skins[i].name;
            opt.value = skins[i].id;
            sel.appendChild(opt);
            if (skins[i].id == Settings.skin.id) opt.selected = true;
        }

        document.getElementById("deleteSkin").disabled = (Settings.skin.id == 0);
        jscolor.refresh();

        SettingsWindow.monitorSkinChanges(true);
    },

    monitorSkinChanges: function (enable)
    {
        var sm = document.getElementById("SkinsMenu");

        var els = sm.getElementsByClassName("jscolorGM");
        for (var i = 0; i < els.length; i++) els[i].onchange = (enable) ? SettingsWindow.addSkinStyles : null;

        els = sm.getElementsByTagName("a");
        for (var i = 0; i < els.length; i++)
            if (enable) els[i].addEventListener("click", SettingsWindow.addSkinStyles, false);
            else els[i].removeEventListener("click", SettingsWindow.addSkinStyles, false);

        var sel = document.getElementById("skinSel");
        if (enable)
        {
            sel.onchange = function (e)
            {
                Settings.SaveSkin();
                Settings.skin = Settings.GetSkin(this.options[this.selectedIndex].value);
                document.getElementById("deleteSkin").disabled = (Settings.skin.id == 0);
                SettingsWindow.displaySkinSettings();
                Beautifier.addDynamicStyles();
                Settings.Save();
            }
        }
        else sel.onchange = null;
    },

    setColours: function (inpts, colours)
    {
        var n = (inpts.length < colours.length) ? 1 : 0;
        for (var i = 0; i < inpts.length; i++)
        {
            inpts[i].value = colours[i + n];
        }
    },

    getColours: function (inpts, colours)
    {
        var n = (inpts.length < colours.length) ? 1 : 0;
        for (var i = 0; i < inpts.length; i++)
        {
            colours[i + n] = inpts[i].value;
        }
    },

    setSwitches: function (btns, value)
    {
        for (var i = 0; i < btns.length; i++)
        {
            if (value & Math.pow(2, i + 1)) btns[i].className = btns[i].className.replace(/Off$/, "On");
            else btns[i].className = btns[i].className.replace(/On$/, "Off");
        }
    },

    getSwitches: function (btns)
    {
        var value = 0;

        for (var i = 0; i < btns.length; i++)
        {
            if (btns[i].className.match(/On$/)) value += Math.pow(2, i + 1);
        }

        return value;
    },

    tabSelected: function (e)
    {
        var name = e.target.value;

        var mtabws = document.getElementsByName("MTabW");
        var mtabs = document.getElementsByName("MTab");
        for (var i = 0; i < mtabs.length; i++)
        {
            if (mtabs[i].value == name)
            {
                mtabs[i].style.backgroundColor = "yellow";
                mtabws[i].style.display = null;
            }
            else
            {
                mtabs[i].style.backgroundColor = null;
                mtabws[i].style.display = "none";
            }
        }
    },

    switchPressed: function (e)
    {
        e.stopPropagation();
        var cn = e.target.className;
        e.target.className = (cn.match(/On$/)) ? cn.replace(/On$/, "Off") : cn.replace(/Off$/, "On");
        return false;
    },


    addSkinStyles: function ()
    {
        SettingsWindow.getSkinSettings();
        Beautifier.addDynamicStyles();
        Settings.SaveSkin();

    },

    getSkinSettings: function ()
    {
        SettingsWindow.getColours(document.getElementById("skinGeneral").getElementsByTagName("input"), Settings.skin.general);
        SettingsWindow.getColours(document.getElementById("skinBorderColor").getElementsByTagName("input"), Settings.skin.border);
        SettingsWindow.getColours(document.getElementById("skinBGC").getElementsByTagName("input"), Settings.skin.bgc);
        SettingsWindow.getColours(document.getElementById("skinFont").getElementsByTagName("input"), Settings.skin.font);

        Settings.skin.border[0] = SettingsWindow.getSwitches(document.getElementById("skinBorderColor").getElementsByTagName("div"));
        Settings.skin.transparency = SettingsWindow.getSwitches(document.getElementById("skinBorderTrans").getElementsByTagName("div"));
        Settings.skin.bgc[0] = SettingsWindow.getSwitches(document.getElementById("SkinBGCBtn").getElementsByTagName("div"));
        Settings.skin.font[0] = SettingsWindow.getSwitches(document.getElementById("SkinFontBtn").getElementsByTagName("div"));
    },

    buttonPressed: function (e)
    {
        switch (e.target.value.toLowerCase())
        {
            case "exit":
                Settings.result.number = SettingsWindow.getSwitches(document.getElementsByName("cNumber"));
                Settings.result.wholerow = SettingsWindow.getSwitches(document.getElementsByName("cWholeRow"));
                Settings.result.remove = SettingsWindow.getSwitches(document.getElementsByName("cRemove"));
                Settings.result.group = SettingsWindow.getSwitches(document.getElementsByName("cGroup"));

                SettingsWindow.getSkinSettings();

                Settings.paging.interval = parseInt(document.getElementById("PInterval").value);
                if (isNaN(Settings.paging.interval) || Settings.paging.interval < 1000) Settings.paging.interval = 1500;
                Settings.result.groupSize = document.getElementById("GroupSel").selectedIndex + 2;

                Settings.Save();
                Beautifier.refreshTable();
                removeNode("Settings");
                MenuHQ.highlight();
                break;
            case "new":
                var name = prompt("Please enter new skin name", "New Skin");
                if (name != null && name != "")
                {
                    Settings.SaveSkin();
                    Settings.skin.name = name;
                    Settings.skin.id = Date.now();
                    Settings.SaveSkin();
                    SettingsWindow.displaySkinSettings();
                }
                break;
            case "delete":
                if (confirm("Delete \"" + Settings.skin.name + "\" skin?"))
                {
                    Settings.DeleteSkin();
                    SettingsWindow.displaySkinSettings();
                    Beautifier.addDynamicStyles();
                }
                break;
        }
    }
}


var MenuHQ =
    {
        addQuickMenu: function ()
        {
            if (IsNull(document.getElementById("QuickMenu")))
            {
                var QM = AddHTMLElement('<div id="QuickMenu"><ul><li><input type="button" value="▶" title="Google Services" /></li><li><hr /></li><li><input type="button" value="S" title="Settings" /></li><li><input type="button" value="R" title="Refresh Table" /></li><li><input type="button" value="➤" title="Skins" /></li><li><hr /></li><li><input type="button" value="P" title="Auto-Pager" /></li><li><input type="button" value="B" title="Blacklist On" /></li><li><input type="button" value="M" title="SafeSearch Off" /></li><li><hr /></li><section id="mColoumns"><li><input type="button" value="1" title="Single Column" /></li><li><input type="button" value="2" title="Two Columns" /></li><li><input type="button" value="3" title="Three Columns" /></li><li><input type="button" value="4" title="Four Columns" /></li></section><li><hr /></li><li><input type="button" value="A" title="About" /></li></ul></div>');

                document.body.appendChild(QM);
                //document.body.insertBefore(QM, document.body.firstElementChild);
                var bts = QM.getElementsByTagName("input");
                for (var i = 0; i < bts.length; i++) bts[i].onclick = MenuHQ.menuSelected;

                MenuHQ.highlight();
            }
        },

        menuSelected: function (e)
        {
            switch (e.target.value)
            {
                case "▶":
                    var jumpMenu = document.getElementById("JumpMenu");
                    if (jumpMenu) removeNode("JumpMenu");
                    else
                    {
                        removeNode("SkinMenu");
                        jumpMenu = document.createElement("ul");
                        jumpMenu.id = "JumpMenu";
                        jumpMenu.className = "contextMenu";
                        jumpMenu.setAttribute("skin", Settings.skin.id);
                        var params = [["Web", null], ["News", "nws"], ["Videos", "vid"], ["Images", "isch"], [null, null], ["Books", "bks"], ["Places", "plcs"],
                            ["Recipes", "rcp"], [null, null], ["Blogs", "blg"], ["Discussions", "dsc"], [null, null], ["Apps", "app"], ["Patents", "pts"]];

                        for (var i = 0; i < params.length; i++)
                        {
                            var li = document.createElement("li");
                            var url = document.URL
                            if (params[i][0])
                            {
                                var a = document.createElement("a");
                                a.textContent = params[i][0];
                                li.appendChild(a);

                                var param = params[i][1];
                                if (url.match(/&tbm=/) == param || (url.match(new RegExp("&tbm=" + param)))) li.style.backgroundColor = "cyan";

                                a.href = Privacy.cleanURL(url).replace(/&tbm=[a-z]+/, "");                                         
                                a.onclick = function (e)
                                {         
                                    e.stopImmediatePropagation();
                                    return true;
                                }
                            }
                            else
                            {
                                var hr = document.createElement("hr");
                                li.appendChild(hr);
                            }

                            jumpMenu.appendChild(li);
                        }

                        jumpMenu.style.top = (e.clientY - 20) + "px";
                        document.body.appendChild(jumpMenu);

                    }
                    break;
                case "S":
                    if (document.getElementById("SettingsExit"))
                    {
                        document.getElementById("SettingsExit").click();
                    }
                    else
                    {
                        Settings.Load();
                        SettingsWindow.showWindow()
                    }
                    break;
                case "R":
                    Settings.Load();
                    Beautifier.refreshTable();
                    break;
                case "➤":
                    var skinMenu = document.getElementById("SkinMenu");
                    if (skinMenu) removeNode("SkinMenu");
                    else
                    {
                        removeNode("JumpMenu");
                        skinMenu = document.createElement("ul");
                        skinMenu.id = "SkinMenu";
                        skinMenu.className = "contextMenu";
                        skinMenu.setAttribute("skin", Settings.skin.id);

                        var skins = Settings.GetSkinNames();

                        for (var i = 0; i < skins.length; i++)
                        {
                            var li = document.createElement("li");
                            li.value = skins[i].id;
                            var a = document.createElement("a");
                            a.textContent = skins[i].name;
                            a.href = "#";
                            li.appendChild(a);

                            if (skins[i].id == Settings.skin.id) li.style.backgroundColor = "cyan";

                            li.onmouseover = function ()
                            {
                                Settings.skin = Settings.GetSkin(this.value);
                                Beautifier.addDynamicStyles();
                            };

                            li.onmouseout = function ()
                            {
                                Settings.skin = Settings.GetSkin(this.parentElement.getAttribute("skin"));
                                Beautifier.addDynamicStyles();
                            };

                            a.onclick = function ()
                            {
                                Settings.skin = Settings.GetSkin(this.parentElement.value);
                                Settings.Save();
                                Beautifier.addDynamicStyles();
                                removeNode("SkinMenu");
                                MenuHQ.highlight();
                                return false;
                            }
                            skinMenu.appendChild(li);
                        }

                        skinMenu.style.top = (e.clientY - 10) + "px";
                        document.body.appendChild(skinMenu);
                    }
                    break;
                case "B":
                    Settings.blacklist.enabled = !Settings.blacklist.enabled;
                    Settings.Save();
                    Beautifier.refreshTable();
                    break;
                case "P":
                    Settings.paging.enabled = !Settings.paging.enabled;
                    Settings.Save();
                    if (Settings.paging.enabled && document.getElementById("pnnext")) Pager.intervalID = setInterval(Pager.loadNextPage, 250);
                    else clearInterval(Pager.intervalID);
                    Pager.loadNextPage();
                    break;

                case "M":
                    var btn = document.getElementById("safeSearchToggler");
                    if (btn) btn.click();
                    break;
                case "1":
                case "2":
                case "3":
                case "4":
                    Settings.columns = parseInt(e.target.value);
                    Settings.Save();
                    Beautifier.refreshTable();
                    break;
                case "A":
                    SettingsWindow.showWindow();
                    document.getElementsByName("MTab")[4].click();
            }

            MenuHQ.highlight();
        },


        highlight: function ()
        {
            var btns = document.getElementById("QuickMenu").getElementsByTagName("input");
            for (var i = 0; i < btns.length; i++)
            {
                switch (btns[i].value)
                {
                    case "▶":
                        btns[i].style.backgroundColor = (document.getElementById("JumpMenu")) ? "lime" : null;
                        break
                    case "S":
                        btns[i].style.backgroundColor = (document.getElementById("Settings")) ? "lime" : null;
                        break;
                    case "➤":
                        btns[i].style.backgroundColor = (document.getElementById("SkinMenu")) ? "lime" : null;
                        break;
                    case "B":
                        btns[i].style.backgroundColor = (Settings.blacklist.enabled) ? "lime" : null;
                        break;
                    case "P":
                        btns[i].style.backgroundColor = (Settings.paging.enabled) ? "lime" : null;
                        break;
                    case "F":
                        btns[i].style.backgroundColor = (Settings.pacify) ? "lime" : null;
                        break;
                    case "M":
                        btns[i].style.backgroundColor = (document.URL.match(/&safe=on/gi)) ? "hotpink" : null;
                        break;

                }
            }

            var cls = document.getElementById("mColoumns").getElementsByTagName("input");
            for (var i = 0; i < cls.length; i++) cls[i].style.backgroundColor = (cls[i].value == Settings.columns) ? "lime" : null;
        }
    };


//num = Number of items to load
var Privacy =
{
    cleanURL: function (url)
    {
        var n = url.indexOf("?");

        var prefix = url.substring(0, n).replace("webhp", "search");

        var postfix = "&" + url.substring(n + 1);

        n = postfix.indexOf("#");
        if (n >= 0) postfix = "&" + postfix.substring(n + 1);

        console.log(postfix);
        //Remove unwanted tags
        for (var i = 0; i < deleteParams.length; i++)
        {
            postfix = postfix.replace(new RegExp("[\&#]" + deleteParams[i] + "=[^&#]+", "gi"), "");
        }


        //Remove duplicated parameters. Used for history
        var params = postfix.match(/[\&#][a-z]+=/gi);
        for (var i = 0; i < params.length; i++)
        {
            var m = postfix.match(new RegExp(params[i] + "[^&#]+", "gi"));
            for (var j = 0; j < m.length - 1; j++) postfix = postfix.replace(m[j], "");
        }

        //Make sure query parameter is at the start
        var q = postfix.match(/&(q=[^&#]+)/);
        if (q) url = prefix + "?" + q[q.length - 1] + postfix.replace(/&q=[^&#]+/ig, "");
        else url = prefix + "?" + postfix;

        return url;
    },

    removeTracking: function (item)
    {
        //var links = document.getElementById("ResultTable").getElementsByTagName("a");
        var links = item.getElementsByTagName("a");

        for (var i = 0; link = links[i], i < links.length; i++)
        {
            var val = link.getAttribute("onmousedown")
            if (val && val.indexOf("return rwt(this,") == 0) link.removeAttribute("onmousedown");
        }
    },


    cleanLinks: function (el)
    {
        var links = el.getElementsByTagName("a");

        for (var i = 0; link = links[i], i < links.length; i++)
        {
            link.href = Privacy.cleanURL(link.href);
        }
    }
}


/*
=============================================================================================
Do not touch. Main function that starts the whole shebang ^_^
=============================================================================================*/

var deleteParams =
[
    //Tags taken from script http://userscripts.org/scripts/show/155719       
    'client',         //Browser Name
    'sclient',        //Browser Name
    'sourceid',       //Source of the query
    'source',         //Source of the query
    'oq',             //What you typed before you made a selection
                        //from the suggestions
    'aq',             //Google Suggest Tracking (Shows which suggestion you choose)
    'pq',             //Previous Query
    'sa',             //Google SERPs navigation behavior tracking
    'swrnum',         //The number of results the initial query returned
    'as_q',           //When searching within results, the query is added as_q
    'oi',             //Universal search: Group name
    'resnum',         //Universal search: Number of a result within the group

    //--- Maybe Tracking Params, but details unknown ---
    'gs_l',           //Location?
    'bav',
    'bvm',
    'bpcl',
    'biw',            //Client display width?
    'bih',            //Client display height?
    'w',
    'h',
    'tbnh',
    'tbnw',
    'fp',
    'ei',
    'usg',
    'sig2',
    'tbs',
    'ved',

    //--- Appearance Setting Params (default: Disabled) ---
    // If you want to delete these params, please reveal the comment out.
    //'tbo',            //tbo=1: Display search toolbar
    //'prmdo',          //prmdo=1: Expand 'services' in toolbar
    //'sout',           //sout=1: Change UI of Google Image Search to old version
    //'esrch',          //esrch=instantpreviews: Enable instant preview
    //'filter',         //filter=1: Filter similar pages
    //'hl',             //Interface language
    //'lr',             //Search target language
    //'ie',             //Query encoding
    //'oe',             //Search result encoding
    'noj',            //noj=1: No JavaScript

    //--- Unknown Params ---
    'pdx',
    'ech',
    'psi',
    'emsg',
    'facrc',
    'imgdii',
    'iact',
    'ndsp',
    'tx',
    'ty',
    //Added by me
    'pbx',
];

//encodeURIComponent();
//decodeURIComponent();

//Privacy.HijackTracking();
Settings.Load(); //Used for auto-complete in IDE

/*
Observe Mutation has slow response so we force callback first. 
NB: calling callback breaks items like #topstuff which have lazy loading content. 
Have to handle such items in a special manner*/
//Observe.callback();
Observe.bodyChanges();