Доброго всем дня!
Тема будет полезна для всех кто пользуется сервисом Паши Дурова telegra.ph рекламируется, для HR, в общем для всех кто гонит трафик через переходники на telegra.ph и просто любопытствующих как я)
Тут на днях бс запостил в лурке новую страницу с ссылками и мне стало любопытно как ужасающе мало траффика можно наковырять в лурке, где в рулетке в группе бс до сих пор не может набраться 70 человек, короче написал на js и выкладываю для всех кому может быть полезно букмарклет/юзерскрипт показывающий всю возможную информацию:
- Просмотры за год + навигация по годам, если страница была создана не в текущем году
- Дата создания страницы + дата внесения изменений
- Строит график просмотров по месяцам + выводит сумму просмотров
Зачем смотреть статистику?
Часто в даркнете нету привычных в обычном вебе инструментов типа яндекс метрики, тут больший процент юзеров без js, часто ссылки не отловятся по реферреру потому что они вообще не ссылки, а текст и написанны через пробел, пользователь их потом сам вбивает в адресную строку, а понимать хотя бы динамику просмотров - это уже что-то
Код в одну строку, готовый к переносу на панель закладок:
javascript:(async()=>{const API_BASE_URL='https://api.telegra.ph/getViews',MONTHS_RU={'01':'Январь','02':'Февраль','03':'Март','04':'Апрель','05':'Май','06':'Июнь','07':'Июль','08':'Август','09':'Сентябрь','10':'Октябрь','11':'Ноябрь','12':'Декабрь'},CHART_CONFIG={type:'bar',options:{responsive:!0,maintainAspectRatio:!1,scales:{y:{beginAtZero:!0}}}},getTimeElapsed=a=>{const b=new Date-a,c=Math.floor(b/864e5),d=Math.floor(b%864e5/36e5);return 0<c?`${c} дн, ${d} ч назад`:`${d} ч назад`},createNavLinks=(a,b)=>{const c=new Date().getFullYear()===parseInt(a),d=parseInt(a)>parseInt(b),e=`${window.location.origin}${window.location.pathname}?year=${a-1}`,f=`${window.location.origin}${window.location.pathname}?year=${parseInt(a)+1}`;return`${d?`<a href="${e}" class="nav-link">← ${a-1}</a>`:'<a></a>'}; <a href="https://donate.torsha.re" class="nav-link">Нужна ваша поддержка</a>${!c?`<a href="${f}" class="nav-link">${parseInt(a)+1} →</a>`:'<a></a>'};`},createExpandableDiv=()=>{const a=document.createElement('div');return a.className='expandableDiv',a.id='expandableDiv',a},createHeader=()=>{const a=document.createElement('div');return a.className='header',a.id='torshare_head',a.innerHTML='<img id="torshare" src=""><h3 id="title_text">API Telegra.ph</h3>',a},fetchViews=async(a,b)=>{try{return await fetch(`${API_BASE_URL}?path=${a}&year=${b}`,{headers:{Authorization:'Bearer 123'}}).then(a=>a.json())}catch(a){console.error('Error fetching views:',a)}},generateRandomGradient=()=>{const a=Math.floor(360*Math.random()),b=(a+60)%360,c=(b+120)%360;return`linear-gradient(45deg, hsl(${a},100%,50%), hsl(${b},100%,50%), hsl(${c},100%,50%))`};const header=createHeader(),expandableDiv=createExpandableDiv(),gradButton=document.getElementById('gradButton'),generateChart=async()=>{try{const a=await fetchViews('test',new Date().getFullYear()),b=a.data.views.map(a=>a.date.split('-').slice(0,2).join('.')).map(a=>`${MONTHS_RU[a.slice(3,5)]} ${a.slice(0,4)}`),c=a.data.views.map(a=>a.views),d=document.getElementById('myChart').getContext('2d');new Chart(d,{...CHART_CONFIG,data:{labels:b,datasets:[{label:'Просмотры',data:c}]}})}catch(a){console.error('Error generating chart:',a)}};gradButton.onclick=()=>{document.body.style.background=generateRandomGradient()},document.body.append(header,expandableDiv),generateChart()})();
Код целиком (можно юзать в kiwi браузере или для tempermonkey)
:
javascript: (function () {
const API_BASE_URL = 'https://api.telegra.ph/getViews';
const MONTHS_RU = {'01': 'Январь', '02': 'Февраль', '03': 'Март', '04': 'Апрель', '05': 'Май', '06': 'Июнь','07': 'Июль', '08': 'Август', '09': 'Сентябрь', '10': 'Октябрь', '11': 'Ноябрь', '12': 'Декабрь'};
const CHART_CONFIG = {type: 'bar', options: {responsive: true, maintainAspectRatio: false,scales: { y: { beginAtZero: true}}}};
const getTimeElapsed = (dateString) => {
const diffMs = new Date() - new Date(dateString);
const days = Math.floor(diffMs / (1000 * 60 * 60 * 24));
const hours = Math.floor((diffMs % (1000 * 60 * 60 * 24)) / (1000 * 60 * 60));
return days > 0 ?
${days} дн, ${hours} ч назад:
${hours} ч назад;
};
const createNavLinks = (currentYear, creationYear) => {
const isCurrentYear = new Date().getFullYear() === parseInt(currentYear);
const showPrevYear = parseInt(currentYear) > parseInt(creationYear);
const prevYearUrl =
${window.location.origin}${window.location.pathname}?year=${currentYear - 1};
const nextYearUrl =
${window.location.origin}${window.location.pathname}?year=${parseInt(currentYear) + 1};
return
${showPrevYear ? <a href="${prevYearUrl}" class="nav-link">← ${currentYear - 1}</a>
: '<a></a>'}; <a href="https://donate.torsha.re" class="nav-link">Нужна ваша поддержка</a>${!isCurrentYear ? <a href="${nextYearUrl}" class="nav-link">${parseInt(currentYear) + 1} →</a>
:'<a></a>'};;};
const createExpandableDiv = () => {
const div = document.createElement('div');
div.className = 'expandableDiv';
div.id = 'expandableDiv';
return div;
};
const createHeader = () => {
const header = document.createElement('div');
header.className = 'header';
header.id="torshare_head";
header.innerHTML =
<img id="torshare" src="" class="toggle-img" title="Развернуть">;
const statsSpan = document.createElement('span');
statsSpan.id = 'hits_summ__and__date_created_updated';
header.appendChild(statsSpan);
return header;
};
const createChartContainer = (currentYear, creationYear) => {
const container = document.createElement('div');
container.className = 'chart-container';
const canvas = document.createElement('canvas');
canvas.id = 'viewsChart';
container.appendChild(canvas);
const navigation = document.createElement('div');
navigation.className = 'navigation';
navigation.innerHTML = createNavLinks(currentYear, creationYear);
container.appendChild(navigation);
return container;
};
const fetchMonthlyViews = async (pathname, year, month) => {
const response = await fetch(
${API_BASE_URL}${pathname}?year=${year}&month=${month});
const data = await response.json();
return data.result?.views || 0;
};
const fetchYearlyData = async (pathname, year) => {
const monthlyData = [];
for (let month = 1; month <= 12; month++) {
const monthStr = month.toString().padStart(2, '0');
const views = await fetchMonthlyViews(pathname, year, monthStr);
monthlyData.push({ month: monthStr, views });
}
return monthlyData;
};
const loadChartLibrary = () => {
return new Promise((resolve) => {
if (window.Chart) {
resolve();
return;
}
const script = document.createElement('script');
script.src = 'https://cdn.jsdelivr.net/npm/chart.js';
script.onload = resolve;
document.body.appendChild(script);
});
};
const renderChart = (data, year) => {
const ctx = document.getElementById('viewsChart').getContext('2d');
const chartData = {
labels: data.map(d => MONTHS_RU[d.month]),
datasets: [{
label:
Просмотры за ${year} год,
data: data.map(d => d.views),
backgroundColor: '#66339988',
borderColor: '#663399',
borderWidth: 2
}]
};
new Chart(ctx, { ...CHART_CONFIG, data: chartData });
};
const init = async () => {
const pathname = window.location.pathname;
const urlYear = new URLSearchParams(window.location.search).get('year');
const currentYear = urlYear && urlYear.length === 4 && !isNaN(urlYear) ? urlYear : new Date().getFullYear();
const publishedMeta = document.querySelector('meta[property="article:published_time"]');
const modifiedMeta = document.querySelector('meta[property="article:modified_time"]');
const publishedTime = publishedMeta.getAttribute('content');
const modifiedTime = modifiedMeta.getAttribute('content');
const creationYear = new Date(publishedTime).getFullYear();
let expandableDiv = document.getElementById('expandableDiv') || createExpandableDiv();
if (!document.getElementById('expandableDiv')) {
const header = createHeader();
const chartContainer = createChartContainer(currentYear, creationYear);
expandableDiv.append(header, chartContainer);
document.body.appendChild(expandableDiv);
}
const yearlyResponse = await fetch(API_BASE_URL+pathname+"?year="+currentYear);
const yearlyData = await yearlyResponse.json();
const monthlyData = await fetchYearlyData(pathname, currentYear);
const statsText = \
Всего просмотров за год: ${yearlyData.result.views} | Создано: ${getTimeElapsed(publishedTime)} | Обновлено: ${getTimeElapsed(modifiedTime)}`;
document.getElementById('hits_summanddate_created_updated').textContent = statsText;
await loadChartLibrary();
renderChart(monthlyData, currentYear);
};
const style=document.createElement('style'); style.textContent=.expandableDiv { position: fixed; left: 0; bottom: 0; width: 100vw; height: 70px; background-color: #fff; padding: 2px; box-sizing: border-box; z-index: 9999; overflow: hidden; transition: height 1s ease-in-out; }.header { display: flex; justify-content: space-between; align-items: center;height:70px; }.toggle-img { height: 50px; cursor: pointer; }.chart-container { width: 100%; height: calc(50vh - 70px);box-sizing:border-box;padding-bottom:15px;}.navigation {display:flex;justify-content:space-between;position: absolute;left:0; bottom: 0px; font-family: monospace; font-size: 14px;width:100vw;background-color:rebeccapurple;}.nav-link:hover { text-decoration: none; background:#fff; color:#663399;}.nav-link{padding-left:5px;padding-right:5px;text-decoration:none;background:rebeccapurple;color:#fff;}
;
document.head.appendChild(style);init().catch(console.error);
document.getElementById('torshare').addEventListener('click', () => {
expandableDiv.style.height = expandableDiv.style.height === '70px' ? '50vh' : '70px';
});
document.getElementById('torshare_head').addEventListener('click', () => {
expandableDiv.style.height = expandableDiv.style.height === '70px' ? '50vh' : '70px';
});document.getElementById('torshare').click();document.getElementById('torshare').click();
})();