Greasy Fork is available in English.

Produs (Proddable Pardus)

Makes Pardus a little more touch-friendly.

// ==UserScript==
// @name            Produs (Proddable Pardus)
// @namespace       Produs
// @description     Makes Pardus a little more touch-friendly.
// @grant           none
// @include         http*://*.pardus.at/*
// @version         9
// @author          Richard Broker (Beigeman)
// @license         MIT License
// ==/UserScript==

var TEXT_NODE = 3;
var PRODUS_CMD_ID_SUFFIX = "ProdusCommand";
var mod = { STARBASE : 0, PLANET : 1, BUILDING : 2,  MY_BUILDING : 3, NEW_BUILDING : 4, WARP : 5, AMBUSH : 6, RETREAT : 7, REFUEL : 8, DOCK : 9, HARVEST : 10, XWARP : 11, WRECK : 12, EXITSB : 13, VIEWAMBUSH : 14, CLEAN_WH : 15};
var PROT = { UNKNOWN : 0, YES : 1, NO : 2 };

var my_document = document;
var nav_size_ver = unsafeWindow.navSizeVer;
var nav_size_hor = unsafeWindow.navSizeHor;
var img_dir = unsafeWindow.imgDir;
var tile_res = unsafeWindow.tileRes;
var current_url = my_document.URL;
var sector, coords, aps_left, field_res, fuel;
var ap_img, res_img, fuel_img;
var starbase, planet, building, my_building, new_building, ambush, retreat, dock, harvest, refuel, warp, xwarp, xwarp_menu, wreck, exitsb, view_ambush, clean_wh, mine_ore;
var partial_refresh = false;
var in_callback = false;

universe();

/* Add a callback to support partial page refreshes. */
if (unsafeWindow.addUserFunction !== undefined) {
    unsafeWindow.addUserFunction(reflowNav);
}

/*
 * Reflow Pardus web pages.
 */
function universe()
{
    reflowButtons();
    reflowLinks();
    reflowNav();
    reflowStarbase();
    reflowPlanet();
    reflowTrade();    
}

