[INTL] Implement Intl.DateTimeFormat.prototype.resolvedOptions ()
[WebKit.git] / Source / JavaScriptCore / runtime / IntlDateTimeFormatConstructor.cpp
1 /*
2  * Copyright (C) 2015 Andy VanWagoner (thetalecrafter@gmail.com)
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26
27 #include "config.h"
28 #include "IntlDateTimeFormatConstructor.h"
29
30 #if ENABLE(INTL)
31
32 #include "Error.h"
33 #include "IntlDateTimeFormat.h"
34 #include "IntlDateTimeFormatPrototype.h"
35 #include "IntlObject.h"
36 #include "JSCJSValueInlines.h"
37 #include "JSCellInlines.h"
38 #include "Lookup.h"
39 #include "SlotVisitorInlines.h"
40 #include "StructureInlines.h"
41
42 namespace JSC {
43
44 STATIC_ASSERT_IS_TRIVIALLY_DESTRUCTIBLE(IntlDateTimeFormatConstructor);
45
46 static EncodedJSValue JSC_HOST_CALL IntlDateTimeFormatConstructorFuncSupportedLocalesOf(ExecState*);
47
48 }
49
50 #include "IntlDateTimeFormatConstructor.lut.h"
51
52 namespace JSC {
53
54 const ClassInfo IntlDateTimeFormatConstructor::s_info = { "Function", &InternalFunction::s_info, &dateTimeFormatConstructorTable, CREATE_METHOD_TABLE(IntlDateTimeFormatConstructor) };
55
56 /* Source for IntlDateTimeFormatConstructor.lut.h
57 @begin dateTimeFormatConstructorTable
58   supportedLocalesOf             IntlDateTimeFormatConstructorFuncSupportedLocalesOf             DontEnum|Function 1
59 @end
60 */
61
62 IntlDateTimeFormatConstructor* IntlDateTimeFormatConstructor::create(VM& vm, Structure* structure, IntlDateTimeFormatPrototype* dateTimeFormatPrototype, Structure* dateTimeFormatStructure)
63 {
64     IntlDateTimeFormatConstructor* constructor = new (NotNull, allocateCell<IntlDateTimeFormatConstructor>(vm.heap)) IntlDateTimeFormatConstructor(vm, structure);
65     constructor->finishCreation(vm, dateTimeFormatPrototype, dateTimeFormatStructure);
66     return constructor;
67 }
68
69 Structure* IntlDateTimeFormatConstructor::createStructure(VM& vm, JSGlobalObject* globalObject, JSValue prototype)
70 {
71     return Structure::create(vm, globalObject, prototype, TypeInfo(ObjectType, StructureFlags), info());
72 }
73
74 IntlDateTimeFormatConstructor::IntlDateTimeFormatConstructor(VM& vm, Structure* structure)
75     : InternalFunction(vm, structure)
76 {
77 }
78
79 void IntlDateTimeFormatConstructor::finishCreation(VM& vm, IntlDateTimeFormatPrototype* dateTimeFormatPrototype, Structure* dateTimeFormatStructure)
80 {
81     Base::finishCreation(vm, ASCIILiteral("DateTimeFormat"));
82     putDirectWithoutTransition(vm, vm.propertyNames->prototype, dateTimeFormatPrototype, DontEnum | DontDelete | ReadOnly);
83     putDirectWithoutTransition(vm, vm.propertyNames->length, jsNumber(0), ReadOnly | DontEnum | DontDelete);
84     m_dateTimeFormatStructure.set(vm, this, dateTimeFormatStructure);
85 }
86
87 static EncodedJSValue JSC_HOST_CALL constructIntlDateTimeFormat(ExecState* state)
88 {
89     // 12.1.2 Intl.DateTimeFormat ([locales [, options]]) (ECMA-402 2.0)
90     // 1. If NewTarget is undefined, let newTarget be the active function object, else let newTarget be NewTarget.
91     JSValue newTarget = state->newTarget();
92     if (newTarget.isUndefined())
93         newTarget = state->callee();
94
95     // 2. Let dateTimeFormat be OrdinaryCreateFromConstructor(newTarget, %DateTimeFormatPrototype%).
96     VM& vm = state->vm();
97     IntlDateTimeFormat* dateTimeFormat = IntlDateTimeFormat::create(vm, jsCast<IntlDateTimeFormatConstructor*>(state->callee()));
98     if (dateTimeFormat && !jsDynamicCast<IntlDateTimeFormatConstructor*>(newTarget)) {
99         JSValue proto = asObject(newTarget)->getDirect(vm, vm.propertyNames->prototype);
100         asObject(dateTimeFormat)->setPrototypeWithCycleCheck(state, proto);
101     }
102
103     // 3. ReturnIfAbrupt(dateTimeFormat).
104     ASSERT(dateTimeFormat);
105
106     // 4. Return InitializeDateTimeFormat(dateTimeFormat, locales, options).
107     JSValue locales = state->argument(0);
108     JSValue options = state->argument(1);
109     dateTimeFormat->initializeDateTimeFormat(*state, locales, options);
110     return JSValue::encode(dateTimeFormat);
111 }
112
113 static EncodedJSValue JSC_HOST_CALL callIntlDateTimeFormat(ExecState* state)
114 {
115     // 12.1.2 Intl.DateTimeFormat ([locales [, options]]) (ECMA-402 2.0)
116     // 1. If NewTarget is undefined, let newTarget be the active function object, else let newTarget be NewTarget.
117     // NewTarget is always undefined when called as a function.
118
119     // 2. Let dateTimeFormat be OrdinaryCreateFromConstructor(newTarget, %DateTimeFormatPrototype%).
120     VM& vm = state->vm();
121     IntlDateTimeFormat* dateTimeFormat = IntlDateTimeFormat::create(vm, jsCast<IntlDateTimeFormatConstructor*>(state->callee()));
122
123     // 3. ReturnIfAbrupt(dateTimeFormat).
124     ASSERT(dateTimeFormat);
125
126     // 4. Return InitializeDateTimeFormat(dateTimeFormat, locales, options).
127     JSValue locales = state->argument(0);
128     JSValue options = state->argument(1);
129     dateTimeFormat->initializeDateTimeFormat(*state, locales, options);
130     return JSValue::encode(dateTimeFormat);
131 }
132
133 ConstructType IntlDateTimeFormatConstructor::getConstructData(JSCell*, ConstructData& constructData)
134 {
135     constructData.native.function = constructIntlDateTimeFormat;
136     return ConstructTypeHost;
137 }
138
139 CallType IntlDateTimeFormatConstructor::getCallData(JSCell*, CallData& callData)
140 {
141     callData.native.function = callIntlDateTimeFormat;
142     return CallTypeHost;
143 }
144
145 bool IntlDateTimeFormatConstructor::getOwnPropertySlot(JSObject* object, ExecState* state, PropertyName propertyName, PropertySlot& slot)
146 {
147     return getStaticFunctionSlot<InternalFunction>(state, dateTimeFormatConstructorTable, jsCast<IntlDateTimeFormatConstructor*>(object), propertyName, slot);
148 }
149
150 EncodedJSValue JSC_HOST_CALL IntlDateTimeFormatConstructorFuncSupportedLocalesOf(ExecState* state)
151 {
152     // 12.2.2 Intl.DateTimeFormat.supportedLocalesOf(locales [, options]) (ECMA-402 2.0)
153
154     // 1. Let availableLocales be %DateTimeFormat%.[[availableLocales]].
155     JSGlobalObject* globalObject = state->callee()->globalObject();
156     const HashSet<String> availableLocales = globalObject->intlDateTimeFormatAvailableLocales();
157
158     // 2. Let requestedLocales be CanonicalizeLocaleList(locales).
159     Vector<String> requestedLocales = canonicalizeLocaleList(*state, state->argument(0));
160     if (state->hadException())
161         return JSValue::encode(jsUndefined());
162
163     // 3. Return SupportedLocales(availableLocales, requestedLocales, options).
164     return JSValue::encode(supportedLocales(*state, availableLocales, requestedLocales, state->argument(1)));
165 }
166
167 void IntlDateTimeFormatConstructor::visitChildren(JSCell* cell, SlotVisitor& visitor)
168 {
169     IntlDateTimeFormatConstructor* thisObject = jsCast<IntlDateTimeFormatConstructor*>(cell);
170     ASSERT_GC_OBJECT_INHERITS(thisObject, info());
171
172     Base::visitChildren(thisObject, visitor);
173
174     visitor.append(&thisObject->m_dateTimeFormatStructure);
175 }
176
177 } // namespace JSC
178
179 #endif // ENABLE(INTL)