b8ddda9928364a254d413bb0ee02fb457dd4e2bf
[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 want a new history item if this JS was called via a user gesture
119     bool userGesture = static_cast<ScriptInterpreter*>(exec->dynamicInterpreter())->wasRunByUserGesture();
120     frame->loader()->scheduleLocationChange(str, activeFrame->loader()->outgoingReferrer(), !userGesture);
121 }
122
123 // Custom functions
124
125 JSValue* JSHTMLDocument::open(ExecState* exec, const List& args)
126 {
127     // For compatibility with other browsers, pass open calls with more than 2 parameters to the window.
128     if (args.size() > 2) {
129         Frame* frame = static_cast<HTMLDocument*>(impl())->frame();
130         if (frame) {
131             Window* window = Window::retrieveWindow(frame);
132             if (window) {
133                 JSObject* functionObject = window->get(exec, "open")->getObject();
134                 if (!functionObject || !functionObject->implementsCall())
135                     return throwError(exec, TypeError);
136                 return functionObject->call(exec, window, args);
137             }
138         }
139         return jsUndefined();
140     }
141
142     // In the case of two parameters or fewer, do a normal document open.
143     static_cast<HTMLDocument*>(impl())->open();
144     return jsUndefined();
145 }
146
147 static String writeHelper(ExecState* exec, const List& args)
148 {
149     // DOM only specifies single string argument, but NS & IE allow multiple
150     // or no arguments.
151     String str = "";
152     for (int i = 0; i < args.size(); ++i)
153         str += args[i]->toString(exec);
154     return str;
155 }
156
157 JSValue* JSHTMLDocument::write(ExecState* exec, const List& args)
158 {
159     static_cast<HTMLDocument*>(impl())->write(writeHelper(exec, args));
160     return jsUndefined();
161 }
162
163 JSValue* JSHTMLDocument::writeln(ExecState* exec, const List& args)
164 {
165     static_cast<HTMLDocument*>(impl())->write(writeHelper(exec, args) + "\n");
166     return jsUndefined();
167 }
168
169 JSValue* JSHTMLDocument::clear(ExecState*, const List&)
170 {
171     // even IE doesn't support this one...
172     // static_cast<HTMLDocument*>(impl())->clear();
173     return jsUndefined();
174 }
175
176 } // namespace WebCore