function reflowNav()
{
    if (current_url.indexOf("main.php") == -1)
        return;

    if (my_document.getElementById("nav"))
        partial_refresh = true;

    /* Get rid of the left column as soon as possible, so the popping is less noticeable. */
    var left_column = my_document.getElementById("tdTabsLeft");
    var middle_column = my_document.getElementById("tdSpaceChart");
    var right_column = my_document.getElementById("tdTabsRight");

    middle_column.setAttribute('width', ((tile_res * nav_size_hor) + 50) + "px");
    left_column.setAttribute('style', 'display:none;');
    middle_column.childNodes[2].setAttribute('align', 'center');
    right_column.setAttribute('align', 'left');


    /* Add ship status & remove previously modded cmd elements. */
    if (partial_refresh)
    {
        var previous_clone = my_document.getElementById('yourship_clone');
        if (previous_clone)
        {
            previous_clone.parentNode.removeChild(previous_clone);
        }
    }

    var otherships = my_document.getElementById('otherships');
    var your_ship = getClone("yourship");
    your_ship.id = "yourship_clone";

    var produs_br = my_document.getElementById('produs_br');
    if (produs_br)
    {
        right_column.insertBefore(your_ship, produs_br);
    }
    else
    {
        right_column.insertBefore(your_ship, otherships);
        var br_container = my_document.createElement('div');
        br_container.id = "produs_br";
        br_container.appendChild(my_document.createElement('br'));
        br_container.appendChild(my_document.createElement('br'));
        right_column.insertBefore(br_container, otherships);
    }

    /* Grab the page elements we're interested in moving. */
    sector = getClone("sector");
    coords = getClone("coords");
    aps_left = getClone("apsleft");
    field_res = getClone("fieldres");
    fuel = getClone("fuel");

    ap_img = getClone("tdStatusApsImg");
    res_img = getClone("tdStatusResImg");
    fuel_img = getClone("tdStatusFuelImg");

    var status_box = getClone("status");

    var links = my_document.getElementsByTagName('a');

    starbase = modNode(my_document.getElementById("aCmdStarbase"), mod.STARBASE);
    planet = modNode(my_document.getElementById("aCmdPlanet"), mod.PLANET);
    building = modNode(my_document.getElementById("aCmdBuilding"), mod.BUILDING);
    my_building = modNode(my_document.getElementById("aCmdOwnBuilding"), mod.MY_BUILDING);
    new_building = modNode(my_document.getElementById("aCmdBuild"), mod.NEW_BUILDING);
    warp = modNode(my_document.getElementById("aCmdWarp"), mod.WARP);
    xwarp = modNode(my_document.getElementById("aCmdWarpX"), mod.XWARP);
    ambush = modNode(my_document.getElementById("aCmdAmbush"), mod.AMBUSH);
    retreat = modNode(my_document.getElementById("aCmdRetreatInfo"), mod.RETREAT);
    refuel = modNode(my_document.getElementById("aCmdTank"), mod.REFUEL);
    dock = modNode(my_document.getElementById("aCmdDock"), mod.DOCK);
    harvest = modNode(my_document.getElementById("aCmdCollect"), mod.HARVEST);
    wreck = modNode(my_document.getElementById("aCmdWreck"), mod.WRECK);
    exitsb = modNode(my_document.getElementById("aCmdExitSb"), mod.EXITSB);
    view_ambush = modNode(my_document.getElementById("aCmdAmbushSettings"), mod.VIEWAMBUSH);
    clean_wh = modNode(my_document.getElementById("aCmdCleanWh"), mod.CLEAN_WH);
    
    let nodelist = null;
    if (partial_refresh)
        nodelist = my_document.querySelectorAll("a[href='javascript:collect(\"asteroids\", 1)']");
    else
        nodelist = my_document.querySelectorAll("a[href='main.php?collect_type=asteroids']");
        
    if (nodelist.length >= 1)
        mine_ore = modNode(nodelist[0], mod.HARVEST);

    var nav_table = getNavTable();
    
    //addGrid(nav_table);

    /* Reflow the elements we've manipulated. */
    prepend_status(nav_table);
    append_commands(nav_table);
}

/*
* Reflows the starbase screen.
*/
function reflowStarbase()
{
    if (current_url.indexOf("starbase.php") == -1)
        return;

    var popularity_value = my_document.getElementById("popularity_value");

    if (popularity_value)
    {
        var popularity_greeting = my_document.getElementById("popularity_greeting");
        popularity_value.parentNode.removeChild(popularity_value);
        popularity_greeting.parentNode.removeChild(popularity_greeting);
    }

    var table = firstEChild(my_document.body);
    var tbody = firstEChild(table);
    var content = tbody.childNodes[1];
    var content_td = content.childNodes[1];

    /* Remove gossiper portraits. */
    var tables = my_document.getElementsByTagName('table');
    var content_div = my_document.getElementsByTagName('div')[0];
    content_div.removeChild(content_div.childNodes[1]); /* Gossipers 1&2. */
    content_div.removeChild(content_div.childNodes[4]); /* Gossipers 3&4 (empty innerHTML if missing, just remove anyway).*/
    
    /* Remove short range scanner notification */
    var table_headers = my_document.getElementsByTagName('th');
    if (table_headers)
    {
        for (var i = 0; i < table_headers.length; i++)
        {
            if (table_headers[i].textContent == "Short Range Scan")
            {
                var srs_table = table_headers[i].parentNode.parentNode.parentNode.parentNode;
                srs_table.parentNode.removeChild(srs_table);                
            }
        }
    }

    /* Remove starbase population display. */
    var popSpan = my_document.getElementsByTagName('span')[1];
    popSpan.parentNode.removeChild(popSpan);

    /* Only NPC starbases have popularity, so we use that to determine conditional actions. */
    if (!popularity_value)
    {
        /* Remove commander name, alliance links. */
        var commander = tables[4];
        commander.parentNode.removeChild(commander);

        /* Remove the custom welcome message. */
        if (tables.length  > 5)
        {
            var welcome_message = tables[5];
            welcome_message.parentNode.removeChild(welcome_message);
        }
    }

    /* Align things properly. */
    content_div.setAttribute('style', 'clear:both');
    tables[3].setAttribute('align', 'left');

    /* Make the service links touch-friendly */
    padLinks();
}

