2011-01-07 Adam Barth <abarth@webkit.org>
[WebKit-https.git] / WebCore / bridge / runtime_array.cpp
1 /*
2  * Copyright (C) 2003, 2008 Apple Inc. All rights reserved.
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 COMPUTER, INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25
26 #include "config.h"
27 #include "runtime_array.h"
28
29 #include <runtime/ArrayPrototype.h>
30 #include <runtime/Error.h>
31 #include <runtime/PropertyNameArray.h>
32 #include "JSDOMBinding.h"
33
34 using namespace WebCore;
35
36 namespace JSC {
37
38 const ClassInfo RuntimeArray::s_info = { "RuntimeArray", &JSArray::info, 0, 0 };
39
40 RuntimeArray::RuntimeArray(ExecState* exec, Bindings::Array* array)
41     // FIXME: deprecatedGetDOMStructure uses the prototype off of the wrong global object
42     // We need to pass in the right global object for "array".
43     : JSArray(deprecatedGetDOMStructure<RuntimeArray>(exec))
44 {
45     setSubclassData(array);
46 }
47
48 RuntimeArray::~RuntimeArray()
49 {
50     delete getConcreteArray();
51 }
52
53 JSValue RuntimeArray::lengthGetter(ExecState*, JSValue slotBase, const Identifier&)
54 {
55     RuntimeArray* thisObj = static_cast<RuntimeArray*>(asObject(slotBase));
56     return jsNumber(thisObj->getLength());
57 }
58
59 JSValue RuntimeArray::indexGetter(ExecState* exec, JSValue slotBase, unsigned index)
60 {
61     RuntimeArray* thisObj = static_cast<RuntimeArray*>(asObject(slotBase));
62     return thisObj->getConcreteArray()->valueAt(exec, index);
63 }
64
65 void RuntimeArray::getOwnPropertyNames(ExecState* exec, PropertyNameArray& propertyNames, EnumerationMode mode)
66 {
67     unsigned length = getLength();
68     for (unsigned i = 0; i < length; ++i)
69         propertyNames.add(Identifier::from(exec, i));
70
71     if (mode == IncludeDontEnumProperties)
72         propertyNames.add(exec->propertyNames().length);
73
74     JSObject::getOwnPropertyNames(exec, propertyNames, mode);
75 }
76
77 bool RuntimeArray::getOwnPropertySlot(ExecState* exec, const Identifier& propertyName, PropertySlot& slot)
78 {
79     if (propertyName == exec->propertyNames().length) {
80         slot.setCacheableCustom(this, lengthGetter);
81         return true;
82     }
83     
84     bool ok;
85     unsigned index = propertyName.toArrayIndex(ok);
86     if (ok) {
87         if (index < getLength()) {
88             slot.setCustomIndex(this, index, indexGetter);
89             return true;
90         }
91     }
92     
93     return JSObject::getOwnPropertySlot(exec, propertyName, slot);
94 }
95
96 bool RuntimeArray::getOwnPropertyDescriptor(ExecState* exec, const Identifier& propertyName, PropertyDescriptor& descriptor)
97 {
98     if (propertyName == exec->propertyNames().length) {
99         PropertySlot slot;
100         slot.setCustom(this, lengthGetter);
101         descriptor.setDescriptor(slot.getValue(exec, propertyName), ReadOnly | DontDelete | DontEnum);
102         return true;
103     }
104     
105     bool ok;
106     unsigned index = propertyName.toArrayIndex(ok);
107     if (ok) {
108         if (index < getLength()) {
109             PropertySlot slot;
110             slot.setCustomIndex(this, index, indexGetter);
111             descriptor.setDescriptor(slot.getValue(exec, propertyName), DontDelete | DontEnum);
112             return true;
113         }
114     }
115     
116     return JSObject::getOwnPropertyDescriptor(exec, propertyName, descriptor);
117 }
118
119 bool RuntimeArray::getOwnPropertySlot(ExecState *exec, unsigned index, PropertySlot& slot)
120 {
121     if (index < getLength()) {
122         slot.setCustomIndex(this, index, indexGetter);
123         return true;
124     }
125     
126     return JSObject::getOwnPropertySlot(exec, index, slot);
127 }
128
129 void RuntimeArray::put(ExecState* exec, const Identifier& propertyName, JSValue value, PutPropertySlot& slot)
130 {
131     if (propertyName == exec->propertyNames().length) {
132         throwError(exec, createRangeError(exec, "Range error"));
133         return;
134     }
135     
136     bool ok;
137     unsigned index = propertyName.toArrayIndex(ok);
138     if (ok) {
139         getConcreteArray()->setValueAt(exec, index, value);
140         return;
141     }
142     
143     JSObject::put(exec, propertyName, value, slot);
144 }
145
146 void RuntimeArray::put(ExecState* exec, unsigned index, JSValue value)
147 {
148     if (index >= getLength()) {
149         throwError(exec, createRangeError(exec, "Range error"));
150         return;
151     }
152     
153     getConcreteArray()->setValueAt(exec, index, value);
154 }
155
156 bool RuntimeArray::deleteProperty(ExecState*, const Identifier&)
157 {
158     return false;
159 }
160
161 bool RuntimeArray::deleteProperty(ExecState*, unsigned)
162 {
163     return false;
164 }
165
166 }