mirror of
https://github.com/ZDoom/qzdoom.git
synced 2024-11-10 14:51:51 +00:00
Implemented Clock Class (#977)
* Simplified some stuff, made up-to-date * Dealt with DST problems * Made SystemTime.Format clearscope, as there is no reason for this function to be limited to the ui
This commit is contained in:
parent
85759e3bd0
commit
5803b78147
5 changed files with 227 additions and 27 deletions
|
@ -570,6 +570,133 @@ CCMD (special)
|
|||
|
||||
|
||||
|
||||
//==========================================================================
|
||||
//
|
||||
// CCMD setdate
|
||||
//
|
||||
// Set the time to a specific value
|
||||
//
|
||||
//==========================================================================
|
||||
|
||||
extern time_t epochoffset;
|
||||
CCMD(setdate)
|
||||
{
|
||||
if (argv.argc() != 3)
|
||||
{
|
||||
Printf("setdate HH:MM:SS DD-MM-YYYY: Set the current date\n");
|
||||
return;
|
||||
}
|
||||
|
||||
time_t today;
|
||||
time(&today);
|
||||
struct tm* timeinfo = localtime(&today);
|
||||
if (timeinfo != nullptr)
|
||||
{
|
||||
auto clock = FString(argv[1]).Split(":");
|
||||
auto date = FString(argv[2]).Split("-");
|
||||
if(clock.Size() != 3 || date.Size() != 3)
|
||||
{
|
||||
Printf("setdate HH:MM:SS DD-MM-YYYY: Set the current date\n");
|
||||
return;
|
||||
}
|
||||
|
||||
if(!clock[0].IsInt())
|
||||
{
|
||||
Printf("Invalid hour\n");
|
||||
return;
|
||||
}
|
||||
if (!clock[1].IsInt())
|
||||
{
|
||||
Printf("Invalid minutes\n");
|
||||
return;
|
||||
}
|
||||
if (!clock[2].IsInt())
|
||||
{
|
||||
Printf("Invalid seconds\n");
|
||||
return;
|
||||
}
|
||||
if (!date[0].IsInt())
|
||||
{
|
||||
Printf("Invalid day\n");
|
||||
return;
|
||||
}
|
||||
if (!date[1].IsInt())
|
||||
{
|
||||
Printf("Invalid month\n");
|
||||
return;
|
||||
}
|
||||
if (!date[2].IsInt())
|
||||
{
|
||||
Printf("Invalid year\n");
|
||||
return;
|
||||
}
|
||||
|
||||
//Set Date
|
||||
timeinfo->tm_hour = int( clock[0].ToLong() );
|
||||
timeinfo->tm_min = int( clock[1].ToLong() );
|
||||
timeinfo->tm_sec = int( clock[2].ToLong() );
|
||||
timeinfo->tm_mday = int( date[0].ToLong() );
|
||||
timeinfo->tm_mon = int( date[1].ToLong() - 1); // Month interally is 0 - 11
|
||||
timeinfo->tm_year = int( date[2].ToLong() - 1900 ); // Year interally is 00 - 138
|
||||
|
||||
time_t newTime = mktime(timeinfo);
|
||||
tm* t_old = localtime(&today);
|
||||
time_t oldTime = mktime(t_old);
|
||||
|
||||
if (newTime == -1 || oldTime == -1)
|
||||
{
|
||||
Printf("Unable to set the date\n");
|
||||
return;
|
||||
}
|
||||
|
||||
epochoffset = newTime - oldTime;
|
||||
|
||||
// This deals with some inconsistent display behaviour for DST
|
||||
// In this case, we want to emulate GCC's behaviour
|
||||
today += epochoffset;
|
||||
struct tm* t_new = localtime(&today);
|
||||
if (t_new != nullptr)
|
||||
{
|
||||
char timeString[1024];
|
||||
if (strftime(timeString, sizeof(timeString), "%H", t_new))
|
||||
{
|
||||
auto hour = FString(timeString).ToLong();
|
||||
if (hour - clock[0].ToLong() == -1 || hour - clock[0].ToLong() == 23)
|
||||
epochoffset += 3600;
|
||||
else if (hour - clock[0].ToLong() == 1 || hour - clock[0].ToLong() == -23)
|
||||
epochoffset -= 3600;
|
||||
}
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
else
|
||||
{
|
||||
Printf("Unable to set the date\n");
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
CCMD(getdate)
|
||||
{
|
||||
time_t now;
|
||||
time(&now);
|
||||
now += epochoffset;
|
||||
struct tm* timeinfo = localtime(&now);
|
||||
if (timeinfo != nullptr)
|
||||
{
|
||||
char timeString[1024];
|
||||
if (strftime(timeString, sizeof(timeString), "%H:%M:%S %d-%m-%Y%n", timeinfo))
|
||||
Printf(timeString);
|
||||
else
|
||||
Printf("Error Retrieving Current Date\n");
|
||||
}
|
||||
else
|
||||
{
|
||||
Printf("Error Retrieving Current Date\n");
|
||||
}
|
||||
}
|
||||
|
||||
//==========================================================================
|
||||
//
|
||||
// CCMD warp
|
||||
|
|
|
@ -164,6 +164,8 @@ bool playeringame[MAXPLAYERS];
|
|||
int consoleplayer; // player taking events
|
||||
int gametic;
|
||||
|
||||
time_t epochoffset = 0; // epoch start in seconds (0 = January 1st, 1970)
|
||||
|
||||
CVAR(Bool, demo_compress, true, CVAR_ARCHIVE|CVAR_GLOBALCONFIG);
|
||||
FString newdemoname;
|
||||
FString newdemomap;
|
||||
|
|
|
@ -2669,20 +2669,84 @@ DEFINE_ACTION_FUNCTION_NATIVE(FLevelLocals, setFrozen, setFrozen)
|
|||
//
|
||||
//=====================================================================================
|
||||
|
||||
static int GetRealTime()
|
||||
extern time_t epochoffset;
|
||||
|
||||
static int GetEpochTime()
|
||||
{
|
||||
time_t now;
|
||||
time(&now);
|
||||
struct tm* timeinfo = localtime(&now);
|
||||
return timeinfo ? timeinfo->tm_sec + timeinfo->tm_min * 60 + timeinfo->tm_hour * 3600 : 0;
|
||||
return now != (time_t)(-1) ? now + epochoffset : (time_t)(-1);
|
||||
}
|
||||
|
||||
DEFINE_ACTION_FUNCTION_NATIVE(_AltHUD, GetRealTime, GetRealTime)
|
||||
//Returns an empty string if the Strf tokens are valid, otherwise returns the problematic token
|
||||
static FString CheckStrfString(FString timeForm)
|
||||
{
|
||||
// Valid Characters after %
|
||||
const char validSingles[] = { 'a','A','b','B','c','C','d','D','e','F','G','g','h','H','I','j','k','l','m','M','n','p','P','r','R','s','S','t','T','u','U','V','w','W','x','X','y','Y','z','Z' };
|
||||
|
||||
timeForm.Substitute("%%", "%a"); //Prevent %% from causing tokenizing problems
|
||||
timeForm = "a" + timeForm; //Prevent %* at the beginning from causing a false error from tokenizing
|
||||
|
||||
auto tokens = timeForm.Split("%");
|
||||
for (auto t : tokens)
|
||||
{
|
||||
bool found = false;
|
||||
// % at end
|
||||
if (t.Len() == 0) return FString("%");
|
||||
|
||||
// Single Character
|
||||
for (int i = 0; i < sizeof(validSingles)/sizeof(validSingles[0]); i++)
|
||||
{
|
||||
if (t[0] == validSingles[i])
|
||||
{
|
||||
found = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (found) continue;
|
||||
return FString("%") + t[0];
|
||||
}
|
||||
return "";
|
||||
}
|
||||
|
||||
static void FormatTime(const FString& timeForm, int timeVal, FString* result)
|
||||
{
|
||||
FString error = CheckStrfString(timeForm);
|
||||
if (!error.IsEmpty())
|
||||
ThrowAbortException(X_FORMAT_ERROR, "'%s' is not a valid format specifier of SystemTime.Format()", error.GetChars());
|
||||
|
||||
time_t val = timeVal;
|
||||
struct tm* timeinfo = localtime(&val);
|
||||
if (timeinfo != nullptr)
|
||||
{
|
||||
char timeString[1024];
|
||||
if (strftime(timeString, sizeof(timeString), timeForm, timeinfo))
|
||||
*result = timeString;
|
||||
}
|
||||
}
|
||||
|
||||
DEFINE_ACTION_FUNCTION_NATIVE(_SystemTime, Now, GetEpochTime)
|
||||
{
|
||||
PARAM_PROLOGUE;
|
||||
ACTION_RETURN_INT(GetRealTime());
|
||||
ACTION_RETURN_INT(GetEpochTime());
|
||||
}
|
||||
|
||||
DEFINE_ACTION_FUNCTION_NATIVE(_SystemTime, Format, FormatTime)
|
||||
{
|
||||
PARAM_PROLOGUE;
|
||||
PARAM_STRING(timeForm);
|
||||
PARAM_INT(timeVal);
|
||||
FString result;
|
||||
FormatTime(timeForm, timeVal, &result);
|
||||
ACTION_RETURN_STRING(result);
|
||||
}
|
||||
|
||||
//=====================================================================================
|
||||
//
|
||||
//
|
||||
//
|
||||
//=====================================================================================
|
||||
|
||||
DEFINE_ACTION_FUNCTION_NATIVE(_AltHUD, GetLatency, Net_GetLatency)
|
||||
{
|
||||
PARAM_PROLOGUE;
|
||||
|
|
|
@ -422,6 +422,12 @@ struct GameInfoStruct native
|
|||
native double normsidemove[2];
|
||||
}
|
||||
|
||||
struct SystemTime
|
||||
{
|
||||
native static ui int Now(); // This returns the epoch time
|
||||
native static clearscope String Format(String timeForm, int timeVal); // This converts an epoch time to a local time, then uses the strftime syntax to format it
|
||||
}
|
||||
|
||||
class Object native
|
||||
{
|
||||
const TICRATE = 35;
|
||||
|
|
|
@ -770,13 +770,12 @@ class AltHud ui
|
|||
// for meaning of all display modes
|
||||
//
|
||||
//---------------------------------------------------------------------------
|
||||
private native static int GetRealTime();
|
||||
|
||||
virtual bool DrawTime(int y)
|
||||
{
|
||||
if (hud_showtime > 0 && hud_showtime <= 9)
|
||||
{
|
||||
int timeSeconds;
|
||||
String timeString;
|
||||
|
||||
if (hud_showtime < 8)
|
||||
{
|
||||
|
@ -787,11 +786,6 @@ class AltHud ui
|
|||
? Level.time
|
||||
: Level.totaltime);
|
||||
timeSeconds = Thinker.Tics2Seconds(timeTicks);
|
||||
}
|
||||
else
|
||||
{
|
||||
timeSeconds = GetRealTime();
|
||||
}
|
||||
|
||||
int hours = timeSeconds / 3600;
|
||||
int minutes = (timeSeconds % 3600) / 60;
|
||||
|
@ -800,8 +794,6 @@ class AltHud ui
|
|||
bool showMillis = 1 == hud_showtime;
|
||||
bool showSeconds = showMillis || (0 == hud_showtime % 2);
|
||||
|
||||
String timeString;
|
||||
|
||||
if (showMillis)
|
||||
{
|
||||
int millis = (Level.time % Thinker.TICRATE) * (1000 / Thinker.TICRATE);
|
||||
|
@ -815,6 +807,15 @@ class AltHud ui
|
|||
{
|
||||
timeString = String.Format("%02i:%02i", hours, minutes);
|
||||
}
|
||||
}
|
||||
else if (hud_showtime == 8)
|
||||
{
|
||||
timeString = SystemTime.Format("%H:%M:%S",SystemTime.Now());
|
||||
}
|
||||
else //if (hud_showtime == 9)
|
||||
{
|
||||
timeString = SystemTime.Format("%H:%M",SystemTime.Now());
|
||||
}
|
||||
|
||||
int characterCount = timeString.length();
|
||||
int width = SmallFont.GetCharWidth("0") * characterCount + 2; // small offset from screen's border
|
||||
|
|
Loading…
Reference in a new issue