Optimized Time preflet:
* use upon-demand initialization in BTimeZone to avoid unnecessary work * renamed BTimeZone::Code() to BTimeZone::ID() and adjusted all callers * avoid using BCountry in the Time preflet for the time being, this means the icon-flags are gone for now (but they could be re-added if the demand is pressing ;-) * group the timezones by regions and then by country instead The performance improvement is considerable and I personally think the new grouping is an improvement, too. Please share your thoughts! git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@38322 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
039ad7e757
commit
750e57b842
@ -9,12 +9,20 @@
|
|||||||
#include <String.h>
|
#include <String.h>
|
||||||
|
|
||||||
|
|
||||||
|
namespace icu_44 {
|
||||||
|
class TimeZone;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class BTimeZone {
|
class BTimeZone {
|
||||||
public:
|
public:
|
||||||
BTimeZone(const char* zoneCode = NULL);
|
BTimeZone(const char* zoneID = NULL);
|
||||||
|
BTimeZone(const BTimeZone& other);
|
||||||
~BTimeZone();
|
~BTimeZone();
|
||||||
|
|
||||||
const BString& Code() const;
|
BTimeZone& operator=(const BTimeZone& source);
|
||||||
|
|
||||||
|
const BString& ID() const;
|
||||||
const BString& Name() const;
|
const BString& Name() const;
|
||||||
const BString& DaylightSavingName() const;
|
const BString& DaylightSavingName() const;
|
||||||
const BString& ShortName() const;
|
const BString& ShortName() const;
|
||||||
@ -24,20 +32,22 @@ public:
|
|||||||
|
|
||||||
status_t InitCheck() const;
|
status_t InitCheck() const;
|
||||||
|
|
||||||
status_t SetTo(const char* zoneCode);
|
status_t SetTo(const char* zoneID);
|
||||||
|
|
||||||
static const char* kNameOfGmtZone;
|
static const char* kNameOfGmtZone;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
BString fCode;
|
icu_44::TimeZone* fIcuTimeZone;
|
||||||
BString fName;
|
|
||||||
BString fDaylightSavingName;
|
|
||||||
BString fShortName;
|
|
||||||
BString fShortDaylightSavingName;
|
|
||||||
int fOffsetFromGMT;
|
|
||||||
bool fSupportsDaylightSaving;
|
|
||||||
|
|
||||||
status_t fInitStatus;
|
status_t fInitStatus;
|
||||||
|
|
||||||
|
mutable uint32 fInitializedFields;
|
||||||
|
mutable BString fZoneID;
|
||||||
|
mutable BString fName;
|
||||||
|
mutable BString fDaylightSavingName;
|
||||||
|
mutable BString fShortName;
|
||||||
|
mutable BString fShortDaylightSavingName;
|
||||||
|
mutable int fOffsetFromGMT;
|
||||||
|
mutable bool fSupportsDaylightSaving;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
@ -405,7 +405,7 @@ BLocale::FormatTime(BString* string, time_t time, bool longFormat,
|
|||||||
|
|
||||||
if (timeZone != NULL) {
|
if (timeZone != NULL) {
|
||||||
ObjectDeleter<TimeZone> icuTimeZone
|
ObjectDeleter<TimeZone> icuTimeZone
|
||||||
= TimeZone::createTimeZone(timeZone->Code().String());
|
= TimeZone::createTimeZone(timeZone->ID().String());
|
||||||
if (icuTimeZone.Get() == NULL)
|
if (icuTimeZone.Get() == NULL)
|
||||||
return B_NO_MEMORY;
|
return B_NO_MEMORY;
|
||||||
timeFormatter->setTimeZone(*icuTimeZone.Get());
|
timeFormatter->setTimeZone(*icuTimeZone.Get());
|
||||||
|
@ -434,8 +434,6 @@ RosterData::_LoadLocaleSettings()
|
|||||||
BMessage settings;
|
BMessage settings;
|
||||||
if (status == B_OK)
|
if (status == B_OK)
|
||||||
status = settings.Unflatten(&file);
|
status = settings.Unflatten(&file);
|
||||||
if (status == B_OK)
|
|
||||||
status = _SetPreferredLanguages(&settings);
|
|
||||||
|
|
||||||
if (status == B_OK) {
|
if (status == B_OK) {
|
||||||
BString codeName;
|
BString codeName;
|
||||||
@ -454,6 +452,9 @@ RosterData::_LoadLocaleSettings()
|
|||||||
|
|
||||||
return B_OK;
|
return B_OK;
|
||||||
}
|
}
|
||||||
|
if (status == B_OK)
|
||||||
|
status = _SetPreferredLanguages(&settings);
|
||||||
|
|
||||||
|
|
||||||
// Something went wrong (no settings file or invalid BMessage), so we
|
// Something went wrong (no settings file or invalid BMessage), so we
|
||||||
// set everything to default values
|
// set everything to default values
|
||||||
@ -481,9 +482,9 @@ RosterData::_LoadTimeSettings()
|
|||||||
if (status == B_OK)
|
if (status == B_OK)
|
||||||
status = settings.Unflatten(&file);
|
status = settings.Unflatten(&file);
|
||||||
if (status == B_OK) {
|
if (status == B_OK) {
|
||||||
BString timeZoneCode;
|
BString timeZoneID;
|
||||||
if (settings.FindString("timezone", &timeZoneCode) == B_OK)
|
if (settings.FindString("timezone", &timeZoneID) == B_OK)
|
||||||
_SetDefaultTimeZone(BTimeZone(timeZoneCode.String()));
|
_SetDefaultTimeZone(BTimeZone(timeZoneID.String()));
|
||||||
else
|
else
|
||||||
_SetDefaultTimeZone(BTimeZone(BTimeZone::kNameOfGmtZone));
|
_SetDefaultTimeZone(BTimeZone(BTimeZone::kNameOfGmtZone));
|
||||||
|
|
||||||
@ -564,6 +565,15 @@ RosterData::_SetDefaultLocale(const BLocale& newLocale)
|
|||||||
{
|
{
|
||||||
fDefaultLocale = newLocale;
|
fDefaultLocale = newLocale;
|
||||||
|
|
||||||
|
UErrorCode icuError = U_ZERO_ERROR;
|
||||||
|
Locale icuLocale = Locale::createCanonical(newLocale.Code());
|
||||||
|
if (icuLocale.isBogus())
|
||||||
|
return B_ERROR;
|
||||||
|
|
||||||
|
Locale::setDefault(icuLocale, icuError);
|
||||||
|
if (!U_SUCCESS(icuError))
|
||||||
|
return B_ERROR;
|
||||||
|
|
||||||
return B_OK;
|
return B_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -573,7 +583,7 @@ RosterData::_SetDefaultTimeZone(const BTimeZone& newZone)
|
|||||||
{
|
{
|
||||||
fDefaultTimeZone = newZone;
|
fDefaultTimeZone = newZone;
|
||||||
|
|
||||||
TimeZone* timeZone = TimeZone::createTimeZone(newZone.Code().String());
|
TimeZone* timeZone = TimeZone::createTimeZone(newZone.ID().String());
|
||||||
if (timeZone == NULL)
|
if (timeZone == NULL)
|
||||||
return B_ERROR;
|
return B_ERROR;
|
||||||
TimeZone::adoptDefault(timeZone);
|
TimeZone::adoptDefault(timeZone);
|
||||||
@ -588,15 +598,6 @@ RosterData::_SetPreferredLanguages(const BMessage* languages)
|
|||||||
BString langName;
|
BString langName;
|
||||||
if (languages != NULL
|
if (languages != NULL
|
||||||
&& languages->FindString("language", &langName) == B_OK) {
|
&& languages->FindString("language", &langName) == B_OK) {
|
||||||
UErrorCode icuError = U_ZERO_ERROR;
|
|
||||||
Locale icuLocale = Locale::createCanonical(langName.String());
|
|
||||||
if (icuLocale.isBogus())
|
|
||||||
return B_ERROR;
|
|
||||||
|
|
||||||
Locale::setDefault(icuLocale, icuError);
|
|
||||||
if (!U_SUCCESS(icuError))
|
|
||||||
return B_ERROR;
|
|
||||||
|
|
||||||
fDefaultLocale.SetCollator(BCollator(langName.String()));
|
fDefaultLocale.SetCollator(BCollator(langName.String()));
|
||||||
fDefaultLanguage.SetTo(langName.String());
|
fDefaultLanguage.SetTo(langName.String());
|
||||||
|
|
||||||
@ -637,7 +638,7 @@ RosterData::_AddDefaultCountryToMessage(BMessage* message) const
|
|||||||
status_t
|
status_t
|
||||||
RosterData::_AddDefaultTimeZoneToMessage(BMessage* message) const
|
RosterData::_AddDefaultTimeZoneToMessage(BMessage* message) const
|
||||||
{
|
{
|
||||||
status_t status = message->AddString("timezone", fDefaultTimeZone.Code());
|
status_t status = message->AddString("timezone", fDefaultTimeZone.ID());
|
||||||
|
|
||||||
// add the offset, too, since that is used by clockconfig when setting
|
// add the offset, too, since that is used by clockconfig when setting
|
||||||
// up timezone state during boot
|
// up timezone state during boot
|
||||||
|
@ -10,6 +10,8 @@
|
|||||||
|
|
||||||
#include <TimeZone.h>
|
#include <TimeZone.h>
|
||||||
|
|
||||||
|
#include <new>
|
||||||
|
|
||||||
#include <unicode/timezone.h>
|
#include <unicode/timezone.h>
|
||||||
#include <ICUWrapper.h>
|
#include <ICUWrapper.h>
|
||||||
|
|
||||||
@ -17,20 +19,93 @@
|
|||||||
const char* BTimeZone::kNameOfGmtZone = "GMT";
|
const char* BTimeZone::kNameOfGmtZone = "GMT";
|
||||||
|
|
||||||
|
|
||||||
BTimeZone::BTimeZone(const char* zoneCode)
|
static const BString skEmptyString;
|
||||||
|
|
||||||
|
|
||||||
|
static const uint32 skNameField = 1U << 0;
|
||||||
|
static const uint32 skDaylightSavingNameField = 1U << 1;
|
||||||
|
static const uint32 skShortNameField = 1U << 2;
|
||||||
|
static const uint32 skShortDaylightSavingNameField = 1U << 3;
|
||||||
|
static const uint32 skLongGenericNameField = 1U << 4;
|
||||||
|
static const uint32 skGenericLocationNameField = 1U << 5;
|
||||||
|
static const uint32 skShortCommonlyUsedNameField = 1U << 6;
|
||||||
|
static const uint32 skSupportsDaylightSavingField = 1U << 7;
|
||||||
|
static const uint32 skOffsetFromGMTField = 1U << 8;
|
||||||
|
|
||||||
|
|
||||||
|
BTimeZone::BTimeZone(const char* zoneID)
|
||||||
|
:
|
||||||
|
fIcuTimeZone(NULL),
|
||||||
|
fInitStatus(B_NO_INIT),
|
||||||
|
fInitializedFields(0)
|
||||||
|
{
|
||||||
|
SetTo(zoneID);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
BTimeZone::BTimeZone(const BTimeZone& other)
|
||||||
|
:
|
||||||
|
fIcuTimeZone(other.fIcuTimeZone == NULL
|
||||||
|
? NULL
|
||||||
|
: other.fIcuTimeZone->clone()),
|
||||||
|
fInitStatus(other.fInitStatus),
|
||||||
|
fInitializedFields(other.fInitializedFields),
|
||||||
|
fZoneID(other.fZoneID),
|
||||||
|
fName(other.fName),
|
||||||
|
fDaylightSavingName(other.fDaylightSavingName),
|
||||||
|
fShortName(other.fShortName),
|
||||||
|
fShortDaylightSavingName(other.fShortDaylightSavingName),
|
||||||
|
fOffsetFromGMT(other.fOffsetFromGMT),
|
||||||
|
fSupportsDaylightSaving(other.fSupportsDaylightSaving)
|
||||||
{
|
{
|
||||||
SetTo(zoneCode);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
BTimeZone::~BTimeZone()
|
BTimeZone::~BTimeZone()
|
||||||
{
|
{
|
||||||
|
delete fIcuTimeZone;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
BTimeZone& BTimeZone::operator=(const BTimeZone& source)
|
||||||
|
{
|
||||||
|
delete fIcuTimeZone;
|
||||||
|
fIcuTimeZone = source.fIcuTimeZone == NULL
|
||||||
|
? NULL
|
||||||
|
: source.fIcuTimeZone->clone();
|
||||||
|
fInitStatus = source.fInitStatus;
|
||||||
|
fInitializedFields = source.fInitializedFields;
|
||||||
|
fZoneID = source.fZoneID;
|
||||||
|
fName = source.fName;
|
||||||
|
fDaylightSavingName = source.fDaylightSavingName;
|
||||||
|
fShortName = source.fShortName;
|
||||||
|
fShortDaylightSavingName = source.fShortDaylightSavingName;
|
||||||
|
fOffsetFromGMT = source.fOffsetFromGMT;
|
||||||
|
fSupportsDaylightSaving = source.fSupportsDaylightSaving;
|
||||||
|
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const BString&
|
||||||
|
BTimeZone::ID() const
|
||||||
|
{
|
||||||
|
return fZoneID;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
const BString&
|
const BString&
|
||||||
BTimeZone::Name() const
|
BTimeZone::Name() const
|
||||||
{
|
{
|
||||||
|
if ((fInitializedFields & skNameField) == 0) {
|
||||||
|
UnicodeString unicodeString;
|
||||||
|
fIcuTimeZone->getDisplayName(false, TimeZone::GENERIC_LOCATION,
|
||||||
|
unicodeString);
|
||||||
|
BStringByteSink sink(&fName);
|
||||||
|
unicodeString.toUTF8(sink);
|
||||||
|
fInitializedFields |= skNameField;
|
||||||
|
}
|
||||||
|
|
||||||
return fName;
|
return fName;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -38,6 +113,15 @@ BTimeZone::Name() const
|
|||||||
const BString&
|
const BString&
|
||||||
BTimeZone::DaylightSavingName() const
|
BTimeZone::DaylightSavingName() const
|
||||||
{
|
{
|
||||||
|
if ((fInitializedFields & skDaylightSavingNameField) == 0) {
|
||||||
|
UnicodeString unicodeString;
|
||||||
|
fIcuTimeZone->getDisplayName(true, TimeZone::GENERIC_LOCATION,
|
||||||
|
unicodeString);
|
||||||
|
BStringByteSink sink(&fDaylightSavingName);
|
||||||
|
unicodeString.toUTF8(sink);
|
||||||
|
fInitializedFields |= skDaylightSavingNameField;
|
||||||
|
}
|
||||||
|
|
||||||
return fDaylightSavingName;
|
return fDaylightSavingName;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -45,6 +129,15 @@ BTimeZone::DaylightSavingName() const
|
|||||||
const BString&
|
const BString&
|
||||||
BTimeZone::ShortName() const
|
BTimeZone::ShortName() const
|
||||||
{
|
{
|
||||||
|
if ((fInitializedFields & skShortNameField) == 0) {
|
||||||
|
UnicodeString unicodeString;
|
||||||
|
fIcuTimeZone->getDisplayName(false, TimeZone::SHORT_COMMONLY_USED,
|
||||||
|
unicodeString);
|
||||||
|
BStringByteSink sink(&fShortName);
|
||||||
|
unicodeString.toUTF8(sink);
|
||||||
|
fInitializedFields |= skShortNameField;
|
||||||
|
}
|
||||||
|
|
||||||
return fShortName;
|
return fShortName;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -52,20 +145,38 @@ BTimeZone::ShortName() const
|
|||||||
const BString&
|
const BString&
|
||||||
BTimeZone::ShortDaylightSavingName() const
|
BTimeZone::ShortDaylightSavingName() const
|
||||||
{
|
{
|
||||||
|
if ((fInitializedFields & skShortDaylightSavingNameField) == 0) {
|
||||||
|
UnicodeString unicodeString;
|
||||||
|
fIcuTimeZone->getDisplayName(true, TimeZone::SHORT_COMMONLY_USED,
|
||||||
|
unicodeString);
|
||||||
|
BStringByteSink sink(&fShortDaylightSavingName);
|
||||||
|
unicodeString.toUTF8(sink);
|
||||||
|
fInitializedFields |= skShortDaylightSavingNameField;
|
||||||
|
}
|
||||||
|
|
||||||
return fShortDaylightSavingName;
|
return fShortDaylightSavingName;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
const BString&
|
|
||||||
BTimeZone::Code() const
|
|
||||||
{
|
|
||||||
return fCode;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int
|
int
|
||||||
BTimeZone::OffsetFromGMT() const
|
BTimeZone::OffsetFromGMT() const
|
||||||
{
|
{
|
||||||
|
if ((fInitializedFields & skOffsetFromGMTField) == 0) {
|
||||||
|
int32_t rawOffset;
|
||||||
|
int32_t dstOffset;
|
||||||
|
UDate nowMillis = 1000 * (double)time(NULL);
|
||||||
|
|
||||||
|
UErrorCode error = U_ZERO_ERROR;
|
||||||
|
fIcuTimeZone->getOffset(nowMillis, FALSE, rawOffset, dstOffset, error);
|
||||||
|
if (!U_SUCCESS(error))
|
||||||
|
fOffsetFromGMT = 0;
|
||||||
|
else {
|
||||||
|
fOffsetFromGMT = (rawOffset + dstOffset) / 1000;
|
||||||
|
// we want seconds, not ms (which ICU gives us)
|
||||||
|
}
|
||||||
|
fInitializedFields |= skOffsetFromGMTField;
|
||||||
|
}
|
||||||
|
|
||||||
return fOffsetFromGMT;
|
return fOffsetFromGMT;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -73,6 +184,11 @@ BTimeZone::OffsetFromGMT() const
|
|||||||
bool
|
bool
|
||||||
BTimeZone::SupportsDaylightSaving() const
|
BTimeZone::SupportsDaylightSaving() const
|
||||||
{
|
{
|
||||||
|
if ((fInitializedFields & skSupportsDaylightSavingField) == 0) {
|
||||||
|
fSupportsDaylightSaving = fIcuTimeZone->useDaylightTime();
|
||||||
|
fInitializedFields |= skSupportsDaylightSavingField;
|
||||||
|
}
|
||||||
|
|
||||||
return fSupportsDaylightSaving;
|
return fSupportsDaylightSaving;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -85,57 +201,27 @@ BTimeZone::InitCheck() const
|
|||||||
|
|
||||||
|
|
||||||
status_t
|
status_t
|
||||||
BTimeZone::SetTo(const char* zoneCode)
|
BTimeZone::SetTo(const char* zoneID)
|
||||||
{
|
{
|
||||||
TimeZone* icuTimeZone;
|
delete fIcuTimeZone;
|
||||||
if (zoneCode == NULL || zoneCode[0] == '\0')
|
fInitializedFields = 0;
|
||||||
icuTimeZone = TimeZone::createDefault();
|
|
||||||
|
if (zoneID == NULL || zoneID[0] == '\0')
|
||||||
|
fIcuTimeZone = TimeZone::createDefault();
|
||||||
else
|
else
|
||||||
icuTimeZone = TimeZone::createTimeZone(zoneCode);
|
fIcuTimeZone = TimeZone::createTimeZone(zoneID);
|
||||||
|
|
||||||
UnicodeString unicodeString;
|
if (fIcuTimeZone == NULL) {
|
||||||
icuTimeZone->getID(unicodeString);
|
fInitStatus = B_NAME_NOT_FOUND;
|
||||||
BStringByteSink converter(&fCode);
|
return fInitStatus;
|
||||||
unicodeString.toUTF8(converter);
|
|
||||||
|
|
||||||
unicodeString.remove();
|
|
||||||
icuTimeZone->getDisplayName(false, TimeZone::LONG, unicodeString);
|
|
||||||
converter.SetTo(&fName);
|
|
||||||
unicodeString.toUTF8(converter);
|
|
||||||
|
|
||||||
unicodeString.remove();
|
|
||||||
icuTimeZone->getDisplayName(true, TimeZone::LONG, unicodeString);
|
|
||||||
converter.SetTo(&fDaylightSavingName);
|
|
||||||
unicodeString.toUTF8(converter);
|
|
||||||
|
|
||||||
unicodeString.remove();
|
|
||||||
icuTimeZone->getDisplayName(false, TimeZone::SHORT, unicodeString);
|
|
||||||
converter.SetTo(&fShortName);
|
|
||||||
unicodeString.toUTF8(converter);
|
|
||||||
|
|
||||||
unicodeString.remove();
|
|
||||||
icuTimeZone->getDisplayName(true, TimeZone::SHORT, unicodeString);
|
|
||||||
converter.SetTo(&fShortDaylightSavingName);
|
|
||||||
unicodeString.toUTF8(converter);
|
|
||||||
|
|
||||||
fSupportsDaylightSaving = icuTimeZone->useDaylightTime();
|
|
||||||
|
|
||||||
int32_t rawOffset;
|
|
||||||
int32_t dstOffset;
|
|
||||||
UDate nowMillis = 1000 * (double)time(NULL);
|
|
||||||
|
|
||||||
UErrorCode error = U_ZERO_ERROR;
|
|
||||||
icuTimeZone->getOffset(nowMillis, FALSE, rawOffset, dstOffset, error);
|
|
||||||
if (!U_SUCCESS(error)) {
|
|
||||||
fOffsetFromGMT = 0;
|
|
||||||
fInitStatus = B_ERROR;
|
|
||||||
} else {
|
|
||||||
fOffsetFromGMT = (rawOffset + dstOffset) / 1000;
|
|
||||||
// we want seconds, not ms (which ICU gives us)
|
|
||||||
fInitStatus = B_OK;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
delete icuTimeZone;
|
UnicodeString unicodeString;
|
||||||
|
fIcuTimeZone->getID(unicodeString);
|
||||||
|
BStringByteSink sink(&fZoneID);
|
||||||
|
unicodeString.toUTF8(sink);
|
||||||
|
|
||||||
|
fInitStatus = B_OK;
|
||||||
|
|
||||||
return fInitStatus;
|
return fInitStatus;
|
||||||
}
|
}
|
||||||
|
@ -2,7 +2,7 @@ SubDir HAIKU_TOP src preferences time ;
|
|||||||
|
|
||||||
SetSubDirSupportedPlatformsBeOSCompatible ;
|
SetSubDirSupportedPlatformsBeOSCompatible ;
|
||||||
|
|
||||||
UsePrivateHeaders locale shared [ FDirName libroot time ] ;
|
UsePrivateHeaders interface locale shared [ FDirName libroot time ] ;
|
||||||
UsePrivateSystemHeaders ;
|
UsePrivateSystemHeaders ;
|
||||||
|
|
||||||
local sources =
|
local sources =
|
||||||
@ -28,7 +28,7 @@ Includes [ FGristFiles $(sources) ] : $(HAIKU_ICU_HEADERS_DEPENDENCY) ;
|
|||||||
|
|
||||||
Preference Time
|
Preference Time
|
||||||
: $(sources)
|
: $(sources)
|
||||||
: be libshared.a $(TARGET_LIBSUPC++) $(HAIKU_LOCALE_LIBS)
|
: be libshared.a $(TARGET_LIBSTDC++) $(HAIKU_LOCALE_LIBS)
|
||||||
: Time.rdef
|
: Time.rdef
|
||||||
;
|
;
|
||||||
|
|
||||||
|
@ -109,12 +109,12 @@ TimeZoneListItem::TimeZone() const
|
|||||||
|
|
||||||
|
|
||||||
const BString&
|
const BString&
|
||||||
TimeZoneListItem::Code() const
|
TimeZoneListItem::ID() const
|
||||||
{
|
{
|
||||||
if (fTimeZone == NULL)
|
if (fTimeZone == NULL)
|
||||||
return skDefaultString;
|
return skDefaultString;
|
||||||
|
|
||||||
return fTimeZone->Code();
|
return fTimeZone->ID();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -29,7 +29,7 @@ public:
|
|||||||
|
|
||||||
bool HasTimeZone() const;
|
bool HasTimeZone() const;
|
||||||
const BTimeZone& TimeZone() const;
|
const BTimeZone& TimeZone() const;
|
||||||
const BString& Code() const;
|
const BString& ID() const;
|
||||||
const BString& Name() const;
|
const BString& Name() const;
|
||||||
int OffsetFromGMT() const;
|
int OffsetFromGMT() const;
|
||||||
|
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
|
||||||
|
#include <map>
|
||||||
#include <new>
|
#include <new>
|
||||||
|
|
||||||
#include <AutoDeleter.h>
|
#include <AutoDeleter.h>
|
||||||
@ -37,6 +38,7 @@
|
|||||||
#include <Window.h>
|
#include <Window.h>
|
||||||
|
|
||||||
#include <syscalls.h>
|
#include <syscalls.h>
|
||||||
|
#include <ToolTip.h>
|
||||||
|
|
||||||
#include <unicode/datefmt.h>
|
#include <unicode/datefmt.h>
|
||||||
#include <unicode/utmscale.h>
|
#include <unicode/utmscale.h>
|
||||||
@ -52,9 +54,21 @@ using BPrivate::gMutableLocaleRoster;
|
|||||||
using BPrivate::ObjectDeleter;
|
using BPrivate::ObjectDeleter;
|
||||||
|
|
||||||
|
|
||||||
|
struct TimeZoneItemLess {
|
||||||
|
bool operator()(const BString& first, const BString& second)
|
||||||
|
{
|
||||||
|
return fCollator.Compare(first.String(), second.String()) < 0;
|
||||||
|
}
|
||||||
|
private:
|
||||||
|
BCollator fCollator;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
TimeZoneView::TimeZoneView(BRect frame)
|
TimeZoneView::TimeZoneView(BRect frame)
|
||||||
:
|
:
|
||||||
BView(frame, "timeZoneView", B_FOLLOW_NONE, B_WILL_DRAW | B_NAVIGABLE_JUMP),
|
BView(frame, "timeZoneView", B_FOLLOW_NONE, B_WILL_DRAW | B_NAVIGABLE_JUMP),
|
||||||
|
fToolTip(NULL),
|
||||||
fCurrentZoneItem(NULL),
|
fCurrentZoneItem(NULL),
|
||||||
fOldZoneItem(NULL),
|
fOldZoneItem(NULL),
|
||||||
fInitialized(false)
|
fInitialized(false)
|
||||||
@ -70,26 +84,9 @@ TimeZoneView::CheckCanRevert()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void
|
|
||||||
TimeZoneView::_Revert()
|
|
||||||
{
|
|
||||||
fCurrentZoneItem = fOldZoneItem;
|
|
||||||
|
|
||||||
if (fCurrentZoneItem != NULL) {
|
|
||||||
int32 currentZoneIndex = fZoneList->IndexOf(fCurrentZoneItem);
|
|
||||||
fZoneList->Select(currentZoneIndex);
|
|
||||||
} else
|
|
||||||
fZoneList->DeselectAll();
|
|
||||||
fZoneList->ScrollToSelection();
|
|
||||||
|
|
||||||
_SetSystemTimeZone();
|
|
||||||
_UpdatePreview();
|
|
||||||
_UpdateCurrent();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
TimeZoneView::~TimeZoneView()
|
TimeZoneView::~TimeZoneView()
|
||||||
{
|
{
|
||||||
|
delete fToolTip;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -165,6 +162,29 @@ TimeZoneView::MessageReceived(BMessage* message)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
bool
|
||||||
|
TimeZoneView::GetToolTipAt(BPoint point, BToolTip** _tip)
|
||||||
|
{
|
||||||
|
TimeZoneListItem* item = static_cast<TimeZoneListItem*>(
|
||||||
|
fZoneList->ItemAt(fZoneList->IndexOf(point)));
|
||||||
|
if (item == NULL || !item->HasTimeZone())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
BString toolTip = item->Text();
|
||||||
|
toolTip << '\n' << item->TimeZone().ShortName() << " / "
|
||||||
|
<< item->TimeZone().ShortDaylightSavingName();
|
||||||
|
|
||||||
|
delete fToolTip;
|
||||||
|
fToolTip = new (std::nothrow) BTextToolTip(toolTip.String());
|
||||||
|
if (fToolTip == NULL)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
*_tip = fToolTip;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void
|
void
|
||||||
TimeZoneView::_UpdateDateTime(BMessage* message)
|
TimeZoneView::_UpdateDateTime(BMessage* message)
|
||||||
{
|
{
|
||||||
@ -195,7 +215,7 @@ TimeZoneView::_InitView()
|
|||||||
fZoneList->SetSelectionMessage(new BMessage(H_CITY_CHANGED));
|
fZoneList->SetSelectionMessage(new BMessage(H_CITY_CHANGED));
|
||||||
fZoneList->SetInvocationMessage(new BMessage(H_SET_TIME_ZONE));
|
fZoneList->SetInvocationMessage(new BMessage(H_SET_TIME_ZONE));
|
||||||
|
|
||||||
_BuildRegionMenu();
|
_BuildZoneMenu();
|
||||||
|
|
||||||
BScrollView* scrollList = new BScrollView("scrollList", fZoneList,
|
BScrollView* scrollList = new BScrollView("scrollList", fZoneList,
|
||||||
B_FOLLOW_ALL, 0, false, true);
|
B_FOLLOW_ALL, 0, false, true);
|
||||||
@ -230,7 +250,7 @@ TimeZoneView::_InitView()
|
|||||||
|
|
||||||
|
|
||||||
void
|
void
|
||||||
TimeZoneView::_BuildRegionMenu()
|
TimeZoneView::_BuildZoneMenu()
|
||||||
{
|
{
|
||||||
BTimeZone defaultTimeZone = NULL;
|
BTimeZone defaultTimeZone = NULL;
|
||||||
be_locale_roster->GetDefaultTimeZone(&defaultTimeZone);
|
be_locale_roster->GetDefaultTimeZone(&defaultTimeZone);
|
||||||
@ -239,75 +259,119 @@ TimeZoneView::_BuildRegionMenu()
|
|||||||
// AddUnder() them (only if there are multiple ones).
|
// AddUnder() them (only if there are multiple ones).
|
||||||
// Finally expand the current country and highlight the active TZ.
|
// Finally expand the current country and highlight the active TZ.
|
||||||
|
|
||||||
BMessage countryList;
|
BMessage zoneList;
|
||||||
be_locale_roster->GetAvailableCountries(&countryList);
|
be_locale_roster->GetAvailableTimeZones(&zoneList);
|
||||||
|
|
||||||
BString countryCode;
|
typedef std::map<BString, TimeZoneListItem*, TimeZoneItemLess> ZoneItemMap;
|
||||||
for (int i = 0; countryList.FindString("countries", i, &countryCode)
|
ZoneItemMap zoneMap;
|
||||||
== B_OK; i++) {
|
const char* supportedRegions[] = {
|
||||||
BCountry country("", countryCode);
|
"Africa", "America", "Antarctica", "Arctic", "Asia", "Atlantic",
|
||||||
BString fullName;
|
"Australia", "Etc", "Europe", "Indian", "Pacific", NULL
|
||||||
country.GetName(fullName);
|
};
|
||||||
|
for (const char** region = supportedRegions; *region != NULL; ++region)
|
||||||
|
zoneMap[*region] = NULL;
|
||||||
|
|
||||||
// Now list the timezones for this country
|
BString zoneID;
|
||||||
BList tzList;
|
for (int i = 0; zoneList.FindString("timeZone", i, &zoneID) == B_OK; i++) {
|
||||||
|
int32 slashPos = zoneID.FindFirst('/');
|
||||||
|
|
||||||
BTimeZone* timeZone;
|
// ignore any "global" timezones, as those are just aliases of regional
|
||||||
TimeZoneListItem* countryItem;
|
// ones
|
||||||
switch (country.GetTimeZones(tzList))
|
if (slashPos <= 0)
|
||||||
{
|
continue;
|
||||||
case 0:
|
|
||||||
// No TZ info for this country. Sorry guys!
|
BString region(zoneID, slashPos);
|
||||||
break;
|
|
||||||
case 1:
|
// just accept timezones from "known" regions, as all others are aliases
|
||||||
// Only one Timezone, no need to add it to the list
|
ZoneItemMap::iterator regionIter = zoneMap.find(region);
|
||||||
timeZone = (BTimeZone*)tzList.ItemAt(0);
|
if (regionIter == zoneMap.end())
|
||||||
countryItem
|
continue;
|
||||||
= new TimeZoneListItem(fullName, &country, timeZone);
|
|
||||||
fZoneList->AddItem(countryItem);
|
TimeZoneListItem* regionItem = regionIter->second;
|
||||||
if (timeZone->Code() == defaultTimeZone.Code())
|
if (regionItem == NULL) {
|
||||||
fCurrentZoneItem = countryItem;
|
regionItem = new TimeZoneListItem(region, NULL, NULL);
|
||||||
break;
|
regionItem->SetOutlineLevel(0);
|
||||||
default:
|
regionItem->SetExpanded(false);
|
||||||
countryItem = new TimeZoneListItem(fullName, &country, NULL);
|
zoneMap[region] = regionItem;
|
||||||
|
}
|
||||||
|
|
||||||
|
BTimeZone* timeZone = new BTimeZone(zoneID);
|
||||||
|
BString tzName = timeZone->Name();
|
||||||
|
if (tzName == "GMT+00:00")
|
||||||
|
tzName = "GMT";
|
||||||
|
int32 openParenthesisPos = tzName.FindFirst('(');
|
||||||
|
BString country;
|
||||||
|
if (openParenthesisPos >= 0) {
|
||||||
|
if (openParenthesisPos > 0)
|
||||||
|
country.SetTo(tzName, openParenthesisPos - 1);
|
||||||
|
tzName.Remove(0, openParenthesisPos + 1);
|
||||||
|
int32 closeParenthesisPos = tzName.FindLast(')');
|
||||||
|
if (closeParenthesisPos >= 0)
|
||||||
|
tzName.Truncate(closeParenthesisPos);
|
||||||
|
}
|
||||||
|
BString fullCountryID = region;
|
||||||
|
if (country.Length() > 0 && country != region)
|
||||||
|
fullCountryID << "/" << country;
|
||||||
|
BString fullZoneID = fullCountryID;
|
||||||
|
fullZoneID << "/" << tzName;
|
||||||
|
|
||||||
|
// skip duplicates
|
||||||
|
ZoneItemMap::iterator zoneIter = zoneMap.find(fullZoneID);
|
||||||
|
if (zoneIter != zoneMap.end()) {
|
||||||
|
delete timeZone;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
TimeZoneListItem* countryItem = NULL;
|
||||||
|
if (country.Length() > 0) {
|
||||||
|
ZoneItemMap::iterator countryIter = zoneMap.find(fullCountryID);
|
||||||
|
if (countryIter == zoneMap.end()) {
|
||||||
|
countryItem = new TimeZoneListItem(country, NULL, NULL);
|
||||||
|
countryItem->SetOutlineLevel(1);
|
||||||
countryItem->SetExpanded(false);
|
countryItem->SetExpanded(false);
|
||||||
fZoneList->AddItem(countryItem);
|
zoneMap[fullCountryID] = countryItem;
|
||||||
|
} else
|
||||||
|
countryItem = countryIter->second;
|
||||||
|
}
|
||||||
|
|
||||||
for (int j = 0;
|
TimeZoneListItem* zoneItem
|
||||||
(timeZone = (BTimeZone*)tzList.ItemAt(j)) != NULL;
|
= new TimeZoneListItem(tzName, NULL, timeZone);
|
||||||
j++) {
|
zoneItem->SetOutlineLevel(countryItem == NULL ? 1 : 2);
|
||||||
TimeZoneListItem* tzItem = new TimeZoneListItem(
|
zoneMap[fullZoneID] = zoneItem;
|
||||||
timeZone->Name(), NULL, timeZone);
|
|
||||||
fZoneList->AddUnder(tzItem, countryItem);
|
if (timeZone->ID() == defaultTimeZone.ID()) {
|
||||||
if (timeZone->Code() == defaultTimeZone.Code())
|
fCurrentZoneItem = zoneItem;
|
||||||
{
|
if (countryItem != NULL)
|
||||||
fCurrentZoneItem = tzItem;
|
|
||||||
countryItem->SetExpanded(true);
|
countryItem->SetExpanded(true);
|
||||||
|
regionItem->SetExpanded(true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// add an artifical (i.e. belongs to no country) entry for GMT/UTC
|
|
||||||
BTimeZone* gmtZone = new(std::nothrow) BTimeZone(BTimeZone::kNameOfGmtZone);
|
|
||||||
TimeZoneListItem* gmtItem
|
|
||||||
= new TimeZoneListItem("- GMT/UTC (Universal Time) -", NULL, gmtZone);
|
|
||||||
fZoneList->AddItem(gmtItem);
|
|
||||||
if (gmtZone->Code() == defaultTimeZone.Code())
|
|
||||||
fCurrentZoneItem = gmtItem;
|
|
||||||
|
|
||||||
fOldZoneItem = fCurrentZoneItem;
|
fOldZoneItem = fCurrentZoneItem;
|
||||||
|
|
||||||
struct ListSorter {
|
ZoneItemMap::iterator zoneIter;
|
||||||
static int compare(const BListItem* first, const BListItem* second)
|
for (zoneIter = zoneMap.begin(); zoneIter != zoneMap.end(); ++zoneIter)
|
||||||
{
|
fZoneList->AddItem(zoneIter->second);
|
||||||
static BCollator collator;
|
|
||||||
return collator.Compare(((BStringItem*)first)->Text(),
|
fZoneList->Select(fZoneList->IndexOf(fCurrentZoneItem));
|
||||||
((BStringItem*)second)->Text());
|
}
|
||||||
}
|
|
||||||
};
|
|
||||||
fZoneList->SortItemsUnder(NULL, false, ListSorter::compare);
|
void
|
||||||
|
TimeZoneView::_Revert()
|
||||||
|
{
|
||||||
|
fCurrentZoneItem = fOldZoneItem;
|
||||||
|
|
||||||
|
if (fCurrentZoneItem != NULL) {
|
||||||
|
int32 currentZoneIndex = fZoneList->IndexOf(fCurrentZoneItem);
|
||||||
|
fZoneList->Select(currentZoneIndex);
|
||||||
|
} else
|
||||||
|
fZoneList->DeselectAll();
|
||||||
|
fZoneList->ScrollToSelection();
|
||||||
|
|
||||||
|
_SetSystemTimeZone();
|
||||||
|
_UpdatePreview();
|
||||||
|
_UpdateCurrent();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -315,12 +379,16 @@ void
|
|||||||
TimeZoneView::_UpdatePreview()
|
TimeZoneView::_UpdatePreview()
|
||||||
{
|
{
|
||||||
int32 selection = fZoneList->CurrentSelection();
|
int32 selection = fZoneList->CurrentSelection();
|
||||||
if (selection < 0)
|
TimeZoneListItem* item
|
||||||
return;
|
= selection < 0
|
||||||
|
? NULL
|
||||||
|
: (TimeZoneListItem*)fZoneList->ItemAt(selection);
|
||||||
|
|
||||||
TimeZoneListItem* item = (TimeZoneListItem*)fZoneList->ItemAt(selection);
|
if (item == NULL || !item->HasTimeZone()) {
|
||||||
if (!item->HasTimeZone())
|
fPreview->SetText("");
|
||||||
|
fPreview->SetTime("");
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
BString timeString = _FormatTime(item);
|
BString timeString = _FormatTime(item);
|
||||||
fPreview->SetText(item->Text());
|
fPreview->SetText(item->Text());
|
||||||
@ -360,8 +428,8 @@ TimeZoneView::_SetSystemTimeZone()
|
|||||||
|
|
||||||
gMutableLocaleRoster->SetDefaultTimeZone(timeZone);
|
gMutableLocaleRoster->SetDefaultTimeZone(timeZone);
|
||||||
|
|
||||||
_kern_set_timezone(timeZone.OffsetFromGMT(), timeZone.Code().String(),
|
_kern_set_timezone(timeZone.OffsetFromGMT(), timeZone.ID().String(),
|
||||||
timeZone.Code().Length());
|
timeZone.ID().Length());
|
||||||
|
|
||||||
fSetZone->SetEnabled(false);
|
fSetZone->SetEnabled(false);
|
||||||
fLastUpdateMinute = -1;
|
fLastUpdateMinute = -1;
|
||||||
|
@ -18,6 +18,7 @@ class BMessage;
|
|||||||
class BPopUpMenu;
|
class BPopUpMenu;
|
||||||
class BOutlineListView;
|
class BOutlineListView;
|
||||||
class BButton;
|
class BButton;
|
||||||
|
class BTextToolTip;
|
||||||
class TTZDisplay;
|
class TTZDisplay;
|
||||||
class TimeZoneListItem;
|
class TimeZoneListItem;
|
||||||
|
|
||||||
@ -31,6 +32,9 @@ public:
|
|||||||
virtual void MessageReceived(BMessage* message);
|
virtual void MessageReceived(BMessage* message);
|
||||||
bool CheckCanRevert();
|
bool CheckCanRevert();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
virtual bool GetToolTipAt(BPoint point, BToolTip** _tip);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void _UpdateDateTime(BMessage* message);
|
void _UpdateDateTime(BMessage* message);
|
||||||
|
|
||||||
@ -41,7 +45,7 @@ private:
|
|||||||
BString _FormatTime(TimeZoneListItem* zoneItem);
|
BString _FormatTime(TimeZoneListItem* zoneItem);
|
||||||
|
|
||||||
void _InitView();
|
void _InitView();
|
||||||
void _BuildRegionMenu();
|
void _BuildZoneMenu();
|
||||||
|
|
||||||
void _Revert();
|
void _Revert();
|
||||||
|
|
||||||
@ -51,6 +55,8 @@ private:
|
|||||||
TTZDisplay* fCurrent;
|
TTZDisplay* fCurrent;
|
||||||
TTZDisplay* fPreview;
|
TTZDisplay* fPreview;
|
||||||
|
|
||||||
|
BTextToolTip* fToolTip;
|
||||||
|
|
||||||
int32 fLastUpdateMinute;
|
int32 fLastUpdateMinute;
|
||||||
|
|
||||||
TimeZoneListItem* fCurrentZoneItem;
|
TimeZoneListItem* fCurrentZoneItem;
|
||||||
|
Loading…
Reference in New Issue
Block a user