/*
* Reflows planet screens.
*/
function reflowPlanet()
{
    if (current_url.indexOf("planet.php") == -1)
        return;

    var popularity_value = my_document.getElementById("popularity_value");

    if (popularity_value)
    {
        var popularity_greeting = my_document.getElementById("popularity_greeting");
        popularity_value.parentNode.removeChild(popularity_value);
        popularity_greeting.parentNode.removeChild(popularity_greeting);
    }

    var table = firstEChild(my_document.body);
    var tbody = firstEChild(table);
    var content = tbody.childNodes[1];
    var content_tbl = content.childNodes[1];
    content_tbl.setAttribute('align', 'left');
    var links = firstEChild(firstEChild(firstEChild((content_tbl.childNodes[3]))));

    /* Remove gossiper portraits. */
    var tables = document.getElementsByTagName('table');
    var content_div = document.getElementsByTagName('div')[0];
    content_div.removeChild(content_div.childNodes[1]); /* Gossipers 1&2. */
    content_div.removeChild(content_div.childNodes[4]); /* Gossipers 3&4 (empty innerHTML if missing, just remove anyway).*/

    /* Remove starbase population display. */
    var popSpan = document.getElementsByTagName('span')[1];
    popSpan.parentNode.removeChild(popSpan);

    links.setAttribute('align', 'left');
    links.removeChild(links.childNodes[2]);
    links.childNodes[1].setAttribute('align', 'left');

    padLinks();
}

function reflowTrade()
{
    if ((current_url.indexOf("_trade.php") == -1) && (current_url.indexOf("blackmarket.php") == -1))
        return;

    var els = my_document.querySelectorAll("input[type=text]");

    for (var i = 0; i < els.length; i++)
    {
        els[i].type = "number";
        els[i].style.height = "35px";
        els[i].style.width = "50px";
    }
}

function reflowButtons()
{
    var els = my_document.querySelectorAll("input[type=submit]");

    for (var i = 0; i < els.length; i++)
    {        
        els[i].style.height = "45px";
        els[i].style.width = (((els[i].value.length) * 6) + 35) + "px";
    }
}

function reflowLinks()
{
    var els = my_document.querySelectorAll("a");

    for (var i = 0; i < els.length; i++)
    {
        if (els[i].querySelector("img"))
            continue;

        improveClickability(els[i]);
        els[i].style.padding = "1em";
    }
}

function addGrid(nav_table)
{
    var navFields = nav_table.querySelectorAll('.nf');
    
    for (var i = 0; i < navFields.length; ++i)
    {
        if (navFields[i].parentNode.tagName != 'A')
            continue;  
    
        navFields[i].style.boxShadow = "inset 0px 0px 0px 1px rgba(200,200,200,1.0)";
        
        if (navFields[i].parentNode.parentNode.className == "navClear")
        {
            navFields[i].parentNode.parentNode.style.boxShadow = "inset 0px 0px 0px 1px rgba(200,200,200,1.0)";
            navFields[i].style.height -= 1;
            navFields[i].style.width -= 1;
        }
    }
    
    var navClear = nav_table.querySelectorAll('.navClear');
    
    for (var i = 0; i < navClear.length; ++i)
    {    
        navClear[i].style.boxShadow = "inset 0px 0px 0px 1px rgba(200,200,200,1.0)";
    }
}

