diff options
author | Charles Kerr <charles.kerr@canonical.com> | 2013-07-26 13:26:03 -0500 |
---|---|---|
committer | Charles Kerr <charles.kerr@canonical.com> | 2013-07-26 13:26:03 -0500 |
commit | 18012d5e396e87d1c0c5534c35c5d0de82c2f7aa (patch) | |
tree | b683918b59569f2441d2922bc8e96759faaab407 | |
parent | 2c8cf9196bea17d20b9afd680f679eb285c5dc81 (diff) | |
download | ayatana-indicator-datetime-18012d5e396e87d1c0c5534c35c5d0de82c2f7aa.tar.gz ayatana-indicator-datetime-18012d5e396e87d1c0c5534c35c5d0de82c2f7aa.tar.bz2 ayatana-indicator-datetime-18012d5e396e87d1c0c5534c35c5d0de82c2f7aa.zip |
add 'terse' time formats for phone menu to match the spec.
-rw-r--r-- | src/datetime-prefs-locations.c | 2 | ||||
-rw-r--r-- | src/service.c | 33 | ||||
-rw-r--r-- | src/settings-shared.h | 29 | ||||
-rw-r--r-- | src/utils.c | 349 | ||||
-rw-r--r-- | src/utils.h | 32 |
5 files changed, 286 insertions, 159 deletions
diff --git a/src/datetime-prefs-locations.c b/src/datetime-prefs-locations.c index 4bbf053..bc044a2 100644 --- a/src/datetime-prefs-locations.c +++ b/src/datetime-prefs-locations.c @@ -428,7 +428,7 @@ update_times (GtkWidget * dlg) if (strzone && *strzone) { GTimeZone * tz = g_time_zone_new (strzone); GDateTime * now_tz = g_date_time_to_timezone (now, tz); - gchar * format = generate_format_string_at_time (now, now_tz); + gchar * format = generate_full_format_string_at_time (now, now_tz); gchar * time_str = g_date_time_format (now_tz, format); gchar * old_time_str; diff --git a/src/service.c b/src/service.c index c8781eb..b52556d 100644 --- a/src/service.c +++ b/src/service.c @@ -461,7 +461,7 @@ get_header_label_format_string (IndicatorDatetimeService * self) { gboolean show_day = g_settings_get_boolean (s, SETTINGS_SHOW_DAY_S); gboolean show_date = g_settings_get_boolean (s, SETTINGS_SHOW_DATE_S); - fmt = generate_format_string_full (show_day, show_date); + fmt = generate_full_format_string (show_day, show_date); } return fmt; @@ -517,7 +517,7 @@ create_phone_header_state (IndicatorDatetimeService * self) /* label */ now = indicator_datetime_service_get_localtime (self); - fmt = _("%I:%M %p"); + fmt = get_terse_time_format_string (now); label = g_date_time_format (now, fmt); g_variant_builder_add (&b, "{sv}", "label", g_variant_new_string (label)); @@ -778,35 +778,36 @@ service_has_alarms (IndicatorDatetimeService * self) } static char * -get_appointment_time_format (struct IndicatorDatetimeAppt * appt, GDateTime * now) +get_appointment_time_format (struct IndicatorDatetimeAppt * appt, + GDateTime * now, + gboolean terse) { char * fmt; gboolean full_day = g_date_time_difference (appt->end, appt->begin) == G_TIME_SPAN_DAY; if (appt->is_daily) { - char * time_string = generate_format_string_full (FALSE, FALSE); - fmt = join_date_and_time_format_strings (_("Daily"), time_string); - g_free (time_string); + const char * time_fmt = terse ? get_terse_time_format_string (appt->begin) + : get_full_time_format_string (); + fmt = join_date_and_time_format_strings (_("Daily"), time_fmt); } else if (full_day) { - /* TRANSLATORS: This is a strftime string for the day for full day events - in the menu. It should most likely be either '%A' for a full text day - (Wednesday) or '%a' for a shortened one (Wed). You should only need to - change for '%a' in the case of langauges with very long day names. */ + /* TRANSLATORS: a strftime(3) format showing full day events. + * "%A" means a full text day (Wednesday), "%a" means abbreviated (Wed). */ fmt = g_strdup (_("%A")); } else { - fmt = generate_format_string_at_time (now, appt->begin); + fmt = terse ? generate_terse_format_string_at_time (now, appt->begin) + : generate_full_format_string_at_time (now, appt->begin); } return fmt; } static void -add_appointments (IndicatorDatetimeService * self, GMenu * menu) +add_appointments (IndicatorDatetimeService * self, GMenu * menu, gboolean terse) { GDateTime * now = indicator_datetime_service_get_localtime (self); GSList * appts; @@ -817,7 +818,7 @@ add_appointments (IndicatorDatetimeService * self, GMenu * menu) for (l=appts; l!=NULL; l=l->next) { struct IndicatorDatetimeAppt * appt = l->data; - char * fmt = get_appointment_time_format (appt, now); + char * fmt = get_appointment_time_format (appt, now, terse); const gint64 unix_time = g_date_time_to_unix (appt->begin); GMenuItem * menu_item; @@ -858,7 +859,7 @@ create_phone_appointments_section (IndicatorDatetimeService * self) g_menu_append_item (menu, menu_item); g_object_unref (menu_item); - add_appointments (self, menu); + add_appointments (self, menu, TRUE); return G_MENU_MODEL (menu); } @@ -872,7 +873,7 @@ create_desktop_appointments_section (IndicatorDatetimeService * self) { GMenuItem * menu_item; - add_appointments (self, menu); + add_appointments (self, menu, FALSE); /* add the 'Add Event…' menuitem */ menu_item = g_menu_item_new (_("Add Event…"), NULL); @@ -1100,7 +1101,7 @@ create_locations_section (IndicatorDatetimeService * self) detailed_action = g_strdup_printf ("indicator.set-location::%s %s", loc->zone, loc->name); - fmt = generate_format_string_at_time (now, loc->local_time); + fmt = generate_full_format_string_at_time (now, loc->local_time); menu_item = g_menu_item_new (label, detailed_action); g_menu_item_set_attribute (menu_item, "x-canonical-type", diff --git a/src/settings-shared.h b/src/settings-shared.h index 8c14f1e..27ce34c 100644 --- a/src/settings-shared.h +++ b/src/settings-shared.h @@ -37,33 +37,4 @@ with this program. If not, see <http://www.gnu.org/licenses/>. #define SETTINGS_LOCATIONS_S "locations" #define SETTINGS_TIMEZONE_NAME_S "timezone-name" -enum { - SETTINGS_TIME_LOCALE = 0, - SETTINGS_TIME_12_HOUR = 1, - SETTINGS_TIME_24_HOUR = 2, - SETTINGS_TIME_CUSTOM = 3 -}; - -/* TRANSLATORS: A format string for the strftime function for - a clock showing 12-hour time without seconds. */ -#define DEFAULT_TIME_12_FORMAT N_("%l:%M %p") - -/* TRANSLATORS: A format string for the strftime function for - a clock showing 24-hour time without seconds. */ -#define DEFAULT_TIME_24_FORMAT N_("%H:%M") - -#define DEFAULT_TIME_FORMAT DEFAULT_TIME_12_FORMAT -#define DEFAULT_TIME_FORMAT_WITH_DAY DEFAULT_TIME_12_FORMAT_WITH_DAY - -/* TRANSLATORS: A format string for the strftime function for - a clock showing the day of the week and the time in 12-hour format without - seconds. */ -#define DEFAULT_TIME_12_FORMAT_WITH_DAY N_("%a %l:%M %p") - -/* TRANSLATORS: A format string for the strftime function for - a clock showing the day of the week and the time in 24-hour format without - seconds. Information is available in this Launchpad answer: - https://answers.launchpad.net/ubuntu/+source/indicator-datetime/+question/149752 */ -#define DEFAULT_TIME_24_FORMAT_WITH_DAY N_("%a %H:%M") - #endif diff --git a/src/utils.c b/src/utils.c index a408f68..1297981 100644 --- a/src/utils.c +++ b/src/utils.c @@ -118,7 +118,7 @@ get_current_zone_name (const gchar * location) } /* Translate msg according to the locale specified by LC_TIME */ -static char * +static const char * T_(const char *msg) { /* General strategy here is to make sure LANGUAGE is empty (since that @@ -158,127 +158,262 @@ gchar * join_date_and_time_format_strings (const char * date_string, const char * time_string) { - /* TRANSLATORS: This is a format string passed to strftime to combine the - * date and the time. The value of "%s\xE2\x80\x82%s" will result in a - * string like this in US English 12-hour time: 'Fri Jul 16 11:50 AM'. - * The space in between date and time is a Unicode en space - * (E28082 in UTF-8 hex). */ - return g_strdup_printf (T_("%s\xE2\x80\x82%s"), date_string, time_string); + gchar * str; + + if (date_string && time_string) + { + /* TRANSLATORS: This is a format string passed to strftime to combine the + * date and the time. The value of "%s\xE2\x80\x82%s" will result in a + * string like this in US English 12-hour time: 'Fri Jul 16 11:50 AM'. + * The space in between date and time is a Unicode en space + * (E28082 in UTF-8 hex). */ + str = g_strdup_printf (T_("%s\xE2\x80\x82%s"), date_string, time_string); + } + else if (date_string) + { + str = g_strdup_printf (T_("%s"), date_string); + } + else /* time_string */ + { + str = g_strdup_printf (T_("%s"), time_string); + } + + return str; +} + +/*** +**** +***/ + +typedef enum +{ + DATE_PROXIMITY_TODAY, + DATE_PROXIMITY_TOMORROW, + DATE_PROXIMITY_WEEK, + DATE_PROXIMITY_FAR +} +date_proximity_t; + +static date_proximity_t +get_date_proximity (GDateTime * now, GDateTime * time) +{ + date_proximity_t prox = DATE_PROXIMITY_FAR; + gint now_year, now_month, now_day; + gint time_year, time_month, time_day; + + /* does it happen today? */ + g_date_time_get_ymd (now, &now_year, &now_month, &now_day); + g_date_time_get_ymd (time, &time_year, &time_month, &time_day); + if ((now_year == time_year) && (now_month == time_month) && (now_day == time_day)) + prox = DATE_PROXIMITY_TODAY; + + /* does it happen tomorrow? */ + if (prox == DATE_PROXIMITY_FAR) + { + GDateTime * tomorrow; + gint tom_year, tom_month, tom_day; + + tomorrow = g_date_time_add_days (now, 1); + g_date_time_get_ymd (tomorrow, &tom_year, &tom_month, &tom_day); + if ((tom_year == time_year) && (tom_month == time_month) && (tom_day == time_day)) + prox = DATE_PROXIMITY_TOMORROW; + + g_date_time_unref (tomorrow); + } + + /* does it happen this week? */ + if (prox == DATE_PROXIMITY_FAR) + { + GDateTime * week; + GDateTime * week_bound; + + week = g_date_time_add_days (now, 6); + week_bound = g_date_time_new_local (g_date_time_get_year(week), + g_date_time_get_month (week), + g_date_time_get_day_of_month(week), + 23, 59, 59.9); + + if (g_date_time_compare (time, week_bound) <= 0) + prox = DATE_PROXIMITY_WEEK; + + g_date_time_unref (week_bound); + g_date_time_unref (week); + } + + return prox; } -/* Tries to figure out what our format string should be. Lots - of translator comments in here. */ +/* + * "Terse" time & date format strings + * + * Used on the phone menu where space is at a premium, these strings + * express the time and date in as brief a form as possible. + * + * Examples from spec: + * 1. "Daily 6:30 AM" + * 2. "5:07 PM" (note date is omitted; today's date is implicit) + * 3. "Daily 12 PM" (note minutes are omitted for on-the-hour times) + * 4. "Tomorrow 7 AM" (note "Tomorrow" is used instead of a day of week) + */ + +static const gchar * +get_terse_date_format_string (date_proximity_t proximity) +{ + const gchar * fmt; + + switch (proximity) + { + case DATE_PROXIMITY_TODAY: fmt = NULL; break; + case DATE_PROXIMITY_TOMORROW: fmt = T_("Tomorrow"); break; + case DATE_PROXIMITY_WEEK: fmt = T_("%a"); break; + default: fmt = T_("%d %b"); break; + } + + return fmt; +} + +const gchar * +get_terse_time_format_string (GDateTime * time) +{ + const gchar * fmt; + + if (g_date_time_get_minute (time) != 0) + fmt = T_("%I:%M %p"); + else + fmt = T_("%I %p"); + + return fmt; +} + gchar * -generate_format_string_full (gboolean show_day, gboolean show_date) +generate_terse_format_string_at_time (GDateTime * now, GDateTime * time) { - gboolean twelvehour = TRUE; + const date_proximity_t prox = get_date_proximity (now, time); + const gchar * date_fmt = get_terse_date_format_string (prox); + const gchar * time_fmt = get_terse_time_format_string (time); + return join_date_and_time_format_strings (date_fmt, time_fmt); +} - GSettings * settings = g_settings_new (SETTINGS_INTERFACE); - gint time_mode = g_settings_get_enum (settings, SETTINGS_TIME_FORMAT_S); - gboolean show_seconds = g_settings_get_boolean (settings, SETTINGS_SHOW_SECONDS_S); - g_object_unref (settings); +/*** +**** FULL +***/ - if (time_mode == SETTINGS_TIME_LOCALE) { - twelvehour = is_locale_12h(); - } else if (time_mode == SETTINGS_TIME_24_HOUR) { - twelvehour = FALSE; - } +static const gchar * +get_full_date_format_string (gboolean show_day, gboolean show_date) +{ + const gchar * fmt; + + if (show_date && show_day) + /* TRANSLATORS: a strftime(3) format showing the date and weekday */ + fmt = T_("%a %b %e"); + else if (show_date) + /* TRANSLATORS: a strftime(3) format showing the date */ + fmt = T_("%b %e"); + else if (show_day) + /* TRANSLATORS: a strftime(3) format showing the weekday */ + fmt = T_("%a"); + else + fmt = NULL; + + return fmt; +} - const gchar * time_string = NULL; - if (twelvehour) { - if (show_seconds) { - /* TRANSLATORS: A format string for the strftime function for - a clock showing 12-hour time with seconds. */ - time_string = T_("%l:%M:%S %p"); - } else { - time_string = T_(DEFAULT_TIME_12_FORMAT); - } - } else { - if (show_seconds) { - /* TRANSLATORS: A format string for the strftime function for - a clock showing 24-hour time with seconds. */ - time_string = T_("%H:%M:%S"); - } else { - time_string = T_(DEFAULT_TIME_24_FORMAT); - } - } - - /* Checkpoint, let's not fail */ - g_return_val_if_fail(time_string != NULL, g_strdup(DEFAULT_TIME_FORMAT)); - - /* If there's no date or day let's just leave now and - not worry about the rest of this code */ - if (!show_date && !show_day) { - return g_strdup(time_string); - } - const gchar * date_string = NULL; - if (show_date && show_day) { - /* TRANSLATORS: This is a format string passed to strftime to represent - the day of the week, the month and the day of the month. */ - date_string = T_("%a %b %e"); - } else if (show_date) { - /* TRANSLATORS: This is a format string passed to strftime to represent - the month and the day of the month. */ - date_string = T_("%b %e"); - } else if (show_day) { - /* TRANSLATORS: This is a format string passed to strftime to represent - the day of the week. */ - date_string = T_("%a"); - } +/* + * "Full" time & date format strings + * + * These are used on the desktop menu & header and honors the + * GSettings entries for 12/24hr mode and whether or not to show seconds. + * + */ - /* Check point, we should have a date string */ - g_return_val_if_fail(date_string != NULL, g_strdup(time_string)); +enum +{ + SETTINGS_TIME_LOCALE = 0, + SETTINGS_TIME_12_HOUR = 1, + SETTINGS_TIME_24_HOUR = 2, + SETTINGS_TIME_CUSTOM = 3 +}; + +const gchar * +get_full_time_format_string (void) +{ + GSettings * settings; + gboolean twelvehour; + gboolean show_seconds; + const gchar * fmt; - return join_date_and_time_format_strings (date_string, time_string); + settings = g_settings_new (SETTINGS_INTERFACE); + + show_seconds = g_settings_get_boolean (settings, SETTINGS_SHOW_SECONDS_S); + + switch (g_settings_get_enum (settings, SETTINGS_TIME_FORMAT_S)) + { + case SETTINGS_TIME_LOCALE: + twelvehour = is_locale_12h(); + break; + + case SETTINGS_TIME_24_HOUR: + twelvehour = FALSE; + break; + + default: + twelvehour = TRUE; + break; + } + + g_object_unref (settings); + + if (twelvehour && show_seconds) + /* TRANSLATORS: a strftime(3) format for 12hr time w/seconds */ + fmt = T_("%l:%M:%S %p"); + else if (twelvehour) + /* TRANSLATORS: a strftime(3) format for 12hr time */ + fmt = T_("%l:%M %p"); + else if (show_seconds) + /* TRANSLATORS: a strftime(3) format for 24hr time w/seconds */ + fmt = T_("%H:%M:%S"); + else + /* TRANSLATORS: a strftime(3) format for 24hr time */ + fmt = T_("%H:%M"); + + return fmt; } gchar * -generate_format_string_at_time (GDateTime * now, GDateTime * time) +generate_full_format_string (gboolean show_day, gboolean show_date) { - /* This is a bit less free-form than for the main "now" time label. */ - /* If it is today, just the time should be shown (e.g. “3:55 PM”) - If it is a different day this week, the day and time should be shown (e.g. “Wed 3:55 PM”) - If it is after this week, the day, date, and time should be shown (e.g. “Wed 21 Apr 3:55 PM”). - In addition, when presenting the times of upcoming events, the time should be followed by the timezone if it is different from the one the computer is currently set to. For example, “Wed 3:55 PM UTC−5”. */ - gboolean show_day = FALSE; - gboolean show_date = FALSE; - - /* First, are we same day? */ - gint time_year, time_month, time_day; - gint now_year, now_month, now_day; - g_date_time_get_ymd(time, &time_year, &time_month, &time_day); - g_date_time_get_ymd(now, &now_year, &now_month, &now_day); - - if (time_year != now_year || - time_month != now_month || - time_day != now_day) { - /* OK, different days so we must at least show the day. */ - show_day = TRUE; - - /* Is it this week? */ - /* Here, we define "is this week" as yesterday, today, or the next five days */ - GDateTime * past = g_date_time_add_days(now, -1); - GDateTime * future = g_date_time_add_days(now, 5); - GDateTime * past_bound = g_date_time_new_local(g_date_time_get_year(past), - g_date_time_get_month(past), - g_date_time_get_day_of_month(past), - 0, 0, 0.0); - GDateTime * future_bound = g_date_time_new_local(g_date_time_get_year(future), - g_date_time_get_month(future), - g_date_time_get_day_of_month(future), - 23, 59, 59.9); - if (g_date_time_compare(time, past_bound) < 0 || - g_date_time_compare(time, future_bound) > 0) { - show_date = TRUE; - } - g_date_time_unref(past); - g_date_time_unref(future); - g_date_time_unref(past_bound); - g_date_time_unref(future_bound); - } + const gchar * date_fmt = get_full_date_format_string (show_day, show_date); + const gchar * time_fmt = get_full_time_format_string (); + return join_date_and_time_format_strings (date_fmt, time_fmt); +} + +gchar * +generate_full_format_string_at_time (GDateTime * now, GDateTime * time) +{ + gboolean show_day; + gboolean show_date; + + switch (get_date_proximity (now, time)) + { + case DATE_PROXIMITY_TODAY: + show_day = FALSE; + show_date = FALSE; + break; + + case DATE_PROXIMITY_TOMORROW: + case DATE_PROXIMITY_WEEK: + show_day = FALSE; + show_date = TRUE; + break; + + default: + show_day = TRUE; + show_date = TRUE; + break; + } - return generate_format_string_full(show_day, show_date); + return generate_full_format_string (show_day, show_date); } diff --git a/src/utils.h b/src/utils.h index 73bd83a..3b0d0a2 100644 --- a/src/utils.h +++ b/src/utils.h @@ -27,13 +27,33 @@ with this program. If not, see <http://www.gnu.org/licenses/>. G_BEGIN_DECLS -gboolean is_locale_12h (void); -void split_settings_location (const gchar * location, gchar ** zone, gchar ** name); -gchar * get_current_zone_name (const gchar * location); -gchar * generate_format_string_full (gboolean show_day, gboolean show_date); -gchar * generate_format_string_at_time (GDateTime * now, GDateTime * time); -gchar * join_date_and_time_format_strings (const char * date, const char * time); +gboolean is_locale_12h (void); +void split_settings_location (const char * location, + char ** zone, + char ** name); + +gchar * get_current_zone_name (const char * location); + +gchar* join_date_and_time_format_strings (const char * date_fmt, + const char * time_fmt); +/*** +**** +***/ + +const gchar * get_terse_time_format_string (GDateTime * time); + +const gchar * get_full_time_format_string (void); + +gchar * generate_terse_format_string_at_time (GDateTime * now, + GDateTime * time); + +gchar * generate_full_format_string (gboolean show_day, + gboolean show_date); + +gchar * generate_full_format_string_at_time (GDateTime * now, + GDateTime * time); + G_END_DECLS #endif |