2007-07-04 Matt Perry <mpComplete@gmail.com>
[WebKit-https.git] / WebCore / bindings / js / JSHTMLDocumentCustom.cpp
1 /*
2  * Copyright (C) 2007 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 "JSHTMLDocument.h"
28
29 #include "Frame.h"
30 #include "FrameLoader.h"
31 #include "HTMLBodyElement.h"
32 #include "HTMLCollection.h"
33 #include "HTMLDocument.h"
34 #include "HTMLElement.h"
35 #include "HTMLIFrameElement.h"
36 #include "HTMLNames.h"
37 #include "JSHTMLCollection.h"
38 #include "kjs_html.h"
39 #include "kjs_window.h"
40
41 namespace WebCore {
42
43 using namespace KJS;
44 using namespace HTMLNames;
45
46 bool JSHTMLDocument::canGetItemsForName(ExecState*, HTMLDocument* doc, const Identifier& propertyName)
47 {
48     return doc->hasNamedItem(propertyName) || doc->hasDocExtraNamedItem(propertyName);
49 }
50
51 JSValue* JSHTMLDocument::nameGetter(ExecState* exec, JSObject* originalObject, const Identifier& propertyName, const PropertySlot& slot)
52 {
53     JSHTMLDocument* thisObj = static_cast<JSHTMLDocument*>(slot.slotBase());
54     HTMLDocument* doc = static_cast<HTMLDocument*>(thisObj->impl());
55
56     String name = propertyName;
57     RefPtr<HTMLCollection> collection = doc->documentNamedItems(name);
58
59     unsigned length = collection->length();
60     if (!length)
61         return jsUndefined();
62
63     if (length == 1) {
64         Node* node = collection->firstItem();
65
66         Frame* frame;
67         if (node->hasTagName(iframeTag) && (frame = static_cast<HTMLIFrameElement*>(node)->contentFrame()))
68             return Window::retrieve(frame);
69
70         return toJS(exec, node);
71     } 
72
73     return toJS(exec, collection.get());
74 }
75
76 // Custom attributes
77
78 JSValue* JSHTMLDocument::all(ExecState* exec) const
79 {
80     // If "all" has been overwritten, return the overwritten value
81     if (JSValue* v = getDirect("all"))
82         return v;
83
84     return toJS(exec, static_cast<HTMLDocument*>(impl())->all().get());
85 }
86
87 void JSHTMLDocument::setAll(ExecState*, JSValue* value)
88 {
89     // Add "all" to the property map.
90     putDirect("all", value);
91 }
92
93 JSValue* JSHTMLDocument::location(ExecState* exec) const
94 {
95     Frame* frame = static_cast<HTMLDocument*>(impl())->frame();
96     if (!frame)
97         return jsNull();
98
99     Window* win = Window::retrieveWindow(frame);
100     ASSERT(win);
101     return win->location();
102 }
103
104 void JSHTMLDocument::setLocation(ExecState* exec, JSValue* value)
105 {
106     Frame* frame = static_cast<HTMLDocument*>(impl())->frame();
107     if (!frame)
108         return;
109
110     String str = value->toString(exec);
111
112     // When assigning location, IE and Mozilla both resolve the URL
113     // relative not the target frame.
114     Frame* activeFrame = static_cast<ScriptInterpreter*>(exec->dynamicInterpreter())->frame();
115     if (activeFrame)
116         str = activeFrame->document()->completeURL(str);
117
118     // We always want a new history item when assigning to document.location.
119     frame->loader()->scheduleLocationChange(str, activeFrame->loader()->outgoingReferrer(), false, true);
120 }
121
122 // Custom functions
123
124 JSValue* JSHTMLDocument::open(ExecState* exec, const List& args)
125 {
126     // For compatibility with other browsers, pass open calls with more than 2 parameters to the window.
127     if (args.size() > 2) {
128         Frame* frame = static_cast<HTMLDocument*>(impl())->frame();
129         if (frame) {
130             Window* window = Window::retrieveWindow(frame);
131             if (window) {
132                 JSObject* functionObject = window->get(exec, "open")->getObject();
133                 if (!functionObject || !functionObject->implementsCall())
134                     return throwError(exec, TypeError);
135                 return functionObject->call(exec, window, args);
136             }
137         }
138         return jsUndefined();
139     }
140
141     // In the case of two parameters or fewer, do a normal document open.
142     static_cast<HTMLDocument*>(impl())->open();
143     return jsUndefined();
144 }
145
146 static String writeHelper(ExecState* exec, const List& args)
147 {
148     // DOM only specifies single string argument, but NS & IE allow multiple
149     // or no arguments.
150     String str = "";
151     for (int i = 0; i < args.size(); ++i)
152         str += args[i]->toString(exec);
153     return str;
154 }
155
156 JSValue* JSHTMLDocument::write(ExecState* exec, const List& args)
157 {
158     static_cast<HTMLDocument*>(impl())->write(writeHelper(exec, args));
159     return jsUndefined();
160 }
161
162 JSValue* JSHTMLDocument::writeln(ExecState* exec, const List& args)
163 {
164     static_cast<HTMLDocument*>(impl())->write(writeHelper(exec, args) + "\n");
165     return jsUndefined();
166 }
167
168 JSValue* JSHTMLDocument::clear(ExecState*, const List&)
169 {
170     // even IE doesn't support this one...
171     // static_cast<HTMLDocument*>(impl())->clear();
172     return jsUndefined();
173 }
174
175 } // namespace WebCore