1 // Copyright 2013 The Chromium Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style license that can be 3 // found in the LICENSE file. 4 5 #include "chrome/browser/chromeos/system/timezone_util.h" 6 7 #include <string> 8 9 #include "base/i18n/rtl.h" 10 #include "base/lazy_instance.h" 11 #include "base/strings/string_util.h" 12 #include "base/strings/stringprintf.h" 13 #include "base/strings/utf_string_conversions.h" 14 #include "base/synchronization/lock.h" 15 #include "base/values.h" 16 #include "chromeos/settings/timezone_settings.h" 17 #include "grit/generated_resources.h" 18 #include "third_party/icu/source/common/unicode/ures.h" 19 #include "third_party/icu/source/common/unicode/utypes.h" 20 #include "third_party/icu/source/i18n/unicode/calendar.h" 21 #include "third_party/icu/source/i18n/unicode/timezone.h" 22 #include "ui/base/l10n/l10n_util.h" 23 24 namespace { 25 26 struct UResClose { 27 inline void operator() (UResourceBundle* b) const { 28 ures_close(b); 29 } 30 }; 31 32 static base::LazyInstance<base::Lock>::Leaky 33 g_timezone_bundle_lock = LAZY_INSTANCE_INITIALIZER; 34 35 // Returns an exemplary city in the given timezone. 36 base::string16 GetExemplarCity(const icu::TimeZone& zone) { 37 // TODO(jungshik): After upgrading to ICU 4.6, use U_ICUDATA_ZONE 38 static const char* zone_bundle_name = NULL; 39 40 // These will be leaked at the end. 41 static UResourceBundle *zone_bundle = NULL; 42 static UResourceBundle *zone_strings = NULL; 43 44 UErrorCode status = U_ZERO_ERROR; 45 { 46 base::AutoLock lock(g_timezone_bundle_lock.Get()); 47 if (zone_bundle == NULL) 48 zone_bundle = ures_open(zone_bundle_name, uloc_getDefault(), &status); 49 50 if (zone_strings == NULL) 51 zone_strings = ures_getByKey(zone_bundle, "zone_strings", NULL, &status); 52 } 53 54 icu::UnicodeString zone_id; 55 zone.getID(zone_id); 56 std::string zone_id_str; 57 zone_id.toUTF8String(zone_id_str); 58 59 // Resource keys for timezones use ':' in place of '/'. 60 ReplaceSubstringsAfterOffset(&zone_id_str, 0, "/", ":"); 61 scoped_ptr_malloc<UResourceBundle, UResClose> zone_item( 62 ures_getByKey(zone_strings, zone_id_str.c_str(), NULL, &status)); 63 icu::UnicodeString city; 64 if (!U_FAILURE(status)) { 65 city = icu::ures_getUnicodeStringByKey(zone_item.get(), "ec", &status); 66 if (U_SUCCESS(status)) 67 return base::string16(city.getBuffer(), city.length()); 68 } 69 70 // Fallback case in case of failure. 71 ReplaceSubstringsAfterOffset(&zone_id_str, 0, ":", "/"); 72 // Take the last component of a timezone id (e.g. 'Baz' in 'Foo/Bar/Baz'). 73 // Depending on timezones, keeping all but the 1st component 74 // (e.g. Bar/Baz) may be better, but our current list does not have 75 // any timezone for which that's the case. 76 std::string::size_type slash_pos = zone_id_str.rfind('/'); 77 if (slash_pos != std::string::npos && slash_pos < zone_id_str.size()) 78 zone_id_str.erase(0, slash_pos + 1); 79 // zone id has '_' in place of ' '. 80 ReplaceSubstringsAfterOffset(&zone_id_str, 0, "_", " "); 81 return ASCIIToUTF16(zone_id_str); 82 } 83 84 // Gets the given timezone's name for visualization. 85 base::string16 GetTimezoneName(const icu::TimeZone& timezone) { 86 // Instead of using the raw_offset, use the offset in effect now. 87 // For instance, US Pacific Time, the offset shown will be -7 in summer 88 // while it'll be -8 in winter. 89 int raw_offset, dst_offset; 90 UDate now = icu::Calendar::getNow(); 91 UErrorCode status = U_ZERO_ERROR; 92 timezone.getOffset(now, false, raw_offset, dst_offset, status); 93 DCHECK(U_SUCCESS(status)); 94 int offset = raw_offset + dst_offset; 95 // |offset| is in msec. 96 int minute_offset = std::abs(offset) / 60000; 97 int hour_offset = minute_offset / 60; 98 int min_remainder = minute_offset % 60; 99 // Some timezones have a non-integral hour offset. So, we need to use hh:mm 100 // form. 101 std::string offset_str = base::StringPrintf(offset >= 0 ? 102 "UTC+%d:%02d" : "UTC-%d:%02d", hour_offset, min_remainder); 103 104 // TODO(jungshik): When coming up with a better list of timezones, we also 105 // have to come up with better 'display' names. One possibility is to list 106 // multiple cities (e.g. "Los Angeles, Vancouver .." in the order of 107 // the population of a country the city belongs to.). 108 // We can also think of using LONG_GENERIC or LOCATION once we upgrade 109 // to ICU 4.6. 110 // In the meantime, we use "LONG" name with "Exemplar City" to distinguish 111 // multiple timezones with the same "LONG" name but with different 112 // rules (e.g. US Mountain Time in Denver vs Phoenix). 113 icu::UnicodeString name; 114 timezone.getDisplayName(dst_offset != 0, icu::TimeZone::LONG, name); 115 base::string16 result(l10n_util::GetStringFUTF16( 116 IDS_OPTIONS_SETTINGS_TIMEZONE_DISPLAY_TEMPLATE, ASCIIToUTF16(offset_str), 117 base::string16(name.getBuffer(), name.length()), GetExemplarCity(timezone))); 118 base::i18n::AdjustStringForLocaleDirection(&result); 119 return result; 120 } 121 122 } // namespace 123 124 namespace chromeos { 125 namespace system { 126 127 // Creates a list of pairs of each timezone's ID and name. 128 scoped_ptr<base::ListValue> GetTimezoneList() { 129 const std::vector<icu::TimeZone*> &timezones = 130 chromeos::system::TimezoneSettings::GetInstance()->GetTimezoneList(); 131 scoped_ptr<base::ListValue> timezoneList(new base::ListValue()); 132 for (std::vector<icu::TimeZone*>::const_iterator iter = timezones.begin(); 133 iter != timezones.end(); ++iter) { 134 const icu::TimeZone* timezone = *iter; 135 base::ListValue* option = new base::ListValue(); 136 option->Append(new base::StringValue( 137 chromeos::system::TimezoneSettings::GetTimezoneID(*timezone))); 138 option->Append(new base::StringValue(GetTimezoneName(*timezone))); 139 timezoneList->Append(option); 140 } 141 return timezoneList.Pass(); 142 } 143 144 } // namespace system 145 } // namespace chromeos 146