swad-core/swad_report.c

886 lines
30 KiB
C
Raw Normal View History

2016-09-08 18:04:30 +02:00
// swad_report.c: report on my use of the platform
/*
SWAD (Shared Workspace At a Distance),
is a web platform developed at the University of Granada (Spain),
and used to support university teaching.
This file is part of SWAD core.
Copyright (C) 1999-2016 Antonio Ca<EFBFBD>as Vargas
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General 3 License as
published by the Free Software Foundation, either version 3 of the
License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
/*****************************************************************************/
/*********************************** Headers *********************************/
/*****************************************************************************/
// #include <stdbool.h> // For boolean type
// #include <stdio.h> // For sprintf
// #include <string.h> // For string functions
2016-09-11 21:25:14 +02:00
#include "swad_database.h"
2016-09-08 19:21:25 +02:00
#include "swad_global.h"
2016-09-11 12:46:52 +02:00
#include "swad_ID.h"
2016-09-08 19:21:25 +02:00
#include "swad_profile.h"
2016-09-08 18:04:30 +02:00
/*****************************************************************************/
/****************************** Public constants *****************************/
/*****************************************************************************/
/*****************************************************************************/
/***************************** Private constants *****************************/
/*****************************************************************************/
2016-10-02 22:54:30 +02:00
#define Rep_MAX_BAR_WIDTH 80 // Maximum width of graphic bar
2016-09-12 10:02:17 +02:00
// #define Rep_BLOCK "&boxH;" // HTML code for a block in graphic bar
// #define Rep_BLOCK "&blk12;" // HTML code for a block in graphic bar
// #define Rep_BLOCK "&block;" // HTML code for a block in graphic bar
// #define Rep_BLOCK "&equiv;" // HTML code for a block in graphic bar
// #define Rep_BLOCK "&bull;" // HTML code for a block in graphic bar
#define Rep_BLOCK "&squf;" // HTML code for a block in graphic bar
2016-09-08 18:04:30 +02:00
/*****************************************************************************/
/****************************** Internal types *******************************/
/*****************************************************************************/
2016-09-09 00:36:44 +02:00
typedef enum
{
Rep_SEE,
Rep_PRINT,
} Rep_SeeOrPrint_t;
2016-09-08 18:04:30 +02:00
/*****************************************************************************/
/************** External global variables from others modules ****************/
/*****************************************************************************/
2016-09-08 19:21:25 +02:00
extern struct Globals Gbl;
2016-09-08 18:04:30 +02:00
/*****************************************************************************/
/************************* Internal global variables *************************/
/*****************************************************************************/
/*****************************************************************************/
/***************************** Private prototypes ****************************/
/*****************************************************************************/
2016-09-09 00:36:44 +02:00
static void Rep_ShowOrPrintMyUsageReport (Rep_SeeOrPrint_t SeeOrPrint);
static void Rep_PutIconToPrintMyUsageReport (void);
2016-10-03 09:40:10 +02:00
static void Rep_WriteHeader (const char *StrCurrentDate);
static void Rep_WriteSectionPlatform (void);
2016-10-03 09:44:32 +02:00
static void Rep_WriteSectionUsrInfo (void);
2016-10-03 09:59:57 +02:00
static void Rep_WriteSectionUsrFigures (struct UsrFigures *UsrFigures,
struct tm *tm_FirstClickTime,
const char *StrCurrentDate,
const char *StrCurrentTime);
2016-10-03 09:40:10 +02:00
2016-10-02 23:26:59 +02:00
static unsigned long Rep_GetMaxHitsPerYear (time_t FirstClickTimeUTC);
2016-09-29 01:34:17 +02:00
static void Rep_GetAndWriteCurrentCrssOfAUsr (const struct UsrData *UsrDat,Rol_Role_t Role,
time_t FirstClickTimeUTC,
2016-09-29 01:48:52 +02:00
struct tm *tm_FirstClickTime,
2016-10-02 17:56:05 +02:00
unsigned long MaxHitsPerYear);
2016-10-02 19:42:47 +02:00
static void Rep_GetAndWriteHistoricCrssOfAUsr (const struct UsrData *UsrDat,Rol_Role_t Role,
time_t FirstClickTimeUTC,
struct tm *tm_FirstClickTime,
unsigned long MaxHitsPerYear);
static void Rep_WriteRowCrsData (long CrsCod,Rol_Role_t Role,
2016-09-29 01:34:17 +02:00
time_t FirstClickTimeUTC,
2016-09-29 01:48:52 +02:00
struct tm *tm_FirstClickTime,
2016-10-02 17:56:05 +02:00
unsigned long MaxHitsPerYear);
2016-10-02 19:42:47 +02:00
static void Rep_ShowMyHitsPerYear (bool AnyCourse,long CrsCod,Rol_Role_t Role,
2016-10-02 17:56:05 +02:00
time_t FirstClickTimeUTC,
struct tm *tm_FirstClickTime,
unsigned long MaxHitsPerYear);
2016-09-12 10:02:17 +02:00
static void Rep_DrawBarNumHits (float HitsNum,float HitsMax,
unsigned MaxBarWidth);
2016-09-12 00:50:24 +02:00
2016-09-08 18:04:30 +02:00
/*****************************************************************************/
/********* Show my usage report (report on my use of the platform) ***********/
/*****************************************************************************/
void Rep_ShowMyUsageReport (void)
2016-09-09 00:36:44 +02:00
{
Rep_ShowOrPrintMyUsageReport (false);
}
void Rep_PrintMyUsageReport (void)
{
Rep_ShowOrPrintMyUsageReport (true);
}
static void Rep_ShowOrPrintMyUsageReport (Rep_SeeOrPrint_t SeeOrPrint)
2016-09-08 18:04:30 +02:00
{
2016-10-02 22:54:30 +02:00
extern const char *Txt_Report;
2016-09-11 22:45:53 +02:00
extern const char *Txt_Courses;
2016-10-02 22:54:30 +02:00
extern const char *Txt_historical_log;
2016-09-12 00:50:24 +02:00
extern const char *Txt_Hits;
2016-09-11 14:28:32 +02:00
struct UsrFigures UsrFigures;
2016-10-02 21:14:07 +02:00
time_t CurrentTime;
struct tm tm_CurrentTime;
2016-09-12 00:50:24 +02:00
struct tm tm_FirstClickTime;
2016-10-02 21:14:07 +02:00
char StrCurrentDate[10+1]; // Example: 2016-10-02
// 1234567890
char StrCurrentTime[8+1]; // Example: 19:03:49
// 12345678
2016-09-11 18:31:11 +02:00
Rol_Role_t Role;
2016-10-02 17:56:05 +02:00
unsigned long MaxHitsPerYear;
2016-10-02 21:14:07 +02:00
/***** Get current date-time *****/
time (&CurrentTime);
if ((gmtime_r (&CurrentTime,&tm_CurrentTime)) != NULL)
{
sprintf (StrCurrentDate,"%04d-%02d-%02d",
1900 + tm_CurrentTime.tm_year, // year
1 + tm_CurrentTime.tm_mon, // month
tm_CurrentTime.tm_mday); // day of the month
sprintf (StrCurrentTime,"%02d:%02d:%02d",
tm_CurrentTime.tm_hour, // hours
tm_CurrentTime.tm_min, // minutes
tm_CurrentTime.tm_sec); // seconds
}
else
{
StrCurrentDate[0] = '\0';
StrCurrentTime[0] = '\0';
}
2016-09-11 22:45:53 +02:00
/***** Start frame *****/
2016-09-11 21:45:29 +02:00
if (SeeOrPrint == Rep_SEE)
2016-10-02 22:54:30 +02:00
Lay_StartRoundFrame (NULL,Txt_Report,
2016-10-02 21:14:07 +02:00
Rep_PutIconToPrintMyUsageReport);
2016-09-11 22:45:53 +02:00
2016-10-02 21:14:07 +02:00
/***** Head *****/
2016-10-03 09:40:10 +02:00
Rep_WriteHeader (StrCurrentDate);
2016-10-02 22:54:30 +02:00
2016-10-03 09:40:10 +02:00
fprintf (Gbl.F.Out,"<div style=\"margin:2em; text-align:left;\">");
2016-10-02 22:54:30 +02:00
/***** Platform *****/
2016-10-03 09:40:10 +02:00
Rep_WriteSectionPlatform ();
2016-10-02 21:14:07 +02:00
2016-09-11 22:45:53 +02:00
/***** Personal information *****/
2016-10-03 09:44:32 +02:00
Rep_WriteSectionUsrInfo ();
2016-09-11 22:45:53 +02:00
/***** Figures *****/
2016-09-11 14:28:32 +02:00
Prf_GetUsrFigures (Gbl.Usrs.Me.UsrDat.UsrCod,&UsrFigures);
if (UsrFigures.FirstClickTimeUTC)
2016-10-03 09:59:57 +02:00
gmtime_r (&UsrFigures.FirstClickTimeUTC,&tm_FirstClickTime);
Rep_WriteSectionUsrFigures (&UsrFigures,&tm_FirstClickTime,
StrCurrentDate,StrCurrentTime);
2016-09-11 22:45:53 +02:00
2016-10-02 23:38:40 +02:00
/***** Global hits *****/
fprintf (Gbl.F.Out,"<h3>%s</h3>",Txt_Hits);
Rep_ShowMyHitsPerYear (true,-1L,Rol_UNKNOWN,
UsrFigures.FirstClickTimeUTC,
&tm_FirstClickTime,
0);
2016-10-02 19:42:47 +02:00
/***** Current courses *****/
2016-10-02 21:14:07 +02:00
fprintf (Gbl.F.Out,"<h3>%s",Txt_Courses);
if (StrCurrentDate[0])
fprintf (Gbl.F.Out," (%s)",StrCurrentDate);
fprintf (Gbl.F.Out,"</h3>"
"<ul>");
2016-09-11 22:45:53 +02:00
2016-10-02 19:42:47 +02:00
/* Number of courses in which the user is student/teacher */
2016-10-02 23:26:59 +02:00
MaxHitsPerYear = Rep_GetMaxHitsPerYear (UsrFigures.FirstClickTimeUTC);
2016-09-11 18:31:11 +02:00
for (Role = Rol_STUDENT;
Role <= Rol_TEACHER;
Role++)
2016-10-02 19:42:47 +02:00
/* List my courses with this role */
Rep_GetAndWriteCurrentCrssOfAUsr (&Gbl.Usrs.Me.UsrDat,Role,
UsrFigures.FirstClickTimeUTC,&tm_FirstClickTime,
MaxHitsPerYear);
2016-09-11 18:31:11 +02:00
2016-09-11 12:46:52 +02:00
fprintf (Gbl.F.Out,"</ul>");
2016-09-11 22:45:53 +02:00
2016-10-02 19:42:47 +02:00
/***** Historic courses *****/
2016-10-02 22:54:30 +02:00
fprintf (Gbl.F.Out,"<h3>%s (%s)</h3>"
2016-10-02 19:42:47 +02:00
"<ul>",
2016-10-02 22:54:30 +02:00
Txt_Courses,Txt_historical_log);
2016-10-02 19:42:47 +02:00
/* Number of courses in which the user clicked as student/teacher */
for (Role = Rol_STUDENT;
Role <= Rol_TEACHER;
Role++)
/* List my courses with this role */
Rep_GetAndWriteHistoricCrssOfAUsr (&Gbl.Usrs.Me.UsrDat,Role,
UsrFigures.FirstClickTimeUTC,&tm_FirstClickTime,
MaxHitsPerYear);
fprintf (Gbl.F.Out,"</ul>");
2016-10-02 17:56:05 +02:00
2016-09-11 22:45:53 +02:00
/***** End frame *****/
fprintf (Gbl.F.Out,"</div>");
2016-09-11 21:45:29 +02:00
if (SeeOrPrint == Rep_SEE)
Lay_EndRoundFrame ();
2016-09-08 18:04:30 +02:00
}
2016-09-09 00:36:44 +02:00
/*****************************************************************************/
/********************* Put icon to print my usage report *********************/
/*****************************************************************************/
static void Rep_PutIconToPrintMyUsageReport (void)
{
extern const char *Txt_Print;
Lay_PutContextualLink (ActPrnMyUsgRep,NULL,
"print64x64.png",
Txt_Print,NULL,
NULL);
}
2016-09-11 21:25:14 +02:00
2016-10-03 09:40:10 +02:00
/*****************************************************************************/
/******************** Write header of user's usage report ********************/
/*****************************************************************************/
static void Rep_WriteHeader (const char *StrCurrentDate)
{
extern const char *Txt_Report_of_use_of_PLATFORM;
/***** Start of header *****/
fprintf (Gbl.F.Out,"<header style=\"margin:2em; text-align:center;\">");
/***** Main title *****/
sprintf (Gbl.Title,Txt_Report_of_use_of_PLATFORM,Cfg_PLATFORM_SHORT_NAME);
fprintf (Gbl.F.Out,"<h1>%s</h1>",Gbl.Title);
/***** Subtitle *****/
fprintf (Gbl.F.Out,"<h2>%s",Gbl.Usrs.Me.UsrDat.FullName);
if (StrCurrentDate[0])
fprintf (Gbl.F.Out,", %s",StrCurrentDate);
fprintf (Gbl.F.Out,"</h2>");
/***** End of header *****/
fprintf (Gbl.F.Out,"</header>");
}
/*****************************************************************************/
/************* Write section for platform in user's usage report *************/
/*****************************************************************************/
static void Rep_WriteSectionPlatform (void)
{
extern const char *Txt_Teaching_platform;
extern const char *Txt_Name;
extern const char *Txt_TAGLINE;
extern const char *Txt_URL;
/***** Start of section *****/
fprintf (Gbl.F.Out,"<section>"
"<h3>%s</h3>"
"<ul>",
Txt_Teaching_platform);
/***** Platform name *****/
fprintf (Gbl.F.Out,"<li>%s: %s, %s</li>",
Txt_Name,
Cfg_PLATFORM_FULL_NAME,Txt_TAGLINE);
/***** Server URL *****/
fprintf (Gbl.F.Out,"<li>%s: <a href=\"%s\">%s</a></li>",
Txt_URL,Cfg_URL_SWAD_SERVER,Cfg_URL_SWAD_SERVER);
/***** End of section *****/
fprintf (Gbl.F.Out,"</ul>"
"</section>");
}
2016-10-02 19:42:47 +02:00
2016-10-03 09:44:32 +02:00
/*****************************************************************************/
/*********** Write section for user's info in user's usage report ************/
/*****************************************************************************/
static void Rep_WriteSectionUsrInfo (void)
{
extern const char *Txt_Personal_information;
extern const char *Txt_Name;
extern const char *Txt_Email;
extern const char *Txt_Country;
extern const char *Txt_Institution;
char CtyName[Cty_MAX_BYTES_COUNTRY_NAME+1];
struct Institution Ins;
/***** Start of section *****/
fprintf (Gbl.F.Out,"<section>"
"<h3>%s</h3>"
"<ul>",
Txt_Personal_information);
/***** User's name *****/
fprintf (Gbl.F.Out,"<li>%s: <strong>%s</strong></li>",
Txt_Name,
Gbl.Usrs.Me.UsrDat.FullName);
/***** User's e-mail *****/
fprintf (Gbl.F.Out,"<li>%s: %s</li>",
Txt_Email,
Gbl.Usrs.Me.UsrDat.Email);
/***** User's country *****/
Cty_GetCountryName (Gbl.Usrs.Me.UsrDat.CtyCod,CtyName);
fprintf (Gbl.F.Out,"<li>%s: %s</li>",
Txt_Country,
CtyName);
/***** User's institution *****/
Ins.InsCod = Gbl.Usrs.Me.UsrDat.InsCod;
Ins_GetDataOfInstitutionByCod (&Ins,Ins_GET_BASIC_DATA);
fprintf (Gbl.F.Out,"<li>%s: %s</li>",
Txt_Institution,
Ins.FullName);
/***** End of section *****/
fprintf (Gbl.F.Out,"</ul>"
"</section>");
}
2016-10-03 09:59:57 +02:00
/*****************************************************************************/
/********* Write section for user's figures in user's usage report ***********/
/*****************************************************************************/
static void Rep_WriteSectionUsrFigures (struct UsrFigures *UsrFigures,
struct tm *tm_FirstClickTime,
const char *StrCurrentDate,
const char *StrCurrentTime)
{
extern const char *Txt_Figures;
extern const char *Txt_TIME_Since;
extern const char *Txt_TIME_until;
extern const char *Txt_day;
extern const char *Txt_days;
extern const char *Txt_Clicks;
extern const char *Txt_Files_uploaded;
extern const char *Txt_file;
extern const char *Txt_files;
extern const char *Txt_public_FILES;
extern const char *Txt_Downloads;
extern const char *Txt_download;
extern const char *Txt_downloads;
extern const char *Txt_Forum_posts;
extern const char *Txt_post;
extern const char *Txt_posts;
extern const char *Txt_Messages_sent;
extern const char *Txt_message;
extern const char *Txt_messages;
unsigned NumFiles;
unsigned NumPublicFiles;
/***** Start of section *****/
fprintf (Gbl.F.Out,"<section>"
"<h3>%s</h3>"
"<ul>",
Txt_Figures);
/***** Time since first click until now *****/
fprintf (Gbl.F.Out,"<li>%s ",Txt_TIME_Since);
if (UsrFigures->FirstClickTimeUTC)
{
if (tm_FirstClickTime != NULL)
{
fprintf (Gbl.F.Out,"%04d-%02d-%02d %02d:%02d:%02d UTC",
1900 + tm_FirstClickTime->tm_year, // year
1 + tm_FirstClickTime->tm_mon, // month
tm_FirstClickTime->tm_mday, // day of the month
tm_FirstClickTime->tm_hour, // hours
tm_FirstClickTime->tm_min, // minutes
tm_FirstClickTime->tm_sec); // seconds
if (StrCurrentDate[0])
fprintf (Gbl.F.Out," %s %s %s UTC",
Txt_TIME_until,StrCurrentDate,StrCurrentTime);
if (UsrFigures->NumDays > 0)
fprintf (Gbl.F.Out," (%d %s)",
UsrFigures->NumDays,
(UsrFigures->NumDays == 1) ? Txt_day :
Txt_days);
}
}
else // Time of first click is unknown
{
fprintf (Gbl.F.Out,"?");
if (StrCurrentDate[0])
fprintf (Gbl.F.Out," - %s %s UTC",StrCurrentDate,StrCurrentTime);
}
fprintf (Gbl.F.Out,"</li>");
/***** Number of clicks *****/
fprintf (Gbl.F.Out,"<li>%s: ",Txt_Clicks);
if (UsrFigures->NumClicks >= 0)
{
fprintf (Gbl.F.Out,"%ld",UsrFigures->NumClicks);
if (UsrFigures->NumDays > 0)
{
fprintf (Gbl.F.Out," (");
Str_WriteFloatNum ((float) UsrFigures->NumClicks /
(float) UsrFigures->NumDays);
fprintf (Gbl.F.Out," / %s)",Txt_day);
}
}
else // Number of clicks is unknown
fprintf (Gbl.F.Out,"?");
fprintf (Gbl.F.Out,"</li>");
/***** Number of files currently published *****/
if ((NumFiles = Brw_GetNumFilesUsr (Gbl.Usrs.Me.UsrDat.UsrCod)))
NumPublicFiles = Brw_GetNumPublicFilesUsr (Gbl.Usrs.Me.UsrDat.UsrCod);
else
NumPublicFiles = 0;
fprintf (Gbl.F.Out,"<li>"
"%s: %u %s (%u %s)"
"</li>",
Txt_Files_uploaded,
NumFiles,
(NumFiles == 1) ? Txt_file :
Txt_files,
NumPublicFiles,Txt_public_FILES);
/***** Number of file views *****/
fprintf (Gbl.F.Out,"<li>%s: ",Txt_Downloads);
if (UsrFigures->NumFileViews >= 0)
{
fprintf (Gbl.F.Out,"%ld %s",
UsrFigures->NumFileViews,
(UsrFigures->NumFileViews == 1) ? Txt_download :
Txt_downloads);
if (UsrFigures->NumDays > 0)
{
fprintf (Gbl.F.Out," (");
Str_WriteFloatNum ((float) UsrFigures->NumFileViews /
(float) UsrFigures->NumDays);
fprintf (Gbl.F.Out," / %s)",Txt_day);
}
}
else // Number of file views is unknown
fprintf (Gbl.F.Out,"?");
fprintf (Gbl.F.Out,"</li>");
/***** Number of posts in forums *****/
fprintf (Gbl.F.Out,"<li>%s: ",Txt_Forum_posts);
if (UsrFigures->NumForPst >= 0)
{
fprintf (Gbl.F.Out,"%ld %s",
UsrFigures->NumForPst,
(UsrFigures->NumForPst == 1) ? Txt_post :
Txt_posts);
if (UsrFigures->NumDays > 0)
{
fprintf (Gbl.F.Out," (");
Str_WriteFloatNum ((float) UsrFigures->NumForPst /
(float) UsrFigures->NumDays);
fprintf (Gbl.F.Out," / %s)",Txt_day);
}
}
else // Number of forum posts is unknown
fprintf (Gbl.F.Out,"?");
fprintf (Gbl.F.Out,"</li>");
/***** Number of messages sent *****/
fprintf (Gbl.F.Out,"<li>%s: ",Txt_Messages_sent);
if (UsrFigures->NumMsgSnt >= 0)
{
fprintf (Gbl.F.Out,"%ld %s",
UsrFigures->NumMsgSnt,
(UsrFigures->NumMsgSnt == 1) ? Txt_message :
Txt_messages);
if (UsrFigures->NumDays > 0)
{
fprintf (Gbl.F.Out," (");
Str_WriteFloatNum ((float) UsrFigures->NumMsgSnt /
(float) UsrFigures->NumDays);
fprintf (Gbl.F.Out," / %s)",Txt_day);
}
}
else // Number of messages sent is unknown
fprintf (Gbl.F.Out,"?");
fprintf (Gbl.F.Out,"</li>");
/***** End of section *****/
fprintf (Gbl.F.Out,"</ul>"
"</section>");
}
2016-10-02 19:42:47 +02:00
/*****************************************************************************/
/************ Get the maximum number of hits per course-year-role ************/
/*****************************************************************************/
// Return the maximum number of hits per year
2016-10-02 23:26:59 +02:00
static unsigned long Rep_GetMaxHitsPerYear (time_t FirstClickTimeUTC)
2016-10-02 19:42:47 +02:00
{
char Query[1024];
MYSQL_RES *mysql_res;
MYSQL_ROW row;
unsigned long MaxHitsPerYear = 0;
sprintf (Query,"SELECT MAX(N) FROM (SELECT "
"CrsCod,"
2016-10-02 23:26:59 +02:00
"YEAR(CONVERT_TZ(ClickTime,@@session.time_zone,'UTC')) AS Year,"
2016-10-02 19:42:47 +02:00
"Role,"
"COUNT(*) AS N"
" FROM log_full"
" WHERE ClickTime>=FROM_UNIXTIME('%ld')"
2016-10-02 23:26:59 +02:00
" AND UsrCod='%ld' AND Role>='%u' AND Role<='%u'"
2016-10-02 19:42:47 +02:00
" GROUP BY CrsCod,Year,Role)"
" AS hits_per_crs_year",
(long) FirstClickTimeUTC,
2016-10-02 23:26:59 +02:00
Gbl.Usrs.Me.UsrDat.UsrCod,
(unsigned) Rol_STUDENT,
(unsigned) Rol_TEACHER);
2016-10-02 19:42:47 +02:00
DB_QuerySELECT (Query,&mysql_res,"can not get last question index");
/***** Get number of users *****/
row = mysql_fetch_row (mysql_res);
if (row[0]) // There are questions
if (sscanf (row[0],"%lu",&MaxHitsPerYear) != 1)
Lay_ShowErrorAndExit ("Error when getting maximum hits.");
return MaxHitsPerYear;
}
2016-09-11 21:25:14 +02:00
/*****************************************************************************/
/************************** Write courses of a user **************************/
/*****************************************************************************/
2016-09-29 01:34:17 +02:00
static void Rep_GetAndWriteCurrentCrssOfAUsr (const struct UsrData *UsrDat,Rol_Role_t Role,
time_t FirstClickTimeUTC,
2016-09-29 01:48:52 +02:00
struct tm *tm_FirstClickTime,
2016-10-02 17:56:05 +02:00
unsigned long MaxHitsPerYear)
2016-09-11 21:25:14 +02:00
{
2016-10-02 19:42:47 +02:00
extern const char *Txt_USER_in_COURSE;
extern const char *Txt_ROLES_SINGUL_Abc[Rol_NUM_ROLES][Usr_NUM_SEXS];
extern const char *Txt_course;
extern const char *Txt_courses;
extern const char *Txt_teachers_ABBREVIATION;
extern const char *Txt_students_ABBREVIATION;
char Query[1024];
MYSQL_RES *mysql_res;
MYSQL_ROW row;
unsigned NumCrss;
unsigned NumCrs;
long CrsCod;
NumCrss = Usr_GetNumCrssOfUsrWithARole (UsrDat->UsrCod,Role);
sprintf (Gbl.Title,Txt_USER_in_COURSE,Txt_ROLES_SINGUL_Abc[Role][Gbl.Usrs.Me.UsrDat.Sex]);
fprintf (Gbl.F.Out,"<li>%s %u %s",
Gbl.Title,
NumCrss,
NumCrss == 1 ? Txt_course :
Txt_courses);
if (NumCrss)
{
fprintf (Gbl.F.Out," (%u %s / %u %s)",
Usr_GetNumUsrsInCrssOfAUsr (Gbl.Usrs.Me.UsrDat.UsrCod,Role,Rol_TEACHER),
Txt_teachers_ABBREVIATION,
Usr_GetNumUsrsInCrssOfAUsr (Gbl.Usrs.Me.UsrDat.UsrCod,Role,Rol_STUDENT),
Txt_students_ABBREVIATION);
/***** Get courses of a user from database *****/
sprintf (Query,"SELECT courses.CrsCod,degrees.FullName,courses.Year,courses.FullName"
" FROM crs_usr,courses,degrees"
" WHERE crs_usr.UsrCod='%ld'"
" AND crs_usr.Role='%u'"
" AND crs_usr.CrsCod=courses.CrsCod"
" AND courses.DegCod=degrees.DegCod"
" ORDER BY degrees.FullName,courses.Year,courses.FullName",
UsrDat->UsrCod,(unsigned) Role);
/***** List the courses (one row per course) *****/
if ((NumCrss = (unsigned) DB_QuerySELECT (Query,&mysql_res,"can not get courses of a user")))
{
/* Heading row */
fprintf (Gbl.F.Out,"<ol>");
/* Write courses */
for (NumCrs = 1;
NumCrs <= NumCrss;
NumCrs++)
{
/* Get next course */
row = mysql_fetch_row (mysql_res);
/* Get course code (row[0]) */
CrsCod = Str_ConvertStrCodToLongCod (row[0]);
/* Write data of this course */
Rep_WriteRowCrsData (CrsCod,Role,
2016-10-02 23:26:59 +02:00
FirstClickTimeUTC,tm_FirstClickTime,
2016-10-02 19:42:47 +02:00
MaxHitsPerYear);
}
/* End table */
fprintf (Gbl.F.Out,"</ol>");
}
/***** Free structure that stores the query result *****/
DB_FreeMySQLResult (&mysql_res);
}
fprintf (Gbl.F.Out,"</li>");
}
/*****************************************************************************/
/************************** Write courses of a user **************************/
/*****************************************************************************/
static void Rep_GetAndWriteHistoricCrssOfAUsr (const struct UsrData *UsrDat,Rol_Role_t Role,
time_t FirstClickTimeUTC,
struct tm *tm_FirstClickTime,
unsigned long MaxHitsPerYear)
{
2016-10-02 22:54:30 +02:00
extern const char *Txt_Hits_as_a_USER;
2016-10-02 19:42:47 +02:00
extern const char *Txt_ROLES_SINGUL_abc[Rol_NUM_ROLES][Usr_NUM_SEXS];
2016-09-11 21:25:14 +02:00
char Query[1024];
MYSQL_RES *mysql_res;
MYSQL_ROW row;
unsigned NumCrss;
unsigned NumCrs;
2016-10-02 19:42:47 +02:00
long CrsCod;
2016-09-11 21:25:14 +02:00
/***** Get courses of a user from database *****/
2016-10-02 19:42:47 +02:00
sprintf (Query,"SELECT CrsCod,"
"COUNT(*) AS N"
" FROM log_full"
" WHERE UsrCod='%ld' AND Role='%u'"
" GROUP BY CrsCod ORDER BY N DESC",
2016-09-11 21:25:14 +02:00
UsrDat->UsrCod,(unsigned) Role);
/***** List the courses (one row per course) *****/
if ((NumCrss = (unsigned) DB_QuerySELECT (Query,&mysql_res,"can not get courses of a user")))
{
/* Heading row */
2016-10-02 22:54:30 +02:00
sprintf (Gbl.Title,Txt_Hits_as_a_USER,
2016-10-02 19:42:47 +02:00
Txt_ROLES_SINGUL_abc[Role][Gbl.Usrs.Me.UsrDat.Sex]);
fprintf (Gbl.F.Out,"<li>%s:"
"<ol>",
Gbl.Title);
2016-09-11 21:25:14 +02:00
/* Write courses */
for (NumCrs = 1;
NumCrs <= NumCrss;
NumCrs++)
{
/* Get next course */
row = mysql_fetch_row (mysql_res);
2016-10-02 19:42:47 +02:00
/* Get course code (row[0]) */
CrsCod = Str_ConvertStrCodToLongCod (row[0]);
2016-09-11 21:25:14 +02:00
/* Write data of this course */
2016-10-02 19:42:47 +02:00
Rep_WriteRowCrsData (CrsCod,Role,
2016-10-02 23:26:59 +02:00
FirstClickTimeUTC,tm_FirstClickTime,
2016-10-02 17:56:05 +02:00
MaxHitsPerYear);
2016-09-11 21:25:14 +02:00
}
2016-10-02 19:42:47 +02:00
/* End of list */
fprintf (Gbl.F.Out,"</ol>"
"</li>");
2016-09-11 21:25:14 +02:00
}
/***** Free structure that stores the query result *****/
DB_FreeMySQLResult (&mysql_res);
}
/*****************************************************************************/
/************** Write the data of a course (result of a query) ***************/
/*****************************************************************************/
2016-10-02 19:42:47 +02:00
static void Rep_WriteRowCrsData (long CrsCod,Rol_Role_t Role,
2016-09-29 01:34:17 +02:00
time_t FirstClickTimeUTC,
2016-09-29 01:48:52 +02:00
struct tm *tm_FirstClickTime,
2016-10-02 17:56:05 +02:00
unsigned long MaxHitsPerYear)
2016-09-11 21:25:14 +02:00
{
extern const char *Txt_YEAR_OF_DEGREE[1+Deg_MAX_YEARS_PER_DEGREE];
extern const char *Txt_teachers_ABBREVIATION;
extern const char *Txt_students_ABBREVIATION;
2016-10-02 20:03:21 +02:00
extern const char *Txt_unknown_removed_course;
2016-10-02 19:55:12 +02:00
extern const char *Txt_no_course_selected;
2016-10-02 19:42:47 +02:00
struct Course Crs;
2016-09-11 21:25:14 +02:00
struct Degree Deg;
2016-10-02 19:42:47 +02:00
/***** Get course data *****/
Crs.CrsCod = CrsCod;
Crs_GetDataOfCourseByCod (&Crs);
/***** Get degree data *****/
Deg.DegCod = Crs.DegCod;
Deg_GetDataOfDegreeByCod (&Deg);
2016-09-11 21:25:14 +02:00
/***** Start row *****/
fprintf (Gbl.F.Out,"<li>");
2016-10-02 19:42:47 +02:00
if (CrsCod > 0) // CrsCod > 0 in log ==> course selected
{
if (Crs.CrsCod > 0) // Course exists
{
/***** Write course full name *****/
fprintf (Gbl.F.Out,"<strong>%s</strong> -",Crs.FullName);
2016-09-11 21:45:29 +02:00
2016-10-02 19:42:47 +02:00
/***** Write year *****/
if (Crs.Year)
fprintf (Gbl.F.Out," %s",Txt_YEAR_OF_DEGREE[Crs.Year]);
2016-09-11 21:45:29 +02:00
2016-10-02 19:42:47 +02:00
/***** Write degree full name *****/
fprintf (Gbl.F.Out," %s",Deg.FullName);
2016-09-11 21:25:14 +02:00
2016-10-02 19:42:47 +02:00
/***** Write number of teachers / students in course *****/
fprintf (Gbl.F.Out," (%u %s / %u %s)",
Usr_GetNumUsrsInCrs (Rol_TEACHER,Crs.CrsCod),Txt_teachers_ABBREVIATION,
Usr_GetNumUsrsInCrs (Rol_STUDENT,Crs.CrsCod),Txt_students_ABBREVIATION);
}
else
2016-10-02 20:03:21 +02:00
fprintf (Gbl.F.Out,"(%s)",Txt_unknown_removed_course);
2016-10-02 19:42:47 +02:00
}
else // CrsCod <= 0 in log ==> no course selected
2016-10-02 19:55:12 +02:00
fprintf (Gbl.F.Out,"(%s)",Txt_no_course_selected);
2016-09-29 01:34:17 +02:00
/***** Write hits per year for this course *****/
2016-10-02 19:42:47 +02:00
fprintf (Gbl.F.Out,"<br />");
Rep_ShowMyHitsPerYear (false,CrsCod,Role,
2016-10-02 23:26:59 +02:00
FirstClickTimeUTC,tm_FirstClickTime,
2016-10-02 17:56:05 +02:00
MaxHitsPerYear);
2016-09-29 01:34:17 +02:00
fprintf (Gbl.F.Out,"</li>");
2016-09-11 21:25:14 +02:00
}
2016-09-12 00:50:24 +02:00
2016-10-02 17:56:05 +02:00
/*****************************************************************************/
/********************** Write my hits grouped by years ***********************/
/*****************************************************************************/
2016-10-02 19:42:47 +02:00
static void Rep_ShowMyHitsPerYear (bool AnyCourse,long CrsCod,Rol_Role_t Role,
2016-10-02 17:56:05 +02:00
time_t FirstClickTimeUTC,
struct tm *tm_FirstClickTime,
unsigned long MaxHitsPerYear)
{
char Query[1024];
char SubQueryCrs[128];
2016-10-02 19:42:47 +02:00
char SubQueryRol[128];
2016-09-12 17:43:23 +02:00
MYSQL_RES *mysql_res;
MYSQL_ROW row;
unsigned long NumRows;
unsigned long NumRow;
unsigned ReadYear;
unsigned LastYear;
unsigned Year;
struct Sta_Hits Hits;
/***** Make the query *****/
2016-10-02 19:42:47 +02:00
if (AnyCourse)
SubQueryCrs[0] = '\0';
else
2016-10-02 17:56:05 +02:00
sprintf (SubQueryCrs," AND CrsCod='%ld'",CrsCod);
2016-10-02 19:42:47 +02:00
if (Role == Rol_UNKNOWN) // Here Rol_UNKNOWN means any role
SubQueryRol[0] = '\0';
2016-09-29 01:34:17 +02:00
else
2016-10-02 19:42:47 +02:00
sprintf (SubQueryRol," AND Role='%u'",(unsigned) Role);
2016-10-02 17:56:05 +02:00
2016-09-12 17:43:23 +02:00
sprintf (Query,"SELECT SQL_NO_CACHE "
2016-10-02 23:26:59 +02:00
"YEAR(CONVERT_TZ(ClickTime,@@session.time_zone,'UTC')) AS Year,"
2016-09-12 17:43:23 +02:00
"COUNT(*) FROM log_full"
" WHERE ClickTime>=FROM_UNIXTIME('%ld')"
2016-10-02 19:42:47 +02:00
" AND UsrCod='%ld'%s%s"
2016-09-12 17:43:23 +02:00
" GROUP BY Year DESC",
(long) FirstClickTimeUTC,
2016-09-29 01:34:17 +02:00
Gbl.Usrs.Me.UsrDat.UsrCod,
2016-10-02 19:42:47 +02:00
SubQueryCrs,
SubQueryRol);
2016-09-12 17:43:23 +02:00
NumRows = DB_QuerySELECT (Query,&mysql_res,"can not get clicks");
/***** Initialize first year *****/
Gbl.DateRange.DateIni.Date.Year = 1900 + tm_FirstClickTime->tm_year;
/***** Initialize LastYear *****/
LastYear = Gbl.Now.Date.Year;
2016-10-02 17:56:05 +02:00
/***** Set maximum number of hits per year *****/
if (MaxHitsPerYear)
/* Set maximum number of hits per year from parameter */
Hits.Max = (float) MaxHitsPerYear;
2016-09-29 01:48:52 +02:00
else
2016-10-02 17:56:05 +02:00
{
/* Compute maximum number of hits per year */
2016-09-29 01:48:52 +02:00
Sta_ComputeMaxAndTotalHits (&Hits,NumRows,mysql_res,1,1);
2016-10-02 17:56:05 +02:00
mysql_data_seek (mysql_res,0);
}
2016-09-12 17:43:23 +02:00
/***** Write rows *****/
for (NumRow = 1;
NumRow <= NumRows;
NumRow++)
{
row = mysql_fetch_row (mysql_res);
/* Get the year (in row[0] is the date in YYYY format) */
if (sscanf (row[0],"%04u",&ReadYear) != 1)
Lay_ShowErrorAndExit ("Wrong date.");
/* Get number hits (in row[1]) */
Hits.Num = Str_GetFloatNumFromStr (row[1]);
for (Year = LastYear;
Year >= ReadYear;
Year--)
{
/* Write the year */
fprintf (Gbl.F.Out,"%04u ",Year);
/* Draw bar proportional to number of hits */
Rep_DrawBarNumHits (Year == LastYear ? Hits.Num :
0.0,
Hits.Max,Rep_MAX_BAR_WIDTH);
}
LastYear = Year;
}
/***** Finally, show the oldest years without clicks *****/
for (Year = LastYear;
Year >= Gbl.DateRange.DateIni.Date.Year;
Year--)
{
/* Write the year */
fprintf (Gbl.F.Out,"%04u ",Year);
/* Draw bar proportional to number of hits */
Rep_DrawBarNumHits (0.0,Hits.Max,Rep_MAX_BAR_WIDTH);
}
}
2016-09-12 01:06:36 +02:00
/*****************************************************************************/
/********************* Draw a bar with the number of hits ********************/
/*****************************************************************************/
2016-09-12 10:02:17 +02:00
static void Rep_DrawBarNumHits (float HitsNum,float HitsMax,
unsigned MaxBarWidth)
2016-09-12 01:06:36 +02:00
{
unsigned BarWidth;
2016-09-12 10:02:17 +02:00
unsigned i;
2016-09-12 01:06:36 +02:00
if (HitsNum != 0.0)
{
/***** Draw bar with a with proportional to the number of hits *****/
BarWidth = (unsigned) (((HitsNum * (float) MaxBarWidth) / HitsMax) + 0.5);
2016-09-12 10:02:17 +02:00
for (i = 0;
i < BarWidth;
i++)
fprintf (Gbl.F.Out,Rep_BLOCK);
2016-09-12 01:06:36 +02:00
/***** Write the number of hits *****/
2016-09-12 10:02:17 +02:00
fprintf (Gbl.F.Out," ");
2016-09-12 01:06:36 +02:00
Str_WriteFloatNum (HitsNum);
}
else
/***** Write the number of clicks *****/
fprintf (Gbl.F.Out,"0");
2016-09-12 10:02:17 +02:00
fprintf (Gbl.F.Out,"<br />");
2016-09-12 01:06:36 +02:00
}