User Interface Script (Big Release World)

User Interface Script

2014-07-22 या दिनांकाला. सर्वात नवीन आवृत्ती पाहा.

ही स्क्रिप्ट इंस्टॉल करण्यासाठी तुम्हाला Tampermonkey, Greasemonkey किंवा Violentmonkey यासारखे एक्स्टेंशन इंस्टॉल करावे लागेल.

ही स्क्रिप्ट इंस्टॉल करण्यासाठी तुम्हाला Tampermonkey किंवा Violentmonkey यासारखे एक्स्टेंशन इंस्टॉल करावे लागेल..

ही स्क्रिप्ट इंस्टॉल करण्यासाठी तुम्हाला Tampermonkey किंवा Violentmonkey यासारखे एक्स्टेंशन इंस्टॉल करावे लागेल..

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

ही स्क्रिप्ट इंस्टॉल करण्यासाठी तुम्हाला Tampermonkey यासारखे एक्स्टेंशन इंस्टॉल करावे लागेल..

ही स्क्रिप्ट इंस्टॉल करण्यासाठी तुम्हाला एक युझर स्क्रिप्ट व्यवस्थापक एक्स्टेंशन इंस्टॉल करावे लागेल.

(माझ्याकडे आधीच युझर स्क्रिप्ट व्यवस्थापक आहे, मला इंस्टॉल करू द्या!)

ही स्टाईल इंस्टॉल करण्यासाठी तुम्हाला Stylus सारखे एक्स्टेंशन इंस्टॉल करावे लागेल.

ही स्टाईल इंस्टॉल करण्यासाठी तुम्हाला Stylus सारखे एक्स्टेंशन इंस्टॉल करावे लागेल.

ही स्टाईल इंस्टॉल करण्यासाठी तुम्हाला Stylus सारखे एक्स्टेंशन इंस्टॉल करावे लागेल.

ही स्टाईल इंस्टॉल करण्यासाठी तुम्हाला एक युझर स्टाईल व्यवस्थापक इंस्टॉल करावे लागेल.

ही स्टाईल इंस्टॉल करण्यासाठी तुम्हाला एक युझर स्टाईल व्यवस्थापक इंस्टॉल करावे लागेल.

ही स्टाईल इंस्टॉल करण्यासाठी तुम्हाला एक युझर स्टाईल व्यवस्थापक इंस्टॉल करावे लागेल.

(माझ्याकडे आधीच युझर स्टाईल व्यवस्थापक आहे, मला इंस्टॉल करू द्या!)

// ==UserScript==
// @name 		User Interface Script (Big Release World)
// @namespace 	http://wofh.ru/
// @author      akasoft,andryxa,Regis,simplexe
// @version     1.3.11.1
// @include     http://w*.wofh.ru/*
// @require     http://ajax.googleapis.com/ajax/libs/jquery/1.10.2/jquery.min.js
// @grant       GM_addStyle
// @description User Interface Script
// ==/UserScript==

String.prototype.trim = function() {
    return this.replace(/^\s+|\s+$/g, '');
};

String.prototype.toHHMMSS = function () {
    var sec_num = parseInt(this, 10); // don't forget the second param
    var hours   = Math.floor(sec_num / 3600);
    var minutes = Math.floor((sec_num - (hours * 3600)) / 60);
    var seconds = sec_num - (hours * 3600) - (minutes * 60);

    if (hours   < 10) {hours   = "0"+hours;}
    if (minutes < 10) {minutes = "0"+minutes;}
    if (seconds < 10) {seconds = "0"+seconds;}
    var time    = hours+':'+minutes+':'+seconds;
    return time;
};

Date.prototype.format = function(str) {
    // yyyy.mm.dd hh:nn:ss
    return str.split('yyyy').join(this.getFullYear()).split('mm').join(LZ(this.getMonth() + 1)).split('dd').join(LZ(this.getDate())).split('hh').join(LZ(this.getHours())).split('nn').join(LZ(this.getMinutes())).split('ss').join(LZ(this.getSeconds()));
};

Date.prototype.formatTime = function() {
    // HH:MM:SS
    return LZ(this.getUTCHours() + (this.getUTCDate() - 1) * 24) + ":" + LZ(this.getUTCMinutes()) + ":" + LZ(this.getUTCSeconds());
};

Math.randomInt = function (min, max) {
    return Math.floor(Math.random() * (max - min + 1)) + min;
};

function _setValue(key, value) {
    window.localStorage.setItem(key + '_' + playerName + '_' + currentHost, value);
}

function _getValue(key, defaultValue) {
    var itm = window.localStorage.getItem(key + '_' + playerName + '_' + currentHost);
    return itm != null ? itm == 'false' ? false : itm : defaultValue;
}

function _serialize(obj) {
    return (obj.toSource) ? obj.toSource() : JSON.stringify(obj);
}

function _deserialize(str) {
    return eval('(' + str + ')');
}

// Отладка
var debug = true;

function _log(text) {
    if (!debug) return;
    console.log(text);
}

function $q(element) {
    if (arguments.length > 1) {
        for (var i = 0, elements = [], length = arguments.length; i < length; i++)
            elements.push($q(arguments[i]));
        return elements;
    }
    if (typeof element == 'string')
        element = document.getElementById(element);
    return element;
}