/* Clones node "node_str" from original document. */
function getClone(node_str)
{
    var node = my_document.getElementById(node_str);

    if (node != null)
        return node.cloneNode(true);

    return null;
}

/*
 * Generic link padding for better clickability on planets/starbases.
 */
function padLinks()
{
    var images = my_document.getElementsByTagName('img');
    for(var i = 0; i < images.length; i++)
    {
        if (images[i].src.indexOf("/factions/sign_") == -1)
            continue;

        if (images[i].parentNode.childNodes.length < 2)
            images[i].parentNode.parentNode.removeChild(images[i].parentNode);
        else
            images[i].parentNode.removeChild(images[i]);

        /* We just removed an element, so adjust our index accordingly. */
        i--;
    }

    var separators = my_document.getElementsByTagName('hr');

    for(var i = 0; i < separators.length; i++)
    {
        separators[i].setAttribute('style', 'display:none');
    }

    var links = my_document.getElementsByTagName('a');

    for(var i = 0; i < links.length; i++)
    {
        if (links[i].href != "javascript:showStory();")
        {
            links[i].setAttribute('style', 'display:block;padding:15px 0;border-top-style:solid;border-bottom-style:solid;border-width:1px;');
        }
        else
        {
            links[i].setAttribute('style', 'display:none');
        }
    }

    var br = my_document.getElementsByTagName('br');
    while(br.length > 0)
    {
        br[0].parentNode.removeChild(br[0]);
    }
}

/*
 * Gets the first child node which is not a text node.
 */
function firstEChild(node)
{
    for (var i = 0; i < node.childNodes.length; i++)
    {
        if (node.childNodes[i].nodeType != TEXT_NODE)
            return node.childNodes[i];
    }

    return null;
}

/*
 * Replaces the given node with a modified version.
 */
function modNode(node, mod_desired)
{
    if (!node)
        return null;

    var mod = getModElement(mod_desired);
    removeChildNodes(node);
    node.appendChild(mod);
    node.id += PRODUS_CMD_ID_SUFFIX;
    improveClickability(node);
    return node;
}

/*
 * Makes the entire containing TD clickable.
 */
function improveClickability(node)
{
    node.setAttribute('style', 'display:inline-block');
    node.setAttribute('width', '100%');
    node.setAttribute('height', '100%');
}

/*
 * Returns true, if the script has detected that there is protection on this tile.
 */
function getProtectionStatus()
{
    var protection_node = my_document.getElementById("tdStatusTerritory");

    if (!protection_node)
        return PROT.UNKNOWN;

    var protection_text = firstEChild(protection_node);

    if (!protection_text)
        return PROT.UNKNOWN;

    switch (protection_text.textContent)
    {
        case "You cannot be attacked here.":
            return PROT.YES;
        case "You can be attacked here.":
            return PROT.NO;
        default:
            return PROT.UNKNOWN;
    }
}

/*
 * Destroys all child nodes of the given node.
 * - Assumes "hasChildNodes" has already been called.
 */
function removeChildNodes(node)
{
    if (node.childNodes == null)
        return;
    
    
    while (node.childNodes.length > 0)
    {
        node.removeChild(node.firstChild );
    }
}

/*
 * Returns an element representing the desired modification.
 */
