OLD | NEW |
1 // Copyright 2013 the V8 project authors. All rights reserved. | 1 // Copyright 2013 the V8 project authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 // ECMAScript 402 API implementation. | 5 // ECMAScript 402 API implementation. |
6 | 6 |
7 /** | 7 /** |
8 * Intl object is a single object that has some named properties, | 8 * Intl object is a single object that has some named properties, |
9 * all of which are constructors. | 9 * all of which are constructors. |
10 */ | 10 */ |
(...skipping 126 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
137 'numberformat': UNDEFINED, | 137 'numberformat': UNDEFINED, |
138 'dateformat': UNDEFINED, | 138 'dateformat': UNDEFINED, |
139 'breakiterator': UNDEFINED | 139 'breakiterator': UNDEFINED |
140 }; | 140 }; |
141 | 141 |
142 /** | 142 /** |
143 * Caches default ICU locale. | 143 * Caches default ICU locale. |
144 */ | 144 */ |
145 var DEFAULT_ICU_LOCALE = UNDEFINED; | 145 var DEFAULT_ICU_LOCALE = UNDEFINED; |
146 | 146 |
| 147 function GetDefaultICULocaleJS() { |
| 148 if (IS_UNDEFINED(DEFAULT_ICU_LOCALE)) { |
| 149 DEFAULT_ICU_LOCALE = %GetDefaultICULocale(); |
| 150 } |
| 151 return DEFAULT_ICU_LOCALE; |
| 152 } |
| 153 |
147 /** | 154 /** |
148 * Unicode extension regular expression. | 155 * Unicode extension regular expression. |
149 */ | 156 */ |
150 var UNICODE_EXTENSION_RE = UNDEFINED; | 157 var UNICODE_EXTENSION_RE = UNDEFINED; |
151 | 158 |
152 function GetUnicodeExtensionRE() { | 159 function GetUnicodeExtensionRE() { |
153 if (IS_UNDEFINED(UNDEFINED)) { | 160 if (IS_UNDEFINED(UNDEFINED)) { |
154 UNICODE_EXTENSION_RE = new GlobalRegExp('-u(-[a-z0-9]{2,8})+', 'g'); | 161 UNICODE_EXTENSION_RE = new GlobalRegExp('-u(-[a-z0-9]{2,8})+', 'g'); |
155 } | 162 } |
156 return UNICODE_EXTENSION_RE; | 163 return UNICODE_EXTENSION_RE; |
(...skipping 284 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
441 // Truncate locale if possible. | 448 // Truncate locale if possible. |
442 var pos = %_Call(StringLastIndexOf, locale, '-'); | 449 var pos = %_Call(StringLastIndexOf, locale, '-'); |
443 if (pos === -1) { | 450 if (pos === -1) { |
444 break; | 451 break; |
445 } | 452 } |
446 locale = %_Call(StringSubstring, locale, 0, pos); | 453 locale = %_Call(StringSubstring, locale, 0, pos); |
447 } while (true); | 454 } while (true); |
448 } | 455 } |
449 | 456 |
450 // Didn't find a match, return default. | 457 // Didn't find a match, return default. |
451 if (IS_UNDEFINED(DEFAULT_ICU_LOCALE)) { | 458 return {'locale': GetDefaultICULocaleJS(), 'extension': '', 'position': -1}; |
452 DEFAULT_ICU_LOCALE = %GetDefaultICULocale(); | |
453 } | |
454 | |
455 return {'locale': DEFAULT_ICU_LOCALE, 'extension': '', 'position': -1}; | |
456 } | 459 } |
457 | 460 |
458 | 461 |
459 /** | 462 /** |
460 * Returns best matched supported locale and extension info using | 463 * Returns best matched supported locale and extension info using |
461 * implementation dependend algorithm. | 464 * implementation dependend algorithm. |
462 */ | 465 */ |
463 function bestFitMatcher(service, requestedLocales) { | 466 function bestFitMatcher(service, requestedLocales) { |
464 // TODO(cira): implement better best fit algorithm. | 467 // TODO(cira): implement better best fit algorithm. |
465 return lookupMatcher(service, requestedLocales); | 468 return lookupMatcher(service, requestedLocales); |
(...skipping 249 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
715 } | 718 } |
716 } | 719 } |
717 return result; | 720 return result; |
718 } | 721 } |
719 | 722 |
720 /** | 723 /** |
721 * Canonicalizes the language tag, or throws in case the tag is invalid. | 724 * Canonicalizes the language tag, or throws in case the tag is invalid. |
722 */ | 725 */ |
723 function canonicalizeLanguageTag(localeID) { | 726 function canonicalizeLanguageTag(localeID) { |
724 // null is typeof 'object' so we have to do extra check. | 727 // null is typeof 'object' so we have to do extra check. |
725 if (typeof localeID !== 'string' && typeof localeID !== 'object' || | 728 if ((!IS_STRING(localeID) && !IS_RECEIVER(localeID)) || |
726 IS_NULL(localeID)) { | 729 IS_NULL(localeID)) { |
727 throw MakeTypeError(kLanguageID); | 730 throw MakeTypeError(kLanguageID); |
728 } | 731 } |
729 | 732 |
| 733 // Optimize for the most common case; a language code alone in |
| 734 // the canonical form/lowercase (e.g. "en", "fil"). |
| 735 if (IS_STRING(localeID) && |
| 736 !IS_NULL(InternalRegExpMatch(/^[a-z]{2,3}$/, localeID))) |
| 737 return localeID; |
| 738 |
730 var localeString = GlobalString(localeID); | 739 var localeString = GlobalString(localeID); |
731 | 740 |
732 if (isValidLanguageTag(localeString) === false) { | 741 if (isValidLanguageTag(localeString) === false) { |
733 throw MakeRangeError(kInvalidLanguageTag, localeString); | 742 throw MakeRangeError(kInvalidLanguageTag, localeString); |
734 } | 743 } |
735 | 744 |
736 // This call will strip -kn but not -kn-true extensions. | |
737 // ICU bug filled - http://bugs.icu-project.org/trac/ticket/9265. | |
738 // TODO(cira): check if -u-kn-true-kc-true-kh-true still throws after | |
739 // upgrade to ICU 4.9. | |
740 var tag = %CanonicalizeLanguageTag(localeString); | 745 var tag = %CanonicalizeLanguageTag(localeString); |
741 if (tag === 'invalid-tag') { | 746 if (tag === 'invalid-tag') { |
742 throw MakeRangeError(kInvalidLanguageTag, localeString); | 747 throw MakeRangeError(kInvalidLanguageTag, localeString); |
743 } | 748 } |
744 | 749 |
745 return tag; | 750 return tag; |
746 } | 751 } |
747 | 752 |
748 | 753 |
749 /** | 754 /** |
(...skipping 1235 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
1985 var useOptions = (IS_UNDEFINED(defaults)) ? options : defaults; | 1990 var useOptions = (IS_UNDEFINED(defaults)) ? options : defaults; |
1986 if (IS_UNDEFINED(locales) && IS_UNDEFINED(options)) { | 1991 if (IS_UNDEFINED(locales) && IS_UNDEFINED(options)) { |
1987 if (IS_UNDEFINED(defaultObjects[service])) { | 1992 if (IS_UNDEFINED(defaultObjects[service])) { |
1988 defaultObjects[service] = new savedObjects[service](locales, useOptions); | 1993 defaultObjects[service] = new savedObjects[service](locales, useOptions); |
1989 } | 1994 } |
1990 return defaultObjects[service]; | 1995 return defaultObjects[service]; |
1991 } | 1996 } |
1992 return new savedObjects[service](locales, useOptions); | 1997 return new savedObjects[service](locales, useOptions); |
1993 } | 1998 } |
1994 | 1999 |
| 2000 function LocaleConvertCase(s, locales, isToUpper) { |
| 2001 // ECMA 402 section 13.1.2 steps 1 through 12. |
| 2002 var language; |
| 2003 // Optimize for the most common two cases. initializeLocaleList() can handle |
| 2004 // them as well, but it's rather slow accounting for over 60% of |
| 2005 // toLocale{U,L}Case() and about 40% of toLocale{U,L}Case("<locale>"). |
| 2006 if (IS_UNDEFINED(locales)) { |
| 2007 language = GetDefaultICULocaleJS(); |
| 2008 } else if (IS_STRING(locales)) { |
| 2009 language = canonicalizeLanguageTag(locales); |
| 2010 } else { |
| 2011 var locales = initializeLocaleList(locales); |
| 2012 language = locales.length > 0 ? locales[0] : GetDefaultICULocaleJS(); |
| 2013 } |
| 2014 |
| 2015 // StringSplit is slwoer than this. |
| 2016 var pos = %_Call(StringIndexOf, language, '-'); |
| 2017 if (pos != -1) |
| 2018 language = %_Call(StringSubstring, language, 0, pos); |
| 2019 |
| 2020 var CUSTOM_CASE_LANGUAGES = ['az', 'el', 'lt', 'tr']; |
| 2021 var langIndex = %_Call(ArrayIndexOf, CUSTOM_CASE_LANGUAGES, language); |
| 2022 if (langIndex == -1) // language-independent case conversion. |
| 2023 return isToUpper ? %StringToUpperCaseI18N(s) : %StringToLowerCaseI18N(s); |
| 2024 return %StringLocaleConvertCase(s, isToUpper, |
| 2025 CUSTOM_CASE_LANGUAGES[langIndex]); |
| 2026 } |
| 2027 |
1995 /** | 2028 /** |
1996 * Compares this and that, and returns less than 0, 0 or greater than 0 value. | 2029 * Compares this and that, and returns less than 0, 0 or greater than 0 value. |
1997 * Overrides the built-in method. | 2030 * Overrides the built-in method. |
1998 */ | 2031 */ |
1999 OverrideFunction(GlobalString.prototype, 'localeCompare', function(that) { | 2032 OverrideFunction(GlobalString.prototype, 'localeCompare', function(that) { |
2000 if (!IS_UNDEFINED(new.target)) { | 2033 if (!IS_UNDEFINED(new.target)) { |
2001 throw MakeTypeError(kOrdinaryFunctionCalledAsConstructor); | 2034 throw MakeTypeError(kOrdinaryFunctionCalledAsConstructor); |
2002 } | 2035 } |
2003 | 2036 |
2004 if (IS_NULL_OR_UNDEFINED(this)) { | 2037 if (IS_NULL_OR_UNDEFINED(this)) { |
(...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2037 var normalizationForm = %_Call(ArrayIndexOf, NORMALIZATION_FORMS, form); | 2070 var normalizationForm = %_Call(ArrayIndexOf, NORMALIZATION_FORMS, form); |
2038 if (normalizationForm === -1) { | 2071 if (normalizationForm === -1) { |
2039 throw MakeRangeError(kNormalizationForm, | 2072 throw MakeRangeError(kNormalizationForm, |
2040 %_Call(ArrayJoin, NORMALIZATION_FORMS, ', ')); | 2073 %_Call(ArrayJoin, NORMALIZATION_FORMS, ', ')); |
2041 } | 2074 } |
2042 | 2075 |
2043 return %StringNormalize(s, normalizationForm); | 2076 return %StringNormalize(s, normalizationForm); |
2044 } | 2077 } |
2045 ); | 2078 ); |
2046 | 2079 |
| 2080 OverrideFunction(GlobalString.prototype, 'toLowerCase', function() { |
| 2081 if (!IS_UNDEFINED(new.target)) { |
| 2082 throw MakeTypeError(kOrdinaryFunctionCalledAsConstructor); |
| 2083 } |
| 2084 CHECK_OBJECT_COERCIBLE(this, "String.prototype.toLowerCase"); |
| 2085 var s = TO_STRING(this); |
| 2086 return %StringToLowerCaseI18N(s); |
| 2087 } |
| 2088 ); |
| 2089 |
| 2090 OverrideFunction(GlobalString.prototype, 'toUpperCase', function() { |
| 2091 if (!IS_UNDEFINED(new.target)) { |
| 2092 throw MakeTypeError(kOrdinaryFunctionCalledAsConstructor); |
| 2093 } |
| 2094 CHECK_OBJECT_COERCIBLE(this, "String.prototype.toLowerCase"); |
| 2095 var s = TO_STRING(this); |
| 2096 return %StringToUpperCaseI18N(s); |
| 2097 } |
| 2098 ); |
| 2099 |
| 2100 OverrideFunction(GlobalString.prototype, 'toLocaleLowerCase', function() { |
| 2101 if (!IS_UNDEFINED(new.target)) { |
| 2102 throw MakeTypeError(kOrdinaryFunctionCalledAsConstructor); |
| 2103 } |
| 2104 CHECK_OBJECT_COERCIBLE(this, "String.prototype.toLocaleLowerCase"); |
| 2105 return LocaleConvertCase(TO_STRING(this), arguments[0], false); |
| 2106 } |
| 2107 ); |
| 2108 |
| 2109 |
| 2110 OverrideFunction(GlobalString.prototype, 'toLocaleUpperCase', function() { |
| 2111 if (!IS_UNDEFINED(new.target)) { |
| 2112 throw MakeTypeError(kOrdinaryFunctionCalledAsConstructor); |
| 2113 } |
| 2114 CHECK_OBJECT_COERCIBLE(this, "String.prototype.toLocaleUpperCase"); |
| 2115 return LocaleConvertCase(TO_STRING(this), arguments[0], true); |
| 2116 } |
| 2117 ); |
| 2118 |
2047 | 2119 |
2048 /** | 2120 /** |
2049 * Formats a Number object (this) using locale and options values. | 2121 * Formats a Number object (this) using locale and options values. |
2050 * If locale or options are omitted, defaults are used. | 2122 * If locale or options are omitted, defaults are used. |
2051 */ | 2123 */ |
2052 OverrideFunction(GlobalNumber.prototype, 'toLocaleString', function() { | 2124 OverrideFunction(GlobalNumber.prototype, 'toLocaleString', function() { |
2053 if (!IS_UNDEFINED(new.target)) { | 2125 if (!IS_UNDEFINED(new.target)) { |
2054 throw MakeTypeError(kOrdinaryFunctionCalledAsConstructor); | 2126 throw MakeTypeError(kOrdinaryFunctionCalledAsConstructor); |
2055 } | 2127 } |
2056 | 2128 |
(...skipping 75 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2132 } | 2204 } |
2133 | 2205 |
2134 var locales = arguments[0]; | 2206 var locales = arguments[0]; |
2135 var options = arguments[1]; | 2207 var options = arguments[1]; |
2136 return toLocaleDateTime( | 2208 return toLocaleDateTime( |
2137 this, locales, options, 'time', 'time', 'dateformattime'); | 2209 this, locales, options, 'time', 'time', 'dateformattime'); |
2138 } | 2210 } |
2139 ); | 2211 ); |
2140 | 2212 |
2141 }) | 2213 }) |
OLD | NEW |