[V8] document.all gets confused about its prototype chain
[WebKit.git] / Source / WebCore / bindings / v8 / custom / V8HTMLCollectionCustom.cpp
1 /*
2  * Copyright (C) 2009 Google 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 are
6  * met:
7  * 
8  *     * Redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer.
10  *     * Redistributions in binary form must reproduce the above
11  * copyright notice, this list of conditions and the following disclaimer
12  * in the documentation and/or other materials provided with the
13  * distribution.
14  *     * Neither the name of Google Inc. nor the names of its
15  * contributors may be used to endorse or promote products derived from
16  * this software without specific prior written permission.
17  * 
18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29  */
30
31 #include "config.h"
32 #include "V8HTMLCollection.h"
33
34 #include "HTMLCollection.h"
35 #include "V8Binding.h"
36 #include "V8HTMLAllCollection.h"
37 #include "V8NamedNodesCollection.h"
38 #include "V8Node.h"
39 #include "V8NodeList.h"
40 #include "V8Proxy.h"
41
42 namespace WebCore {
43
44 static v8::Handle<v8::Value> getNamedItems(HTMLCollection* collection, AtomicString name)
45 {
46     Vector<RefPtr<Node> > namedItems;
47     collection->namedItems(name, namedItems);
48
49     if (!namedItems.size())
50         return v8::Handle<v8::Value>();
51
52     if (namedItems.size() == 1)
53         return toV8(namedItems.at(0).release());
54
55     return toV8(V8NamedNodesCollection::create(namedItems));
56 }
57
58 static v8::Handle<v8::Value> getItem(HTMLCollection* collection, v8::Handle<v8::Value> argument)
59 {
60     v8::Local<v8::Uint32> index = argument->ToArrayIndex();
61     if (index.IsEmpty()) {
62         v8::Local<v8::String> asString = argument->ToString();
63         if (asString.IsEmpty())
64             return v8::Handle<v8::Value>();
65         v8::Handle<v8::Value> result = getNamedItems(collection, toWebCoreString(asString));
66
67         if (result.IsEmpty())
68             return v8::Undefined();
69
70         return result;
71     }
72
73     RefPtr<Node> result = collection->item(index->Uint32Value());
74     return toV8(result.release());
75 }
76
77 v8::Handle<v8::Value> V8HTMLCollection::namedPropertyGetter(v8::Local<v8::String> name, const v8::AccessorInfo& info)
78 {
79     INC_STATS("DOM.HTMLCollection.NamedPropertyGetter");
80
81     if (!info.Holder()->GetRealNamedPropertyInPrototypeChain(name).IsEmpty())
82         return notHandledByInterceptor();
83     if (info.Holder()->HasRealNamedCallbackProperty(name))
84         return notHandledByInterceptor();
85
86     HTMLCollection* imp = V8HTMLCollection::toNative(info.Holder());
87     return getNamedItems(imp, v8StringToAtomicWebCoreString(name));
88 }
89
90 v8::Handle<v8::Value> V8HTMLCollection::itemCallback(const v8::Arguments& args)
91 {
92     INC_STATS("DOM.HTMLCollection.item()");
93     HTMLCollection* imp = V8HTMLCollection::toNative(args.Holder());
94     return getItem(imp, args[0]);
95 }
96
97 v8::Handle<v8::Value> V8HTMLCollection::namedItemCallback(const v8::Arguments& args)
98 {
99     INC_STATS("DOM.HTMLCollection.namedItem()");
100     HTMLCollection* imp = V8HTMLCollection::toNative(args.Holder());
101     v8::Handle<v8::Value> result = getNamedItems(imp, toWebCoreString(args[0]));
102
103     if (result.IsEmpty())
104         return v8::Undefined();
105
106     return result;
107 }
108
109 v8::Handle<v8::Value> V8HTMLCollection::callAsFunctionCallback(const v8::Arguments& args)
110 {
111     INC_STATS("DOM.HTMLCollection.callAsFunction()");
112     if (args.Length() < 1)
113         return v8::Undefined();
114
115     HTMLCollection* imp = V8HTMLCollection::toNative(args.Holder());
116
117     if (args.Length() == 1)
118         return getItem(imp, args[0]);
119
120     // If there is a second argument it is the index of the item we want.
121     String name = toWebCoreString(args[0]);
122     v8::Local<v8::Uint32> index = args[1]->ToArrayIndex();
123     if (index.IsEmpty())
124         return v8::Undefined();
125
126     unsigned current = index->Uint32Value();
127     Node* node = imp->namedItem(name);
128     while (node) {
129         if (!current)
130             return toV8(node);
131
132         node = imp->nextNamedItem(name);
133         current--;
134     }
135
136     return v8::Undefined();
137 }
138
139 v8::Handle<v8::Value> toV8(HTMLCollection* impl)
140 {
141     if (impl->type() == DocAll)
142         return toV8(static_cast<HTMLAllCollection*>(impl));
143     return V8HTMLCollection::wrap(impl);
144 }
145
146 } // namespace WebCore