function getModElement(mod_desired)
{
    var new_element = my_document.createElement('img');

    switch (mod_desired)
    {
        case mod.STARBASE:
            new_element.setAttribute('src', getImgSrcByPartName("/foregrounds/starbase"));
        break;

        case mod.PLANET:
            new_element.setAttribute('src', getImgSrcByPartName("/foregrounds/planet"));
        break;

        case mod.BUILDING:
        case mod.MY_BUILDING:
            new_element.setAttribute('src', '');
        break;

        case mod.WARP:

            var src = getImgSrcByPartName("wormhole.png");

            if (src == null)
            {
                src = getImgSrcByPartName("foregrounds/wormholeseal_");
            }

            new_element.setAttribute('src', src);
        break;

        case mod.XWARP:
            var src = getImgSrcByPartName("foregrounds/xhole.png");

            if (src === null)
            {
                src = getImgSrcByPartName("foregrounds/yhole.png");
            }
            new_element.setAttribute('src', src);
        break;

        case mod.WRECK:
            new_element.setAttribute('src', getImgSrcByPartName(img_dir + "/foregrounds/wreck_"));
        break;


        case mod.AMBUSH:
            new_element.setAttribute('src', "");
        break;

        case mod.RETREAT:
            new_element.setAttribute('src', '');
        break;

        case mod.NEW_BUILDING:
            new_element.setAttribute('src', "");
        break;

        case mod.REFUEL:
            new_element.setAttribute('src', '');
        break;

        case mod.DOCK:
            new_element.setAttribute('src', '');
        break;

        case mod.HARVEST:
            new_element.setAttribute('src', '');
        break;

        case mod.EXITSB:
            new_element.setAttribute('src', '');
        break;
        
        case mod.VIEWAMBUSH:
            new_element.setAttribute('src', '');
        break;
        
        case mod.CLEAN_WH:
            new_element.setAttribute('src', '');
        break;

        default:
            alert("Undefined desired mod: " + mod_desired);
        break;
    }

    new_element.setAttribute('height', '48');
    new_element.setAttribute('width', '48');

    return new_element;
}

/*
 * Tries to find the correct image for a link by a part filepath match.
 */
function getImgSrcByPartName(match)
{
    var imgs = my_document.getElementsByTagName('img');

    for(var i = 0; i < imgs.length; i++)
    {
        if (imgs[i].src.indexOf(match) != -1)
        {
            return imgs[i].src;
        }
    }

    return null;
}

/*
 * Retrieves a reference to the ship information area.
 */
function getShipInfoTable()
{
    var tables = my_document.getElementsByTagName("table");

    for(i = 0; i < tables.length; i++)
    {
        if ((tables[i].border == 0) && (tables[i].align == "center"))
            return tables[i];
    }
}

/*
 * Retrieves a reference to the nav tiles.
 */
function getNavTable()
{
    var area = my_document.getElementById("navareatransition");

    if (area)
        return area;
    else
        return my_document.getElementById("navarea");
}

function createMaxWidthSpacer(spacer_tr)
{
    for(var i = 0; i < nav_size_hor; i++)
    {
        create_spacer(spacer_tr);
    }
}

function addProtectionStatusImage(protection_td)
{

    var protection_href = my_document.createElement('a');
    var protection_img;
    var protection = getProtectionStatus();
    
    protection_td.style = "border: 1px solid rgba(200, 200, 200, 0.5);";

     // Add the link to check for current protection status, and apply relevant image.
    if (protection == PROT.UNKNOWN)
    {
        protection_img = my_document.createElement('img');
        protection_img.setAttribute('src', '');

        if (partial_refresh)
            protection_href.setAttribute('href', 'javascript:clusterprot()');
        else
            protection_href.setAttribute('href', current_url + '?ccp=1');

        protection_href.appendChild(protection_img);
        protection_td.appendChild(protection_href);
        improveClickability(protection_href);
    }
    else if (protection == PROT.NO)
    {
        protection_img = my_document.createElement('img');
        protection_img.setAttribute('src', '');
        protection_td.appendChild(protection_img);
    }
    else if (protection == PROT.YES)
    {
        protection_img = my_document.createElement('img');
        protection_img.setAttribute('src', '');
        protection_td.appendChild(protection_img);
    }

    return protection_td;
}


/*
 * Adds status information retrieved earlier and hacks it into a new row above the nav tiles.
 */
