swad-core/swad_notice.c

910 lines
34 KiB
C
Raw Normal View History

2014-12-01 23:55:08 +01:00
// swad_notice.c: notices (yellow notes)
/*
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.
2015-01-01 14:34:06 +01:00
Copyright (C) 1999-2015 Antonio Ca<EFBFBD>as Vargas
2014-12-01 23:55:08 +01:00
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public 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 <linux/limits.h> // For PATH_MAX
#include <linux/stddef.h> // For NULL
#include <stdlib.h> // For exit, system, malloc, calloc, free, etc.
#include <string.h>
#include "swad_database.h"
#include "swad_global.h"
#include "swad_notice.h"
#include "swad_notification.h"
#include "swad_parameter.h"
#include "swad_RSS.h"
/*****************************************************************************/
/************** External global variables from others modules ****************/
/*****************************************************************************/
extern struct Globals Gbl;
/*****************************************************************************/
/***************************** Private constants *****************************/
/*****************************************************************************/
#define Not_MAX_CHARS_ON_NOTICE 40 // Maximum number of characters in notices (when not expanded)
const unsigned Not_ContainerWidth[Not_NUM_TYPES_LISTING] =
{
2015-09-28 18:28:29 +02:00
125, // Not_LIST_BRIEF_NOTICES
500, // Not_LIST_FULL_NOTICES
2014-12-01 23:55:08 +01:00
};
const unsigned Not_MaxCharsURLOnScreen[Not_NUM_TYPES_LISTING] =
{
15, // Not_LIST_BRIEF_NOTICES
50, // Not_LIST_FULL_NOTICES
};
/*****************************************************************************/
/***************************** Private prototypes ****************************/
/*****************************************************************************/
static void Not_DrawANotice (Not_Listing_t TypeNoticesListing,
long NotCod,
2015-10-26 14:49:42 +01:00
time_t TimeUTC,
2014-12-01 23:55:08 +01:00
const char *Content,
long UsrCod,
2015-11-01 20:21:59 +01:00
Not_Status_t Status,
2015-11-01 18:54:10 +01:00
bool ICanEditNotices);
2014-12-01 23:55:08 +01:00
static long Not_InsertNoticeInDB (const char *Content);
static void Not_UpdateNumUsrsNotifiedByEMailAboutNotice (long NotCod,unsigned NumUsrsToBeNotifiedByEMail);
static void Not_PutHiddenParamNotCod (long NotCod);
static long Not_GetParamNotCod (void);
/*****************************************************************************/
/***************************** Write a new notice ****************************/
/*****************************************************************************/
void Not_ShowFormNotice (void)
{
2015-07-28 00:16:09 +02:00
extern const char *The_ClassForm[The_NUM_THEMES];
2014-12-01 23:55:08 +01:00
extern const char *Txt_The_notice_you_enter_here_will_appear_as_a_yellow_note_;
extern const char *Txt_New_notice;
extern const char *Txt_MSG_Message;
extern const char *Txt_Create_notice;
sprintf (Gbl.Message,Txt_The_notice_you_enter_here_will_appear_as_a_yellow_note_,
Gbl.CurrentCrs.Crs.FullName);
Lay_ShowAlert (Lay_INFO,Gbl.Message);
2015-04-11 17:33:14 +02:00
/***** Start form *****/
2014-12-01 23:55:08 +01:00
Act_FormStart (ActRcvNot);
2015-03-24 17:47:26 +01:00
/***** Start frame *****/
2015-04-12 18:01:06 +02:00
Lay_StartRoundFrameTable (NULL,2,Txt_New_notice);
2015-03-24 17:47:26 +01:00
/***** Message body *****/
fprintf (Gbl.F.Out,"<tr>"
2015-07-28 00:16:09 +02:00
"<td class=\"%s RIGHT_TOP\">"
2014-12-23 22:47:09 +01:00
"%s: "
"</td>"
2015-08-06 14:16:11 +02:00
"<td class=\"LEFT_TOP\">"
"<textarea name=\"Content\" cols=\"30\" rows=\"10\">"
"</textarea>"
2014-12-01 23:55:08 +01:00
"</td>"
2015-03-24 17:47:26 +01:00
"</tr>",
2015-07-28 00:16:09 +02:00
The_ClassForm[Gbl.Prefs.Theme],
2014-12-01 23:55:08 +01:00
Txt_MSG_Message);
2015-03-24 17:47:26 +01:00
2015-04-11 23:46:21 +02:00
/***** Button to create notice and end frame *****/
2015-04-12 18:01:06 +02:00
Lay_EndRoundFrameTableWithButton (Lay_CREATE_BUTTON,Txt_Create_notice);
2015-03-24 17:47:26 +01:00
2015-04-11 17:33:14 +02:00
/***** End form *****/
2015-03-13 00:16:02 +01:00
Act_FormEnd ();
2014-12-01 23:55:08 +01:00
}
/*****************************************************************************/
/******* Receive a new notice from a form and store it in database ***********/
/*****************************************************************************/
void Not_ReceiveNotice (void)
{
extern const char *Txt_Notice_created;
long NotCod;
unsigned NumUsrsToBeNotifiedByEMail;
char Content[Cns_MAX_BYTES_TEXT+1];
/***** Get the text of the notice *****/
Par_GetParAndChangeFormat ("Content",Content,Cns_MAX_BYTES_TEXT,
Str_TO_RIGOROUS_HTML,true);
/***** Create a new notice in database *****/
NotCod = Not_InsertNoticeInDB (Content);
/***** Update RSS of current course *****/
RSS_UpdateRSSFileForACrs (&Gbl.CurrentCrs.Crs);
/***** Write message of success *****/
Lay_ShowAlert (Lay_SUCCESS,Txt_Notice_created);
/***** Notify by e-mail about the new notice *****/
if ((NumUsrsToBeNotifiedByEMail = Ntf_StoreNotifyEventsToAllUsrs (Ntf_EVENT_NOTICE,NotCod)))
Not_UpdateNumUsrsNotifiedByEMailAboutNotice (NotCod,NumUsrsToBeNotifiedByEMail);
Ntf_ShowAlertNumUsrsToBeNotifiedByEMail (NumUsrsToBeNotifiedByEMail);
}
/*****************************************************************************/
/******************* Insert a notice in the table of notices *****************/
/*****************************************************************************/
// Return the code of the new inserted notice
static long Not_InsertNoticeInDB (const char *Content)
{
char Query[256+Cns_MAX_BYTES_TEXT];
/***** Insert notice in the database *****/
sprintf (Query,"INSERT INTO notices (CrsCod,UsrCod,CreatTime,Content,Status)"
" VALUES ('%ld','%ld',NOW(),'%s','%u')",
Gbl.CurrentCrs.Crs.CrsCod,Gbl.Usrs.Me.UsrDat.UsrCod,Content,(unsigned) Not_ACTIVE_NOTICE);
return DB_QueryINSERTandReturnCode (Query,"can not create notice");
}
/*****************************************************************************/
/*********** Update number of users notified in table of notices *************/
/*****************************************************************************/
static void Not_UpdateNumUsrsNotifiedByEMailAboutNotice (long NotCod,unsigned NumUsrsToBeNotifiedByEMail)
{
char Query[512];
/***** Update number of users notified *****/
sprintf (Query,"UPDATE notices SET NumNotif='%u' WHERE NotCod='%ld'",
NumUsrsToBeNotifiedByEMail,NotCod);
DB_QueryUPDATE (Query,"can not update the number of notifications of a notice");
}
/*****************************************************************************/
/******************************* List notices ********************************/
/*****************************************************************************/
void Not_ListNotices (void)
{
Not_ShowNotices (Not_LIST_FULL_NOTICES);
}
/*****************************************************************************/
/***************** Mark as hidden a notice that was active *******************/
/*****************************************************************************/
void Not_HideActiveNotice (void)
{
char Query[256];
long NotCod;
/***** Get the code of the notice to hide *****/
NotCod = Not_GetParamNotCod ();
/***** Set notice as hidden *****/
sprintf (Query,"UPDATE notices SET Status='%u'"
" WHERE NotCod='%ld' AND CrsCod='%ld'",
(unsigned) Not_OBSOLETE_NOTICE,
NotCod,Gbl.CurrentCrs.Crs.CrsCod);
DB_QueryUPDATE (Query,"can not hide notice");
/***** Update RSS of current course *****/
RSS_UpdateRSSFileForACrs (&Gbl.CurrentCrs.Crs);
}
/*****************************************************************************/
/****************** Mark as active a notice that was hidden ******************/
/*****************************************************************************/
void Not_RevealHiddenNotice (void)
{
char Query[256];
long NotCod;
2015-11-01 20:21:59 +01:00
/***** Get the code of the notice to reveal *****/
2014-12-01 23:55:08 +01:00
NotCod = Not_GetParamNotCod ();
2015-11-01 20:21:59 +01:00
/***** Set notice as active *****/
2014-12-01 23:55:08 +01:00
sprintf (Query,"UPDATE notices SET Status='%u'"
" WHERE NotCod='%ld' AND CrsCod='%ld'",
(unsigned) Not_ACTIVE_NOTICE,
NotCod,Gbl.CurrentCrs.Crs.CrsCod);
2015-11-01 20:21:59 +01:00
DB_QueryUPDATE (Query,"can not reveal notice");
2014-12-01 23:55:08 +01:00
/***** Update RSS of current course *****/
RSS_UpdateRSSFileForACrs (&Gbl.CurrentCrs.Crs);
}
/*****************************************************************************/
2015-11-01 20:21:59 +01:00
/******************************* Remove a notice *****************************/
2014-12-01 23:55:08 +01:00
/*****************************************************************************/
void Not_DeleteNotice (void)
{
char Query[512];
long NotCod;
/***** Get the code of the notice to delete *****/
NotCod = Not_GetParamNotCod ();
/***** Remove notice *****/
/* Copy notice to table of deleted notices */
sprintf (Query,"INSERT IGNORE INTO notices_deleted (NotCod,CrsCod,UsrCod,CreatTime,Content,NumNotif)"
" SELECT NotCod,CrsCod,UsrCod,CreatTime,Content,NumNotif"
" FROM notices"
" WHERE NotCod='%ld' AND CrsCod='%ld'",
NotCod,Gbl.CurrentCrs.Crs.CrsCod);
DB_QueryINSERT (Query,"can not remove notice");
/* Remove notice */
sprintf (Query,"DELETE FROM notices"
" WHERE NotCod='%ld' AND CrsCod='%ld'",
NotCod,Gbl.CurrentCrs.Crs.CrsCod);
DB_QueryDELETE (Query,"can not remove notice");
/***** Mark possible notifications as removed *****/
Ntf_SetNotifAsRemoved (Ntf_EVENT_NOTICE,NotCod);
/***** Update RSS of current course *****/
RSS_UpdateRSSFileForACrs (&Gbl.CurrentCrs.Crs);
}
/*****************************************************************************/
/**************************** Show (expand) a notice *************************/
/*****************************************************************************/
void Not_ShowANotice (void)
{
/***** Get the code of the notice to highlight *****/
Gbl.CurrentCrs.Notices.HighlightNotCod = Not_GetParamNotCod ();
}
/*****************************************************************************/
/***************************** Show the notices ******************************/
/*****************************************************************************/
void Not_ShowNotices (Not_Listing_t TypeNoticesListing)
{
extern const char *Txt_New_notice;
extern const char *Txt_All_notices;
2015-11-01 18:54:10 +01:00
extern const char *Txt_No_notices;
2014-12-01 23:55:08 +01:00
char Query[512];
MYSQL_RES *mysql_res;
MYSQL_ROW row;
char StrWidth[10+2+1];
char PathRelRSSFile[PATH_MAX+1];
long NotCod;
2015-10-26 14:49:42 +01:00
unsigned long NumNot;
unsigned long NumNotices;
2014-12-01 23:55:08 +01:00
char Content[Cns_MAX_BYTES_TEXT+1];
2015-10-26 14:49:42 +01:00
time_t TimeUTC;
2014-12-01 23:55:08 +01:00
long UsrCod;
unsigned UnsignedNum;
2015-11-01 20:21:59 +01:00
Not_Status_t Status;
2014-12-01 23:55:08 +01:00
bool ICanEditNotices;
/***** A course must be selected (Gbl.CurrentCrs.Crs.CrsCod > 0) *****/
if (Gbl.CurrentCrs.Crs.CrsCod > 0)
{
2015-11-01 18:54:10 +01:00
ICanEditNotices = (TypeNoticesListing == Not_LIST_FULL_NOTICES &&
2015-04-07 21:44:24 +02:00
(Gbl.Usrs.Me.LoggedRole == Rol_TEACHER ||
2015-11-01 18:54:10 +01:00
Gbl.Usrs.Me.LoggedRole == Rol_SYS_ADM));
2014-12-01 23:55:08 +01:00
if (ICanEditNotices)
{
2015-04-02 14:22:21 +02:00
fprintf (Gbl.F.Out,"<div class=\"CONTEXT_MENU\">");
2015-04-02 18:39:49 +02:00
Act_PutContextualLink (ActWriNot,NULL,"new",Txt_New_notice);
2015-04-02 13:38:05 +02:00
fprintf (Gbl.F.Out,"</div>");
2014-12-01 23:55:08 +01:00
}
/***** Show highlighted notice *****/
if (TypeNoticesListing == Not_LIST_FULL_NOTICES &&
Gbl.CurrentCrs.Notices.HighlightNotCod > 0)
{
2015-10-26 14:49:42 +01:00
sprintf (Query,"SELECT UNIX_TIMESTAMP(CreatTime) AS F,UsrCod,Content,Status"
2014-12-01 23:55:08 +01:00
" FROM notices"
" WHERE NotCod='%ld' AND CrsCod='%ld'",
Gbl.CurrentCrs.Notices.HighlightNotCod,
Gbl.CurrentCrs.Crs.CrsCod);
if (DB_QuerySELECT (Query,&mysql_res,"can not get notice from database"))
{
row = mysql_fetch_row (mysql_res);
2015-10-26 14:49:42 +01:00
/* Get creation time (row[0] holds the UTC date-time) */
TimeUTC = Dat_GetUNIXTimeFromStr (row[0]);
/* Get user code (row[1]) */
2014-12-01 23:55:08 +01:00
UsrCod = Str_ConvertStrCodToLongCod (row[1]);
2015-10-26 14:49:42 +01:00
/* Get the content (row[2]) and insert links*/
2014-12-01 23:55:08 +01:00
strncpy (Content,row[2],Cns_MAX_BYTES_TEXT);
Str_InsertLinkInURLs (Content,Cns_MAX_BYTES_TEXT,
Not_MaxCharsURLOnScreen[TypeNoticesListing]);
if (TypeNoticesListing == Not_LIST_BRIEF_NOTICES)
Str_LimitLengthHTMLStr (Content,Not_MAX_CHARS_ON_NOTICE);
2015-10-26 14:49:42 +01:00
/* Get status of the notice (row[3]) */
2015-11-01 20:21:59 +01:00
Status = Not_OBSOLETE_NOTICE;
2014-12-01 23:55:08 +01:00
if (sscanf (row[3],"%u",&UnsignedNum) == 1)
if (UnsignedNum < Not_NUM_STATUS)
2015-11-01 20:21:59 +01:00
Status = (Not_Status_t) UnsignedNum;
2014-12-01 23:55:08 +01:00
/* Draw the notice */
2015-11-01 18:54:10 +01:00
Not_DrawANotice (TypeNoticesListing,
2015-10-26 14:49:42 +01:00
Gbl.CurrentCrs.Notices.HighlightNotCod,
2015-11-01 20:21:59 +01:00
TimeUTC,Content,UsrCod,Status,
2015-11-01 18:54:10 +01:00
ICanEditNotices);
2014-12-01 23:55:08 +01:00
}
/* Free structure that stores the query result */
DB_FreeMySQLResult (&mysql_res);
}
/***** Get notices from database *****/
switch (TypeNoticesListing)
{
case Not_LIST_BRIEF_NOTICES:
2015-10-26 14:49:42 +01:00
sprintf (Query,"SELECT NotCod,UNIX_TIMESTAMP(CreatTime) AS F,UsrCod,Content,Status"
2014-12-01 23:55:08 +01:00
" FROM notices"
" WHERE CrsCod='%ld' AND Status='%u'"
" ORDER BY CreatTime DESC",
Gbl.CurrentCrs.Crs.CrsCod,
(unsigned) Not_ACTIVE_NOTICE);
break;
case Not_LIST_FULL_NOTICES:
2015-10-26 14:49:42 +01:00
sprintf (Query,"SELECT NotCod,UNIX_TIMESTAMP(CreatTime) AS F,UsrCod,Content,Status"
2014-12-01 23:55:08 +01:00
" FROM notices"
" WHERE CrsCod='%ld'"
" ORDER BY CreatTime DESC",
Gbl.CurrentCrs.Crs.CrsCod);
break;
}
NumNotices = DB_QuerySELECT (Query,&mysql_res,"can not get notices from database");
if (TypeNoticesListing == Not_LIST_FULL_NOTICES)
{
if (NumNotices)
{
2015-11-01 18:54:10 +01:00
/***** Start frame *****/
2015-09-24 18:02:21 +02:00
sprintf (StrWidth,"%upx",
2015-09-28 18:28:29 +02:00
Not_ContainerWidth[Not_LIST_FULL_NOTICES] + 50);
2015-11-01 18:54:10 +01:00
Lay_StartRoundFrame (StrWidth,Txt_All_notices);
2014-12-01 23:55:08 +01:00
}
else
Lay_ShowAlert (Lay_INFO,Txt_No_notices);
}
/***** Link to RSS file *****/
if (TypeNoticesListing == Not_LIST_BRIEF_NOTICES)
{
/* Create RSS file if not exists */
sprintf (PathRelRSSFile,"%s/%s/%ld/%s/%s",
Cfg_PATH_SWAD_PUBLIC,Cfg_FOLDER_CRS,Gbl.CurrentCrs.Crs.CrsCod,Cfg_RSS_FOLDER,Cfg_RSS_FILE);
if (!Fil_CheckIfPathExists (PathRelRSSFile))
RSS_UpdateRSSFileForACrs (&Gbl.CurrentCrs.Crs);
/* Put a link to the RSS file */
2015-08-06 14:16:11 +02:00
fprintf (Gbl.F.Out,"<div class=\"CENTER_MIDDLE\">"
2014-12-01 23:55:08 +01:00
"<a href=\"");
RSS_WriteRSSLink (Gbl.F.Out,Gbl.CurrentCrs.Crs.CrsCod);
fprintf (Gbl.F.Out,"\" target=\"_blank\">"
"<img src=\"%s/rss16x16.gif\""
2015-07-22 13:32:56 +02:00
" alt=\"RSS\" title=\"RSS\""
" class=\"ICON16x16\" />"
2014-12-01 23:55:08 +01:00
"</a>"
"</div>",
Gbl.Prefs.IconsURL);
}
/***** Show the notices *****/
for (NumNot = 0;
NumNot < NumNotices;
NumNot++)
{
row = mysql_fetch_row (mysql_res);
2015-10-26 14:49:42 +01:00
/* Get notice code (row[0]) */
2014-12-01 23:55:08 +01:00
if (sscanf (row[0],"%ld",&NotCod) != 1)
Lay_ShowErrorAndExit ("Wrong code of notice.");
2015-10-26 14:49:42 +01:00
/* Get creation time (row[1] holds the UTC date-time) */
TimeUTC = Dat_GetUNIXTimeFromStr (row[1]);
/* Get user code (row[2]) */
2014-12-01 23:55:08 +01:00
UsrCod = Str_ConvertStrCodToLongCod (row[2]);
2015-10-26 14:49:42 +01:00
/* Get the content (row[3]) and insert links */
2014-12-01 23:55:08 +01:00
strncpy (Content,row[3],Cns_MAX_BYTES_TEXT);
Str_InsertLinkInURLs (Content,Cns_MAX_BYTES_TEXT,
Not_MaxCharsURLOnScreen[TypeNoticesListing]);
if (TypeNoticesListing == Not_LIST_BRIEF_NOTICES)
Str_LimitLengthHTMLStr (Content,Not_MAX_CHARS_ON_NOTICE);
2015-10-26 14:49:42 +01:00
/* Get status of the notice (row[4]) */
2015-11-01 20:21:59 +01:00
Status = Not_OBSOLETE_NOTICE;
2014-12-01 23:55:08 +01:00
if (sscanf (row[4],"%u",&UnsignedNum) == 1)
if (UnsignedNum < Not_NUM_STATUS)
2015-11-01 20:21:59 +01:00
Status = (Not_Status_t) UnsignedNum;
2014-12-01 23:55:08 +01:00
/* Draw the notice */
2015-11-01 18:54:10 +01:00
Not_DrawANotice (TypeNoticesListing,
2015-10-26 14:49:42 +01:00
NotCod,
2015-11-01 20:21:59 +01:00
TimeUTC,Content,UsrCod,Status,
2015-11-01 18:54:10 +01:00
ICanEditNotices);
2014-12-01 23:55:08 +01:00
}
if (TypeNoticesListing == Not_LIST_FULL_NOTICES && NumNotices)
2015-11-01 18:54:10 +01:00
/***** End frame *****/
Lay_EndRoundFrame ();
2014-12-01 23:55:08 +01:00
/***** Free structure that stores the query result *****/
DB_FreeMySQLResult (&mysql_res);
/***** Mark possible notification as seen *****/
Ntf_SetNotifAsSeen (Ntf_EVENT_NOTICE,
-1L,
Gbl.Usrs.Me.UsrDat.UsrCod);
}
}
/*****************************************************************************/
/********************* Draw a notice as a yellow note ************************/
/*****************************************************************************/
static void Not_DrawANotice (Not_Listing_t TypeNoticesListing,
long NotCod,
2015-10-26 14:49:42 +01:00
time_t TimeUTC,
2014-12-01 23:55:08 +01:00
const char *Content,
long UsrCod,
2015-11-01 20:21:59 +01:00
Not_Status_t Status,
2015-11-01 18:54:10 +01:00
bool ICanEditNotices)
2014-12-01 23:55:08 +01:00
{
2015-07-28 00:16:09 +02:00
extern const char *The_ClassForm[The_NUM_THEMES];
2014-12-01 23:55:08 +01:00
extern const char *Txt_NOTICE_Active_SINGULAR;
extern const char *Txt_NOTICE_Active_Mark_as_obsolete;
extern const char *Txt_NOTICE_Obsolete_SINGULAR;
extern const char *Txt_NOTICE_Obsolete_Mark_as_active;
extern const char *Txt_See_full_notice;
extern const char *Txt_Remove;
2015-11-01 14:11:46 +01:00
static const char *ContainerClass[Not_NUM_STATUS] =
{
"NOTICE_CONTAINER_ACTIVE", // Not_ACTIVE_NOTICE
"NOTICE_CONTAINER_OBSOLETE", // Not_OBSOLETE_NOTICE
};
static const char *DateClass[Not_NUM_STATUS] =
{
"NOTICE_DATE_ACTIVE", // Not_ACTIVE_NOTICE
"NOTICE_DATE_OBSOLETE", // Not_OBSOLETE_NOTICE
};
static const char *TextClass[Not_NUM_STATUS] =
{
"NOTICE_TEXT_ACTIVE", // Not_ACTIVE_NOTICE
"NOTICE_TEXT_OBSOLETE", // Not_OBSOLETE_NOTICE
};
static const char *AuthorClass[Not_NUM_STATUS] =
{
"NOTICE_AUTHOR_ACTIVE", // Not_ACTIVE_NOTICE
"NOTICE_AUTHOR_OBSOLETE", // Not_OBSOLETE_NOTICE
};
2015-10-26 14:49:42 +01:00
static unsigned UniqueId = 0;
2014-12-01 23:55:08 +01:00
struct UsrData UsrDat;
/***** Start yellow note *****/
2015-11-01 14:11:46 +01:00
fprintf (Gbl.F.Out,"<div class=\"%s\" style=\"width:%upx;\">",
2015-11-01 20:21:59 +01:00
ContainerClass[Status],
2015-09-24 18:02:21 +02:00
Not_ContainerWidth[TypeNoticesListing]);
2014-12-01 23:55:08 +01:00
/***** Write the date in the top part of the yellow note *****/
/* Write symbol to indicate if notice is obsolete or active */
if (TypeNoticesListing == Not_LIST_FULL_NOTICES)
{
2015-11-01 14:11:46 +01:00
if (ICanEditNotices)
{
2015-11-01 18:54:10 +01:00
/* Form to remove notice */
2015-11-01 14:11:46 +01:00
Act_FormStart (ActRemNot);
Not_PutHiddenParamNotCod (NotCod);
fprintf (Gbl.F.Out,"<div class=\"CONTEXT_OPT ICON_HIGHLIGHT\">"
"<input type=\"image\""
" src=\"%s/delon16x16.gif\""
" alt=\"%s\" title=\"%s\""
" class=\"ICON16x16\" />"
"</div>",
Gbl.Prefs.IconsURL,
Txt_Remove,
Txt_Remove);
Act_FormEnd ();
2015-11-01 20:21:59 +01:00
/* Put form to change the status of the notice */
switch (Status)
2014-12-01 23:55:08 +01:00
{
case Not_ACTIVE_NOTICE:
Act_FormStart (ActHidNot);
Not_PutHiddenParamNotCod (NotCod);
2015-11-01 14:11:46 +01:00
fprintf (Gbl.F.Out,"<div class=\"CONTEXT_OPT ICON_HIGHLIGHT\">"
"<input type=\"image\""
2015-09-05 12:04:30 +02:00
" src=\"%s/visible_on16x16.gif\""
" alt=%s\" title=\"%s\""
2015-11-01 14:11:46 +01:00
" class=\"ICON16x16\" />"
"</div>",
2014-12-01 23:55:08 +01:00
Gbl.Prefs.IconsURL,
Txt_NOTICE_Active_Mark_as_obsolete,
Txt_NOTICE_Active_Mark_as_obsolete);
break;
case Not_OBSOLETE_NOTICE:
Act_FormStart (ActRevNot);
Not_PutHiddenParamNotCod (NotCod);
2015-11-01 14:11:46 +01:00
fprintf (Gbl.F.Out,"<div class=\"CONTEXT_OPT ICON_HIGHLIGHT\">"
"<input type=\"image\""
2015-09-05 12:04:30 +02:00
" src=\"%s/hidden_on16x16.gif\""
" alt=\"%s\" title=\"%s\""
2015-11-01 14:11:46 +01:00
" class=\"ICON16x16\" />"
"</div>",
2014-12-01 23:55:08 +01:00
Gbl.Prefs.IconsURL,
Txt_NOTICE_Obsolete_Mark_as_active,
Txt_NOTICE_Obsolete_Mark_as_active);
break;
}
2015-11-01 14:11:46 +01:00
Act_FormEnd ();
}
else // Don't put forms
/* Status of the notice */
2015-11-01 20:21:59 +01:00
switch (Status)
2014-12-01 23:55:08 +01:00
{
case Not_ACTIVE_NOTICE:
fprintf (Gbl.F.Out,"<span title=\"%s\">"
"<img src=\"%s/visible_off16x16.gif\""
2015-09-05 12:04:30 +02:00
" alt=\"%s\" title=\"%s\""
" class=\"ICON16x16\" />"
2014-12-01 23:55:08 +01:00
"</span>",
Txt_NOTICE_Active_SINGULAR,
Gbl.Prefs.IconsURL,
2015-07-22 13:32:56 +02:00
Txt_NOTICE_Active_SINGULAR,
2014-12-01 23:55:08 +01:00
Txt_NOTICE_Active_SINGULAR);
break;
case Not_OBSOLETE_NOTICE:
fprintf (Gbl.F.Out,"<span title=\"%s\">"
"<img src=\"%s/hidden_off16x16.gif\""
2015-07-22 13:32:56 +02:00
" alt=\"%s\" title=\"%s\""
" class=\"ICON16x16\" />"
2014-12-01 23:55:08 +01:00
"</span>",
Txt_NOTICE_Obsolete_SINGULAR,
Gbl.Prefs.IconsURL,
2015-07-22 13:32:56 +02:00
Txt_NOTICE_Obsolete_SINGULAR,
2014-12-01 23:55:08 +01:00
Txt_NOTICE_Obsolete_SINGULAR);
break;
}
}
2015-11-01 14:11:46 +01:00
/* Write the date */
2015-10-26 14:49:42 +01:00
UniqueId++;
2015-11-01 14:11:46 +01:00
fprintf (Gbl.F.Out,"<div class=\"%s\">",
2015-11-01 20:21:59 +01:00
DateClass[Status]);
2014-12-01 23:55:08 +01:00
if (TypeNoticesListing == Not_LIST_BRIEF_NOTICES)
{
/* Form to view full notice */
Act_FormStart (ActShoNot);
Not_PutHiddenParamNotCod (NotCod);
2015-11-01 20:21:59 +01:00
Act_LinkFormSubmit (Txt_See_full_notice,DateClass[Status]);
2014-12-01 23:55:08 +01:00
}
2015-10-26 14:49:42 +01:00
fprintf (Gbl.F.Out,"<span id=\"notice_date_%u\"></span>",
UniqueId);
2014-12-01 23:55:08 +01:00
if (TypeNoticesListing == Not_LIST_BRIEF_NOTICES)
2015-03-13 00:16:02 +01:00
{
fprintf (Gbl.F.Out,"</a>");
Act_FormEnd ();
}
2015-10-26 14:49:42 +01:00
fprintf (Gbl.F.Out,"<script type=\"text/javascript\">"
"writeLocalDateTimeFromUTC('notice_date_%u',%ld,'<br />');"
"</script>"
"</div>",
UniqueId,(long) TimeUTC);
2014-12-01 23:55:08 +01:00
/***** Write the content of the notice *****/
if (TypeNoticesListing == Not_LIST_BRIEF_NOTICES)
{
2015-11-01 21:19:33 +01:00
fprintf (Gbl.F.Out,"<div class=\"NOTICE_TEXT_BRIEF\">%s</div>",
Content);
2014-12-01 23:55:08 +01:00
/* Form to view full notice */
2015-11-01 21:19:33 +01:00
fprintf (Gbl.F.Out,"<div class=\"CENTER_MIDDLE\">");
2014-12-01 23:55:08 +01:00
Act_FormStart (ActShoNot);
Not_PutHiddenParamNotCod (NotCod);
2015-07-28 00:16:09 +02:00
Act_LinkFormSubmit (Txt_See_full_notice,The_ClassForm[Gbl.Prefs.Theme]);
2014-12-01 23:55:08 +01:00
fprintf (Gbl.F.Out,"<img src=\"%s/ellipsis32x32.gif\""
2015-07-22 13:32:56 +02:00
" alt=\"%s\" title=\"%s\""
" class=\"ICON32x32\" />"
2015-03-13 00:16:02 +01:00
"</a>",
2014-12-01 23:55:08 +01:00
Gbl.Prefs.IconsURL,
2015-07-22 13:32:56 +02:00
Txt_See_full_notice,
2014-12-01 23:55:08 +01:00
Txt_See_full_notice);
2015-03-13 00:16:02 +01:00
Act_FormEnd ();
fprintf (Gbl.F.Out,"</div>");
2014-12-01 23:55:08 +01:00
}
2015-11-01 21:19:33 +01:00
else
fprintf (Gbl.F.Out,"<div class=\"%s\">%s</div>",
TextClass[Status],Content);
2014-12-01 23:55:08 +01:00
/***** Write the author *****/
2015-11-01 14:11:46 +01:00
fprintf (Gbl.F.Out,"<div class=\"%s\">",
2015-11-01 20:21:59 +01:00
AuthorClass[Status]);
2014-12-01 23:55:08 +01:00
Usr_UsrDataConstructor (&UsrDat);
UsrDat.UsrCod = UsrCod;
if (Usr_ChkUsrCodAndGetAllUsrDataFromUsrCod (&UsrDat)) // Get from the database the data of the autor
Usr_RestrictLengthAndWriteName (&UsrDat,18);
Usr_UsrDataDestructor (&UsrDat);
fprintf (Gbl.F.Out,"</div>");
fprintf (Gbl.F.Out,"</div>");
}
/*****************************************************************************/
/********************* Write first characters of a notice *******************/
/*****************************************************************************/
// This function may be called inside a web service, so don't report error
void Not_GetNotifNotice (char *SummaryStr,char **ContentStr,long NotCod,unsigned MaxChars,bool GetContent)
{
char Query[512];
MYSQL_RES *mysql_res;
MYSQL_ROW row;
SummaryStr[0] = '\0'; // Return nothing on error
/***** Get subject of message from database *****/
sprintf (Query,"SELECT Content FROM notices WHERE NotCod='%ld'",
NotCod);
if (!mysql_query (&Gbl.mysql,Query))
if ((mysql_res = mysql_store_result (&Gbl.mysql)) != NULL)
{
/***** Result should have a unique row *****/
if (mysql_num_rows (mysql_res) == 1)
{
/***** Get sumary / content *****/
row = mysql_fetch_row (mysql_res);
/***** Copy summary *****/
strcpy (SummaryStr,row[0]);
if (MaxChars)
Str_LimitLengthHTMLStr (SummaryStr,MaxChars);
/***** Copy content *****/
if (GetContent)
{
if ((*ContentStr = (char *) malloc (strlen (row[0])+1)) == NULL)
Lay_ShowErrorAndExit ("Error allocating memory for notification content.");
strcpy (*ContentStr,row[0]);
}
}
mysql_free_result (mysql_res);
}
}
/*****************************************************************************/
/*************************** Get number of notices ***************************/
/*****************************************************************************/
// Returns the number of (active or obsolete) notices
// sent from this location (all the platform, current degree or current course)
2015-11-01 20:21:59 +01:00
unsigned Not_GetNumNotices (Sco_Scope_t Scope,Not_Status_t Status,unsigned *NumNotif)
2014-12-01 23:55:08 +01:00
{
char Query[1024];
MYSQL_RES *mysql_res;
MYSQL_ROW row;
unsigned NumNotices;
/***** Get number of notices from database *****/
switch (Scope)
{
2015-02-01 20:17:24 +01:00
case Sco_SCOPE_SYS:
2014-12-01 23:55:08 +01:00
sprintf (Query,"SELECT COUNT(*),SUM(NumNotif)"
" FROM notices"
" WHERE Status='%u'",
2015-11-01 20:21:59 +01:00
Status);
2014-12-01 23:55:08 +01:00
break;
2015-03-09 10:40:31 +01:00
case Sco_SCOPE_CTY:
sprintf (Query,"SELECT COUNT(*),SUM(notices.NumNotif)"
" FROM institutions,centres,degrees,courses,notices"
" WHERE institutions.CtyCod='%ld'"
" AND institutions.InsCod=centres.InsCod"
" AND centres.CtrCod=degrees.CtrCod"
" AND degrees.DegCod=courses.DegCod"
" AND courses.CrsCod=notices.CrsCod"
" AND notices.Status='%u'",
Gbl.CurrentCty.Cty.CtyCod,
2015-11-01 20:21:59 +01:00
Status);
2015-03-09 10:40:31 +01:00
break;
2015-02-01 20:17:24 +01:00
case Sco_SCOPE_INS:
2014-12-01 23:55:08 +01:00
sprintf (Query,"SELECT COUNT(*),SUM(notices.NumNotif)"
" FROM centres,degrees,courses,notices"
" WHERE centres.InsCod='%ld'"
" AND centres.CtrCod=degrees.CtrCod"
" AND degrees.DegCod=courses.DegCod"
" AND courses.CrsCod=notices.CrsCod"
" AND notices.Status='%u'",
Gbl.CurrentIns.Ins.InsCod,
2015-11-01 20:21:59 +01:00
Status);
2014-12-01 23:55:08 +01:00
break;
2015-02-01 20:17:24 +01:00
case Sco_SCOPE_CTR:
2014-12-01 23:55:08 +01:00
sprintf (Query,"SELECT COUNT(*),SUM(notices.NumNotif)"
" FROM degrees,courses,notices"
" WHERE degrees.CtrCod='%ld'"
" AND degrees.DegCod=courses.DegCod"
" AND courses.CrsCod=notices.CrsCod"
" AND notices.Status='%u'",
Gbl.CurrentCtr.Ctr.CtrCod,
2015-11-01 20:21:59 +01:00
Status);
2014-12-01 23:55:08 +01:00
break;
2015-02-01 20:17:24 +01:00
case Sco_SCOPE_DEG:
2014-12-01 23:55:08 +01:00
sprintf (Query,"SELECT COUNT(*),SUM(notices.NumNotif)"
" FROM courses,notices"
" WHERE courses.DegCod='%ld'"
" AND courses.CrsCod=notices.CrsCod"
" AND notices.Status='%u'",
Gbl.CurrentDeg.Deg.DegCod,
2015-11-01 20:21:59 +01:00
Status);
2014-12-01 23:55:08 +01:00
break;
2015-02-01 20:17:24 +01:00
case Sco_SCOPE_CRS:
2014-12-01 23:55:08 +01:00
sprintf (Query,"SELECT COUNT(*),SUM(NumNotif)"
" FROM notices"
" WHERE CrsCod='%ld'"
" AND Status='%u'",
Gbl.CurrentCrs.Crs.CrsCod,
2015-11-01 20:21:59 +01:00
Status);
2014-12-01 23:55:08 +01:00
break;
default:
Lay_ShowErrorAndExit ("Wrong scope.");
break;
}
DB_QuerySELECT (Query,&mysql_res,"can not get number of notices");
/***** Get number of notices *****/
row = mysql_fetch_row (mysql_res);
if (sscanf (row[0],"%u",&NumNotices) != 1)
Lay_ShowErrorAndExit ("Error when getting number of notices.");
/***** Get number of notifications by e-mail *****/
if (row[1])
{
if (sscanf (row[1],"%u",NumNotif) != 1)
Lay_ShowErrorAndExit ("Error when getting number of notifications of notices.");
}
else
*NumNotif = 0;
/***** Free structure that stores the query result *****/
DB_FreeMySQLResult (&mysql_res);
return NumNotices;
}
/*****************************************************************************/
/************************ Get number of deleted notices **********************/
/*****************************************************************************/
// Returns the number of deleted notices
// sent from this location (all the platform, current degree or current course)
unsigned Not_GetNumNoticesDeleted (Sco_Scope_t Scope,unsigned *NumNotif)
{
char Query[1024];
MYSQL_RES *mysql_res;
MYSQL_ROW row;
unsigned NumNotices;
/***** Get number of notices from database *****/
switch (Scope)
{
2015-02-01 20:17:24 +01:00
case Sco_SCOPE_SYS:
2014-12-01 23:55:08 +01:00
sprintf (Query,"SELECT COUNT(*),SUM(NumNotif)"
" FROM notices_deleted");
break;
2015-03-09 10:40:31 +01:00
case Sco_SCOPE_CTY:
sprintf (Query,"SELECT COUNT(*),SUM(notices_deleted.NumNotif)"
" FROM institutions,centres,degrees,courses,notices_deleted"
" WHERE institutions.CtyCod='%ld'"
" AND institutions.InsCod=centres.InsCod"
" AND centres.CtrCod=degrees.CtrCod"
" AND degrees.DegCod=courses.DegCod"
" AND courses.CrsCod=notices_deleted.CrsCod",
Gbl.CurrentCty.Cty.CtyCod);
break;
2015-02-01 20:17:24 +01:00
case Sco_SCOPE_INS:
2014-12-01 23:55:08 +01:00
sprintf (Query,"SELECT COUNT(*),SUM(notices_deleted.NumNotif)"
" FROM centres,degrees,courses,notices_deleted"
" WHERE centres.InsCod='%ld'"
" AND centres.CtrCod=degrees.CtrCod"
" AND degrees.DegCod=courses.DegCod"
" AND courses.CrsCod=notices_deleted.CrsCod",
Gbl.CurrentIns.Ins.InsCod);
break;
2015-02-01 20:17:24 +01:00
case Sco_SCOPE_CTR:
2014-12-01 23:55:08 +01:00
sprintf (Query,"SELECT COUNT(*),SUM(notices_deleted.NumNotif)"
" FROM degrees,courses,notices_deleted"
" WHERE degrees.CtrCod='%ld'"
" AND degrees.DegCod=courses.DegCod"
" AND courses.CrsCod=notices_deleted.CrsCod",
Gbl.CurrentCtr.Ctr.CtrCod);
break;
2015-02-01 20:17:24 +01:00
case Sco_SCOPE_DEG:
2014-12-01 23:55:08 +01:00
sprintf (Query,"SELECT COUNT(*),SUM(notices_deleted.NumNotif)"
" FROM courses,notices_deleted"
" WHERE courses.DegCod='%ld'"
" AND courses.CrsCod=notices_deleted.CrsCod",
Gbl.CurrentDeg.Deg.DegCod);
break;
2015-02-01 20:17:24 +01:00
case Sco_SCOPE_CRS:
2014-12-01 23:55:08 +01:00
sprintf (Query,"SELECT COUNT(*),SUM(NumNotif)"
" FROM notices_deleted"
" WHERE CrsCod='%ld'",
Gbl.CurrentCrs.Crs.CrsCod);
break;
default:
Lay_ShowErrorAndExit ("Wrong scope.");
break;
}
DB_QuerySELECT (Query,&mysql_res,"can not get number of deleted notices");
/***** Get number of notices *****/
row = mysql_fetch_row (mysql_res);
if (sscanf (row[0],"%u",&NumNotices) != 1)
Lay_ShowErrorAndExit ("Error when getting number of deleted notices.");
/***** Get number of notifications by e-mail *****/
if (row[1])
{
if (sscanf (row[1],"%u",NumNotif) != 1)
Lay_ShowErrorAndExit ("Error when getting number of notifications of deleted notices.");
}
else
*NumNotif = 0;
/***** Free structure that stores the query result *****/
DB_FreeMySQLResult (&mysql_res);
return NumNotices;
}
/*****************************************************************************/
/*************** Put parameter with the code of a notice *********************/
/*****************************************************************************/
static void Not_PutHiddenParamNotCod (long NotCod)
{
Par_PutHiddenParamLong ("NotCod",NotCod);
}
/*****************************************************************************/
/*************** Get parameter with the code of a notice *********************/
/*****************************************************************************/
static long Not_GetParamNotCod (void)
{
char LongStr[1+10+1]; // String that holds the notice code
long NotCod;
/* Get notice code */
Par_GetParToText ("NotCod",LongStr,1+10);
if (sscanf (LongStr,"%ld",&NotCod) != 1)
Lay_ShowErrorAndExit ("Wrong code of notice.");
return NotCod;
}