Вход Регистрация
Файл: js/al/common.js
Строк: 7167
<?php
/*
vk.al = -1 - (обычная загрузка страниц через фрейм) ответ(вся реализация) во фрейме.
vk.al = 1 - (обычный ajax запрос) ответ через сплиты "<!>".
vk.al = 2 - если браузер не поддерживает history.pushState (). 
vk.al = 3 - нормальная работа, (сгенерированная сервером) загрузка страницы.
vk.al = 4 - перезагрузка страницы.
*/
if (vk.al == 1) { // Перезагружает страницу. 
  
if (location['search'] || location.pathname != '/') {
    
location.replace('/');
  }
} else { 
  if (
vk.al == && !history.pushStatevk.al 2;
  if (!
location['search'] && location.pathname == '/index.php') {
    
location.replace('/');
  }
  
vk.version false;
}

window.__debugMode true// Don't turn it off

if (!window._ua) {
  var 
_ua navigator.userAgent.toLowerCase();
}

if (!
window.locDomain) {
  var 
locDomain location.host.toString().match(/[a-zA-Z]+.[a-zA-Z]+.?$/)[0];
}

if (!
window.StaticFiles) {
  var 
StaticFiles = {};
}

// Парсинг json.
var parseJSON = (window.JSON && JSON.parse) ? function (obj) {
  try { return 
JSON.parse(obj); } catch (e) {
    
topError('<b>parseJSON:</b> ' e.message, {dt: -1type5answerobj});
    return eval(
'('+obj+')');
  }
} : function(
obj) {
  return eval(
'('+obj+')');
}

// cur.destroy - выполнения функций в destroy при переходе через nav.go.
// cur.nav - выполнения функций в nav при при переходе через nav.go (для перехвата запроса на текущ. стр.).
var cur = {destroy: [], nav: []};

// Cоответствие браузера, взрощает true если браузер совподает, иначе false
var browser = {
  
version: (_ua.match( /.+(?:me|ox|on|rv|it|era|ie)[/: ]([d.]+)/ ) || [0,'0'])[1],
  
opera: /opera/i.test(_ua),
  
msie: (/msie/i.test(_ua) && !/opera/i.test(_ua)),
  
msie6: (/msie 6/i.test(_ua) && !/opera/i.test(_ua)),
  
msie7: (/msie 7/i.test(_ua) && !/opera/i.test(_ua)),
  
msie8: (/msie 8/i.test(_ua) && !/opera/i.test(_ua)),
  
msie9: (/msie 9/i.test(_ua) && !/opera/i.test(_ua)),
  
mozilla: /firefox/i.test(_ua),
  
chrome: /chrome/i.test(_ua),
  
safari: (!(/chrome/i.test(_ua)) && /webkit|safari|khtml/i.test(_ua)),
  
iphone: /iphone/i.test(_ua),
  
ipod: /ipod/i.test(_ua),
  
iphone4: /iphone.*OS 4/i.test(_ua),
  
ipod4: /ipod.*OS 4/i.test(_ua),
  
ipad: /ipad/i.test(_ua),
  
android: /android/i.test(_ua),
  
bada: /bada/i.test(_ua),
  
mobile: /iphone|ipod|ipad|opera mini|opera mobi|iemobile/i.test(_ua),
  
msie_mobile: /iemobile/i.test(_ua),
  
safari_mobile: /iphone|ipod|ipad/i.test(_ua),
  
opera_mobile: /opera mini|opera mobi/i.test(_ua),
  
opera_mini: /opera mini/i.test(_ua),
  
mac: /mac/i.test(_ua)
};

// Кроссбраузерная инсталляция флеш контейнера.  
(function() {
  var 
flash = [000], axon 'ShockwaveFlash.ShockwaveFlash';
  var 
wrapType 'embed'wrapParam 'type="application/x-shockwave-flash" ';
  var 
escapeAttr = function(v) {
    return 
v.toString().replace('&''&amp;').replace('"''&quot;');
  }
  if (
navigator.plugins && navigator.mimeTypes.length) { // Версия флеша в браузере. 
    
var navigator.plugins['Shockwave Flash'];
    if (
&& x.description) {
      var 
ver x.description.replace(/([a-zA-Z]|s)+/, '').replace(/(s+r|s+b[0-9]+)/, '.').split('.');
      for (var 
03; ++iflash[i] = ver[i] || 0;
    }
  } else {
    if (
_ua.indexOf('Windows CE') >= 0) {
      var 
axo truever 6;
      while (
axo) {
        try {
          ++
ver;
          
axo = new ActiveXObject(axon '.' ver);
          
flash[0] = ver;
        } catch(
e) {}
      }
    } else {
      try {
        var 
axo = new ActiveXObject(axon '.7');
        
flash axo.GetVariable('$version').split(' ')[1].split(',');
      } catch (
e) {}
    }
    
wrapType 'object';
    
wrapParam 'classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000" ';
  }
  
browser.flashwrap = (wrapType == 'embed') ? function(optsparams) {
    
params extend({
      
idopts.id,
      
nameopts.id,
      
widthopts.width,
      
heightopts.height,
      
styleopts.style,
      
preventhideopts.preventhide
    
}, params);
    if (
browser.flash >= opts.version) {
      
params.src opts.url;
    } else {
      
params.src opts.express;
    }
    var 
paramsStr = [];
    for (var 
i in params) {
      var 
params[i];
      if (
!== undefined && !== null) {
        
paramsStr.push('="' escapeAttr(p) + '" ');
      }
    }
    return 
'<embed ' wrapParam paramsStr.join('') + '/>';
  } : function(
optsparams) {
    if (
browser.flash >= opts.version) {
      
params.movie opts.url;
    } else {
      
params.movie opts.express;
    }
    var 
attr = {
      
idopts.id,
      
widthopts.width,
      
heightopts.height,
      
styleopts.style,
      
preventhideopts.preventhide
    
}
    var 
attrStr = [];
    for (var 
i in attr) {
      var 
attr[i];
      if (
!== undefined && !== null) {
        
attrStr.push('="' escapeAttr(p) + '" ');
      }
    }
    var 
paramsStr = [];
    for (var 
i in params) {
      var 
params[i];
      if (
!== undefined && !== null) {
        
paramsStr.push('<param name="' '" value="' escapeAttr(p) + '" />');
      }
    }
    return 
'<object ' wrapParam attrStr.join('') +'>' paramsStr.join('') + '</object>';
  }
  if (
flash[0] < 7flash = [000];
  
browser.flash intval(flash[0]);
  
browser.flashfull = {
    
majorbrowser.flash,
    
minorintval(flash[1]),
    
revintval(flash[2])
  }
  
setCookie('remixflash'intval(flash[0]) + '.' intval(flash[1]) + '.' intval(flash[2]), 30);
})();

// Установка куков с размерами экрана для андроидов.
if (browser.android) {
  
setCookie('remixscreen_width'window.screen.width365);
  
setCookie('remixscreen_height'window.screen.height365);
  
setCookie('remixscreen_dpr'window.devicePixelRatio || 1365);
}

// Удаление статических файлов, если зашли не с IE.
if (!browser.msie6) {
  
delete StaticFiles['ie6.css'];
}
if (!
browser.msie7) {
  
delete StaticFiles['ie7.css'];
}

// Инсталляция, преобразование имени статик. файла для id div-ов в utils. 
for (var i in StaticFiles) {
  var 
StaticFiles[i];
  
f.= (i.indexOf('.css') != -1) ? 'css' 'js'
  
f.i.replace(/[/.]/g'_'); //превращает "." в "_"
  
f.0;
  
f.0;
}

window.locHost location.host;
window.locProtocol location.protocol;
window.__dev true;///[a-z0-9_-]+.[a-z0-9_-]+.[a-z0-9_-]+.[a-z0-9_-]+/i.test(locHost);
if (!__dev__debugMode false;
window.locHash location.hash.replace('#/''').replace('#!'''); // Убирает #/ и #!.
window.locBase location.toString().replace(/#.+$/, ''); // Убирает хэш теги.

// Всплывающее окно сверху. 
function topMsg(textsecondscolor) {
  if (!
colorcolor '#D6E5F7';
  if (!
text) {
    
hide('system_msg');
  } else {
    
clearTimeout(window.topMsgTimer);
    var 
el ge('system_msg');
    
el.style.backgroundColor color;
    
el.innerHTML text;
    
show(el);
    if (
seconds) {
      
window.topMsgTimer setTimeout(topMsg.pbind(false), seconds 1000);
    }
  }
}

// Всплывающее окно ошибки сверху.
function topError(textopts) {
  if (!
optsopts = {};
  if (
text.message) {
    var 
text;
    
text '<b>JavaScript error:</b> ' e.message;
    
opts.stack e.stack;
    if (
e.stack && __debugModetext += '<br/>' e.stack.replace(/n/g'<br/>');
    try { 
console.log(e.stack); } catch (e2) {};
  }
  if (!
opts.stack) {
    try {eval(
'0 = 1');} catch(e) {
      
opts.stack e.stack;
    }
  }

  if (
opts.dt != -1) {
    
topMsg(textopts.dt'#FFB4A3');
  }
  if (!
__dev && !ge('debuglogwrap')) {
    
delete(opts.dt);
    
ajax.plainpost('/errors.php'extend(opts, {msgopts.msg || textmodule: (window.cur || {}).moduleidvk.idhostlocHostlangvk.langloc: (window.nav || {}).strLocrealloclocation.toString()}));
  }
}

//...
function langNumeric(countvarsformatNum) {
  if (!
vars || !window.langConfig) { return count; }
  var 
res;
  if (!
isArray(vars)) {
    
res vars;
  } else {
    
res vars[1];
    if(
count != Math.floor(count)) {
      
res vars[langConfig.numRules['float']];
    } else {
      
each(langConfig.numRules['int'], function(i,v){
        if (
v[0] == '*') { res vars[v[2]]; return false; }
        var 
v[0] ? count v[0] : count;
        if(
indexOf(v[1], c) != -1) { res vars[v[2]]; return false; }
      });
    }
  }
  if (
formatNum) {
    var 
count.toString().split('.'), = [];
    for(var 
n[0].length 3> -3-= 3) {
      
c.unshift(n[0].slice(03));
    }
    
n[0] = c.join(langConfig.numDel);
    
count n.join(langConfig.numDec);
  }
  
res = (res || '%s').replace('%s'count);
  return 
res;
}

//...
function langSex(sexvars) {
  if (!
isArray(vars)) return vars;
  var 
res vars[1];
  if (!
window.langConfig) return res;
  
each(langConfig.sexRules, function(i,v){
    if (
v[0] == '*') { res vars[v[1]]; return false; }
    if (
sex == v[0] && vars[v[1]]) { res vars[v[1]]; return false; }
  });
  return 
res;
}

// Ищет значение в cur.lang, lang, langpack и возвращает найденое значение. 
function getLang() {
  try {
    var 
args = Array.prototype.slice.call(arguments);
    var 
key args.shift();
    if (!
key) return '...';
    var 
val = (window.cur.lang && window.cur.lang[key]) || (window.lang && window.lang[key]) || (window.langpack && window.langpack[key]) || window[key];
    if (!
val) {
      var 
res key.split('_');
      
res.shift();
      return 
res.join(' ');
    }
    if (
isFunction(val)) {
      return 
val.apply(nullargs);
    } else if (
args[0] !== undefined || isArray(val)) {
      return 
langNumeric(args[0], valargs[1]);
    } else {
      return 
val;
    }
  } catch(
e) {
    
debugLog('lang error:' e.message '(' + Array.prototype.slice.call(arguments).join(', ') + ')');
  }
}

// Debug Log

var _logTimer = (new Date()).getTime();
var 
_debugLogHist = [];
var 
_debugLogHistOffset 0;
var 
_debugLogHistShow false;

// Выводит лог в див и консоль.
function debugLog(msg){
  try {
    var 
'[' + (((new Date()).getTime() - _logTimer) / 1000) + '] ';
    if (
ge('debuglog')) {
      if (
msg === null) {
        
msg '[NULL]';
      } else if (
msg === undefined) {
        
msg '[UNDEFINED]';
      }
      
ge('debuglog').appendChild(ce('div', {innerHTMLmsg.toString().replace('<''&lt;').replace('>''&gt;')}));
    }
    if (
window.console && console.log) {
      var 
args = Array.prototype.slice.call(arguments);
      
args.unshift(t);
      if (
browser.msie || browser.mobile) {
        
console.log(args.join(' '));
      } else {
        
console.log.apply(consoleargs);
      }
    }
  } catch (
e) {
  }
}

// Пишит историю логов в массив
function debugLogHist(msg){
  try {
    var 
'[' + (((new Date()).getTime() - _logTimer) / 1000) + '] 'line = new Array(57).join('=');
    if (
ge('debuglog')) {
      
msg msg.toString().replace('<''&lt;').replace('>''&gt;')+'<br/>';
      
msg line '<br/>' msg line '<br/>';
      
_debugLogHistOffset++;
      if (
_debugLogHistOffset >= 20) { // Удаления лога с начала массива. (От переполнения массива) 
        
_debugLogHist.splice(0_debugLogHistOffset 19);
        
_debugLogHistOffset 19;
      }
      if (!
_debugLogHist[_debugLogHistOffset]) { _debugLogHist[_debugLogHistOffset] = ''; }
      
_debugLogHist[_debugLogHistOffset] += msg;
    }
  } catch (
e) {
  }
}
// Инфа о DOM элементе.
function debugEl(el) {
  return 
el && (((el.tagName || '').toLowerCase() + (el.className '.' el.className.replace(/s+/g'.') : '') + (el.id && !/^__vk/.test(el.id) ? '#' el.id '')) || el.toString()) || '[NULL]';
}
function 
__bf() {}

//
// DOM
//

// getElementById
function ge(el) {
  return (
typeof el == 'string' || typeof el == 'number') ? document.getElementById(el) : el;
}

// getElementsByTagName 
function geByTag(searchTagnode) {
  return (
node || document).getElementsByTagName(searchTag);
}

// Подобие $() jquery, возвращает первый элемент
// В опере работает быстрее чем getElementsByTagName 
function geByTag1(searchTagnode) {
  
node node || document;
  return 
node.querySelector && node.querySelector(searchTag) || geByTag(searchTagnode)[0];
}

// Возвращает один или несколько элементов по searchClass, 
// Если передан tag то вернет один элемент по tag.
function geByClass(searchClassnodetag) {
  
node node || document;
  
tag tag || '*';
  var 
classElements = [];

  if (!
browser.msie8 && node.querySelectorAll && tag != '*') {
    return 
node.querySelectorAll(tag '.' searchClass);
  }
  if (
node.getElementsByClassName) {
    var 
nodes node.getElementsByClassName(searchClass);
    if (
tag != '*') {
      
tag tag.toUpperCase();
      for (var 
0nodes.lengthl; ++i) {
        if (
nodes[i].tagName.toUpperCase() == tag) {
          
classElements.push(nodes[i]);
        }
      }
    } else {
      
classElements = Array.prototype.slice.call(nodes);
    }
    return 
classElements;
  }

  var 
els geByTag(tagnode);
  var 
pattern = new RegExp('(^|\s)' searchClass '(\s|$)');
  for (var 
0els.lengthl; ++i) {
    if (
pattern.test(els[i].className)) {
      
classElements.push(els[i]);
    }
  }
  return 
classElements;
}

// Вернет только один элемент по searchClass и tag.
function geByClass1(searchClassnodetag) {
  
node node || document;
  
tag tag || '*';
  return !
browser.msie8 && node.querySelector && node.querySelector(tag '.' searchClass) || geByClass(searchClassnodetag)[0];
}

// Создает и возвращает эелемент.
function ce(tagNameattrstyle) {
  var 
el document.createElement(tagName);
  if (
attrextend(elattr);
  if (
stylesetStyle(elstyle);
  return 
el;
}

// Создает и добавляет в DocumentFragment проанализированный html.
// Работает быстрее чем (цикличная) обычная вставка данных в ноду.
window.cf = (function(doc) {
  var 
frag doc.createDocumentFragment(),
      
elem doc.createElement('div'),
      
range doc.createRange && doc.createRange();
  
frag.appendChild(elem);
  
range && range.selectNodeContents(elem);

  return 
range && range.createContextualFragment ?
    function (
html) {
      if (!
html) return doc.createDocumentFragment();
      return 
range.createContextualFragment(html);
    } :
    function (
html) {
      if (!
html) return doc.createDocumentFragment();
      
elem.innerHTML html;
      var 
frag doc.createDocumentFragment();
      while (
elem.firstChild) {
        
frag.appendChild(elem.firstChild);
      }
      return 
frag;
    };
})(
document);

// Удаляет элемент.
function re(el) {
  
el ge(el);
  if (
el && el.parentNodeel.parentNode.removeChild(el);
  return 
el;
}

// Возвращает первый элемент передановго html
function se(html) {return ce('div', {innerHTMLhtml}).firstChild;}

// Ищет и заменяет ключи передановго масива на значеня в строке.  
function rs(htmlrepl) {
  
each (repl, function(kv) {
    
html html.replace(new RegExp('%' '%''g'), v);
  });
  return 
html;
}

// Заменяет старые ссылки хранилище данных на новые
function psr(html) {
  if (
locProtocol != 'https:') return html;
  
html html.replace(/http://cs(d+).(userapi.com|vk.com)/([a-z0-9/_:]+.jpg)/gi, 'https://pp.userapi.com/c$1/$3');
  
html html.replace(/http://cs(d+).(userapi.com|vk.com)//gi, 'https://ps.userapi.com/c$1/');
  
html html.replace(/http://video(d+).vkadre.ru//gi, 'https://ps.userapi.com/v$1/');
  
return html;
}

// Премещения по DOM.
function domEL(elp) {
  
'previousSibling' 'nextSibling';
  while (
el && !el.tagNameel el[p];
  return 
el;
}
function 
domNS(el) {
  return 
domEL((el || {}).nextSibling);
}
function 
domPS(el) {
  return 
domEL((el || {}).previousSibling1);
}
function 
domFC(el) {
  return 
domEL((el || {}).firstChild);
}
function 
domLC(el) {
  return 
domEL((el || {}).lastChild1);
}
function 
domPN(el) {
  return (
el || {}).parentNode;
}

// Показывает элемент
function show(elem) {
  if (
arguments.length 1) {
    for (var 
0arguments.lengthl; ++i) {
      
show(arguments[i]);
    }
    return;
  }
  
elem ge(elem);
  if (!
elem || !elem.style) return;
  var 
old elem.olddisplaynewStyle 'block'tag elem.tagName.toLowerCase();
  
elem.style.display old || '';


  if (
getStyle(elem'display') == 'none') {
    if (
hasClass(elem'inline')) {
      
newStyle 'inline';
    } else if (
tag == 'tr' && !browser.msie) {
      
newStyle 'table-row';
    } else if (
tag == 'table' && !browser.msie) {
      
newStyle 'table';
    } else {
      
newStyle 'block';
    }
    
elem.style.display elem.olddisplay newStyle;
  }
}

// Скрывает элемент.
function hide(elem) {
  var 
arguments.length;
  if (
1) {
    for (var 
0li++) {
      
hide(arguments[i]);
    }
    return;
  }
  
elem ge(elem);
  if (!
elem || !elem.style) return;
  var 
getStyle(elem'display');
  
elem.olddisplay = (!= 'none') ? '';
  
elem.style.display 'none';
}

// Проверяет не скрыт ли елемент, вернет bool.
function isVisible(elem) {
  
elem ge(elem);
  if (!
elem || !elem.style) return false;
  return 
getStyle(elem'display') != 'none';
}

// Переключатель.
// Если элемент скрыт - показывает, если открыт - скрывает.
function toggle(elemv) {
  if (
=== undefined) {
    
= !isVisible(elem);
  }
  if (
v) {
    
show(elem);
  } else {
    
hide(elem);
  }
}

// Скрывает или показывает все флеш контейнера. 
var hfTimeout 0;
function 
toggleFlash(showtimeout) {
  
//if (/mac/i.test(navigator.userAgent)) return;
  
clearTimeout(hfTimeout);
  if (
timeout 0) {
    
hfTimeout setTimeout(function() {toggleFlash(show0)}, timeout);
    return;
  }

  var 
vis show 'visible' 'hidden';

  
triggerEvent(documentshow 'unblock' 'block');

  var 
= function() {
    if (
this.getAttribute('preventhide')) {
      return;
    } else if (
this.id == 'flash_app' && browser.msie) {

      
show setStyle(this, {position'static'top0}) : setStyle(this, {position'absolute'top'-5000px'});
    } else {
      
this.style.visibility vis;
    }
  };
  
each(geByTag('embed'), f);
  
each(geByTag('object'), f);
}

// Возвращает положения элемента по XY.
function getXY(objforFixed) {
  
obj ge(obj);
  if (!
obj) return [0,0];

  var 
left 0top 0poslastLeft;
  if (
obj.offsetParent) {
    do {
      
left += (lastLeft obj.offsetLeft);
      
top += obj.offsetTop;
      
pos getStyle(obj'position');
      if (
pos == 'fixed' || pos == 'absolute' || (pos == 'relative')) {
        
left -= obj.scrollLeft;
        
top -= obj.scrollTop;
        if (
pos == 'fixed' && !forFixed) {
          
left += ((obj.offsetParent || {}).scrollLeft || bodyNode.scrollLeft || htmlNode.scrollLeft);
          
top += ((obj.offsetParent || {}).scrollTop || bodyNode.scrollTop || htmlNode.scrollTop);
        }
      }
    } while (
obj obj.offsetParent);
  }
  if (
forFixed && browser.msie && intval(browser.version) < 9) {
    if (
lastLeft) {
      
left += ge('page_layout').offsetLeft;
    }
  }
  return [
left,top];
}

// Возвращает размер елемента, если передан 2 пар. отнимает границы(padding,border).
function getSize(elemwithoutBounds) {
  
elem ge(elem);
  var 
= [00], de document.documentElement;
  if (
elem == document) {
    
=  [Math.max(
        
de.clientWidth,
        
bodyNode.scrollWidthde.scrollWidth,
        
bodyNode.offsetWidthde.offsetWidth
      
), Math.max(
        
de.clientHeight,
        
bodyNode.scrollHeightde.scrollHeight,
        
bodyNode.offsetHeightde.offsetHeight
      
)];
  } else if (
elem){
    function 
getWH() {
      
= [elem.offsetWidthelem.offsetHeight];
      if (!
withoutBounds) return;
      var 
padding 0border 0;
      
each(s, function(iv) {
        var 
which ? ['Top''Bottom'] : ['Left''Right'];
        
each(which, function(){
          
s[i] -= parseFloat(getStyle(elem'padding' this)) || 0;
          
s[i] -= parseFloat(getStyle(elem'border' this 'Width')) || 0;
        });
      });
      
= [Math.round(s[0]), Math.round(s[1])];
    }
    if (!
isVisible(elem)) {
      var 
props = {position'absolute'visibility'hidden'display'block'};
      var 
old = {};
      
each(props, function(iv) {
        
old[i] = elem.style[i];
        
elem.style[i] = v;
      });
      
getWH();
      
each(props, function(iv) {
        
elem.style[i] = old[i];
      });
    } else 
getWH();

  }
  return 
s;
}

// Возвращает маштаб. На сколько увеличина страница. 1 = 100%.
function getZoom() {
  var 
r1 ge('zoom_test_1') || document.body.appendChild(ce('div', {id'zoom_test_1'}, {left'10%'position'absolute'visibility'hidden'})),
      
r2 ge('zoom_test_2') || document.body.appendChild(ce('div', {id'zoom_test_2'}, {leftr1.offsetLeft 'px'position'absolute'visibility'hidden'}));
  return 
r2.offsetLeft r1.offsetLeft;
}

//
//  Useful utils
//

// Вызывает текущую функцию с контекстом window и передают текущие параметры.
Function.prototype.pbind = function() {
  var 
args = Array.prototype.slice.call(arguments);
  
args.unshift(window);
  return 
this.bind.apply(thisargs);
};
Function.
prototype.bind = function() {
  var 
func thisargs = Array.prototype.slice.call(arguments);
  var 
obj args.shift();
  return function() {
    var 
curArgs = Array.prototype.slice.call(arguments);
    return 
func.apply(objargs.concat(curArgs));
  }
}

function 
rand(mima) { return Math.random() * (ma mi 1) + mi; } 
function 
irand(mima) { return Math.floor(rand(mima)); }
// Проверка на соответствие типов
function isFunction(obj) {return Object.prototype.toString.call(obj) === '[object Function]'; }
function 
isArray(obj) { return Object.prototype.toString.call(obj) === '[object Array]'; }
function 
isObject(obj) { return Object.prototype.toString.call(obj) === '[object Object]'; }
function 
isEmpty(o) { if(Object.prototype.toString.call(o) !== '[object Object]') {return false;} for(var i in o){ if(o.hasOwnProperty(i)){return false;} } return true; }

function 
vkNow() { return +new Date; } // Unix Time.
function vkImage() { return window.Image ? (new Image()) : ce('img'); } // IE8 workaround.
function trim(text) { return (text || '').replace(/^s+|s+$/g''); } // Вырезает пробелы
function stripHTML(text) { return text text.replace(/<(?:.|s)*?>/g, '') : ''; } // Вырезает все html теги
function escapeRE(s) { return s ? s.replace(/([.*+?^${}()|[]/\])/g, '\$1') : ''; } // Экранирует []/(){}.+?|*

function intval(value) {
  if (value === true) return 1;
  return parseInt(value) || 0;
}
function floatval(value) {
  if (value === true) return 1;
  return parseFloat(value) || 0;
}

// Вернет только положительные числа. 
function positive(value) {
  value = intval(value);
  return value < 0 ? 0 : value;
}

// Заменяет код html на обычные скобки и тд.
function winToUtf(text) {
  var m, i, j, code;
  m = text.match(/&#[0-9]{2}[0-9]*;/gi);
  for (j in m) {
    var v = '' + m[j]; // buggy IE6
    code = intval(v.substr(2, v.length - 3));
    if (code >= 32 && ('&#' + code + ';' == v)) { // buggy IE6
      text = text.replace(v, String.fromCharCode(code));
    }
  }
  return text.replace(/&quot;/gi, '"').replace(/&lt;/gi, '<').replace(/&gt;/gi, '>').replace(/&amp;/gi, '&');
}

// Возвращает текст заменяя html разметку.
function replaceEntities(str) {
  return se('<textarea>' + (str.replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;')) + '</textarea>').value;
}

// Замена html разметки
function clean(str) {
  return str ? str.replace(/&/g, '&amp;').replace(/</g, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;').replace(/'/g, '&#039;') : '';
}

//
//  Arrays, objects
//

// Аналог foreach.
function each(object, callback) {
  var name, i = 0, length = object.length;

  if (length === undefined) {
    for (name in object)
      if (callback.call(object[name], name, object[name]) === false)
        break;
  } else {
    for (var value = object[0];
      i < length && callback.call(value, i, value) !== false;
        value = object[++i]) {}
  }

  return object;
}

// Возвращает позицию значеня в массиве.
function indexOf(arr, value, from) {
  for (var i = from || 0, l = (arr || []).length; i < l; i++) {
    if (arr[i] == value) return i;
  }
  return -1;
}

// Возвращает bool тип, проверка на наличия значеня в массиве.
function inArray(value, arr) {
  return indexOf(arr, value) != -1;
}

// Клонирует массив, объект.
function clone(obj, req) {
  var newObj = isArray(obj) ? [] : {};
  for (var i in obj) {
    if (/webkit/i.test(_ua) && (i == 'layerX' || i == 'layerY')) continue;
    if (req && typeof(obj[i]) === 'object' && i !== 'prototype') {
      newObj[i] = clone(obj[i]);
    } else {
      newObj[i] = obj[i];
    }

  }
  return newObj;
}

// Расхождения массивов
function arrayKeyDiff(a) {   // Computes the difference of arrays by keys and values
  var arr_dif = {}, i = 1, argc = arguments.length, argv = arguments, key, found;
  for (key in a){
    found = false;
    for (i = 1; i < argc; i++){
      if (argv[i][key] && (argv[i][key] == a[key])){
        found = true;
      }
    }
    if (!found) {
      arr_dif[key] = a[key];
    }
  }
  return arr_dif;
}

// Extending object by another
// Слив объектов в один (переданный первичным) 
// Если первый пар. true то переберает рекурсией вложенность
function extend() {
  var a = arguments, target = a[0] || {}, i = 1, l = a.length, deep = false, options;

  if (typeof target === 'boolean') {
    deep = target;
    target = a[1] || {};
    i = 2;
  }

  if (typeof target !== 'object' && !isFunction(target)) target = {};

  for (; i < l; ++i) {
    if ((options = a[i]) != null) {
      for (var name in options) {
        var src = target[name], copy = options[name];

        if (target === copy) continue;

        if (deep && copy && typeof copy === 'object' && !copy.nodeType) {
          target[name] = extend(deep, src || (copy.length != null ? [] : {}), copy);
        } else if (copy !== undefined) {
          target[name] = copy;
        }
      }
    }
  }

  return target;
}


//
// CSS classes
//

// Наличие атрибуда Class.
function hasClass(obj, name) {
  obj = ge(obj);
  return obj && (new RegExp('(\s|^)' + name + '(\s|$)')).test(obj.className);
}

// Добавляет атрибут class или добовляет к class значение
function addClass(obj, name) {
  if ((obj = ge(obj)) && !hasClass(obj, name)) {
    obj.className = (obj.className ? obj.className + ' ' : '') + name;
  }
}

// Удаляет значение из class.
function removeClass(obj, name) {
  if (obj = ge(obj)) {
    obj.className = trim((obj.className || '').replace((new RegExp('(\s|^)' + name + '(\s|$)')), ' '));
  }
}

// Если class есть - удаляет, нету - создает
function toggleClass(obj, name, v) {
  if (v === undefined) {
    v = !hasClass(obj, name);
  }
  (v ? addClass : removeClass)(obj, name);
}

// Заменяет class.
function replaceClass(obj, oldName, newName) {
  removeClass(obj, oldName);
  addClass(obj, newName);
}

// Get computed style.
function getStyle(elem, name, force) {
  elem = ge(elem);
  if (isArray(name)) { var res = {}; each(name, function(i,v){res[v] = getStyle(elem, v);}); return res; }
  if (force === undefined) {
    force = true;
  }
  if (!force && name == 'opacity' && browser.msie) {
    var filter = elem.style['filter'];
    return filter ? (filter.indexOf('opacity=') >= 0 ?
      (parseFloat(filter.match(/opacity=([^)]*)/)[1] ) / 100) + '' : '1') : '';
  }
  if (!force && elem.style && (elem.style[name] || name == 'height')) {
    return elem.style[name];
  }

  var ret, defaultView = document.defaultView || window;
  if (defaultView.getComputedStyle) {
    name = name.replace(/([A-Z])/g, '-$1').toLowerCase();
    var computedStyle = defaultView.getComputedStyle(elem, null);
    if (computedStyle) {
      ret = computedStyle.getPropertyValue(name);
    }
  } else if (elem.currentStyle) {
    if (name == 'opacity' && browser.msie) {
      var filter = elem.currentStyle['filter'];
      return filter && filter.indexOf('opacity=') >= 0 ?
        (parseFloat(filter.match(/opacity=([^)]*)/)[1]) / 100) + '' : '1';
    }
    var camelCase = name.replace(/-(w)/g, function(all, letter){
      return letter.toUpperCase();
    });
    ret = elem.currentStyle[name] || elem.currentStyle[camelCase];
    //dummy fix for ie
    if (ret == 'auto') {
      ret = 0;
    }

    if (!/^d+(px)?$/i.test(ret) && /^d/.test(ret)) {
      var style = elem.style, left = style.left, rsLeft = elem.runtimeStyle.left;

      elem.runtimeStyle.left = elem.currentStyle.left;
      style.left = ret || 0;
      ret = style.pixelLeft + 'px';

      style.left = left;
      elem.runtimeStyle.left = rsLeft;
    }
  }

  if (force && (name == 'width' || name == 'height')) {
    var ret2 = getSize(elem, true)[({'width': 0, 'height': 1})[name]];
    ret = (intval(ret) ? Math.max(floatval(ret), ret2) : ret2) + 'px';
  }

  return ret;
}

// Установка стиля элементу.
function setStyle(elem, name, value){
  elem = ge(elem);
  if (!elem) return;
  if (typeof name == 'object') return each(name, function(k, v) { setStyle(elem,k,v); });
  if (name == 'opacity') {
    if (browser.msie) {
      if ((value + '').length) {
        if (value !== 1) {
          elem.style.filter = 'alpha(opacity=' + value * 100 + ')';
        } else {
          elem.style.filter = '';
        }
      } else {
        elem.style.cssText = elem.style.cssText.replace(/filters*:[^;]*/gi, '');
      }
      elem.style.zoom = 1;
    };
    elem.style.opacity = value;
  } else {
    try{
    var isN = typeof(value) == 'number';
    if (isN && (/height|width/i).test(name)) value = Math.abs(value);
    elem.style[name] = isN && !(/z-?index|font-?weight|opacity|zoom|line-?height/i).test(name) ? value + 'px' : value;
    } catch(e){debugLog([name, value]);}
  }
}

//
// Store data connected to element
//

var vkExpand = 'VK' + vkNow(), vkUUID = 0, vkCache = {};

// Для совместимости событий
// Запись и чтение vkCache.
function data(elem, name, data) {
  var id = elem[vkExpand], undefined;
  if (!id) {
    id = elem[vkExpand] = ++vkUUID;
  }

  if (data !== undefined) {
    if (!vkCache[id]) {
      vkCache[id] = {};
      if (__debugMode) vkCache[id].__elem = elem;
    }
    vkCache[id][name] = data;
  }

  return name ? vkCache[id] && vkCache[id][name] : id;
}

// Удаляет элементы массива, el - массив, остальные аргументы - элементы которые надо удалить 
function removeAttr(el) {
  for (var i = 0, l = arguments.length; i < l; ++i) {
    var n = arguments[i];
    if (el[n] === undefined) continue;
    try {
      delete el[n];
    } catch(e) {
      try {
        el.removeAttribute(n);
      } catch(e) {}
    }
  }
}

//  Удаляет ячейки vkCache
function removeData(elem, name) {
  var id = elem ? elem[vkExpand] : false;
  if (!id) return;

  if (name) {
    if (vkCache[id]) {
      delete vkCache[id][name];
      name = '';

      var count = 0;
      for (name in vkCache[id]) {
        if (name !== '__elem') {
          count++;
          break;
        }
      }

      if (!count) {
        removeData(elem);
      }
    }
  } else {
    removeEvent(elem);
    removeAttr(elem, vkExpand);
    delete vkCache[id];
  }
}

// Отчищает ячейку vkCache, и удаляет событие btnevents с элемента
function cleanElems() {
  var a = arguments;
  for (var i = 0; i < a.length; ++i) {
    var el = ge(a[i]);
    if (el) {
      removeData(el);
      removeAttr(el, 'btnevents');
    }
  }
}

// Simple FX Анимация
function animate(el, params, speed, callback) {
  el = ge(el);
  if (!el) return;
  var _cb = isFunction(callback) ? callback : function() {};
  var options = extend({}, typeof speed == 'object' ? speed : {duration: speed, onComplete: _cb});
  var fromArr = {}, toArr = {}, visible = isVisible(el), self = this, p;
  options.orig = {};
  params = clone(params);
  if (params.discrete) {
    options.discrete = 1;
    delete(params.discrete);
  }
  if (browser.iphone)
    options.duration = 0;
  var tween = data(el, 'tween'), i, name, toggleAct = visible ? 'hide' : 'show';
  if (tween && tween.isTweening) {
    options.orig = extend(options.orig, tween.options.orig);
    tween.stop(false);
    if (tween.options.show) toggleAct = 'hide';
    else if (tween.options.hide) toggleAct = 'show';
  }
  for (p in params)  {
    if (!tween && (params[p] == 'show' && visible || params[p] == 'hide' && !visible)) {
      return options.onComplete.call(this, el);
    }
    if ((p == 'height' || p == 'width') && el.style) {
      if (!params.overflow) {
        if (options.orig.overflow == undefined) {
          options.orig.overflow = getStyle(el, 'overflow');
        }
        el.style.overflow = 'hidden';
      }
      if (!hasClass(el, 'inl_bl') && el.tagName != 'TD') {
        el.style.display = 'block';
      }
    }
    if (/show|hide|toggle/.test(params[p])) {
      if (params[p] == 'toggle') {
        params[p] = toggleAct;
      }
      if (params[p] == 'show') {
        var from = 0;
        options.show = true;
        if (options.orig[p] == undefined) {
          options.orig[p] = getStyle(el, p, false) || '';
          setStyle(el, p, 0);
        }

        var o;
        if (p == 'height' && browser.msie6) {
          o = '0px';
          el.style.overflow = '';
        } else {
          o = options.orig[p];
        }

        var old = el.style[p];
        el.style[p] = o;
        params[p] = parseFloat(getStyle(el, p, true));
        el.style[p] = old;

        if (p == 'height' && browser.msie && !params.overflow) {
          el.style.overflow = 'hidden';
        }
      } else {
        if (options.orig[p] == undefined) {
          options.orig[p] = getStyle(el, p, false) || '';
        }
        options.hide = true;
        params[p] = 0;
      }
    }
  }
  if (options.show && !visible) {
    show(el);
  }
  tween = new Fx.Base(el, options);
  each(params, function(name, to) {
    if (/backgroundColor|borderBottomColor|borderLeftColor|borderRightColor|borderTopColor|color|borderColor|outlineColor/.test(name)) {
      var p = (name == 'borderColor') ? 'borderTopColor' : name;
      from = getColor(el, p);
      to = getRGB(to);
    } else {
      var parts = to.toString().match(/^([+-]=)?([d+-.]+)(.*)$/),
        start = tween.cur(name, true) || 0;
      if (parts) {
        to = parseFloat(parts[2]);
        if (parts[1]) {
          to = ((parts[1] == '-=' ? -1 : 1) * to) + to;
        }
      }

      if (options.hide && name == 'height' && browser.msie6) {
        el.style.height = '0px';
        el.style.overflow = '';
      }
      from = tween.cur(name, true);
      if (options.hide && name == 'height' && browser.msie6) {
        el.style.height = '';
        el.style.overflow = 'hidden';
      }
      if (from == 0 && (name == 'width' || name == 'height'))
        from = 1;

      if (name == 'opacity' && to > 0 && !visible) {
        setStyle(el, 'opacity', 0);
        from = 0;
        show(el);
      }
    }
    if (from != to || (isArray(from) && from.join(',') == to.join(','))) {
      fromArr[name] = from;
      toArr[name] = to;
    }
  });
  tween.start(fromArr, toArr);
  data(el, 'tween', tween);

  return tween;
}

// Показвает элемент с opacity.
function fadeTo(el, speed, to, callback) {
  return animate(el, {opacity: to}, speed, callback);
}

// Вспомогательный объект для анимации.
var Fx = {
  Transitions: {
    linear: function(t, b, c, d) { return c*t/d + b; },
    sineInOut: function(t, b, c, d) { return -c/2 * (Math.cos(Math.PI*t/d) - 1) + b; },
    halfSine: function(t, b, c, d) { return c * (Math.sin(Math.PI * (t/d) / 2)) + b; },
    easeOutBack: function(t, b, c, d) { var s = 1.70158; return c*((t=t/d-1)*t*((s+1)*t + s) + 1) + b; },
    easeInCirc: function(t, b, c, d) { return -c * (Math.sqrt(1 - (t/=d)*t) - 1) + b; },
    easeOutCirc: function(t, b, c, d) { return c * Math.sqrt(1 - (t=t/d-1)*t) + b; },
    easeInQuint: function(t, b, c, d) { return c*(t/=d)*t*t*t*t + b; },
    easeOutQuint: function(t, b, c, d) { return c*((t=t/d-1)*t*t*t*t + 1) + b; },
    easeOutCubic: function(t, b, c, d) { return c*((t=t/d-1)*t*t + 1) + b;}
  },
  Attrs: [
    [ 'height', 'marginTop', 'marginBottom', 'paddingTop', 'paddingBottom' ],
    [ 'width', 'marginLeft', 'marginRight', 'paddingLeft', 'paddingRight' ],
    [ 'opacity', 'left', 'top' ]
  ],
  Timers: [],
  TimerId: null
}, fx = Fx;

// Инициализация для анимации.
Fx.Base = function(el, options, name) {
  this.el = ge(el);
  this.name = name;
  this.options = extend({
    onComplete: function() {},
    transition: options.transition || Fx.Transitions.sineInOut,
    duration: 500
  }, options || {});
}

// Возвращает объект с нужным Fx.Attrs элементом. 
function genFx(type, num) {
  var obj = {};
  each(Fx.Attrs.concat.apply([], Fx.Attrs.slice(0, num)), function() {
    obj[this] = type;
  });
  return obj;
};

// Сливает в window стандартные функции для анимации.
each({slideDown: genFx('show', 1),
  slideUp: genFx('hide', 1),
  slideToggle: genFx('toggle', 1),
  fadeIn: {opacity: 'show'},
  fadeOut: {opacity: 'hide'},
  fadeToggle: {opacity: 'toggle'}}, function(f, v) {
  window[f] = function(el, speed, callback) { return animate(el, v, speed, callback); }
});

// Основной механизм анимации. Счет времени, промежутков и т.д
Fx.Base.prototype = {
  start: function(from, to){
    this.from = from;
    this.to = to;
    this.time = vkNow();
    this.isTweening = true;

    var self = this;
    function t(gotoEnd) {
      return self.step(gotoEnd);
    }
    t.el = this.el;
    if (t() && Fx.Timers.push(t) && !Fx.TimerId) {
      Fx.TimerId = setInterval(function() {
        var timers = Fx.Timers, l = timers.length;
        for (var i = 0; i < l; i++) {
          if (!timers[i]()) {
            timers.splice(i--, 1);
            l--;
          }
        }
        if (!l) {
          clearInterval(Fx.TimerId);
          Fx.TimerId = null;
        }
      }, 13);
    }
    return this;
  },

  stop: function(gotoEnd) {
    var timers = Fx.Timers;

    for (var i = timers.length - 1; i >= 0; i--) {
      if (timers[i].el == this.el ) {
        if (gotoEnd) {
          timers[i](true);
        }
        timers.splice(i, 1);
      }
    }
    this.isTweening = false;
  },

  step: function(gotoEnd) {
    var time = vkNow();
    if (!gotoEnd && time < this.time + this.options.duration) {
      this.cTime = time - this.time;
      this.now = {};
      for (p in this.to) {
        // color fx
        if (isArray(this.to[p])) {
          var color = [], j;
          for (j = 0; j < 3; j++) {
            if (this.from[p] === undefined || this.to[p] === undefined) {
              return false;
            }
            color.push(Math.min(parseInt(this.compute(this.from[p][j], this.to[p][j])), 255));
          }
          this.now[p] = color;
        } else {
          this.now[p] = this.compute(this.from[p], this.to[p]);
          if (this.options.discrete) this.now[p] = intval(this.now[p]);
        }
      }
      this.update();
      return true;
    } else {
      setTimeout(this.options.onComplete.bind(this, this.el), 10);
      this.now = extend(this.to, this.options.orig);
      this.update();
      if (this.options.hide) hide(this.el);
      this.isTweening = false;
      return false;
    }
  },

  compute: function(from, to){
    var change = to - from;
    return this.options.transition(this.cTime, from, change, this.options.duration);
  },

  update: function(){
    for (var p in this.now) {
      if (isArray(this.now[p])) setStyle(this.el, p, 'rgb(' + this.now[p].join(',') + ')');
      else this.el[p] != undefined ? (this.el[p] = this.now[p]) : setStyle(this.el, p, this.now[p]);
    }
  },

  cur: function(name, force){
    if (this.el[name] != null && (!this.el.style || this.el.style[name] == null))
      return this.el[name];
    return parseFloat(getStyle(this.el, name, force)) || 0;
  }
};

// Возвращает цвет переведеный в RGB [255,255,255]
function getRGB(color) {
  var result;
  if (color && isArray(color) && color.length == 3)
    return color;
  if (result = /rgb(s*([0-9]{1,3})s*,s*([0-9]{1,3})s*,s*([0-9]{1,3})s*)/.exec(color))
    return [parseInt(result[1]), parseInt(result[2]), parseInt(result[3])];
  if (result = /rgb(s*([0-9]+(?:.[0-9]+)?)%s*,s*([0-9]+(?:.[0-9]+)?)%s*,s*([0-9]+(?:.[0-9]+)?)%s*)/.exec(color))
    return [parseFloat(result[1])*2.55, parseFloat(result[2])*2.55, parseFloat(result[3])*2.55];
  if (result = /#([a-fA-F0-9]{2})([a-fA-F0-9]{2})([a-fA-F0-9]{2})/.exec(color))
    return [parseInt(result[1],16), parseInt(result[2],16), parseInt(result[3],16)];
  if (result = /#([a-fA-F0-9])([a-fA-F0-9])([a-fA-F0-9])/.exec(color))
    return [parseInt(result[1]+result[1],16), parseInt(result[2]+result[2],16), parseInt(result[3]+result[3],16)];
}

// Возвращает цвет элемента в формате RGB[255,255,255]
function getColor(elem, attr) {
  var color;
  do {
    color = getStyle(elem, attr);
    if (!color.indexOf('rgba')) color = '';
    if (color != '' && color != 'transparent' || elem.nodeName.toLowerCase() == 'body') {
      break;
    }
    attr = 'backgroundColor';
  } while (elem = elem.parentNode);
  return getRGB(color);
}

// Скролинг по оси Y.
function scrollToY(y, speed, anim, topLink) {
  if (speed == undefined) speed = 400;
  if (speed) {
    var updT = function() {
      if (anim === 2) {
        updSideTopLink();
      }
    }
    if (browser.msie6) {
      animate(pageNode, {scrollTop: y}, speed, updT);
    } else {
      animate(htmlNode, {scrollTop: y, transition: Fx.Transitions.easeInCirc}, speed, updT);
      animate(bodyNode, {scrollTop: y, transition: Fx.Transitions.easeInCirc}, speed, updT);
    }
  } else {
    if (anim && anim !== 2) {
      var diff = scrollGetY() - y;
      if (Math.abs(diff) > 6) {
        scrollToY(y+(diff > 0 ? 6 : -6), 0, 2);
      }
      setTimeout(scrollToY.pbind(y, 100, 2), 0);
      return;
    }
    window.scroll(0, y);
    if (browser.msie6) {
      pageNode.scrollTop = y;
    }
    if (!anim) {
      updSideTopLink();
    }
  }
}

// Скролинг к началу страницы.
function scrollToTop(speed) {
  return scrollToY(0, speed);
}

// Возвращает место положения скролла по Y. 
function scrollGetY() {
  return window.pageYOffset || scrollNode.scrollTop || document.documentElement.scrollTop;
}

// Ставит цвет у элемента на 4 секунды запоминая предыдущий цвет в кеше.
function notaBene(el, color, nofocus) {
  el = ge(el);
  if (!el) return;

  if (!nofocus) elfocus(el);
  if (data(el, 'backstyle') === undefined) data(el, 'backstyle', el.style.backgroundColor || '');
  var oldBack = data(el, 'back') || data(el, 'back', getStyle(el, 'backgroundColor'));
  var colors = {notice: '#FFFFE0', warning: '#FAEAEA'};
  setStyle(el, 'backgroundColor', colors[color] || color || colors.warning);
  setTimeout(animate.pbind(el, {backgroundColor: oldBack}, 300, function() {
    el.style.backgroundColor = data(el, 'backstyle');
  }), 400);
}

// Установка атрибута title элементу.
function setTitle(el) {
  el = ge(el);
  if (!el || el.titleSet) return;
  if (el.scrollWidth > el.clientWidth) {
    el.setAttribute('title', el.innerText || el.textContent);
  } else {
    var b = geByTag1('b', el);
    if (b && b.scrollWidth > b.clientWidth) {
      el.setAttribute('title', el.innerText || el.textContent);
    } else {
      el.removeAttribute('title');
    }
  }
  el.titleSet = 1;
}

//
// Events
//

var KEY = window.KEY = {
  LEFT: 37,
  UP: 38,
  RIGHT: 39,
  DOWN: 40,
  DEL: 8,
  TAB: 9,
  RETURN: 13,
  ENTER: 13,
  ESC: 27,
  PAGEUP: 33,
  PAGEDOWN: 34,
  SPACE: 32
};

// Добавления событие(я).
function addEvent(elem, types, handler, custom, context) {
  elem = ge(elem);
  if (!elem || elem.nodeType == 3 || elem.nodeType == 8) { // 3 - Node.TEXT_NODE, 8 - Node.COMMENT_NODE
    return;
  }

  // Контекст выполнения обработчика.
  var realHandler = context ? function() {
    var newHandler = function(e) {
      var prevData = e.data;
      e.data = context;
      var ret = handler.apply(this, [e]);
      e.data = prevData;
      return ret;
    }
    newHandler.handler = handler;
    return newHandler;
  }() : handler;

  // For IE
  if (elem.setInterval && elem != window) elem = window;

  var events = data(elem, 'events') || data(elem, 'events', {}),
      handle = data(elem, 'handle') || data(elem, 'handle', function() {
        _eventHandle.apply(arguments.callee.elem, arguments);
      });
  // to prevent a memory leak
  handle.elem = elem;

  each(types.split(/s+/), function(index, type) {
    if (!events[type]) {
      events[type] = [];
      if (!custom && elem.addEventListener) {
        elem.addEventListener(type, handle, false);
      } else if (!custom && elem.attachEvent) {
        elem.attachEvent('on' + type, handle);
      }
    }
    events[type].push(realHandler);
  });

  elem = null;
}

// Удаляет событие(я).
function removeEvent(elem, types, handler) {
  elem = ge(elem);
  if (!elem) return;
  var events = data(elem, 'events');
  if (!events) return;
  if (typeof(types) != 'string') {
    for (var i in events) {
      removeEvent(elem, i);
    }
    return;
  }

  each(types.split(/s+/), function(index, type) {
    if (!isArray(events[type])) return;
    var l = events[type].length;
    if (isFunction(handler)) {
      for (var i = l - 1; i >= 0; i--) {
        if (events[type][i] && (events[type][i] === handler || events[type][i].handler === handler)) {
          events[type].splice(i, 1);
          l--;
          break;
        }
      }
    } else {
      for (var i = 0; i < l; i++) {
        delete events[type][i];
      }
      l = 0;
    }
    if (!l) {
      if (elem.removeEventListener) {
        elem.removeEventListener(type, data(elem, 'handle'), false);
      } else if (elem.detachEvent) {
        elem.detachEvent('on' + type, data(elem, 'handle'));
      }
      delete events[type];
    }
  });
  if (isEmpty(events)) {
    removeData(elem, 'events')
    removeData(elem, 'handle')
  }
}

// Заменяет действующее событие.
// Ставит type и target у события.
// type - тип события. target - элемент у которого произошло событие.
function triggerEvent(elem, type, ev, now) {
  elem = ge(elem);
  var handle = data(elem, 'handle');
  if (handle) {
    var f = function() {
      handle.call(elem, extend((ev || {}), {type: type, target: elem}))
    };
    now ? f() : setTimeout(f, 0);
  }
}

// Перехватывает событие на всплытие и отключает реакцию браузера. 
function cancelEvent(event) {
  event = (event || window.event);
  if (!event) return false;
  event = (event.originalEvent || event);
  if (event.preventDefault) event.preventDefault();
  if (event.stopPropagation) event.stopPropagation();
  event.cancelBubble = true;
  event.returnValue = false;
  return false;
}

// Для перехвата событий на всплытие. 
function _eventHandle(event) {
  event = normEvent(event);

  var handlers = data(this, 'events');
  if (!handlers || typeof(event.type) != 'string' || !handlers[event.type] || !handlers[event.type].length) {
    return;
  }

  for (var i in (handlers[event.type] || [])) {
    if (event.type == 'mouseover' || event.type == 'mouseout') {
      var parent = event.relatedElement;
      while (parent && parent != this) {
        try { parent = parent.parentNode; }
        catch(e) { parent = this; }
      }
      if (parent == this) {
        continue
      }
    }
    var ret = handlers[event.type][i].apply(this, arguments);
    if (ret === false || ret === -1) {
      cancelEvent(event);
    }
    if (ret === -1) {
      return false;
    }
  }
}

// Делает Кросс-браузерность события
function normEvent(event) {
  event = event || window.event;

  var originalEvent = event;
  event = clone(originalEvent);
  event.originalEvent = originalEvent;

  if (!event.target) {
    event.target = event.srcElement || document;
  }

  // check if target is a textnode (safari)
  if (event.target.nodeType == 3) {
    event.target = event.target.parentNode;
  }

  if (!event.relatedTarget && event.fromElement) {
    event.relatedTarget = event.fromElement == event.target;
  }

  if (event.pageX == null && event.clientX != null) {
    var doc = document.documentElement, body = bodyNode;
    event.pageX = event.clientX + (doc && doc.scrollLeft || body && body.scrollLeft || 0) - (doc.clientLeft || 0);
    event.pageY = event.clientY + (doc && doc.scrollTop || body && body.scrollTop || 0) - (doc.clientTop || 0);
  }

  if (!event.which && ((event.charCode || event.charCode === 0) ? event.charCode : event.keyCode)) {
    event.which = event.charCode || event.keyCode;
  }

  if (!event.metaKey && event.ctrlKey) {
    event.metaKey = event.ctrlKey;
  } else if (!event.ctrlKey && event.metaKey && browser.mac) {
    event.ctrlKey = event.metaKey;
  }

  // click: 1 == left; 2 == middle; 3 == right
  if (!event.which && event.button) {
    event.which = (event.button & 1 ? 1 : ( event.button & 2 ? 3 : ( event.button & 4 ? 2 : 0 ) ));
  }

  return event;
}

// Prevent memory leaks in IE
// Отчистка от событий не связанных с window
addEvent(window, 'unload', function() {
  for (var id in vkCache) {
    if (vkCache[id].handle && vkCache[id].handle.elem != window) {
      removeEvent(vkCache[id].handle.elem);
    }
  }
});

addEvent(window, 'load', function() {
  vk.loaded = true;
  updSideTopLink();
});

// Вывод инфы о всех элементах - событий. 
if (__debugMode) {
  function __checkData() {
    var r = [];
    for (var i in vkCache) {
      var c = vkCache[i], el;
      if (!c || !(el = c.__elem)) continue;
      var id = el.id;
      if (!id) el.id = id = '__vk' + irand(1000000, 9999999);
      if (ge(id) == el) continue;
      var d = [];
      for (var j in c) {
        if (j == '__elem' || j == 'handle' && c.events) continue;
        if (j == 'events') {
          var e = [];
          for (var k in c[j]) e.push(k + '(' + c[j][k].length + ')');
          d.push('{' + e.join(', ') + '}');
        } else {
          d.push(j);
        }
      }
      r.push(debugEl(el) + ': ' + d.join(', '));
    }
    return r.join('<br>');
  }
  addEvent(document, 'keydown', function(e) {
    if (e.keyCode == 120 && !e.charCode) {
      showFastBox({title: 'Debug'}, __checkData());
    }
  });
}

var _layerAnim = false;
// Layers
var layers = {
  // Показывает слой. 
  sh: (!_layerAnim || browser.msie || browser.iphone) ? function(el, done) {
    show(el);
    if (done) done();
  } : function(el, done) {
    fadeIn(el, 200, done);
  },
  // Скрывает слой.
  hd: (!_layerAnim || browser.msie || browser.iphone) ? function(el, done) {
    hide(el);
    if (done) done();
  } : function(el, done) {
    fadeOut(el, 200, done);
  },
  visible: false,
  _show: function(el, con, opacity, color) { // Показывает слой и блок с контентом
    setStyle(el, {opacity: opacity || '', backgroundColor: color || ''});
    if (!layers.visible) {
      toggleFlash();
      if (browser.mozilla) {
        window._oldScroll = htmlNode.scrollTop;
        pageNode.style.height = (_oldScroll + lastWindowHeight) + 'px';
        pageNode.style.marginTop = -_oldScroll + 'px';
      } else if (!browser.msie6) {
        (browser.msie7 ? htmlNode : bodyNode).style.overflow = 'hidden';
      }
    }
    layers.visible = true;
    if (con.visibilityHide) {
      removeClass(con, 'box_layer_hidden');
    } else {
      show(con); // Показывает див с контентом
    }
    layers.sh(el);
    window.updateWndVScroll && updateWndVScroll();
  },
  _hide: function(el, con) { // Скрывает слой и блок с контентом
    var done = function() {
      if (con && con.visibilityHide) {
        addClass(con, 'box_layer_hidden');
      } else {
        hide(con); // Скрывает див с контентом
      }
      if (!isVisible(layerWrap) && !cur._inLayer && (!isVisible(boxLayerWrap) || boxLayerWrap.visibilityHide)
        && ((window.mvcur && mvcur.minimized)
          || !isVisible(window.mvLayerWrap))
        && !isVisible(window.wkLayerWrap)) {
        layers.visible = false;
        toggleFlash(true);
        if (browser.mozilla) {
          pageNode.style.height = 'auto';
          pageNode.style.marginTop = '0px';
          if (window._oldScroll) {
            htmlNode.scrollTop = _oldScroll;
          }
        } else if (!browser.msie6) {
          (browser.msie7 ? htmlNode : bodyNode).style.overflow = 'auto';
        }
      }
      window.updateWndVScroll && updateWndVScroll();
    }
    layers.hd(el, done);
  }
};

// Ресайз блока с рекламой.
window.__seenAds = 2 - intval(getCookie('remixseenads'));
function updSeenAdsInfo() {
  var top = (getXY('left_ads', true) || {})[1];
  if (!top || !vk.id) return;

  var friendsHeight = (isVisible('left_friends') ? getSize(ge('left_friends'))[1] : 0)
  var adsY = getXY('left_ads', true)[1];
  var ads = Math.floor((lastWindowHeight - adsY + friendsHeight) / 211);
  if (ads < 0) {
    ads = 0;
  } else if (ads > 2) {
    ads = 2;
  }
  if (__seenAds !== ads) {
    __seenAds = ads;
    setCookie('remixseenads', 2 - ads, 30);
  }
}

// Обновления при скролле 
window.__scrLeft = 0;
function updSideTopLink(resized) {
  if (!window.scrollNode || browser.mobile) return;

  var pb = ge('page_body'), // Контент (без шапки и меню).
      xy = getXY(pb), // xy контента 
      st = scrollGetY(), // На сколько страница прокручена от вверха
      scl = bodyNode.scrollLeft, // на сколько страница прокручена влево 
      sb = ge('side_bar'), // Левая часть (Меню)
      is = isVisible(sb); // Есть ли левая часть (Меню)
  
  // Размер от вверха контента + сайтбар(меню) - позиция скролла
  // Вернет 0 если страница проскролится ниже блока с меню, 
  // иначе оставшийся число скролла до конца блока с меню.  
  window._stlSideTop = Math.max(xy[1] - (browser.mozilla ? getXY(pageNode)[1] : 0) + (is ? getSize(sb)[1] : 0) - (_fixedNav ? 0 : st), 0);

  // Установка блоку "наверх" и кнопкам позиции в зависимости от скролла. 
  if (resized || scl != __scrLeft) {
    var pl = ge('page_layout'), // весь контент
        // vk.rtl если true то блок "наверх" будет справа. 
        l = vk.rtl ? pl.offsetLeft + pl.ge('page_layout').offsetLeft  : 0, // отступ слева + отступ слева или 0 
        w = vk.rtl ? (lastWindowWidth - l) : pl.offsetLeft; // общаяя ширина - отступ слева или отсуп слева

    setStyle(_stlLeft, {
      left: l,
      width: Math.max(w, 0),
      height: lastWindowHeight
    });

    var sl = vk.rtl ? (xy[0] + pb.offsetWidth + 5) : w, sw = vk.rtl ? (l - sl) : (xy[0] - 5 - sl);
    
    setStyle(_stlSide, {
      left: sl - scl,
      width: Math.max(sw, 0)
    });
    __scrLeft = scl;
  }

  // Установка высоты и ширины блоку с кнопками. 
  setStyle(_stlSide, {
    top: _stlSideTop,
    height: Math.max(lastWindowHeight - _stlSideTop, 0)
  });
  __adsUpdate();


  if (is) {
    updateFixedMenu(st);
  } else {
    hide(_fMenu);
  }

  var mx = 200, // позиция скролла
      vis = _tbLink.loc || _stlWas || (st > mx), // если есть сслыка назад или тек. позиция скролла больше 200 то true
      o = 0, 
      f = false, 
      regBar = (st > 250) && cur._regBar; // true если тек.скролл больше 250 или есть регБар

  // 
  if (!vis) {
    if (_stlShown !== 0) {
      hide(_stlLeft, _stlSide);
      _stlShown = 0;
    }
  } else {
    if (_stlShown !== 1) {
      show(_stlLeft, _stlSide);
      addClass(_stlLeft, 'stl_active');
      addClass(_stlSide, 'stl_active');
      _stlShown = 1;
    }
    if (_tbLink.loc && cur._noUpLink) {
      st = 0;
    }
    if (_stlWas && st > 500) {
      _stlWas = 0;
    }
    if (st > mx) {
      o = (st - mx) / mx;
      if (_stlWasSet || _stlBack) {
        _stlWasSet = _stlBack = 0; f = 1;
        val(_stlText, getLang('global_to_top'));
        removeClass(_stlText, 'down');
        removeClass(_stlText, 'back');
      }
    } else {
      o = (mx - st) / mx;
      if (_stlWas) {
        if (!_stlWasSet) {
          _stlWasSet = 1; f = 0;
          val(_stlText, '');
          addClass(_stlText, 'down');
          if (_stlBack) {
            _stlBack = 0;
            removeClass(_stlText, 'back');
          }
        }
      } else if (_tbLink.loc) {
        if (!_stlBack) {
          _stlBack = 1; f = _tbLink.fast ? 1 : 0;
          val(_stlText, getLang('global_back'));
          addClass(_stlText, 'back');
          if (_stlWasSet) {
            _stlWasSet = 0;
            removeClass(_stlText, 'down');
          }
        }
      }
    }
    if (f !== false) toggleClass(_stlLeft, 'over_fast', hasClass(_stlLeft, 'over') && f);
    setStyle(_stlLeft, {
      opacity: Math.min(Math.max(o, 0), 1)
    });
  }
  if (!vk.id) {
    if (!_regBar && regBar) {
      _regBar = 1;
      val(ge('reg_bar_content'), cur._regBar);
      animate(ge('reg_bar'), {top: 0, transition: Fx.Transitions.sineInOut}, 400);
      animate(ge('stl_bg'), {paddingTop: 60, transition: Fx.Transitions.sineInOut}, 400);
    } else if (_regBar && !regBar) {
      _regBar = 0;
      animate(ge('reg_bar'), {top: -56, transition: Fx.Transitions.sineInOut}, 400);
      animate(ge('stl_bg'), {paddingTop: 13, transition: Fx.Transitions.sineInOut}, 400);
    }
  }
}

window.__adsLoaded = vkNow();
function __adsGetAjaxParams(ajaxParams, ajaxOptions) {
  __adsGetAjaxParams = function() {
    return window.AdsLight && AdsLight.getAjaxParams.apply(AdsLight.getAjaxParams, arguments) || {al_ad: null};
  };
  var result = stManager.add(['aes_light.js'], __adsGetAjaxParams.pbind(ajaxParams, ajaxOptions));
  return result || {al_ad: null};
}
function __adsUpdate() {
  __adsUpdate = function() {
    window.AdsLight && AdsLight.updateBlock();
  };
  stManager.add(['aes_light.js'], __adsUpdate);
}
function __adsSet(adsHtml, adsSection, adsCanShow) {
  __adsSet = function() {
    window.AdsLight && AdsLight.setNewBlock.apply(AdsLight.setNewBlock, arguments);
  };
  stManager.add(['aes_light.js'], __adsSet.pbind(adsHtml, adsSection, adsCanShow));
}

function updGlobalPlayer() {
  var w = window, de = document.documentElement;
  if (!w.pageNode) return;
  var _gp = ge('gp'), _gb = ge('gp_back'), _a = window.audioPlayer;
  if (_a && _a.gpHidden || cur.gpHidden) {
    if (isVisible(_gp)) toggleGlobalPlayer(false);
    return;
  }
  if (!_gp || !_gb) return;
  var pb = ge('page_body'), pbxy = getXY(pb), pbsz = getSize(pb);
  var sbw = sbWidth();
  var dwidth = Math.max(intval(w.innerWidth), intval(de.clientWidth)) - sbw * (browser.msie && (intval(browser.version) < 9) ? 0 : 1) - 1;
  var gpW = 154, gpL = vk.rtl ? pbxy[0] - gpW + 6 : pbxy[0] + pbsz[0] - 6;
  if (gpL + gpW <= dwidth && !vk.rtl || vk.rtl && gpL >= 0) {
    removeClass(_gp, 'reverse');
    setStyle(_gb, { width: gpW + 'px' });
    setStyle(_gp, {
      width: gpW + 'px',
      bottom: 'auto',
      left: gpL + 'px'
    });
    var headFixed = ge('head_play_btn') && getStyle(ge('page_header'), 'position') == 'fixed',
    gpAbs = !headFixed && scrollGetY() < 29;
    if (browser.mac) {
      var gpTop = gpAbs || headFixed ? 49 : 20;
      setStyle(_gp, {top: gpTop});
      toggleClass(_gp, 'abs', gpAbs);
    } else {
      var gpTop = headFixed ? 49 : Math.max(Math.min(49 - scrollGetY(), 49), 20),
          oldStyle = getStyle(_gp, 'top'), oldTop = parseInt(oldStyle);
      if ((oldTop || oldStyle == 'auto') && gpTop != oldTop) {
        cssAnim(_gp, {top: gpTop}, {duration: oldStyle == 'auto' ? 0 : 100, func: 'ease-in-out'}, function() {});
      }
    }
    show(_gp);
  } else {
    addClass(_gp, 'reverse');
    gpL = vk.rtl ? pbxy[0] + pbsz[0] + 9 : pbxy[0] - gpW - 9;
    if (gpL + gpW <= dwidth && vk.rtl || !vk.rtl && gpL >= 0) {
      setStyle(_gb, { width: gpW + 'px' });
      setStyle(_gp, {
        width: gpW + 'px',
        top: 'auto',
        bottom: '20px',
        left: gpL + 'px'
      });
      show(_gp);
    } else {
      hide(_gp);
    }
  }
}

function toggleGlobalPlayer(open) {
  var _gp = ge('gp'), _a = window.audioPlayer;
  if (!_gp) return;
  if (open) {
    if (cur.gpHidden || cur.gpShowing || _a && _a.gpHidden || ge('head_play_btn') && scrollGetY() < 33) return;
    // updGlobalPlayer();
    cur.gpShowing = true;
    cur.gpHiding = false;
    fadeIn(_gp, 300, function() {
      cur.gpShowing = false;
    });
  } else {
    if (cur.gpHiding) return;
    // updGlobalPlayer();
    cur.gpHiding = true;
    cur.gpShowing = false;
    fadeOut(_gp, 300, function() {
      cur.gpHiding = false;
    });
  }
}

// Ресайз боксов, лейеров и т.д 
function onBodyResize(force) {
  var w = window, de = document.documentElement;
  if (!w.pageNode) return;

  var dwidth = Math.max(intval(w.innerWidth), intval(de.clientWidth));
  var dheight = Math.max(intval(w.innerHeight), intval(de.clientHeight));
  var sbw = sbWidth(), changed = false;

  if (browser.mobile) {
    dwidth = Math.max(dwidth, intval(bodyNode.scrollWidth));
    dheight = Math.max(dheight, intval(bodyNode.scrollHeight));
  } else if (browser.msie7) {
    if (htmlNode.scrollHeight > htmlNode.offsetHeight && !layers.visible) {
      dwidth += sbw + 1;
    }
  } else if (browser.msie8) {
    if (htmlNode.scrollHeight + 3 > htmlNode.offsetHeight && !layers.visible) {
      dwidth += sbw + 1;
    }
  }
  if (w.lastWindowWidth != dwidth || force === true) {
    changed = true;
    w.lastInnerWidth = w.lastWindowWidth = dwidth;
    layerWrap.style.width = boxLayerWrap.style.width = dwidth + 'px';
    var layerWidth = layer.style.width = boxLayer.style.width = (dwidth - sbw - 2) + 'px';
    if (window.mvLayerWrap && !mvcur.minimized) {
      mvLayerWrap.style.width = dwidth + 'px';
      mvLayer.style.width = layerWidth;
    }
    if (window.wkLayerWrap) {
      wkLayerWrap.style.width = dwidth + 'px';
      wkLayer.style.width = layerWidth;
    }

    if (bodyNode.offsetWidth < vk.width + sbw + 2) {
      dwidth = vk.width + sbw + 2;
    }
    if (dwidth) {
      for (var el = pageNode.firstChild; el; el = el.nextSibling) {
        if (!el.tagName) continue;
        for (var e = el.firstChild; e; e = e.nextSibling) {
          if (e.className == 'scroll_fix') {
            e.style.width = ((w.lastInnerWidth = (dwidth - sbw * (browser.msie7 ? 2 : 1) - 1)) - 1) + 'px';
          }
        }
      }
    }
    if (_pads.shown) Pads.reposition();
  }
  if (w.lastWindowHeight != dheight || force === true) {
    changed = true;
    w.lastWindowHeight = dheight;
    layerBG.style.height = boxLayerBG.style.height =
    layerWrap.style.height = boxLayerWrap.style.height = dheight + 'px';
    if (window.mvLayerWrap && !mvcur.minimized) {
      mvLayerWrap.style.height = dheight + 'px';
    }
    if (window.wkLayerWrap) {
      wkLayerWrap.style.height = dheight + 'px';
    }
    if (_pads.layerBG) _pads.layerBG.style.height = dheight + 'px';
    if (browser.mozilla && layers.visible) {
      pageNode.style.height = (_oldScroll + dheight) + 'px';
    } else if (browser.msie6) {
      pageNode.style.height = dheight + 'px';
    }
  }
  updSideTopLink(1);
  // Для ресайза всех открытых боксов 
  if (changed && w.curRBox && w.curRBox.boxes && window.getWndInner) {
    var wndInner = getWndInner();
    each (curRBox.boxes, function() {this._wnd_resize(wndInner[0], wndInner[1])});
  }
  setTimeout(updSeenAdsInfo, 0);
  updGlobalPlayer();
  if (window.Pads && Pads.setPos && window._pads && (_pads.shown == 'mus' || _pads.shown == 'game')) {
    Pads.setPos(_pads.shown);
  }
  if (window.tooltips) {
    tooltips.rePositionAll();
  }
}

// Обновления левой части, плеера и pads.
function onBodyScroll() {
  if (!window.pageNode) return;

  updSideTopLink();
  updGlobalPlayer();
  if (_pads.shown) Pads.onScroll();
}


if (!window._pads) _pads = {cache: {}};


function onDocumentClick(e) {
  if (checkEvent(e)) {
    return true;
  }
  if (_pads.shown) {
    for (var el = e.target; el != bodyNode; el = domPN(el)) {
      if (!el || el.id && el.id.match(/^(box_|mv_|wk_|pad_)?(layer|pad)_(bg|wrap)$/) || (el.className || '').match(/(^|s)tt(s|$)/)) break;
    }
    if (el == bodyNode) Pads.hide();
  }

  if (cur.onMouseClick && cur.onMouseClick(e)) return;
  if (!(e = (window.event || e.originalEvent || e))) {
    return true;
  }
  var sel = trim((
    window.getSelection && window.getSelection() ||
    document.getSelection && document.getSelection() ||
    document.selection && document.selection.createRange().text || ''
  ).toString());
  if (sel) {
    return true;
  }

  var i = 4, target = e.target || e.srcElement, href, path, hash, w = window;
  while (target && target != bodyNode && target.tagName != 'A' && i--) {
    target = target.parentNode;
  }
  if (!target || target.tagName != 'A' || target.onclick || target.onmousedown) {
    return true;
  }
  href = target.href;
  if (href && (target.getAttribute('target') || nav.baseBlank)) {
    try {
      w._opener.contentWindow.open(href, '_blank');
      setTimeout(w._reopen, 0);
      return cancelEvent(e);
    } catch(er) {
      return true;
    }
  }
  if (!href.indexOf(location.protocol)) {
    href = href.replace(location.protocol + '//', '');
  }
  if (!href.indexOf(location.hostname)) {
    href = href.replace(location.hostname, '');
  }
  href = href.replace(/^(vkontakte.ru/|vk.com/)/, '/');
  var opts = {};
  if ((hash = href.match(/^/(.+?)#[!/](.+?)$/)) && !hash[1].match(/^app(d+)/)) {
    opts.permanent = hash[1];
    href = '/' + hash[2];
  }
  if (href.match(/#$/) || !(path = href.match(/^/(.*?)(?|#|$)/))) {
    return true;
  }
  path = path[1];
  if (path.indexOf('.php') > 0 || path.indexOf('/') > 0 || path.match(/^(doc-?d+_d+|graffitid+|regd+|images|.js|js/|.css|css/|source)/)) {
    return true;
  }
  try {
    nav.go(href, e, opts);
    return cancelEvent(e);
  } catch (e) {
    return true;
  }
}

// Обработка нажатия Ctrl + Enter.
function onCtrlEnter(ev, handler) {
  ev = ev || window.event;
  if (ev.keyCode == 10 || ev.keyCode == 13 && (ev.ctrlKey || ev.metaKey && browser.mac)) {
    handler();
    cancelEvent(ev);
  }
}

// Установка icon.
function setFavIcon(href, force) {
  if (!window.icoNode) return;
  if (icoNode.href == href && !force) return;
  var ico = ce('link', {rel: 'shortcut icon', type: 'image/gif', href: href});
  headNode.replaceChild(ico, icoNode);
  icoNode = ico;
}

// Предварительная установка icon.
(function() {
var step = 1, timer, to, func = false;
if (browser.mozilla) {
  func = function() {
    setFavIcon('/images/icons/prgicon.gif');
  }
} else if (browser.chrome || browser.opera && !browser.opera_mobile) {
  func = function() {
    step = step % 4 + 1;
    setFavIcon('/images/icons/prgicon' +  step + '.gif');
    timer = setTimeout(arguments.callee, 250);
  }
}
window.showTitleProgress = function(timeout) {
  if (browser.mozilla || browser.chrome) return;
  if (timeout > 0) {
    to = setTimeout(showTitleProgress.pbind(false), timeout);
    return;
  }
  if (timer) {
    return;
  }
  if (document.body) {
    document.body.style.cursor = 'progress';
  }
  if (func) func();
}
window.hideTitleProgress = function() {
  if (browser.mozilla || browser.chrome) return;
  clearTimeout(to);
  document.body.style.cursor = 'default';
  if (timer) {
    clearTimeout(timer);
    timer = false;
  }
  if (browser.mozilla || browser.chrome || browser.opera && !browser.opera_mobile) {
    setFavIcon('/images/favicon' + (vk.intnat ? '_vk' : 'new') + (window.devicePixelRatio >= 2 ? '_2x' : '') + '.ico');
  }
}
})();

// Проверка события у блока "наверх".
function _stlClick(e) {
  return checkEvent(e) || cancelEvent(e);
}

// Нажатия кнопки "наверх" - скроллинг.
function _stlMousedown(e) {
  e = e || window.event;
  if (checkEvent(e)) {
    return;
  }
  if (!__afterFocus) {
    if (_stlWasSet && _stlWas) {
      var to = _stlWas;
      _stlWas = 0;
      scrollToY(to, 0, true);
    } else if (_stlBack === 1) {
      _tbLink.onclick();
    } else {
      _stlWas = scrollGetY();
      scrollToY(0, 0, true);
    }
  }
  return cancelEvent(e);
}

// Наведения мыши на блок "наверх".
function _stlMouseover(e) {
  var ev = (e ? e.originalEvent || e : window.event || {}),
      over = (ev.type == 'mouseover') && (ev.pageX > 0 || ev.clientX > 0) && !hasClass(_fMenu, 'over');

  toggleClass(_stlLeft, 'over', over);
  toggleClass(_stlLeft, 'over_fast', over && (_stlBack === 0 || _tbLink.fast) && _stlWasSet === 0);
  toggleClass(_stlSide, 'over', over);
}

vk.width = 791;
function domStarted() {
  window.headNode = geByTag1('head');
  extend(window, {
    icoNode:  geByTag1('link', headNode),
    bodyNode: geByTag1('body'),
    htmlNode: geByTag1('html'),
    utilsNode: ge('utils'),
    _fixedNav: false,
    _tbLink: {}
  });
  bodyNode.onresize = onBodyResize.pbind(false);
  debugLog('dom started');
  if (!utilsNode) return;

  if (browser.mozilla) {
    addClass(bodyNode, 'firefox');
  } else if (browser.msie6) {
    addClass(bodyNode, 'nofixed');
  } else if (browser.mobile) {
    addClass(bodyNode, 'mobfixed');
  }

  for (var i in StaticFiles) {
    var f = StaticFiles[i];
    f.l = 1;
    if (f.t == 'css') {
      utilsNode.appendChild(ce('div', {id: f.n}));
    }
  }

  var l = ge('layer_bg'), lw = l.nextSibling, bl = ge('box_layer_bg'), blw = bl.nextSibling;
  extend(window, {
    _reopen: function() {
      re(window._opener);
      window._opener = utilsNode.appendChild(ce('iframe'));
    },
    layerBG: l,
    boxLayerBG: bl,
    layerWrap: lw,
    layer: lw.firstChild,
    boxLayerWrap: blw,
    boxLayer: blw.firstChild,
    boxLoader: blw.firstChild.firstChild,
    _stlSide: ge('stl_side'),
    _stlLeft: ge('stl_left'),
    _stlShown: 0,
    _stlWas: 0,
    _stlWasSet: 0,
    _stlBack: 0,
    _regBar: 0,
    __afterFocus: false,
    __needBlur: false
  });
  _reopen();
  if (!browser.mobile) {
    var s = {
      className: 'fixed',
      onclick: _stlClick,
      onmousedown: _stlMousedown,
      onmouseover: _stlMouseover,
      onmouseout: _stlMouseover
    };
    val(_stlLeft, '<div id="stl_bg"><nobr id="stl_text">' + getLang('global_to_top') + '</nobr></div>');
    extend(_stlLeft, s);
    extend(_stlSide, s);
    window._stlBg = _stlLeft.firstChild;
    window._stlText = _stlBg.firstChild;
    addEvent(window, 'blur', function(e) {
      __needBlur = false;
    });
    addEvent(window, 'focus', function(e) {
      if (__needBlur || browser.mac) {
        return; // opera fix
      }
      __afterFocus = __needBlur = true;
      setTimeout(function() {
        __afterFocus = false;
      }, 10);
    });
  }

  // Box.
  addEvent(boxLayerWrap, 'click', __bq.hideLastCheck);

  extend(layers, {
    show: layers._show.pbind(l, lw),
    boxshow: layers._show.pbind(bl, blw),
    wrapshow: layers._show.pbind(l),
    hide: layers._hide.pbind(l, lw),
    boxhide: layers._hide.pbind(bl, blw),
    wraphide: layers._hide.pbind(l)
  });

  hab.init();
  if (window._retinaInit) {
    window._retinaInit();
  } else {
    window._initedCheck = 1;
  }
}

window._fMenu = window._fMenuShown = false;
window._fMenuHidden = true;
window._fMenuLnks = {};
function initFixedMenu() {
  var lnks = {fr: 'friends?section=requests', ph: 'albums?act=added', vid: 'video?section=tagged', msg: 'im', gr: 'groups', nws: 'feed?section=notifications', ap: 'apps'}, text = [], i, el;
  for (i in lnks) {
    text.push('<a class="fmenu_item fl_r" onmouseover="Pads.preload('' + i + '')" onmousedown="return Pads.show('' + i + '', event)" href="' + lnks[i] + '"><span class="fmenu_text inl_bl"></span><span id="fmenu_' + i + '" class="fmenu_icon inl_bl"></span></a>');
  }
  re(_fMenu);
  extend(window, {
    _fMenu: _stlSide.appendChild(ce('div', {id: 'fmenu', innerHTML: text.join(''), onmousedown: function(e) {
      e = e || window.event;
      return (checkEvent(e) || browser.msie6) ? true : cancelEvent(e);
    }})),
    _fMenuLnks: {}
  });
  _fMenu.onmouseover = addClass.pbind(_fMenu, 'over');
  _fMenu.onmouseout = removeClass.pbind(_fMenu, 'over');
  el = domFC(_fMenu);
  for (i in lnks) {
    _fMenuLnks[i] = el;
    el = domNS(el);
  }
  updateFixedMenu(false, true);
}
function updateFixedMenu(st, refr) {
  if (!window._fMenu) return;

  var os = _fMenuShown, oh = _fMenuHidden;
  if (refr) {
    if (!(_fMenuShown = !!(vk.id > 0 && isVisible('side_bar')))) {
      _fMenuHidden = true;
      hide(_fMenu);
      if (_pads.shown) Pads.setPos(_pads.shown);
    }
  }
  if (!_fMenuShown) return;

  if (refr) {
    for (var i in _fMenuLnks) {
      var c = intval(vk.counts[i]), el = _fMenuLnks[i];
      toggle(el, (c > 0) || !c && (i == 'msg' || i == 'nwvs'));
      val(domFC(el), c > 1000 ? ('..' + (c % 1000) + '&nbsp;') : (c > 0 ? ('+' + c + '&nbsp;') : ''));
    }
    if (!os) {
      show(_fMenu);
      setStyle(_fMenu, {visibility: 'hidden', opacity: 0});
      if (_pads.shown) Pads.setPos(_pads.shown);
    }
  }
  if (st === false) st = scrollGetY();
  if ((_fMenuHidden = (_stlSideTop > 10)) != oh) {
    if (oh) {
      setStyle(_fMenu, {visibility: ''});
      animate(_fMenu, {opacity: 1}, 100);
    } else {
      animate(_fMenu, {opacity: 0}, 100, function() { setStyle(_fMenu, {visibility: 'hidden'}); });
    }
  }
}

vk.started = vkNow();
function domReady() {
  if (!utilsNode) return;

  var sb = ge('side_bar');
  extend(window, {
    pageNode: ge('page_wrap'),
    _fixedNav: (sb && getStyle(sb, 'position') === 'fixed'),
    _tbLink: ge('top_back_link')
  });

  window.scrollNode = browser.msie6 ? pageNode : ((browser.chrome || browser.safari) ? bodyNode : htmlNode);

  if (vk.al == 1) {
    showTitleProgress();
  }

  var dt = Math.max(vkNow() - vk.started, 10), speed = intval((vk.contlen || 1) / dt * 1000);
  if (browser.mozilla && browser.version >= 4) {
    speed /= 2.5;
  } else if (browser.mozilla) {
    speed *= 1.5;
  } else if (browser.msie && browser.version >= 7) {
    speed /= 1.5;
  } else if (browser.msie) {
    speed *= 2.5;
  }
  __stm.lowlimit = intval(Math.max(2000000 / speed, 1) * 150);
  __stm.highlimit = __stm.lowlimit * 6;
  __stm.lowlimit = Math.min(__stm.lowlimit, 600);

  onBodyResize();
  setTimeout(onBodyResize.pbind(false), 0);
  setInterval(function() { // some strange chrome bug..
    if (!bodyNode.onresize) {
      bodyNode.onresize = onBodyResize.pbind(false);
    }
  }, 1000);

  if (_pads.shown) Pads.updateHeight();

  var scrolledNode = browser.msie6 ? pageNode : window;
  addEvent(scrolledNode, 'scroll', onBodyScroll);

  var dl = ge('debuglog');
  if (dl && !window._debugLogHist[window._debugLogHistOffset]) {
    window._debugLogHist[window._debugLogHistOffset] = dl.innerHTML;
  }

  setTimeout(initFixedMenu, 0);
}
function onDomReady(f) {
  f();
}

// Ajax
function serializeForm(form) {
  if (typeof(form) != 'object') {
    return false;
  }
  var result = {};
  var g = function(n) {
    return geByTag(n, form);
  };
  var nv = function(i, e){
    if (!e.name) return;
    if (e.type == 'text' || !e.type) {
      result[e.name] = val(e);
    } else {
      result[e.name] = (browser.msie && !e.value && form[e.name]) ? form[e.name].value : e.value;
    }
  };
  each(g('input'), function(i, e) {
    if ((e.type != 'radio' && e.type != 'checkbox') || e.checked) return nv(i, e);
  });
  each(g('select'), nv);
  each(g('textarea'), nv);

  return result;
}

// Кодирует из массива в валидный URL 
function ajx2q(qa) {
  var query = [], enc = function(str) {
    if (window._decodeEr && _decodeEr[str]) {
      return str;
    }
    try {
      return encodeURIComponent(str);
    } catch (e) { return str; }
  };

  for (var key in qa) {
    if (qa[key] == null || isFunction(qa[key])) continue;
    if (isArray(qa[key])) {
      for (var i = 0, c = 0, l = qa[key].length; i < l; ++i) {
        if (qa[key][i] == null || isFunction(qa[key][i])) {
          continue;
        }
        query.push(enc(key) + '[' + c + ']=' + enc(qa[key][i]));
        ++c;
      }
    } else {
      query.push(enc(key) + '=' + enc(qa[key]));
    }
  }
  query.sort(); 
  return query.join('&'); // 
}

// Декодирует из URL в массив с валид значениями.   
function q2ajx(qa) {
  if (!qa) return {};
  var query = {}, dec = function(str) {
    try {
      return decodeURIComponent(str);
    } catch (e) {
      window._decodeEr = window._decodeEr || {};
      _decodeEr[str] = 1;
      return str;
    }
  };
  qa = qa.split('&');
  each(qa, function(i, a) {
    var t = a.split('=');
    if (t[0]) {
      var v = dec(t[1] + '');
      if (t[0].substr(t.length - 2) == '[]') {
        var k = dec(t[0].substr(0, t.length - 2));
        if (!query[k]) {
          query[k] = [];
        }
        query[k].push(v);
      } else {
        query[dec(t[0])] = v;
      }
    }
  });
  return query;
}

// Загрузка статик. файлов с возможностью передачи обработчика. 
var stManager = {
  _waiters: [],
  _wait: function() {
    var l = __stm._waiters.length, checked = {}, handlers = [];
    if (!l) {
      clearInterval(__stm._waitTimer);
      __stm._waitTimer = false;
      return;
    }
    for (var j = 0; j < l; ++j) {
      var wait = __stm._waiters[j][0];
      for (var i = 0, ln = wait.length; i < ln; ++i) {
        var f = wait[i];
        if (!checked[f]) {
          if (!StaticFiles[f].l && StaticFiles[f].t == 'css' && getStyle(StaticFiles[f].n, 'display') == 'none') {
            __stm.done(f);
          }
          if (StaticFiles[f].l) {
            checked[f] = 1;
          } else {
            checked[f] = -1;
            if (vk.loaded) {
              var c = ++StaticFiles[f].c;
              if (c > __stm.lowlimit && stVersions[f] > 0 || c > __stm.highlimit) {
                if (stVersions[f] < 0) {
                  topError('<b>Error:</b> Could not load <b>' + f + '</b>.', {dt: 5, type: 1, msg: 'Failed to load with ' + __stm.lowlimit + '/' + __stm.highlimit + ' limits (' + ((vkNow() - vk.started) / 100) + ' ticks passed)', file: f});
                  StaticFiles[f].l = 1;
                  checked[f] = 1;
                } else {
                  topMsg('Some problems with loading <b>' + f + '</b>...', 5);
                  stVersions[f] = irand(-10000, -1);
                  __stm._add(f, StaticFiles[f]);
                }
              }
            }
          }
        }
        if (checked[f] > 0) {
          wait.splice(i, 1);
          --i; --ln;
        }
      }
      if (!wait.length) {
        handlers.push(__stm._waiters.splice(j, 1)[0][1]);
        --j; --l;
      }
    }
    for (var j = 0, l = handlers.length; j < l; ++j) {
      handlers[j]();
    }
  },
  _addCss: function(text) {
    var elem = headNode.appendChild(ce('style', {
      type: 'text/css',
      media: 'screen'
    }));
    if (elem.sheet) {
      elem.sheet.insertRule(text, 0);
    } else if (elem.styleSheet) {
      elem.styleSheet.cssText = text;
    }
  },
  _srcPrefix: function (f, v) {
    if (!vk.stDomains ||
        __dev ||
        f.indexOf('.js') == -1 && f.indexOf('.css') == -1 ||
        f.indexOf('lang') != -1 ||
        f.indexOf('dyn-') != -1 ||
        f.indexOf('loader_nav') != -1 ||
        location.protocol == 'https:') {
      return '';
    }
    if (f.indexOf('.css') != -1) {
      return 'http://st0.userapi.com';
    }
    f = f.replace(/[^a-zd.-_]/ig, '');
    var src = intval(v), n = f.length, i;
    for (i = 0; i < n; i++) {
      src += f.charCodeAt(i);
    }
    // debugLog(f, src, vk.stDomains);
    return 'http://st' + ((src % vk.stDomains) + 1) + '.userapi.com';
  },
  _add: function(f, old) {
    var name = f.replace(/[/.]/g, '_'),
        f_ver = stVersions[f],
        f_full = f + '?' + f_ver,
        f_prefix = stManager._srcPrefix(f, f_ver);

    if (old && old.l && old.t == 'css') {
      __stm._addCss('#' + name + ' {display: block; }');
    }
    StaticFiles[f] = {v: f_ver, n: name, l: 0, c: 0};
    if (f.indexOf('.js') != -1) {
      var p = '/js/';
      if (stTypes.fromLib[f]) {
        p += 'lib/';
      } else if (!/^langd/i.test(f) && !stTypes.fromRoot[f]) {
        p += 'al/';
      }
      headNode.appendChild(ce('script', {
        type: 'text/javascript',
        src: f_prefix + p + f_full
      }));

      StaticFiles[f].t = 'js';
    } else if (f.indexOf('.css') != -1) {
      var p = '/css/' + (stTypes.fromRoot[f] ? '' : 'al/');
      headNode.appendChild(ce('link', {
        type: 'text/css',
        rel: 'stylesheet',
        href: f_prefix + p + f_full
      }));

      StaticFiles[f].t = 'css';

      if (!ge(name)) {
        utilsNode.appendChild(ce('div', {id: name}));
      }
    }
  },

  add: function(files, callback) {
    var wait = [], de = document.documentElement;
    if (!isArray(files)) files = [files];
    for (var i in files) {
      var f = files[i];
      if (f.indexOf('?') != -1) {
        f = f.split('?')[0];
      }
      if (/^langd/i.test(f)) {
        stVersions[f] = stVersions['lang'];
      } else if (!stVersions[f]) {
        stVersions[f] = 1;
      }
// Opera Speed Dial fix
      var opSpeed = browser.opera && de.clientHeight == 768 && de.clientWidth == 1024;
      if ((opSpeed || __debugMode) && !(browser.iphone || browser.ipad) && f != 'common.js' && f != 'common.css' && stVersions[f] > 0 && stVersions[f] < 1000000000) stVersions[f] += irand(1000000000, 2000000000);
      var old = StaticFiles[f];
      if (!old || old.v != stVersions[f]) {
        __stm._add(f, old);
      }
      if (callback && !StaticFiles[f].l) {
        wait.push(f);
      }
    }
    if (!callback) return;
    if (!wait.length) {
      return callback();
    }
    __stm._waiters.push([wait, callback]);
    if (!__stm._waitTimer) {
      __stm._waitTimer = setInterval(__stm._wait, 100);
    }
  },
  done: function(f) {
    if (stVersions[f] < 0) {
      topMsg('<b>Warning:</b> Something is bad, please <b><a href="/page-777107_43991681">clear your cache</a></b> and restart your browser.', 10);
    }
    StaticFiles[f].l = 1;
  }
}, __stm = stManager;

function requestBox(box, onDone, onFail) {
  box.setOptions({onDestroy: onFail});
  box.onDone = function() {
    onDone.apply(null, arguments);
  }
  return box;
}
function activateMobileBox(opts) {
  return requestBox(showBox('activation.php', {
    act: 'activate_mobile_box',
    hash: opts.hash
  }), function() {
    vk.nophone = 0;
    opts.onDone();
  }, opts.onFail);
}
function validateMobileBox(opts) {
  return requestBox(showBox('activation.php', {
    act: 'validate_box',
    captcha: opts.acceptCaptcha ? 1 : '',
    hash: opts.hash
  }, {stat: ['uncommon.css']}), opts.onDone, opts.onFail);
}

var ajaxCache = {};
var globalAjaxCache = {};
var ajax = {
  _init: function() {// Кроссбраузерная установка ajax-а в _req.
    var r = false;
    try {
      if (r = new XMLHttpRequest()) {
        ajax._req = function() { return new XMLHttpRequest(); }
        return;
      }
    } catch(e) {}
    each(['Msxml2.XMLHTTP', 'Microsoft.XMLHTTP'], function() {
      try {
        var t = '' + this;
        if (r = new ActiveXObject(t)) {
          (function(n) {
            ajax._req = function() { return new ActiveXObject(n); }
          })(t);
          return false;
        }
      } catch(e) {}
    });
    if (!ajax._req) {
      location.replace('/badbrowser.php');
    }
  },
  _getreq: function() {// Возвращает экземпляр ajaxa, если его нету, создает.
    if (!ajax._req) ajax._init();
    return ajax._req();
  },
  _frameover: function(js, params) {
    var node = iframeTransport.parentNode;
    node.innerHTML = '';
    utilsNode.removeChild(node);
    iframeTransport = false;
    if (js || params) {
      /*var onOver = function() {
        eval('(function(){' + js + ';})()');
      }*/
      /*if (!ajax.onFrameOver) {
        ajax.onFrameOver = onOver;
      } else {
        onOver();
      }*/
      ajax.framegot(false, false, js, params);
    }
    ajax.framegot(false);
    if (cur.onFrameBlocksDone) {
      cur.onFrameBlocksDone();
    }
    ajax.tOver = new Date().getTime();
    if (window._test) _test();
  },
  _receive: function(cont, html, js, bench, params) {
    var c = cont && ge(cont);
    if (c && html) {
      if (!c.firstChild) {
        val(c, html);
      } else {
        c.appendChild(cf(html));
      }
    }
    if (js) {
      var scr = '(function(){' + js + ';})()';
      if (__debugMode) {
        eval(scr);
      } else try {
        eval(scr);
      } catch (e) {
        topError(e, {dt: 15, type: 8, url: ajax._frameurl, js: js, answer: Array.prototype.slice.call(arguments).join('<!>')});
      }
      if (bench) {
        ajax.tModule = cur.module;
      }
    }
    if (params && 'leftads' in params) {
      __adsSet(params.leftads, params.ads_section || '', params.ads_can_show);
    }
    ajax._framenext();
  },
  framedata: false,
  _framenext: function() {
    if (!(ajax.framedata || {}).length) return;
    var d = ajax.framedata.shift();
    if (d === true) {
      ajax._framenext();
    } else if (d === false) {
      ajax.framedata = false;
    } else {
      setTimeout(ajax._receive.pbind(d[0], d[1], d[2], true, d[3]), 0);
    }
  },
  framegot: function(c, h, j, p) {
    if (!ajax.framedata) return;
    ajax.framedata.push((h === undefined && j === undefined && p === undefined) ? c : [c, h, j, p]);
    if (ajax.framedata.length == 1) {
      ajax._framenext();
    }
  },
  framepost: function(url, query, done) {
    if (window.iframeTransport) {
      ajax._frameover();
    }
    window.iframeTransport = utilsNode.appendChild(ce('div', {innerHTML: '<iframe></iframe>'})).firstChild;
    ajax._framedone = done;
    ajax.framedata = [true];
    url += '?' + ((typeof(query) != 'string') ? ajx2q(query) : query);
    url += (url.charAt(url.length - 1) != '?' ? '&' : '') + '_rndVer=' + irand(0, 99999);
    ajax._frameurl = iframeTransport.src = url;
  },
  plainpost: function(url, query, done, fail, urlonly) {
    var r = ajax._getreq();
    var q = (typeof(query) != 'string') ? ajx2q(query) : query;
    r.onreadystatechange = function() {
      if (r.readyState == 4) {
        if (r.status >= 200 && r.status < 300) {
          if (done) done(r.responseText, r);
        } else { // e.g sleep
          if (fail) fail(r.responseText, r);
        }
      }
    }
    try {
      r.open('POST', url, true);
    } catch(e) {
      return false;
    }
    if (!urlonly) {
      r.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
      r.setRequestHeader('X-Requested-With', 'XMLHttpRequest');
    }
    r.send(q);
    return r;
  },
  post: function(url, query, options) {
    if (url.substr(0, 1) != '/') url = '/' + url;
    var o = extend({_captcha: false, _box: false}, options || {}), q = extend({al: o.frame ? -1 : 1}, query);
    if (o.progress) {
      if (!o.showProgress) {
        o.showProgress = show.pbind(o.progress);
      }
      if (!o.hideProgress) {
        o.hideProgress = hide.pbind(o.progress);
      }
    }
    if (o.loader) {
      o.showProgress = function() {
        boxRefreshCoords(boxLoader);
        show(boxLoader);
        show(boxLayerWrap);
      }
      o.hideProgress = function() {
        hide(boxLoader);
        hide(boxLayerWrap);
      }
    }
    return ajax._post(url, q, o);
  },
  preload: function(url, query, data) {
    if (url.substr(0, 1) != '/') url = '/' + url;
    ajaxCache[url + '#' + ajx2q(query)] = data;
  },
  _debugLog: function(text) {
     window._updateDebug = function() {
       var dlw = ge('debuglogwrap');
       if (dlw) {
         dlw.innerHTML = text;
         window._debugLogHist[window._debugLogHistOffset] += ge('debuglog').innerHTML;
         if (window._debugLogHistShow) {
           ge('debuglog').innerHTML = window._debugLogHist.join('');
           var lnk = ge('debugloghistlnk');
           if (lnk) { lnk.innerHTML = 'Show last'; }
         }
         window._updateDebug = false;
       }
     }
  },
  _parseRes: function(answer) {
    window._updateDebug = false;
    for (var i = 0, l = answer.length; i < l; ++i) {
      var ans = answer[i];
      if (ans.substr(0, 2) == '<!') {
        var from = ans.indexOf('>');
        var type = ans.substr(2, from - 2);
        ans = ans.substr(from + 1);
        switch (type) {
        case 'json' :
          answer[i] = parseJSON(ans);
          break;
        case 'int'  : answer[i] = intval(ans); break;
        case 'float': answer[i] = floatval(ans); break;
        case 'bool' : answer[i] = intval(ans) ? true : false; break;
        case 'null' : answer[i] = null; break;
        case 'debug':
          ajax._debugLog(ans);
          answer.pop(); // <!debug> must be last one
          l--;
        break;
        }
      }
    }
  },
  _post: function(url, q, o) {
    if (!q.captcha_sid && o.showProgress) o.showProgress();
    var cacheKey = false;
    extend(q, __adsGetAjaxParams(q, o));
    if (o.cache) {
      var boldq = clone(q);
      delete boldq.al;
      delete boldq.al_ad;
      delete boldq.ads_section;
      delete boldq.captcha_sid;
      delete boldq.captcha_key;
      cacheKey = url + '#' + ajx2q(boldq);
    }
    var hideBoxes = function() {
      for (var i = 0, l = arguments.length; i < l; ++i) {
        var box = arguments[i];
        if (box && box.isVisible()) {
          box.setOptions({onHide: false, onDestroy: false});
          box.hide();
        }
      }
      return false;
    }
    var fail = function(text, r) {
      if (o.hideProgress) o.hideProgress();
      if (o._suggest) cleanElems(o._suggest);
      o._suggest = o._captcha = o._box = hideBoxes(o._captcha, o._box);
      if (text.indexOf('The page is temporarily unavailable') != -1 && __dev) {
        ajax._post(url, q, o);
        return false;
      }
      if (!o.onFail || o.onFail(text) !== true) {
        topError(text, {dt: 5, type: 3, status: r.status, url: url, query: q && ajx2q(q)});
      }
    }
    if (o.stat) {
      var statAct = false;
      stManager.add(o.stat, function() {
        if (statAct) {
          statAct();
        }
        o.stat = false;
      });
    }
    // Process response function
    var processResponse = function(code, answer) {
      if (o.cache) {
        var answ = ajaxCache[cacheKey];
        if (answ && answ._loading) {
          setTimeout(function() {
            for (var i in answ._callbacks) {
              answ._callbacks[i](code, answer);
            }
          }, 0);
          delete ajaxCache[cacheKey];
        }
      }
      if (o.stat) {
        o.stat = false;
        statAct = processResponse.pbind(code, answer);
        return false;
      }
      if (o.cache && !o.forceGlobalCache) {
        if (!code) {
          ajaxCache[cacheKey] = answer;
        }
      }

      // Parse response
      if (o.hideProgress) o.hideProgress();
      if (code != 2) {
        if (o._captcha) {
          if (o._suggest) cleanElems(o._suggest);
          o._suggest = o._captcha = hideBoxes(o._captcha);
        }
        o._box = hideBoxes(o._box);
      }
      switch (code) {
      case 1: // email not confirmed
        if (ge('confirm_mail')) {
          showFastBox({
            width: 430,
            title: ge('confirm_mail_title').value,
            onDestroy: o.onFail
          }, '<div class="confirm_mail">' + ge('confirm_mail').innerHTML + '</div>');
        } else {
          topMsg('<b>Error!</b> Email is not confirmed!');
        }
        break;
      case 2: // captcha
        var resend = function(sid, key) {
          var nq = extend(q, {captcha_sid: sid, captcha_key: key});
          var no = o.cache ? extend(o, {cache: -1}) : o;
          ajax._post(url, nq, no);
        }
        var addText = '';
        // if (vk.nophone == 1 && !vk.nomail) {
        //   addText = getLang('global_try_to_activate').replace('{link}', '<a class="phone_validation_link">').replace('{/link}', '</a>');
        //   addText = '<div class="phone_validation_suggest">' + addText + '</div>';
        // }
        o._captcha = showCaptchaBox(answer[0], intval(answer[1]), o._captcha, {
          onSubmit: resend,
          addText: addText,
          onDestroy: function() {
            if (o.onFail) o.onFail();
          }
        });
        o._suggest = geByClass1('phone_validation_link', o._captcha.bodyNode);
        if (o._suggest) {
          addEvent(o._suggest, 'click', function() {
            o._box = validateMobileBox({onDone: o._captcha.submit});
          });
        }
        break;
      case 11: // mobile validation needed
      case 12:
        var no = o.cache ? extend(o, {cache: -1}) : o;
        o._box = validateMobileBox({acceptCaptcha: (code == 11), onDone: function(sid, key) {
          vk.nophone = 0;
          if (sid) o._captcha = curBox();
          ajax._post(url, sid ? extend(q, {captcha_sid: sid, captcha_key: key}) : q, no);
        }, onFail: o.onFail, hash: answer[0]});
        break;
      case 3: // auth failed
        var no = o.cache ? extend(o, {cache: -1}) : o;
        window.onReLoginDone = ajax._post.pbind(url, q, no);
        window.onReLoginFailed = function(toRoot) {
          if (toRoot === -1) {
            location.href = location.href.replace(/^http:/, 'https:');
          } else if (toRoot) {
            nav.go('/');
          } else {
            window.onReLoginDone();
          }
        }

        utilsNode.appendChild(ce('iframe', {src: vk.loginscheme + '://login.vk.com/?' + ajx2q({role: 'al_frame', _origin: locProtocol + '//' + locHost, ip_h: (answer[0] || vk.ip_h)})}));
        break;
      case 4: // redirect
        if (intval(answer[1])) { // ajax layout redirect
          nav.go(answer[0], false, {nocur: (answer[1] === '2'), noback: (answer[1] === true) ? true : false, showProgress: o.showProgress, hideProgress: o.hideProgress});
        } else {
          hab.stop();
          location.href = answer[0];
        }
        break;
      case 5: // reload
        nav.reload({force: intval(answer[0]), from: 1, url: url, query: q && ajx2q(q)}); // force reload
        break;
      case 6: // mobile activation needed
        var no = o.cache ? extend(o, {cache: -1}) : o;
        o._box = activateMobileBox({onDone: ajax._post.pbind(url, q, no), onFail: o.onFail, hash: answer[0]});
        break;
      case 7: // message
        if (o.onFail) o.onFail();
        topMsg(answer[0], 10);
        break;
      case 8: // error
        if (o.onFail) {
          if (o.onFail(answer[0])) {
            return;
          }
        }
        topError(answer[0] + (answer[2] ? ' #'+answer[2] : ''), {dt: answer[1] ? 0 : 10, type: 4, url: url, query: q && ajx2q(q)});
        break;
      case 9: // votes payment
        if (o.fromBox) {
          if (o.onDone) { // page, box or other
            o.onDone.apply(window, answer);
          }
          break;
        }
        o._box = showFastBox(answer[0], answer[1]);
        var no = extend(clone(o), {showProgress: o._box.showProgress, hideProgress: o._box.hideProgress});
        if (o.cache) {
          no.cache = -1;
        }
        o._box = requestBox(o._box, function() {
          if (isVisible(o._box.progress)) return;
          ajax._post(url, extend(q, {_votes_ok: 1}), no);
        }, o.onFail);
        o._box.evalBox(answer[2]);
        break;
      case 10: //zero zone
        o._box = showFastBox({
          title: answer[0] || getLang('global_charged_zone_title'),
          onHide: o.onFail
        }, answer[1], getLang('global_charged_zone_continue'), function() {
          var nq = extend(q, {charged_confirm: answer[3]});
          ajax._post(url, nq, o);
        }, getLang('global_cancel'));
        break;
      default:
        if (code == -1) {
          var adsCanShow = answer.pop();
          var adsHtml    = answer.pop();
          __adsSet(adsHtml, null, adsCanShow);
        }
        if (o.onDone) { // page, box or other
          o.onDone.apply(window, answer);
        }
        break;
      }
      if (window._updateDebug) _updateDebug();
    }
    var done = function(text, data) { // data - for iframe transport post
      if (o.bench) {
        ajax.tDone = new Date().getTime();
      }
      text = text.replace(/^<!--/, '').replace(/-<>->/g, '-->');
      if (!trim(text).length) {
        data = [8, getLang('global_unknown_error')];
        text = stVersions['nav'] + '<!><!>' + vk.lang + '<!>' + stVersions['lang'] + '<!>8<!>' + data[1];
      }
      var answer = text.split('<!>');

      var navVersion = intval(answer.shift());
      if (!navVersion) {
        return fail('<pre>' + text + '</pre>', {status: -1});
      }

      // First strict check for index.php reloading, in vk.al == 1 mode.
      if (vk.version && vk.version != navVersion) {
        if (navVersion && answer.length > 4) {
          nav.reload({force: true, from: 2, url: url, query: q && ajx2q(q)});
        } else {
          if (nav.strLoc) {
            location.replace(locBase);
          } else {
            topError('Server error.', {type: 100});
          }
        }
        return;
      }
      vk.version = false;

      // Common response fields
      var newStatic = answer.shift();
      var langId = intval(answer.shift());
      var langVer = intval(answer.shift());

      if (o.frame) answer = data;

      var code = intval(answer.shift());
      if (vk.lang != langId && o.canReload) { // Lang changed
        nav.reload({force: true, from: 3, url: url, query: q && ajx2q(q)});
        return;
      }

      // Wait for attached static files
      var waitResponseStatic = function() {
        var st = ['common.css'];
        if (browser.msie6) {
          st.push('ie6.css');
        } else if (browser.msie7) {
          st.push('ie7.css');
        }
        if (newStatic) {
          newStatic = newStatic.split(',');
          for (var i = 0, l = newStatic.length; i < l; ++i) {
            st.push(newStatic[i]);
          }
        }
        if (stVersions['lang'] < langVer) {
          stVersions['lang'] = langVer;
          for (var i in StaticFiles) {
            if (/^langd/i.test(i)) {
              st.push(i);
            }
          }
        }

        if (!o.frame) {
          try {
            ajax._parseRes(answer);
          } catch(e) {
            topError('<b>JSON Error:</b> ' + e.message, {type: 5, answer: answer.join('<!>'), url: url, query: q && ajx2q(q)});
          }
        }
        stManager.add(st, processResponse.pbind(code, answer));
      }

      // Static managing function
      if (navVersion <= stVersions['nav']) {
        return waitResponseStatic();
      }
      headNode.appendChild(ce('script', {
        type: 'text/javascript',
        src: '/js/loader_nav' + navVersion + '_' + vk.lang + '.js'
      }));
      setTimeout(function() {
        if (navVersion <= stVersions['nav']) {
          return waitResponseStatic();
        }
        setTimeout(arguments.callee, 100);
      }, 0);
    }
    if (o.cache > 0 || o.forceGlobalCache) {
      var answer = ajaxCache[cacheKey];
      if (answer && answer._loading) {
        answer._callbacks.push(processResponse);
        return;
      } else {
        if (answer && !o.forceGlobalCache) {
          processResponse(0, answer);
          if (o.cache === 3) delete ajaxCache[cacheKey];
          return;
        } else if (answer = globalAjaxCache[cacheKey]) {
          if (answer == -1 || isFunction(answer)) {
            globalAjaxCache[cacheKey] = o.onDone;
          } else {
            o.onDone.apply(window, answer);
          }
          if (o.hideProgress) o.hideProgress();
          return;
        }
      }

    }
    ajaxCache[cacheKey] = {_loading: 1, _callbacks: []};
    debugLogHist(url + (q ? ': ' + ajx2q(q).split('&').join('&amp;') : ''));
    return o.frame ? ajax.framepost(url, q, done) : ajax.plainpost(url, q, done, fail);
  },
  tGetParam: function() {
    if (!ajax.tStart || !ajax.tModule) return;
    var d = ajax.tDone - ajax.tStart;
    var p = ajax.tProcess - ajax.tDone;
    var r = ajax.tRender - ajax.tProcess;
    var o = ajax.tOver - ajax.tStart;
    var res = [d, p, r, o, ajax.tModule];
    for (var i in res) {
      if (res[i] < 0) return false;
    }
    ajax.tStart = false;
    return res.join(',');
  }
}

// Кроссбраузерная работа с url. 
// Изменение, установка url, и получение установленного url.
function HistoryAndBookmarks(params) {
  // strict check for cool hash display in ff.
  var fixEncode = function(loc) {
    var h = loc.split('#');
    var l = h[0].split('?');
    return l[0] + (l[1] ? ('?' + ajx2q(q2ajx(l[1]))) : '') + (h[1] ? ('#' + h[1]) : '');
  }

  var frame = null, withFrame = browser.msie6 || browser.msie7;
  var frameDoc = function() {
    return frame.contentDocument || (frame.contentWindow ? frame.contentWindow.document : frame.document);
  }

  var options = extend({onLocChange: function() {}}, params);

  var getLoc = function(skipFrame) {
    var loc = '';
    if (vk.al == 3) {
      loc = (location.pathname || '') + (location.search || '') + (location.hash || '');
    } else {
      if (withFrame && !skipFrame) {
        try {
          loc = frameDoc().getElementById('loc').innerHTML.replace(/&lt;/ig, '<').replace(/&gt;/ig, '>').replace(/&quot;/ig, '"').replace(/&amp;/ig, '&');
        } catch(e) {
          loc = curLoc;
        }
      } else {
        loc = (location.toString().match(/#(.*)/) || {})[1] || '';
        if (loc.substr(0, 1) != vk.navPrefix) {
          loc = (location.pathname || '') + (location.search || '') + (location.hash || '');
        }
      }
    }
    if (!loc && vk.al > 1) {
      loc = (location.pathname || '') + (location.search || '');
    }
    return fixEncode(loc.replace(/^(/|!)/, ''));
  }

  var curLoc = getLoc(true);

  var setFrameContent = function(loc) {
    try {
      var d = frameDoc();
      d.open();

      d.write('<script type="text/javascript">var u=navigator.userAgent,d=location.host.toString().match(/[a-zA-Z]+\.[a-zA-Z]+$/)[0];if(/opera/i.test(u)||!/msie 6/i.test(u)||document.domain!=d)document.domain=d;</script>' +
        '<div id="loc">' +
          loc.replace('&', '&amp;').replace('"', '&quot;').replace('>', '&gt;').replace('<', '&lt;') +
        '</div>'
      );

      d.close();
    } catch(e) {}
  }

  var setLoc = function(loc) {
    //curLoc = fixEncode(loc.replace(/#(/|!)?/, ''));
    curLoc = fixEncode(loc);
    var l = (location.toString().match(/#(.*)/) || {})[1] || '';
    if (!l && vk.al > 1) {
      l = (location.pathname || '') + (location.search || '');
    }
    l = fixEncode(l);
    if (l.replace(/^(/|!)/, '') != curLoc) {
      if (vk.al == 3) {
        try {
          history.pushState({}, '', '/' + curLoc);
          return;
        } catch(e) {}
      }
      window.chHashFlag = true;
      location.hash = '#' + vk.navPrefix + curLoc;
      if (withFrame && getLoc() != curLoc) {
        setFrameContent(curLoc);
      }
    }
  }

  var locChecker = function() {
    var loc = getLoc(true);
    if (loc != curLoc) {
      if (browser.msie6) {
        if (reloadCheckFlood({force: true, from: 6})) return;
        location.reload(true);
      } else {
        setFrameContent(loc);
      }
    }
  }

  var checker = function(force) {
    var l = getLoc();
    if (l == curLoc && force !== true) {
      return;
    }
    options.onLocChange(l);

    curLoc = l;
    if (withFrame && location.hash.replace('#' + vk.navPrefix, '') != l) {
      location.hash = '#' + vk.navPrefix + l;
    }
  }
  var checkTimer;
  var frameChecker = function() {
    try {
      if (frame.contentWindow.document.readyState != 'complete') {
        return;
      }
    } catch(e) {
      return;
    }
    checker();
  }
  var init = function() {
    if (vk.al == 1) {
      checker(true);
    }
    if (vk.al == 3) {
      addEvent(window, 'popstate', checker);
      if (browser.safari) {
        addEvent(window, 'hashchange', checker);
      }
    } else if (withFrame) {
      frame = ce('iframe', {id: 'hab_frame'});
      frame.attachEvent('onreadystatechange', frameChecker);
      frame.src = '/al_loader.php?act=hab_frame&loc=' + encodeURIComponent(curLoc);

      utilsNode.appendChild(frame);

      checkTimer = setInterval(locChecker, 200);
    } else if ('onhashchange' in window) {
      addEvent(window, 'hashchange', function() {
        if (window.chHashFlag) {
          window.chHashFlag = false;
        } else {
          checker();
        }
      });
    } else {
      checkTimer = setInterval(checker, 200);
    }
  }
  return {
    setLoc: setLoc,
    getLoc: getLoc,
    init: init,
    setOptions: function(params) {
      options = extend(options, params);
    },
    checker: checker,
    stop: function() {
      if (vk.al < 3) {
        clearInterval(checkTimer);
        if (withFrame) {
          frame.detachEvent('onreadystatechange', frameChecker);
        }
      } else if (vk.al == 3) {
        removeEvent(window, 'popstate', checker);
      }
    }
  }
}

// Инициализация и определения обработчика на изменения url.
window.hab = new HistoryAndBookmarks({onLocChange: function(loc) {
  // hist - скрытие всех лейеров. 
  // back - для вывода кэша страницы из globalHistory.  
  nav.go('/' + loc, undefined, {back: true, hist: true});
}});

// Проверка события на клик с зажатой клавишей. 
function checkEvent(e) {
  return ((e = (e || window.event)) && (e.type == 'click' || e.type == 'mousedown' || e.type == 'mouseup') && (e.which > 1 || e.button > 1 || e.ctrlKey || e.shiftKey || browser.mac && e.metaKey));
}
function checkOver(e, target) {
  if (!e) return true;
  e = e.originalEvent || e;
  target = target || e.target;
  var related = e.fromElement || e.relatedTarget;
  if (!related || related == target || related == target.parentNode) {
    return true;
  }
  while (related != target && related.parentNode && related.parentNode != bodyNode) {
    related = related.parentNode;
  }
  return (related != target);
}

// onmouseover на блок с напоминаниями.
function leftBlockOver(block) {
  var timer = 'timer'
  var over = 1;
  if (!block.id) {
    block = ge('left_hide' + block);
    over = 0;
  }
  if (over || !block.timer) {
    if (block.showing) {
      removeAttr(block, 'showing');
    } else {
      animate(block, {opacity: over ? 1 : 0.5}, 200);
      if (over) {
        block.showing = 1;
      }
    }
  }
  if (block.timer) {
    clearTimeout(block.timer);
    removeAttr(block, 'timer');
  }
}

// onmouseout на блок с напоминаниями.
function leftBlockOut(block) {
  var opacity = 0.5;
  if (!block.id) {
    block = ge('left_hide' + block);
    opacity = 0;
  }
  block.timer = setTimeout(function() {
    animate(block, {opacity: opacity}, 200);
    removeAttr(block, 'timer');
  }, 1);
}

// Скрыть блок с напоминанием.
function leftBlockHide(block, hash, blockData) {
  var query = {act: 'hide_block', block: block, hash: hash};
  if (blockData) {
    query.block = blockData;
  }
  ajax.post('al_index.php', query, {onDone: updSeenAdsInfo});
  hide('left_block' + block);
}

// Скрыть блок с рекламой.
function leftAdBlockClose(blockIdSuffix, closeLink) {

  setStyle('left_hide' + blockIdSuffix, {visibility: 'hidden'});

  ajax.post(closeLink, {}, {noAds: true, onDone: onDone});

  function onDone(response) {
    if (!response.done) return;

    var elemInfo  = ge('ads_ad_close_info_' + blockIdSuffix);

    if (!elemInfo) return false;

    setStyle(elemInfo, {display: 'block', opacity: 0})

    setTimeout(onClose, 0);
  }
  function onClose() {
    animate('ads_ad_close_info_' + blockIdSuffix, {opacity: 1}, 200, onCloseComplete);
  }
  function onCloseComplete() {
    setStyle('ads_ad_box2_' + blockIdSuffix, {visibility: 'hidden'});
  }
}

// Старые функции для работы с друзьями. Сейчас все в profile.js 
function leftBlockFriendHide(blockId, blockData, hash) {
  var container = ge('left_friends');
  if (!container) { return; }
  var blocks = geByClass('left_friend_block', container);
  var showedIds = [];
  for (var i in blocks) {
    var t = blocks[i].id.split('_');
    showedIds.push(t[2]);
  }
  var query = {act: 'hide_block', block: blockData, showed: showedIds.join(','), hash: hash};
  ajax.post('al_index.php', query, {onDone: function(html) {
    if (html) {
      var block = se(html);
      if (geByClass('left_friend_block', container).length >= 2) {
        hide(block);
      }
      container.insertBefore(block, geByClass1('left_friend_all_link', container));
    }
    if (!geByClass('left_friend_block', container).length) {
      hide('left_friend_all_link');
    }
  }});

  var hideBl = ge('left_block' + blockId);
  var newBl = blocks.pop();
  if (newBl && !isVisible(newBl)) {
    show(newBl)
    hideBl.parentNode.replaceChild(newBl, hideBl);
  } else {
    re(hideBl);
  }
}
function leftBlockToggleFriend(mid, type, hash, act, ev) {
  if (act) {
    stManager.add(['tooltips.css', 'tooltips.js']);
  }
  cur.mfid = mid;

  ajax.post('al_friends.php', {act: act ? 'add' : 'remove', mid: mid, mf_type: type, hash: hash, from: 'leftblock'}, {
    onDone: function(text, ttText, ttScript) {
      if (!text) return nav.reload();
      var fs = ge('left_friend_status_'+mid);
      cleanElems(fs.firstChild);
      if (text) {
        show(fs);
        val(fs, text);
      } else {
        hide(fs);
      }
      if (ttText) {
        ajax.preload('al_friends.php', {act: 'friend_tt', mid: mid}, [ttText, ttScript])
        setTimeout(leftBlockFriendTooltip, 0);
      }
    },
    showProgress: function() {
      var tt = (ge('left_friend_subscribed') || {}).tt;
      if (tt) {
        tt.hide({fasthide: 1});
        tt.destroy();
      }
      ge('left_friend_status_'+mid).innerHTML = '<img src="/images/upload'+(window.devicePixelRatio >= 2 ? '_2x' : '')+'.gif" />';
    },
    hideProgress: function() { hide('left_friend_status_'+mid); },
    onFail: function(text) {
      if (!text) return;

      showFastBox(getLang('global_error'), text);
      return true;
    }
  });
  cancelEvent(ev);
}
function leftBlockFriendTooltip() {
  return showTooltip(ge('left_friend_subscribed'), {
    url: 'al_friends.php',
    params: {act: 'friend_tt', mid: cur.mfid, from: 'leftblock'},
    slide: 15,
    hidedt: 500,
    shift: [20, -1, 3],
    className: 'preq_tt',
    forcetodown: true
  });
}

// Сторонние сервисы сбора статистики.
function updateOtherCounters(url, referrer) {
  if (vk.zero) return;
  referrer = referrer || document.referrer;
  vkImage().src = locProtocol + '//counter.yadro.ru/hit?r' + escape(referrer) + (window.screen === undefined ? '' : ';s' + screen.width + '*' + screen.height + '*' + (screen.colorDepth ? screen.colorDepth : screen.pixelDepth)) + ';u' + escape(url) + ';' + Math.random() + '';
  vkImage().src = locProtocol + '//' + (locProtocol == 'https:' ? 'sb' : 'b') + '.scorecardresearch.com/p?c1=2&c2=13765216&c3=&c4=' + escape(url) + '&c5=&c9=' + escape(referrer) + 'c15=&cv=2.0&cj=1&rn=' + Math.random();
}

// Установка размера страницы и параметров отображения.
function handlePageView(params) {
  var footer = ge('footer_wrap');

  var width          = (params.width === undefined) ? vk.width : params.width;
  var widthDec       = (params.width_dec === undefined) ? vk.width_dec : params.width_dec;
  var widthDecFooter = (params.width_dec_footer === undefined) ? vk.width_dec_footer : params.width_dec_footer;

  if (vk.noleftmenu != params.noleftmenu || vk.nobottommenu != params.nobottommenu || vk.width != params.width || vk.width_dec_footer != params.width_dec_footer) {
    if (vk.noleftmenu != params.noleftmenu && params.noleftmenu) {
      hide('side_bar');
    }
    if (vk.nobottommenu != params.nobottommenu) {
      if (params.nobottommenu) {
        hide('bottom_nav');
      } else {
        show('bottom_nav');
      }
    }
    if (params.noleftmenu && params.nobottommenu) {
      ge('page_body').className = 'simple';
      if (footer) {
        footer.className = 'simple';
        footer.style.width = 'auto';
      }
    } else {
      ge('page_body').className = 'fl_r';
      if (footer) {
        footer.className = 'fl_r';
        footer.style.width = (width - widthDecFooter) + 'px';
      }
    }
  }

  if (vk.notopmenu != params.notopmenu) {
    if (params.notopmenu) {
      hide('quick_search', 'qsearch_border', 'top_search', 'top_invite_link', 'top_menu_wrap');
    } else {
      show('quick_search', 'qsearch_border', 'top_search', 'top_invite_link', 'top_menu_wrap');
    }
  }

  if (width != vk.width || widthDec != vk.width_dec) {
    ge('page_layout').style.width = width + 'px';
    ge('page_body').style.width = (width - widthDec) + 'px';
    if (ge('ts_wrap') && hasClass(ge('ts_wrap'), 'vk')) ge('ts_wrap').style.width = (width - 191) + 'px';
    setTimeout(updSideTopLink.pbind(true), 0);

    _tbLink.style.maxWidth = (_tbLink.parentNode.offsetWidth - 35) + 'px';
  }

  if (vk.noleftmenu != params.noleftmenu && !params.noleftmenu) {
    show('side_bar');
  }

  vk.noleftmenu       = params.noleftmenu;
  vk.nobottommenu     = params.nobottommenu;
  vk.notopmenu        = params.notopmenu;
  vk.width            = width;
  vk.width_dec        = widthDec;
  vk.width_dec_footer = widthDecFooter;

  vk.no_ads           = params.no_ads;
  vk.ad_preview       = params.ad_preview;
}

// Установка счетчиков в меню.
function handleSetCount(e, v, id) {
  var newCnt = '+', attr = '', maxCntLen = (id == 'spr') ? 5 : ((vk.fs > 11) ? 2 : 3),
      plus = '+',
      newCntWrap = '<span class="inl_bl left_count">+</span>',
      curCntWp = geByClass1('left_count_wrap', e, 'span'),
      curOver = hasClass(geByClass1('left_row', e, 'a'), 'left_nav_over'),
      curCntEl = geByClass1('left_count', e, 'span'), curCntV = val(curCntEl);
  if (v) {
    newCnt = v.toString();
    if (newCnt.length > maxCntLen) {
      attr = ' title="' + stripHTML(langNumeric(v, '%s', true)) + '"';
      newCnt = '..' + newCnt.substr(newCnt.length - maxCntLen);
      plus = '';
    } else if (v <= 0) {
      plus = '';
    }
    newCnt = plus + newCnt;
    newCntWrap = '<span class="inl_bl left_count" ' + attr + '>' + newCnt + '</span>';
  }
  var onDone = function () {
    val(curCntWp, newCntWrap);
    (!v ? addClass : removeClass)(curCntWp, 'left_void');
    setStyle(curCntWp, {opacity: ''});
  }, el = domFC(_fMenuLnks[id]);
  if (id && newCnt != val(curCntEl) && (newCnt || curCntEl)) {
    Pads.invalidate(id);
  }
  if (curCntV && (curCntV != '+' || curOver)) {
    if (newCnt && newCnt != '+') {
      animateCount(curCntEl, newCnt, {str: 'auto', onDone: onDone});
      if (el) animateCount(el, newCnt + '&nbsp;', {str: 'auto'});
    } else if (curOver) {
      var fakeEl = bodyNode.appendChild(se('<span class="left_count_wrap"><span class="inl_bl left_count">+</span></span>'));
      plusWidth = getSize(domFC(fakeEl))[0];
      re(fakeEl);
      if (curCntV.charAt(0) != '+') val(curCntEl, curCntV.replace('..', '+'));
      animate(curCntEl, {width: plusWidth}, 100, onDone);
      if (el) animate(el, {width: 0, discrete: 1}, 100, function() { val(el, el.style.width = ''); });
    } else {
      animate(curCntWp, {opacity: 0}, 100, onDone);
      if (el) animate(el, {width: 0, discrete: 1}, 100, function() { val(el, el.style.width = ''); });
    }
  } else {
    onDone();
    setStyle(curCntWp, {opacity: 0});
    animate(curCntWp, {opacity: 1}, 100);
    if (el && newCnt && newCnt != '+') {
      val(el, newCnt + '&nbsp');
      var to = el.offsetWidth;
      el.style.width = '1px';
      animate(el, {width: to, discrete: 1}, 100, function() { el.style.width = ''; });
    }
  }
}

// Подготовка. 
// Сохранение параметров, установка размеров страницы и установка счетчиков меню.
vk.counts = {};  
function handlePageParams(params) {
  vk.id = params.id;
  if (params.pads !== vk.pads) {
    ((vk.pads = params.pads) ? addClass : removeClass)(bodyNode, 'pads');
  }
  if (params.pvbig !== undefined) vk.pvbig = params.pvbig;
  if (params.pvdark !== undefined) vk.pvdark = params.pvdark;
  cur._level = params.level;

  handlePageView(params);

  var confMail = ge('confirm_mail_wrap'), chPhone = ge('change_phone_wrap'), mv = ge('mvk_footer_lnk');
  if (mv) {
    mv.firstChild.href = params.mvklnk || 'http://m.vk.com/';
  }
  if (confMail) {
    if (!params.nomail) {
      confMail.parentNode.removeChild(confMail);
    } else if (params.notopbars) {
      hide(confMail);
    } else {
      show(confMail);
    }
  }
  vk.nomail = params.nomail;
  if (chPhone) {
    if (!params.chphone) {
      re(chPhone);
    } else if (params.notopbars) {
      hide(chPhone);
    } else {
      show(chPhone);
    }
  }

  vk.nophone = intval(params.nophone);

  if (vk.id) {
    var leftBlocksElem = ge('left_blocks');
    if (leftBlocksElem) {
      leftBlocksElem.innerHTML = (params.leftblocks || '');
    }
  }
  if ('leftads' in params && params.leftads !== 0) {
    __adsSet(params.leftads, params.ads_section || '', params.ads_can_show);
  }

  var currentURL = locProtocol+'//'+location.host+'/'+params.loc;
  var referrer = (document.URL == currentURL) ? '' : document.URL;
  setTimeout(updateOtherCounters.pbind(currentURL, referrer), 10);

  if (!params.counters) return;
  var cnts = (params.counters || '').split(',');

  var i = 0, setEl = ge('l_set'), sep = setEl && setEl.nextSibling || false, sh = false, ids = ['fr', 'ph', 'vid', 'msg', 'nts', 'gr', 'ev', 'wsh', 'ap', 'ads', 'nws', 'wk', 'docs'];
  var lnks = ['friends', 'albums' + vk.id, 'video', '', 'notes', 'groups', 'events', 'gifts.php?act=wishlist', 'apps', 'ads?act=office', 'feed', 'pages', 'docs'];
  var adds = ['', 'act=added', 'section=tagged', '', 'act=comments', '', 'tab=invitations', '', '', '', 'section=notifications'];

  // Установка счетчиков в vk.counts.
  if (!params.handlecnts) {
    for (; i < 13; ++i) {
      vk.counts[ids[i]] = cnts[i];
    }
    return;
  }

  for (; i < 13; ++i) {
    handlePageCount(ids[i], cnts[i], lnks[i], adds[i]);
  }

  // Установка счетчиков приложениям. 

  // Перебор элементов после сепаратора. 
  for (var e = sep.nextSibling; e; e = e.nextSibling) {
    if (e.tagName && e.tagName.toLowerCase() == 'li' && isVisible(e)) {
      sh = true;
      break;
    }
  }
  // Если есть елементы после сепаратора то показать сепаратор, иначе скрыть. 
  (sh ? show : hide)(sep);
  for (var l = cnts.length; i < l; ++i) {
    var id_v = cnts[i].split(':'), e = ge('l_app' + intval(id_v[0]));
    if (!e) continue;

    handleSetCount(e, intval(id_v[1]));
  }
  setTimeout(updSeenAdsInfo, 0);
}

// Изменения счетчиков в меню. 
function handlePageCount(id, value, lnk, add) {
  var v = intval(value);
  if (vk.counts === undefined) vk.counts = {};
  if (vk.counts[id] === v) return;
  vk.counts[id] = v;

  var e = ge('l_' + id), toAdd, hc = hasClass(domFC(e), 'left_nav_over');
  if (e) {
    handleSetCount(e, (v > 0) ? v : 0, id);

    if (add && lnk) {
      toAdd = (v > 0 && add) ? ('?' + add) : '';
      e.firstChild.href = '/' + lnk + toAdd;
    }
  }
  if (v >= 0 || !hc) {
    toggle(e, v >= 0);
    toggle(_fMenuLnks[id], !!((v > 0) || (v == 0) && geByClass1('left_count_persist', e) || hc));
  }
}

// Выполнение hide и destroy функций у cur._back 
function processDestroy(c) {
  if (c._back && c._back.hide && c == cur) {
    for (var i in c._back.hide) {
      try {
        c._back.hide[i]();
      } catch(e) {
        try { console.log(e.stack); } catch (e2) {}
      }
    }
  }
  if (!c.destroy || !c.destroy.length) return;
  for (var i in c.destroy) {
    try {
      c.destroy[i](c);
    } catch(e) {
      try { console.log(e.stack); } catch (e2) {}
    }
  }
}

// Удаление объекта из globalHistory если текущ. url совпадает с тем, который в истории. 
var globalHistory = [];
function globalHistoryDestroy(loc) {
  for (var i = 0, l = globalHistory.length; i < l; ++i) {
    if (globalHistory[i].loc == loc) {
      var h = globalHistory.splice(i, 1)[0];
      processDestroy(h.cur);
      h.content.innerHTML = '';
      --i; --l;
    }
  }
}

// Показывает или скрывает кнопку назад.  
function showBackLink(loc, text, fast) {
  var l = loc;
  loc = (loc || '').replace(/^//, '');
  _tbLink.loc = loc;
  if (fast === undefined) {
    fast = 0;
    if (loc) {
      for (var i = 0, len = globalHistory.length; i < len; ++i) {
        if (globalHistory[i].loc == loc) {
          fast = 1;
          break;
        }
      }
    }
  }
  if (l) {
    _tbLink.style.maxWidth = (_tbLink.parentNode.offsetWidth - 35) + 'px';
    extend(_tbLink, {href: '/' + loc, innerHTML: text, fast: fast});
    show(_tbLink);
    _stlWas = 0;
  } else {
    hide(_tbLink);
  }
  if (!hasClass(ge('ts_wrap'), 'vk')) TopSearch.toggleTsWrap(!l);
  updSideTopLink(1);
}

// Проверка на частоту перезагрузки страниц, 
// если кол-во перезагрузки более 5 раз и менее 20 сек. 
// то выдаст ошибку и вернет true иначе false
function reloadCheckFlood(opts) {
  opts = opts || {};
  var reloaded = ls.get('last_reloaded') || [],
      reloadedLen = 1,
      reloadLimit = 5,
      reloadStop;

  reloaded.unshift(vkNow());
  if ((reloadedLen = reloaded.length) > reloadLimit) {
    reloaded.splice(reloadLimit, reloadedLen - reloadLimit);
    reloadedLen = reloadLimit;
  }
  ls.set('last_reloaded', reloaded);
  reloadStop = (reloadedLen == reloadLimit && reloaded[0] - reloaded[reloadLimit - 1] < 20000);

  debugLog('reload check flood', reloadStop, reloaded, reloadedLen, reloadLimit, reloaded[0] - reloaded[reloadLimit - 1]);
  if (reloadStop) {
    topError(
      '<b>Reloading error</b>, please check internet connection, <b><a href="/page-777107_43991681">clear your cache</a></b> and restart your browser.<br>If problem remains, please <a href="/support?act=new">report it here</a>.',
      {dt: 15, type: 6, msg: 'Reload error, from ' + (opts.from || 0) + ', forced ' + (opts.force || 0) + ', url ' + (opts.url || '') + ', query ' + (opts.query || '') }
    );
    return true;
  }
  return false;
}

var nav = {
  getData: function(loc) { // Получение данных роутинга. 
    if (loc.length) {
      for (var i in navMap) {
        if (i[0] == '<') continue;
        var m = loc.match(new RegExp('^' + i, 'i'));
        if (m) {
          return {url: navMap[i][0], files: navMap[i][1]};
        }
      }
      var m = loc.match(/^[a-z0-9-_]+.php$/i);
      if (m) {
        return {url: loc};
      }
      return {url: navMap['<other>'][0], files: navMap['<other>'][1]};
    }
    return {url: navMap['<void>'][0], files: navMap['<void>'][1]};


  },
  reload: function(opts) {
    if (reloadCheckFlood(opts)) return; // Проверка на частоту перезагрузок. 
    opts = opts || {};
    if (opts.force) { // Полная перезагрузка. 
      hab.stop();
      location.href = '/' + nav.strLoc;
    } else { // Перезагрузка контента без (перехвата) cur.nav и zNav. 
      nav.go('/' + nav.strLoc, undefined, extend({nocur: true}, opts));
    }
  },
  go: function(loc, ev, opts) {
    if (checkEvent(ev)) return; // Не выполнять, если был не одиночный клик.

    // Подготовка. (дэкод - энкод текущ. url).
    if (loc.tagName && loc.tagName.toLowerCase() == 'a' && loc.href) {
      if (loc.target == '_blank' || nav.baseBlank) {
        return;
      }
      loc = loc.href;
    }
    var strLoc = '', objLoc = {}, changed = {};
    if (typeof(loc) == 'string') {
      loc = loc.replace(new RegExp('^(' + locProtocol + '//' + locHost + ')?/?', 'i'), '');
      strLoc = loc;
      objLoc = nav.fromStr(loc);
    } else {
      if (!loc[0]) loc[0] = '';
      strLoc = nav.toStr(loc);
      objLoc = loc;
    }

    // Аудио.
    var _a = window.audioPlayer, aid = currentAudioId();
    opts = opts || {};
    if (_a && aid && (_a.gpDisabled || !ge('gp')) && !opts.back) _a.stop();
    if (_a && (_a.player && _a.player.paused() || !_a.player)) {
      _a.gpHidden = true;
      if (!opts.back && window.audioPlaylist && ls.get('pad_pltime')) {
        delete window.audioPlaylist;
        if (_a.id) _a.id = null;
      }
    }

    if (!opts.nocur) {
      // Синхронизация текущ. url с уже установленным. 
      changed = clone(objLoc);
      for (var i in nav.objLoc) {
        if (nav.objLoc[i] == changed[i]) {
          delete(changed[i]);
        } else if (changed[i] === undefined) {
          changed[i] = false;
        }
      }
      //
      if (zNav(clone(changed), {hist: opts.hist}) === false) {
        nav.setLoc(strLoc);
        return false;
      }
    }

    // Выполнения текущих cur.nav функций (перехват запроса на текущ. стр.).
    if (!opts.nocur && (vk.loaded || !changed['0'])) {
      for (var i in (cur.nav || {})) {
        if (cur.nav[i](clone(changed), nav.objLoc, objLoc, opts) === false) {
          return false;
        }
      }
    }

    // Перезагрузка страницы.
    if (vk.al == 4 || (!vk.loaded && (!window.audioPlayer || !audioPlayer.player) && changed['0'])) {
      setTimeout(function() {
        location.href = '/' + (strLoc || '').replace('%23', '#');
      }, 0);
      return false;
    }

    if (opts.back) {
      // Выполнение onBack для перехвата вывода при нажатии назад. 
      if (cur._back && cur._back.onBack) {
        return cur._back.onBack();
      }
      // Вывод из истории если url в истории совпадает с переданным(текущ).
      for (var i = 0, l = globalHistory.length; i < l; ++i) {
        if (globalHistory[i].loc == strLoc) {
          var h = globalHistory.splice(i, 1)[0];
          var wNode = ge('wrap3'), tNode = ge('title');
          var onback = cur._onback;

          if (window.tooltips) tooltips.destroyAll();
          processDestroy(cur);
          radioBtns = h.radioBtns;
          ajaxCache = h.ajaxCache;
          boxQueue.hideAll();
          if (layers.fullhide) layers.fullhide(true);

          showBackLink();

          if (!hasClass(ge('ts_wrap'), 'vk')) TopSearch.toggleTsWrap(true);

          cur = h.cur;
          if (gSearch.on) gSearch.hide();
          setTimeout(function() {
            wNode.innerHTML = '';
            wNode.parentNode.replaceChild(h.content, wNode);
            if (vk.width != h.width) {
              handlePageView(h);
            }
            scrollToY(h.scrollTop, 0);
            document.title = h.htitle;
            tNode.innerHTML = h.title;
            setStyle(tNode.parentNode, 'display', h.hideHeader ? 'none' : 'block');
            for (var i = 0, l = cur._back.show.length; i < l; ++i) cur._back.show[i]();
            if (onback) {
              for (var i = 0, l = onback.length; i < l; ++i) onback[i]();
            }
            nav.setLoc(strLoc);
            var b = h.back || {};
            setTimeout(function() {
              showBackLink(b[0], b[1], b[2]);
              if (nav.objLoc.z || nav.objLoc.w) {
                zNav({z: nav.objLoc.z, w: nav.objLoc.w}, {});
              }
            }, 10);
            var _a = window.audioPlayer, aid = currentAudioId();
            if (_a && aid && _a.hideCurrentTrack && _a.showCurrentTrack) {
              _a.hideCurrentTrack();
              _a.showCurrentTrack();
            }
          }, 10);
          return false;
        }
      }
    }

    var dest = objLoc[0];
    delete(objLoc[0]);

    var where = nav.getData(dest);
    if (!opts.noframe) {
      opts.tstat = ajax.tGetParam();
      ajax.tStart = new Date().getTime();
      opts.bench = true;
    }
    if (!opts.params || !opts.params._ref) {
      opts.params = extend(opts.params || {}, {_ref: nav.objLoc[0] || ''});
    }
    if (where.files) {
      stManager.add(where.files);
    }
    where.params = extend({__query: dest, al_id: vk.id}, objLoc, opts.params || {});
    if (opts.cl_id) {
      where.params.fr_click = cur.oid+','+opts.cl_id+','+cur.options.fr_click;
    }
    if (opts.tstat) {
      where.params._tstat = opts.tstat;
    }
    if (opts.permanent) {
      where.params._permanent = opts.permanent;
    }
    var done = function(title, html, js, params) {
      try {
        params._id = params.id;
      } catch(e) {
        return topError(e, {dt: 15, type: 6, msg: 'Error: ' + e.message + ', (params undefined?), title: ' + title + ', html: ' + html + ', js: ' + js, url: where.url, query: ajx2q(where.params), answer: arguments.length});
      }

      if (opts.bench) {
        ajax.tProcess = new Date().getTime();
      }

      if (stVersions['common.js'] > StaticFiles['common.js'].v) {
        nav.setLoc(params.loc || nav.strLoc);
        if (reloadCheckFlood({force: true, from: 4})) return;
        location.reload(true);
        return;
      }
      var newPage = (where.params.al_id === undefined) || (where.params.al_id != params.id) || params.fullPage;
      var _back = cur._back, wNode = ge('wrap3'), tNode = ge('title'), hist = false;
      if (strLoc == (cur._back || {}).loc || newPage || opts.back) {
        _back = false;
      }
      if (opts.noback || params.level && (!cur._level || params.level <= cur._level) && opts.noback !== false) {
        _back = false;
        if (opts.noback || (cur._level && params.level < cur._level)) {
          showBackLink();
        }
      }
      if (window.tooltips) tooltips.destroyAll();
      if (gSearch.on) gSearch.hide();
      if (_back) {
        revertLastInlineVideo();
        hist = {
          loc: _back.loc || nav.strLoc,
          cur: cur,
          radioBtns: radioBtns,
          ajaxCache: ajaxCache,
          scrollTop: scrollGetY(),
          htitle: document.title.toString(),
          width: vk.width,
          width_dec: vk.width_dec,
          width_dec_footer: vk.width_dec_footer,
          noleftmenu: vk.noleftmenu,
          notopmenu: vk.notopmenu,
          nobottommenu: vk.nobottommenu,
          back: _tbLink.loc ? [_tbLink.loc, val(_tbLink), _tbLink.fast] : false
        };
        if (tNode && tNode.parentNode && !isVisible(tNode.parentNode)) {
          hist.hideHeader = true;
        }
        globalHistoryDestroy(hist.loc);
        if (globalHistory.length > 2) {
          var h = globalHistory.shift();
          processDestroy(h.cur);
          h.content.innerHTML = '';
        }
        for (var i = 0, l = cur._back.hide.length; i < l; ++i) cur._back.hide[i]();
        showBackLink(hist.loc, _back.text, 1);
      } else {
        _tbLink.fast = 0;
        processDestroy(cur);
      }
      radioBtns = {};
      ajaxCache = {};
      boxQueue.hideAll();
      if (layers.fullhide) layers.fullhide(true);

      cur = {destroy: [], nav: []};
      _stlWas = 0;
      if (newPage) {
        cleanElems('quick_login_button', 'quick_expire', 'search_form', 'top_links', 'bottom_nav')
        while (globalHistory.length) {
          var h = globalHistory.shift();
          processDestroy(h.cur);
          h.content.innerHTML = '';
        }
        pageNode.innerHTML = html;
        _tbLink = ge('top_back_link');
        _tbLink.style.maxWidth = (_tbLink.parentNode.offsetWidth - 35) + 'px';
        if (!browser.mobile) onBodyResize(true);
      } else {
        if (_back) {
          var newW = ce('div', {id: 'wrap3'});
          extend(hist, {
            content: wNode.parentNode.replaceChild(newW, wNode),
            title: tNode.innerHTML
          });
          globalHistory.push(hist);
          wNode = newW;
        }
        wNode.innerHTML = html;
        tNode.innerHTML = title;
        (title ? show : hide)(tNode.parentNode);
      }

      // Сохранение парам., установка размера страницы и установка счетчиков в меню.
      handlePageParams(params);  

      if (newPage) {
        initFixedMenu();
      }

      if (!opts.noscroll && !params.noscroll) scrollToTop(0);

      if (opts.bench) {
        ajax.tRender = new Date().getTime();
      }

      nav.curLoc = params.loc;
      if (js) {
        eval('(function(){' + js + ';})()');
      }
      ajax._framenext();

      if (opts.onDone) opts.onDone();

      if (browser.mobile) onBodyResize();
      setTimeout(function() {
        nav.setLoc(params.loc || '');
        var _a = window.audioPlayer, aid = currentAudioId();
        if (_a && aid && _a.showCurrentTrack) _a.showCurrentTrack();
        if (!isVisible('ts_wrap') && hasClass(ge('ts_wrap'), 'vk')) TopSearch.toggleSettings();
        if (TopSearch && TopSearch.tsNeedsClear) {
          TopSearch.clear();
          TopSearch.toggleInput(false);
          delete TopSearch.tsNeedsClear;
        }
      }, browser.chrome ? 100 : 50);
    }
    updSeenAdsInfo();
    __adsLoaded = vkNow();
    if (_pads.shown) Pads.hide();
    ajax.post(where.url, where.params, {onDone: function() {
      var a = arguments;
      if (__debugMode) {
        done.apply(null, a);
      } else try {
        done.apply(null, a);
      } catch (e) {
        topError(e, {dt: 15, type: 6, url: where.url, query: ajx2q(where.params), js: a[2], answer: Array.prototype.slice.call(arguments).join('<!>')});
      }
    }, onFail: opts.onFail || function(text) {
      if (!text) return;

      setTimeout(showFastBox(getLang('global_error'), text).hide, __debugMode ? 30000 : 3000);
      return true;
    }, frame: opts.noframe ? 0 : 1, canReload: true, showProgress: opts.showProgress || showTitleProgress, hideProgress: opts.hideProgress || hideTitleProgress, cache: opts.search ? 1 : '', bench: opts.bench});
    //setTimeout(nav.showPreload.pbind(ev, where), 0);
    return false;
  },
  setLoc: function(loc) { // Установка url.
    if (typeof(loc) == 'string') {
      nav.strLoc = loc;
      nav.objLoc = nav.fromStr(loc);
    } else {
      nav.strLoc = nav.toStr(loc);
      nav.objLoc = loc;
    }
    hab.setLoc(nav.strLoc);
  },
  change: function(loc, ev, opts) { // Запрос текущей страницы с переданными парам. 
    var params = clone(nav.objLoc);
    each(loc, function(i,v) {
      if (v === false) {
        delete params[i];
      } else {
        params[i] = v;
      }
    });
    return nav.go(params, ev, opts);
  },
  fromStr: function(str) { // Разбивает строку запроса на объект.
    str = str.split('#');
    var res = str[0].split('?');
    var param = {'0': res[0] || ''}
    if (str[1]) {
      param['#'] = str[1];
    }
    return extend(q2ajx(res[1] || ''), param);
  },
  toStr: function(obj) { // Объект запроса превращает в строку. 
    obj = clone(obj);
    var hash = obj['#'] || '';
    var res = obj[0] || '';
    delete(obj[0]);
    delete(obj['#']);
    var str = ajx2q(obj);
    return (str ? (res + '?' + str) : res) + (hash ? ('#' + hash) : '');
  },
  init: function() { // Получение и сохранение в поля объекта nav текущего url. 
    nav.strLoc = hab.getLoc();
    nav.objLoc = nav.fromStr(nav.strLoc);
  }
}

nav.init();

//
// Cookies
//

var _cookies;
function _initCookies() {
  _cookies = {};
  var ca = document.cookie.split(';');
  var re = /^[s]*([^s]+?)$/i;
  for (var i = 0, l = ca.length; i < l; i++) {
    var c = ca[i].split('=');
    if (c.length == 2) {
     _cookies[c[0].match(re)[1]] = unescape(c[1].match(re) ? c[1].match(re)[1] : '');
    }
  }
}
function getCookie(name) {
  _initCookies();
  return _cookies[name];
}
function setCookie(name, value, days, secure) {
  var expires = '';
  if (days) {
    var date = new Date();
    date.setTime(date.getTime()+(days*24*60*60*1000));
    expires = '; expires='+date.toGMTString();
  }
  var domain = locDomain;
  document.cookie = name + '='+escape(value) + expires + '; path=/' + (domain ? '; domain=.' + domain : '') + ((secure && locProtocol == 'https:') ? '; secure' : '');
}

//
// Time offset stuff
//
if (vk.time && !browser.opera_mobile) setTimeout(function() {
  var t = new Date(), time = [0, t.getMonth() + 1, t.getDate(), t.getHours(), t.getMinutes()];
  if (time[1] == 1 && vk.time[1] == 12) {
    vk.time[1] = 0;
  } else if (time[1] == 12 && vk.time[1] == 1) {
    time[1] = 0;
  } else if (time[1] > vk.time[1] + 1 || vk.time[1] > time[1] + 1) {
    time[1] = vk.time[1] = time[2] = vk.time[2] = 0;
  }
  if (time[1] > vk.time[1] && time[2] == 1) {
    if (vk.time[2] == 31 || (vk.time[1] == 4 || vk.time[1] == 6 || vk.time[1] == 9 || vk.time[1] == 11) && vk.time[2] == 30 || vk.time[1] == 2 && (vk.time[2] == 29 || vk.time[2] == 28 && (vk.time[0] % 4))) {
      vk.time[2] = 0;
    } else {
      vk.time[2] = time[2] = 0;
    }
  } else if (vk.time[1] > time[1] && vk.time[2] == 1) {
    if (time[2] == 31 || (time[1] == 4 || time[1] == 6 || time[1] == 9 || time[1] == 11) && time[2] == 30 || time[1] == 2 && (time[2] == 29 || time[2] == 28 && (vk.time[0] % 4))) {
      time[2] = 0;
    } else {
      time[2] = vk.time[2] = 0;
    }
  }
  if (time[2] > vk.time[2] + 1 || vk.time[2] > time[2] + 1) {
    time[2] = vk.time[2] = 0;
  }

  var realDt = (((time[2] - vk.time[2]) * 24 + (time[3] - vk.time[3])) * 60 + (time[4] - vk.time[4])) * 60;
  if (realDt < -16.5 * 3600) {
    realDt += 24 * 3600;
  } else if (realDt > 9.5 * 3600) {
    realDt -= 24 * 3600;
  }
  var finalDt = 0, minDt = Math.abs(realDt), dts = [-12, -11, -10, -9, -8, -7, -6, -5, -4.5, -4, -3.5, -3, -2.5, -2, -1, 0, 1, 2, 2.5, 3, 3.5, 4, 4.5, 5, 5.5, 5.75, 6, 6.5, 7, 8, 8.5, 9, 9.5, 10, 11, 12, 13];
  for (var i in dts) {
    var dt = Math.round((dts[i] - 4) * 3600), checkDt = Math.abs(realDt - dt);
    if (checkDt < minDt) {
      minDt = checkDt;
      finalDt = dt;
    }
  }
  debugLog('Real time offset: ' + (realDt / 3600) + ', minDt: ' + minDt + ', final time offset: ' + (finalDt / 3600));
  vk.dt = finalDt;
  if (getCookie('remixdt') != vk.dt) {
    setCookie('remixdt', vk.dt, 365);
  }
  if (window.devicePixelRatio >= 2 && !browser.iphone) {
    var rtc = intval(getCookie('remixrt'));
    if (!(rtc & 1)) {
      setCookie('remixrt', rtc + 1, 365);
      window._retinaInit = function() {
        stManager.add(['retina.css']);
        addClass(document.body, 'is_2x');
      };
      if (window._initedCheck) {
        window._retinaInit();
      }
    }
  }
}, 0);
//
// Other stuff
//

function dispatchIntro(step, params) {
  if (typeof dispatchIntroEvent != 'undefined') {dispatchIntroEvent(step, params);}
}

function parseLatin(text){
  var outtext = text;
  var lat1 = ['yo','zh','kh','ts','ch','sch','shch','sh','eh','yu','ya','YO','ZH','KH','TS','CH','SCH','SHCH','SH','EH','YU','YA',"'"];
  var rus1 = ['ё', 'ж', 'х', 'ц', 'ч', 'щ',  'щ',   'ш', 'э', 'ю', 'я', 'Ё', 'Ж', 'Х', 'Ц', 'Ч', 'Щ',  'Щ',   'Ш', 'Э', 'Ю', 'Я', 'ь'];
  for (var i = 0, l = lat1.length; i < l; i++) {
    outtext = outtext.split(lat1[i]).join(rus1[i]);
  }
  var lat2 = 'abvgdezijklmnoprstufhcyABVGDEZIJKLMNOPRSTUFHCYёЁ';
  var rus2 = 'абвгдезийклмнопрстуфхцыАБВГДЕЗИЙКЛМНОПРСТУФХЦЫеЕ';
  for (var i = 0, l = lat2.length; i < l; i++) {
    outtext = outtext.split(lat2.charAt(i)).join(rus2.charAt(i));
  }
  return (outtext == text) ? null : outtext;
}

function __phCheck(el, back, editable, focus, blur) {
  var shown = el.phshown, ph = el.phcont;
  if (editable) {
    var v = (el.textContent !== undefined ? el.textContent : el.innerText) || geByTag('img', el).length;
    if (v && browser.opera && v.match(/^[ ]+$/)) {
      v = '';
    }
  } else {
    var v = el.value;
  }
  if (shown && (back && v || !back && (focus || v))) {
    hide(ph);
    el.phshown = false;
  } else if (!shown && !v && (back || blur)) {
    show(ph);
    el.phshown = true;
  }
  if (back && !v) {
    if (focus) {
      clearTimeout(el.phanim);
      el.phanim = setTimeout(function() {
        animate(ph.firstChild.firstChild, {color: '#C0C8D0'}, 200);
      }, 100);
    }
    if (blur) {
      clearTimeout(el.phanim);
      el.phanim = setTimeout(function() {
        animate(ph.firstChild.firstChild, {color: '#777777'}, 200);
      }, 100);
    }
  }
}
function placeholderSetup(id, opts) {
  var el = ge(id), ph, o = opts ? clone(opts) : {};
  if (!el || (el.phevents && !o.reload) || !(ph = (el.getAttribute('placeholder') || el.placeholder))) {
    return;
  }

  el.setAttribute('placeholder', '');

  var pad = {}, dirs = ['Top', 'Bottom', 'Left', 'Right'];
  if (o.pad) {
    pad = o.pad;
  } else {
    if (o.fast) {
      for (var i = 0; i < 4; ++i) {
        pad['padding' + dirs[i]] = 3;
        pad['margin' + dirs[i]] = 0;
      }
      extend(pad, o.styles || {});
    } else {
      var prop = [];
      for (var i = 0; i < 4; ++i) {
        prop.push('margin' + dirs[i]);
        prop.push('padding' + dirs[i]);
      }
      pad = getStyle(el, prop);
    }
    for (var i = 0; i < 4; ++i) { // add border 1px
      var key = 'margin' + dirs[i];
      pad[key] = (intval(pad[key]) + 1) + 'px';
    }
  }

  if (o.reload) {
    var prel = el.previousSibling;
    if (prel && hasClass(prel, 'input_back_wrap')) re(prel);
  }
  var b1 = el.phcont = el.parentNode.insertBefore(ce('div', {className: 'input_back_wrap no_select', innerHTML: '
<div class="input_back"><div class="input_back_content">' + ph + '</div></div>
  '}), el), b = b1.firstChild, c = b.firstChild;
  setStyle(b, pad);

  var cv = __phCheck.pbind(el, o.back, o.editable), checkValue = browser.mobile ? cv : function(f, b) {
    setTimeout(cv.pbind(f, b), 0);
  }

  if (browser.msie && !browser.msie8) {
    setStyle(b, {marginTop: 1});
  }
  el.phonfocus = function(hid) {
    el.focused = true;
    cur.__focused = el;
    if (hid === true) {
      setStyle(el, {backgroundColor: '#FFF'});
      hide(b);
    }
    checkValue(true, false);
  }
  el.phonblur = function() {
    cur.__focused = el.focused = false;
    show(b);
    checkValue(false, true);
  }
  el.phshown = true, el.phanim = null;

  if (el.value || (o.editable && ((el.textContent !== undefined ? el.textContent : el.innerText) || geByTag('img', el).length))) {
    el.phshown = false;
    hide(b1);
  }

  if (!browser.opera_mobile) {
    addEvent(b1, 'focus click', function(ev) {
      if (o.editableFocus) {
        setTimeout(o.editableFocus.pbind(el), 0);
        el.phonfocus();
      } else {
        el.blur(); el.focus();
      }
    });
    addEvent(el, 'focus'+(o.editable ? ' click' : ''), el.phonfocus);
    addEvent(el, 'keydown paste cut input', checkValue);
  }
  addEvent(el, 'blur', el.phonblur);
  el.check = checkValue;

  el.getValue = function() {
    return el.value;
  }
  el.setValue = function(v) {
    el.value = v;
    __phCheck(el, o.back,  o.editable);
  }
  el.phevents = true;
  el.phonsize = function() {};

  if (o.global) return;

  if (!o.reload) {
    if (!cur.__phinputs) {
      cur.__phinputs = [];
      cur.destroy.push(function() {
        for (var i = 0, l = cur.__phinputs.length; i < l; ++i) {
          removeData(cur.__phinputs[i]);
        }
      });
    }
    cur.__phinputs.push(el);
  }
}

// Установка или получнене значений из контейнеров или полей. 
function val(input, value, nofire) {
  input = ge(input);
  if (!input) 
    return;

  if (value !== undefined) 
  {
    if (input.setValue) 
    {
      input.setValue(value);
      !nofire && input.phonblur && input.phonblur();
    } 
    else if (input.tagName == 'INPUT' || input.tagName == 'TEXTAREA') 
    {
      input.value = value
    } 
    else {
      input.innerHTML = value
    }
  }
  return input.getValue ? input.getValue() :
         (((input.tagName == 'INPUT' || input.tagName == 'TEXTAREA') ? input.value : input.innerHTML) || '');
}

// Установка фокуса и указателя элементу. 
function elfocus(el, from, to) {
  el = ge(el);
  try {
    el.focus();
    if (from === undefined || from === false) from = el.value.length;
    if (to === undefined || to === false) to = from;
    if (el.createTextRange) {
      var range = el.createTextRange();
      range.collapse(true);
      range.moveEnd('character', to);
      range.moveStart('character', from);
      range.select();
    } else if (el.setSelectionRange) {
      el.setSelectionRange(from, to);
    }
  } catch(e) {}
}

// --- Боксы разобраны 30.05.13. Нужно прокомментировать!!!
// Message box
var _message_box_guid = 0, _message_boxes = [], _show_flash_timeout = 0;

var __bq = boxQueue = { // Очередь. 
  hideAll: function() { // Закрывает все боксы. 
    if (__bq.count()) {
      var box = _message_boxes[__bq._boxes.pop()];
      box._in_queue = false;
      box._hide();
    }
    while (__bq.count()) {
      var box = _message_boxes[__bq._boxes.pop()];
      box._in_queue = false;
    }
  },
  hideLast: function(check, e) { // Закрывает текущий бокс. 
    if (__bq.count()) {
      var box = _message_boxes[__bq._boxes[__bq.count() - 1]];
      if (check === true && (box.changed || __bq.skip)) {
        __bq.skip = false;
        return;
      }
      box.hide();
    }
    if (e && e.type == 'click') return cancelEvent(e);
  },
  hideBGClick: function(e) { // Клик по слою.
    if (e && e.target && /^box_layer/.test(e.target.id)) {
      __bq.hideLast();
    }
  },
  count: function() { // Кол-во боксов в очереде. 
    return __bq._boxes.length;
  },
  _show: function(guid) { // Показывает бокс определяя очредь.  
    var box = _message_boxes[guid];
    if (!box || box._in_queue) return;
    if (__bq.count()) {
      _message_boxes[__bq._boxes[__bq.count() - 1]]._hide(true, true);
    } else if (window.tooltips) {
      tooltips.hideAll();
    }
    box._in_queue = true;
    var notFirst = __bq.count() ? true : false;
    __bq.curBox = guid;
    box._show(notFirst || __bq.currHiding, notFirst);
    __bq._boxes.push(guid);
  },
  _hide: function(guid) { // Закрывает бокс. Если в очереди есть боксы, то открывает предыдущий.
    var box = _message_boxes[guid];
    if (!box || !box._in_queue || __bq._boxes[__bq.count() - 1] != guid || !box.isVisible()) return;
    box._in_queue = false;
    __bq._boxes.pop();
    box._hide(__bq.count() ? true : false);
    if (__bq.count()) {
      var prev_guid = __bq._boxes[__bq.count() - 1];
      __bq.curBox = prev_guid;
      _message_boxes[prev_guid]._show(true, true, true);
    }
  },
  _boxes: [],
  curBox: 0
}

__bq.hideLastCheck = __bq.hideLast.pbind(true);
function curBox() { var b = _message_boxes[__bq.curBox]; return (b && b.isVisible()) ? b : null; }

if (!browser.mobile) {
  addEvent(document, 'keydown', function(e) {
    if (e.keyCode == KEY.ESC && __bq.count()) {
      __bq.hideLast();
      return -1;
    }
  });
}

function boxRefreshCoords(cont) {
  var height = window.innerHeight ? window.innerHeight : (document.documentElement.clientHeight ? document.documentElement.clientHeight : boxLayerBG.offsetHeight);
  var top = browser.mobile ? intval(window.pageYOffset) : 0;
  containerSize = getSize(cont);
  cont.style.marginTop = Math.max(10, top + (height - containerSize[1]) / 3) + 'px';
}

function MessageBox(options) {
  var defaults = {
    title: false,
    width: 410,
    height: 'auto',
    animSpeed: 0,
    bodyStyle: '',
    selfDestruct: true,
    progress: false,
    hideOnBGClick: false,
    hideButtons: false,
    onShow: false,
    onHideAttempt: false,
    onBeforeHide: false,
    onHide: false,
    onClean: false,
    onDestroy: false
  };

  options = extend(defaults, options);

  var buttonsCount = 0,
      boxContainer, boxBG, boxLayout;
  var boxTitleWrap, boxTitle, boxCloseButton, boxBody;
  var boxControlsWrap, boxControls, boxButtons, boxProgress, boxControlsText;
  var guid = _message_box_guid++, visible = false;

  if (!options.progress) options.progress = 'box_progress' + guid;

  var controlsStyle = options.hideButtons ? ' style="display: none"' : '';
  boxContainer = ce('div', {
    className: 'popup_box_container',
    innerHTML: '
<div class="box_layout" onclick="__bq.skip=true;">
<div class="box_title_wrap"><div class="box_x_button"></div><div class="box_title"></div></div>
<div class="box_body" style="' + options.bodyStyle + '"></div>
<div class="box_controls_wrap"' + controlsStyle + '><div class="box_controls">
<table cellspacing="0" cellpadding="0" class="fl_r"><tr></tr></table>
<div class="progress" id="' + options.progress + '"></div>
<div class="box_controls_text"></div>
</div></div>
</div>'
  }, {
    display: 'none'
  });
  hide(boxContainer);

  boxLayout = domFC(boxContainer);

  boxTitleWrap = domFC(boxLayout);
  boxCloseButton = domFC(boxTitleWrap);
  boxTitle = domNS(boxCloseButton);

  if (options.noCloseButton) hide(boxCloseButton);

  boxBody = domNS(boxTitleWrap);

  boxControlsWrap = domNS(boxBody);
  boxControls = domFC(boxControlsWrap);
  boxButtons = domFC(boxControls);
  boxProgress = domNS(boxButtons);
  boxControlsText = domNS(boxProgress);

  boxLayer.appendChild(boxContainer);

  refreshBox();
  boxRefreshCoords(boxContainer);

  // Refresh box properties
  function refreshBox() {
    // Set title
    if (options.title) {
      boxTitle.innerHTML = options.title;
      removeClass(boxBody, 'box_no_title');
      show(boxTitleWrap);
    } else {
      addClass(boxBody, 'box_no_title');
      hide(boxTitleWrap);
    }

    // Set box dimensions
    boxContainer.style.width = typeof(options.width) == 'string' ? options.width : options.width + 'px';
    boxContainer.style.height = typeof(options.height) == 'string' ? options.height : options.height + 'px';
  }

  // Add button
  function addButton(label, onclick, type) {
    ++buttonsCount;
    if (type == 'no') type = 'gray';
    if (type == 'yes') type = 'blue';
    var buttonWrap = ce('div', {
      className: 'button_' + (type ? type : 'blue'),
      innerHTML: '<button>' + label + '</button>'
    }), row = boxButtons.rows[0], cell = row.insertCell(0);
    cell.appendChild(buttonWrap);
    createButton(buttonWrap.firstChild, onclick);
    return buttonWrap;
  }

  // Add custom controls text
  function setControlsText(text) {
    boxControlsText.innerHTML = text;
  }

  // Remove buttons
  function removeButtons() {
    var row = boxButtons.rows[0];
    while (row.cells.length) {
      cleanElems(row.cells[0]);
      row.deleteCell(0);
    }
  }

  var destroyMe = function() {
    if (isFunction(options.onClean)) options.onClean();
    if (isFunction(options.onDestroy)) options.onDestroy();
    removeButtons();
    cleanElems(boxContainer, boxCloseButton, boxTitleWrap, boxControlsWrap);
    boxLayer.removeChild(boxContainer);
    delete _message_boxes[guid];
  }

  // Hide box
  var hideMe = function(showingOther, tempHiding) {
    if (!visible) return;
    visible = false;

    var speed = (showingOther === true) ? 0 : options.animSpeed;

    if (options.hideOnBGClick) {
      removeEvent(document, 'click', __bq.hideBGClick);
    }

    if (isFunction(options.onBeforeHide)) {
      options.onBeforeHide();
    }

    if (_layerAnim && !showingOther) {
      layers.boxhide();
    }

    var onHide = function() {
      if (__bq.currHiding == _message_boxes[guid]) {
        __bq.currHiding = false;
      }
      if (!_layerAnim && !_message_boxes[guid].shOther && !showingOther) {
        layers.boxhide();
      }
      if (!tempHiding && options.selfDestruct) {
        destroyMe();
      } else {
        hide(boxContainer);
      }
      if (isFunction(options.onHide)) {
        options.onHide();
      }
    }
    if (speed > 0) {
      __bq.currHiding = _message_boxes[guid];
      fadeOut(boxContainer, speed, onHide);
    } else {
      onHide();
    }
  }

  // Show box
  function showMe(noAnim, notFirst, isReturned) {
    if (visible || !_message_boxes[guid]) return;
    visible = true;

    var speed = (noAnim === true || notFirst) ? 0 : options.animSpeed;

    if (options.hideOnBGClick) {
      addEvent(document, 'click', __bq.hideBGClick);
    }

    // Show blocking background
    if (!notFirst) {
      layers.boxshow();
    }

    if (__bq.currHiding) {
      __bq.currHiding.shOther = true;
      var cont = __bq.currHiding.bodyNode.parentNode.parentNode;
      data(cont, 'tween').stop(true);
    }

    // Show box
    if (speed > 0) {
      fadeIn(boxContainer, speed);
    } else {
      show(boxContainer);
    }

    boxRefreshCoords(boxContainer);
    if (options.onShow) {
      options.onShow(isReturned);
    }

    _message_box_shown = true;
  }

  var fadeToColor = function(color) {
    return function() {
      animate(this, {backgroundColor: color}, 200);
    }
  }
  addEvent(boxCloseButton, 'mouseover', fadeToColor('#FFFFFF'));
  addEvent(boxCloseButton, 'mouseout', fadeToColor('#9DB7D4'));
  addEvent(boxCloseButton, 'click', __bq.hideLast);

  var retBox = _message_boxes[guid] = {
    guid: guid,
    _show: showMe,
    _hide: hideMe,

    bodyNode: boxBody,

    // Show box
    show: function() {
      __bq._show(guid);
      return this;
    },
    progress: boxProgress,
    showProgress: function() {
      hide(boxControlsText);
      show(boxProgress);
    },
    hideProgress: function() {
      hide(boxProgress);
      show(boxControlsText);
    },

    // Hide box
    hide: function(attemptParam) {
      if (isFunction(options.onHideAttempt) && !options.onHideAttempt(attemptParam)) return false;
      __bq._hide(guid);
      return true;
    },

    isVisible: function() {
      return visible;
    },
    bodyHeight: function() {
      return getStyle(boxBody, 'height');
    },

    // Insert html content into the box
    content: function(html) {
      if (options.onClean) options.onClean();
      boxBody.innerHTML = html;
      boxRefreshCoords(boxContainer);
      refreshBox();
      return this;
    },

    // Add button
    addButton: function(label, onclick, type, returnBtn) {
      var btn = addButton(label, onclick ? onclick : this.hide, type);
      return (returnBtn) ? btn : this;
    },

    setButtons: function(yes, onYes, no, onNo) {
      var b = this.removeButtons();
      if (!yes) return b.addButton(box_close);
      if (no) b.addButton(no, onNo, 'no');
      return b.addButton(yes, onYes);
    },

    // Set controls text
    setControlsText: setControlsText,

    // Remove buttons
    removeButtons: function() {
      removeButtons();
      return this;
    },

    destroy: destroyMe,

    // Update box options
    setOptions: function(newOptions) {
      if (options.hideOnBGClick) {
        removeEvent(document, 'click', __bq.hideBGClick);
      }
      options = extend(options, newOptions);
      if ('bodyStyle' in newOptions) {
        var items = options.bodyStyle.split(';');
        for (var i = 0, l = items.length; i < l; ++i) {
          var name_value = items[i].split(':');
          if (name_value.length > 1 && name_value[0].length) {
            boxBody.style[trim(name_value[0])] = trim(name_value[1]);
            if (boxBody.style.setProperty) {
              boxBody.style.setProperty(trim(name_value[0]), trim(name_value[1]), '');
            }
          }
        }
      }
      if (options.hideOnBGClick) {
        addEvent(document, 'click', __bq.hideBGClick);
      }
      toggle(boxControlsWrap, !options.hideButtons);
      refreshBox();
      boxRefreshCoords(boxContainer);
      return this;
    },
    evalBox: function(js, url, params) {
      var scr = '((function() { return function() { var box = this; ' + (js || '') + ';}; })())'; // IE :( Пичаль и безысходность...
      if (__debugMode) {
        var fn = eval(scr);
        fn.apply(this, [url, params]);
      } else try {
        var fn = eval(scr);
        fn.apply(this, [url, params]);
      } catch (e) {
        topError(e, {dt: 15, type: 7, url: url, query: params ? ajx2q(params) : undefined, js: js});
      }
    }
  }
  return retBox;
}

function showBox(url, params, options, e) {
  if (checkEvent(e)) return false;

  var opts = options || {},
      boxParams = opts.params || {},
      box = new MessageBox(boxParams), p = {
    onDone: function(title, html, js) {
      if (!box.isVisible()) return;
      try {
        show(boxLayerBG);
        box.setOptions({title: title, hideButtons: boxParams.hideButtons || false});
        if (opts.showProgress) {
          box.show();
        } else {
          show(box.bodyNode);
        }
        box.content(html);
        box.evalBox(js, url, params);
        if (opts.onDone) opts.onDone();
      } catch(e) {
        topError(e, {dt: 15, type: 103, url: url, query: ajx2q(params), answer: Array.prototype.slice.call(arguments).join('<!>')});
        if (box.isVisible()) box.hide();
      }
    },
    onFail: function(error) {
      box.failed = true;
      setTimeout(box.hide, 0);
      if (isFunction(opts.onFail)) return opts.onFail(error);
    },
    cache: opts.cache,
    stat: opts.stat,
    fromBox: true
  };

  if (opts.prgEl) {
    opts.showProgress = showGlobalPrg.pbind(opts.prgEl, {cls: opts.prgClass, w: opts.prgW, h: opts.prgH, hide: true});
    opts.hideProgress = hide.pbind('global_prg');
  }
  if (opts.showProgress) {
    extend(p, {
      showProgress: opts.showProgress,
      hideProgress: opts.hideProgress
    });
  } else {
    box.setOptions({title: false, hideButtons: true}).show();
    if (__bq.count() < 2) {
      hide(boxLayerBG);
    }
    hide(box.bodyNode);
    p.showProgress = function() {
      show(boxLoader);
      boxRefreshCoords(boxLoader);
    }
    p.hideProgress = hide.pbind(boxLoader);
  }
  box.removeButtons().addButton(getLang('global_close'));

  ajax.post(url, params, p);
  return box;
}

function showTabbedBox(url, params, options, e) {
  options = options || {};
  options.stat = options.stat || [];
  options.stat.push('box.js', 'boxes.css');
  return showBox(url, params, options, e)
}

function showFastBox(o, c, yes, onYes, no, onNo) {
  return (new MessageBox(typeof(o) == 'string' ? {title: o} : o)).content(c).setButtons(yes, onYes, no, onNo).show();
}

function showCaptchaBox(sid, dif, box, o) {
  var done = function(e) {
    if (e && e.keyCode !== undefined && e.keyCode != 10 && e.keyCode != 13) return;
    var key = geByTag1('input', box.bodyNode);
    if (!trim(key.value) && e !== true) {
      elfocus(key);
      return;
    }
    var imgs = geByTag1('img', box.bodyNode);
    var captcha = imgs[0], loader = imgs[1];
    removeEvent(key);
    removeEvent(captcha);
    show(geByClass1('progress', box.bodyNode));
    hide(key);
    o.onSubmit(sid, key.value);
  }
  var was_box = box ? true : false;
  var difficulty = intval(dif) ? '' : '&s=1';
  var imgSrc = o.imgSrc || '/captcha.php?sid=' + sid + difficulty;
  if (!was_box) {
    var content = '
<div class="captcha">
  <div><img src="' + imgSrc + '"/></div>
  <div><input type="text" class="big_text" maxlength="7" placeholder="' + getLang('global_captcha_input_here') + '" /><div class="progress" /></div></div>
</div>' + (o.addText || '');
    box = showFastBox({
      title: getLang('captcha_enter_code'),
      width: 305,
      onHide: o.onHide,
      onDestroy: o.onDestroy || false
    }, content, getLang('captcha_send'), function() {
      box.submit();
    }, getLang('captcha_cancel'), function() {
      var key = geByTag1('input', box.bodyNode);
      var captcha = geByTag1('img', box.bodyNode);
      removeEvent(key);
      removeEvent(captcha);
      box.hide();
    });
  }
  box.submit = done.pbind(true);
  box.changed = true;
  var key = geByTag1('input', box.bodyNode);
  var captcha = geByTag1('img', box.bodyNode);
  if (was_box) {
    key.value = '';
    captcha.src = '/captcha.php?sid=' + sid + difficulty;
    hide(geByClass1('progress', box.bodyNode));
  }
  show(key);
  addEvent(key, 'keypress', done);
  addEvent(captcha, 'click', function() {
    this.src = '/captcha.php?sid=' + sid + difficulty + '&v=' + irand(1000000, 2000000);
  });
  elfocus(key);
  return box;
}

// Three-state button

function createButton(el, onClick) {
  el = ge(el);
  if (!el || el.btnevents) return;
  var p = el.parentNode;

  if (hasClass(p, 'button_blue') || hasClass(p, 'button_gray')) {
    if (isFunction(onClick)) {
      el.onclick = onClick.pbind(el);
    }
    return;
  }
  var hover = false;
  addEvent(el, 'click mousedown mouseover mouseout', function(e) {
    if (hasClass(p, 'locked')) return;
    switch (e.type) {
    case 'click':
      if (!hover) return;
      el.className = 'button_hover';
      onClick(el);
      return cancelEvent(e);
    break;
    case 'mousedown':
      el.className = 'button_down';
    break;
    case 'mouseover':
      hover = true;
      el.className = 'button_hover';
    break;
    case 'mouseout':
      el.className = 'button';
      hover = false;
    break;
    }
  });
  el.btnevents = true;
}

function lockButton(el) {
  if (!(el = ge(el))) return;
  var btn = (el.tagName.toLowerCase() == 'button'), d = btn ? 2 : ((browser.msie6 || browser.msie7) ? 2 : 4), tEl = btn ? el : geByClass1('file_button_text', el);
  if (!btn && !hasClass(el, 'file_button') || buttonLocked(el)) return;
  var lock = ce('span', {className: 'button_lock'});
  el.parentNode.insertBefore(lock, el);
  el['old_width'] = el.style.width;
  el['old_height'] = el.style.height;
  var s = getSize(el.parentNode);
  setStyle(el, {width: s[0] - d, height: s[1] - d});
  if (browser.msie6 || browser.msie7) {
    tEl['old_html'] = tEl.innerHTML; tEl.innerHTML = '';
  } else {
    tEl.style.textIndent = '-9999px';
  }
}
function unlockButton(el) {
  if (!(el = ge(el))) return;
  var lock = geByClass1('button_lock', el.parentNode, 'span'), btn = (el.tagName.toLowerCase() == 'button'), tEl = btn ? el : geByClass1('file_button_text', el);
  if (!lock) return;
  el.parentNode.removeChild(lock);
  el.style.width = el['old_width'];
  el.style.height = el['old_height'];
  if (browser.msie6 || browser.msie7) tEl.innerHTML = tEl['old_html'];
  tEl.style.textIndent = '';
}
function buttonLocked(el) {
  if (!(el = ge(el))) return;
  return geByClass1('button_lock', el.parentNode, 'span') ? true : false;
}

function sbWidth() {
  if (window._sbWidth === undefined) {
    var t = ce('div', {innerHTML: '<div style="height: 75px;">1<br>1</div>'}, {
      overflowY: 'scroll',
      position: 'absolute',
      width: '50px',
      height: '50px'
    });
    bodyNode.appendChild(t);
    window._sbWidth = Math.max(0, t.offsetWidth - t.firstChild.offsetWidth - 1);
    bodyNode.removeChild(t);
  }
  return window._sbWidth;
}

function checkTextLength(maxLen, inp, warn, nobr, cut) {
  var value = (inp.getValue) ? inp.getValue() : inp.value,
      lastLen = inp.lastLen || 0;
  if (inp.lastLen === value.length) return;
  inp.lastLen = value.length;
  var countRealLen = function(text, nobr) {
    var spec = {'&': 5, '<': 4, '>': 4, '"': 6, "n": (nobr ? 1 : 4), "r": 0, '!': 5, "'": 5, '$': 6, '\': 6};
    if (cut) spec[','] = 5;
    var res = 0;
    for (var i = 0, l = text.length; i < l; i++) {
      var k = spec[text.charAt(i)], c = text.charCodeAt(i);
      if (k !== undefined) res += k;
      else if ((c > 0x80 && c < 0xC0) || c > 0x500) res += ('&#' + c + ';').length;
      else res += 1;
    }
    return res;
  }
  var realLen = countRealLen(value, nobr);
  warn = ge(warn);
  if (realLen > maxLen - 100) {
    show(warn);
    if (realLen > maxLen) {
      if (cut) {
        var cutVal = val(inp, value.substr(0, Math.min(maxLen, lastLen)));
        inp.lastLen = cutVal.length;
        warn.innerHTML = getLang('text_N_symbols_remain', 0);
      } else {
        warn.innerHTML = getLang('text_exceeds_symbol_limit', realLen - maxLen);
      }
    } else {
      warn.innerHTML = getLang('text_N_symbols_remain', maxLen - realLen);
    }
  } else {
    hide(warn);
  }
}

// Установка автоматической длины textarea при заполнении поля. 
function autosizeSetup(el, options) {
  el = ge(el);
  if (!el) return;
  if (el.autosize) {
    el.autosize.update();
    return;
  }

  options.minHeight = intval(options.minHeight) || intval(getStyle(el, 'height'));
  options.maxHeight = intval(options.maxHeight);

  var elwidth = intval(getStyle(el, 'width')),
      fs = getStyle(el, 'fontSize'),
      lh;
  if (elwidth < 1) {
    elwidth = intval(getStyle(el, 'width', false));
  }
  if (fs.indexOf('em') > 0) {
    fs = floatval(fs) * vk.fs;
  }
  fs = intval(fs);
  el.autosize = {
    options: options,
    helper: ce('textarea', {className: 'ashelper'}, {
      width: elwidth,
      height: 10,
      fontFamily: getStyle(el, 'fontFamily'),
      fontSize: fs + 'px',
      lineHeight: (lh = getStyle(el, 'lineHeight'))
    }),
    handleEvent: function(v, e) {
      var ch = e.charCode ? String.fromCharCode(e.charCode) : e.charCode;
      if (ch === undefined) {
        ch = String.fromCharCode(e.keyCode);
        if (e.keyCode == 10 || e.keyCode == 13) {
          ch = 'n';
        } else if (!browser.msie && e.keyCode <= 40) {
          ch = '';
        }
      }
      if (!ch) {
        return v;
      }
      if (!browser.msie) {
        return v.substr(0, el.selectionStart) + ch + v.substr(el.selectionEnd);
      }
      var r = document.selection.createRange();
      if (r.text) {
        v = v.replace(r.text, '');
      }
      return v + ch;
    },
    update: function(e) {
      var value = el.value;
      if (e && e.type != 'blur' && e.type != 'keyup' && (!browser.msie || e.type == 'keypress')) {
        if (!e.ctrlKey && !e.altKey && !e.metaKey) {
          value = el.autosize.handleEvent(value, e);
        }
      }
      if (!value) {
        value = ' ';
      }
      if (el.autosize.helper.value != value) {
        el.autosize.helper.value = value;
      }
      var opts = el.autosize.options;

      var oldHeight = getSize(el, true)[1];
      var newHeight = el.autosize.helper.scrollHeight, df;
      if (opts.exact && (df = newHeight % lh) > 2) {
        newHeight -= (df - 2);
      }
      if (newHeight < opts.minHeight) {
        newHeight = opts.minHeight;
      }
      var newStyle = {overflow: 'hidden'}, curOverflow = getStyle(el, 'overflow').indexOf('auto') > -1 ? 'auto' : 'hidden';
      if (opts.maxHeight && newHeight > opts.maxHeight) {
        newHeight = opts.maxHeight;
        extend(newStyle, {overflow: 'auto', overflowX: 'hidden'});
      }
      if (oldHeight != newHeight || curOverflow != newStyle.overflow) {
        newStyle.height = newHeight;
        setStyle(el, newStyle);
        if (isFunction(opts.onResize)) {
          opts.onResize(newHeight);
        }
      }
    }
  }
  if (options.exact) {
    if (lh == 'normal') lh = '120%';
    lh = intval((lh.indexOf('%') > 0) ? fs * intval(lh) / 100 : lh);
  }
  utilsNode.appendChild(el.autosize.helper);
  if (browser.opera_mobile) {
    setStyle(el, {overflow: 'hidden'});
    el.autosize.update();
    addEvent(el, 'blur', el.autosize.update);
  } else {
    addEvent(el, 'keydown keyup keypress', el.autosize.update);
    setTimeout(function() {
      setStyle(el, {overflow: 'hidden', resize: 'none'});
      el.autosize.update();
      var t = val(el); val(el, ' ', true); val(el, t, true);
    }, 0);
  }
}

function goAway(lnk, prms, e) {
  if ((prms || {}).h != -1 || checkEvent(e)) {
    return true;
  }
  if ((prms || {}).h != -1) {
    var m = lnk.match(/https?://([a-zA-Z0-9-_.]+.)?(vk.com|vkontakte.ru)(/|$)/i);
    if (m && m[1].toLowerCase() != 'api.') {
      location.href = lnk;
      return false;
    }
    var no_warning = intval(getCookie('remixsettings_bits'));
    if (/https?://([a-zA-Z0-9-_]+.)(vk.com|vkontakte.ru)(/|$)/i.test(locBase) || no_warning & 1) {
      window.open('/away.php?to=' + encodeURIComponent(lnk) + ((prms && prms.h !== undefined) ? '&h=' + prms.h : ''), '_blank');
      return false;
    }
  }
  var params = extend({act: 'a_go', to: lnk}, prms || {});
  return !showBox('away.php', params, {}, e);
}

function isChecked(el) {
  el = ge(el);
  return hasClass(el, 'on') ? 1 : '';
}
function checkbox(el, v) {
  el = ge(el);
  if (!el || hasClass(el, 'disabled')) return;

  if (v === undefined) {
    v = !isChecked(el);
  }
  if (v) {
    addClass(el, 'on');
  } else {
    removeClass(el, 'on');
  }
  return false;
}

function disable(el, v) {
  el = ge(el);

  if (v === undefined) {
    v = !hasClass(el, 'disabled');
  }
  if (v) {
    addClass(el, 'disabled');
  } else {
    removeClass(el, 'disabled');
  }
  return false;
}

var radioBtns = {};
function radioval(name) {
  return radioBtns[name] ? radioBtns[name].val : false;
}
function radiobtn(el, v, name) {
  if (!radioBtns[name]) return;
  each(radioBtns[name].els, function() {
    if (this == el) {
      addClass(this, 'on');
    } else {
      removeClass(this, 'on');
    }
  });
  return radioBtns[name].val = v;
}

function renderFlash(cont, opts, params, vars) {
  if (!opts.url || !opts.id) {
    return false;
  }
  opts = extend({
    version: 9,
    width: 1,
    height: 1
  }, opts);
  var f = opts.url;
  if (!stVersions[f]) {
    stVersions[f] = '';
  }
  if (__debugMode && stVersions[f] < 1000000) stVersions[f] += irand(1000000, 2000000);

  if (stVersions[f]) {
    opts.url += ((opts.url.indexOf('?') == -1) ? '?' : '&') + '_stV=' + stVersions[f];
  }

  params = extend({
    quality: 'high',
    flashvars: ajx2q(vars)
  }, params);
  if (browser.flash < opts.version) {
//    if (opts.express) {
//      params.flashvars += '&MMplayerType=PlugIn&MMredirectURL=' + encodeURIComponent(locBase + location.hash);
//    } else {
      return false;
//    }
  }
  ge(cont).innerHTML = browser.flashwrap(opts, params);
  return true;
}

function playAudioNew() {
  var args = arguments;
  if (args[args.length - 1] !== false) args = Array.prototype.slice.apply(arguments).concat([true]);
  stManager.add(['audioplayer.js', 'audioplayer.css'], function() {
    audioPlayer.operate.apply(null, args);
  });
}

function showAudioClaimWarning(owner_id, id, delete_hash, claim_id, title) {
  var claimText = getLang(claim_id > 0 ? 'audio_claim_warning_objection' : (claim_id == 0 ? 'audio_claim_warning_text' : 'audio_claim_warning'));
  claimText = claimText.split('{audio}').join('<b>' + title + '</b>');
  claimText = claimText.split('{objection_link}').join('<a href="/help?act=cc_objection&claim=' + claim_id + '&content=audio' + owner_id + '_' + id + '">' + getLang('audio_claim_objection') + '</a>');
  claimText = claimText.split('{delete_link}').join('<a href="#" onclick="deleteAudioOnClaim(' + owner_id + ',' + id + ','' + delete_hash + ''); return false;">' + getLang('audio_claim_delete') + '</a>');
  cur.claimWarning = showFastBox(getLang('audio_claim_warning_title'), claimText);
}

function deleteAudioOnClaim(owner_id, id, delete_hash) {
  if (cur.silent) {
    cur.onSilentLoad = function() {
      deleteAudioOnClaim(owner_id, id, delete_hash);
    };
    return;
  }
  if (cur.deleting) {
    return false;
  }
  cur.deleting = true;
  var el = ge('audio' + id);
  var h = getSize(geByClass1('play_btn', el))[1];
  ajax.post('/audio', {act: 'delete_audio', oid: owner_id, aid: id, hash: delete_hash, restore: 1}, {
    onDone: function(text, delete_all) {
      if (cur.claimWarning) {
        cur.claimWarning.hide();
      }
      cur.deleting = false;
      if (!cur.deletedAudios) cur.deletedAudios = [];
      cur.deletedAudios[id] = ge('audio'+id).innerHTML;
      el.innerHTML = text;
      setStyle(geByClass1('dld', el), {height: h+'px'});
      el.style.cursor = 'auto';
      el.setAttribute('nosorthandle', '1');
      if (delete_all) {
        cur.summaryLang.delete_all = delete_all;
      }
      if (cur.audios && cur.audiosIndex) {
        cur.audiosIndex.remove(cur.audios[id]);
        cur.audios[id].deleted = true;
      }
      cur.sectionCount--;
      if (Audio) {
        Audio.changeSummary();
      }
    }
  });
  return false;
}

window.__qlTimer = null;
window.__qlClear = function() { clearTimeout(__qlTimer); setTimeout(function() { clearTimeout(__qlTimer); }, 2000); }
window.onLoginDone = function () {
  __qlClear();
  nav.reload({force: true, from: 6});
}
window.onLogout = function() {
  __qlClear();
  if (window.audioPlayer) {
    audioPlayer.stop();
    toggleGlobalPlayer(false);
  }
  window.Notifier && Notifier.standby();
  window.FastChat && FastChat.standby();
  window.Page && Page.postsClear();
  if (ls.checkVersion()) {
    try {
      window.localStorage.clear();
    } catch (e) {}
  }
  nav.reload({from: 5});
}

function onLoginFailed(code, opts) {
  __qlClear();
  switch (code) {
    case -1: location.href = location.href.replace(/^http:/, 'https:'); break;
    case 4: location.href = '/login.php?m=1&email=' + opts.email; break;
    default: location.href = '/login.php'; break;
  }
}
function onLoginCaptcha(sid, dif) {
  __qlClear();
  unlockButton(window.__qfBtn);
  window.qloginBox = showCaptchaBox(sid, dif, window.qloginBox, {onSubmit: function(sid, key) {
    ge('quick_captcha_sid').value = sid;
    ge('quick_captcha_key').value = key;
    ge('quick_login_form').submit();
  }, onHide: function() { window.qloginBox = false; }});
}

function callHub(func, count) {
  this.count = count || 1;
  this.done = function(c) {
    this.count -= c || 1;
    if (this.count <= 0) {
      func();
    }
  };
}

function showWriteMessageBox(e, id) {
  gSearch.hide(e, true);
  if (cur.onFreindMessage) cur.onFreindMessage();
  stManager.add(['mail.css', 'wide_dd.css', 'page.css', 'page.js', 'wide_dd.js', 'writebox.js']);
  var box = showBox('al_mail.php', {act: 'write_box', to: id}, {stat: ['mail.css', 'wide_dd.css'], cache: 1}, e);
  if (box) cancelEvent(e);
  return !box;
}

function startVideocall(e, id, with_video) {
  if (checkEvent(e)) return true;
  gSearch.hide(e, true);
  var tt = (ge('videocall_btn') || {}).tt;
  if (tt && tt.hide && tt.destroy) {
    tt.hide({fasthide: 1});
    tt.destroy();
  }
  tt = (ge('profile_am_subscribed') || {}).tt;
  if (tt && tt.hide && tt.destroy) {
    tt.hide({fasthide: 1});
    tt.destroy();
  }

  var stat = ['call.js', 'call.css', 'notifier.js', 'notifier.css'];

  var hub = new callHub(function() {
    Call.rtmpServer = hub.data[0];
    Call.rtmfpServer = hub.data[1];
    window.langpack = extend(window.langpack || {}, hub.data[2]);
    vk.vc_h = hub.data[3];
    Call.start(id, with_video);
  }, 2);

  stManager.add(stat, function() {
    hub.done();
  });

  var options = {onDone: function() {
    hub.data = arguments;
    hub.done();
  }, params: {act: 'init'}, loader: true};

  ajax.post('call.php', options.params, options);
  return false;
}

var gSearch = new (function() {
  this.on = 0;
  var self = this;
  this.hub = new callHub(function() {
    if (self.onShow) self.onShow();
  }, 2);
  this.hintsHub = new callHub(function() {
    self.showStartHints();
  }, 2);
  this.load = function() {
    if (!ge('quick_search')) return;
    if (this.loading) return;
    this.loading = true;
    stManager.add('qsearch.js', function() {
      self.hub.done();
    });
    ajax.post('hints.php', {act: 'a_start_hints'}, {onDone: function(text) {
      self.startHintsText = trim(text);
      self.hintsHub.done();
    }});
  },
  this.show = function(e, noAnim) {
    if (!ge('quick_search')) return;
    if (this.on) {
      return this.go(e);
    }
    this.on = 1;
    show(self.sCont);
    placeholderSetup('search_input');
    ge('search_input').setAttribute('autocomplete', 'off');
    addClass(ge('qsearch_link'), 'active');
    this.prev_content = ge('content');
    if (!this.qsearch_cont) {
      this.qsearch_cont = ce('div', {id: 'content', innerHTML: '<div style="padding: 200px; text-align: center;"><img src="/images/progress7.gif"/></div>'});
    }
    // hide('header');
    this.prev_content.parentNode.replaceChild(this.qsearch_cont, this.prev_content);
    if (!this.loading) this.load();
    self.hub.done();
    self.hintsHub.done();
    if (e) return cancelEvent(e);
  };
  this.go = function(e) {
    var url = '/gsearch.php?section=' + (self.last_section || 'people') + '&q=' + trim(ge('search_input').value) + '&name=1';
    cancelEvent(e || window.event);
    location.href = url;
    return false;
  };

  this.hide = function(e, force) {
    if (!ge('quick_search')) return;
    if ((self.active && !force) || !self.on) return;
    self.on = 0;
    toggleFlash();
    if (self.beforeHide && self.beforeHide()) {
      return true;
    }
    if (ge('search_input').setValue) {
      ge('search_input').setValue('');
    } else {
      ge('search_input').value = '';
    }
    // show('header');
    hide(self.sCont);
    removeClass(ge('qsearch_link'), 'active');
    self.qsearch_cont.parentNode.replaceChild(self.prev_content, self.qsearch_cont);
  };
  this.init = function(options) {
    this.sCont = ge('quick_search');
    this.opt = options || {};
  };
  var qsearch_start = false, qsearch_requested = false;
  this.preload = function() {
    var url = '/al_search.php', params = {__query: 'search', al_id: vk.id, preload: 1}, q = url + '#' + ajx2q(params);
    if (globalAjaxCache[q] !== undefined) return;
    globalAjaxCache[q] = -1;
    ajax.post(url, extend(params, {al: 1}), {onDone: function() {
      var cb = globalAjaxCache[q], args = Array.prototype.slice.call(arguments);
      globalAjaxCache[q] = args;
      if (isFunction(cb)) {
        cb.apply(window, args);
      }
    }, onFail: function() {
      delete globalAjaxCache[q];
    }});
  };
})();

// opts: {url: '...', params: {}} or {text: '...'} or {content: '...'}
var _cleanHide = function(el) {
  if (el.temphide) {
    removeEvent(el, 'mouseout', el.temphide);
    removeAttr(el, 'temphide');
    removeAttr(el, 'showing');
  }
}

function showTooltip(el, opts) {
  if (!vk.loaded && !opts.noload) return;

  if (!el) return;
  if (!el.temphide) {
    el.temphide = function() {
      el.showing = false;
    }
    addEvent(el, 'mouseout', el.temphide);
  }
  el.showing = true;
  if (el.tt == 'loadingstat') return;

  if (!el.tt) {
    el.tt = 'loadingstat';
  }

  cur.cancelTooltip = false;

  if (opts.stat) stManager.add(opts.stat);
  stManager.add(['tooltips.js', 'tooltips.css'], function() {
    if (el.tt == 'loadingstat') el.tt = false;

    if (!el.showing || cur.cancelTooltip) return;
    _cleanHide(el);

    if (!el.tt || !el.tt.el || opts.force) {
      tooltips.create(el, opts);
      if (opts.onCreate) opts.onCreate();
    }
    tooltips.show(el, opts);
  });
}
function reportAd(ad_id) {
  showBox('/reports.php?act=a_report_ad_box', {ad_id: ad_id}, {params: {width: 350}, stat: ['ui_controls.js', 'ui_controls.css']});
}

function updateMoney(balance) {
  if (balance === undefined || balance === false) return;
  vk.balance = balance;
  var els = geByClass('votes_balance_nom');
  for (var i in els) {
    els[i].innerHTML = balance+' '+getLang('votes_flex', balance);
  }
}

function zNav(changed, opts) {
  var z = changed.z, f = changed.f, w = changed.w;
  delete changed.z;
  delete changed.f;
  delete changed.w;


  if (f) {
    handleScroll(f);
    if (z === undefined) return false;
  }
  if (!isEmpty(changed)) return;
  if (w) {
    showWiki({w: w}, w == 'note_new', false, {onLoaded: z && z.match(/^single-?d+$/) && stManager.add.pbind(['single_pv.css', 'single_pv.js'], function() { ge(z).onclick(); })});
    if (z === undefined) return false;
  } else if (z == 'giftbox') {
    return !showTabbedBox('/al_gifts.php', {act: 'get_gift_box', tab: 'birthday', mid: opts.id || 0, link: nav.objLoc[0]}, {stat: ['gifts.css', 'ui_controls.js', 'ui_controls.css'], cache: 1}, window.event);
  }
  if (z === false || w === false) {
    var wkHide = (!window.wkcur || !wkcur.shown || layers.fullhide != WkView.hide);
    if (layers.fullhide && (wkHide || w === false)) {
      layers.fullhide(opts.hist ? 2 : false);
    }
    return false;
  }

  if (!z) return;
  var zType = z.match(/^([a-z]+)(-?d+(?:_d+)?)/?(.*)/i);
  if (zType) {
    var onFail = function() {
      delete nav.objLoc.z;
      nav.setLoc(nav.objLoc);
      return true;
    };
    switch (zType[1]) {
      case 'photo':
        showPhoto(zType[2], zType[3], extend(opts || {}, {onFail: onFail, noHistory: true}));
        return false;
        break;
      case 'albums':
        showAlbums(zType[2], extend(opts || {}, {onFail: onFail, noHistory: true}));
        return false;
        break;
      case 'album':
        showAlbum(zType[2], extend(opts || {}, {onFail: onFail, noHistory: true}));
        return false;
        break;
      case 'video':
        showVideo(zType[2], zType[3], extend(opts || {}, {onFail: onFail, noLocChange: 1}));
        return false;
        break;
      case 'screen':
        stManager.add(['index.js'], function() { setTimeout(JoinPhotoview.show.pbind(intval(zType[2])), 0); });
        return false;
        break;
      case 'single':
        if (w === undefined) {
          stManager.add(['single_pv.css', 'single_pv.js'], ge(z).onclick);
        }
        return false;
        break;
    }
  }
}

function handleScroll(scroll) {
  scroll = scroll.split(',');
  var named = cur.named || {},
      scrollEl = scroll[0] && (named[scroll[0]] || ge(scroll[0])) || false,
      focusEl = scroll[1] && (named[scroll[1]] || ge(scroll[1])) || false;

  if (!scrollEl && !focusEl) {
    scrollEl = document.getElementsByName(scroll[0])[0];
    if (!scrollEl) {
      return;
    }
  }

  setTimeout(function() {
    scrollEl && scrollToY(getXY(scrollEl)[1], 0);
    focusEl && elfocus(focusEl);
  }, 300);
}

function showGlobalPrg(img, opts) {
  var xy = getXY(img), sz = getSize(img), o = opts || {}, w = o.w || 32, h = o.h || 13, el = ge('global_prg');
  el.className = o.cls || 'progress';
  setStyle(el, {
    left: xy[0] + Math.floor((sz[0] - w) / 2),
    top: xy[1] + Math.floor((sz[1] - h) / 2),
    width: w, height: h,
    display: 'block'
  });
  if (o.hide) {
    img.style.visibility = 'hidden';
  }
}
function showPhoto(photoId, listId, options, ev) {
  if (cur.viewAsBox) return cur.viewAsBox();
  if (checkEvent(ev) || (cur._editMode && cur._editMode(ev))) return;
  var stat = ['photoview.js', 'photoview.css'], phv = window.Photoview;
  if (options.img) {
    options.showProgress = showGlobalPrg.pbind(options.img, {cls: 'progress_inv_img', w: 46, h: 16});
    options.hideProgress = hide.pbind('global_prg');
  }
  if (phv && (phv.showPhoto(photoId, listId, options) === false)) {
    return false;
  }
  var doShow = true;
  if (options.temp) {
    stManager.add(stat, function() {
      extend(cur, {
        pvCancelLoad: function() {
          doShow = false;
        },
        pvData: cur.pvData || {}
      });
      cur.pvData.temp = [options.temp];
      Photoview.show('temp', 0);
    });
  }
  extend(options, {onDone: function(lst) {
    Photoview.list(photoId, listId, lst);
    Photoview.loaded.apply(window, arguments);
    if (!doShow) return;
    Photoview.showPhoto(photoId, listId, options, true);
  }, stat: stat, cache: 1});
  ajax.post('al_photos.php', {act: 'show', photo: photoId, list: listId, module: cur.module || ''}, options);

  return false;
}
function showAlbums(ownerId, options, ev) {
  if (cur.viewAsBox) return cur.viewAsBox();
  if (checkEvent(ev)) return;
  stManager.add(['photoview.js', 'photoview.css'], function() {
    Photoview.showAlbums(ownerId, options);
  });
  return false;
}

function showAlbum(albumRaw, options, ev) {
  if (cur.viewAsBox) return cur.viewAsBox();
  if (checkEvent(ev)) return;
  stManager.add(['photoview.js', 'photoview.css'], function() {
    Photoview.showAlbum(albumRaw, options);
  });
  return false;
}

window._videoLastInlined = false;
function showVideo(videoId, listId, options, ev) {
  if (cur.viewAsBox) return cur.viewAsBox();
  if (checkEvent(ev)) return true;

  if (window.mvcur && mvcur.mvShown && mvcur.minimized && mvcur.videoRaw == videoId) {
    videoview.unminimize();
    return false;
  }

  var claim = nav.objLoc.claim;

  var stat = ['videoview.js', 'videoview.css'];

  var hub = new callHub(function() {
    revertLastInlineVideo();
    videoview.showVideo.apply(videoview, hub.data);
  }, 2);

  if (!options) {
    options = {};
  }

  stManager.add(stat, function() {
    videoview.show(ev, videoId, listId, options);
    hub.done();
  });

  extend(options, {onDone: function() {
    hub.data = arguments;
    hub.done();
  }, cache: (listId != 'status')});

  if (!options.params) {
    options.params = {act: 'show', video: videoId, list: listId, autoplay: (options.autoplay) ? 1 : 0, ad_video: options.ad_video, module: cur.module || ''};
  }
  if (claim) {
    options.params.claim = claim;
  }
  ajax.post('al_video.php', options.params, options);
  return false;
}

function showInlineVideo(videoId, listId, options, ev, thumb) {
  if (checkEvent(ev)) return true;

  if (window.mvcur && mvcur.mvShown) {
    return showVideo(videoId, listId, options, ev);
  }

  options = options || {};
  options.params = options.params || {act: 'show_inline', video: videoId, list: listId, autoplay: (options.autoplay) ? 1 : 0};
  var h = thumb.clientHeight,
      w = thumb.clientWidth,
      btn = geByClass1('video_play_inline', thumb, 'div');

  extend(options.params, {width: w, height: h});
  options.onDone = function (title, html, js, opts) {
    revertLastInlineVideo();
    hide(thumb);
    var videoWrap = ce('div', {id: 'page_video_inline_wrap' + videoId, className: 'page_video_inline_wrap', innerHTML: html}, {width: w, height: h}),
        videoBg = ge('video_background' + videoId);
    _videoLastInlined = [videoWrap, thumb]
    thumb.parentNode.appendChild(videoWrap);
    videoBg && setStyle(geByTag1('img', videoBg), {width: w, height: h});
    try {
      eval('(function () {' + js + '})();');
    } catch (e) {
      debugLog('video inline error', e.message, e.stack, e, js);
    }
    var _n = window.Notifier, _a = window.audioPlayer;
    if (_n) setTimeout(function() { _n.lcSend('video_start'); }, 0);
    if (_a && _a.player && !_a.player.paused()) {
      _a.pauseTrack();
      _a.pausedByVideo = 1;
    }
  };
  options.showProgress = function () {
    addClass(btn, 'video_play_inline_loading');
  };
  options.hideProgress = function () {
    removeClass(btn, 'video_play_inline_loading');
  };
  stManager.add('videoview.js', function() {
    ajax.post('al_video.php', options.params, options);
  });
  return false;
}

function revertLastInlineVideo(ancestor) {
  if (!_videoLastInlined) {
    return;
  }
  var current, found = false;
  if ((ancestor = ge(ancestor)) &&
      (current = _videoLastInlined[0])) {
    while (current = current.parentNode) {
      if (current == ancestor) {
        found = true;
        break;
      }
    }
    if (!found) {
      return;
    }
  }
  re(_videoLastInlined[0]);
  show(_videoLastInlined[1]);
  _videoLastInlined = false;
}

function showWiki(page, edit, e, opts) {
  if (checkEvent(e)) return true;
  var opts = opts || {};
  var stat = opts.stat || ['wkview.js' ,'wkview.css', 'wk.css', 'wk.js'];
  edit && stat.push('wysiwyg.js', 'wysiwyg.css');
  ajax.post('wkview.php', extend({act: 'show', loc: nav.objLoc[0]}, page), {
    stat: stat,
    loader: 1,
    onDone: function(title, html, options, script) {
      WkView.show(title, html, extend(options, opts), script, e);
    },
    onFail: function(text) {
      return WkView.showError(text);
    }
  });
  return cancelEvent(e);
}

function videoCallback(params) {
  var method = params.shift();
  if (window.videoview && videoview.playerCallback[method]) {
    videoview.playerCallback[method].apply(videoview, params);
  } else {
    throw Error('Unregistered player callback: ' + method);
  }
}

function showApp(ev, aid, needInstall, ref, mid, options) {
  if (!options) {
    options = {};
  }
  var params = extend({w: 'app'+aid}, options);
  if (ref) {
    if (isObject(ref)) {
      params = extend(params, ref);
    } else {
      params['ref'] = ref;
    }
  }
  if (cur.apps && cur.apps[aid] || !needInstall) {
    delete params['w'];
    return nav.go('/app' + aid + (mid ? '_' + mid : '') + nav.toStr(params), ev);
  }
  if (mid) {
    params['mid'] = mid;
  }
  var wikiOptions = {stat: ['wkview.js' ,'wkview.css', 'apps.js', 'apps.css']};
  return showWiki(params, false, ev, wikiOptions);
}

// Показ сообщения по центру. 
function showDoneBox(msg, opts) {
  opts = opts || {};
  var l = (opts.w || 380) + 20;
  var style = opts.w ? ' style="width: ' + opts.w + 'px;"' : '';
  var pageW = bodyNode.offsetWidth,
      resEl = ce('div', {
      className: 'top_result_baloon_wrap fixed',
      innerHTML: '<div class="top_result_baloon"' + style + '>' + msg + '</div>'
  }, {left: (pageW - l) / 2});
  bodyNode.insertBefore(resEl, pageNode);
  boxRefreshCoords(resEl);
  var out = opts.out || 2000;
  var _fadeOut = function() {
    setTimeout(function() {
      if (opts.permit && !opts.permit()) {
        _fadeOut();
        return;
      }
      fadeOut(resEl.firstChild, 500, function() {
        re(resEl);
        if (opts.callback) {
          opts.callback();
        }
      });
    }, out);
  }
  _fadeOut();
}

function animateCount(el, newCount, opts) {
  el = ge(el);
  opts = opts || {};

  if (opts.str) {
    newCount = trim(newCount.toString()) || '';
  } else {
    newCount = positive(newCount);
  }
  if (!el) return;
  if (browser.msie6 || browser.mobile && !browser.safari_mobile && !browser.android) {
    val(el, newCount || '');
    return;
  }

  var curCount = data(el, 'curCount'),
      nextCount = data(el, 'nextCount');

  if (typeof nextCount == 'number' || opts.str && typeof nextCount == 'string') {
    if (newCount != nextCount) {
      data(el, 'nextCount', newCount);
    }
    return;
  }
  if (typeof curCount == 'number' || opts.str && typeof curCount == 'string') {
    if (newCount != curCount) {
      data(el, 'nextCount', newCount);
    }
    return;
  }
  if (opts.str) {
    curCount = trim(val(el).toString()) || '';
  } else {
    curCount = positive(val(el));
  }
  if (opts.str === 'auto') {
    opts.str = !curCount.match(/^d+$/) || !newCount.match(/^d+$/);
    if (!opts.str) {
      curCount = positive(curCount);
      newCount = positive(newCount);
    }
  }
  if (curCount == newCount) {
    return;
  }
  data(el, 'curCount', newCount);
  var incr = opts.str ? (curCount.length == newCount.length ? curCount < newCount : curCount.length < newCount.length) : curCount < newCount,
      big = (incr ? newCount : curCount).toString(),
      small = (incr ? curCount : newCount).toString(),
      constPart = [],
      constEndPart = [],
      bigPart = '',
      smallPart = '',
      i, l, j;

  if (!opts.str) {
    small = ((new Array(big.length - small.length + 1)).join('0')) + small;
  }
  for (i = 0, l = big.length; i < l; i++) {
    if ((j = big.charAt(i)) !== small.charAt(i)) {
      break;
    }
    constPart.push(j);
  }
  bigPart = big.substr(i);
  smallPart = small.substr(i);

  if (opts.str) {
    for (i = bigPart.length; i > 0; i--) {
      if ((j = bigPart.charAt(i)) !== smallPart.charAt(i)) {
        break;
      }
      constEndPart.unshift(j);
    }
    if (constEndPart.length) {
      bigPart = bigPart.substr(0, i + 1);
      smallPart = smallPart.substr(0, i + 1);
    }
  }

  constPart = constPart.join('').replace(/s$/, '&nbsp;');
  constEndPart = constEndPart.join('').replace(/^s/, '&nbsp;');

  if (!trim(val(el))) {
    val(el, '&nbsp;');
  }
  var h = el.clientHeight || el.offsetHeight;
  val(el, '<div class="counter_wrap inl_bl"></div>');
  var wrapEl = el.firstChild,
      constEl1, constEl2, animwrapEl, animEl,
      vert = true;

  if (constPart.length) {
    wrapEl.appendChild(constEl1 = ce('div', {className: 'counter_const inl_bl', innerHTML: constPart}));
  }
  if (!constPart.length) {
    smallPart = smallPart.replace(/^0+/, '');
  }
  if (!smallPart || smallPart == '0' && !constPart.length) {
    smallPart = '&nbsp;';
    vert = constPart.length ? true : false;
  }

  wrapEl.appendChild(animwrapEl = ce('div', {className: 'counter_anim_wrap inl_bl'}));
  animwrapEl.appendChild(animEl = ce('div', {
    className: 'counter_anim ' + (incr ? 'counter_anim_inc' : 'counter_anim_dec'),
    innerHTML: '<div class="counter_anim_big"><span class="counter_anim_big_c">' + bigPart + '</span></div>' +
               (vert ? '<div class="counter_anim_small"><span class="counter_anim_small_c">' + smallPart + '</span></div>' : '')
  }, vert ? {marginTop: incr ? -h : 0} : {right: 'auto', left: 0}));
  if (opts.str) {
    setStyle(animEl, {textAlign: 'left', right: 'auto', left: 0});
  }

  var bigW = geByClass1('counter_anim_big_c', animEl, 'span').offsetWidth,
      smallW = vert ? (smallPart == '&nbsp;' ? bigW : geByClass1('counter_anim_small_c', animEl, 'span').offsetWidth) : 0;

  if (constEndPart.length) {
    wrapEl.appendChild(constEl2 = ce('div', {className: 'counter_const inl_bl', innerHTML: constEndPart}));
  }

  setStyle(wrapEl, {width: (constEl1 && constEl1.offsetWidth || 0) + (constEl2 && constEl2.offsetWidth || 0) + bigW + 0})

  if (browser.csstransitions === undefined) {
    var b = browser, bv = floatval(b.version);
    browser.csstransitions =
      (b.chrome && bv >= 9.0) ||
     (b.mozilla && bv >= 4.0) ||
     (b.opera && bv >= 10.5) ||
     (b.safari && bv >= 3.2) ||
     (b.safari_mobile) ||
     (b.android);
  }
  var css3 = browser.csstransitions;
  setStyle(animwrapEl, {width: incr ? smallW : bigW});

  // return debugLog(css3, incr, curCount, newCount, animwrapEl, animEl, geByClass1('counter_anim_big_c', animEl, 'span'), geByClass1('counter_anim_small_c', animEl, 'span'), h, bigW, smallW);
  var onDone = function () {
    val(el, newCount || ' ');
    var next = data(el, 'nextCount');
    data(el, 'curCount', false);
    data(el, 'nextCount', false);
    if (typeof next == 'number' || opts.str && typeof next == 'string') {
      setTimeout(animateCount.pbind(el, next, opts), 0);
    }
    opts.onDone && opts.onDone();
  }, margin = vert ? {marginTop: incr ? 0 : -h} : {marginRight: incr ? -smallW : 0};
  if (css3) {
    getStyle(animwrapEl, 'width');
    addClass(animwrapEl, 'counter_css_anim_wrap');
    if (bigW != smallW) {
      setStyle(animwrapEl, {width: incr ? bigW : smallW});
    }
    if (vert) setStyle(animEl, margin);
    setTimeout(onDone, 300);
  } else {
    if (bigW != smallW) {
      animate(animwrapEl, {width: incr ? bigW : smallW}, {duration: 100});
    }
    if (vert) {
      animate(animEl, margin, {duration: 300, transition: Fx.Transitions.easeOutCirc, onComplete: onDone});
    } else {
      setTimeout(onDone, 300);
    }
  }
}

TopSearch = {
  cache: {},
  lists: {},
  init: function(){
    if (this.inited) return false;
    var tsInput = ge('ts_input'),
        tsInputWrap = ge('ts_input_wrap'),
        tsWrap = ge('ts_wrap'),
        tsCont = ge('ts_cont_wrap'),
        tsOnline = ge('ts_friends_online');
    extend(cur, {
      onChatFriendsUpdated: this.onChatFriendsUpdated
    });
    if (!tsInput) return false;
    hide(tsOnline);
    tsOnline.onclick = function() {
      if (curFastChat.clistBox.visible)
        FastChat.clistHide();
      else
        FastChat.clistShow();
    }
    tsOnline.onmouseover = function() {
      showTooltip(tsOnline, {text: getLang('head_fr_online_tip'), shift: [11,0,3], showdt: 0, black: 1});
    }
    addEvent(tsInput, 'focus', function() {
      TopSearch.deselect();
      if (trim(val(this))) addClass(tsCont.firstChild, 'active');
      TopSearch.toggleInput(true);
    });
    addEvent(tsInput, 'keydown', function(e) {
      switch (e.keyCode) {
        case KEY.DOWN:
        case KEY.UP:
          TopSearch.moveSelection(e.keyCode);
          cancelEvent(e);
          break;
        case KEY.ENTER:
          var curLink = geByClass1('active', tsCont);
          if (curLink) TopSearch.select(curLink, e);
          else {
            var q = trim(val(this));
            if (q) {
              tsInput.blur();
              hide(tsWrap);
              TopSearch.tsNeedsClear = true;
              nav.go('/search?c[section]=auto&c[q]='+encodeURIComponent(q));
            }
          }
          break;
        case KEY.ESC:
          TopSearch.clear();
          TopSearch.toggleInput(false);
          break;
      }
    });
    addEvent(tsInput, 'keyup', function(e) {
      switch (e.keyCode) {
        case KEY.DOWN:
        case KEY.UP:
        case KEY.ENTER:
        case KEY.ESC:
          cancelEvent(e);
          break;
        default:
          TopSearch.prepareRows(trim(val(this)));
          break;
      }
    });
    addEvent(tsInput, 'paste', function() {
      setTimeout(function() {
        TopSearch.prepareRows(trim(val(tsInput)));
      }, 10);
    });
    addEvent(document, 'mousedown', function(e) { // removed touchstart because of feed lags @izhukov
      if (!isVisible('ts_wrap') && hasClass(ge('ts_wrap'), 'vk')) TopSearch.toggleSettings();
      tsInput.blur();
      TopSearch.toggleInput(false);
    });
    this.inited = true;
  },
  onChatFriendsUpdated: function(onlineCount){
    var el = ge('ts_friends_online');
    if (!el) return;
    if (onlineCount > 0) {
      if (!isVisible(el)) {
        val(el, onlineCount);
      } else {
        animateCount(el, onlineCount, {right: 0});
      }
      show(el);
    } else {
      hide(el);
    }
  },
  clear: function() {
    var tsInput = ge('ts_input');
    tsInput.setValue('');
    tsInput.blur();
    tsInput.phonblur();
    this.prepareRows();
  },
  select: function(el, event, peer) {
    if (checkEvent(event)) return true;
    var tsWrap = ge('ts_cont_wrap'),
        tsInput = ge('ts_input'),
        tsInputWrap = ge('ts_input_wrap'),
        tsInputLength = trim(val(tsInput)).length,
        hintType = el.getAttribute('hinttype');
    if (!tsInputLength) {
      tsInput.blur();
      this.toggleInput(false);
    }
    if (peer && hasClass(event.target, 'ts_contact_status')) {
      ajax.post('al_search.php', {act: 'save_metrics', ql: tsInputLength, mk: 'chat_box'});
      this.writeBox(peer);
      this.clear();
      this.toggleInput(false);
      return false;
    }
    hide(tsWrap);
    this.tsNeedsClear = true;

    var res = nav.go(el, event);
    ajax.post('al_search.php', {act: 'save_metrics', ql: tsInputLength, mk: hintType});
    return res;
  },
  deselect: function() {
    var tsWrap = ge('ts_cont_wrap');
    each(geByClass('active', tsWrap), function(i, v) {removeClass(v, 'active')});
  },
  itemOver: function(el, state, event) {
    if (state == 1) TopSearch.deselect();
    var canWrite = inArray(el.getAttribute('hintType'), ['h_friends', 'h_correspondents', 'h_chats']);
    toggleClass(el, 'write', canWrite);
    if (state) addClass(el, 'active');
  },
  moveSelection: function(key) {
    var tsWrap = ge('ts_cont_wrap'), curLink = geByClass1('active', tsWrap), newLink;
    switch (key) {
      case KEY.UP:
        newLink = curLink ? this.getNextNode(curLink, -1, 'a') || curLink : false;
        break;
      case KEY.DOWN:
        newLink = curLink ? this.getNextNode(curLink, 1, 'a') || curLink : tsWrap.firstChild;
        break;
    }
    this.deselect();
    if (newLink) {
      addClass(newLink, 'active');
    }
    return false;
  },
  getNextNode: function(node, dir, tag) {
    var curNode = node;
    while (1) {
      curNode = dir > 0 ? curNode.nextSibling : curNode.previousSibling;
      if (!curNode) {
        return false;
      } else if (tag && curNode.tagName && curNode.tagName.toLowerCase() == tag || !tag && curNode) {
        return curNode;
      }
    }
  },
  highlightInput: function(s) {
    if (true || !hasClass(ge('ts_wrap'), 'vk')) return false;
    var tsCont = ge('ts_cont_wrap'),
        tsInput = ge('ts_input'),
        tsInputWrap = ge('ts_input_wrap');
    if (!isVisible(tsCont) && !trim(val(tsInput)) && !TopSearch.animationStarted) {
      if (s) {
        animate(tsInputWrap, {opacity: 0.65}, {duration: 300, transition: Fx.Transitions.easeOutCubic});
      } else {
        animate(tsInputWrap, {opacity: 0.5}, {duration: 300, transition: Fx.Transitions.easeOutCubic});
      }
    }
  },
  toggleInput: function(s, noAnimation) {
    if (!hasClass(ge('ts_wrap'), 'vk')) {
      toggle('ts_cont_wrap', s);
      // toggleClass(geByClass1('ts', ge('ts_input_wrap')), 'dark', !s);
      return;
    }
    var tsCont = ge('ts_cont_wrap'),
        tsInput = ge('ts_input'),
        tsInputWrap = ge('ts_input_wrap'),
        duration = noAnimation ? 0 : 150;
    TopSearch.animationStarted = 1;
    if (s) {
      animate(tsInputWrap, {opacity: 1, width: '170px'}, {duration: duration, transition: Fx.Transitions.easeOutCubic, onComplete: function() {
        show(tsCont);
        delete TopSearch.animationStarted;
      }});
      if (vk.rtl) {
        var newBG = browser.msie8 || browser.msie7 ? {backgroundPositionX: '152px'} : {backgroundPosition: '152px 5px'};
        animate(tsInput, newBG, {duration: duration, transition: Fx.Transitions.easeOutCubic});
      }
      if (cur.onTsShow) {
        cur.onTsShow();
      }
    } else {
      hide(tsCont);
      if (!trim(val(tsInput))) {
        animate(tsInputWrap, {width: '130px'}, {duration: duration, transition: Fx.Transitions.easeOutCubic, onComplete: function() {
          delete TopSearch.animationStarted;
        }});
        if (vk.rtl) {
          var newBG = browser.msie8 || browser.msie7 ? {backgroundPositionX: '112px'} : {backgroundPosition: '112px 5px'};
          animate(tsInput, newBG, {duration: duration, transition: Fx.Transitions.easeOutCubic});
        }
        if (cur.onTsHide) {
          cur.onTsHide();
        }
      }
    }
  },
  toggleSettings: function() {
    var tsSet = ge('ts_settings'),
        tsWrap = ge('ts_wrap'),
        tsInput = ge('ts_input'),
        s = isVisible(tsWrap),
        oldIE = browser.msie && (intval(browser.version) < 9);
    if (s) {
      addClass(tsSet, 'active');
      if (oldIE) {
        setStyle(tsSet, {opacity: 1});
        hide(tsWrap);
        cur.tsUpdated && cur.tsUpdated();
      } else {
        animate(tsSet, {opacity: 1}, 200);
        fadeOut(tsWrap, 200, cur.tsUpdated);
      }
      tsInput.blur();
      this.toggleInput(false);
    } else {
      removeClass(tsSet, 'active');
      if (oldIE) {
        setStyle(tsSet, {opacity: 0.7});
        show(tsWrap);
        cur.tsUpdated && cur.tsUpdated();
      } else {
        animate(tsSet, {opacity: 0.7}, 200);
        fadeIn(tsWrap, 200, cur.tsUpdated);
      }
    }
  },
  toggleTsWrap: function(s) {
    var tsWrap = ge('ts_wrap'), isVis = isVisible(tsWrap);
    clearTimeout(window.toggleTsWrapTO);
    if (s && !isVis) {
      window.toggleTsWrapTO = setTimeout(fadeIn.pbind(tsWrap, 100), 150);
    } else if (!s && isVis) {
      window.toggleTsWrapTO = setTimeout(fadeOut.pbind(tsWrap, 100), 150);
    }
  },
  settingsOver: function() {
    var tsSet = ge('ts_settings');
    if (!hasClass(tsSet, 'active')) {
      animate(tsSet, {opacity: 1}, 200);
    }
  },
  settingsOut: function() {
    var tsSet = ge('ts_settings');
    if (!hasClass(tsSet, 'active')) {
      animate(tsSet, {opacity: 0.7}, 200);
    }
  },
  getList: function(type) {
    switch (type) {
      case 'friends':
        return (this.lists['friends'] || /*window.curFastChat && curFastChat.friends||*/ this.topFriends || {});
      case 'publics':
      case 'events':
      case 'groups':
      case 'apps':
      case 'search':
        return this.lists[type] || {};
    }
    return {};
  },
  onlines: function() {
    return (window.curFastChat && curFastChat.onlines || this.lists['onlines'] || {});
  },
  initFriendsList: function() {
    if (this.friendsLoaded) return false;
    if (cur.initingFL || vk.isBanned) return false;
    var isEmpty = function(obj){
      for(var i in obj){return false;}
      return true;
    }
    var getTopFriends = function() {
      cur.initingFL = true;
      ajax.post('al_search.php', {act: 'get_top_friends'}, {
        cache: 1,
        onDone: function(friends) {
          delete cur.initingFL;
          TopSearch.topFriends = friends;
          TopSearch.updateCache('friends');
          if (!cur.tsStr) TopSearch.prepareRows();
          getAllPages();
        },
        onFail: function() {
          delete cur.initingFL;
        }
      });
    }
    var getAllPages = function() {
      if (TopSearch.friendsLoaded) return false;
      cur.initingFL = true;
      ajax.post('al_search.php', {act: 'get_pages'}, {
        cache: 1,
        onDone: function(res) {
          delete cur.initingFL;
          if (!TopSearch.friendsLoaded) {
            each(res, function(i, v) {
              TopSearch.lists[i] = v;
              if (i != 'onlines') TopSearch.updateCache(i);
            });
            TopSearch.friendsLoaded = true;
          }
        },
        onFail: function() {
          delete cur.initingFL;
        }
      });
    }
    var fr = this.getList('friends');
    if (isEmpty(fr)) getTopFriends();
    else {
      TopSearch.updateCache('friends');
      if (!cur.tsStr) TopSearch.prepareRows();
      getAllPages();
    }
  },
  parseLatKeys: function(text) {
    var outtext = text, i;
        lat = "qwertyuiop[]asdfghjkl;'zxcvbnm,./`",
        rus = "йцукенгшщзхъфывапролджэячсмитьбю.ё";
    for (i = 0; i < lat.length; i++) {
      outtext = outtext.split(lat.charAt(i)).join(rus.charAt(i));
    }
    return (outtext == text) ? false : outtext;
  },
  parseCyr: function(text) {
    var outtext = text, i,
        lat1 = ['yo','zh','kh','ts','ch','sch','shch','sh','eh','yu','ya','YO','ZH','KH','TS','CH','SCH','SHCH','SH','EH','YU','YA',"'"],
        rus1 = ['ё', 'ж', 'х', 'ц', 'ч', 'щ',  'щ',   'ш', 'э', 'ю', 'я', 'Ё', 'Ж', 'Х', 'Ц', 'Ч', 'Щ',  'Щ',   'Ш', 'Э', 'Ю', 'Я', 'ь'],
        lat2 = 'abvgdezijklmnoprstufhcyABVGDEZIJKLMNOPRSTUFHCYёЁ',
        rus2 = 'абвгдезийклмнопрстуфхцыАБВГДЕЗИЙКЛМНОПРСТУФХЦЫеЕ';
    for (i = 0; i < rus1.length; i++) {
      outtext = outtext.split(rus1[i]).join(lat1[i]);
    }
    for (i = 0; i < rus2.length; i++) {
      outtext = outtext.split(rus2.charAt(i)).join(lat2.charAt(i));
    }
    return (outtext == text) ? false : outtext;
  },

  getSimilarQueries: function(q) {
    q = q.toLowerCase();
    var queries = [q], t;
    if (t = parseLatin(q)) queries.push(t);
    if (t = TopSearch.parseLatKeys(q)) queries.push(t);
    if (t = TopSearch.parseCyr(q)) queries.push(t);
    return queries;
  },
  searchCache: function(listName, q) {
    var list = TopSearch.getList(listName);
    if (!q) return false;
    var queries = this.getSimilarQueries(q), query, t, i, j, cached, name, re, fr, cache;
    if (this.cache[listName][q] !== undefined) return queries;
    cache = this.cache[listName][q] = {};
    for (i in queries) {
      query = queries[i];
      if (cached = this.cache[listName][' ' + query.charAt(0).toLowerCase()]) {
        re = new RegExp('(^|\s|\()' + escapeRE(query), 'gi');
        for (j in cached) {
          fr = list[j + '_'];
          if (!isArray(fr)) continue;
          if (fr[0].match(re) !== null) cache[j] = 1;
        }
      }
    }
    j = 0;
    for (i in cache) j++;
    cache._num = j;
    return queries;
  },
  updateCache: function(listName, items) {
    var list = items || this.getList(listName);
    var name, cursor, letter;
    this.cache[listName] = this.cache[listName] || {};
    for (i in list) {
      name = list[i][0];
      i = intval(i);
      cursor = 0;
      while (1) {
        letter = ' ' + name.charAt(cursor).toLowerCase();
        this.cache[listName][letter] = this.cache[listName][letter] || {};
        this.cache[listName][letter][i] = 1;
        cursor = name.indexOf(' ', cursor + 1);
        if (cursor == -1) break;
        ++cursor;
      }
    }
  },
  listSearch: function(listName, q, limit, excludeIds) {
    var results = [], re = false, filterList = {}, listCache = listName + 'Cache';
    if (q) {
      TopSearch.searchCache(listName, q);
      filterList = this.cache[listName] && this.cache[listName][q] || {};
    } else {
      each (this.getList(listName), function(k) {
        var mid = intval(k); filterList[mid] = 1;
      });
    }
    // Why should we walk through the whole list?
    each (this.getList(listName), function(k) {
      var mid = intval(k), matches = filterList[mid];
      if (excludeIds && excludeIds[mid]) return;
      if (!matches) return;
      if (!(limit--)) return false;
      results.push([mid, this]);
    });
    return results;
  },
  row: function(mid, href, photo, name, online, re, hintType, info) {
    var peer = 0, typeAttr = '';
    if (re) name = name.replace(re, '$1<em class="ts_clist_hl">$2</em>');
    if (inArray(hintType, ['h_friends', 'h_correspondents', 'h_chats'])) {
      peer = mid;
    }
    if (!info) info = '';
    return '<a href="' + href + '" class="ts_contact ' + (online ? (online != 1 ? 'ts_contact_mobile' : 'ts_contact_online') : '') +' clear_fix" id="ts_contact' + mid + '" onclick="return TopSearch.select(this, event, '+peer+');" onmousedown="event.cancelBubble = true;" onmouseover="TopSearch.itemOver(this, 1, event);"  onmouseout="TopSearch.itemOver(this, 0, event);" hinttype="'+hintType+'"><span class="ts_contact_photo fl_l"><img src="' + photo + '"/></span><span class="ts_contact_name fl_l">' + name + '<div class="ts_contact_info">'+info+'</div></span><div class="ts_contact_status"></div></a>';
  },
  rowSearchSep: function() {
    return '<div id="ts_search_sep">'+getLang('head_search_results')+'</div>';
  },
  prepareRows: function(q) {
    var html = [],
        limit = total = 8,
        filterList = {},
        tsWrap = ge('ts_cont_wrap'),
        curLink = geByClass1('active', tsWrap),
        activeId = curLink ? curLink.id  : '', searchLists = {}, noSep = false;
    if (!tsWrap) return false;
    if (cur.tsStr && cur.tsStr == q) return false;

    var rowsNum = 0, excludeIds = {}, re;
    if (q) {
      re = [];
      each(this.getSimilarQueries(q), function() {
        re.push(escapeRE(this));
      });
      re = new RegExp("([ -]|^|s|&nbsp;|b)(" + re.join('|') + ")", "gi");
      cur.lastRe = re;
      limit--;
      searchLists = {apps: 1, events: 1, publics: 2, groups: 4, friends: 10, search: 10};
    } else {
      searchLists = {friends: 10};
    }

    each(searchLists, function(i, v) {
      if (!TopSearch.cache[i]) {
        return;
      }
      var results = TopSearch.listSearch(i, q, v, excludeIds), items = [], row;
      if (isEmpty(results)) return;
      for (var j in results) {
        if (!limit || limit < total - v) break;
        items.push(results[j]);
        limit--;
        rowsNum++;
      }
      if (!items.length) return;
      if (i != 'search') items.reverse();

      for (var j in items) {
        var item = items[j][1], mid = intval(items[j][0]);
        if (i == 'search' && !noSep && (item[3].substr(0, 2) == 's_' || item[3] == 'h_mutual_friends') && html.length > 0) {
          html.push(TopSearch.rowSearchSep());
          noSep = true;
        } else if (i == 'search' && !html.length) {
          noSep = true;
        }

        var online = mid > 0 ? TopSearch.onlines()[mid] : false,
          name = item[0], href = item[2],
          info = item[4],
          type = (i == 'search') ? item[3] : 'h_'+i;
          excludeIds[mid] = 1;

        row = TopSearch.row(mid, href, item[1], name, online, re, type, info);

        (i == 'search') ? html.push(row) : html.unshift(row);
      }

    });
    if (q) {
      if (!html.length) {
        noSep = true;
      }
      var show_q = (q.length > 27) ? q.substr(0, 25) + '..' : q,
          langKey = (q[0] == '#') ? getLang('top_search_news') : getLang('top_search');
      html.unshift('<a href="/search?c[section]=auto&c[q]='+encodeURIComponent(q)+'" class="ts_search_link clear_fix active" id="ts_search_link" onclick="return TopSearch.select(this, event);" onmousedown="event.cancelBubble = true;" onmouseover="TopSearch.itemOver(this, 1, event);"><span class="ts_contact_name fl_l">'+langKey.replace('{query}', '<span class="ts_query">'+clean(show_q)+'</span>')+'</span><div class="ts_contact_status "></div></a>');
    }

    TopSearch.friendsRows = tsWrap.innerHTML = html.join('');
    if (window.tsHintsEnabled) {
      if (limit && q && q[0] != '#') {
        this.hintsSearch(q, cur.lastRe || false, limit, noSep);
      }
    }
    if (html || q) cur.tsStr = q;
    if (activeId && ge(activeId)) addClass(ge(activeId), 'active');
  },
  hintsSearch: function(q, re, limit, noSep) {
    var tsInput = ge('ts_input'), tsWrap = ge('ts_cont_wrap'), curLink, activeId;
    ajax.post('al_search.php', {act: 'get_pages_hints', q: q}, {
      cache: 1,
      onDone: function(pages) {
        if (trim(val(tsInput)) != q) {
          return false;
        }
        if (!pages) {
          return false;
        }
        var rows = [], hasSep = ge('ts_search_sep') ? true : false;

        TopSearch.lists['search'] = TopSearch.lists['search'] || {};
        each (pages, function(k) {
          TopSearch.lists['search'][k] = this;
          var mid = intval(k), name = this[0], href = this[2], hintType = this[3], info = this[4];
          if (ge('ts_contact'+mid)) return true;
          if (!(limit--)) return false;
          if (!noSep && (hintType.substr(0, 2) == 's_' || hintType == 'h_mutual_friends')) {
            rows.push(TopSearch.rowSearchSep());
            noSep = true;
          }
          rows.push(TopSearch.row(mid, href, this[1], name, false, re, hintType, info));
          return true;
        });
        TopSearch.updateCache('search', pages);
        if (rows.length) {
          curLink = geByClass1('active', tsWrap);
          activeId = curLink ? curLink.id  : '';
          tsWrap.innerHTML += rows.join('');
          if (activeId && ge(activeId)) addClass(ge(activeId), 'active');
        }
      }
    });
  },
  writeBox: function(peer) {
    if (window.curFastChat && curFastChat.inited && window.FastChat) { // black movable rbox
      FastChat.selectPeer(peer);
    } else if (peer > 0 && peer < 2e9) {
      showWriteMessageBox(false, peer);
    } else { // no box for multichats or emails
      nav.go('/im?sel=' + peer);
    }
  }
}

function mentionOver(el) {
  showTooltip(el, {
    url: 'al_wall.php',
    params: {act: 'mention_tt', mention: el.getAttribute('mention_id'), from: 'wall'},
    shift: [65, 7, 7],
    hidedt: 500,
    showdt: 500,
    slide: 15
  });
}

function mentionClick(el, ev) {
  if (el && el.tt && el.tt.hide) {
    el.tt.hide();
  }
  return nav.go(el, ev);
}

function headPlayPause(event) {
  var aid = currentAudioId();
  if (!window.audioPadShown && !aid) Pads.show('mus');
  if (!aid) {
    aid = ls.get('audio_id');
    if (aid) {
      window.padPlClicked = true;
    }
  }
  if (aid) {
    playAudioNew(aid);
  } else {
    var plist = padAudioPlaylist();
    if (plist && plist.start) {
      playAudioNew(plist.start);
    } else {
      addClass(ge('head_play_btn'), 'playing');
      window.onPlaylistLoaded = function() {
        var plist = padAudioPlaylist();
        if (plist && plist.start) {
          playAudioNew(plist.start);
        }
      }
    }
  }
  if (event) cancelEvent(event);
}

function currentAudioId() {
  // return ls.get('audio_id') || window.audioPlayer && audioPlayer.id;
  return window.audioPlayer && audioPlayer.id;
}

function padAudioPlaylist() {
  return window.audioPlaylist || ls.get('pad_playlist');
}

var _postsSeen = {}, _postsSaved = {}, _postsSaveTimer, _postsSendTimer, _postsCleanTimer;
var ls = {
  checkVersion: function() {
    return (window.localStorage !== undefined && window.JSON !== undefined);
  },
  set: function(k, v) {
    this.remove(k);
    try {
      return (ls.checkVersion()) ? localStorage.setItem(k, JSON.stringify(v)) : false;
    } catch (e) {
      return false;
    }
  },
  get: function(k) {
    if (!ls.checkVersion()) {
      return false;
    }
    try {
      return JSON.parse(localStorage.getItem(k));
    } catch (e) {
      return false;
    };
  },
  remove: function(k) {
    try { localStorage.removeItem(k); } catch(e) {};
  }
}

var mobilePromo = showBox.pbind('al_login.php', {act: 'mobile', box: 1});
function mobileOnlineTip(el, opts) {
  var sh = (opts.right ? 278 : 35) + (browser.opera ? 1 : 0);
  return showTooltip(el, {
    url: 'al_login.php',
    params: {act: 'mobile_tt', mid: opts.mid, was: opts.was},
    slide: 15,
    ajxdt: 200,
    showdt: 200,
    hidedt: 200,
    toup: false,
    shift: [sh, 0, 0],
    className: 'mobile_tt' + (opts.right ? ' mobile_tt_right' : '')
  });
}

var Pads = {
  preload: function(id) {
    var st = ['pads.css', 'pads.js', 'notifier.js'];
    if (id == 'msg') st.push('pads_im.css', 'pads_im.js', 'page.css', 'page.js');
    stManager.add(st, function() { Pads.preload(id); });
  },
  show: function(id, ev) {
    if (checkEvent(ev) === true || browser.msie6) return;
    var st = ['pads.css', 'pads.js', 'notifier.js'];
    if (id == 'msg') st.push('pads_im.css', 'pads_im.js', 'page.css', 'page.js');
    stManager.add(st, function() { Pads.show(id); });
    return cancelEvent(ev);
  },
  invalidate: __bf
};

function cssAnim(obj, prep, opts, callb) {
  var v = intval(browser.version);
  if (obj && ((browser.chrome && v > 14) || (browser.mozilla && v > 13) || (browser.opera && v > 2))) {
    var callbWrap;
    var st = 'all '+opts.duration+'ms '+(opts.func || 'ease-out');
    obj.style.WebkitTransition = st;
    obj.style.MozTransition = st;
    obj.style.OTransition = st;
    obj.style.transition = st;
    var callbWrap = function() {
      if (browser.opera) {
        obj.removeEventListener('oTransitionEnd', callbWrap);
      } else {
        removeEvent(obj, 'webkitTransitionEnd transitionend msTransitionEnd oTransitionEnd', callbWrap);
      }
      obj.style.WebkitTransition = '';
      obj.style.MozTransition = '';
      obj.style.OTransition = '';
      obj.style.transition = '';
      if (callb) callb();
      return false;
    }
    if (callb) {
      if (browser.opera) {
        obj.addEventListener('oTransitionEnd', callbWrap)
      } else {
        addEvent(obj, 'webkitTransitionEnd transitionend msTransitionEnd oTransitionEnd', callbWrap);
      }
    }
    setTimeout(setStyle.pbind(obj, prep), 0);
  } else {
    animate(obj, prep, extend(opts, {onComplete: callb}));
  }
}

try{stManager.done('common.js');}catch(e){}
?>
Онлайн: 0
Реклама