function prepend_status(nav_table)
{
    var tbody = firstEChild(nav_table);
    var first_row = tbody.childNodes[0];
    var status_tr_1 = my_document.createElement('tr');
    var status_tr_2 = my_document.createElement('tr');
    var spacer_tr = my_document.createElement('tr');
    var table;
    var sector_td = create_status_td();
    var coords_td = create_status_td();
    var aps_td = create_status_td();
    var res_td = create_status_td();
    var fuel_td = create_status_td();
    var protection_td = create_status_td();

    createMaxWidthSpacer(spacer_tr);

    addProtectionStatusImage(protection_td);

    // Complete the table data cells entry.
    sector_td.appendChild(sector);
    coords_td.appendChild(coords);
    aps_td.appendChild(ap_img);
    aps_td.appendChild(create_spacer_text());
    aps_td.appendChild(aps_left);
    res_td.appendChild(res_img);
    res_td.appendChild(create_spacer_text());
    res_td.appendChild(field_res);
    fuel_td.appendChild(fuel_img);
    fuel_td.appendChild(create_linebreak());
    fuel_td.appendChild(fuel);

    // Add all the TDs to the TRs.
    status_tr_1.appendChild(sector_td);
    status_tr_1.appendChild(coords_td);
    status_tr_1.appendChild(aps_td);
    status_tr_2.appendChild(res_td);
    status_tr_2.appendChild(fuel_td);
    status_tr_2.appendChild(protection_td);

    var status_table = create_status_table();
    status_tr_1.setAttribute("width", "100%");
    status_table.appendChild(status_tr_1);
    status_table.appendChild(status_tr_2);
    status_table.appendChild(spacer_tr);

    // Prepend desired elements. Include hack for different behaviour
    // with partial refreshes enabled.
    if (partial_refresh)
    {
        var previous_status_table = my_document.getElementById("produs_status_table");
        var nav = my_document.getElementById("nav");
        var navtransition = my_document.getElementById("navtransition");

        /* Remove any previously added information before adding more. */
        if    (previous_status_table)
        {
            previous_status_table.parentNode.removeChild(previous_status_table);
        }

        /* Add new status information. */
        status_table.id = "produs_status_table";
        nav.parentNode.id = "produs_nav_id";
        nav.style.top = "92px";
        navtransition.style.top = "92px";
        nav_table.parentNode.parentNode.insertBefore(status_table, nav_table.parentNode);
    }
    else
    {
        nav_table.parentNode.insertBefore(status_table, nav_table);
    }
}

function create_status_td()
{
    var td = my_document.createElement('td');
    td.setAttribute('align', 'center');
    td.setAttribute('width', '33.33%');
    return td;
}

/*
 * Creates a table to fill max width
 */
function create_status_table()
{
    var span_table = my_document.createElement('table');

    span_table.setAttribute('width', '100%');
    span_table.setAttribute('height', '64px');
    span_table.setAttribute('cellspacing', '0');
    span_table.setAttribute('cellpadding', '0');
    span_table.setAttribute('border', '0');

    return span_table;
}


/*
 * Creates a table inside the tr, which spans all of the nav columns, allowing cleaner layout.
 */
function create_max_colspan_table(tr)
{
    var spanning_td = my_document.createElement('td');
    var span_table = my_document.createElement('table');
    var span_tbody = my_document.createElement('tbody');

    span_table.setAttribute('width', '100%');
    spanning_td.setAttribute('colspan', nav_size_hor);

    span_table.appendChild(span_tbody);
    spanning_td.appendChild(span_table);
    tr.appendChild(spanning_td);

    return span_tbody;
}

/*
 * Fills out the row with a separator bar, to separate new areas from the nav tiles.
 */
function create_spacer(tr)
{
    var spacer_td = my_document.createElement('td');
    var spacer_text = my_document.createTextNode('\u00A0');

    spacer_td.setAttribute('style', 'background-image:url(' + img_dir + '/text7.png);background-repeat:repeat-x;');
    spacer_td.appendChild(spacer_text);

    tr.appendChild(spacer_td);
}


