summaryrefslogtreecommitdiff
path: root/Source/WebCore/platform/text/LocaleICU.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'Source/WebCore/platform/text/LocaleICU.cpp')
-rw-r--r--Source/WebCore/platform/text/LocaleICU.cpp376
1 files changed, 376 insertions, 0 deletions
diff --git a/Source/WebCore/platform/text/LocaleICU.cpp b/Source/WebCore/platform/text/LocaleICU.cpp
new file mode 100644
index 000000000..88151ea78
--- /dev/null
+++ b/Source/WebCore/platform/text/LocaleICU.cpp
@@ -0,0 +1,376 @@
+/*
+ * 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 <limits>
+#include <unicode/udatpg.h>
+#include <unicode/uloc.h>
+#include <wtf/DateMath.h>
+#include <wtf/text/StringBuilder.h>
+
+using namespace icu;
+using namespace std;
+
+namespace WebCore {
+
+std::unique_ptr<Locale> Locale::create(const AtomicString& locale)
+{
+ return std::make_unique<LocaleICU>(locale.string().utf8().data());
+}
+
+LocaleICU::LocaleICU(const char* locale)
+ : m_locale(locale)
+{
+}
+
+LocaleICU::~LocaleICU()
+{
+#if !UCONFIG_NO_FORMATTING
+ unum_close(m_numberFormat);
+#endif
+#if ENABLE(DATE_AND_TIME_INPUT_TYPES)
+ udat_close(m_shortDateFormat);
+ udat_close(m_mediumTimeFormat);
+ udat_close(m_shortTimeFormat);
+#endif
+}
+
+#if !UCONFIG_NO_FORMATTING
+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<UChar> buffer(bufferLength);
+ status = U_ZERO_ERROR;
+ unum_getSymbol(m_numberFormat, symbol, buffer.data(), bufferLength, &status);
+ if (U_FAILURE(status))
+ return String();
+ return String::adopt(WTFMove(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<UChar> 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(WTFMove(buffer));
+}
+#endif
+
+void LocaleICU::initializeLocaleData()
+{
+#if !UCONFIG_NO_FORMATTING
+ 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<String, DecimalSymbolsSize> 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));
+#endif
+}
+
+#if ENABLE(DATE_AND_TIME_INPUT_TYPES)
+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);
+}
+
+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<UChar> buffer(length);
+ status = U_ZERO_ERROR;
+ udat_toPattern(dateFormat, TRUE, buffer.data(), length, &status);
+ if (U_FAILURE(status))
+ return emptyString();
+ return String::adopt(WTFMove(buffer));
+}
+
+std::unique_ptr<Vector<String>> LocaleICU::createLabelVector(const UDateFormat* dateFormat, UDateFormatSymbolType type, int32_t startIndex, int32_t size)
+{
+ if (!dateFormat)
+ return std::make_unique<Vector<String>>();
+ if (udat_countSymbols(dateFormat, type) != startIndex + size)
+ return std::make_unique<Vector<String>>();
+
+ auto labels = std::make_unique<Vector<String>>();
+ 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<String>>();
+ Vector<UChar> buffer(length);
+ status = U_ZERO_ERROR;
+ udat_getSymbols(dateFormat, type, startIndex + i, buffer.data(), length, &status);
+ if (U_FAILURE(status))
+ return std::make_unique<Vector<String>>();
+ labels->append(String::adopt(WTFMove(buffer)));
+ }
+ return WTFMove(labels);
+}
+
+static std::unique_ptr<Vector<String>> createFallbackMonthLabels()
+{
+ auto labels = std::make_unique<Vector<String>>();
+ 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<String>& 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;
+}
+
+static std::unique_ptr<Vector<String>> createFallbackAMPMLabels()
+{
+ auto labels = std::make_unique<Vector<String>>();
+ 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<UChar> buffer(length);
+ status = U_ZERO_ERROR;
+ udatpg_getBestPattern(patternGenerator, skeleton, skeletonLength, buffer.data(), length, &status);
+ if (U_SUCCESS(status))
+ format = String::adopt(WTFMove(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<String>& 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<String>& 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<String>& 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<String>& LocaleICU::timeAMPMLabels()
+{
+ initializeDateTimeFormat();
+ return m_timeAMPMLabels;
+}
+
+#endif
+
+} // namespace WebCore
+