OLD | NEW |
(Empty) | |
| 1 /* |
| 2 * Copyright (C) 2007, 2008 Apple Inc. All rights reserved. |
| 3 * Copyright (C) 2008 Matt Lilek <webkit@mattlilek.com> |
| 4 * Copyright (C) 2009 Google Inc. All rights reserved. |
| 5 * |
| 6 * Redistribution and use in source and binary forms, with or without |
| 7 * modification, are permitted provided that the following conditions are |
| 8 * met: |
| 9 * |
| 10 * * Redistributions of source code must retain the above copyright |
| 11 * notice, this list of conditions and the following disclaimer. |
| 12 * * Redistributions in binary form must reproduce the above |
| 13 * copyright notice, this list of conditions and the following disclaimer |
| 14 * in the documentation and/or other materials provided with the |
| 15 * distribution. |
| 16 * * Neither the name of Google Inc. nor the names of its |
| 17 * contributors may be used to endorse or promote products derived from |
| 18 * this software without specific prior written permission. |
| 19 * |
| 20 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| 21 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| 22 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| 23 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| 24 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| 25 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| 26 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| 27 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| 28 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| 29 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| 30 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| 31 */ |
| 32 |
| 33 #include "config.h" |
| 34 #include "JSInspectorController.h" |
| 35 |
| 36 #include "Console.h" |
| 37 #if ENABLE(DATABASE) |
| 38 #include "Database.h" |
| 39 #include "JSDatabase.h" |
| 40 #endif |
| 41 #include "ExceptionCode.h" |
| 42 #include "Frame.h" |
| 43 #include "FrameLoader.h" |
| 44 #include "InspectorController.h" |
| 45 #include "InspectorResource.h" |
| 46 #include "JavaScriptProfile.h" |
| 47 #include "JSDOMWindow.h" |
| 48 #include "JSInspectedObjectWrapper.h" |
| 49 #include "JSInspectorCallbackWrapper.h" |
| 50 #include "JSNode.h" |
| 51 #include "JSRange.h" |
| 52 #include "Node.h" |
| 53 #include "Page.h" |
| 54 #include "TextIterator.h" |
| 55 #include "VisiblePosition.h" |
| 56 #include <profiler/Profile.h> |
| 57 #include <profiler/Profiler.h> |
| 58 #include <runtime/JSArray.h> |
| 59 #include <runtime/JSLock.h> |
| 60 #include <wtf/Vector.h> |
| 61 |
| 62 #if ENABLE(JAVASCRIPT_DEBUGGER) |
| 63 #include "JavaScriptCallFrame.h" |
| 64 #include "JavaScriptDebugServer.h" |
| 65 #include "JSJavaScriptCallFrame.h" |
| 66 #endif |
| 67 |
| 68 using namespace JSC; |
| 69 |
| 70 namespace WebCore { |
| 71 |
| 72 JSValuePtr JSInspectorController::profiles(JSC::ExecState* exec, const JSC::ArgL
ist&) |
| 73 { |
| 74 JSLock lock(false); |
| 75 ArgList result; |
| 76 const Vector<RefPtr<Profile> >& profiles = impl()->profiles(); |
| 77 |
| 78 for (size_t i = 0; i < profiles.size(); ++i) |
| 79 result.append(toJS(exec, profiles[i].get())); |
| 80 |
| 81 return constructArray(exec, result); |
| 82 } |
| 83 |
| 84 JSValuePtr JSInspectorController::highlightDOMNode(JSC::ExecState* exec, const J
SC::ArgList& args) |
| 85 { |
| 86 if (args.size() < 1) |
| 87 return jsUndefined(); |
| 88 |
| 89 JSQuarantinedObjectWrapper* wrapper = JSQuarantinedObjectWrapper::asWrapper(
args.at(exec, 0)); |
| 90 if (!wrapper) |
| 91 return jsUndefined(); |
| 92 |
| 93 Node* node = toNode(wrapper->unwrappedObject()); |
| 94 if (!node) |
| 95 return jsUndefined(); |
| 96 |
| 97 impl()->highlight(node); |
| 98 |
| 99 return jsUndefined(); |
| 100 } |
| 101 |
| 102 JSValuePtr JSInspectorController::addResourceSourceToFrame(ExecState* exec, cons
t ArgList& args) |
| 103 { |
| 104 if (args.size() < 2) |
| 105 return jsUndefined(); |
| 106 |
| 107 bool ok = false; |
| 108 unsigned identifier = args.at(exec, 0).toUInt32(exec, ok); |
| 109 if (!ok) |
| 110 return jsUndefined(); |
| 111 |
| 112 RefPtr<InspectorResource> resource = impl()->resources().get(identifier); |
| 113 ASSERT(resource); |
| 114 if (!resource) |
| 115 return jsUndefined(); |
| 116 |
| 117 String sourceString = resource->sourceString(); |
| 118 if (sourceString.isEmpty()) |
| 119 return jsUndefined(); |
| 120 |
| 121 return jsBoolean(impl()->addSourceToFrame(resource->mimeType, sourceString,
toNode(args.at(exec, 1)))); |
| 122 } |
| 123 |
| 124 JSValuePtr JSInspectorController::addSourceToFrame(ExecState* exec, const ArgLis
t& args) |
| 125 { |
| 126 if (args.size() < 3) |
| 127 return jsUndefined(); |
| 128 |
| 129 String mimeType = args.at(exec, 0).toString(exec); |
| 130 if (exec->hadException()) |
| 131 return jsUndefined(); |
| 132 |
| 133 String sourceString = args.at(exec, 1).toString(exec); |
| 134 if (exec->hadException()) |
| 135 return jsUndefined(); |
| 136 |
| 137 return jsBoolean(impl()->addSourceToFrame(mimeType, sourceString, toNode(arg
s.at(exec, 1)))); |
| 138 } |
| 139 |
| 140 JSValuePtr JSInspectorController::getResourceDocumentNode(ExecState* exec, const
ArgList& args) |
| 141 { |
| 142 if (args.size() < 1) |
| 143 return jsUndefined(); |
| 144 |
| 145 bool ok = false; |
| 146 unsigned identifier = args.at(exec, 0).toUInt32(exec, ok); |
| 147 if (!ok) |
| 148 return jsUndefined(); |
| 149 |
| 150 RefPtr<InspectorResource> resource = impl()->resources().get(identifier); |
| 151 ASSERT(resource); |
| 152 if (!resource) |
| 153 return jsUndefined(); |
| 154 |
| 155 Frame* frame = resource->frame.get(); |
| 156 Document* document = frame->document(); |
| 157 |
| 158 if (document->isPluginDocument() || document->isImageDocument() || document-
>isMediaDocument()) |
| 159 return jsUndefined(); |
| 160 |
| 161 // FIXME: I am not sure if this is actually needed. Can we just use exec? |
| 162 ExecState* resourceExec = toJSDOMWindowShell(resource->frame.get())->window(
)->globalExec(); |
| 163 |
| 164 JSLock lock(false); |
| 165 return JSInspectedObjectWrapper::wrap(resourceExec, toJS(resourceExec, docum
ent)); |
| 166 } |
| 167 |
| 168 JSValuePtr JSInspectorController::search(ExecState* exec, const ArgList& args) |
| 169 { |
| 170 if (args.size() < 2) |
| 171 return jsUndefined(); |
| 172 |
| 173 Node* node = toNode(args.at(exec, 0)); |
| 174 if (!node) |
| 175 return jsUndefined(); |
| 176 |
| 177 String target = args.at(exec, 1).toString(exec); |
| 178 if (exec->hadException()) |
| 179 return jsUndefined(); |
| 180 |
| 181 ArgList result; |
| 182 RefPtr<Range> searchRange(rangeOfContents(node)); |
| 183 |
| 184 ExceptionCode ec = 0; |
| 185 do { |
| 186 RefPtr<Range> resultRange(findPlainText(searchRange.get(), target, true,
false)); |
| 187 if (resultRange->collapsed(ec)) |
| 188 break; |
| 189 |
| 190 // A non-collapsed result range can in some funky whitespace cases still
not |
| 191 // advance the range's start position (4509328). Break to avoid infinite
loop. |
| 192 VisiblePosition newStart = endVisiblePosition(resultRange.get(), DOWNSTR
EAM); |
| 193 if (newStart == startVisiblePosition(searchRange.get(), DOWNSTREAM)) |
| 194 break; |
| 195 |
| 196 result.append(toJS(exec, resultRange.get())); |
| 197 |
| 198 setStart(searchRange.get(), newStart); |
| 199 } while (true); |
| 200 |
| 201 return constructArray(exec, result); |
| 202 } |
| 203 |
| 204 #if ENABLE(DATABASE) |
| 205 JSValuePtr JSInspectorController::databaseTableNames(ExecState* exec, const ArgL
ist& args) |
| 206 { |
| 207 if (args.size() < 1) |
| 208 return jsUndefined(); |
| 209 |
| 210 JSQuarantinedObjectWrapper* wrapper = JSQuarantinedObjectWrapper::asWrapper(
args.at(exec, 0)); |
| 211 if (!wrapper) |
| 212 return jsUndefined(); |
| 213 |
| 214 Database* database = toDatabase(wrapper->unwrappedObject()); |
| 215 if (!database) |
| 216 return jsUndefined(); |
| 217 |
| 218 ArgList result; |
| 219 |
| 220 Vector<String> tableNames = database->tableNames(); |
| 221 unsigned length = tableNames.size(); |
| 222 for (unsigned i = 0; i < length; ++i) |
| 223 result.append(jsString(exec, tableNames[i])); |
| 224 |
| 225 return constructArray(exec, result); |
| 226 } |
| 227 #endif |
| 228 |
| 229 JSValuePtr JSInspectorController::inspectedWindow(ExecState*, const ArgList&) |
| 230 { |
| 231 JSDOMWindow* inspectedWindow = toJSDOMWindow(impl()->inspectedPage()->mainFr
ame()); |
| 232 return JSInspectedObjectWrapper::wrap(inspectedWindow->globalExec(), inspect
edWindow); |
| 233 } |
| 234 |
| 235 JSValuePtr JSInspectorController::setting(ExecState* exec, const ArgList& args) |
| 236 { |
| 237 if (args.size() < 1) |
| 238 return jsUndefined(); |
| 239 |
| 240 String key = args.at(exec, 0).toString(exec); |
| 241 if (exec->hadException()) |
| 242 return jsUndefined(); |
| 243 |
| 244 const InspectorController::Setting& setting = impl()->setting(key); |
| 245 |
| 246 switch (setting.type()) { |
| 247 default: |
| 248 case InspectorController::Setting::NoType: |
| 249 return jsUndefined(); |
| 250 case InspectorController::Setting::StringType: |
| 251 return jsString(exec, setting.string()); |
| 252 case InspectorController::Setting::DoubleType: |
| 253 return jsNumber(exec, setting.doubleValue()); |
| 254 case InspectorController::Setting::IntegerType: |
| 255 return jsNumber(exec, setting.integerValue()); |
| 256 case InspectorController::Setting::BooleanType: |
| 257 return jsBoolean(setting.booleanValue()); |
| 258 case InspectorController::Setting::StringVectorType: { |
| 259 ArgList stringsArray; |
| 260 const Vector<String>& strings = setting.stringVector(); |
| 261 const unsigned length = strings.size(); |
| 262 for (unsigned i = 0; i < length; ++i) |
| 263 stringsArray.append(jsString(exec, strings[i])); |
| 264 return constructArray(exec, stringsArray); |
| 265 } |
| 266 } |
| 267 } |
| 268 |
| 269 JSValuePtr JSInspectorController::setSetting(ExecState* exec, const ArgList& arg
s) |
| 270 { |
| 271 if (args.size() < 2) |
| 272 return jsUndefined(); |
| 273 |
| 274 String key = args.at(exec, 0).toString(exec); |
| 275 if (exec->hadException()) |
| 276 return jsUndefined(); |
| 277 |
| 278 InspectorController::Setting setting; |
| 279 |
| 280 JSValuePtr value = args.at(exec, 0); |
| 281 if (value.isUndefined() || value.isNull()) { |
| 282 // Do nothing. The setting is already NoType. |
| 283 ASSERT(setting.type() == InspectorController::Setting::NoType); |
| 284 } else if (value.isString()) |
| 285 setting.set(value.toString(exec)); |
| 286 else if (value.isNumber()) |
| 287 setting.set(value.toNumber(exec)); |
| 288 else if (value.isBoolean()) |
| 289 setting.set(value.toBoolean(exec)); |
| 290 else { |
| 291 JSArray* jsArray = asArray(value); |
| 292 if (!jsArray) |
| 293 return jsUndefined(); |
| 294 Vector<String> strings; |
| 295 for (unsigned i = 0; i < jsArray->length(); ++i) { |
| 296 String item = jsArray->get(exec, i).toString(exec); |
| 297 if (exec->hadException()) |
| 298 return jsUndefined(); |
| 299 strings.append(item); |
| 300 } |
| 301 setting.set(strings); |
| 302 } |
| 303 |
| 304 if (exec->hadException()) |
| 305 return jsUndefined(); |
| 306 |
| 307 impl()->setSetting(key, setting); |
| 308 |
| 309 return jsUndefined(); |
| 310 } |
| 311 |
| 312 JSValuePtr JSInspectorController::wrapCallback(ExecState* exec, const ArgList& a
rgs) |
| 313 { |
| 314 if (args.size() < 1) |
| 315 return jsUndefined(); |
| 316 |
| 317 return JSInspectorCallbackWrapper::wrap(exec, args.at(exec, 0)); |
| 318 } |
| 319 |
| 320 JSValuePtr JSInspectorController::currentCallFrame(ExecState* exec, const ArgLis
t&) |
| 321 { |
| 322 JavaScriptCallFrame* callFrame = impl()->currentCallFrame(); |
| 323 if (!callFrame || !callFrame->isValid()) |
| 324 return jsUndefined(); |
| 325 |
| 326 // FIXME: I am not sure if this is actually needed. Can we just use exec? |
| 327 ExecState* globalExec = callFrame->scopeChain()->globalObject()->globalExec(
); |
| 328 |
| 329 JSLock lock(false); |
| 330 return JSInspectedObjectWrapper::wrap(globalExec, toJS(exec, callFrame)); |
| 331 } |
| 332 |
| 333 } // namespace WebCore |
OLD | NEW |