/*
 * Creates a line break.
 */
function create_linebreak()
{
    return my_document.createElement('BR');
}

/*
 * Used to separate images from text.
 */
function create_spacer_text()
{
    return my_document.createTextNode(" ");
}

function create_bordered_element(element_type)
{
    var el = my_document.createElement(element_type);
    el.style = "border: 1px solid rgba(200, 200, 200, 0.5);";
    return el;
}

/*
 * Attaches commands to the end of the the nav screen tiles.
 */
function append_commands(nav_table)
{
    var rows;
    var tbody = firstEChild(nav_table);
    var commands = new Array();
    var commands_tr = my_document.createElement('tr');
    var table;
    var commands_tr_1 = my_document.createElement('tr');
    var commands_tr_2 = my_document.createElement('tr');
    var commands_tr_3 = my_document.createElement('tr');
    var spacer_tr = my_document.createElement('tr');
    var context_td_1, building_td, ambush_td, harvest_td, dock_td, refuel_td, retreat_td, xwarp_td, view_ambush_td, clean_wh_td, mine_ore_td;

    createMaxWidthSpacer(spacer_tr);
    
    context_td_1 = create_bordered_element('td');

    if (starbase)
    {
        context_td_1.appendChild(starbase);
    }
    else if (planet)
    {
        context_td_1.appendChild(planet);
    }
    else if (building)
    {
        context_td_1.appendChild(building);
    }
    else if (my_building)
    {
        context_td_1.appendChild(my_building);
    }
    else if (exitsb)
    {
        context_td_1.appendChild(exitsb);
    }
    else if (warp)
    {
        context_td_1.appendChild(warp);
    }
    else if (xwarp)
    {
        context_td_1.appendChild(xwarp);
        xwarp_td = create_bordered_element('td');
        var warp_box = my_document.getElementById('xholebox');
        warp_box.firstChild.setAttribute('style', 'width:48px;height:48px;text-align:center;padding:12px 0;');
        warp_box.firstChild.firstChild.text = "?";
        xwarp_td.appendChild(warp_box);
    }
    else if (wreck)
    {
        context_td_1.appendChild(wreck);
    }
    else
    {
        var closed_wh = getImgSrcByPartName("wormholeseal_closed");

        if (closed_wh)
        {
            var new_element = create_bordered_element('img');
            new_element.setAttribute('height', '48');
            new_element.setAttribute('width', '48');
            new_element.setAttribute('src', closed_wh);
            context_td_1.appendChild(new_element);
        }
    }

    /* If we're on a nex hole, then ensure that we can select a destination. */
    if (xwarp_td)
    {
        commands.push(xwarp_td);
    }

    /* Add to the command array, there wont be something here if the PR callback is made without the page having changed. */
    if (context_td_1.childNodes.length > 0)
        commands.push(context_td_1);

    if (new_building)
    {
        building_td = create_bordered_element('td');
        building_td.appendChild(new_building);
        commands.push(building_td);
    }

    if (harvest)
    {
        harvest_td = create_bordered_element('td');
        harvest_td.appendChild(harvest);
        commands.push(harvest_td);
    }
    
    if (mine_ore)
    {
        mine_ore_td = create_bordered_element('td');
        mine_ore_td.appendChild(mine_ore);
        commands.push(mine_ore_td);
    }

    if (refuel)
    {
        refuel_td = create_bordered_element('td');
        refuel_td.appendChild(refuel);
        commands.push(refuel_td);
    }
    
    if (clean_wh)
    {
        clean_wh_td = create_bordered_element('td');
        clean_wh_td.appendChild(clean_wh);
        commands.push(clean_wh_td);
    }


    /* Add optional commands if they exist. */
    if (ambush)
    {
        ambush_td = create_bordered_element('td');
        ambush_td.appendChild(ambush);
        commands.push(ambush_td);
    }

    if (retreat)
    {
        retreat_td = create_bordered_element('td');
        retreat_td.appendChild(retreat);
        commands.push(retreat_td);
    }
    
    if (view_ambush)
    {
        view_ambush_td = create_bordered_element('td');
        view_ambush_td.appendChild(view_ambush);
        commands.push(view_ambush_td);
    }

    if (dock)
    {
        dock_td = create_bordered_element('td');
        dock_td.appendChild(dock);
        commands.push(dock_td);
    }

    // Make things a little bit tidier.
    tidy_td_arr(commands);

    // Add the new table entries.
    if (commands.length > 0)
    {
        var produs_commands_spacer = my_document.getElementById("produs_commands_spacer");
        if (produs_commands_spacer)
            produs_commands_spacer.parentNode.removeChild(produs_commands_spacer);
    }
            
    tbody.appendChild(spacer_tr);
    table = create_max_colspan_table(commands_tr);

    if (commands.length <= nav_size_hor)
    {
        rows = 1;
        add_one_row(commands, commands_tr_1);
        table.appendChild(commands_tr_1);
    }
    else if (commands.length >= (nav_size_hor * 2))
    {
        rows = 3;
        add_three_rows(commands, commands_tr_1, commands_tr_2, commands_tr_3);
        table.appendChild(commands_tr_1);
        table.appendChild(commands_tr_2);
        table.appendChild(commands_tr_3);
    }
    else
    {
        rows = 2;
        add_two_rows(commands, commands_tr_1, commands_tr_2);
        table.appendChild(commands_tr_1);
        table.appendChild(commands_tr_2);
    }

    /* Hacks for partial refresh. */
    if (partial_refresh && (commands.length > 0))
    {
        /* Remove existing tr before adding new content. */
        var produs_commands = my_document.getElementById("produs_commands");
        if (produs_commands)
            produs_commands.parentNode.removeChild(produs_commands);

        var produs_nav = my_document.getElementById("produs_nav_id");
        produs_nav.style.height = ((tile_res * nav_size_ver) + (rows * 50) + 115) + "px";
        produs_nav.style.backgroundImage = "";

        commands_tr.id = "produs_commands";
        spacer_tr.id = "produs_commands_spacer";
    }

    // Add the new table entries.    
    tbody.appendChild(commands_tr);
}

