Greasy Fork is available in English.

habrActivity

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

// ==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>')