/* * Copyright (C) 2011,2012 Google Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following disclaimer * in the documentation and/or other materials provided with the * distribution. * * Neither the name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #include "LocaleICU.h" #include "LocalizedStrings.h" #include #include #include #include #include using namespace icu; using namespace std; namespace WebCore { std::unique_ptr Locale::create(const AtomicString& locale) { return std::make_unique(locale.string().utf8().data()); } LocaleICU::LocaleICU(const char* locale) : m_locale(locale) , m_numberFormat(0) , m_shortDateFormat(0) , m_didCreateDecimalFormat(false) , m_didCreateShortDateFormat(false) #if ENABLE(DATE_AND_TIME_INPUT_TYPES) , m_mediumTimeFormat(0) , m_shortTimeFormat(0) , m_didCreateTimeFormat(false) #endif { } LocaleICU::~LocaleICU() { unum_close(m_numberFormat); udat_close(m_shortDateFormat); #if ENABLE(DATE_AND_TIME_INPUT_TYPES) udat_close(m_mediumTimeFormat); udat_close(m_shortTimeFormat); #endif } String LocaleICU::decimalSymbol(UNumberFormatSymbol symbol) { UErrorCode status = U_ZERO_ERROR; int32_t bufferLength = unum_getSymbol(m_numberFormat, symbol, 0, 0, &status); ASSERT(U_SUCCESS(status) || status == U_BUFFER_OVERFLOW_ERROR); if (U_FAILURE(status) && status != U_BUFFER_OVERFLOW_ERROR) return String(); Vector buffer(bufferLength); status = U_ZERO_ERROR; unum_getSymbol(m_numberFormat, symbol, buffer.data(), bufferLength, &status); if (U_FAILURE(status)) return String(); return String::adopt(buffer); } String LocaleICU::decimalTextAttribute(UNumberFormatTextAttribute tag) { UErrorCode status = U_ZERO_ERROR; int32_t bufferLength = unum_getTextAttribute(m_numberFormat, tag, 0, 0, &status); ASSERT(U_SUCCESS(status) || status == U_BUFFER_OVERFLOW_ERROR); if (U_FAILURE(status) && status != U_BUFFER_OVERFLOW_ERROR) return String(); Vector buffer(bufferLength); status = U_ZERO_ERROR; unum_getTextAttribute(m_numberFormat, tag, buffer.data(), bufferLength, &status); ASSERT(U_SUCCESS(status)); if (U_FAILURE(status)) return String(); return String::adopt(buffer); } void LocaleICU::initializeLocaleData() { if (m_didCreateDecimalFormat) return; m_didCreateDecimalFormat = true; UErrorCode status = U_ZERO_ERROR; m_numberFormat = unum_open(UNUM_DECIMAL, 0, 0, m_locale.data(), 0, &status); if (!U_SUCCESS(status)) return; Vector symbols; symbols.append(decimalSymbol(UNUM_ZERO_DIGIT_SYMBOL)); symbols.append(decimalSymbol(UNUM_ONE_DIGIT_SYMBOL)); symbols.append(decimalSymbol(UNUM_TWO_DIGIT_SYMBOL)); symbols.append(decimalSymbol(UNUM_THREE_DIGIT_SYMBOL)); symbols.append(decimalSymbol(UNUM_FOUR_DIGIT_SYMBOL)); symbols.append(decimalSymbol(UNUM_FIVE_DIGIT_SYMBOL)); symbols.append(decimalSymbol(UNUM_SIX_DIGIT_SYMBOL)); symbols.append(decimalSymbol(UNUM_SEVEN_DIGIT_SYMBOL)); symbols.append(decimalSymbol(UNUM_EIGHT_DIGIT_SYMBOL)); symbols.append(decimalSymbol(UNUM_NINE_DIGIT_SYMBOL)); symbols.append(decimalSymbol(UNUM_DECIMAL_SEPARATOR_SYMBOL)); symbols.append(decimalSymbol(UNUM_GROUPING_SEPARATOR_SYMBOL)); ASSERT(symbols.size() == DecimalSymbolsSize); setLocaleData(symbols, decimalTextAttribute(UNUM_POSITIVE_PREFIX), decimalTextAttribute(UNUM_POSITIVE_SUFFIX), decimalTextAttribute(UNUM_NEGATIVE_PREFIX), decimalTextAttribute(UNUM_NEGATIVE_SUFFIX)); } bool LocaleICU::initializeShortDateFormat() { if (m_didCreateShortDateFormat) return m_shortDateFormat; m_shortDateFormat = openDateFormat(UDAT_NONE, UDAT_SHORT); m_didCreateShortDateFormat = true; return m_shortDateFormat; } UDateFormat* LocaleICU::openDateFormat(UDateFormatStyle timeStyle, UDateFormatStyle dateStyle) const { const UChar gmtTimezone[3] = {'G', 'M', 'T'}; UErrorCode status = U_ZERO_ERROR; return udat_open(timeStyle, dateStyle, m_locale.data(), gmtTimezone, WTF_ARRAY_LENGTH(gmtTimezone), 0, -1, &status); } #if ENABLE(DATE_AND_TIME_INPUT_TYPES) static String getDateFormatPattern(const UDateFormat* dateFormat) { if (!dateFormat) return emptyString(); UErrorCode status = U_ZERO_ERROR; int32_t length = udat_toPattern(dateFormat, TRUE, 0, 0, &status); if (status != U_BUFFER_OVERFLOW_ERROR || !length) return emptyString(); Vector buffer(length); status = U_ZERO_ERROR; udat_toPattern(dateFormat, TRUE, buffer.data(), length, &status); if (U_FAILURE(status)) return emptyString(); return String::adopt(buffer); } std::unique_ptr> LocaleICU::createLabelVector(const UDateFormat* dateFormat, UDateFormatSymbolType type, int32_t startIndex, int32_t size) { if (!dateFormat) return std::make_unique>(); if (udat_countSymbols(dateFormat, type) != startIndex + size) return std::make_unique>(); auto labels = std::make_unique>(); labels->reserveCapacity(size); for (int32_t i = 0; i < size; ++i) { UErrorCode status = U_ZERO_ERROR; int32_t length = udat_getSymbols(dateFormat, type, startIndex + i, 0, 0, &status); if (status != U_BUFFER_OVERFLOW_ERROR) return std::make_unique>(); Vector buffer(length); status = U_ZERO_ERROR; udat_getSymbols(dateFormat, type, startIndex + i, buffer.data(), length, &status); if (U_FAILURE(status)) return std::make_unique>(); labels->append(String::adopt(buffer)); } return WTFMove(labels); } #endif #if ENABLE(DATE_AND_TIME_INPUT_TYPES) static std::unique_ptr> createFallbackMonthLabels() { auto labels = std::make_unique>(); labels->reserveCapacity(WTF_ARRAY_LENGTH(WTF::monthFullName)); for (unsigned i = 0; i < WTF_ARRAY_LENGTH(WTF::monthFullName); ++i) labels->append(WTF::monthFullName[i]); return WTFMove(labels); } const Vector& LocaleICU::monthLabels() { if (m_monthLabels) return *m_monthLabels; if (initializeShortDateFormat()) { m_monthLabels = createLabelVector(m_shortDateFormat, UDAT_MONTHS, UCAL_JANUARY, 12); if (m_monthLabels) return *m_monthLabels; } m_monthLabels = createFallbackMonthLabels(); return *m_monthLabels; } #endif #if ENABLE(DATE_AND_TIME_INPUT_TYPES) static std::unique_ptr> createFallbackAMPMLabels() { auto labels = std::make_unique>(); labels->reserveCapacity(2); labels->append("AM"); labels->append("PM"); return WTFMove(labels); } void LocaleICU::initializeDateTimeFormat() { if (m_didCreateTimeFormat) return; // We assume ICU medium time pattern and short time pattern are compatible // with LDML, because ICU specific pattern character "V" doesn't appear // in both medium and short time pattern. m_mediumTimeFormat = openDateFormat(UDAT_MEDIUM, UDAT_NONE); m_timeFormatWithSeconds = getDateFormatPattern(m_mediumTimeFormat); m_shortTimeFormat = openDateFormat(UDAT_SHORT, UDAT_NONE); m_timeFormatWithoutSeconds = getDateFormatPattern(m_shortTimeFormat); UDateFormat* dateTimeFormatWithSeconds = openDateFormat(UDAT_MEDIUM, UDAT_SHORT); m_dateTimeFormatWithSeconds = getDateFormatPattern(dateTimeFormatWithSeconds); udat_close(dateTimeFormatWithSeconds); UDateFormat* dateTimeFormatWithoutSeconds = openDateFormat(UDAT_SHORT, UDAT_SHORT); m_dateTimeFormatWithoutSeconds = getDateFormatPattern(dateTimeFormatWithoutSeconds); udat_close(dateTimeFormatWithoutSeconds); auto timeAMPMLabels = createLabelVector(m_mediumTimeFormat, UDAT_AM_PMS, UCAL_AM, 2); if (!timeAMPMLabels) timeAMPMLabels = createFallbackAMPMLabels(); m_timeAMPMLabels = *timeAMPMLabels; m_didCreateTimeFormat = true; } String LocaleICU::dateFormat() { if (!m_dateFormat.isNull()) return m_dateFormat; if (!initializeShortDateFormat()) return ASCIILiteral("yyyy-MM-dd"); m_dateFormat = getDateFormatPattern(m_shortDateFormat); return m_dateFormat; } static String getFormatForSkeleton(const char* locale, const UChar* skeleton, int32_t skeletonLength) { String format = ASCIILiteral("yyyy-MM"); UErrorCode status = U_ZERO_ERROR; UDateTimePatternGenerator* patternGenerator = udatpg_open(locale, &status); if (!patternGenerator) return format; status = U_ZERO_ERROR; int32_t length = udatpg_getBestPattern(patternGenerator, skeleton, skeletonLength, 0, 0, &status); if (status == U_BUFFER_OVERFLOW_ERROR && length) { Vector buffer(length); status = U_ZERO_ERROR; udatpg_getBestPattern(patternGenerator, skeleton, skeletonLength, buffer.data(), length, &status); if (U_SUCCESS(status)) format = String::adopt(buffer); } udatpg_close(patternGenerator); return format; } String LocaleICU::monthFormat() { if (!m_monthFormat.isNull()) return m_monthFormat; // Gets a format for "MMMM" because Windows API always provides formats for // "MMMM" in some locales. const UChar skeleton[] = { 'y', 'y', 'y', 'y', 'M', 'M', 'M', 'M' }; m_monthFormat = getFormatForSkeleton(m_locale.data(), skeleton, WTF_ARRAY_LENGTH(skeleton)); return m_monthFormat; } String LocaleICU::shortMonthFormat() { if (!m_shortMonthFormat.isNull()) return m_shortMonthFormat; const UChar skeleton[] = { 'y', 'y', 'y', 'y', 'M', 'M', 'M' }; m_shortMonthFormat = getFormatForSkeleton(m_locale.data(), skeleton, WTF_ARRAY_LENGTH(skeleton)); return m_shortMonthFormat; } String LocaleICU::timeFormat() { initializeDateTimeFormat(); return m_timeFormatWithSeconds; } String LocaleICU::shortTimeFormat() { initializeDateTimeFormat(); return m_timeFormatWithoutSeconds; } String LocaleICU::dateTimeFormatWithSeconds() { initializeDateTimeFormat(); return m_dateTimeFormatWithSeconds; } String LocaleICU::dateTimeFormatWithoutSeconds() { initializeDateTimeFormat(); return m_dateTimeFormatWithoutSeconds; } const Vector& LocaleICU::shortMonthLabels() { if (!m_shortMonthLabels.isEmpty()) return m_shortMonthLabels; if (initializeShortDateFormat()) { if (auto labels = createLabelVector(m_shortDateFormat, UDAT_SHORT_MONTHS, UCAL_JANUARY, 12)) { m_shortMonthLabels = *labels; return m_shortMonthLabels; } } m_shortMonthLabels.reserveCapacity(WTF_ARRAY_LENGTH(WTF::monthName)); for (unsigned i = 0; i < WTF_ARRAY_LENGTH(WTF::monthName); ++i) m_shortMonthLabels.append(WTF::monthName[i]); return m_shortMonthLabels; } const Vector& LocaleICU::standAloneMonthLabels() { if (!m_standAloneMonthLabels.isEmpty()) return m_standAloneMonthLabels; if (initializeShortDateFormat()) { if (auto labels = createLabelVector(m_shortDateFormat, UDAT_STANDALONE_MONTHS, UCAL_JANUARY, 12)) { m_standAloneMonthLabels = *labels; return m_standAloneMonthLabels; } } m_standAloneMonthLabels = monthLabels(); return m_standAloneMonthLabels; } const Vector& LocaleICU::shortStandAloneMonthLabels() { if (!m_shortStandAloneMonthLabels.isEmpty()) return m_shortStandAloneMonthLabels; if (initializeShortDateFormat()) { if (auto labels = createLabelVector(m_shortDateFormat, UDAT_STANDALONE_SHORT_MONTHS, UCAL_JANUARY, 12)) { m_shortStandAloneMonthLabels = *labels; return m_shortStandAloneMonthLabels; } } m_shortStandAloneMonthLabels = shortMonthLabels(); return m_shortStandAloneMonthLabels; } const Vector& LocaleICU::timeAMPMLabels() { initializeDateTimeFormat(); return m_timeAMPMLabels; } #endif } // namespace WebCore