/*
 * Configures the tds to have a tidy layout for each nav screen size.
 */
function tidy_td_arr(tds)
{
    for(var i = 0; i < tds.length; i++)
    {
        tds[i].setAttribute('align', 'center');
        if (tds.length < nav_size_hor)
        {
            tds[i].setAttribute('width', (100 / tds.length) + "%");
        }
        else
        {
            tds[i].setAttribute('width', (100 / nav_size_hor) + "%");
        }
    }
}

/*
 * Just adds one tr below the nav tiles.
 */
function add_one_row(commands, tr1)
{
    for(var i = 0; i < commands.length; i++)
    {
        tr1.appendChild(commands[i]);
    }
}

/*
 * Splits commands into two trs below the nav tiles, we have too many
 * to fit in one row.
 */
function add_two_rows(commands, tr1, tr2)
{
    for(var i = 0; i < commands.length; i++)
    {
        if (i < nav_size_hor)
        {
            tr1.appendChild(commands[i]);
        }
        else
        {
            tr2.appendChild(commands[i]);
        }
    }
}

/*
 * Adds three rows, only needed when sitting on one's own building.
 */
function add_three_rows(commands, tr1, tr2, tr3)
{
    for( var i = 0; i < commands.length; i++)
    {
        if (i < nav_size_hor)
        {
            tr1.appendChild(commands[i]);
        }
        else if (i >= (nav_size_hor * 2))
        {
            tr3.appendChild(commands[i]);
        }
        else
        {
            tr2.appendChild(commands[i]);
        }
    }
}