// одно expression --> в 1 или более элементов
function $x(expression, parent) {
    var results = [];
    var query = document.evaluate(expression, $q(parent) || document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
    for (var i = 0, length = query.snapshotLength; i < length; i++)
        results.push(query.snapshotItem(i));
    return results;
}


function $c(className, parent) {
    "use strict";
    return (parent || document).getElementsByClassName(className)
}

// создание элемента
function $e(tag, content, attributes, style, parent) {

    var result = document.createElement(tag);
    if (content)
        result.innerHTML = content;

    if (attributes)
        for (var a in attributes)
            result.setAttribute(a, attributes[a]);

    if (style)
        for (var a in style)
            result.style[a] = style[a];

    if (parent) {
        parent = $q(parent) || document;
        parent.appendChild(result);
    }
    return result;
}

function $t(text) {
    return document.createTextNode(text);
}

function $n(element) {
    if (arguments.length > 1) {
        for (var i = 0, elements = [], length = arguments.length; i < length; i++)
            elements.push($n(arguments[i]));
        return elements;
    }
    if (typeof element == 'string')
        element = document.getElementsByName(element)[0];
    return element;
}

function LZ(n) {
    return (n > 9 ? n : '0' + n);
}

function isInt(x) {
    var y = parseInt(x);
    if (isNaN(y)) return false;
    return x == y && x.toString() == y.toString();
}

function _addStyle(css) {
    var NSURI = 'http://www.w3.org/1999/xhtml';
    var hashead = document.getElementsByTagName('head')[0];
    var parentel = hashead || document.documentElement;
    var newElement = document.createElementNS(NSURI, 'link');
    newElement.setAttributeNS(NSURI, 'rel', 'stylesheet');
    newElement.setAttributeNS(NSURI, 'type', 'text/css');
    newElement.setAttributeNS(NSURI, 'href', 'data:text/css,' + encodeURIComponent(css));
    if (hashead) {
        parentel.appendChild(newElement);
    } else {
        parentel.insertBefore(newElement, parentel.firstChild);
    }
}

function __addStyle(cssStyle) {
    try {
        GM_addStyle(cssStyle);
    } catch (e) {
        _addStyle(cssStyle);
    }
}

var isMinMenu = false;
var playerName = '';
var currentHost = '';

var wofh = unsafeWindow.wofh;
var utils = unsafeWindow.utils;
var Build = unsafeWindow.Build;
var JSN =  unsafeWindow.JSN;
var lib = unsafeWindow.lib;
/* Идея.
 Есть несколько панелек с однотипным видом и поведением.
 Div-контейнер с заголовком и кнопкой, по щелчку по которой сворачивается/разворачивается контент
 Меняется текст заголовка и цвет фона заголовка
 Меняется контент
 Сохраняются положение (left, top) контейнера и видимость контента

 Описатель панели (Pane)
 {
 pane: { id: 'pane', left: '10px', top: '10px' },
 title: { id: 'panetitle', },
 button: { id: 'panebutton', image: '' },
 caption: { id: 'panecaption', text: '', bkcolor: '' },
 content: { id: 'panecontent', visible: false, text: '' }
 }/**/

/* массив используемых картинок
 0 - кнопка панелек
 1 - pause, png
 2 - play, png
 3 - del, png
 4 - city1, gif, желтый
 5 - красный
 6 - синий
 7 - зелёный
 8 - салатовый
 9 - белый
 10 - фиолетовый /**/

var imageList = [
    "",
    "",
    "",
    "",

    "",
    "",
    "",
    "",
    "",
    "",
    ""
];
//FIXME: пофиксить 
var unitSpeedHolder = eval('({0: 4, 1: 4, 2: 5, 3: 5, 4: 6, 5: 3, 6: 7, 7: 9, 8: 4, 9: 8, 10: 10, 11: 6, 12: 4, 13: 7, 14: 12, 15: 8, 16: 7, 17: 6, 18: 3, 19: 11, 20: 5, 21: 4, 22: 6, 23: 5, 24: 9, 25: 5, 26: 6, 27: 10, 28: 6, 29: 5, 30: 4, 31: 12, 32: 7, 33: 7, 34: 7, 35: 7, 36: 5, 37: 5, 38: 4, 39: 3, 40: 5, 41: 2, 42: 2, 43: 1, 44: 2, 45: 2, 46: 2, 47: 7, 48: 6, 49: 7, 50: 7, 51: 8, 52: 7, 53: 9, 54: 8, 55: 8, 56: 1, 57: 4, 58: 7, 59: 13, 60: 1, 61: 1, 62: 1, 63: 1, 64: 1, 65: 1, 66: 9, 67: 9, 68: 11, 69: 10, 70: 7, 71: 10, 72: 9, 73: 11, 74: 5, 75: 6, 76: 7, 77: 8, 78: 5, 79: 6, 80: 5, 81: 3, 82: 11, 83: 9, 84: 10, 85: 2, 86: 10, 87: 7, 88: 12, 89: 15, 90: 12, 91: 13, 92: 1, 93: 7, 94: 20, 95: 16, 96: 6, 97: 6, 98: 6, 99: 6})');
var buildLevelMap = {13:5, 14: 1, 19:0, 34: 1, 40: 0, 51:0, 67: 0};
var Holder = (function() {

    var actualConstructor = function() {
        this.currentHost = '';
        this.playerName = '';
        this.isMinMenu = false;
        this.buildingInProgress = false;
        this.cities = null;
        arguments.callee.instances.push(this);
    }
    actualConstructor.instances = [];
    return actualConstructor;
}
        )();

Holder.prototype.setTownPanel = function(panel) {
    this.townPanel = panel;
}

Holder.prototype.updateTownPanel = function() {
    this.townPanel.caption.text = ' Города (' + getArrayCount(this.townInfo) + ')';
    this.townPanel.updateCaption();
    this.townPanel.renderContent(this);
}

Holder.prototype.collectPageInfo = function() {

    // имя игрока из заголовка страницы
    playerName = document.title.split('—')[1].trim();
    currentHost = location.host;
    isMinMenu = _getValue('isMinMenu', false);

    this.cities = _deserialize(_getValue('cities', '{ current: null, list: [] }'));
    // собираем инфу о городах игрока: название, координаты, id
    //  cities = { current: null, list: [] };
    // ищем select со списком городов (по аттрибуту name)
    var items = $x("//div[@id='town_select']/div[contains(@class, 'iteml1')]");
    for (var i in items) {
        var s = items[i].innerHTML;
        var id = items[i].getAttribute('value');
        if (!this.cities.list[i])
            this.cities.list[i] = {};

        this.cities.list[i].id = id;
        this.cities.list[i].name = s.trim();
        if (items[i].getAttribute('class').indexOf('iteml1_hover') != -1)
            this.cities.current = i;
    }
    // ищем строительство

    if (!this.cities.list[this.cities.current].building)
        this.cities.list[this.cities.current].building = {};
    this.cities.list[this.cities.current].building.at = new Date().getTime();

    if (document.location.href.indexOf('town') != -1) {
        this.buildingInProgress = $q('buildp') != null;
        if (this.buildingInProgress) {
            if (!this.cities.list[this.cities.current].building.item)
                this.cities.list[this.cities.current].building.item = {};
            var b = $x("//div[@class='task_b' or @class='task_d']/a[contains(@href,'build?pos=')]");
            if (b.length > 0) {
                var sid = parseInt(b[0].href.match(/pos=(\d+)/)[1]);
                var s = $x("//div[@class='task_b']/div[@class='task_desc']/b");
                var d = $x("//div[@class='task_d']/div[@class='task_desc']/b");
                if (s.length > 0) { //здание строится
                    var shtml = s[0].innerHTML;
                    // обновить информацию
                    this.cities.list[this.cities.current].building.item.caption = shtml.substring(0, shtml.indexOf('<br>'));
                    this.cities.list[this.cities.current].building.item.level = parseInt(shtml.substring(shtml.indexOf('—') + 2, shtml.length));
                    this.cities.list[this.cities.current].building.item.sid = sid;
                } else if (d.length > 0) { //здание разрушается
                    var shtml = d[0].innerHTML;
                    // обновить информацию
                    this.cities.list[this.cities.current].building.item.caption = shtml.substring(0, shtml.indexOf('<br>'));
                    this.cities.list[this.cities.current].building.item.level = parseInt(shtml.substring(shtml.indexOf('—') + 2, shtml.length));
                    this.cities.list[this.cities.current].building.item.sid = sid;
                }
            }
            var btime = $q('buildt').innerHTML;
            if (btime) {
                var s2 = btime.match(/\>(\d+)\</);
                if (s2 && s2[1])
                    btime = s2[1];
                this.cities.list[this.cities.current].building.item.time = 1000 * parseInt(btime);
            }
            var bprogress = $q('buildp').innerHTML;
            if (bprogress) {
                var s2 = bprogress.match(/\>(\d+)\</);
                if (s2 && s2[1])
                    bprogress = s2[1];
                this.cities.list[this.cities.current].building.item.progress = 1000 * parseInt(bprogress);
            }
        } else
            this.cities.list[this.cities.current].building.item = null;
    }

    _setValue('cities', _serialize(this.cities));
}

Holder.prototype.injectTownInfo = function() {
    this.townInfo = _deserialize(_getValue('townlist', '{}'));
    if (document.location.href.indexOf('mapinfo?o=') == -1 && document.location.href.indexOf('mapinfo?x=') == -1)
        return;
    // вставляем div с инфой и командами под ссылками

    var form = $x("//form[@action='postmessage' and @method='post']");

    if (form.length == 0) {
        return;
    }

    var tid = -1;
    var data = $x("//div[@class='pagetitle']/div");
    //тут, видимо был блок, который удалялся.
    //data = data[0].innerHTML.match(/^(.*)\((.*)\)$/);
    //старый трим не робит, дергаем контекстом
    var tname = data[0].textContent.trim();
    //_log(tname);
    //тут я так понял надо дергать ссылку на город
    //нет, все-таки на местность
    tid = data[1].textContent.split(' ')[0];
    // _log(tid);
    data = $x("//div[@class='inf_t_l_b2']/dl/dd/a[contains(@href, 'account?id=')]");
    var pid = data[0].href.match(/id=(\d+)/)[1];
    var pname = data[0].innerHTML.trim();
    var country = null;
    data = $x("//div[@class='inf_t_l_b2']/dl/dd/a[contains(@href, 'countryinfo?id=')]");
    if (data.length == 1) {
        country = { id: data[0].href.match(/id=(\d+)/)[1], name: data[0].innerHTML.trim() };
    } else {
        country = { id: -1, name: 'В стране не состоит' };
    }

    var popdate = new Date().getTime();

    var popvalue = 0;
    var tclimate = '';
    var trelief = '';
    var trace = '';

    data = $x("//div[@class='inf_t_l_b2']/dl/dd/img[contains(@class, 'res')]");
    if (data.length > 0) {
        popvalue = parseInt(data[0].parentNode.lastChild.nodeValue);
    }
    /*
     var i=0;
     while (i <data.length) {
     var o = data[i].innerHTML;
     i++;
     if(!data[i-1].nextSibling)
     continue;
     var o1 = data[i-1].nextSibling.innerHTML;

     if (o && o1) {
     if (o.indexOf('Население') != -1) {
     _log("data: " + o1);
     popvalue = parseInt(o1.substring(0, o1.indexOf('<img')));
     } else if (o.indexOf('Климат') != -1)
     tclimate = o1;
     else if (o.indexOf('Рельеф') != -1)
     trelief = o1;
     else if (o.indexOf('Раса') != -1)
     trace = o1;
     }
     }
     */
    var town = {id: tid, name: tname, climate: tclimate, relief: trelief, race: trace,
        player: { id: pid, name: pname },
        country: country,
        pop: [
            {date: popdate, value: popvalue}
        ]};

    // втавка панельки
    form = form[0];

    var inner = '<a id="townadd" href="javascript: void(0)">Добавить/обновить город</a>';
/*
    inner += '<br/><a id="playeradd" href="javascript: void(0)">Добавить игрока во враги</a>';
    if (town.country.id > -1) {
        inner += '<br/><a id="countryadd" href="javascript: void(0)">Добавить страну во враги</a>';
    }
*/

    if (this.townInfo[town.id]) {
        var t = this.townInfo[town.id];
        var s2 = '';
        if (t.pop.length > 0) {
            var j = t.pop.length - 1;
            while (j >= 0) {
                if (!t.pop[j]) {
                    j--;
                    continue;
                }
                var d = new Date(t.pop[j].date);
                if (s2.length > 0)
                    s2 += ', ';
                s2 += t.pop[j].value + '<sub>' + d.format('dd.mm hh:nn') + '</sub>';
                j--;
            }
        }
        else
            s2 = '-';

        var d = new Date(popdate);
        s2 = '<b>' + popvalue + '<sub>' + d.format('dd.mm hh:nn') + '</sub></b>, ' + s2;
        inner += '<br /><br />' + s2;
    }

    var elem = $e('div', inner, { id: 'towndiv' });
    var infDiv = $c("inf_line")[0];
    infDiv.parentNode.insertBefore(elem, infDiv.nextSibling);

    $q('townadd').addEventListener('click', clickAddTown(town, this), false);
/*
    $q('playeradd').addEventListener('click', this.clickAddPlayer(town.player), false);
    if (town.country.id > -1) {
        $q('countryadd').addEventListener('click', this.clickAddCountry(town.country), false);
    }
*/

}

function clickAddTown(town, holder) {
    return function () {
        if (holder.townInfo[town.id]) {
            var found = false;
            for (var i in holder.townInfo[town.id].pop) {
                var p = holder.townInfo[town.id].pop[i];
                if (Math.abs(town.pop[0].date - p.date) <= 60 * 1000) {
                    found = true;
                    p.date = town.pop[0].date;
                    p.value = town.pop[0].value;
                    break;
                }
            }
            if (!found)
                holder.townInfo[town.id].pop.push(town.pop[0]);
            holder.townInfo[town.id].country = town.country;
            holder.townInfo[town.id].name = town.name;
        } else {
            holder.townInfo[town.id] = town;
        }
        holder.saveTownInfo();
        holder.updateTownPanel();
    }
}

/*
Holder.prototype.clickAddPlayer = function(player) {
    return function () {
        if (holder.enemyInfo.players[player.id] != player.name) {
            holder.enemyInfo.players[player.id] = player.name;
        }
    }
}

Holder.prototype.clickAddCountry = function(country) {
    return function () {
        if (holder.enemyInfo.countries[country.id] != country.name) {
            holder.enemyInfo.countries[country.id] = country.name;
        }
    }
}
*/

Holder.prototype.saveTownInfo = function() {
    _setValue('townlist', _serialize(this.townInfo));
}


var Panel = (function() {

    var Pane = function(id, left, top) {
        this.id = id + 'pane';
        this.leftKey = id + 'pane_left';
        this.topKey = id + 'pane_top';
        this.left = left || '10px';
        this.top = top || (160 + 50 * panelCount) + 'px';
    }
    Pane.prototype.className = 'pane';

    var Title = function(id, normalBackground, alertBackground) {
        this.id = id + 'panetitle';
        this.normalBackground = normalBackground || '#90DD43';
        this.alertBackground = alertBackground || 'red';
        __addStyle('#' + id + 'pane { z-index: ' + (3030 - panelCount * 5) + '; background-color: ' + this.normalBackground + ';}');
    }
    Title.prototype.className = 'panetitle';

    var Button = function(id, imageIndex) {
        this.id = id + 'panebutton';
        this.image = imageList[0];
    }
    Button.prototype.className = 'panebutton';

    var Caption = function(id, text) {
        this.id = id + 'panecaption';
        this.text = text ? text : 'Тоже панель';
    }
    Caption.prototype.className = 'panecaption';

    var Content = function(id) {
        this.id = id + 'panecontent';
        this.visibleKey = id + 'pane_visible';
        this.visible = _getValue(this.visibleKey, false);
        this.text = 'Пусто';
    }
    Content.prototype.className = 'panecontent';

    var panelCss = '.pane { border: 2px groove black; \
    position: absolute; padding: 5px 5px; -moz-border-radius: 5px; } \
    .panebutton img { vertical-align: middle; } \
    .panetitle { cursor: move; text-align: left; vertical-align: middle; \
    padding: 1px 1px; } \
    .panecaption { vertical-align: middle; \
    white-space: nowrap; color:#000000; } \
    .panecontent { padding: 2px 0px; text-align: left; } \
    .panecontent div { padding: 0px 0px; text-align: left; font-size: 80%; } \
	#towndiv { padding: 5px 5px; background-color: #ffff66; } \
	.towns { border: 1px solid #999966; padding: 0px 0px;  font-size: 80%; \
    max-width: 500px;} \
    .towns tr, .towns td { border: 1px solid #999966; padding: 0px 0px; color: #000000; } \
    .towns td img { height: 16px; width: 16px; vertical-align: middle; } \
	.tactics { font-size: 80%; } \
    .tactics, .tactics tr, .tactics td { border-style: none; color:#000} \
	.trainTable { border:1px solid #000000;} \
    .trainTable td {text-align:center; border-left:1px solid #000000; color: #000000} \
	.tradeTable { border:1px solid #000000;} \
	.tradeTable td { text-align:center; border-left:1px solid #000000; border-top:1px solid #000000; color: #000000} \
	'
    __addStyle(panelCss);

    var panelCount = 0;
    var actualConstructor = function (id, caption, normalBackground, alertBackground) {
        this.pane = new Pane(id);
        this.title = new Title(id, normalBackground, alertBackground);
        this.button = new Button(id, 0);
        this.caption = new Caption(id, caption);
        this.content = new Content(id);
        arguments.callee.instances.push(this);
        panelCount++;
    }
    actualConstructor.instances = [];
    return actualConstructor;
})();

Panel.prototype.updateCaption = function() {
    $q(this.caption.id).innerHTML = (this.content.visible || !isMinMenu) ? this.caption.text : '';
}

Panel.prototype.action = function(render) {
    //_log('reder=' + render + ' ' + this.content.visibleKey + ':' + this.content.visible);
    !render && (this.content.visible = !this.content.visible);
    this.updateCaption();
    var content = $q(this.content.id);
    var button = $q(this.button.id);
    if (this.content.visible) {
        content.style.display = '';
        button.setAttribute('title', 'Спрятать');
        //_log(this.content.visibleKey + ':' + this.content.visible);
    }
    else {
        content.style.display = 'none';
        button.setAttribute('title', 'Показать');
    }
    _setValue(this.content.visibleKey, this.content.visible);
}

Panel.prototype.render = function() {
    var inner = [];
    inner.push('<div id="');
    inner.push(this.title.id);
    inner.push('" class="');
    inner.push(this.title.className);
    inner.push('">');
    inner.push('<a id="');
    inner.push(this.button.id);
    inner.push('" class="');
    inner.push(this.button.className);
    inner.push('" href="javascript: void(0)"><img src="');
    inner.push(this.button.image);
    inner.push('" /></a>');
    inner.push('<span id="');
    inner.push(this.caption.id);
    inner.push('" class="');
    inner.push(this.caption.className);
    inner.push('">');
    inner.push('</span></div><div id="');
    inner.push(this.content.id);
    inner.push('" class="');
    inner.push(this.content.className);
    inner.push(' " style="display:none" >');
    inner.push(this.content.text);
    inner.push('</div>');
    var element = $e('div', inner.join(''), {id: this.pane.id},
    {left: this.pane.left,top: this.pane.top}, document.body);
    element.className = this.pane.className;
    this.updateCaption();
    $q(this.button.id).addEventListener('click', bindMethod(this, this.action), false);


    _setValue(this.content.visibleKey, this.content.visible);

    function bindMethod(o, f) {
        return function() {
            return f.apply(o);
        }
    }
}

Panel.prototype.renderContent = function(holder) {

}


function onUSLoad() {
    function correctUI() {
        var unitbtn = $x("//div[@class='tunit_btns']");
        for (var i in unitbtn) {
            unitbtn[i].style.padding = '3px 92px 25px';
        }
    }

    if (document.body.innerHTML.length == 0) {
        // С 10.10.2009
        // Ошибка на сервере или ещё почему страничка не загружена
        // пробуем рефрешить её через 30-60 сек
        document.body.innerHTML = 'О-о! Ошибка при загрузке страницы. \
      Возможно, на сервере технические работы или проблемы с каналом Интернет. \
      Попытка обновления начнётся автоматически через 30..60 секунд...';

        var tid = window.setInterval(function() {
            location.href = document.URL;
        }, (20 + getRandomInt(10, 40)) * 1000);

        return;
    }
    _log('loading...');
    var holder = new Holder();
    try {
        holder.collectPageInfo();
    } catch (e) {
        _log(e);
    }

    /* стили панелек
     #tradepane { z-index: 3040; background-color: #99ccff; } \
     #buildpane { z-index: 3030; background-color: #ffcc33; } \
     #trainpane { z-index: 3025; background-color: #1C9205; } \
     #citypane { z-index: 3020; background-color: #ffff66; } \
     #mappane { z-index: 3010; background-color: #efefef; } \
     #enemypane {z-index: 3005; background-color: #ffffff; }\
     #pane { z-index: 3000; background-color: #90DD43; } \
     */
    var trade = new Panel('trade', ' Торговля', '#99ccff');

    trade.renderContent = function(holder) {
        var text = '<table class="tradeTable"><tr><td>N</td><td>Операция</td><td>Товар</td><td>Время</td></tr>';
        
        for (var i in holder.cities.list) {
            
            var c = holder.cities.list[i];
            var s = c.name;
            if ((document.location.href.indexOf("market") != -1 || document.location.href.indexOf("town") != -1)  && wofh.town.id == c.id) {
				//_log(c);
                c.traders = {};
                c.traders.total = wofh.town.trade.traders.count;
                c.traders.market = wofh.town.trade.traders.reserve;
                if (wofh.events != null) {
                var result = 0;
                wofh.events.forEach(function(entry) {
                    switch(entry.event) {
                        case 108:
                        case 109:
                        case 111:
                           if (c.id == entry.town1)
                              result += parseInt(entry.data.split(',')[1].split(':')[1]);
                           break;
                    }
                });
                c.traders.free = c.traders.total - result;
            } else {
                c.traders.free = c.traders.total-wofh.town.trade.traders.busy;
            }
                holder.cities.list[i].traders = c.traders;
                _setValue('cities', _serialize(holder.cities));
                
            }
            if (c.traders) {
                s += ' (' + c.traders.free + '/' + c.traders.market + '/' + c.traders.total + ')';
            }
            if (i == holder.cities.current)
                s = '<b>' + s + '</b>';
            s = '<a href="http://' + currentHost + '/town?tid=' + c.id + '">' + s + '</a>';
            s += ' <a href="http://' + currentHost + '/market?target=' + holder.cities.list[i].id + '#send'
                    + '"><img title="Транспортировать ресурсы" class="ibut ib2" src="/p/spacer.gif"></a>';
            s += '  <a href="http://' + currentHost + '/market?target=' + holder.cities.list[i].id + '#stream&info'
                    + '"><img title="Начать снабжение ресурсами" class="ibut ib3" src="/p/spacer.gif"></a>'
            s = '<tr><td colspan="4">' + s + '</td></tr>';
            /*
            var num = 1;
            if (wofh.events != null) {
            wofh.events.forEach(function(entry) {
                        switch(entry.event) {
                            case 108:
                            case 109:
                                    _log("o");
                                    s += '<tr><td>';
                                    s += num;
                                    s += '</td><td>';
                                    s += entry.data.split(',')[1].split(':')[0];
                                    s += '</td><td>';
                                    s += utils.parseResString(entry.data.split(',')[0].split(':')[1]);
                                    s += '</td><td>';
                                    s += '<span id="op_' + i + '_' + num + '" >' + new Date(entry.time - new Date().getTime()).formatTime() + '</span>';
                                    s += '</td></tr>';
                                    num++;

                                break;
                        }
                });
            }
            */
            text += s;
        }
        $q(this.content.id).innerHTML = text;
    }
    var build = new Panel('build', ' Строения', '#FFCC33');

    build.renderContent = function(holder) {
        var text = '';
        for (var i in holder.cities.list) {
            var c = holder.cities.list[i];
            var s = c.name;
            if (i == holder.cities.current)
                s = '<b>' + s + '</b>';
            s = '<a href="' + applyTidToUri(c.id, 'town?', true) + '">' + s + '</a>';
            s = '<td>' + s + '</td><td>';

            if (c.building) {
                s += '(' + (new Date(c.building.at)).format('hh:nn:ss') + ') ';
                var item = c.building.item;
                if (item == null)
                    s += 'Ничего не строится'
                else {
                    // _log('showBuilding: '+c.building.toSource());
                    var complete = c.building.at + item.time - item.progress;
                    s += ' <a href="' + applyTidToUri(c.id, 'build?pos=' + item.sid) + '">' + item.caption + ' x' + item.level + '</a> в ' + (new Date(complete)).format('hh:nn:ss');
                }
            }
            else
                s += ' Нет информации';
            s += '</td>';
            text += '<tr>' + s + '</tr>';
        }
        text += '<tr><td><a id="show_build" href="javascript:void(0)">Рассчитать прочность города</a></td><td></td></tr>';

        $q(this.content.id).innerHTML = '<table class="tactics" cellpadding="0" cellspacing="0">' + text + '</table>';
        $q('show_build').addEventListener('click', buildingInfo(), false);

    }

    var train = new Panel('train', ' Тренировка', '#90DD43');

    train.renderContent = function(holder) {
        var text = '';
        for (var ci in holder.cities.list) {
            var c = holder.cities.list[ci];
            var s = c.name;
            if (ci == holder.cities.current)
                s = '<b>' + s + '</b>';
            s = '<a href="' + applyTidToUri(c.id, 'town?', true) + '">' + s + '</a>';
            s = '<td>' + s + '&nbsp;</td><td>';
            //
            s += '<table class="trainTable" >';
            s += '<tr>';

            //Динамическая секция по всем строениям
            var trInfoArr = holder.cities.list[ci].train;
            for (var i in trInfoArr) {
                if (trInfoArr[i]) {
                    s += '<td>&nbsp;<a href="' + applyTidToUri(c.id, 'http://' + currentHost + '/build?pos=' + i) + '">' + trInfoArr[i].building + ' x' + trInfoArr[i].level + '</a>&nbsp;</td>';
                }
            }
            s += '</tr><tr>';
            for (var i in trInfoArr) {
                if (trInfoArr[i]) {
                    if (trInfoArr[i].unitId > -1) {
                        s += '<td><img src="p/spacer.gif" class="unit u' + trInfoArr[i].unitId + '"> (' + trInfoArr[i].unitCount + ')</td>';
                    } else {
                        s += '<td>&nbsp;</td>';
                    }
                }
            }
            s += '</tr><tr>';
            for (var i in trInfoArr) {
                if (trInfoArr[i]) {
                    if (trInfoArr[i].timeEnd) {
                        s += '<td>&nbsp;<span id="tr_' + ci + '_' + i + '" >' + new Date(trInfoArr[i].timeEnd - new Date().getTime() + new Date().getTimezoneOffset() * 60 * 1000).formatTime() + '</span>&nbsp;';
                    } else
                        s += '<td>&nbsp;';
                    s += '<div style="display:inline; font-size:100%; padding-left:20px"><a style="color:red" id="rm_' + ci + '_' + i + '" href="javascript:void(0);">X</a></div></td>';
                }
            }
            // -- конец

            s += '</tr></table>';
            s += '</td>';
            text += '<tr>' + s + '</tr>';
            ;
        }
        text = '<table class="tactics" >' + text + '</table>';
        $q(this.content.id).innerHTML = text;
        for (var ci in holder.cities.list) {
            var trInfoArr = holder.cities.list[ci].train;
            for (var i in trInfoArr) {
                if (trInfoArr[i]) {
                    $q('rm_' + ci + '_' + i).addEventListener('click', onRemoveTrainBox(ci, i, holder, this), false);
                }
            }
        }
    }

    var city = new Panel('city', ' Города', '#ffff66');

    city.renderContent = function(holder) {
        this.caption.text = ' Города (' + getArrayCount(holder.townInfo) + ')';
        //$q(desc.caption.id).innerHTML = s;
        var s = '<tr><td>ID/X/Y</td><td>Город</td><td>Игрок</td><td>Население</td><td>-</td></tr>';
        for (var i in holder.townInfo) {
            var t = holder.townInfo[i];
            var ti = getTownView(i, holder);
            var cn = '';
            if (t.country)
                cn = ', ' + t.country.name;
            s += '<tr>';
            s += '<td><a id="tgom' + i + '" href="javascript: void(0)" title="Свернуть и перейти на карту">' + t.id + '</a></td>';
            s += '<td><a id="tgot' + i + '" href="javascript: void(0)" title="Свернуть и перейти на город">' + t.name + '</a> <a id="tdelc' + i + '" href="javascript: void(0)" title="Удалить город из списка"><img src="' + imageList[3] + '"></a> <a id="tmapc' + i + '" class="lbl" href="javascript: void(0)" title="' + ti.hint + ', переключить видимость">' + ti.caption + '</a></td>';
            s += '<td><a id="tgop' + i + '" href="javascript: void(0)" title="Свернуть и перейти на игрока">' + t.player.name + cn + '</a></td>';
            s += '<td>';

            var s2 = '';
            if (t.pop.length > 0) {
                var j = t.pop.length - 1;
                while (j >= 0) {
                    if (!t.pop[j]) {
                        j--;
                        continue;
                    }
                    var d = new Date(t.pop[j].date);
                    if (s2.length > 0)
                        s2 += '<br />';
                    var aPop = t.pop[j].value;
                    if (s2.length == 0)
                        aPop = '<b>' + aPop + '</b>';
                    s2 += aPop;
                    s2 += '<sub>' + d.format('dd.mm hh:nn') + '</sub>';
                    s2 += ' <a id="tdelp' + i + j + '" href="javascript: void(0)" title="Удалить население из списка"><img src="' + imageList[3] + '"></a>';
                    j--;
                }
            }
            else
                s2 = '-';
            s += s2;
            s += '</td>';
            s += '<td>-</td>';
            s += '</tr>';
        }
        $q(this.content.id).innerHTML = '<table class="towns" cellpadding="0" cellspacing="0"><tbody>' + s + '</tbody></table>';
        for (var i in holder.townInfo) {                
            var t = holder.townInfo[i];
            $q('tgom' + i).addEventListener('click', onShowMap(t.id, this), false);
            $q('tgot' + i).addEventListener('click', onShowTown(t.id, this), false);
            $q('tgop' + i).addEventListener('click', onShowPlayer(t.player.id, this), false);
            $q('tdelc' + i).addEventListener('click', onDelTown(t.id, holder), false);
            $q('tmapc' + i).addEventListener('click', onChangeViewTown(t.id, holder), false); // на карте
            if (t.pop.length > 0) {
                var j = t.pop.length - 1;
                while (j >= 0) {
                    $q('tdelp' + i + j).addEventListener('click', onDelPop(t.id, j, holder), false);
                    j--;
                }
            }
        }
    }
/*
    var enemy = new Panel('enemy', ' Враги', '#ffffff');
*/    
    
    if (!$q('settingspane')) {
        __addStyle('#settingspane a { color:#ffffff;}');
        var sett = new Panel('settings', 'Min', '#000000');
// поднял выше        
        sett.pane.top = '360px';

        sett.action = function(render) {
            if (render)
                return;
            isMinMenu = !isMinMenu;
            _setValue('isMinMenu', isMinMenu)
            for (var i in Panel.instances)
                Panel.instances[i].updateCaption();
        };

        sett.updateCaption = function() {
            $q(this.button.id).innerHTML = (isMinMenu ? 'Max' : 'Min');
        }
    } else {
        var action = function() {
            isMinMenu = !isMinMenu;
            for (var i in Panel.instances)
                Panel.instances[i].updateCaption();

        };
        $q('settingspanebutton').addEventListener('click', action, false);
    }

    holder.setTownPanel(city);
    holder.injectTownInfo();

    for (var i in Panel.instances) {
        Panel.instances[i].render();
        try {
            Panel.instances[i].renderContent(holder);
        } catch (e) {
            _log(e);
        }
        Panel.instances[i].action(true);
    }

    //применить цветовую раскраску
    try {
        resourceColorer();
    } catch(e) {
        _log(e);
    }
    //распознаем транспортировку ресов
    if (document.location.href.indexOf('market') != -1){
    	injectCityListInTrade(holder);
        injectTradeOps(holder);
        
    }
    //распознаем отправку войск
    if (document.location.href.indexOf('getarmy') != -1) {
        injectCities($x("//form[@action='getarmy']/p"), 'DIV', holder);
        injectArmySpeed();
    }
    //распознаем отзыв подкрепления
    if (document.location.href.indexOf('turnout?home') !=-1) {
        injectArmySpeed();
    }
    // Распознаем форму тренировки войск
/*
    if (document.location.href.indexOf('trainpage') != -1 || document.location.href.indexOf('build?pos=') != -1) {
        trameTimeCalculate();
        collectTrainInfo(holder, train);
    }
*/
    if (document.location.href.indexOf('town') != -1 || document.location.href.indexOf('build?pos=') != -1) {
        trameTimeCalculate();
        collectTrainInfo(holder, train);
    }
    

    if (document.location.href.indexOf('account?id=') != -1) {
        parseCityListPage();
    }

    if (document.location.href.indexOf('map') != -1) {
        parseMapPage();
        var allPlayerCitiesHolder = _deserialize(_getValue("all_player_cities_holder", '{}'));
        var cities = _deserialize(_getValue('cities', '{ current: null, list: [] }'));
        var baseCoords = allPlayerCitiesHolder[cities.list[cities.current].id];
        if (baseCoords) {
            var linkDiv = $x("//div[@id='map_all']/div[contains(@style, 'position:absolute')]")[0];
            $e('DIV', '<a id="inf_tt_ico" class="a_osn_yes"></a>', {id:'inf_tt_ico_a'}, null, linkDiv);
            document.getElementById('inf_tt_ico_a').addEventListener('click', switchMapHelp);
        }
        $x("//div[@id='landlnks']")[0].addEventListener("DOMNodeInserted", onMapDivNodeInserted, false);
    }
    
    //распознаем форму Экономики города
    if (document.location.href.indexOf('economics') != -1) {
        injectEconomicCalc(holder);
    }

 
    if (document.location.href.indexOf('market?stream') != -1) {
        injectTPCities();
       streamSort();
    }
/*
    if (document.location.href.indexOf('buildinfo?id=') != -1) {
        parseBuildingInfo();
    }
*/
    if (document.location.href.indexOf('countryinfo') !=-1) {
        var players = $x("//table[@class='def_table players']/tbody/tr/td/a[@class='ulink0']");
        var line = "";
        for (var i in players) {
            line += players[i].innerHTML + '|';
        }
        _log(line);
    }
/*
    if (document.location.href.indexOf('town') != -1) {
        var blocks = $x("//div[contains(@style,'/p/if/stone/sh_sl.png')]");
        if (blocks.length == 0) {
            $e('DIV', null, {id: 'en_build_info_panel'}, {cursor:'pointer', background: 'url(/p/if/stone/sh_sl.png)', position: 'absolute', left: '-7px', top:'444px', zIndex:10000, width:'29px', height:'26px'}, $x("//div[@class='balka2_ balka2_p1']/div[@id='cont04']")[0]);
            $q('en_build_info_panel').addEventListener('click', changeBuildPanelView(), false);
        }
    }

    if (_getValue('build_info_panel', false) && document.location.href.indexOf('town') != -1) {
        addBuildInfoPanel();
    }
*/

    if ((document.location.href.indexOf('market') != -1 || document.location.href.indexOf('trade') != -1) && document.location.href.indexOf('adv') != -1 && document.location.href.indexOf('buy') != -1) {
        injectAdvTrade();
		$x("//table[contains(@class, 'trade_table')]")[0].addEventListener("DOMNodeInserted", onTradeTableNodeInserted, false);
    }

    applyHelpToBuildings();

    trainTimer(holder, train);
    var help_div = document.createElement('DIV');
    help_div.id = 'help_div';
    help_div.style.position = 'absolute';
    help_div.style.zIndex = 10000;
    help_div.style.border = '1px solid #333';
    help_div.style.backgroundColor = '#EEE';
    help_div.style.width = '190px';
    help_div.style.height = '180px';
    help_div.style.color = '#000';
    help_div.style.fontSize = '.8em';
    help_div.style.visibility = 'hidden';
    help_div.style.display = 'none';
    document.body.appendChild(help_div);
}


// common functions

//Применить цветовую раскраску к складу:
//фиолетовый - склад заполниться менее чем за 6 часов
//зеленый - склад опустеет более чем за 24 часа/склад заполняется
//оранжевый - склад опустеет менее чем через 24 часа но более чем за 12 часов
//бардовый - склад опустеет менее чем за 12 часов но более чем за 6 часов
//красный жирный - склад опустеет менее чем за 6 часов
//В диалоге постройки здания применить цвета к ресурсам, необходимым для постройки
//зеленый - ресурсов достаточно для постройки
//красный - ресурсов недостаточно для постройки
var arr = new Array();
function resourceColorer() {
    var lis = $x("//ul[@id='myres']/li");
    //Добавить информацию о максимальном количестве
    var capacity = null;
    for (var i in lis) {
        var cap = $x("./center/b/a/span[@id='storemax']", lis[i])[0];
        if (cap) {
            capacity = parseInt(cap.innerHTML);
            continue;
        }
        if (!capacity) continue;
        var i_nodes = $x("./i", lis[i]);
        if (i_nodes.length > 0 && i_nodes.length < 4) {
            i_nodes[1].parentNode.insertBefore($t("/"), i_nodes[1]);
            i_nodes[1].parentNode.insertBefore($e("i", capacity), i_nodes[1]);
        } else {
            i_nodes = $x("./img", lis[i]);
            if (i_nodes.length > 0) {
                i_nodes[0].parentNode.appendChild($t("/" + capacity));
            }
        }
    }
    for (var i in lis) {
        var imgs = $x("img", lis[i]);
        if (imgs.length == 0)
            continue;
        var resId = parseInt(imgs[0].className.substring(5, imgs[0].className.length));
        if (resId < 2)
            continue;
        var iss = $x("i", lis[i]);
        if (iss.length < 2) {
            var nodes = lis[i].childNodes;
            var txt = '';
            for (k in nodes) {
                if (nodes[k].nodeName == '#text') {
                    txt = txt + nodes[k].nodeValue;
                }
            }
            var art = txt.match(/(\d+)\/(\d+)/);
            if (art && art.length > 1)
                arr[resId] = parseInt(art[1]);
            else
                continue;
            imgs[0].title = imgs[0].title + ' - осталось места: ' + formatNumber(parseInt(art[2]) - parseInt(art[1]), 3);
            continue;
        }
        var nres = parseInt(iss[0].innerHTML);
        var nmax = parseInt(iss[1].innerHTML);
        imgs[0].title = imgs[0].title + ' - осталось места: ' + formatNumber(nmax - nres, 3);
        if (iss.length > 3) {
            var min = parseFloat(iss[2].innerHTML.substring(1));
            if (iss[2].innerHTML.indexOf('-') > -1) {
                if (min * 24 < nres) {
                    setColor(iss[0], iss[1], iss[2], 'green', nres / min);
                } else if (min * 12 < nres) {
                    setColor(iss[0], iss[1], iss[2], '#C27811', nres / min);
                } else if (min * 6 < nres) {
                    setColor(iss[0], iss[1], iss[2], '#A52A2A', nres/min);
		} else {
			setColor(iss[0], iss[1], iss[2], 'red', nres/min);
			applyBold(iss[0]);
			applyBold(iss[1]);
			applyBold(iss[2]);
		}
    } else if (iss[2].innerHTML.indexOf('+') > -1 ) {
		if (min * 6 + nres > nmax) {
			setColor(iss[0], iss[1], iss[2], '#5D0EDC', (nmax-nres)/min);
		} else
			setColor(iss[0], iss[1], iss[2], 'green', (nmax-nres)/min);
    }
    if (nres >= 100000) {
      applyBold(iss[0]);
    }
  }
   arr[resId] = nres;
 }


if (location.href.indexOf('build?pos=') > -1) {
  //раскрасить улучшение
  var fonts = $x("//div[@class='build_title']")
  for (var i in fonts)
	processNode(fonts[i], arr);
  //раскрасить перестройку
  fonts = $x("//div[@class='build_rebuild']/table/tbody/tr/td")
  for (var i in fonts)
	processNode(fonts[i], arr);
  //раскрасить список строительства
  var itms = $x("//div[@id='build_prc_cont']");
  for (var j in itms) {
	processNode(itms[j], arr);
  }

 }

 var nasI = $x("//div[@class='chcol2 chcol_p1']/div[@class='aC']/nobr/span");
 if (nasI && nasI.length > 0) {
   //расчитать время до макс. населения
   var nasCurr = parseFloat($x("./span", nasI[0])[0].innerHTML);
   var nasMax = parseFloat($x("./span", nasI[0])[1].innerHTML);
   var dt = $x("//div[@class='aC']/nobr")[1];
   var temp = parseFloat(dt.innerHTML.match(/(\d+.\d|\d+)/)[1]);
   var ttime = 0;
   if (temp > 0) {
     ttime = 24*(nasMax - nasCurr)/temp;
   }
   var clr = 'green';
   if (ttime < 0) {
     clr = 'red';
     ttime = -1*ttime;
   }
   var nobr = $e('nobr', null, null, null, $x("//div[@class='chcol2 chcol_p1']/div[@class='aC']")[0]);
   $e('span', '[' + Math.round(ttime*10)/10 + ']', null, {float: 'right', fontWeight: 'bold', color: clr}, nobr);

 }

}

 function applyBold(iss) {
		iss.style.fontWeight='bold';
 }

function processNode(node, arr) {
    var spans = $x("./span", node);
    for (var spi in spans) {
        var nd = spans[spi];
        var childs = nd.childNodes;
        var i = 0;
        while (i < childs.length) {
            var nod = childs[i];
            if (nod.nodeName == 'IMG') {
                var txt = nod.className.substring(5, nod.className.length)
                var colorSpan = document.createElement('SPAN');
                var val = parseInt(childs[i + 1].nodeValue);
                if (arr[parseInt(txt)] >= val)
                    colorSpan.style.color = 'green';
                else {
                    colorSpan.style.color = '#A52A2A';
                    nod.title = nod.title + ': Нехватает ' + formatNumber(val - arr[parseInt(txt)], 3);
                }
                colorSpan.style.fontWeight = 'bold';
                colorSpan.innerHTML = childs[i + 1].nodeValue;
                nd.removeChild(childs[i + 1]);
                nd.insertBefore(colorSpan, nod.nextSibling);
            }
            i++;
        }
    }
}

 function setColor(el, el1, el2, clr, ttime){
 		el.style.color = clr;
 		el1.style.color = clr;
 		el2.style.color = clr;
 		var nfo = document.createElement('span');
 		nfo.setAttribute('style', 'float:right;font-weight:bold; color:' + clr);
 		nfo.innerHTML = '[' + Math.round(ttime * 10) / 10 + ']';
 		el1.parentNode.appendChild(nfo);
 }

function applyTidToUri(tid, url, notUsePrefix) {
  var s = document.location.href;
  if (url)
    s = url;
  var pref = '&';
  if (notUsePrefix)
     pref = '';
  if (s.indexOf('tid=') == -1)
    return s+pref+'tid='+tid;
  else
    return s.replace(/tid\=\d+/, 'tid='+tid);
}

function formatNumber(num, digits) {
  var snum = new String(num);
  var ln = snum.length;
  var fmtNum = '';
  for (var i=ln-1; i>=0; i--) {
    fmtNum = snum.charAt(i) + fmtNum;
    if (i > 0 && (ln - i) % digits == 0) {
      fmtNum = '.' + fmtNum;
    }
  }
  return fmtNum;
}

function injectCityListInTrade(holder) {
    //reformat cities in trade list to new format
    var pTrade = $x("//select[@name='id']/option", null);
    for (var i in pTrade) {
        if (pTrade[i].value)
            pTrade[i].value = pTrade[i].value + '/0/0';
        else
            pTrade[i].value = '0/0/0';
    }
    var sTrade = $x("//select[@name='id']", null)[0];
    sTrade.options[sTrade.options.length] = new Option('-----------------', '0/0/0');
    for (var i in holder.townInfo) {
        var town = holder.townInfo[i];
        if (town.view == 2 || town.view == 4) {
            var coords = parseCityCoordinats(town.id);
            sTrade.options[sTrade.options.length] = new Option(town.name, coords[0]+'/'+coords[1]+'/'+coords[2]);
        }
    }
    sTrade.removeAttribute('onchange');
    sTrade.addEventListener('change', changeTradeCoord, false);
}

function parseCityCoordinats(id) {
    var inf = id.match(/(\d+)\/(\d+)\/(\d+)/);
    if (inf && inf.length > 3) {
        return [inf[1], inf[2], inf[3]];
    }
    inf = id.match(/(\d+)\/(\d+)/);
    if (inf && inf.length > 2) {
        return [0, inf[1], inf[2]];
    }
}

function injectTPCities() {
    var tplinks = $x("//div[@class='tradeway_3_sh']/div/table[@class='def_table_cw']/tbody/tr/td/a[contains(@href, 'towninfo?id=')]");
    if (tplinks.length > 0) {
        var sel = $x("//select[@name='id']")[0];
        $e('OPTION', '------ТП------', {disabled:true, name: 'id', value: '0'}, null, sel);
        for (var i in tplinks) {
            var id = parseInt(tplinks[i].href.match(/id=(\d+)/)[1]);
            $e('OPTION', tplinks[i].innerHTML, {name: 'id', value:id},null, sel);
        }
    }
}

function injectArmySpeed() {
    var armyTd = $x("//table[@id='armyetable']/tbody/tr/td");
    for (var i in armyTd) {
        var id = parseInt($x("./a", armyTd[i])[0].href.match(/id=(\d+)/)[1]);
        $e('TD', '<img class="icnu ic8" src="/p/_.gif" title="Скорость">' + unitSpeedHolder[id], null, null, armyTd[i].parentNode);
    }
}

function injectCities(pTrade, container, holder) {
    var txt = '<select id="injCities" name="id" ><option value="0/0/0"></option>';
	for (var ci in holder.cities.list) {
		var c = holder.cities.list[ci];
		txt +='<option value="' + c.id + '/0/0">'+ c.name + '</option>';
	}
	txt +='</select>';
	$e(container, txt, null, null, pTrade[0]);
	document.getElementById('injCities').addEventListener('change', changeTradeCoord, false);

}

function changeTradeCoord() {
    var inf = parseCityCoordinats($x("//select[@name='id']", null)[0].value);
    if (!inf) inf = document.getElementById('injCities').value;

    if (document.location.href.indexOf("market") != -1)
        document.location = 'http://' + currentHost + '/market?target=' + inf[0];
    else {
        _log("changeTradeCoord: " + document.location.href);
        var xCoord = $x("//input[@type='text' and @name='x']")[0];
        var yCoord = $x("//input[@type='text' and @name='y']")[0];
        var oCoord = $x("//input[@type='text' and @name='o']")[0];
        xCoord.value = inf[1];
        yCoord.value = inf[2];
        oCoord.value = inf[0];
    }
}

function getTownView(id, holder) {
  var view = 0;
  if ('view' in holder.townInfo[id])
    view = holder.townInfo[id].view;
  var result = { caption: '?', hint: '' };
  switch (view) {
    case 0:
    default:
      result.caption = '-';
      result.hint = 'не показано';
      break;
    case 1:
      result.caption = 'М<img src="'+imageList[3+view]+'" />';
      result.hint = 'мой город';
      break;
    case 2:
      result.caption = 'С<img src="'+imageList[3+view]+'" />';
      result.hint = 'участник страны';
      break;
    case 3:
      result.caption = 'К<img src="'+imageList[3+view]+'" />';
      result.hint = 'кормушка';
      break;
    case 4:
      result.caption = 'Д<img src="'+imageList[3+view]+'" />';
      result.hint = 'друг';
      break;
    case 5:
      result.caption = 'В<img src="'+imageList[3+view]+'" />';
      result.hint = 'враг';
      break;
  }
  return result;
}

function onShowMap(id, cityPanel){
  return function(){
  	try {
		cityPanel.action();
		var inf = parseCityCoordinats(id);
		location.href = 'http://' + currentHost + '/map?o=' + inf[0] + '&x=' + inf[1] + '&y=' + inf[2];
	} catch (e) {
		_log(e);
	}
  }
}

function onShowTown(id, cityPanel){
  return function(){
    cityPanel.action();
	var inf = parseCityCoordinats(id);
	location.href = 'http://' + currentHost + '/mapinfo?o=' + inf[0] + '&x=' + inf[1] + '&y=' + inf[2];
  }
}

function onShowPlayer(id, cityPanel){
  return function(){
    cityPanel.action();
    location.href = 'http://' + currentHost + '/account?id=' + id;
  }
}

function onDelTown(id, holder){
  return function(){
    // удаляем элемент
    delete holder.townInfo[id];
    holder.saveTownInfo();
    holder.updateTownPanel();
  }
}


function onChangeViewTown(id, holder) {
  return function(){
    // меняем видимость
    var view = 0;
    if ('view' in holder.townInfo[id])
      view = holder.townInfo[id].view;
    view++;
    if (view > 5)
      view = 0;
    holder.townInfo[id].view = view;

    holder.saveTownInfo();
    holder.updateTownPanel();
  }
}

function onDelPop(id, item, holder){
	return function(){
    	// удаляем элемент
    	holder.townInfo[id].pop.splice(item, 1);
    	holder.saveTownInfo();
    	holder.updateTownPanel();
  	}
}

function getArrayCount(arr) {
  var result = 0;
  for (var i in arr)
      result++;
  return result;
}

function injectTradeOps(holder) {
    if (wofh.town.trade.traders.busy > 0) {
       holder.cities.list[holder.cities.current].traders.total = wofh.town.trade.traders.count;
       holder.cities.list[holder.cities.current].traders.free =  wofh.town.trade.traders.count - wofh.town.trade.traders.busy;
       holder.cities.list[holder.cities.current].traders.market = wofh.town.trade.traders.reserve;
       _setValue('cities', _serialize(holder.cities));
    }
    var tradeContainer = new TradeContainer(holder.cities.list[holder.cities.current].traders.free * 250);
    tradeContainer.init();
}


//К форме тренировки войск добавляется количество юнитов, которое можно создать в течении суток
// напр. Требушет  [Пища] 100 [Древесина] 600 [Металл] 100 [Житель] 5  5:47:13(4.1)
function trameTimeCalculate() {
	var armyTd = $x("//form[@action='train']/div/div/table/tbody/tr/td[@class='utm']/span");
 	for (var i in armyTd) {
    	var txt = armyTd[i].innerHTML;
        var radio_id = $x("./td[@class='urd']/input", armyTd[i].parentNode.parentNode)[0].getAttribute('id');
        var input_id = $x("./div[@class='tunit_btns']/div[@class='cnt']/input", armyTd[i].parentNode.parentNode.parentNode.parentNode.parentNode.parentNode.parentNode)[0].getAttribute('id');
       	armyTd[i].innerHTML = txt + "<a href=\"javascript:void(0)\" id=\"setup_" + i + "\">[" + calcTime(txt) + "]</a>";
        document.getElementById('setup_' + i).addEventListener('click', setTraneCount(radio_id, input_id, Math.round(calcTime(txt)), false));
 	}
}

function setTraneCount(radio_id, input_id, count) {
    return function() {
        $q(radio_id).checked = true;
        $q(input_id).value = count;


        var evObj = document.createEvent('UIEvents');
        evObj.initUIEvent( 'keyup', true, true, window, 1 );
        evObj.keyCode = 13;
        $q(input_id).dispatchEvent(evObj);
    }
}

function calcTime(str) {
	var tm = str.split(':');
  	var tm1 = tm[1];
  	if (tm1.indexOf('0') == 0) {
    	tm1 = tm1.substring(1,tm1.length);
  	}
  	var tt = parseInt(tm[0])*60*60 + parseInt(tm1)*60 + parseInt(tm[2]);
  	return Math.round(24*60*60/tt*10)/10;
}

function collectTrainInfo(holder, panel) {
    var timeToFinish;


	holder.cities.list[holder.cities.current].train = {};
    if (document.location.href.indexOf('town') != -1 || document.location.href.indexOf('build?pos=') != -1) {
        for (var i=0; i < wofh.town.events.items.length; i++) {
            var event = wofh.town.events.items[i];
            
            if (event.cls.name == 'masstrn') {
                //_log(lib.units.list[event.data.unit].name);
                
                var trInfo = {};
                trInfo.building = lib.builds[wofh.town.build.slots[event.slot].id].name;
                trInfo.level = wofh.town.build.slots[event.slot].level;
                trInfo.unitId = event.data.unit;
                trInfo.unitCount = event.data.count;                
                trInfo.timeEnd = wofh.core.calcTrainEventData(wofh.town, wofh.town.build.slots[event.slot], wofh.town.events.items[i]).time;
                holder.cities.list[holder.cities.current].train[i] = trInfo;
		        _setValue('cities', _serialize(holder.cities));
		        panel.renderContent(holder);
            }
        }
    }
    /*
	if (document.location.href.indexOf('build?pos=') != -1) {
		var tr = $x("//div[@class='tunit_cont']/form[contains(@action, 'train')]");
		var buildDiv =  $x("//div[@class='build_train']/div[@class='desc']");
        if (tr.length == 0 && buildDiv.length == 0)
			return;

        var trInfo = {};
        var bpos = parseInt(document.location.href.match(/pos=(\d+)/)[1]);
        var info = $x("//div[@id='cont04']/div[@class='pagetitle']/div");
        var txt = info[0].innerHTML;
        var nfo = txt.match(/<a .*>(.*)<\/a> - (\d+)/);
        trInfo.building = nfo[1];
        trInfo.level = parseInt(nfo[2]);
        if (tr.length > 0) {
            // Присутвует форма тренировки - здание свободно
            trInfo.unitId = -1;
            trInfo.unitCount = 0;
            trInfo.timeEnd = 0;
        } else {
			if (buildDiv.length > 0) {
				// идет производство войск
				var hr = $x("a[contains(@href,'unitinfo?id=')]", buildDiv[0])[0].href;
				trInfo.unitId = parseInt(hr.match(/id=(\d+)/)[1]);
				
				// быстрохак
				if (currentHost.indexOf("waysofhistory") != -1) {
			        trInfo.unitCount = parseInt(buildDiv[0].innerHTML.match(/Ready:\s*(\d+)\s*of\s*(\d+)/)[2]);
			        var s = $x("span/span[contains(@id, 'timer')]", buildDiv[0]);
                    var found = null;
                    if (s.length > 0) {
                        s = s[0].innerHTML;
                        found = s.match(/(\d+)\:(\d+)\:(\d+)/);
                    } else {
                        s = $x("./span[@class='td']", buildDiv[0])[0].innerHTML;
                        var days = s.match(/(\d+).(\d+) days/);
                        var minutes = (parseInt(days[1]) + parseInt(days[2])/10) * 24 * 60 * 60;
                        found = ['0', '0', '00', '' + Math.round(minutes)] ;
                    }
                    var found2 = found[2];
                    if (found2.indexOf('0') == 0) {
                        found2 = found2.substring(1,found2.length);
                    }
                    trInfo.timeEnd = new Date().getTime() + (parseInt(found[1])*60*60 + parseInt(found2)*60 + parseInt(found[3]))*1000; 
			    } else {
			        trInfo.unitCount = parseInt(buildDiv[0].innerHTML.match(/Готово:\s*(\d+)\s*из\s*(\d+)/)[2]);
                    var s = $x("./span[@class='td build_train_timer']", buildDiv[0])[0].getAttribute("data-time");
                    timeToFinish = parseInt(s, 10) * 1000;
                    trInfo.timeEnd = new Date().getTime() + timeToFinish;
                }
            }
		}
        holder.cities.list[holder.cities.current].train[bpos] = trInfo;
		_setValue('cities', _serialize(holder.cities));
		panel.renderContent(holder);
	} else {

	var trExists = $x("//div[@id='cont04']/div[@class='pagecont3']/span/div[@class='build_train']");
	if (trExists.length ==0)
	  return;
	var trainDiv = $x("//div[@id='cont04']/div[@class='pagecont3']");
    var trNodes = trainDiv[0].childNodes;
	var trCnt = -1;
	for (var i in trNodes) {
	    var trInfo = null;
	    try {
			if (trNodes[i].nodeName == 'P') {
				//Получаем информацию о военном строении и его уровне
				var descr = $x(".//a[contains(@href,'build?pos=')]/span", trNodes[i]);
				if (descr.length == 0)
					continue;
				trInfo = {};
				trInfo.building = descr[0].innerHTML;

				var hr = $x(".//a[contains(@href,'build?pos=')]", trNodes[i])[0].href;
				trCnt = parseInt(hr.match(/pos=(\d+)/)[1]);

				var s = trNodes[i].lastChild.lastChild.nodeValue;
				trInfo.level = parseInt(s.match(/\d+/)[0]);
				trInfo.unitId = -1;
				holder.cities.list[holder.cities.current].train[trCnt] = trInfo;
			}
			if (trNodes[i].nodeName == 'SPAN') {
				//Получаем информацию о строящихся войсках
				var trInfo = holder.cities.list[holder.cities.current].train[trCnt];
				var buildDiv =  $x(".//div/div[@class='desc']", trNodes[i]);
				if (buildDiv.length > 0) {
					//	идет производство войск
					var hr = $x(".//a[contains(@href,'unitinfo?id=')]", buildDiv[0])[0].href;
					trInfo.unitId = parseInt(hr.match(/id=(\d+)/)[1]);
					
					// быстрохак 
					if (currentHost.indexOf("waysofhistory") != -1) {
					    
			            trInfo.unitCount = parseInt(buildDiv[0].innerHTML.match(/Ready:\s*(\d+)\s*of\s*(\d+)/)[2]);
			            var s = $x("span/span[contains(@id, 'timer')]", buildDiv[0]);
                        var found = null;
                        if (s.length > 0) {
                            s = s[0].innerHTML;
                            found = s.match(/(\d+)\:(\d+)\:(\d+)/);
                        } else {
                            s = $x("./span[@class='td']", buildDiv[0])[0].innerHTML;
                            var days = s.match(/(\d+).(\d+) days/);
                            var minutes = (parseInt(days[1]) + parseInt(days[2])/10) * 24 * 60 * 60;
                            found = ['0', '0', '00', '' + Math.round(minutes)] ;
                        }
                        var found2 = found[2];
                        if (found2.indexOf('0') == 0) {
                            found2 = found2.substring(1,found2.length);
                        }
                        alert (found);
                        trInfo.timeEnd = new Date().getTime() + (parseInt(found[1])*60*60 + parseInt(found2)*60 + parseInt(found[3]))*1000;
			            
			        } else {
			            trInfo.unitCount = parseInt(buildDiv[0].innerHTML.match(/Готово:\s+(\d+)\s*из\s*(\d+)/)[2]);
					    var s = $x("./span[@class='td build_train_timer']", buildDiv[0])[0].getAttribute("data-time");
                        timeToFinish = parseInt(s, 10) * 1000;
                        trInfo.timeEnd = new Date().getTime() + timeToFinish;
                    }
                    
					holder.cities.list[holder.cities.current].train[trCnt] = trInfo;
				}
			}
	    } catch (except) {
			_log(except);
	    }

	    if (trInfo && trCnt >-1)
			holder.cities.list[holder.cities.current].train[trCnt] = trInfo;
	}
	_setValue('cities', _serialize(holder.cities));
    
    }
    */
}

function onRemoveTrainBox(ci, i, holder, panel){
  return function(){
    if (confirm("Вы точно хотите удалить строение из списка?")) {
    	delete holder.cities.list[ci].train[i];
      	_setValue('cities', _serialize(holder.cities));
		panel.renderContent(holder);
    }
  }
}

function trainTimer(holder, panel) {
	for (var ci in holder.cities.list) {
		var trInfoArr = holder.cities.list[ci].train;
		for (var i in trInfoArr) {
			if (trInfoArr[i] && trInfoArr[i].timeEnd) {
				if (trInfoArr[i].timeEnd - new Date().getTime() <=0) {
					document.getElementById("tr_" + ci + "_" + i).innerHTML = 'постройка окончена';
					//Ахтунг, здание простаивает!!! А игрок не в курсе!
					$q(panel.title.id).style.backgroundColor = panel.title.alertBackground;
				} else {
					document.getElementById("tr_" + ci + "_" + i).innerHTML =  new Date(trInfoArr[i].timeEnd - new Date().getTime()).formatTime();
				}
			}
		}
	}
	var visible = _getValue(panel.content.visibleKey, panel.content.visible);
	if (visible)
		for (var ci in holder.cities.list) {
			var trInfoArr = holder.cities.list[ci].tradeOperation;
			for (var i in trInfoArr) {
				if (trInfoArr[i] && trInfoArr[i].time) {
					if (trInfoArr[i].time - new Date().getTime() <=0) {
						document.getElementById("op_" + ci + "_" + i).innerHTML = 'просрочено';
					} else {
						document.getElementById("op_" + ci + "_" + i).innerHTML =  new Date(trInfoArr[i].time - new Date().getTime()).formatTime();
					}
				}
			}
		}
		setTimeout(function(){
			trainTimer(holder, panel)
		}, 1000);
}

var nas = 0;
var cr = 0;
var pp = 0;
function injectEconomicCalc(holder) {

	var infoDivs = $x("//div[@id='econstats']/table/tbody/tr/td");
	var pNode = infoDivs[0].lastChild;
	var eNode = infoDivs[2];
	var cNode = infoDivs[4];

	nas = parseInt(pNode.nodeValue);
	infoDivs[0].replaceChild($e('SPAN', nas, {id: 'info_div_0_value'}), pNode);

	pp = eNode.innerHTML.match(/(\d+)/)[1];
	cr = cNode.innerHTML.match(/(\d+)/)[1];

	eNode.replaceChild($t('Потенциал: '), eNode.firstChild);
	eNode.insertBefore($e('SPAN', pp, {id: 'info_div_1_value'}), eNode.lastChild);
	eNode.insertBefore($t(' '), eNode.lastChild);

	cNode.replaceChild($t('Коррупция: '), cNode.firstChild);
	cNode.insertBefore($e('SPAN', cr, {id: 'info_div_2_value'}), cNode.lastChild);
	cNode.insertBefore($t('% '), cNode.lastChild);

	var trElement = $x("//div[@id='econstats']/table/tbody");
	var changeNode = $e('TD', null, {colspan : 7}, {color: '#000000'}, $e('TR', null, null, null, trElement[0]));

	var content = [];
	content.push('Население: ');
	content.push('<a href="javascript: void(0)" id="info_div_0_add_10" >+10</a>');
	content.push('/<a href="javascript: void(0)" id="info_div_0_rm_10" >-10</a>   ');
	content.push('<a href="javascript: void(0)" id="info_div_0_add_100" >+100</a>');
	content.push('/<a href="javascript: void(0)" id="info_div_0_rm_100" >-100</a>  ');
	content.push('<a href="javascript: void(0)" id="info_div_0_add_1000" >+1K</a>');
	content.push('/<a href="javascript: void(0)" id="info_div_0_rm_1000" >-1K</a>');
	content.push('  Уровень суда: ');
	var lvls = [1,1.3,1.7,2.1,2.6,3.1,3.6,4.1,4.6,5.2,5.8,6.3,6.9,7.5,8.1,8.7,9.4,10,10.6,11.3,11.9];
	content.push('<select id="info_sel_change">');
	for (var i in lvls) {
		content.push('<option value="');
		content.push(lvls[i]);
		content.push('">Суд ');
		content.push(i);
		content.push('</option>');
	}
	content.push('</select>');

	changeNode.innerHTML = content.join('');

  $q('info_sel_change').addEventListener('change', onRecalcCorrupt(holder), false);

  $q('info_div_0_add_10').addEventListener('click', onRecalcNas(10), false);
  $q('info_div_0_rm_10').addEventListener('click', onRecalcNas(-10), false);
  $q('info_div_0_add_100').addEventListener('click', onRecalcNas(100), false);
  $q('info_div_0_rm_100').addEventListener('click', onRecalcNas(-100), false);
  $q('info_div_0_add_1000').addEventListener('click', onRecalcNas(1000), false);
  $q('info_div_0_rm_1000').addEventListener('click', onRecalcNas(-1000), false);

}


function onRecalcCorrupt(holder){
  return function(){
    var effect = $q('info_sel_change').value;

    var corr = -20;
    for (var c in holder.cities.list) {
      corr+=20;
    }
    cr = corr/effect;
    if (cr > 90)
      cr = 90;
    $q("info_div_2_value").innerHTML = Math.round(cr);
    recalcEconomic();
  }
}

function onRecalcNas(val){
  return function(){
    if (nas + val < 0)
      return;
    nas = nas + val;
    pp = Math.pow(nas, 0.85) + 10;
    $q("info_div_0_value").innerHTML = nas;
    $q("info_div_1_value").innerHTML = Math.round(pp);

    recalcEconomic();
  }
}


function recalcEconomic() {
  var datas = $x("//div[@id='epro']/table[@id='tbprod']/tbody/tr");
  for (var i in datas) {
      var cur_value = 0;
      var next = false;
      var tds = $x("td", datas[i]);
      for (var j in tds) {
		if (tds[j].getAttribute('class') != 'tc') {
			if (next) {
				var foundBonus = tds[j].innerHTML.match(/(\d+)\+(\d+)/);
				tds[j].innerHTML = Math.round(cur_value);
				if (foundBonus) {
					tds[j].innerHTML = tds[j].innerHTML + '+' + foundBonus[2];
				}
				continue;
			}
			if (tds[j].innerHTML.indexOf('=') > -1) {
				next = true;
				continue;
			}
			var ext_found = tds[j].innerHTML.match(/(\d+) \* (\d+)\%( \* (\d+)%)?/);
			if (ext_found) {
				cur_value += pp * (100 - cr) / 100;
				cur_value = cur_value * parseInt(ext_found[2]) / 100;
				if (ext_found[3] && ext_found[4]) {
					cur_value = cur_value * parseInt(ext_found[4]) / 100;
				}
				continue;
			}
			var found = tds[j].innerHTML.match(/(\d+)/);
			if (found) {
				var value = found[1];
				if (tds[j].innerHTML.indexOf('%') > -1) {
					cur_value = cur_value * parseInt(value) / 100;
				} else  {
					cur_value += pp *(100 - cr) / 100;
				}
			}
		}
      }
      next = false;
  }
}
function changeBuildPanelView() {
    return function() {
        _setValue('build_info_panel', !_getValue('build_info_panel'));
        window.location.reload();
    }
}

function buildingInfo() {
    return function () {
        var total = 0;
        var line = "";
        for (var i = 0; i < wofh.town.build.slots.length; i++) {
            try {
                var id = wofh.town.build.slots[i].id;
                if (id >= 0) {
                    var bTitle = lib.builds[id].name;
                    line += bTitle + ' - ';
                    var cc = 0;
                    var bhelper = new Build(id);
                    var level = wofh.town.build.slots[i].level;
                    for (var j = 1; j <= level; j++) {
                        var cost = bhelper.getCostLevel(j);
                        for (var k in cost) {
                            if (cost[k] > 0) {
                                total += cost[k];
                                cc += cost[k];
                            }
                        }
                    }
                    line += cc;
                    line += '\n';
                }
            } catch (e) {
                _log(e.toString());
            }
        }
        alert('Прочность города: ' + total + '\n' + line);
    }
}

function applyHelpToBuildings() {

    //looking for script injected panel
    var buildings = $x("//div[@class='pmaincont']/a[contains(@href, 'build?pos=') and contains(@class, 'build_icon_s')]", null);
    
    //not found? looking for native panel
    if (buildings.length == 0) {
        if ($x("//div[@class='pmaincont']/a[contains(@href, 'build?pos=')]/div[@class='pmaintg']").length > 0) {
            buildings = $x("//div[@class='pmaincont']/a[contains(@href, 'build?pos=')]");
        }
    }

    for (var i in buildings) {
        //_log(buildings[i]);
        buildings[i].addEventListener("mouseover", showTooltip(buildings[i]), false);
        buildings[i].removeAttribute("title") ;
        buildings[i].addEventListener("mouseout", hideTooltip, false);

        //buildings[i].addEventListener('mousedown', showHelp(buildings[i].href), false);
    }
    
}

function showHelpMess(bld) {
    return function() {
        _log(bld);
    }
}

function showTooltip(link) {
    return function() {
        var tt = $q('help_div');
        var href = $x('./img',link);
        if (href.length > 0) {
            href = href[0].src;
        } else {
            href = link.style.backgroundImage;
        }
	    var id = parseInt(href.match(/\/p(\/g)?\/b\/(\d+)_/)[2]);
        var mapbuildings = $x("//area[contains(@href, 'build?pos=')]");
        var title = "";
        for (var i in mapbuildings) {
            if (link.href == mapbuildings[i].href)
               title = mapbuildings[i].title.match(/уровень\s*(\d+)/);
        }
        var level = parseInt(title[1]);
        var bhelper = new Build(id);
        var text = link.title + '<hr><b>Сл. уровень.</b><BR>';
        if (level >=20) {
            text += '<b>Максимальный уровень</b>';
        } else {
            var cost = bhelper.getCostLevel(level + 1);
		    for (var k in cost) {
                var  st = '';
                if (arr[k] >= cost[k])
 					st = 'green';
 				else
 					st = '#A52A2A';
                if (cost[k] > 0) {
                text += '<nobr><span style="color: ' + st + '"><img class="res r' + k + '" src="/p/_.gif">'+ cost[k] + '</span></nobr> '; 
                }
		    }
            text += '<nobr><span><img src="/p/_.gif" class="stimer"> ' + bhelper.getTime(level + 1).toString().toHHMMSS() + '</span></nobr>';

        }
        text += '<hr><b>Тек. эффект/стоимость/спад</b><br>';
        var cpay = 0;
        if (wofh.account.race == 3) {
            cpay = bhelper.getPay(level, true);
            
        } else {
            cpay = bhelper.getPay(level, false);
        }
        if (cpay < 0)
            cpay = 0;
        var cungrown = (utils.oddFunc(bhelper.getAttributes(id).ungrown, level) * wofh.town.pop.incReal / wofh.town.pop.inc);
        text += bhelper.getEffectLevel(level) + '&nbsp;/&nbsp;<img class="res r1" src="/p/_.gif" >' + cpay + '&nbsp;/&nbsp;' + cungrown.toFixed(2);
        if (level < 20) {
            var npay = 0;
            if (wofh.account.race == 3) {
                npay = bhelper.getPay(level + 1, true);

            } else {
                npay = bhelper.getPay(level + 1, false);
            }
            if (npay < 0)
                npay = 0;
            var nungrown = (utils.oddFunc(bhelper.getAttributes(id).ungrown, level + 1) * wofh.town.pop.incReal / wofh.town.pop.inc);
            text += '<hr><b>+1 ур. эффект/стоимость/спад</b><br>';
            text += bhelper.getEffectLevel(level + 1) + '&nbsp;/&nbsp;<img class="res r1" src="/p/_.gif" >' + npay + '&nbsp;/&nbsp;' + nungrown.toFixed(2);
        }
        tt.innerHTML = '<div>' + text + '</div>';

        tt.style.left = _getX(link) + 'px';
        tt.style.top = _getY(link) - 180 + 'px';
        tt.style.visibility = 'visible';
        tt.style.display = '';
    }
}

function hideTooltip() {
    $q('help_div').style.visibility = 'hidden';
}

function _getX(e){
    var x = 0;
    while (e) {
        x += e.offsetLeft;
        e = e.offsetParent;
    }
    return x;
}

function _getY(e){
    var y = 0;
    while (e) {
        y += e.offsetTop;
        e = e.offsetParent;
    }
    return y;
}

// end common functions

function parseBuildingInfo() {
	var datas = $x("//table[@class='def_table']/tbody/tr");
	var id = document.location.href.match(/id=(\d+)/)[1];
	var infoLine = id + ':{';
	for (var i in datas) {
		var children = datas[i].childNodes;
		var time = children[0].childNodes[0].getAttribute("title");
		var lvl = parseInt(children[0].childNodes[1].nodeValue);
		
		var cost = $x("./nobr", children[1]);
		var costClass = '';
		for (var j in cost) {
			var className = cost[j].childNodes[0].getAttribute("class");
			var resType = className.match(/res r(\d+)/)[1];
			var resCount = cost[j].childNodes[1].nodeValue;
			costClass += resType + ':' + resCount;
			if (j < cost.length - 1)
				costClass +=', '; 
		}
		
		var price = 0;
		if (children[2].childNodes.length > 0) {
			price = parseInt(children[2].childNodes[1].nodeValue);
		}
		var effect = children[3].innerHTML;
		var spad = 0;
		if (children[4].innerHTML.length > 0) {
			spad = parseFloat(children[4].innerHTML);
		}
		
		infoLine += lvl + ': {time:\\\'' + time + '\\\', cost:{' + costClass +'}, price:' + price + ', effect:\\\'' + effect + '\\\', spad:' + spad + '}';
		if (i < datas.length - 1)
				infoLine +=', '; 
	}
	infoLine +='}';
	_log(infoLine);
}

function addBuildInfoPanel() {


    var text = '<div class="pmaincont">';
    var blds = $x("//div[@id='cont04']/div[@class='tmain']/div[contains(@style, 'width:auto')]");
    var c_url = null;
    var c_title = null;
    var c_lvl = null;
    var textAreas = [];
    var merr_img = null;
    for (var i in blds) {
        if (i == 0) {
            try {
            var img = $x("./a/img", blds[i])[0];
            c_url = img.src;
            c_title = img.title;
            try {c_lvl = $x("./a/div/div",blds[i])[0].className; } catch (e) {};
            var group = img.src.match(/.*p\/(?:g\/)?b\/(\d+)_(.*)_.*/);
            if (group) {
            var lvlId = buildLevelMap[parseInt(group[1])];
            if (!lvlId && lvlId !=0) {
                merr_img = '<img width="45" height="46" src="' + img.src + '" />';
                if (group[2].indexOf("b") !=-1) lvlId =  group[2].substring(0, group[2].length -1); else lvlId= group[2];
            }
            c_url = '&quot;/p/b/bg/ss0.png&quot;';
            } else {
                c_url = '&quot;/p/b/sw.gif&quot;';
            }
            } catch (e) {_log(e)}
            continue;
        }
        var href = $x("./a",blds[i])[0].href;
        var img = $x("./a/img", blds[i])[0];
        var lvlClass = null;
        var group = img.src.match(/.*p\/(?:g\/)?b\/(\d+)_(.*)_.*/);
        var style =  '/p/b/s0/sm.png';
        if (group) {
            style = img.src;
        } else {
            group = img.src.match(/.*p\/(?:g\/)?b\/s(\d+)\/(.*).png/);
            if (group && group[2].indexOf('a') == -1) style = '/p/b/s' + group[1] + '/s.png';
        }
        try {lvlClass = $x("./a/div/div",blds[i])[0].className; } catch (e) {};

        var area_index = parseInt(href.match(/pos=(\d+)/)[1]);
        var area_text = '<a href="' + href + '" style="background: url(/p/b/bg/ss0.png)" title="' + img.title + '">'
         + '<img width="45" height="46" src="' + style + '" />';
        if (lvlClass && group[2].indexOf("b") != -1 && group[2].length > 1)  {
            area_text +='<div class="pmaintb"><div class="' + lvlClass + '"></div></div>'
        } else if (lvlClass && lvlClass.indexOf('mnnd') != -1) {
            area_text +='<div class="pmaintd"><div class="' + lvlClass + '"></div></div>';
        } else if (lvlClass) {
            area_text +='<div class="pmaint"><div class="' + lvlClass + '"></div></div>';
        }
        area_text += '</a>';
        textAreas[area_index] = area_text;
    }
    for (var i = 1; i<19; i++) {
        if (textAreas[i]) {
            text += textAreas[i];
        } else {
            text += '<a href="/build/pos=' + i + '" style="background: url(/p/b/bg/ss0.png)" >'
         + '<img width="45" height="46" src="/p/b/s0/sm.png" />'
        }
    }


    try {
    text += '<a href="build?pos=0" style="left:-95px; top:-70px; poisition:absolute; z-index:1000; background: url('
     + c_url + ')" title="' + c_title + '">';
        if (merr_img) text += merr_img;
    if (c_lvl) {
        text += '<div class="lalala-pmaint"><div class="' + c_lvl + '"></div></div>';
    } } catch (e) {_log(e)};
    text += '</a>'
    text += '</div>';
    var div = $e('DIV', text, {class : 'pmain'}, {display: 'block'});
    var table = $x("//div[@id='cont04']/div[@class='tmain']")[0];
    table.parentNode.appendChild(div);
}

function calcDistance(x1, y1, x2, y2) {
  return Math.sqrt((x1 - x2) * (x1 - x2) + (y1 - y2) * (y1 - y2));
}

function roundNumber(num, dec) {
	var result = Math.round(num*Math.pow(10,dec))/Math.pow(10,dec);
	return result;
}

function onMapDivNodeInserted() {
	if (injectMapDivProcessing)
		return;
	parseMapPage();
}

function switchMapHelp() {
    mapHelpEnabled = !mapHelpEnabled;
    document.getElementById('inf_tt_ico').setAttribute('class', !mapHelpEnabled ? 'a_osn_yes' : 'a_pos_no');
    var allPlayerCitiesHolder = _deserialize(_getValue("all_player_cities_holder", '{}'));
    var cities = _deserialize(_getValue('cities', '{ current: null, list: [] }'));
    var baseCoords = allPlayerCitiesHolder[cities.list[cities.current].id];
    if (baseCoords) {
        var landLinks = $x("//div[@id='landlnks']/a");
        for (var i in landLinks) {
            var coords = landLinks[i].href.match(/x=(\d+)&y=(\d+)/);
            var range = calcDistance(baseCoords.x, baseCoords.y, parseInt(coords[1]), parseInt(coords[2]));
            applyMapOpacity(landLinks[i], range, mapHelpEnabled);
        }
    }
}

function applyMapOpacity(link, range, enabled) {
    if (enabled) {
        link.style.opacity = 0.4;
        link.style.backgroundColor = range > 25 ? '#e83838' : '#0ce70c';
    } else {
        link.style.opacity = null;
        link.style.backgroundColor = null;
    }
}

var mapHelpEnabled = false;
var injectMapDivProcessing = false;
function parseMapPage() {
    injectMapDivProcessing = true;
    var allPlayerCitiesHolder = _deserialize(_getValue("all_player_cities_holder", '{}'));
    var cities = _deserialize(_getValue('cities', '{ current: null, list: [] }'));
    var baseCoords = allPlayerCitiesHolder[cities.list[cities.current].id];
    if (baseCoords) {
    var landLinks = $x("//div[@id='landlnks']/a");
    for (var i in landLinks) {
         if (landLinks[i].getAttribute('ws_processed') == 'true')
            continue;
         var coords = landLinks[i].href.match(/x=(\d+)&y=(\d+)/);
         var range = calcDistance(baseCoords.x, baseCoords.y, parseInt(coords[1]), parseInt(coords[2]));
         landLinks[i].setAttribute('tooltip', 'Расстояние: ' + roundNumber(range, 3) + '. ' + landLinks[i].getAttribute('tooltip'));
         if (mapHelpEnabled) {
             applyMapOpacity(landLinks[i], range, true);
         }
         landLinks[i].setAttribute('ws_processed', 'true');

    }
    }
    injectMapDivProcessing = false;
}

function parseCityListPage() {

    //определяем персональную страницу игрока
    var pname = $x("//div[@class='ptitle2_ ptitle2_p1']/h2")[0].innerHTML.match(/[^\s]*? (.*)/)[1];
    var infoTr = $x("//div[@id='tb_town']/div[@class='town_row tB']");
    if (pname == playerName) {
        var allPlayerCitiesHolder = {};
        for (var i in infoTr) {
            var centroid = $x("./div[@class='conbut2']/div[@class='M_bt M_bt_cent fR']/a", infoTr[i])[0];
            var coords = centroid.href.match(/x=(\d+)&y=(\d+)/);
            var townLink = $x("./*/div[@class='t_name fwB']/a", infoTr[i])[0];
            var id = townLink.getAttribute("href").match(/id=(\d+)/)[1];
            allPlayerCitiesHolder[id] = {x: parseInt(coords[1]), y: parseInt(coords[2])};
        }
        _setValue("all_player_cities_holder", _serialize(allPlayerCitiesHolder));
        _log(allPlayerCitiesHolder);
    }
    var allPlayerCitiesHolder = _deserialize(_getValue("all_player_cities_holder", '{}'));
    var cities = _deserialize(_getValue('cities', '{ current: null, list: [] }'));
    var baseCoords = allPlayerCitiesHolder[cities.list[cities.current].id];
    for (var i in infoTr) {
            var centroid = $x("./div[@class='conbut2']/div[@class='M_bt M_bt_cent fR']/a", infoTr[i])[0];
            var coords = centroid.href.match(/x=(\d+)&y=(\d+)/);
            if (coords.length > 2) {
                var divInfoHolder = $x("./div[@class='t_n_cont']/div[@class='t_dist fR']", infoTr[i])[0];
                var range = calcDistance(baseCoords.x, baseCoords.y, parseInt(coords[1]), parseInt(coords[2]));
                if (divInfoHolder.innerHTML == '')
                divInfoHolder.appendChild($t('Расстояние: ' + roundNumber(range, 3)));
            }
    }


    
    var infoTr = $x("//div[@id='tb_town']/div[@class='town_row tB']");
    var pname = $x("//div[@class='ptitle2_ ptitle2_p1']/h2")[0].innerHTML.match(/[^\s]*? (.*)/)[1];
    var plId = document.location.href.match(/id=(\d+)/)[1];
    var allPlayersHolder = _deserialize(_getValue("all_players_holder", '{data: []}'));
    var cityInfoList = [];
    for (var i in infoTr) {
        var cityInfo = {};
        cityInfoList[i] = cityInfo;

        var cview = $x("./div", $x("./div", infoTr[i])[1]);
        if (cview.length > 1) {
            cityInfo.mestorod = cview[1].title;
        }

        if (!cityInfo.mestorod) {
            cityInfo.mestorod = 'нет';
        }
        var ct = $x("./div", infoTr[i])[3];
        cityInfo.link = $x("./div/a", ct)[0].href;
        cityInfo.name = $x("./div/a", ct)[0].innerHTML;
        cityInfo.nas = $x("./div", ct)[1].lastChild.nodeValue;

        var rast = $x('./nobr', $x("./div", ct)[2])[0];
        if (rast) {
            cityInfo.rast = rast.childNodes[0].nodeValue.match(/(\d+.\d)/)[1];
            cityInfo.put = rast.childNodes[1].innerHTML;
        } else {
            cityInfo.rast = '';
            cityInfo.put = '';
        }
    }
    var divNfo = $e('DIV');
    $x("//div[@id='tb_town']")[0].appendChild(divNfo);
    var lastData = allPlayersHolder.data[plId];
    divNfo.appendChild($e('DIV', 'Последнее обновление: ' + (lastData ? lastData.date : 'никогда'), {id: 'last_data_info_div'}));
    divNfo.appendChild($e('A', 'Добавить/Обновить', {id: 'last_data_info_href',  href: 'javascript:void(0)'}));
    $x("//div[@id='tb_town']")[0].appendChild($e('A', 'Показать таблицу', {id: 'show_table_link', href: 'javascript:void(0)'}));
    $q('show_table_link').addEventListener('click', showCityInfoList(allPlayersHolder), false);
    $q('last_data_info_href').addEventListener('click', addToCityInfoList(cityInfoList, plId, pname, allPlayersHolder), false);
    _log(cityInfoList);
    
}
function addToCityInfoList(cityInfoList, plId, pname, allPlayersHolder) {
    return function() {
        allPlayersHolder.data[plId] = {};
        allPlayersHolder.data[plId].date = new Date();
        allPlayersHolder.data[plId].player = pname;
        allPlayersHolder.data[plId].data = cityInfoList;
        $q('last_data_info_div').innerHTML = 'Последнее обновление: ' + allPlayersHolder.data[plId].date;
        _setValue("all_players_holder", _serialize(allPlayersHolder));
    }
}
function showCityInfoList(allPlayersHolder) {
    return function() {
        var text = '<div align="right"><a id="city_list_div_close" href="javascript:void(0);">Закрыть</a></div><table border="5px" style="color:black;"><tr><td>Игрок</td><td>id</td><td>Обновление</td></tr>';
        var textarea = 'Игрок\tID\tГород\tСсылка\tМесторождение\tНаселение\tРасстояние\tПуть\n';
        for (var i in allPlayersHolder.data) {
            if (!allPlayersHolder.data[i])
                continue;
            text += '<tr><td>' + allPlayersHolder.data[i].player +
                    '</td><td>' + i +
                    '</td><td>' + allPlayersHolder.data[i].date +
                    '</td></tr>';

            for (var j in allPlayersHolder.data[i].data) {
                var cityInfo = allPlayersHolder.data[i].data[j];
                textarea += allPlayersHolder.data[i].player + '\t'
                         + i + '\t' + cityInfo.name + '\t' + cityInfo.link + '\t'
                         + cityInfo.mestorod + '\t' + cityInfo.nas + '\t'
                         + cityInfo.rast + '\t' + cityInfo.put + '\n';
            }

        }
        text += '</table><textarea rows="10" cols="25">';
        text += textarea;
        text += '</textarea>';
        var tt = $e('DIV', text, {id: 'city_list_div', class: 'pane'}, {zIndex: 4000, left: '0px', top:'0px', backgroundColor: '#FF6'}, document.body);
        var table = $x("//div[@id='tb_town']")[0];
        tt.style.left = _getX(table) + 'px';
        tt.style.top = _getY(table) + 'px';
        document.getElementById('city_list_div_close').addEventListener('click', closeCityInfoList, false);
    }
}

function closeCityInfoList() {
    document.body.removeChild($q('city_list_div'));
}
function onTradeTableNodeInserted() {
	if (injectAdvTradeProcessing)
		return;
	injectAdvTrade()
}

var injectAdvTradeProcessing = false;
function injectAdvTrade(){

  injectAdvTradeProcessing = true;
  var tradeTr = $x("//table[contains(@class, 'trade_table')]/tbody/tr[contains(@class, 'tbl_row')]");

  for (var i in tradeTr) {
    var tradeTd = $x("td", tradeTr[i]);
	if (tradeTr[i].getAttribute('ws_processed') == 'true')
		continue;
    if (tradeTd.length == 7) {
	
	
	  var maxval = parseInt(tradeTd[2].lastChild.nodeValue!=null?tradeTd[2].lastChild.nodeValue:tradeTd[2].lastChild.lastChild.nodeValue); //максимум, который выставил продавец/покупатель
	  //покупаем или продаем?
	  var isSell = document.getElementById('r1').checked;
	  var maxMessage = 'max - 1';
	  var maxValue = maxval - 250;
	  
	  if (isSell) { //продаем. Пересчитать возможный максимум.
	        var maxByTraders = parseInt($x("//span[@id='inpb']")[0].innerHTML); //максимум, который могут перевезти торговцы.
			var maxByResources = parseInt($x("./nobr/img", tradeTd[2])[0].getAttribute('class').match(/res r(\d+)/)[1]);
			maxByResources = isInt(arr[maxByResources]) ? arr[maxByResources] : 0; //максимум данного ресурса на складе
			
			maxval = Math.min(maxval, maxByTraders, maxByResources);
			maxMessage = 'max';
			maxValue = maxval;
	  }
      
	  var l = $e('a', '+1', { href: 'javascript: void(0)', id: 'incb' + i });
      var l2 = $e('a', '+4', { href: 'javascript: void(0)', id: 'incbt' + i });
      var l3 = $e('a', '-1', { href: 'javascript: void(0)', id: 'decb' + i });
      var l4 = $e('a', '-4', { href: 'javascript: void(0)', id: 'decbt' + i });
      var lmax = $e('a', maxMessage, { href: 'javascript: void(0)', id: 'maxb' +i});

      tradeTd[5].appendChild($e('br'));
      tradeTd[5].appendChild(l);
      tradeTd[5].appendChild($t(' '));
      tradeTd[5].appendChild(l2);

      tradeTd[4].appendChild($e('br'));
      tradeTd[4].appendChild(l3);
      tradeTd[4].appendChild($t(' '));
      tradeTd[4].appendChild(l4);

      tradeTd[3].appendChild($e('br'));
      tradeTd[3].appendChild(lmax);

      var inpu = $x(".//input[@name='count']", tradeTd[6])[0];
      $q('incb' + i).addEventListener('click', onIncValue(inpu, 1, maxval), false);
      $q('incbt' + i).addEventListener('click', onIncValue(inpu, 4, maxval), false);
      $q('decb' + i).addEventListener('click', onIncValue(inpu, -1, maxval), false);
      $q('decbt' + i).addEventListener('click', onIncValue(inpu, -4, maxval), false);
      $q('maxb' + i).addEventListener('click', onMaxValue(inpu, maxValue), false);
	  
	  tradeTr[i].setAttribute('ws_processed', 'true');

    }
  }
  injectAdvTradeProcessing = false;

}

function onIncValue(elem, cnt, maxval) {
  return function() {
    var incval = 250*cnt;
    var s = elem.value;
    if (s.length == 0)
      s = 0;
    else
      s = parseInt(s);
    if (s + incval > maxval)
      s = maxval;
    else
      s += incval;
    if (s < 0) {
      s = 0;
    }
    elem.value = s;
  }
}

function onMaxValue(elem, maxval) {
	return function() 
	{        
        elem.value = maxval;
	}
}

var Resource = (function() {
        var actualConstructor = function(id, max, container, parent) {
            this.id = id;
            this.max = max;
            this.container = container;
            this.parent = parent;
            arguments.callee.instances.push(this);
        }
        actualConstructor.instances = [];
        return actualConstructor;
    })();

    Resource.prototype.capacity = function() {
        return this.container.value.length == 0 || isNaN(this.container.value) ? 0 : parseInt(this.container.value);
    }

    Resource.prototype.onAdd = function(count) {
        var mmax = this.parent.capacity(this.id);
        if (this.max < mmax)  mmax = this.max;

        var incval = 250*count;

        var s = this.capacity();
    	if (s + incval > mmax)
      		s = mmax;
    	else
      		s += incval;
    	if (s < 0) {
      		s = 0;
    	}
    	this.container.value = s;
        JSN.market.pages.send.calcCurCapacity();
    }

    Resource.prototype.onMax = function() {
        var mmax = this.parent.capacity(this.id);
        if (this.max < mmax) mmax = this.max;
        this.container.value = mmax;
        JSN.market.pages.send.calcCurCapacity();
    }

var TradeContainer = (function() {

    var actualConstructor = function(size) {
        this.size = size;
        this.resources = {};
        arguments.callee.instances.push(this);
    }
    actualConstructor.instances = [];
    return actualConstructor;
    }
)();

    TradeContainer.prototype.registerResource = function(id, max, container) {
        this.resources[id] = new Resource(id, max, container, this);
        addToContainer(container, $e('label', 'max', {href: 'javascript: void(0)', id: 'maxb' + id }, {color:'red'}));
        $q('maxb' + id).addEventListener('click', onMax(id, this), false);
        addToContainer(container, $e('label', '-4', { href: 'javascript: void(0)', id: 'decbt' + id }, {color: '#008000'}));
        $q('decbt' + id).addEventListener('click', onAdd(id, -4, this), false);
        addToContainer(container, $e('label', '-1', { href: 'javascript: void(0)', id: 'decb' + id }, {color: '#008000'}));
        $q('decb' + id).addEventListener('click', onAdd(id, -1, this), false);
        addToContainer(container, $e('label', '+4', { href: 'javascript: void(0)', id: 'incbt' + id }, {color: '#008000'}));
        $q('incbt' + id).addEventListener('click', onAdd(id, 4, this), false);
        addToContainer(container, $e('label', '+1', { href: 'javascript: void(0)', id: 'incb' + id }, {color: '#008000'}));
        $q('incb' + id).addEventListener('click', onAdd(id, 1, this), false);
    }

    function addToContainer(container, element) {
        container.parentNode.insertBefore(element, container.nextSibling);
        container.parentNode.insertBefore($t(' '), element);
    }

    TradeContainer.prototype.capacity = function(id) {
        var total = 0;
        for (var i in this.resources) {
            if (id != i)
            total += this.resources[i].capacity();
        }
        return this.size - total;
    }

    function onAdd(id, count, tradeContainer) {
        return function() {
            tradeContainer.resources[id].onAdd(count);
        }
    }

    function onMax(id, tradeContainer) {
        return function() {
            tradeContainer.resources[id].onMax();
        }
    }

    TradeContainer.prototype.init = function() {
        var elements = $x("//table[@id='inp']/tbody/tr/td/input[@type='text' and contains(@name, 'res')]");
        for (var i in elements) {
            var resId = elements[i].name.match(/res(\d+)/)[1];
            this.registerResource(resId, Math.floor(arr[resId]/250)*250, elements[i]);
        }
    }

var sortType = 0;
var sortOrder = 1;

function streamSort() {

    var sortArray = [];
    var tabs = $x("//table[@class='def_table trway']", null);
    var tabIndex = tabs.length > 1 ? 1 : 0;
    var lines = $x("./tbody/tr", tabs[tabIndex]);
    for (var i in lines) {
        var columns = $x("./td", lines[i]);
        var cHolder = {city: {}, res: {}, pay: {}, traders: {}, optype: {}, time: {}, panel: ''};
        cHolder.city.text = columns[0].innerHTML;
        cHolder.city.value = $x("./a", columns[0])[0].innerHTML;
        cHolder.res.text = columns[1].innerHTML;
        cHolder.res.value = parseInt(columns[1].lastChild.nodeValue);
        cHolder.res.type = parseInt($x("./img", columns[1])[0].getAttribute('class').match(/res r(\d+)/)[1]);
        cHolder.pay.text = columns[2].innerHTML;
        cHolder.pay.value = parseFloat(columns[2].lastChild.nodeValue);
        cHolder.traders.text = columns[3].innerHTML;
        cHolder.traders.value = 0;
        if (columns[3].childNodes.length > 0) {
            cHolder.traders.value = parseInt(columns[3].firstChild.nodeValue);
        }
        cHolder.optype.text = columns[4].innerHTML;
        cHolder.optype.value = columns[4].innerHTML == 'Покупка' ? true : false;
        cHolder.time.text = columns[5].innerHTML;
        try {
            cHolder.time.value = parseFloat(columns[5].innerHTML.match(/(\d+\.\d+) /)[1]);
        } catch (e) {
            cHolder.time.value = 0;
        }
        cHolder.panel = columns[6].innerHTML;
        sortArray[i] = cHolder;
    }

    var columns = $x("./thead/tr/td", tabs[tabIndex]);
    for (var i in columns) {
        columns[i].innerHTML='<a id="sort_col_' + i + '" href="javascript:void(0)" >' + columns[i].innerHTML + '</a>';
        document.getElementById('sort_col_' + i).addEventListener('click', startSort(i, sortArray), false);
    }
}

function startSort(type, sortArray) {
    return function() {
        if  (sortType == type) {
            sortOrder = -1 * sortOrder;
        } else {
            sortType = type;
            sortOrder = 1;
        }
        sortArray.sort(streamComparator);
        var tabs = $x("//table[@class='def_table trway']", null);
        var lines = $x("./tbody/tr", tabs[tabs.length - 1]);
        for (var i in lines) {
            var columns = $x("./td", lines[i]);
            columns[0].innerHTML = sortArray[i].city.text;
            columns[1].innerHTML = sortArray[i].res.text;
            columns[2].innerHTML = sortArray[i].pay.text;
            columns[3].innerHTML = sortArray[i].traders.text;
            columns[4].innerHTML = sortArray[i].optype.text;
            columns[5].innerHTML = sortArray[i].time.text;
            columns[6].innerHTML = sortArray[i].panel;
        }
    }
}

function streamComparator(a, b) {
    if (sortType == 0) {
        //sort by city
            var x = a.city.value.toLowerCase();
            var y = b.city.value.toLowerCase();
            return sortOrder*((x < y) ? -1 : ((x > y) ? 1 : 0));
    }
    if (sortType ==1) { //sort by resource type/count
            if (a.res.type == b.res.type) {
                return sortOrder*(a.res.value - b.res.value);
            } else {
                return sortOrder*(a.res.type - b.res.type);
            }
    }
    if (sortType ==2) { //sort by payment value
            return sortOrder*(a.pay.value - b.pay.value);
    }
    if (sortType ==3) { //sort by traders count
            return sortOrder*(a.traders.value - b.traders.value);
    }
    if (sortType ==4) { //sort by operation type
            return sortOrder*(a.optype.value == b.optype.value ? 0 : a.optype.value ? -1 : 1);
    }
    if (sortType ==5) { //sort by time
            return sortOrder*(a.time.value - b.time.value);
    }
    return 0;

}


function onUSUnload() {}

var is_chrome = /chrome/.test( navigator.userAgent.toLowerCase() );
if (!is_chrome){
    window.addEventListener('load', onUSLoad, false);
    window.addEventListener('unload', onUSUnload, false);
}else {
    onUSLoad();

}