зеркало из https://github.com/mozilla/gecko-dev.git
Bug 1431957 - Move Intl.Collator self-hosted code to a new builtin/intl/Collator.js file. r=anba
--HG-- rename : js/src/builtin/Intl.js => js/src/builtin/intl/Collator.js extra : rebase_source : 18f48d145d9eef44c491253356b4450811a1d7c4
This commit is contained in:
Родитель
57c5806aa2
Коммит
e35f51daf6
|
@ -1512,437 +1512,3 @@ function getInternals(obj) {
|
|||
setInternalProperties(internals, internalProps);
|
||||
return internalProps;
|
||||
}
|
||||
|
||||
|
||||
/********** Intl.Collator **********/
|
||||
|
||||
|
||||
/**
|
||||
* Mapping from Unicode extension keys for collation to options properties,
|
||||
* their types and permissible values.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 10.1.1.
|
||||
*/
|
||||
var collatorKeyMappings = {
|
||||
kn: {property: "numeric", type: "boolean"},
|
||||
kf: {property: "caseFirst", type: "string", values: ["upper", "lower", "false"]}
|
||||
};
|
||||
|
||||
|
||||
/**
|
||||
* Compute an internal properties object from |lazyCollatorData|.
|
||||
*/
|
||||
function resolveCollatorInternals(lazyCollatorData) {
|
||||
assert(IsObject(lazyCollatorData), "lazy data not an object?");
|
||||
|
||||
var internalProps = std_Object_create(null);
|
||||
|
||||
// Step 7.
|
||||
internalProps.usage = lazyCollatorData.usage;
|
||||
|
||||
// Step 8.
|
||||
var Collator = collatorInternalProperties;
|
||||
|
||||
// Step 9.
|
||||
var collatorIsSorting = lazyCollatorData.usage === "sort";
|
||||
var localeData = collatorIsSorting
|
||||
? Collator.sortLocaleData
|
||||
: Collator.searchLocaleData;
|
||||
|
||||
// Compute effective locale.
|
||||
// Step 14.
|
||||
var relevantExtensionKeys = Collator.relevantExtensionKeys;
|
||||
|
||||
// Step 15.
|
||||
var r = ResolveLocale(callFunction(Collator.availableLocales, Collator),
|
||||
lazyCollatorData.requestedLocales,
|
||||
lazyCollatorData.opt,
|
||||
relevantExtensionKeys,
|
||||
localeData);
|
||||
|
||||
// Step 16.
|
||||
internalProps.locale = r.locale;
|
||||
|
||||
// Steps 17-19.
|
||||
var key, property, value, mapping;
|
||||
var i = 0, len = relevantExtensionKeys.length;
|
||||
while (i < len) {
|
||||
// Step 19.a.
|
||||
key = relevantExtensionKeys[i];
|
||||
if (key === "co") {
|
||||
// Step 19.b.
|
||||
property = "collation";
|
||||
value = r.co === null ? "default" : r.co;
|
||||
} else {
|
||||
// Step 19.c.
|
||||
mapping = collatorKeyMappings[key];
|
||||
property = mapping.property;
|
||||
value = r[key];
|
||||
if (mapping.type === "boolean")
|
||||
value = value === "true";
|
||||
}
|
||||
|
||||
// Step 19.d.
|
||||
internalProps[property] = value;
|
||||
|
||||
// Step 19.e.
|
||||
i++;
|
||||
}
|
||||
|
||||
// Compute remaining collation options.
|
||||
// Steps 21-22.
|
||||
var s = lazyCollatorData.rawSensitivity;
|
||||
if (s === undefined) {
|
||||
// In theory the default sensitivity for the "search" collator is
|
||||
// locale dependent; in reality the CLDR/ICU default strength is
|
||||
// always tertiary. Therefore use "variant" as the default value for
|
||||
// both collation modes.
|
||||
s = "variant";
|
||||
}
|
||||
internalProps.sensitivity = s;
|
||||
|
||||
// Step 24.
|
||||
internalProps.ignorePunctuation = lazyCollatorData.ignorePunctuation;
|
||||
|
||||
// Step 25.
|
||||
internalProps.boundFormat = undefined;
|
||||
|
||||
// The caller is responsible for associating |internalProps| with the right
|
||||
// object using |setInternalProperties|.
|
||||
return internalProps;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Returns an object containing the Collator internal properties of |obj|.
|
||||
*/
|
||||
function getCollatorInternals(obj) {
|
||||
assert(IsObject(obj), "getCollatorInternals called with non-object");
|
||||
assert(IsCollator(obj), "getCollatorInternals called with non-Collator");
|
||||
|
||||
var internals = getIntlObjectInternals(obj);
|
||||
assert(internals.type === "Collator", "bad type escaped getIntlObjectInternals");
|
||||
|
||||
// If internal properties have already been computed, use them.
|
||||
var internalProps = maybeInternalProperties(internals);
|
||||
if (internalProps)
|
||||
return internalProps;
|
||||
|
||||
// Otherwise it's time to fully create them.
|
||||
internalProps = resolveCollatorInternals(internals.lazyData);
|
||||
setInternalProperties(internals, internalProps);
|
||||
return internalProps;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Initializes an object as a Collator.
|
||||
*
|
||||
* This method is complicated a moderate bit by its implementing initialization
|
||||
* as a *lazy* concept. Everything that must happen now, does -- but we defer
|
||||
* all the work we can until the object is actually used as a Collator. This
|
||||
* later work occurs in |resolveCollatorInternals|; steps not noted here occur
|
||||
* there.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 10.1.1.
|
||||
*/
|
||||
function InitializeCollator(collator, locales, options) {
|
||||
assert(IsObject(collator), "InitializeCollator called with non-object");
|
||||
assert(IsCollator(collator), "InitializeCollator called with non-Collator");
|
||||
|
||||
// Steps 1-2 (These steps are no longer required and should be removed
|
||||
// from the spec; https://github.com/tc39/ecma402/issues/115).
|
||||
|
||||
// Lazy Collator data has the following structure:
|
||||
//
|
||||
// {
|
||||
// requestedLocales: List of locales,
|
||||
// usage: "sort" / "search",
|
||||
// opt: // opt object computed in InitializeCollator
|
||||
// {
|
||||
// localeMatcher: "lookup" / "best fit",
|
||||
// kn: true / false / undefined,
|
||||
// kf: "upper" / "lower" / "false" / undefined
|
||||
// }
|
||||
// rawSensitivity: "base" / "accent" / "case" / "variant" / undefined,
|
||||
// ignorePunctuation: true / false
|
||||
// }
|
||||
//
|
||||
// Note that lazy data is only installed as a final step of initialization,
|
||||
// so every Collator lazy data object has *all* these properties, never a
|
||||
// subset of them.
|
||||
var lazyCollatorData = std_Object_create(null);
|
||||
|
||||
// Step 3.
|
||||
var requestedLocales = CanonicalizeLocaleList(locales);
|
||||
lazyCollatorData.requestedLocales = requestedLocales;
|
||||
|
||||
// Steps 4-5.
|
||||
//
|
||||
// If we ever need more speed here at startup, we should try to detect the
|
||||
// case where |options === undefined| and then directly use the default
|
||||
// value for each option. For now, just keep it simple.
|
||||
if (options === undefined)
|
||||
options = std_Object_create(null);
|
||||
else
|
||||
options = ToObject(options);
|
||||
|
||||
// Compute options that impact interpretation of locale.
|
||||
// Step 6.
|
||||
var u = GetOption(options, "usage", "string", ["sort", "search"], "sort");
|
||||
lazyCollatorData.usage = u;
|
||||
|
||||
// Step 10.
|
||||
var opt = new Record();
|
||||
lazyCollatorData.opt = opt;
|
||||
|
||||
// Steps 11-12.
|
||||
var matcher = GetOption(options, "localeMatcher", "string", ["lookup", "best fit"], "best fit");
|
||||
opt.localeMatcher = matcher;
|
||||
|
||||
// Step 13, unrolled.
|
||||
var numericValue = GetOption(options, "numeric", "boolean", undefined, undefined);
|
||||
if (numericValue !== undefined)
|
||||
numericValue = numericValue ? "true" : "false";
|
||||
opt.kn = numericValue;
|
||||
|
||||
var caseFirstValue = GetOption(options, "caseFirst", "string", ["upper", "lower", "false"], undefined);
|
||||
opt.kf = caseFirstValue;
|
||||
|
||||
// Compute remaining collation options.
|
||||
// Step 20.
|
||||
var s = GetOption(options, "sensitivity", "string",
|
||||
["base", "accent", "case", "variant"], undefined);
|
||||
lazyCollatorData.rawSensitivity = s;
|
||||
|
||||
// Step 23.
|
||||
var ip = GetOption(options, "ignorePunctuation", "boolean", undefined, false);
|
||||
lazyCollatorData.ignorePunctuation = ip;
|
||||
|
||||
// Step 26.
|
||||
//
|
||||
// We've done everything that must be done now: mark the lazy data as fully
|
||||
// computed and install it.
|
||||
initializeIntlObject(collator, "Collator", lazyCollatorData);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Returns the subset of the given locale list for which this locale list has a
|
||||
* matching (possibly fallback) locale. Locales appear in the same order in the
|
||||
* returned list as in the input list.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 10.2.2.
|
||||
*/
|
||||
function Intl_Collator_supportedLocalesOf(locales /*, options*/) {
|
||||
var options = arguments.length > 1 ? arguments[1] : undefined;
|
||||
|
||||
var availableLocales = callFunction(collatorInternalProperties.availableLocales,
|
||||
collatorInternalProperties);
|
||||
var requestedLocales = CanonicalizeLocaleList(locales);
|
||||
return SupportedLocales(availableLocales, requestedLocales, options);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Collator internal properties.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 9.1 and 10.2.3.
|
||||
*/
|
||||
var collatorInternalProperties = {
|
||||
sortLocaleData: collatorSortLocaleData,
|
||||
searchLocaleData: collatorSearchLocaleData,
|
||||
_availableLocales: null,
|
||||
availableLocales: function() // eslint-disable-line object-shorthand
|
||||
{
|
||||
var locales = this._availableLocales;
|
||||
if (locales)
|
||||
return locales;
|
||||
|
||||
locales = intl_Collator_availableLocales();
|
||||
addSpecialMissingLanguageTags(locales);
|
||||
return (this._availableLocales = locales);
|
||||
},
|
||||
relevantExtensionKeys: ["co", "kn", "kf"]
|
||||
};
|
||||
|
||||
|
||||
/**
|
||||
* Returns the actual locale used when a collator for |locale| is constructed.
|
||||
*/
|
||||
function collatorActualLocale(locale) {
|
||||
assert(typeof locale === "string", "locale should be string");
|
||||
|
||||
// If |locale| is the default locale (e.g. da-DK), but only supported
|
||||
// through a fallback (da), we need to get the actual locale before we
|
||||
// can call intl_isUpperCaseFirst. Also see BestAvailableLocaleHelper.
|
||||
var availableLocales = callFunction(collatorInternalProperties.availableLocales,
|
||||
collatorInternalProperties);
|
||||
return BestAvailableLocaleIgnoringDefault(availableLocales, locale);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Returns the default caseFirst values for the given locale. The first
|
||||
* element in the returned array denotes the default value per ES2017 Intl,
|
||||
* 9.1 Internal slots of Service Constructors.
|
||||
*/
|
||||
function collatorSortCaseFirst(locale) {
|
||||
var actualLocale = collatorActualLocale(locale);
|
||||
if (intl_isUpperCaseFirst(actualLocale))
|
||||
return ["upper", "false", "lower"];
|
||||
|
||||
// Default caseFirst values for all other languages.
|
||||
return ["false", "lower", "upper"];
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Returns the default caseFirst value for the given locale.
|
||||
*/
|
||||
function collatorSortCaseFirstDefault(locale) {
|
||||
var actualLocale = collatorActualLocale(locale);
|
||||
if (intl_isUpperCaseFirst(actualLocale))
|
||||
return "upper";
|
||||
|
||||
// Default caseFirst value for all other languages.
|
||||
return "false";
|
||||
}
|
||||
|
||||
|
||||
function collatorSortLocaleData() {
|
||||
/* eslint-disable object-shorthand */
|
||||
return {
|
||||
co: intl_availableCollations,
|
||||
kn: function() {
|
||||
return ["false", "true"];
|
||||
},
|
||||
kf: collatorSortCaseFirst,
|
||||
default: {
|
||||
co: function() {
|
||||
// The first element of the collations array must be |null|
|
||||
// per ES2017 Intl, 10.2.3 Internal Slots.
|
||||
return null;
|
||||
},
|
||||
kn: function() {
|
||||
return "false";
|
||||
},
|
||||
kf: collatorSortCaseFirstDefault,
|
||||
}
|
||||
};
|
||||
/* eslint-enable object-shorthand */
|
||||
}
|
||||
|
||||
|
||||
function collatorSearchLocaleData() {
|
||||
/* eslint-disable object-shorthand */
|
||||
return {
|
||||
co: function() {
|
||||
return [null];
|
||||
},
|
||||
kn: function() {
|
||||
return ["false", "true"];
|
||||
},
|
||||
kf: function() {
|
||||
return ["false", "lower", "upper"];
|
||||
},
|
||||
default: {
|
||||
co: function() {
|
||||
return null;
|
||||
},
|
||||
kn: function() {
|
||||
return "false";
|
||||
},
|
||||
kf: function() {
|
||||
return "false";
|
||||
},
|
||||
}
|
||||
};
|
||||
/* eslint-enable object-shorthand */
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Function to be bound and returned by Intl.Collator.prototype.format.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 12.3.2.
|
||||
*/
|
||||
function collatorCompareToBind(x, y) {
|
||||
// Step 1.
|
||||
var collator = this;
|
||||
|
||||
// Step 2.
|
||||
assert(IsObject(collator), "collatorCompareToBind called with non-object");
|
||||
assert(IsCollator(collator), "collatorCompareToBind called with non-Collator");
|
||||
|
||||
// Steps 3-6
|
||||
var X = ToString(x);
|
||||
var Y = ToString(y);
|
||||
|
||||
// Step 7.
|
||||
return intl_CompareStrings(collator, X, Y);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Returns a function bound to this Collator that compares x (converted to a
|
||||
* String value) and y (converted to a String value),
|
||||
* and returns a number less than 0 if x < y, 0 if x = y, or a number greater
|
||||
* than 0 if x > y according to the sort order for the locale and collation
|
||||
* options of this Collator object.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 10.3.2.
|
||||
*/
|
||||
function Intl_Collator_compare_get() {
|
||||
// Step 1.
|
||||
var collator = this;
|
||||
|
||||
// Steps 2-3.
|
||||
if (!IsObject(collator) || !IsCollator(collator))
|
||||
ThrowTypeError(JSMSG_INTL_OBJECT_NOT_INITED, "Collator", "compare", "Collator");
|
||||
|
||||
var internals = getCollatorInternals(collator);
|
||||
|
||||
// Step 4.
|
||||
if (internals.boundCompare === undefined) {
|
||||
// Step 4.a.
|
||||
var F = collatorCompareToBind;
|
||||
|
||||
// Steps 4.b-d.
|
||||
var bc = callFunction(FunctionBind, F, collator);
|
||||
internals.boundCompare = bc;
|
||||
}
|
||||
|
||||
// Step 5.
|
||||
return internals.boundCompare;
|
||||
}
|
||||
_SetCanonicalName(Intl_Collator_compare_get, "get compare");
|
||||
|
||||
|
||||
/**
|
||||
* Returns the resolved options for a Collator object.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 10.3.3 and 10.4.
|
||||
*/
|
||||
function Intl_Collator_resolvedOptions() {
|
||||
// Check "this Collator object" per introduction of section 10.3.
|
||||
if (!IsObject(this) || !IsCollator(this))
|
||||
ThrowTypeError(JSMSG_INTL_OBJECT_NOT_INITED, "Collator", "resolvedOptions", "Collator");
|
||||
|
||||
var internals = getCollatorInternals(this);
|
||||
|
||||
var result = {
|
||||
locale: internals.locale,
|
||||
usage: internals.usage,
|
||||
sensitivity: internals.sensitivity,
|
||||
ignorePunctuation: internals.ignorePunctuation
|
||||
};
|
||||
|
||||
var relevantExtensionKeys = collatorInternalProperties.relevantExtensionKeys;
|
||||
for (var i = 0; i < relevantExtensionKeys.length; i++) {
|
||||
var key = relevantExtensionKeys[i];
|
||||
var property = (key === "co") ? "collation" : collatorKeyMappings[key].property;
|
||||
_DefineDataProperty(result, property, internals[property]);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
|
|
|
@ -0,0 +1,422 @@
|
|||
/* This Source Code Form is subject to the terms of the Mozilla Public
|
||||
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
||||
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
||||
|
||||
/* Portions Copyright Norbert Lindenberg 2011-2012. */
|
||||
|
||||
/**
|
||||
* Mapping from Unicode extension keys for collation to options properties,
|
||||
* their types and permissible values.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 10.1.1.
|
||||
*/
|
||||
var collatorKeyMappings = {
|
||||
kn: {property: "numeric", type: "boolean"},
|
||||
kf: {property: "caseFirst", type: "string", values: ["upper", "lower", "false"]}
|
||||
};
|
||||
|
||||
/**
|
||||
* Compute an internal properties object from |lazyCollatorData|.
|
||||
*/
|
||||
function resolveCollatorInternals(lazyCollatorData) {
|
||||
assert(IsObject(lazyCollatorData), "lazy data not an object?");
|
||||
|
||||
var internalProps = std_Object_create(null);
|
||||
|
||||
// Step 7.
|
||||
internalProps.usage = lazyCollatorData.usage;
|
||||
|
||||
// Step 8.
|
||||
var Collator = collatorInternalProperties;
|
||||
|
||||
// Step 9.
|
||||
var collatorIsSorting = lazyCollatorData.usage === "sort";
|
||||
var localeData = collatorIsSorting
|
||||
? Collator.sortLocaleData
|
||||
: Collator.searchLocaleData;
|
||||
|
||||
// Compute effective locale.
|
||||
// Step 14.
|
||||
var relevantExtensionKeys = Collator.relevantExtensionKeys;
|
||||
|
||||
// Step 15.
|
||||
var r = ResolveLocale(callFunction(Collator.availableLocales, Collator),
|
||||
lazyCollatorData.requestedLocales,
|
||||
lazyCollatorData.opt,
|
||||
relevantExtensionKeys,
|
||||
localeData);
|
||||
|
||||
// Step 16.
|
||||
internalProps.locale = r.locale;
|
||||
|
||||
// Steps 17-19.
|
||||
var key, property, value, mapping;
|
||||
var i = 0, len = relevantExtensionKeys.length;
|
||||
while (i < len) {
|
||||
// Step 19.a.
|
||||
key = relevantExtensionKeys[i];
|
||||
if (key === "co") {
|
||||
// Step 19.b.
|
||||
property = "collation";
|
||||
value = r.co === null ? "default" : r.co;
|
||||
} else {
|
||||
// Step 19.c.
|
||||
mapping = collatorKeyMappings[key];
|
||||
property = mapping.property;
|
||||
value = r[key];
|
||||
if (mapping.type === "boolean")
|
||||
value = value === "true";
|
||||
}
|
||||
|
||||
// Step 19.d.
|
||||
internalProps[property] = value;
|
||||
|
||||
// Step 19.e.
|
||||
i++;
|
||||
}
|
||||
|
||||
// Compute remaining collation options.
|
||||
// Steps 21-22.
|
||||
var s = lazyCollatorData.rawSensitivity;
|
||||
if (s === undefined) {
|
||||
// In theory the default sensitivity for the "search" collator is
|
||||
// locale dependent; in reality the CLDR/ICU default strength is
|
||||
// always tertiary. Therefore use "variant" as the default value for
|
||||
// both collation modes.
|
||||
s = "variant";
|
||||
}
|
||||
internalProps.sensitivity = s;
|
||||
|
||||
// Step 24.
|
||||
internalProps.ignorePunctuation = lazyCollatorData.ignorePunctuation;
|
||||
|
||||
// Step 25.
|
||||
internalProps.boundFormat = undefined;
|
||||
|
||||
// The caller is responsible for associating |internalProps| with the right
|
||||
// object using |setInternalProperties|.
|
||||
return internalProps;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns an object containing the Collator internal properties of |obj|.
|
||||
*/
|
||||
function getCollatorInternals(obj) {
|
||||
assert(IsObject(obj), "getCollatorInternals called with non-object");
|
||||
assert(IsCollator(obj), "getCollatorInternals called with non-Collator");
|
||||
|
||||
var internals = getIntlObjectInternals(obj);
|
||||
assert(internals.type === "Collator", "bad type escaped getIntlObjectInternals");
|
||||
|
||||
// If internal properties have already been computed, use them.
|
||||
var internalProps = maybeInternalProperties(internals);
|
||||
if (internalProps)
|
||||
return internalProps;
|
||||
|
||||
// Otherwise it's time to fully create them.
|
||||
internalProps = resolveCollatorInternals(internals.lazyData);
|
||||
setInternalProperties(internals, internalProps);
|
||||
return internalProps;
|
||||
}
|
||||
|
||||
/**
|
||||
* Initializes an object as a Collator.
|
||||
*
|
||||
* This method is complicated a moderate bit by its implementing initialization
|
||||
* as a *lazy* concept. Everything that must happen now, does -- but we defer
|
||||
* all the work we can until the object is actually used as a Collator. This
|
||||
* later work occurs in |resolveCollatorInternals|; steps not noted here occur
|
||||
* there.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 10.1.1.
|
||||
*/
|
||||
function InitializeCollator(collator, locales, options) {
|
||||
assert(IsObject(collator), "InitializeCollator called with non-object");
|
||||
assert(IsCollator(collator), "InitializeCollator called with non-Collator");
|
||||
|
||||
// Steps 1-2 (These steps are no longer required and should be removed
|
||||
// from the spec; https://github.com/tc39/ecma402/issues/115).
|
||||
|
||||
// Lazy Collator data has the following structure:
|
||||
//
|
||||
// {
|
||||
// requestedLocales: List of locales,
|
||||
// usage: "sort" / "search",
|
||||
// opt: // opt object computed in InitializeCollator
|
||||
// {
|
||||
// localeMatcher: "lookup" / "best fit",
|
||||
// kn: true / false / undefined,
|
||||
// kf: "upper" / "lower" / "false" / undefined
|
||||
// }
|
||||
// rawSensitivity: "base" / "accent" / "case" / "variant" / undefined,
|
||||
// ignorePunctuation: true / false
|
||||
// }
|
||||
//
|
||||
// Note that lazy data is only installed as a final step of initialization,
|
||||
// so every Collator lazy data object has *all* these properties, never a
|
||||
// subset of them.
|
||||
var lazyCollatorData = std_Object_create(null);
|
||||
|
||||
// Step 3.
|
||||
var requestedLocales = CanonicalizeLocaleList(locales);
|
||||
lazyCollatorData.requestedLocales = requestedLocales;
|
||||
|
||||
// Steps 4-5.
|
||||
//
|
||||
// If we ever need more speed here at startup, we should try to detect the
|
||||
// case where |options === undefined| and then directly use the default
|
||||
// value for each option. For now, just keep it simple.
|
||||
if (options === undefined)
|
||||
options = std_Object_create(null);
|
||||
else
|
||||
options = ToObject(options);
|
||||
|
||||
// Compute options that impact interpretation of locale.
|
||||
// Step 6.
|
||||
var u = GetOption(options, "usage", "string", ["sort", "search"], "sort");
|
||||
lazyCollatorData.usage = u;
|
||||
|
||||
// Step 10.
|
||||
var opt = new Record();
|
||||
lazyCollatorData.opt = opt;
|
||||
|
||||
// Steps 11-12.
|
||||
var matcher = GetOption(options, "localeMatcher", "string", ["lookup", "best fit"], "best fit");
|
||||
opt.localeMatcher = matcher;
|
||||
|
||||
// Step 13, unrolled.
|
||||
var numericValue = GetOption(options, "numeric", "boolean", undefined, undefined);
|
||||
if (numericValue !== undefined)
|
||||
numericValue = numericValue ? "true" : "false";
|
||||
opt.kn = numericValue;
|
||||
|
||||
var caseFirstValue = GetOption(options, "caseFirst", "string", ["upper", "lower", "false"], undefined);
|
||||
opt.kf = caseFirstValue;
|
||||
|
||||
// Compute remaining collation options.
|
||||
// Step 20.
|
||||
var s = GetOption(options, "sensitivity", "string",
|
||||
["base", "accent", "case", "variant"], undefined);
|
||||
lazyCollatorData.rawSensitivity = s;
|
||||
|
||||
// Step 23.
|
||||
var ip = GetOption(options, "ignorePunctuation", "boolean", undefined, false);
|
||||
lazyCollatorData.ignorePunctuation = ip;
|
||||
|
||||
// Step 26.
|
||||
//
|
||||
// We've done everything that must be done now: mark the lazy data as fully
|
||||
// computed and install it.
|
||||
initializeIntlObject(collator, "Collator", lazyCollatorData);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the subset of the given locale list for which this locale list has a
|
||||
* matching (possibly fallback) locale. Locales appear in the same order in the
|
||||
* returned list as in the input list.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 10.2.2.
|
||||
*/
|
||||
function Intl_Collator_supportedLocalesOf(locales /*, options*/) {
|
||||
var options = arguments.length > 1 ? arguments[1] : undefined;
|
||||
|
||||
var availableLocales = callFunction(collatorInternalProperties.availableLocales,
|
||||
collatorInternalProperties);
|
||||
var requestedLocales = CanonicalizeLocaleList(locales);
|
||||
return SupportedLocales(availableLocales, requestedLocales, options);
|
||||
}
|
||||
|
||||
/**
|
||||
* Collator internal properties.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 9.1 and 10.2.3.
|
||||
*/
|
||||
var collatorInternalProperties = {
|
||||
sortLocaleData: collatorSortLocaleData,
|
||||
searchLocaleData: collatorSearchLocaleData,
|
||||
_availableLocales: null,
|
||||
availableLocales: function() // eslint-disable-line object-shorthand
|
||||
{
|
||||
var locales = this._availableLocales;
|
||||
if (locales)
|
||||
return locales;
|
||||
|
||||
locales = intl_Collator_availableLocales();
|
||||
addSpecialMissingLanguageTags(locales);
|
||||
return (this._availableLocales = locales);
|
||||
},
|
||||
relevantExtensionKeys: ["co", "kn", "kf"]
|
||||
};
|
||||
|
||||
/**
|
||||
* Returns the actual locale used when a collator for |locale| is constructed.
|
||||
*/
|
||||
function collatorActualLocale(locale) {
|
||||
assert(typeof locale === "string", "locale should be string");
|
||||
|
||||
// If |locale| is the default locale (e.g. da-DK), but only supported
|
||||
// through a fallback (da), we need to get the actual locale before we
|
||||
// can call intl_isUpperCaseFirst. Also see BestAvailableLocaleHelper.
|
||||
var availableLocales = callFunction(collatorInternalProperties.availableLocales,
|
||||
collatorInternalProperties);
|
||||
return BestAvailableLocaleIgnoringDefault(availableLocales, locale);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the default caseFirst values for the given locale. The first
|
||||
* element in the returned array denotes the default value per ES2017 Intl,
|
||||
* 9.1 Internal slots of Service Constructors.
|
||||
*/
|
||||
function collatorSortCaseFirst(locale) {
|
||||
var actualLocale = collatorActualLocale(locale);
|
||||
if (intl_isUpperCaseFirst(actualLocale))
|
||||
return ["upper", "false", "lower"];
|
||||
|
||||
// Default caseFirst values for all other languages.
|
||||
return ["false", "lower", "upper"];
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the default caseFirst value for the given locale.
|
||||
*/
|
||||
function collatorSortCaseFirstDefault(locale) {
|
||||
var actualLocale = collatorActualLocale(locale);
|
||||
if (intl_isUpperCaseFirst(actualLocale))
|
||||
return "upper";
|
||||
|
||||
// Default caseFirst value for all other languages.
|
||||
return "false";
|
||||
}
|
||||
|
||||
function collatorSortLocaleData() {
|
||||
/* eslint-disable object-shorthand */
|
||||
return {
|
||||
co: intl_availableCollations,
|
||||
kn: function() {
|
||||
return ["false", "true"];
|
||||
},
|
||||
kf: collatorSortCaseFirst,
|
||||
default: {
|
||||
co: function() {
|
||||
// The first element of the collations array must be |null|
|
||||
// per ES2017 Intl, 10.2.3 Internal Slots.
|
||||
return null;
|
||||
},
|
||||
kn: function() {
|
||||
return "false";
|
||||
},
|
||||
kf: collatorSortCaseFirstDefault,
|
||||
}
|
||||
};
|
||||
/* eslint-enable object-shorthand */
|
||||
}
|
||||
|
||||
function collatorSearchLocaleData() {
|
||||
/* eslint-disable object-shorthand */
|
||||
return {
|
||||
co: function() {
|
||||
return [null];
|
||||
},
|
||||
kn: function() {
|
||||
return ["false", "true"];
|
||||
},
|
||||
kf: function() {
|
||||
return ["false", "lower", "upper"];
|
||||
},
|
||||
default: {
|
||||
co: function() {
|
||||
return null;
|
||||
},
|
||||
kn: function() {
|
||||
return "false";
|
||||
},
|
||||
kf: function() {
|
||||
return "false";
|
||||
},
|
||||
}
|
||||
};
|
||||
/* eslint-enable object-shorthand */
|
||||
}
|
||||
|
||||
/**
|
||||
* Function to be bound and returned by Intl.Collator.prototype.format.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 12.3.2.
|
||||
*/
|
||||
function collatorCompareToBind(x, y) {
|
||||
// Step 1.
|
||||
var collator = this;
|
||||
|
||||
// Step 2.
|
||||
assert(IsObject(collator), "collatorCompareToBind called with non-object");
|
||||
assert(IsCollator(collator), "collatorCompareToBind called with non-Collator");
|
||||
|
||||
// Steps 3-6
|
||||
var X = ToString(x);
|
||||
var Y = ToString(y);
|
||||
|
||||
// Step 7.
|
||||
return intl_CompareStrings(collator, X, Y);
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a function bound to this Collator that compares x (converted to a
|
||||
* String value) and y (converted to a String value),
|
||||
* and returns a number less than 0 if x < y, 0 if x = y, or a number greater
|
||||
* than 0 if x > y according to the sort order for the locale and collation
|
||||
* options of this Collator object.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 10.3.2.
|
||||
*/
|
||||
function Intl_Collator_compare_get() {
|
||||
// Step 1.
|
||||
var collator = this;
|
||||
|
||||
// Steps 2-3.
|
||||
if (!IsObject(collator) || !IsCollator(collator))
|
||||
ThrowTypeError(JSMSG_INTL_OBJECT_NOT_INITED, "Collator", "compare", "Collator");
|
||||
|
||||
var internals = getCollatorInternals(collator);
|
||||
|
||||
// Step 4.
|
||||
if (internals.boundCompare === undefined) {
|
||||
// Step 4.a.
|
||||
var F = collatorCompareToBind;
|
||||
|
||||
// Steps 4.b-d.
|
||||
var bc = callFunction(FunctionBind, F, collator);
|
||||
internals.boundCompare = bc;
|
||||
}
|
||||
|
||||
// Step 5.
|
||||
return internals.boundCompare;
|
||||
}
|
||||
_SetCanonicalName(Intl_Collator_compare_get, "get compare");
|
||||
|
||||
/**
|
||||
* Returns the resolved options for a Collator object.
|
||||
*
|
||||
* Spec: ECMAScript Internationalization API Specification, 10.3.3 and 10.4.
|
||||
*/
|
||||
function Intl_Collator_resolvedOptions() {
|
||||
// Check "this Collator object" per introduction of section 10.3.
|
||||
if (!IsObject(this) || !IsCollator(this))
|
||||
ThrowTypeError(JSMSG_INTL_OBJECT_NOT_INITED, "Collator", "resolvedOptions", "Collator");
|
||||
|
||||
var internals = getCollatorInternals(this);
|
||||
|
||||
var result = {
|
||||
locale: internals.locale,
|
||||
usage: internals.usage,
|
||||
sensitivity: internals.sensitivity,
|
||||
ignorePunctuation: internals.ignorePunctuation
|
||||
};
|
||||
|
||||
var relevantExtensionKeys = collatorInternalProperties.relevantExtensionKeys;
|
||||
for (var i = 0; i < relevantExtensionKeys.length; i++) {
|
||||
var key = relevantExtensionKeys[i];
|
||||
var property = (key === "co") ? "collation" : collatorKeyMappings[key].property;
|
||||
_DefineDataProperty(result, property, internals[property]);
|
||||
}
|
||||
return result;
|
||||
}
|
|
@ -746,6 +746,7 @@ selfhosted.inputs = [
|
|||
'builtin/Function.js',
|
||||
'builtin/Generator.js',
|
||||
'builtin/Intl.js',
|
||||
'builtin/intl/Collator.js',
|
||||
'builtin/intl/CurrencyDataGenerated.js',
|
||||
'builtin/intl/DateTimeFormat.js',
|
||||
'builtin/intl/IntlObject.js',
|
||||
|
|
Загрузка…
Ссылка в новой задаче