mirror of
https://github.com/FreshRSS/FreshRSS.git
synced 2026-05-18 05:14:34 -04:00
@@ -204,25 +204,6 @@ class FreshRSS_index_Controller extends Minz_ActionController {
|
||||
}
|
||||
}
|
||||
|
||||
public function statsAction () {
|
||||
if (!$this->view->loginOk) {
|
||||
Minz_Error::error (
|
||||
403,
|
||||
array ('error' => array (Minz_Translate::t ('access_denied')))
|
||||
);
|
||||
}
|
||||
|
||||
Minz_View::prependTitle (Minz_Translate::t ('stats') . ' · ');
|
||||
|
||||
$statsDAO = new FreshRSS_StatsDAO ();
|
||||
Minz_View::appendScript (Minz_Url::display ('/scripts/flotr2.min.js?' . @filemtime(PUBLIC_PATH . '/scripts/flotr2.min.js')));
|
||||
$this->view->repartition = $statsDAO->calculateEntryRepartition();
|
||||
$this->view->count = ($statsDAO->calculateEntryCount());
|
||||
$this->view->feedByCategory = $statsDAO->calculateFeedByCategory();
|
||||
$this->view->entryByCategory = $statsDAO->calculateEntryByCategory();
|
||||
$this->view->topFeed = $statsDAO->calculateTopFeed();
|
||||
}
|
||||
|
||||
public function aboutAction () {
|
||||
Minz_View::prependTitle (Minz_Translate::t ('about') . ' · ');
|
||||
}
|
||||
|
||||
67
app/Controllers/statsController.php
Normal file
67
app/Controllers/statsController.php
Normal file
@@ -0,0 +1,67 @@
|
||||
<?php
|
||||
|
||||
class FreshRSS_stats_Controller extends Minz_ActionController {
|
||||
|
||||
public function indexAction() {
|
||||
$statsDAO = new FreshRSS_StatsDAO ();
|
||||
Minz_View::appendScript (Minz_Url::display ('/scripts/flotr2.min.js?' . @filemtime(PUBLIC_PATH . '/scripts/flotr2.min.js')));
|
||||
$this->view->repartition = $statsDAO->calculateEntryRepartition();
|
||||
$this->view->count = ($statsDAO->calculateEntryCount());
|
||||
$this->view->feedByCategory = $statsDAO->calculateFeedByCategory();
|
||||
$this->view->entryByCategory = $statsDAO->calculateEntryByCategory();
|
||||
$this->view->topFeed = $statsDAO->calculateTopFeed();
|
||||
}
|
||||
|
||||
public function idleAction() {
|
||||
$statsDAO = new FreshRSS_StatsDAO ();
|
||||
$feeds = $statsDAO->calculateFeedLastDate();
|
||||
$idleFeeds = array();
|
||||
$now = new \DateTime();
|
||||
$feedDate = clone $now;
|
||||
$lastWeek = clone $now;
|
||||
$lastWeek->modify('-1 week');
|
||||
$lastMonth = clone $now;
|
||||
$lastMonth->modify('-1 month');
|
||||
$last3Month = clone $now;
|
||||
$last3Month->modify('-3 month');
|
||||
$last6Month = clone $now;
|
||||
$last6Month->modify('-6 month');
|
||||
$lastYear = clone $now;
|
||||
$lastYear->modify('-1 year');
|
||||
|
||||
foreach ($feeds as $feed) {
|
||||
$feedDate->setTimestamp($feed['last_date']);
|
||||
if ($feedDate >= $lastWeek) {
|
||||
continue;
|
||||
}
|
||||
if ($feedDate < $lastWeek) {
|
||||
$idleFeeds['last_week'][] = $feed['name'];
|
||||
}
|
||||
if ($feedDate < $lastMonth) {
|
||||
$idleFeeds['last_month'][] = $feed['name'];
|
||||
}
|
||||
if ($feedDate < $last3Month) {
|
||||
$idleFeeds['last_3_month'][] = $feed['name'];
|
||||
}
|
||||
if ($feedDate < $last6Month) {
|
||||
$idleFeeds['last_6_month'][] = $feed['name'];
|
||||
}
|
||||
if ($feedDate < $lastYear) {
|
||||
$idleFeeds['last_year'][] = $feed['name'];
|
||||
}
|
||||
}
|
||||
|
||||
$this->view->idleFeeds = array_reverse($idleFeeds);
|
||||
}
|
||||
|
||||
public function firstAction() {
|
||||
if (!$this->view->loginOk) {
|
||||
Minz_Error::error(
|
||||
403, array('error' => array(Minz_Translate::t('access_denied')))
|
||||
);
|
||||
}
|
||||
|
||||
Minz_View::prependTitle(Minz_Translate::t('stats') . ' · ');
|
||||
}
|
||||
|
||||
}
|
||||
@@ -175,6 +175,26 @@ AND f.id = e.id_feed
|
||||
GROUP BY id
|
||||
ORDER BY count DESC
|
||||
LIMIT 10
|
||||
SQL;
|
||||
$stm = $this->bd->prepare($sql);
|
||||
$stm->execute();
|
||||
return $stm->fetchAll(PDO::FETCH_ASSOC);
|
||||
}
|
||||
|
||||
/**
|
||||
* Calculates the last publication date for each feed
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function calculateFeedLastDate() {
|
||||
$sql = <<<SQL
|
||||
SELECT MAX(f.name) AS name
|
||||
, MAX(date) AS last_date
|
||||
FROM {$this->prefix}feed AS f,
|
||||
{$this->prefix}entry AS e
|
||||
WHERE f.id = e.id_feed
|
||||
GROUP BY f.id
|
||||
ORDER BY name
|
||||
SQL;
|
||||
$stm = $this->bd->prepare($sql);
|
||||
$stm->execute();
|
||||
|
||||
@@ -46,6 +46,14 @@ return array (
|
||||
'no_query_filter' => 'No filter',
|
||||
'about' => 'About',
|
||||
'stats' => 'Statistics',
|
||||
'stats_idle' => 'Idle feeds',
|
||||
'stats_main' => 'Main statistics',
|
||||
|
||||
'last_week' => 'Last week',
|
||||
'last_month' => 'Last month',
|
||||
'last_3_month' => 'Last three months',
|
||||
'last_6_month' => 'Last six months',
|
||||
'last_year' => 'Last year',
|
||||
|
||||
'your_rss_feeds' => 'Your RSS feeds',
|
||||
'add_rss_feed' => 'Add a RSS feed',
|
||||
|
||||
@@ -46,6 +46,14 @@ return array (
|
||||
'no_query_filter' => 'Aucun filtre appliqué',
|
||||
'about' => 'À propos',
|
||||
'stats' => 'Statistiques',
|
||||
'stats_idle' => 'Flux inactifs',
|
||||
'stats_main' => 'Statistiques principales',
|
||||
|
||||
'last_week' => 'La dernière semaine',
|
||||
'last_month' => 'Le dernier mois',
|
||||
'last_3_month' => 'Les derniers trois mois',
|
||||
'last_6_month' => 'Les derniers six mois',
|
||||
'last_year' => 'La dernière année',
|
||||
|
||||
'your_rss_feeds' => 'Vos flux RSS',
|
||||
'add_rss_feed' => 'Ajouter un flux RSS',
|
||||
|
||||
9
app/layout/aside_stats.phtml
Normal file
9
app/layout/aside_stats.phtml
Normal file
@@ -0,0 +1,9 @@
|
||||
<ul class="nav nav-list aside">
|
||||
<li class="nav-header"><?php echo Minz_Translate::t ('stats'); ?></li>
|
||||
<li class="item<?php echo Minz_Request::actionName () == 'index' ? ' active' : ''; ?>">
|
||||
<a href="<?php echo _url ('stats', 'index'); ?>"><?php echo Minz_Translate::t ('stats_main'); ?></a>
|
||||
</li>
|
||||
<li class="item<?php echo Minz_Request::actionName () == 'idle' ? ' active' : ''; ?>">
|
||||
<a href="<?php echo _url ('stats', 'idle'); ?>"><?php echo Minz_Translate::t ('stats_idle'); ?></a>
|
||||
</li>
|
||||
</ul>
|
||||
@@ -76,7 +76,7 @@ if (Minz_Configuration::canLogIn()) {
|
||||
<li class="separator"></li>
|
||||
<li class="item"><a href="<?php echo _url ('configure', 'users'); ?>"><?php echo Minz_Translate::t ('users'); ?></a></li>
|
||||
<li class="separator"></li>
|
||||
<li class="item"><a href="<?php echo _url ('index', 'stats'); ?>"><?php echo Minz_Translate::t ('stats'); ?></a></li>
|
||||
<li class="item"><a href="<?php echo _url ('stats', 'index'); ?>"><?php echo Minz_Translate::t ('stats'); ?></a></li>
|
||||
<li class="item"><a href="<?php echo _url ('index', 'logs'); ?>"><?php echo Minz_Translate::t ('logs'); ?></a></li>
|
||||
<li class="item"><a href="<?php echo _url ('index', 'about'); ?>"><?php echo Minz_Translate::t ('about'); ?></a></li>
|
||||
<?php
|
||||
|
||||
19
app/views/stats/idle.phtml
Normal file
19
app/views/stats/idle.phtml
Normal file
@@ -0,0 +1,19 @@
|
||||
<?php $this->partial('aside_stats'); ?>
|
||||
|
||||
<div class="post content">
|
||||
<a href="<?php echo _url ('index', 'index'); ?>"><?php echo _t ('back_to_rss_feeds'); ?></a>
|
||||
|
||||
<h1><?php echo _t ('stats_idle'); ?></h1>
|
||||
|
||||
<?php foreach ($this->idleFeeds as $period => $feeds){ ?>
|
||||
<div class="stat">
|
||||
<h2><?php echo _t ($period); ?></h2>
|
||||
|
||||
<ul>
|
||||
<?php foreach ($feeds as $feed){ ?>
|
||||
<li><?php echo $feed; ?></li>
|
||||
<?php } ?>
|
||||
</ul>
|
||||
</div>
|
||||
<?php } ?>
|
||||
</div>
|
||||
127
app/views/stats/index.phtml
Normal file
127
app/views/stats/index.phtml
Normal file
@@ -0,0 +1,127 @@
|
||||
<?php $this->partial('aside_stats'); ?>
|
||||
|
||||
<div class="post content">
|
||||
<a href="<?php echo _url ('index', 'index'); ?>"><?php echo _t ('back_to_rss_feeds'); ?></a>
|
||||
|
||||
<h1><?php echo _t ('stats_main'); ?></h1>
|
||||
|
||||
<div class="stat">
|
||||
<h2><?php echo _t ('stats_entry_repartition'); ?></h2>
|
||||
<table>
|
||||
<thead>
|
||||
<tr>
|
||||
<th> </th>
|
||||
<th><?php echo _t ('main_stream'); ?></th>
|
||||
<th><?php echo _t ('all_feeds'); ?></th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<tr>
|
||||
<th><?php echo _t ('status_total'); ?></th>
|
||||
<td class="numeric"><?php echo formatNumber($this->repartition['main_stream']['total']); ?></td>
|
||||
<td class="numeric"><?php echo formatNumber($this->repartition['all_feeds']['total']); ?></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th><?php echo _t ('status_read'); ?></th>
|
||||
<td class="numeric"><?php echo formatNumber($this->repartition['main_stream']['read']); ?></td>
|
||||
<td class="numeric"><?php echo formatNumber($this->repartition['all_feeds']['read']); ?></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th><?php echo _t ('status_unread'); ?></th>
|
||||
<td class="numeric"><?php echo formatNumber($this->repartition['main_stream']['unread']); ?></td>
|
||||
<td class="numeric"><?php echo formatNumber($this->repartition['all_feeds']['unread']); ?></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th><?php echo _t ('status_favorites'); ?></th>
|
||||
<td class="numeric"><?php echo formatNumber($this->repartition['main_stream']['favorite']); ?></td>
|
||||
<td class="numeric"><?php echo formatNumber($this->repartition['all_feeds']['favorite']); ?></td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
|
||||
<div class="stat">
|
||||
<h2><?php echo _t ('stats_entry_per_day'); ?></h2>
|
||||
<div id="statsEntryPerDay" style="height: 300px"></div>
|
||||
</div>
|
||||
|
||||
<div class="stat">
|
||||
<h2><?php echo _t ('stats_feed_per_category'); ?></h2>
|
||||
<div id="statsFeedPerCategory" style="height: 300px"></div>
|
||||
<div id="statsFeedPerCategoryLegend"></div>
|
||||
</div>
|
||||
|
||||
<div class="stat">
|
||||
<h2><?php echo _t ('stats_entry_per_category'); ?></h2>
|
||||
<div id="statsEntryPerCategory" style="height: 300px"></div>
|
||||
<div id="statsEntryPerCategoryLegend"></div>
|
||||
</div>
|
||||
|
||||
<div class="stat">
|
||||
<h2><?php echo _t ('stats_top_feed'); ?></h2>
|
||||
<table>
|
||||
<thead>
|
||||
<tr>
|
||||
<th><?php echo _t ('feed'); ?></th>
|
||||
<th><?php echo _t ('category'); ?></th>
|
||||
<th><?php echo _t ('stats_entry_count'); ?></th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<?php foreach ($this->topFeed as $feed): ?>
|
||||
<tr>
|
||||
<td><?php echo $feed['name']; ?></td>
|
||||
<td><?php echo $feed['category']; ?></td>
|
||||
<td class="numeric"><?php echo formatNumber($feed['count']); ?></td>
|
||||
</tr>
|
||||
<?php endforeach;?>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
"use strict";
|
||||
function initStats() {
|
||||
if (!window.Flotr) {
|
||||
if (window.console) {
|
||||
console.log('FreshRSS waiting for Flotr…');
|
||||
}
|
||||
window.setTimeout(initStats, 50);
|
||||
return;
|
||||
}
|
||||
// Entry per day
|
||||
Flotr.draw(document.getElementById('statsEntryPerDay'),
|
||||
[<?php echo $this->count ?>],
|
||||
{
|
||||
grid: {verticalLines: false},
|
||||
bars: {horizontal: false, show: true},
|
||||
xaxis: {noTicks: 6, showLabels: false, tickDecimals: 0},
|
||||
yaxis: {min: 0},
|
||||
mouse: {relative: true, track: true, trackDecimals: 0, trackFormatter: function(obj) {return numberFormat(obj.y);}}
|
||||
});
|
||||
// Feed per category
|
||||
Flotr.draw(document.getElementById('statsFeedPerCategory'),
|
||||
<?php echo $this->feedByCategory ?>,
|
||||
{
|
||||
grid: {verticalLines: false, horizontalLines: false},
|
||||
pie: {explode: 10, show: true, labelFormatter: function(){return '';}},
|
||||
xaxis: {showLabels: false},
|
||||
yaxis: {showLabels: false},
|
||||
mouse: {relative: true, track: true, trackDecimals: 0, trackFormatter: function(obj) {return obj.series.label + ' - '+ numberFormat(obj.y) + ' ('+ (obj.fraction * 100).toFixed(1) + '%)';}},
|
||||
legend: {container: document.getElementById('statsFeedPerCategoryLegend'), noColumns: 3}
|
||||
});
|
||||
// Entry per category
|
||||
Flotr.draw(document.getElementById('statsEntryPerCategory'),
|
||||
<?php echo $this->entryByCategory ?>,
|
||||
{
|
||||
grid: {verticalLines: false, horizontalLines: false},
|
||||
pie: {explode: 10, show: true, labelFormatter: function(){return '';}},
|
||||
xaxis: {showLabels: false},
|
||||
yaxis: {showLabels: false},
|
||||
mouse: {relative: true, track: true, trackDecimals: 0, trackFormatter: function(obj) {return obj.series.label + ' - '+ numberFormat(obj.y) + ' ('+ (obj.fraction * 100).toFixed(1) + '%)';}},
|
||||
legend: {container: document.getElementById('statsEntryPerCategoryLegend'), noColumns: 3}
|
||||
});
|
||||
}
|
||||
initStats();
|
||||
</script>
|
||||
@@ -1,9 +1,11 @@
|
||||
<?php $this->partial('aside_stats'); ?>
|
||||
|
||||
<div class="post content">
|
||||
<a href="<?php echo _url ('index', 'index'); ?>"><?php echo Minz_Translate::t ('back_to_rss_feeds'); ?></a>
|
||||
|
||||
<h1><?php echo Minz_Translate::t ('stats'); ?></h1>
|
||||
|
||||
<div class="stat">
|
||||
<a href="<?php echo _url ('index', 'index'); ?>"><?php echo Minz_Translate::t ('back_to_rss_feeds'); ?></a>
|
||||
|
||||
<h1><?php echo Minz_Translate::t ('stats_main'); ?></h1>
|
||||
|
||||
<div class="stat">
|
||||
<h2><?php echo Minz_Translate::t ('stats_entry_repartition'); ?></h2>
|
||||
<table>
|
||||
<thead>
|
||||
Reference in New Issue
Block a user