diff options
Diffstat (limited to 'src/print_ddate.c')
-rw-r--r-- | src/print_ddate.c | 212 |
1 files changed, 212 insertions, 0 deletions
diff --git a/src/print_ddate.c b/src/print_ddate.c new file mode 100644 index 0000000..be6776a --- /dev/null +++ b/src/print_ddate.c @@ -0,0 +1,212 @@ +// vim:ts=8:expandtab +#include <time.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> + +/* define fixed output-Strings */ +char *season_long[5] = { + "Chaos", + "Discord", + "Confusion", + "Bureaucracy", + "The Aftermath" +}; + +char *season_short[5] = { + "Chs", + "Dsc", + "Cfn", + "Bcy", + "Afm" +}; + +char *day_long[5] = { + "Sweetmorn", + "Boomtime", + "Pungenday", + "Prickle-Prickle", + "Setting Orange" +}; + +char *day_short[5] = { + "SM", + "BT", + "PD", + "PP", + "SO" +}; + +char *holidays[10] = { + "Mungday", + "Mojoday", + "Syaday", + "Zaraday", + "Maladay", + "Chaoflux", + "Discoflux", + "Confuflux", + "Bureflux", + "Afflux" +}; + +/* A helper-struct, taking the discordian date */ +struct disc_time { + int year; + int season; + int week_day; + int season_day; + int st_tibs_day; +}; + +/* Print the date *dt in format *format */ +int format_output(char *format, struct disc_time *dt) { + char *i; + char *tibs_end = 0; + + for (i = format; *i != '\0'; i++) { + if (*i != '%') { + putchar(*i); + continue; + } + switch (*(i+1)) { + /* Weekday in long and abbreviation */ + case 'A': + printf("%s", day_long[dt->week_day]); + break; + case 'a': + printf("%s", day_short[dt->week_day]); + break; + /* Season in long and abbreviation */ + case 'B': + printf("%s", season_long[dt->season]); + break; + case 'b': + printf("%s", season_short[dt->season]); + break; + /* Day of the season (ordinal and cardinal) */ + case 'd': + printf("%d", dt->season_day + 1); + break; + case 'e': + printf("%d", dt->season_day + 1); + switch (dt->season_day) { + case 0: + printf("st"); + break; + case 1: + printf("nd"); + break; + case 2: + printf("rd"); + break; + default: + printf("th"); + break; + } + break; + /* YOLD */ + case 'Y': + printf("%d", dt->year); + break; + /* Holidays */ + case 'H': + if (dt->season_day == 4) { + printf(holidays[dt->season]); + } + if (dt->season_day == 49) { + printf(holidays[dt->season + 5]); + } + break; + /* Stop parsing the format string, except on Holidays */ + case 'N': + if (dt->season_day != 4 && dt->season_day != 49) { + return 0; + } + break; + /* Newline- and Tabbing-characters */ + case 'n': + printf("\n"); + break; + case 't': + printf("\t"); + break; + /* The St. Tib's Day replacement */ + case '{': + tibs_end = strstr(i, "%}"); + if (tibs_end == NULL) { + i++; + break; + } + if (dt->st_tibs_day) { + /* We outpt "St. Tib's Day... */ + printf("St. Tib's Day"); + } else { + /* ...or parse the substring between %{ and %} ... */ + *i = '\0'; + if (!format_output(i + 2, dt)) return 0; + *i = '%'; + } + /* ...and continue with the rest */ + i = tibs_end + 2; + break; + case '}': + i++; + break; + default: + /* No escape-sequence, so we just skip */ + printf("%%%c",*(i+1)); + break; + } + i++; + } + return 1; +} + +/* Get the current date and convert it to discordian */ +struct disc_time *get_ddate() { + time_t current_time = time(NULL); + + if (current_time == (time_t) -1) { + return NULL; + } + + struct tm *current_tm = localtime(¤t_time); + + if (current_tm == NULL) { + return NULL; + } + + /* We have to know, whether we have to insert St. Tib's Day, so whether it's a leap + year in gregorian calendar */ + int is_leap_year = !(current_tm->tm_year % 4) && + (!(current_tm->tm_year % 400) || current_tm->tm_year % 100); + + struct disc_time *dt = malloc(sizeof(dt)); + if (is_leap_year && current_tm->tm_yday == 59) { + /* On St. Tibs Day we don't have to define a date */ + dt->st_tibs_day = 1; + } else { + dt->st_tibs_day = 0; + dt->season_day = current_tm->tm_yday % 73; + if (is_leap_year && current_tm->tm_yday > 59) { + dt->week_day = (current_tm->tm_yday - 1) % 5; + } else { + dt->week_day = current_tm->tm_yday % 5; + } + } + dt->year = current_tm->tm_year + 3066; + dt->season = current_tm->tm_yday / 73; + return dt; +} + +void print_ddate(const char *format) { + struct disc_time *dt = get_ddate(); + if (dt == NULL) { + return; + } + char *form = strdup(format); + format_output(form, dt); + free(dt); + free(form); +} |