| Index: chrome/browser/ui/webui/options2/chromeos/system_settings_provider.cc
|
| diff --git a/chrome/browser/ui/webui/options2/chromeos/system_settings_provider.cc b/chrome/browser/ui/webui/options2/chromeos/system_settings_provider.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..c951164fb4ff6e47480ddfcc5b076e5495a71beb
|
| --- /dev/null
|
| +++ b/chrome/browser/ui/webui/options2/chromeos/system_settings_provider.cc
|
| @@ -0,0 +1,333 @@
|
| +// Copyright (c) 2011 The Chromium Authors. All rights reserved.
|
| +// Use of this source code is governed by a BSD-style license that can be
|
| +// found in the LICENSE file.
|
| +
|
| +#include "chrome/browser/ui/webui/options2/chromeos/system_settings_provider.h"
|
| +
|
| +#include <string>
|
| +
|
| +#include "base/i18n/rtl.h"
|
| +#include "base/lazy_instance.h"
|
| +#include "base/memory/scoped_ptr.h"
|
| +#include "base/stl_util.h"
|
| +#include "base/string_util.h"
|
| +#include "base/stringprintf.h"
|
| +#include "base/synchronization/lock.h"
|
| +#include "base/time.h"
|
| +#include "base/utf_string_conversions.h"
|
| +#include "base/values.h"
|
| +#include "chrome/browser/chromeos/cros/cros_library.h"
|
| +#include "chrome/browser/chromeos/cros_settings.h"
|
| +#include "chrome/browser/chromeos/cros_settings_names.h"
|
| +#include "chrome/browser/chromeos/login/user_manager.h"
|
| +#include "grit/generated_resources.h"
|
| +#include "ui/base/l10n/l10n_util.h"
|
| +#include "unicode/calendar.h"
|
| +#include "unicode/timezone.h"
|
| +#include "unicode/ures.h"
|
| +
|
| +namespace {
|
| +
|
| +// TODO(jungshik): Using Enumerate method in ICU gives 600+ timezones.
|
| +// Even after filtering out duplicate entries with a strict identity check,
|
| +// we still have 400+ zones. Relaxing the criteria for the timezone
|
| +// identity is likely to cut down the number to < 100. Until we
|
| +// come up with a better list, we hard-code the following list as used by
|
| +// Android.
|
| +static const char* kTimeZones[] = {
|
| + "Pacific/Majuro",
|
| + "Pacific/Midway",
|
| + "Pacific/Honolulu",
|
| + "America/Anchorage",
|
| + "America/Los_Angeles",
|
| + "America/Tijuana",
|
| + "America/Denver",
|
| + "America/Phoenix",
|
| + "America/Chihuahua",
|
| + "America/Chicago",
|
| + "America/Mexico_City",
|
| + "America/Costa_Rica",
|
| + "America/Regina",
|
| + "America/New_York",
|
| + "America/Bogota",
|
| + "America/Caracas",
|
| + "America/Barbados",
|
| + "America/Manaus",
|
| + "America/Santiago",
|
| + "America/St_Johns",
|
| + "America/Sao_Paulo",
|
| + "America/Araguaina",
|
| + "America/Argentina/Buenos_Aires",
|
| + "America/Godthab",
|
| + "America/Montevideo",
|
| + "Atlantic/South_Georgia",
|
| + "Atlantic/Azores",
|
| + "Atlantic/Cape_Verde",
|
| + "Africa/Casablanca",
|
| + "Europe/London",
|
| + "Europe/Amsterdam",
|
| + "Europe/Belgrade",
|
| + "Europe/Brussels",
|
| + "Europe/Sarajevo",
|
| + "Africa/Windhoek",
|
| + "Africa/Brazzaville",
|
| + "Asia/Amman",
|
| + "Europe/Athens",
|
| + "Asia/Beirut",
|
| + "Africa/Cairo",
|
| + "Europe/Helsinki",
|
| + "Asia/Jerusalem",
|
| + "Europe/Minsk",
|
| + "Africa/Harare",
|
| + "Asia/Baghdad",
|
| + "Europe/Moscow",
|
| + "Asia/Kuwait",
|
| + "Africa/Nairobi",
|
| + "Asia/Tehran",
|
| + "Asia/Baku",
|
| + "Asia/Tbilisi",
|
| + "Asia/Yerevan",
|
| + "Asia/Dubai",
|
| + "Asia/Kabul",
|
| + "Asia/Karachi",
|
| + "Asia/Oral",
|
| + "Asia/Yekaterinburg",
|
| + "Asia/Calcutta",
|
| + "Asia/Colombo",
|
| + "Asia/Katmandu",
|
| + "Asia/Almaty",
|
| + "Asia/Rangoon",
|
| + "Asia/Krasnoyarsk",
|
| + "Asia/Bangkok",
|
| + "Asia/Shanghai",
|
| + "Asia/Hong_Kong",
|
| + "Asia/Irkutsk",
|
| + "Asia/Kuala_Lumpur",
|
| + "Australia/Perth",
|
| + "Asia/Taipei",
|
| + "Asia/Seoul",
|
| + "Asia/Tokyo",
|
| + "Asia/Yakutsk",
|
| + "Australia/Adelaide",
|
| + "Australia/Darwin",
|
| + "Australia/Brisbane",
|
| + "Australia/Hobart",
|
| + "Australia/Sydney",
|
| + "Asia/Vladivostok",
|
| + "Pacific/Guam",
|
| + "Asia/Magadan",
|
| + "Pacific/Auckland",
|
| + "Pacific/Fiji",
|
| + "Pacific/Tongatapu",
|
| +};
|
| +
|
| +static base::LazyInstance<base::Lock,
|
| + base::LeakyLazyInstanceTraits<base::Lock> >
|
| + g_timezone_bundle_lock = LAZY_INSTANCE_INITIALIZER;
|
| +
|
| +struct UResClose {
|
| + inline void operator() (UResourceBundle* b) const {
|
| + ures_close(b);
|
| + }
|
| +};
|
| +
|
| +string16 GetExemplarCity(const icu::TimeZone& zone) {
|
| + // TODO(jungshik): After upgrading to ICU 4.6, use U_ICUDATA_ZONE
|
| + static const char* zone_bundle_name = NULL;
|
| +
|
| + // These will be leaked at the end.
|
| + static UResourceBundle *zone_bundle = NULL;
|
| + static UResourceBundle *zone_strings = NULL;
|
| +
|
| + UErrorCode status = U_ZERO_ERROR;
|
| + {
|
| + base::AutoLock lock(g_timezone_bundle_lock.Get());
|
| + if (zone_bundle == NULL)
|
| + zone_bundle = ures_open(zone_bundle_name, uloc_getDefault(), &status);
|
| +
|
| + if (zone_strings == NULL)
|
| + zone_strings = ures_getByKey(zone_bundle, "zone_strings", NULL, &status);
|
| + }
|
| +
|
| + icu::UnicodeString zone_id;
|
| + zone.getID(zone_id);
|
| + std::string zone_id_str;
|
| + zone_id.toUTF8String(zone_id_str);
|
| +
|
| + // resource keys for timezones use ':' in place of '/'.
|
| + ReplaceSubstringsAfterOffset(&zone_id_str, 0, "/", ":");
|
| + scoped_ptr_malloc<UResourceBundle, UResClose> zone_item(
|
| + ures_getByKey(zone_strings, zone_id_str.c_str(), NULL, &status));
|
| + icu::UnicodeString city;
|
| + if (!U_FAILURE(status)) {
|
| + city = icu::ures_getUnicodeStringByKey(zone_item.get(), "ec", &status);
|
| + if (U_SUCCESS(status))
|
| + return string16(city.getBuffer(), city.length());
|
| + }
|
| +
|
| + // Fallback case in case of failure.
|
| + ReplaceSubstringsAfterOffset(&zone_id_str, 0, ":", "/");
|
| + // Take the last component of a timezone id (e.g. 'Baz' in 'Foo/Bar/Baz').
|
| + // Depending on timezones, keeping all but the 1st component
|
| + // (e.g. Bar/Baz) may be better, but our current list does not have
|
| + // any timezone for which that's the case.
|
| + std::string::size_type slash_pos = zone_id_str.rfind('/');
|
| + if (slash_pos != std::string::npos && slash_pos < zone_id_str.size())
|
| + zone_id_str.erase(0, slash_pos + 1);
|
| + // zone id has '_' in place of ' '.
|
| + ReplaceSubstringsAfterOffset(&zone_id_str, 0, "_", " ");
|
| + return ASCIIToUTF16(zone_id_str);
|
| +}
|
| +
|
| +} // namespace anonymous
|
| +
|
| +namespace chromeos {
|
| +
|
| +SystemSettingsProvider::SystemSettingsProvider(
|
| + const NotifyObserversCallback& notify_cb)
|
| + : CrosSettingsProvider(notify_cb) {
|
| + for (size_t i = 0; i < arraysize(kTimeZones); i++) {
|
| + timezones_.push_back(icu::TimeZone::createTimeZone(
|
| + icu::UnicodeString(kTimeZones[i], -1, US_INV)));
|
| + }
|
| + system::TimezoneSettings::GetInstance()->AddObserver(this);
|
| + timezone_value_.reset(base::Value::CreateStringValue(GetKnownTimezoneID(
|
| + system::TimezoneSettings::GetInstance()->GetTimezone())));
|
| +}
|
| +
|
| +SystemSettingsProvider::~SystemSettingsProvider() {
|
| + system::TimezoneSettings::GetInstance()->RemoveObserver(this);
|
| + STLDeleteElements(&timezones_);
|
| +}
|
| +
|
| +void SystemSettingsProvider::DoSet(const std::string& path,
|
| + const base::Value& in_value) {
|
| + // Non-guest users can change the time zone.
|
| + if (UserManager::Get()->IsLoggedInAsGuest())
|
| + return;
|
| +
|
| + if (path == kSystemTimezone) {
|
| + string16 value;
|
| + if (!in_value.IsType(Value::TYPE_STRING) || !in_value.GetAsString(&value))
|
| + return;
|
| + const icu::TimeZone* timezone = GetTimezone(value);
|
| + if (!timezone)
|
| + return;
|
| + system::TimezoneSettings::GetInstance()->SetTimezone(*timezone);
|
| + timezone_value_.reset(
|
| + base::Value::CreateStringValue(GetKnownTimezoneID(*timezone)));
|
| + }
|
| +}
|
| +
|
| +const base::Value* SystemSettingsProvider::Get(const std::string& path) const {
|
| + if (path == kSystemTimezone)
|
| + return timezone_value_.get();
|
| + return NULL;
|
| +}
|
| +
|
| +// The timezone is always trusted.
|
| +bool SystemSettingsProvider::GetTrusted(const std::string& path,
|
| + const base::Closure& callback) {
|
| + return true;
|
| +}
|
| +
|
| +bool SystemSettingsProvider::HandlesSetting(const std::string& path) const {
|
| + return path == kSystemTimezone;
|
| +}
|
| +
|
| +void SystemSettingsProvider::Reload() {
|
| + // TODO(pastarmovj): We can actually cache the timezone here to make returning
|
| + // it faster.
|
| +}
|
| +
|
| +void SystemSettingsProvider::TimezoneChanged(const icu::TimeZone& timezone) {
|
| + // Fires system setting change notification.
|
| + timezone_value_.reset(
|
| + base::Value::CreateStringValue(GetKnownTimezoneID(timezone)));
|
| + NotifyObservers(kSystemTimezone);
|
| +}
|
| +
|
| +ListValue* SystemSettingsProvider::GetTimezoneList() {
|
| + ListValue* timezoneList = new ListValue();
|
| + for (std::vector<icu::TimeZone*>::iterator iter = timezones_.begin();
|
| + iter != timezones_.end(); ++iter) {
|
| + const icu::TimeZone* timezone = *iter;
|
| + ListValue* option = new ListValue();
|
| + option->Append(Value::CreateStringValue(GetTimezoneID(*timezone)));
|
| + option->Append(Value::CreateStringValue(GetTimezoneName(*timezone)));
|
| + timezoneList->Append(option);
|
| + }
|
| + return timezoneList;
|
| +}
|
| +
|
| +string16 SystemSettingsProvider::GetTimezoneName(
|
| + const icu::TimeZone& timezone) {
|
| + // Instead of using the raw_offset, use the offset in effect now.
|
| + // For instance, US Pacific Time, the offset shown will be -7 in summer
|
| + // while it'll be -8 in winter.
|
| + int raw_offset, dst_offset;
|
| + UDate now = icu::Calendar::getNow();
|
| + UErrorCode status = U_ZERO_ERROR;
|
| + timezone.getOffset(now, false, raw_offset, dst_offset, status);
|
| + DCHECK(U_SUCCESS(status));
|
| + int offset = raw_offset + dst_offset;
|
| + // offset is in msec.
|
| + int minute_offset = std::abs(offset) / 60000;
|
| + int hour_offset = minute_offset / 60;
|
| + int min_remainder = minute_offset % 60;
|
| + // Some timezones have a non-integral hour offset. So, we need to
|
| + // use hh:mm form.
|
| + std::string offset_str = base::StringPrintf(offset >= 0 ?
|
| + "UTC+%d:%02d" : "UTC-%d:%02d", hour_offset, min_remainder);
|
| +
|
| + // TODO(jungshik): When coming up with a better list of timezones, we also
|
| + // have to come up with better 'display' names. One possibility is to list
|
| + // multiple cities (e.g. "Los Angeles, Vancouver .." in the order of
|
| + // the population of a country the city belongs to.).
|
| + // We can also think of using LONG_GENERIC or LOCATION once we upgrade
|
| + // to ICU 4.6.
|
| + // In the meantime, we use "LONG" name with "Exemplar City" to distinguish
|
| + // multiple timezones with the same "LONG" name but with different
|
| + // rules (e.g. US Mountain Time in Denver vs Phoenix).
|
| + icu::UnicodeString name;
|
| + timezone.getDisplayName(dst_offset != 0, icu::TimeZone::LONG, name);
|
| + string16 result(l10n_util::GetStringFUTF16(
|
| + IDS_OPTIONS_SETTINGS_TIMEZONE_DISPLAY_TEMPLATE, ASCIIToUTF16(offset_str),
|
| + string16(name.getBuffer(), name.length()), GetExemplarCity(timezone)));
|
| + base::i18n::AdjustStringForLocaleDirection(&result);
|
| + return result;
|
| +}
|
| +
|
| +string16 SystemSettingsProvider::GetTimezoneID(
|
| + const icu::TimeZone& timezone) {
|
| + icu::UnicodeString id;
|
| + timezone.getID(id);
|
| + return string16(id.getBuffer(), id.length());
|
| +}
|
| +
|
| +const icu::TimeZone* SystemSettingsProvider::GetTimezone(
|
| + const string16& timezone_id) {
|
| + for (std::vector<icu::TimeZone*>::iterator iter = timezones_.begin();
|
| + iter != timezones_.end(); ++iter) {
|
| + const icu::TimeZone* timezone = *iter;
|
| + if (GetTimezoneID(*timezone) == timezone_id) {
|
| + return timezone;
|
| + }
|
| + }
|
| + return NULL;
|
| +}
|
| +
|
| +string16 SystemSettingsProvider::GetKnownTimezoneID(
|
| + const icu::TimeZone& timezone) const {
|
| + for (std::vector<icu::TimeZone*>::const_iterator iter = timezones_.begin();
|
| + iter != timezones_.end(); ++iter) {
|
| + const icu::TimeZone* known_timezone = *iter;
|
| + if (known_timezone->hasSameRules(timezone))
|
| + return GetTimezoneID(*known_timezone);
|
| + }
|
| +
|
| + // Not able to find a matching timezone in our list.
|
| + return string16();
|
| +}
|
| +
|
| +} // namespace chromeos
|
|
|