habrActivity

view user activity in Habr comments for Fx-Opera-Chrome

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

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

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

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

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

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @id             habrActivity
// @name           habrActivity
// @version        5.2014.12.31
// @author         spmbt0
// @description    view user activity in Habr comments for Fx-Opera-Chrome
// @include        http://habrahabr.ru/*
// @include        http://geektimes.ru/*
// @exclude        http://habrahabr.ru/api/*
// @update 4 win.console; remove panel from Geektimes articles;
// @icon 
// @namespace https://greasyfork.org/users/2323
// ==/UserScript==
(function(win, noConsole, css, hActHelp){
var d = document
	,dBody = d.body || d.documentElement
	,$q = function(q, f){return (f||d).querySelector(q)}
	,uHead = $q('.user_header')
	,uName2a = $q('.username a', uHead)
	,uName2 = uName2a && uName2a.innerHTML
	,comms = $q('.comments_list');
if(uName2 && comms && $q('.user_comments')){ //=====(всё работает только на странице комментариев некоторого пользователя)=====

try{ //для оповещения об ошибках в Fx
var NOWdate = new Date()
,lh = location.href
,HRU =location.protocol +'//'+ location.host
,lStorRoot ='habrAct_'
,setLocStor = function(name, hh){ if(!win.localStorage) return;
	localStorage[lStorRoot + name] = JSON.stringify({h: hh});
}
,getLocStor = function(name){
	return (JSON.parse(localStorage && localStorage[lStorRoot + name] ||'{"h":{}}')).h;
}
,removeLocStor = function(name){localStorage.removeItem(lStorRoot + name);}
,$qA = function(q, f){return (f||d).querySelectorAll(q)}
,$pd = function(ev){ev.preventDefault();}
,$sp = function(ev){ev.stopPropagation();}
,$x = function(el, h){ //===extend===
	if(h)
		for(var i in h)
			el[i] = h[i];
	return el;
}
,$e = function(g){ //===создать или использовать имеющийся элемент===
	//g={el|clone,IF+ifA,q|[q,el],cl|(clAdd,clRemove),ht,cs,at,on,revent,apT,prT,bef,aft,f+fA}
	if(g.ht || g.at){
		var at = g.at ||{}; if(g.ht) at.innerHTML = g.ht;}
	if(typeof g.IF =='function')
		g.IF = g.IF.apply(g, g.ifA ||[]);
	g.el = g.el || g.clone || g.IF && g.IF.attributes && g.IF ||'DIV';
	var o = g.clone && g.clone.cloneNode(!0)
			|| (typeof g.el =='string' ? d.createElement(g.el) : g.el);
	if(o && (g.IF===undefined || g.IF) && (!g.q || g.q && (g.dQ = g.q instanceof Array ? $q(g.q[0], g.q[1]) : $q(g.q)) ) ){ //выполнять, если существует; g.dQ - результат селектора для функций IF,f
		if(g.cl)
			o.className = g.cl;
		else{
			if(g.clAdd)
				o.classList.add(g.clAdd);
			if(g.clRemove)
				o.classList.remove(g.clRemove);
		}
		if(g.cs)
			$x(o.style, g.cs);
		if(at)
			for(var i in at){
				if(i=='innerHTML') o[i] = at[i];
				else o.setAttribute(i, at[i]);}
		if(g.on)
			for(var i in g.on)
				o.addEventListener(i, g.on[i],!1);
		if(g.revent)
			for(var i in g.revent)
				o.removeEventListener(i, g.revent[i],!1);
		g.apT && g.apT.appendChild(o); //ставится по ориентации, если новый
		g.prT && (g.prT.firstChild
			? g.prT.insertBefore(o, g.prT.firstChild)
			: g.prT.appendChild(o) );
		g.bef && g.bef.parentNode.insertBefore(o, g.bef);
		g.aft && (g.aft.nextSibling
			? g.aft.parentNode.insertBefore(o, g.aft.nextSibling)
			: g.aft.parentNode.appendChild(o) );
		if(typeof g.f =='function')
			g.f.apply(g, g.fA ||[]); //this - это g
	}
	return o;
}
,parents = function(cl, elem){
	for(var el = elem; el!=null && !RegExp(cl).test(el.className); el = el.parentNode);
	return el;
}
,prev = function(cl, elem){
	for(var el = elem; el!=null && !RegExp(cl).test(el.className); el = el.previousSibling);
	return el;
}
,next = function(cl, elem){
	for(var el = elem; el!=null && !RegExp(cl).test(el.className); el = el.nextSibling);
	return el;
}
,toTime = function(dat){
	var yestDate = new Date(+NOWdate - 86400000)
		,datMonth ={"января":0,"февраля":1,"марта":2,"апреля":3,"мая":4,"июня":5,
			"июля":6,"августа":7,"сентября":8,"октября":9,"ноября":10,"декабря":11}
		,datFull = dat.innerHTML
		,datYest = /вчера/.test(datFull)
		,dateText = datFull.replace(/ в /,' ').replace(/(сегодня |вчера )/,'')
		,datArr = dateText.match(/(\d+)\s+([а-яё]+)\s+(\d{4})?/i);
	if(!datArr)
		datArr =[0,(datYest ? yestDate : NOWdate).getDate(),(datYest ? yestDate : NOWdate).getMonth(),(datYest ? yestDate : NOWdate).getFullYear()];
	var altArr = dateText.match(/(\d+)\:(\d+),([а-яё]+)\s*(\d+)\s*([а-яё]+)\s*(\d{4})?/i); //ччммддЧЧММГГГГ?
	if(altArr)
		datArr =[0,altArr[4],altArr[5],altArr[6]];
	//'altArr'.wcl(altArr)
	var mon = datArr && datMonth[datArr[2]] || datMonth[datArr[2]] !=0 && datArr[2] || datMonth[datArr[2]];
	//'datArr'.wcl(datArr, mon)
	if(datArr && !datArr[3])
		datArr[3] = NOWdate.getFullYear();
	var ret2 = new Date(datArr[3], mon, datArr[1], dateText.replace(/(.*?)(\d{1,2}):(\d\d)(.*)/,'$2'), dateText.replace(/(.*?)(\d{1,2}):(\d\d)(.*)/,'$3') ).getTime();
	if(+NOWdate < ret2)
		ret2 = new Date(datArr[3] -1, mon, datArr[1], dateText.replace(/(.*?)(\d{1,2}):(\d\d)(.*)/,'$2'), dateText.replace(/(.*?)(\d{1,2}):(\d\d)(.*)/,'$3') ).getTime();
	return ret2;
}
,getYearWeekDay = function(dd){
	var d = new Date(dd), jan1 = new Date(d.getFullYear(),0,1);
	return [d.getFullYear(), Math.ceil(((dd - jan1) /86400000 + jan1.getDay() -1)/7)
		, d.getDay() + 7*(d.getDay()==0), d.getDate(), d.getMonth()];
}
,wcl = function(a){ a = a ||''; //консоль как метод строки или функция, с отключением по noConsole
	if(win.console && (!noConsole || this =='ER_global:'))
		win.console.log.apply(win.console, this instanceof String
			? ["'=="+ this +"'"].concat([].slice.call(arguments)) : arguments);
};
String.prototype.wcl = wcl;

var hActUsersTmpl ={ //шаблон основной (индексной) записи в хранилище
	dataLen: 0 //число записей в хранилище вида habrActDataNNNNN, где NNNNN - число
	,dataCount: 25 //огранчитель цикла чтения, страниц
	,dateStart: 1000 //ограничитель периода чтения, до 365 дней, сравнивает дату начала цикла и посл.
	,userS: []  //массив имён пользователей, для которых имеются данные
}
,readAutoIntervalMax = 365
,hActDataTmpl ={ //шаблон записи данных в хранилище; запись имеет вид habrActData[число], от 1 до dataLen
	user:'' //пользователь, для которого со страницы комментариев сняли данные
	,data: [] //массив дат (или сложнее)
};
(function(css){ var u = 'undefined'; //addRules
	if(typeof GM_addStyle !=u){ GM_addStyle(css);
	}else if(typeof addStyle !=u){ addStyle(css);
	}else{
		var node = d.createElement('style');
		node.type ='text/css';
		node.appendChild(d.createTextNode(css));
		(d.getElementsByTagName('head')[0] || dBody).appendChild(node);}
})(css);

var readPage
	,ww =0
	,startButt = $e({el:'button', ht:'Старт'
		,on:{click: readPage = function(ev){ //читать статистику со страницы и переходить к следующей
			if(ev){
				var handStart =1;
				hActUsers = getLocStor('users');
				hActUsers.userS = hActUsers.userS ||[];
				hActUsers.dataLen = hActUsers.dataLen ||0;
			}
			var infoS = $qA('.info', comms)
				,datC =[];
			clearTimeout(ww);
			if(startButt.innerHTML =='Стоп'){
				startButt.innerHTML ='Старт';
				hActUsers.dataCount =0;
				setLocStor('users', hActUsers);
				return;
			}
			hActUsers.dataLen++;
			startButt.innerHTML ='Стоп';
			if(handStart)
				hActUsers.dataCount = hActUsersTmpl.dataCount;
			for(var i in infoS){ var iI = infoS[i]; if(iI.attributes){
				var sco = $q('.voting .score',iI)
					,apm = sco && sco.title.match(/\d+/g)
					,text = $q('.message', iI.parentNode);
				datC.push(//{date: 
					toTime($q('time',iI))/1000
					//,plus: apm && apm[1]
					//,minus: apm && apm[2]
					//,textLen: text && text.innerHTML.replace(/\t/g,'').length}
				);
			}}
			var pageA = lh.match(/^.+?page(\d+).*/);
			'datC'.wcl(datC, pageA, ( + NOWdate/1000 - datC[datC.length -1]) /86400)
			$e({el: $q('.msg', startButt.parentNode) ||'span'
				,cl:'msg'
				,ht:'<br>&nbsp; До '+(0|((NOWdate/1000 - datC[datC.length -1]) /86400))
					+' дней от сегодня; интервал - '+ (0|((datC[0] - datC[datC.length -1]) /86400))+' дней'
				,bef: $q('.clear',uHead)
			});
			if(handStart)
				hActUsers.dateStart = datC[0];
			if((hActUsers.dateStart - datC[datC.length -1]) /86400 < readAutoIntervalMax)
				ww = setTimeout(function(){
					location.href = pageA
						? lh.replace(/page(\d+)/,'page'+ (+pageA[1] +1))
						: lh +'page2/';
				}, 3000);
			else{
				startButt.innerHTML ='Старт';
				hActUsers.dataCount =0;
			}
			if(!handStart)
				hActUsers.dataCount--;
			var noNewUser =0;
			for(var i in hActUsers.userS){ var uI = hActUsers.userS[i];
				if(uI == uName2){
					noNewUser =1; break;}
			}
			if(!noNewUser)
				hActUsers.userS.push(uName2);
			setLocStor('users', hActUsers); //накапливать статистику в хранилище
			wcl('2nd', hActUsers.dataLen);
			setLocStor('data'+ hActUsers.dataLen, {user: uName2, data: datC});
		}}
		,bef: $q('.clear', uHead)
	}),
	hActUsers = getLocStor('users');
hActUsers.userS = hActUsers.userS ||[];
hActUsers.dataLen = hActUsers.dataLen ||0;
if(hActUsers.dataCount >0){ //автозапуск анализа страницы
	wcl('1st');readPage();}

var helpHAct,
helpButt = $e({el:'button', ht:'Подробности'
	,on:{click: function(ev){ if(!helpHAct || helpHAct.style.display=='none'){
		if(!helpHAct){
			helpHAct = $e({cl:'helpHAct'
				,ht: hActHelp
				,on:{click: function(){this.style.display ='none';}}
				,apT: dBody
			});
			$q('.in', helpHAct).addEventListener('click', function(ev){$sp(ev);},!1);
		}else
			helpHAct.style.display ='block';
	}else helpHAct.style.display ='none'; } }
	,aft: startButt
}),
eraseButt = $e({el:'button', ht:'Стереть'
	,on:{click: function(ev){
		var hActUsers = getLocStor('users');
		if(confirm('Удалить всю статистику комментариев пользователей ('+ hActUsers.dataLen +' записей)?')){
			if(hActUsers.dataLen){
				for(var i = +hActUsers.dataLen; i >=1; i--)
					removeLocStor('data'+ i);
				removeLocStor('users');
			}
			$q('.diag', selUser) && $q('.diag', selUser).classList.add('empty');
		}
	}}
	,aft: startButt
}),
dataButt = $e({el:'button', ht:'Данные'
	,on:{click: function(ev){
		hActUsers = getLocStor('users');
		selectUser(showData);
		this.blur();
	}}
	,aft: startButt
}),
selUser,
selectUser = function(f){ //предложить выбор пользователей
	if(!selUser || selUser.style.display=='none'){
		if(!selUser){
			selUser = $e({cl:'selUser'
				,ht: '<div class=under></div><div class=in><h2><input class="inUser" title="перейти -видео на комментарии пользователя; Ctrl-Enter - в новом окне"><span class="titl">Выбрать пользователя</span></h2><div class="diag empty"></div></div>'
				,on:{click: function(){this.style.display ='none';}}
				,apT: dBody
			});
			$q('.in', selUser).addEventListener('click', function(ev){$sp(ev);},!1);
			$q('.inUser',selUser).addEventListener('keyup',function(ev){ if(ev.keyCode ==13){
				var lnk = HRU +'/users/'+ this.value +'/comments/';
				if(!ev.ctrlKey)
					location.href = lnk;
				else
					window.open(lnk,"_blank")
			}},!1);
		}else{
			selUser.style.display ='block';
			var sC = $q('.in', selUser).childNodes;
			for(var i = sC.length -1; i >=0; i--){ var sI = sC[i];
				'sI'.wcl(sI.tagName, sI.className)
				if(dI.attributes &&(sI.tagName =='BUTTON'|| sI.className =='hADel') )
					selUser.removeChild(sI);
			}
		}
		for(var i in hActUsers.userS){ var uI = hActUsers.userS[i];
			$e({el:'button'
				,ht: uI
				,on:{click: function(){
					var bS = $qA('button', this.parentNode);
					for(var j in bS) if(bS[j].attributes)
						bS[j].classList.remove('hAActive');
					this.classList.add('hAActive');
					f(this.innerHTML);
					this.blur();
				}}
				,bef: $q('.diag', selUser)
			});
			$e({el:'button',cl:'hADel'
				,ht:'X'
				,at:{title:'Удалить из хранилища','data-user': uI}
				,on:{click: function(){
					var t = this, u;
					del1user(u = this.getAttribute('data-user'));
					t.previousSibling.parentNode.removeChild(t.previousSibling);
					setTimeout(function(){t.parentNode.removeChild(t);},1);
					for(var i in hActUsers.userS){ var uI = hActUsers.userS[i];
						if(u == uI){
							hActUsers.userS.splice(i, 1); break;
					}}
					setLocStor('users', hActUsers);
				}}
				,bef: $q('.diag', selUser)
			});
		}
		var hU = hActUsers.userS;
		'hU'.wcl(hU,uName2)
		if(hU && hU.length ==1)
			f(hU[0]),$q('button',selUser).classList.add('hAActive');
		else if(hU && uName2){
			var bS = $qA('button', hU[0].parentNode);
			for(var i in bS) if(bS[i].attributes){
				bS[i].classList.remove('hAActive');
				if(bS[i].innerHTML == uName2)
					f(uName2),bS[i].classList.add('hAActive');
			}
		}
	}else selUser.style.display ='none';
},
showData = function(user){ //показать диаграмму активности
	var dA =[]
		,diag = $q('.diag', selUser)
		,dC = diag.childNodes;
	for(var i = dC.length -1; i >=0; i--){ var dI = dC[i]; if(dI.attributes)
		diag.removeChild(dI);}
	for(var i =1; i <= +hActUsers.dataLen; i++){ //всё ранее прочитанное по юзеру
		if(!RegExp(' '+ i +' ').test(hActUsers.removed ||'') ){
			var dat = getLocStor('data'+ i);
			if(user == dat.user)
				dA = dA.concat(dat.data);
	}}
	diag.style.height ='100px';
	dA.sort(function(a, b){return a - b});
	var ymd0 = getYearWeekDay(dA[0]*1000), ymdE =[] //год, неделя, день, число, месяц, кол-во комм.
		,diagLeftMargin =7
		,diagTopMargin =7
		,stripeN =1 //число полос для диаграммы
		,stripePeriod = 110
		,nComm =0
		,diagInnerWid = diag.offsetWidth -diagLeftMargin *2 -10 //ограничения ширины 1 полосы
		,wasDoubles =0, dAClean =[]; //слежение за дублями
	for(var i in dA){
		var ymd = getYearWeekDay(dA[i]*1000);
		ymd[5] = ymdE[0]==ymd[0] && ymdE[1]==ymd[1] && ymdE[2]==ymd[2] ? ymdE[5]+1 : 1;
		if(dA[i] == ymdE[6]){ //игнор дублей чтения
			wasDoubles =1; continue;}
		dAClean.push(dA[i]);
		nComm++;
		ymdE = [ymd[0], ymd[1], ymd[2], ymd[3], ymd[4], ymd[5], dA[i]];
		var leftFirstShift = ((ymd[0]-ymd0[0])*52 + (ymd[1]-ymd0[1]) )*10
			,iStripe = Math.floor(leftFirstShift / diagInnerWid);
		stripeN = Math.max(stripeN, iStripe +1);
		$e({cl:'comm'
			,cs:{left: (leftFirstShift % diagInnerWid) + diagLeftMargin +'px'
				,top: iStripe * stripePeriod + ymd[2] *10 + diagTopMargin +'px'}
			,at:{title: ymd[5] +' / '+ ymd[3]+'янвфевмарапрмайиюниюлавгсеноктноядек'.match(/.../g)[ymd[4]]+(ymd[0] +'').substr(2,2) }
			,apT: diag
		});
	}
	if(wasDoubles){ //удалить юзера и записать его же из очищенного от дублей массива dAClean
		del1user(user);
		wcl('delU')
		setLocStor('data'+ ++hActUsers.dataLen, {user: uName2, data: dAClean});
		setLocStor('users', hActUsers); //сохранить .removed
	}
	if(stripeN >1)
		diag.style.height = 110 * stripeN +'px';
	diag.classList.remove('empty');
	var hActTitle = '%N комментари%W <span title="%T">с %D0 по %D1</span>', days;
	if(ymd)
		$q('h2 .titl', selUser).innerHTML = hActTitle
			.replace(/%N/,nComm).replace(/%W/, nComm % 10 >0 && nComm % 10 <5 && Math.floor(nComm % 100 / 10) !=1 ? (nComm % 10 ==1 ?'й':'я'):'ев')
			.replace(/%D0/,ymd0[3] +'.'+ (ymd0[4]+1) +'.'+ (ymd0[0] +'').substr(2,2))
			.replace(/%D1/,ymd[3] +'.'+ (ymd[4]+1) +'.'+ (ymd[0] +'').substr(2,2))
			.replace('%T', Math.ceil(days = (dA[dA.length-1] - dA[0]) /86400) +' дней, '+ (nComm/days).toFixed(2) +' комм./д.' );
},
del1user = function(user){ //удалить данные 1 пользователя
	for(var i =1; i <= +hActUsers.dataLen; i++){ //всё ранее прочитанное по юзеру
		if(!RegExp(' '+ i +' ').test(hActUsers.removed ||'') ){
			var dat = getLocStor('data'+ i);
			if(user == dat.user){
				var dat = removeLocStor('data'+ i);
				hActUsers.removed = (hActUsers.removed ||' ')+ i +' ';
	}}}
};


}catch(er){
	'ER_global:'.wcl(er +' (line '+(er.lineNumber||'')+')')}; //для оповещения об ошибках в Fx
}} //=====/(конец основных операций)=====
)(typeof unsafeWindow !='undefined'? unsafeWindow: (function(){return this})(), 'noConsole',

	/*===== css =====*/

'.helpHAct,.selUser{position: absolute; z-index: 1201; top: 0; width: 100%; height: 100%;}'
+'.helpHAct .under,.selUser .under{position: fixed; z-index: 1200; width: 100%; height: 100%; opacity:0.1; background:#777}'
+'.helpHAct .in, .selUser .in{position: relative;  z-index: 1201;max-width: 48em; margin: 120px auto 280px; padding: 12px; border: 1px solid #bbb; box-shadow: 0 0 7px 4px #a4b39D/*#afb6af*/;  background: #fff;}'
+'.selUser .in .inUser{float: right; padding: 1px 1px 3px; border: 1px solid #ccc;}'
+'.selUser .in h2{margin-bottom: 6px;}'
+'.selUser .diag{position: relative; height: 100px; margin-top: 12px; background: url() 6px 0;}'
+'.selUser .comm{position: absolute; width: 8px; height: 8px; opacity: 0.2; background: #8b4;}'
+'.selUser .hAActive, .user_header .hAActive{background-color: #f6f8e0/*#fff2d8*/}'
+'.selUser button, .user_header button{height: 1.6em; margin: 10px 0 0 1px; padding: 0 4px 1px; line-height: 1.3em;'
	+'box-shadow: 0 0 2px rgba(255, 255, 255, 0.4) inset, 0 0 2px rgba(0, 0, 0, 0.2); transition-duration: 0.2s; text-shadow: 0 1px 0 rgba(255, 255, 255, 0.8); border-radius: 3px; border: 1px solid #e9d8e8; border-color: #e9DeD8 #bccbbb #9eae9e; background-color: #c8ead0/*#cfe8c4*/;}'
+'.selUser button{margin-top: 5px}.user_header .rating{margin-right: 14px;}'
+'.selUser button:hover,.user_header button:hover{background-color: #f2fddb/*#fcfcfc*/;}'
+'.selUser .diag.empty{background-image: url()}'
+'.selUser .hADel{position: relative; height: 14px; line-height: 1px; border-radius: 6px; margin:-0.6em; top:-0.9em; left: -3px; padding:0 0 1px; border: 1px solid #d99; font-size:10px; background-color: #ea7852; opacity: 0.15;}.selUser .hADel:hover{background-color: #E65E30; color: #fbb; opacity: 0.8}'
+'.selUser button:hover +.hADel{opacity: 0.5}',

	/*===== help string =====*/

'<div class=in><h2>Просмотр активности комментариев пользователей</h2><br><br>После установки юзерскрипта <b>habrActivity</b> на страницах комментариев пользователей (и только на них) появляются 4 кнопки: "Старт", "Данные", "Стереть", "Подробности".<br><br>'

+'При нажатии на "<b>Старт</b>" запускается цикл чтения данных с перебором страниц комментариев для данного пользователя, а кнопка меняет название на "Стоп". Прочитывается не более 25 страниц в автоматическом режиме и читается интервал комментариев за не более 365 дней. В любой момент цикл автоматического чтения страниц останавливается вручную кнопкой "<b>Стоп</b>". Если нужно читать больше 365 дней, чтение возобновляется по кнопке "Старт".<br><br>'

+'С каждой страницы собирается информация о датах создания комментариев и, возможно, впоследствии более сложная, и запоминается в хранилище. Таким образом, накапливается информация о комментариях разных пользователей в разные периоды времени. Хранятся только массивы чисел - например, даты написания комментариев. В последующих версиях возможно расширить статистику на сбор оценок, объёма текстов, количество ссылок и картинок и подобное.<br><br>'

+'Чтобы просмотреть накопленные данные, нажимают кнопку "<b>Данные</b>". Если в браузере на данный момент хранится информация о более 1 пользователе, появится список кнопок с именами пользователей. Нажав одну из кнопок, переходим на просмотр статистики.<br><br>'

+'Просмотр статистики по датам комментариев организован аналогично инфографике Гитхаба - показ активности комментариев пользователя по дням года и дням недели. Отображается не более 420 последних дней (60 недель). Данные подготовлены для "музыкальной иллюстрации активности" http://habrahabr.ru/post/173085/ .<br><br>'

+'В результате, получаются 3 полезные вещи: смотрим активность любого пользователя в комментариях, видим наглядный график, прослушиваем в задумчивости озвучивание его. Делаем выводы.<br><br>'

+'Чтобы стереть все данные пользователя, нажимается кнопка "<b>Стереть</b>". Память браузера очищается от накопленных данных.<br><br></div>')