Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(6)

Unified Diff: sky/engine/core/inspector/InjectedScriptSource.js

Issue 727593004: Wire up the Inspector V8 Debugger (Closed) Base URL: git@github.com:domokit/mojo.git@master
Patch Set: Actually works Created 6 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: sky/engine/core/inspector/InjectedScriptSource.js
diff --git a/sky/engine/core/inspector/InjectedScriptSource.js b/sky/engine/core/inspector/InjectedScriptSource.js
new file mode 100644
index 0000000000000000000000000000000000000000..a5054c15c182c85d05397323153570db09b9a2d8
--- /dev/null
+++ b/sky/engine/core/inspector/InjectedScriptSource.js
@@ -0,0 +1,1731 @@
+/*
+ * Copyright (C) 2007 Apple Inc. All rights reserved.
+ * Copyright (C) 2013 Google Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright
+ * notice, this list of conditions and the following disclaimer.
+ * 2. Redistributions in binary form must reproduce the above copyright
+ * notice, this list of conditions and the following disclaimer in the
+ * documentation and/or other materials provided with the distribution.
+ * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ * its contributors may be used to endorse or promote products derived
+ * from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+/**
+ * @param {InjectedScriptHost} InjectedScriptHost
+ * @param {Window} inspectedWindow
+ * @param {number} injectedScriptId
+ */
+(function (InjectedScriptHost, inspectedWindow, injectedScriptId) {
+
+/**
+ * Protect against Object overwritten by the user code.
+ * @suppress {duplicate}
+ */
+var Object = /** @type {function(new:Object, *=)} */ ({}.constructor);
+
+/**
+ * @param {!Array.<T>} array
+ * @param {...} var_args
+ * @template T
+ */
+function push(array, var_args)
+{
+ for (var i = 1; i < arguments.length; ++i)
+ array[array.length] = arguments[i];
+}
+
+/**
+ * @param {!Arguments.<T>} array
+ * @param {number=} index
+ * @return {!Array.<T>}
+ * @template T
+ */
+function slice(array, index)
+{
+ var result = [];
+ for (var i = index || 0, j = 0; i < array.length; ++i, ++j)
+ result[j] = array[i];
+ return result;
+}
+
+/**
+ * @param {!Array.<T>} array1
+ * @param {!Array.<T>} array2
+ * @return {!Array.<T>}
+ * @template T
+ */
+function concat(array1, array2)
+{
+ var result = [];
+ for (var i = 0; i < array1.length; ++i)
+ push(result, array1[i]);
+ for (var i = 0; i < array2.length; ++i)
+ push(result, array2[i]);
+ return result;
+}
+
+/**
+ * @param {*} obj
+ * @return {string}
+ */
+function toString(obj)
+{
+ // We don't use String(obj) because it could be overriden.
+ return "" + obj;
+}
+
+/**
+ * @param {*} obj
+ * @return {string}
+ */
+function toStringDescription(obj)
+{
+ if (typeof obj === "number" && obj === 0 && 1 / obj < 0)
+ return "-0"; // Negative zero.
+ return "" + obj;
+}
+
+/**
+ * Please use this bind, not the one from Function.prototype
+ * @param {function(...)} func
+ * @param {?Object} thisObject
+ * @param {...} var_args
+ * @return {function(...)}
+ */
+function bind(func, thisObject, var_args)
+{
+ var args = slice(arguments, 2);
+
+ /**
+ * @param {...} var_args
+ */
+ function bound(var_args)
+ {
+ return InjectedScriptHost.callFunction(func, thisObject, concat(args, slice(arguments)));
+ }
+ bound.toString = function()
+ {
+ return "bound: " + func;
+ };
+ return bound;
+}
+
+/**
+ * @param {T} obj
+ * @return {T}
+ * @template T
+ */
+function nullifyObjectProto(obj)
+{
+ if (obj && typeof obj === "object")
+ obj.__proto__ = null;
+ return obj;
+}
+
+/**
+ * @param {*} obj
+ * @return {boolean}
+ */
+function isUInt32(obj)
+{
+ return typeof obj === "number" && obj >>> 0 === obj && (obj > 0 || 1 / obj > 0);
+}
+
+/**
+ * FireBug's array detection.
+ * @param {*} obj
+ * @return {boolean}
+ */
+function isArrayLike(obj)
+{
+ if (typeof obj !== "object")
+ return false;
+ try {
+ if (typeof obj.splice === "function")
+ return isUInt32(obj.length);
+ } catch (e) {
+ }
+ return false;
+}
+
+/**
+ * @param {number} a
+ * @param {number} b
+ * @return {number}
+ */
+function max(a, b)
+{
+ return a > b ? a : b;
+}
+
+/**
+ * FIXME: Remove once ES6 is supported natively by JS compiler.
+ * @param {*} obj
+ * @return {boolean}
+ */
+function isSymbol(obj)
+{
+ var type = typeof obj;
+ return (type === "symbol");
+}
+
+/**
+ * @constructor
+ */
+var InjectedScript = function()
+{
+ /** @type {number} */
+ this._lastBoundObjectId = 1;
+ /** @type {!Object.<number, (!Object|symbol)>} */
+ this._idToWrappedObject = { __proto__: null };
+ /** @type {!Object.<number, string>} */
+ this._idToObjectGroupName = { __proto__: null };
+ /** @type {!Object.<string, !Array.<number>>} */
+ this._objectGroups = { __proto__: null };
+ /** @type {!Object.<string, !Object>} */
+ this._modules = { __proto__: null };
+}
+
+/**
+ * @type {!Object.<string, boolean>}
+ * @const
+ */
+InjectedScript.primitiveTypes = {
+ "undefined": true,
+ "boolean": true,
+ "number": true,
+ "string": true,
+ __proto__: null
+}
+
+InjectedScript.prototype = {
+ /**
+ * @param {*} object
+ * @return {boolean}
+ */
+ isPrimitiveValue: function(object)
+ {
+ // FIXME(33716): typeof document.all is always 'undefined'.
+ return InjectedScript.primitiveTypes[typeof object] && !this._isHTMLAllCollection(object);
+ },
+
+ /**
+ * @param {*} object
+ * @param {string} groupName
+ * @param {boolean} canAccessInspectedWindow
+ * @param {boolean} generatePreview
+ * @return {!RuntimeAgent.RemoteObject}
+ */
+ wrapObject: function(object, groupName, canAccessInspectedWindow, generatePreview)
+ {
+ if (canAccessInspectedWindow)
+ return this._wrapObject(object, groupName, false, generatePreview);
+ return this._fallbackWrapper(object);
+ },
+
+ /**
+ * @param {*} object
+ * @return {!RuntimeAgent.RemoteObject}
+ */
+ _fallbackWrapper: function(object)
+ {
+ var result = { __proto__: null };
+ result.type = typeof object;
+ if (this.isPrimitiveValue(object))
+ result.value = object;
+ else
+ result.description = toString(object);
+ return /** @type {!RuntimeAgent.RemoteObject} */ (result);
+ },
+
+ /**
+ * @param {boolean} canAccessInspectedWindow
+ * @param {!Object} table
+ * @param {!Array.<string>|string|boolean} columns
+ * @return {!RuntimeAgent.RemoteObject}
+ */
+ wrapTable: function(canAccessInspectedWindow, table, columns)
+ {
+ if (!canAccessInspectedWindow)
+ return this._fallbackWrapper(table);
+ var columnNames = null;
+ if (typeof columns === "string")
+ columns = [columns];
+ if (InjectedScriptHost.subtype(columns) === "array") {
+ columnNames = [];
+ for (var i = 0; i < columns.length; ++i)
+ columnNames[i] = toString(columns[i]);
+ }
+ return this._wrapObject(table, "console", false, true, columnNames, true);
+ },
+
+ /**
+ * @param {*} object
+ */
+ inspectNode: function(object)
+ {
+ this._inspect(object);
+ },
+
+ /**
+ * @param {*} object
+ * @return {*}
+ */
+ _inspect: function(object)
+ {
+ if (arguments.length === 0)
+ return;
+
+ var objectId = this._wrapObject(object, "");
+ var hints = { __proto__: null };
+
+ InjectedScriptHost.inspect(objectId, hints);
+ return object;
+ },
+
+ /**
+ * This method cannot throw.
+ * @param {*} object
+ * @param {string=} objectGroupName
+ * @param {boolean=} forceValueType
+ * @param {boolean=} generatePreview
+ * @param {?Array.<string>=} columnNames
+ * @param {boolean=} isTable
+ * @return {!RuntimeAgent.RemoteObject}
+ * @suppress {checkTypes}
+ */
+ _wrapObject: function(object, objectGroupName, forceValueType, generatePreview, columnNames, isTable)
+ {
+ try {
+ return new InjectedScript.RemoteObject(object, objectGroupName, forceValueType, generatePreview, columnNames, isTable);
+ } catch (e) {
+ try {
+ var description = injectedScript._describe(e);
+ } catch (ex) {
+ var description = "<failed to convert exception to string>";
+ }
+ return new InjectedScript.RemoteObject(description);
+ }
+ },
+
+ /**
+ * @param {!Object|symbol} object
+ * @param {string=} objectGroupName
+ * @return {string}
+ */
+ _bind: function(object, objectGroupName)
+ {
+ var id = this._lastBoundObjectId++;
+ this._idToWrappedObject[id] = object;
+ var objectId = "{\"injectedScriptId\":" + injectedScriptId + ",\"id\":" + id + "}";
+ if (objectGroupName) {
+ var group = this._objectGroups[objectGroupName];
+ if (!group) {
+ group = [];
+ this._objectGroups[objectGroupName] = group;
+ }
+ push(group, id);
+ this._idToObjectGroupName[id] = objectGroupName;
+ }
+ return objectId;
+ },
+
+ /**
+ * @param {string} objectId
+ * @return {!Object}
+ */
+ _parseObjectId: function(objectId)
+ {
+ return nullifyObjectProto(InjectedScriptHost.eval("(" + objectId + ")"));
+ },
+
+ /**
+ * @param {string} objectGroupName
+ */
+ releaseObjectGroup: function(objectGroupName)
+ {
+ if (objectGroupName === "console")
+ delete this._lastResult;
+ var group = this._objectGroups[objectGroupName];
+ if (!group)
+ return;
+ for (var i = 0; i < group.length; i++)
+ this._releaseObject(group[i]);
+ delete this._objectGroups[objectGroupName];
+ },
+
+ /**
+ * @param {string} methodName
+ * @param {string} args
+ * @return {*}
+ */
+ dispatch: function(methodName, args)
+ {
+ var argsArray = InjectedScriptHost.eval("(" + args + ")");
+ var result = InjectedScriptHost.callFunction(this[methodName], this, argsArray);
+ if (typeof result === "undefined") {
+ inspectedWindow.console.error("Web Inspector error: InjectedScript.%s returns undefined", methodName);
+ result = null;
+ }
+ return result;
+ },
+
+ /**
+ * @param {string} objectId
+ * @param {boolean} ownProperties
+ * @param {boolean} accessorPropertiesOnly
+ * @return {!Array.<!RuntimeAgent.PropertyDescriptor>|boolean}
+ */
+ getProperties: function(objectId, ownProperties, accessorPropertiesOnly)
+ {
+ var parsedObjectId = this._parseObjectId(objectId);
+ var object = this._objectForId(parsedObjectId);
+ var objectGroupName = this._idToObjectGroupName[parsedObjectId.id];
+
+ if (!this._isDefined(object) || isSymbol(object))
+ return false;
+ object = /** @type {!Object} */ (object);
+ var descriptors = this._propertyDescriptors(object, ownProperties, accessorPropertiesOnly);
+
+ // Go over properties, wrap object values.
+ for (var i = 0; i < descriptors.length; ++i) {
+ var descriptor = descriptors[i];
+ if ("get" in descriptor)
+ descriptor.get = this._wrapObject(descriptor.get, objectGroupName);
+ if ("set" in descriptor)
+ descriptor.set = this._wrapObject(descriptor.set, objectGroupName);
+ if ("value" in descriptor)
+ descriptor.value = this._wrapObject(descriptor.value, objectGroupName);
+ if (!("configurable" in descriptor))
+ descriptor.configurable = false;
+ if (!("enumerable" in descriptor))
+ descriptor.enumerable = false;
+ if ("symbol" in descriptor)
+ descriptor.symbol = this._wrapObject(descriptor.symbol, objectGroupName);
+ }
+ return descriptors;
+ },
+
+ /**
+ * @param {string} objectId
+ * @return {!Array.<!Object>|boolean}
+ */
+ getInternalProperties: function(objectId)
+ {
+ var parsedObjectId = this._parseObjectId(objectId);
+ var object = this._objectForId(parsedObjectId);
+ var objectGroupName = this._idToObjectGroupName[parsedObjectId.id];
+ if (!this._isDefined(object) || isSymbol(object))
+ return false;
+ object = /** @type {!Object} */ (object);
+ var descriptors = [];
+ var internalProperties = InjectedScriptHost.getInternalProperties(object);
+ if (internalProperties) {
+ for (var i = 0; i < internalProperties.length; i++) {
+ var property = internalProperties[i];
+ var descriptor = {
+ name: property.name,
+ value: this._wrapObject(property.value, objectGroupName),
+ __proto__: null
+ };
+ push(descriptors, descriptor);
+ }
+ }
+ return descriptors;
+ },
+
+ /**
+ * @param {string} functionId
+ * @return {!DebuggerAgent.FunctionDetails|string}
+ */
+ getFunctionDetails: function(functionId)
+ {
+ var parsedFunctionId = this._parseObjectId(functionId);
+ var func = this._objectForId(parsedFunctionId);
+ if (typeof func !== "function")
+ return "Cannot resolve function by id.";
+ var details = nullifyObjectProto(InjectedScriptHost.functionDetails(func));
+ if ("rawScopes" in details) {
+ var objectGroupName = this._idToObjectGroupName[parsedFunctionId.id];
+ var rawScopes = details.rawScopes;
+ delete details.rawScopes;
+ var scopes = [];
+ for (var i = 0; i < rawScopes.length; ++i)
+ scopes[i] = InjectedScript.CallFrameProxy._createScopeJson(rawScopes[i].type, rawScopes[i].object, objectGroupName);
+ details.scopeChain = scopes;
+ }
+ return details;
+ },
+
+ /**
+ * @param {string} objectId
+ * @return {!Array.<!Object>|string}
+ */
+ getCollectionEntries: function(objectId)
+ {
+ var parsedObjectId = this._parseObjectId(objectId);
+ var object = this._objectForId(parsedObjectId);
+ if (!object || typeof object !== "object")
+ return "Could not find object with given id";
+ var entries = InjectedScriptHost.collectionEntries(object);
+ if (!entries)
+ return "Object with given id is not a collection";
+ var objectGroupName = this._idToObjectGroupName[parsedObjectId.id];
+ for (var i = 0; i < entries.length; ++i) {
+ var entry = nullifyObjectProto(entries[i]);
+ if ("key" in entry)
+ entry.key = this._wrapObject(entry.key, objectGroupName);
+ entry.value = this._wrapObject(entry.value, objectGroupName);
+ entries[i] = entry;
+ }
+ return entries;
+ },
+
+ /**
+ * @param {string} objectId
+ */
+ releaseObject: function(objectId)
+ {
+ var parsedObjectId = this._parseObjectId(objectId);
+ this._releaseObject(parsedObjectId.id);
+ },
+
+ /**
+ * @param {number} id
+ */
+ _releaseObject: function(id)
+ {
+ delete this._idToWrappedObject[id];
+ delete this._idToObjectGroupName[id];
+ },
+
+ /**
+ * @param {!Object} object
+ * @param {boolean=} ownProperties
+ * @param {boolean=} accessorPropertiesOnly
+ * @return {!Array.<!Object>}
+ */
+ _propertyDescriptors: function(object, ownProperties, accessorPropertiesOnly)
+ {
+ var descriptors = [];
+ var propertyProcessed = { __proto__: null };
+
+ /**
+ * @param {?Object} o
+ * @param {!Array.<string|symbol>} properties
+ */
+ function process(o, properties)
+ {
+ for (var i = 0; i < properties.length; ++i) {
+ var property = properties[i];
+ if (propertyProcessed[property])
+ continue;
+
+ var name = property;
+ if (isSymbol(property))
+ name = injectedScript._describe(property);
+
+ try {
+ propertyProcessed[property] = true;
+ var descriptor = nullifyObjectProto(InjectedScriptHost.suppressWarningsAndCallFunction(Object.getOwnPropertyDescriptor, Object, [o, property]));
+ if (descriptor) {
+ if (accessorPropertiesOnly && !("get" in descriptor || "set" in descriptor))
+ continue;
+ } else {
+ // Not all bindings provide proper descriptors. Fall back to the writable, configurable property.
+ if (accessorPropertiesOnly)
+ continue;
+ try {
+ descriptor = { name: name, value: o[property], writable: false, configurable: false, enumerable: false, __proto__: null };
+ if (o === object)
+ descriptor.isOwn = true;
+ push(descriptors, descriptor);
+ } catch (e) {
+ // Silent catch.
+ }
+ continue;
+ }
+ } catch (e) {
+ if (accessorPropertiesOnly)
+ continue;
+ var descriptor = { __proto__: null };
+ descriptor.value = e;
+ descriptor.wasThrown = true;
+ }
+
+ descriptor.name = name;
+ if (o === object)
+ descriptor.isOwn = true;
+ if (isSymbol(property))
+ descriptor.symbol = property;
+ push(descriptors, descriptor);
+ }
+ }
+
+ for (var o = object; this._isDefined(o); o = o.__proto__) {
+ // First call Object.keys() to enforce ordering of the property descriptors.
+ process(o, Object.keys(/** @type {!Object} */ (o)));
+ process(o, Object.getOwnPropertyNames(/** @type {!Object} */ (o)));
+ if (Object.getOwnPropertySymbols)
+ process(o, Object.getOwnPropertySymbols(/** @type {!Object} */ (o)));
+
+ if (ownProperties) {
+ if (object.__proto__ && !accessorPropertiesOnly)
+ push(descriptors, { name: "__proto__", value: object.__proto__, writable: true, configurable: true, enumerable: false, isOwn: true, __proto__: null });
+ break;
+ }
+ }
+
+ return descriptors;
+ },
+
+ /**
+ * @param {string} expression
+ * @param {string} objectGroup
+ * @param {boolean} injectCommandLineAPI
+ * @param {boolean} returnByValue
+ * @param {boolean} generatePreview
+ * @return {*}
+ */
+ evaluate: function(expression, objectGroup, injectCommandLineAPI, returnByValue, generatePreview)
+ {
+ return this._evaluateAndWrap(null, null, expression, objectGroup, false, injectCommandLineAPI, returnByValue, generatePreview);
+ },
+
+ /**
+ * @param {string} objectId
+ * @param {string} expression
+ * @param {string} args
+ * @param {boolean} returnByValue
+ * @return {!Object|string}
+ */
+ callFunctionOn: function(objectId, expression, args, returnByValue)
+ {
+ var parsedObjectId = this._parseObjectId(objectId);
+ var object = this._objectForId(parsedObjectId);
+ if (!this._isDefined(object))
+ return "Could not find object with given id";
+
+ if (args) {
+ var resolvedArgs = [];
+ args = InjectedScriptHost.eval(args);
+ for (var i = 0; i < args.length; ++i) {
+ try {
+ resolvedArgs[i] = this._resolveCallArgument(args[i]);
+ } catch (e) {
+ return toString(e);
+ }
+ }
+ }
+
+ try {
+ var objectGroup = this._idToObjectGroupName[parsedObjectId.id];
+ var func = InjectedScriptHost.eval("(" + expression + ")");
+ if (typeof func !== "function")
+ return "Given expression does not evaluate to a function";
+
+ return { wasThrown: false,
+ result: this._wrapObject(InjectedScriptHost.callFunction(func, object, resolvedArgs), objectGroup, returnByValue),
+ __proto__: null };
+ } catch (e) {
+ return this._createThrownValue(e, objectGroup, false);
+ }
+ },
+
+ /**
+ * Resolves a value from CallArgument description.
+ * @param {!RuntimeAgent.CallArgument} callArgumentJson
+ * @return {*} resolved value
+ * @throws {string} error message
+ */
+ _resolveCallArgument: function(callArgumentJson)
+ {
+ callArgumentJson = nullifyObjectProto(callArgumentJson);
+ var objectId = callArgumentJson.objectId;
+ if (objectId) {
+ var parsedArgId = this._parseObjectId(objectId);
+ if (!parsedArgId || parsedArgId["injectedScriptId"] !== injectedScriptId)
+ throw "Arguments should belong to the same JavaScript world as the target object.";
+
+ var resolvedArg = this._objectForId(parsedArgId);
+ if (!this._isDefined(resolvedArg))
+ throw "Could not find object with given id";
+
+ return resolvedArg;
+ } else if ("value" in callArgumentJson) {
+ var value = callArgumentJson.value;
+ if (callArgumentJson.type === "number" && typeof value !== "number")
+ value = Number(value);
+ return value;
+ }
+ return undefined;
+ },
+
+ /**
+ * @param {?function(string):*} evalFunction
+ * @param {?Object} object
+ * @param {string} expression
+ * @param {string} objectGroup
+ * @param {boolean} isEvalOnCallFrame
+ * @param {boolean} injectCommandLineAPI
+ * @param {boolean} returnByValue
+ * @param {boolean} generatePreview
+ * @param {!Array.<!Object>=} scopeChain
+ * @return {!Object}
+ */
+ _evaluateAndWrap: function(evalFunction, object, expression, objectGroup, isEvalOnCallFrame, injectCommandLineAPI, returnByValue, generatePreview, scopeChain)
+ {
+ var wrappedResult = this._evaluateOn(evalFunction, object, objectGroup, expression, isEvalOnCallFrame, injectCommandLineAPI, scopeChain);
+ if (!wrappedResult.exceptionDetails) {
+ return { wasThrown: false,
+ result: this._wrapObject(wrappedResult.result, objectGroup, returnByValue, generatePreview),
+ __proto__: null };
+ }
+ return this._createThrownValue(wrappedResult.result, objectGroup, generatePreview, wrappedResult.exceptionDetails);
+ },
+
+ /**
+ * @param {*} value
+ * @param {string} objectGroup
+ * @param {boolean} generatePreview
+ * @param {!DebuggerAgent.ExceptionDetails=} exceptionDetails
+ * @return {!Object}
+ */
+ _createThrownValue: function(value, objectGroup, generatePreview, exceptionDetails)
+ {
+ var remoteObject = this._wrapObject(value, objectGroup, false, generatePreview && !(value instanceof Error));
+ if (!remoteObject.description){
+ try {
+ remoteObject.description = toStringDescription(value);
+ } catch (e) {}
+ }
+ return { wasThrown: true, result: remoteObject, exceptionDetails: exceptionDetails, __proto__: null };
+ },
+
+ /**
+ * @param {?function(string):*} evalFunction
+ * @param {?Object} object
+ * @param {string} objectGroup
+ * @param {string} expression
+ * @param {boolean} isEvalOnCallFrame
+ * @param {boolean} injectCommandLineAPI
+ * @param {!Array.<!Object>=} scopeChain
+ * @return {*}
+ */
+ _evaluateOn: function(evalFunction, object, objectGroup, expression, isEvalOnCallFrame, injectCommandLineAPI, scopeChain)
+ {
+ // Only install command line api object for the time of evaluation.
+ // Surround the expression in with statements to inject our command line API so that
+ // the window object properties still take more precedent than our API functions.
+
+ injectCommandLineAPI = injectCommandLineAPI && !("__commandLineAPI" in inspectedWindow);
+ var injectScopeChain = scopeChain && scopeChain.length && !("__scopeChainForEval" in inspectedWindow);
+
+ try {
+ var prefix = "";
+ var suffix = "";
+ if (injectCommandLineAPI) {
+ inspectedWindow.__commandLineAPI = new CommandLineAPI(this._commandLineAPIImpl, isEvalOnCallFrame ? object : null);
+ prefix = "with (__commandLineAPI || { __proto__: null }) {";
+ suffix = "}";
+ }
+ if (injectScopeChain) {
+ inspectedWindow.__scopeChainForEval = scopeChain;
+ for (var i = 0; i < scopeChain.length; ++i) {
+ prefix = "with (__scopeChainForEval[" + i + "] || { __proto__: null }) {" + (suffix ? " " : "") + prefix;
+ if (suffix)
+ suffix += " }";
+ else
+ suffix = "}";
+ }
+ }
+
+ if (prefix)
+ expression = prefix + "\n" + expression + "\n" + suffix;
+ var wrappedResult = evalFunction ? InjectedScriptHost.callFunction(evalFunction, object, [expression]) : InjectedScriptHost.evaluateWithExceptionDetails(expression);
+ if (objectGroup === "console" && !wrappedResult.exceptionDetails)
+ this._lastResult = wrappedResult.result;
+ return wrappedResult;
+ } finally {
+ if (injectCommandLineAPI)
+ delete inspectedWindow.__commandLineAPI;
+ if (injectScopeChain)
+ delete inspectedWindow.__scopeChainForEval;
+ }
+ },
+
+ /**
+ * @param {?Object} callFrame
+ * @param {number} asyncOrdinal
+ * @return {!Array.<!InjectedScript.CallFrameProxy>|boolean}
+ */
+ wrapCallFrames: function(callFrame, asyncOrdinal)
+ {
+ if (!callFrame)
+ return false;
+
+ var result = [];
+ var depth = 0;
+ do {
+ result[depth] = new InjectedScript.CallFrameProxy(depth, callFrame, asyncOrdinal);
+ callFrame = callFrame.caller;
+ ++depth;
+ } while (callFrame);
+ return result;
+ },
+
+ /**
+ * @param {!Object} topCallFrame
+ * @param {!Array.<!Object>} asyncCallStacks
+ * @param {string} callFrameId
+ * @param {string} expression
+ * @param {string} objectGroup
+ * @param {boolean} injectCommandLineAPI
+ * @param {boolean} returnByValue
+ * @param {boolean} generatePreview
+ * @return {*}
+ */
+ evaluateOnCallFrame: function(topCallFrame, asyncCallStacks, callFrameId, expression, objectGroup, injectCommandLineAPI, returnByValue, generatePreview)
+ {
+ var parsedCallFrameId = nullifyObjectProto(InjectedScriptHost.eval("(" + callFrameId + ")"));
+ var callFrame = this._callFrameForParsedId(topCallFrame, parsedCallFrameId, asyncCallStacks);
+ if (!callFrame)
+ return "Could not find call frame with given id";
+ if (parsedCallFrameId["asyncOrdinal"])
+ return this._evaluateAndWrap(null, null, expression, objectGroup, false, injectCommandLineAPI, returnByValue, generatePreview, callFrame.scopeChain);
+ return this._evaluateAndWrap(callFrame.evaluateWithExceptionDetails, callFrame, expression, objectGroup, true, injectCommandLineAPI, returnByValue, generatePreview);
+ },
+
+ /**
+ * @param {!Object} topCallFrame
+ * @param {string} callFrameId
+ * @return {*}
+ */
+ restartFrame: function(topCallFrame, callFrameId)
+ {
+ var callFrame = this._callFrameForId(topCallFrame, callFrameId);
+ if (!callFrame)
+ return "Could not find call frame with given id";
+ var result = callFrame.restart();
+ if (result === false)
+ result = "Restart frame is not supported";
+ return result;
+ },
+
+ /**
+ * @param {!Object} topCallFrame
+ * @param {string} callFrameId
+ * @return {*} a stepIn position array ready for protocol JSON or a string error
+ */
+ getStepInPositions: function(topCallFrame, callFrameId)
+ {
+ var callFrame = this._callFrameForId(topCallFrame, callFrameId);
+ if (!callFrame)
+ return "Could not find call frame with given id";
+ var stepInPositionsUnpacked = JSON.parse(callFrame.stepInPositions);
+ if (typeof stepInPositionsUnpacked !== "object")
+ return "Step in positions not available";
+ return stepInPositionsUnpacked;
+ },
+
+ /**
+ * Either callFrameId or functionObjectId must be specified.
+ * @param {!Object} topCallFrame
+ * @param {string|boolean} callFrameId or false
+ * @param {string|boolean} functionObjectId or false
+ * @param {number} scopeNumber
+ * @param {string} variableName
+ * @param {string} newValueJsonString RuntimeAgent.CallArgument structure serialized as string
+ * @return {string|undefined} undefined if success or an error message
+ */
+ setVariableValue: function(topCallFrame, callFrameId, functionObjectId, scopeNumber, variableName, newValueJsonString)
+ {
+ var setter;
+ if (typeof callFrameId === "string") {
+ var callFrame = this._callFrameForId(topCallFrame, callFrameId);
+ if (!callFrame)
+ return "Could not find call frame with given id";
+ setter = bind(callFrame.setVariableValue, callFrame);
+ } else {
+ var parsedFunctionId = this._parseObjectId(/** @type {string} */ (functionObjectId));
+ var func = this._objectForId(parsedFunctionId);
+ if (typeof func !== "function")
+ return "Cannot resolve function by id.";
+ setter = bind(InjectedScriptHost.setFunctionVariableValue, InjectedScriptHost, func);
+ }
+ var newValueJson;
+ try {
+ newValueJson = InjectedScriptHost.eval("(" + newValueJsonString + ")");
+ } catch (e) {
+ return "Failed to parse new value JSON " + newValueJsonString + " : " + e;
+ }
+ var resolvedValue;
+ try {
+ resolvedValue = this._resolveCallArgument(newValueJson);
+ } catch (e) {
+ return toString(e);
+ }
+ try {
+ setter(scopeNumber, variableName, resolvedValue);
+ } catch (e) {
+ return "Failed to change variable value: " + e;
+ }
+ return undefined;
+ },
+
+ /**
+ * @param {!Object} topCallFrame
+ * @param {string} callFrameId
+ * @return {?Object}
+ */
+ _callFrameForId: function(topCallFrame, callFrameId)
+ {
+ var parsedCallFrameId = nullifyObjectProto(InjectedScriptHost.eval("(" + callFrameId + ")"));
+ return this._callFrameForParsedId(topCallFrame, parsedCallFrameId, []);
+ },
+
+ /**
+ * @param {!Object} topCallFrame
+ * @param {!Object} parsedCallFrameId
+ * @param {!Array.<!Object>} asyncCallStacks
+ * @return {?Object}
+ */
+ _callFrameForParsedId: function(topCallFrame, parsedCallFrameId, asyncCallStacks)
+ {
+ var asyncOrdinal = parsedCallFrameId["asyncOrdinal"]; // 1-based index
+ if (asyncOrdinal)
+ topCallFrame = asyncCallStacks[asyncOrdinal - 1];
+ var ordinal = parsedCallFrameId["ordinal"];
+ var callFrame = topCallFrame;
+ while (--ordinal >= 0 && callFrame)
+ callFrame = callFrame.caller;
+ return callFrame;
+ },
+
+ /**
+ * @param {!Object} objectId
+ * @return {!Object|symbol}
+ */
+ _objectForId: function(objectId)
+ {
+ return this._idToWrappedObject[objectId.id];
+ },
+
+ /**
+ * @param {string} objectId
+ * @return {!Object|symbol}
+ */
+ findObjectById: function(objectId)
+ {
+ var parsedObjectId = this._parseObjectId(objectId);
+ return this._objectForId(parsedObjectId);
+ },
+
+ /**
+ * @param {string} objectId
+ * @return {?Node}
+ */
+ nodeForObjectId: function(objectId)
+ {
+ var object = this.findObjectById(objectId);
+ if (!object || this._subtype(object) !== "node")
+ return null;
+ return /** @type {!Node} */ (object);
+ },
+
+ /**
+ * @param {string} name
+ * @return {!Object}
+ */
+ module: function(name)
+ {
+ return this._modules[name];
+ },
+
+ /**
+ * @param {string} name
+ * @param {string} source
+ * @return {?Object}
+ */
+ injectModule: function(name, source)
+ {
+ delete this._modules[name];
+ var moduleFunction = InjectedScriptHost.eval("(" + source + ")");
+ if (typeof moduleFunction !== "function") {
+ inspectedWindow.console.error("Web Inspector error: A function was expected for module %s evaluation", name);
+ return null;
+ }
+ var module = InjectedScriptHost.callFunction(moduleFunction, inspectedWindow, [InjectedScriptHost, inspectedWindow, injectedScriptId, this]);
+ this._modules[name] = module;
+ return module;
+ },
+
+ /**
+ * @param {*} object
+ * @return {boolean}
+ */
+ _isDefined: function(object)
+ {
+ return !!object || this._isHTMLAllCollection(object);
+ },
+
+ /**
+ * @param {*} object
+ * @return {boolean}
+ */
+ _isHTMLAllCollection: function(object)
+ {
+ // document.all is reported as undefined, but we still want to process it.
+ return (typeof object === "undefined") && InjectedScriptHost.isHTMLAllCollection(object);
+ },
+
+ /**
+ * @param {*} obj
+ * @return {?string}
+ */
+ _subtype: function(obj)
+ {
+ if (obj === null)
+ return "null";
+
+ if (this.isPrimitiveValue(obj))
+ return null;
+
+ var subtype = InjectedScriptHost.subtype(obj);
+ if (subtype)
+ return subtype;
+
+ if (isArrayLike(obj))
+ return "array";
+
+ // If owning frame has navigated to somewhere else window properties will be undefined.
+ return null;
+ },
+
+ /**
+ * @param {*} obj
+ * @return {?string}
+ */
+ _describe: function(obj)
+ {
+ if (this.isPrimitiveValue(obj))
+ return null;
+
+ var subtype = this._subtype(obj);
+
+ if (subtype === "regexp")
+ return toString(obj);
+
+ if (subtype === "date")
+ return toString(obj);
+
+ if (subtype === "node") {
+ var description = obj.nodeName.toLowerCase();
+ switch (obj.nodeType) {
+ case 1 /* Node.ELEMENT_NODE */:
+ description += obj.id ? "#" + obj.id : "";
+ var className = obj.className;
+ description += (className && typeof className === "string") ? "." + className.trim().replace(/\s+/g, ".") : "";
+ break;
+ case 10 /*Node.DOCUMENT_TYPE_NODE */:
+ description = "<!DOCTYPE " + description + ">";
+ break;
+ }
+ return description;
+ }
+
+ var className = InjectedScriptHost.internalConstructorName(obj);
+ if (subtype === "array") {
+ if (typeof obj.length === "number")
+ className += "[" + obj.length + "]";
+ return className;
+ }
+
+ // NodeList in JSC is a function, check for array prior to this.
+ if (typeof obj === "function")
+ return toString(obj);
+
+ if (isSymbol(obj)) {
+ try {
+ return InjectedScriptHost.callFunction(Symbol.prototype.toString, obj) || "Symbol";
+ } catch (e) {
+ return "Symbol";
+ }
+ }
+
+ if (obj instanceof Error && !!obj.message)
+ return className + ": " + obj.message;
+
+ return className;
+ }
+}
+
+/**
+ * @type {!InjectedScript}
+ * @const
+ */
+var injectedScript = new InjectedScript();
+
+/**
+ * @constructor
+ * @param {*} object
+ * @param {string=} objectGroupName
+ * @param {boolean=} forceValueType
+ * @param {boolean=} generatePreview
+ * @param {?Array.<string>=} columnNames
+ * @param {boolean=} isTable
+ */
+InjectedScript.RemoteObject = function(object, objectGroupName, forceValueType, generatePreview, columnNames, isTable)
+{
+ this.type = typeof object;
+ if (injectedScript.isPrimitiveValue(object) || object === null || forceValueType) {
+ // We don't send undefined values over JSON.
+ if (this.type !== "undefined")
+ this.value = object;
+
+ // Null object is object with 'null' subtype.
+ if (object === null)
+ this.subtype = "null";
+
+ // Provide user-friendly number values.
+ if (this.type === "number") {
+ this.description = toStringDescription(object);
+ // Override "value" property for values that can not be JSON-stringified.
+ switch (this.description) {
+ case "NaN":
+ case "Infinity":
+ case "-Infinity":
+ case "-0":
+ this.value = this.description;
+ break;
+ }
+ }
+
+ return;
+ }
+
+ object = /** @type {!Object} */ (object);
+
+ this.objectId = injectedScript._bind(object, objectGroupName);
+ var subtype = injectedScript._subtype(object);
+ if (subtype)
+ this.subtype = subtype;
+ var className = InjectedScriptHost.internalConstructorName(object);
+ if (className)
+ this.className = className;
+ this.description = injectedScript._describe(object);
+
+ if (generatePreview && (this.type === "object" || injectedScript._isHTMLAllCollection(object)))
+ this.preview = this._generatePreview(object, undefined, columnNames, isTable);
+}
+
+InjectedScript.RemoteObject.prototype = {
+ /**
+ * @param {!Object} object
+ * @param {?Array.<string>=} firstLevelKeys
+ * @param {?Array.<string>=} secondLevelKeys
+ * @param {boolean=} isTable
+ * @return {!RuntimeAgent.ObjectPreview} preview
+ */
+ _generatePreview: function(object, firstLevelKeys, secondLevelKeys, isTable)
+ {
+ var preview = { __proto__: null };
+ preview.lossless = true;
+ preview.overflow = false;
+ preview.properties = [];
+
+ var firstLevelKeysCount = firstLevelKeys ? firstLevelKeys.length : 0;
+
+ var propertiesThreshold = {
+ properties: isTable ? 1000 : max(5, firstLevelKeysCount),
+ indexes: isTable ? 1000 : max(100, firstLevelKeysCount)
+ };
+
+ try {
+ var descriptors = injectedScript._propertyDescriptors(object);
+
+ if (firstLevelKeys) {
+ var nameToDescriptors = { __proto__: null };
+ for (var i = 0; i < descriptors.length; ++i) {
+ var descriptor = descriptors[i];
+ nameToDescriptors["#" + descriptor.name] = descriptor;
+ }
+ descriptors = [];
+ for (var i = 0; i < firstLevelKeys.length; ++i)
+ descriptors[i] = nameToDescriptors["#" + firstLevelKeys[i]];
+ }
+
+ this._appendPropertyDescriptors(preview, descriptors, propertiesThreshold, secondLevelKeys, isTable);
+ if (propertiesThreshold.indexes < 0 || propertiesThreshold.properties < 0)
+ return preview;
+
+ // Add internal properties to preview.
+ var internalProperties = InjectedScriptHost.getInternalProperties(object) || [];
+ for (var i = 0; i < internalProperties.length; ++i) {
+ internalProperties[i] = nullifyObjectProto(internalProperties[i]);
+ internalProperties[i].enumerable = true;
+ }
+ this._appendPropertyDescriptors(preview, internalProperties, propertiesThreshold, secondLevelKeys, isTable);
+
+ } catch (e) {
+ preview.lossless = false;
+ }
+
+ return preview;
+ },
+
+ /**
+ * @param {!RuntimeAgent.ObjectPreview} preview
+ * @param {!Array.<!Object>} descriptors
+ * @param {!Object} propertiesThreshold
+ * @param {?Array.<string>=} secondLevelKeys
+ * @param {boolean=} isTable
+ */
+ _appendPropertyDescriptors: function(preview, descriptors, propertiesThreshold, secondLevelKeys, isTable)
+ {
+ for (var i = 0; i < descriptors.length; ++i) {
+ if (propertiesThreshold.indexes < 0 || propertiesThreshold.properties < 0)
+ break;
+
+ var descriptor = descriptors[i];
+ if (!descriptor)
+ continue;
+ if (descriptor.wasThrown) {
+ preview.lossless = false;
+ continue;
+ }
+ if (!descriptor.enumerable && !descriptor.isOwn)
+ continue;
+
+ var name = descriptor.name;
+ if (name === "__proto__")
+ continue;
+ if (this.subtype === "array" && name === "length")
+ continue;
+
+ if (!("value" in descriptor)) {
+ preview.lossless = false;
+ this._appendPropertyPreview(preview, { name: name, type: "accessor", __proto__: null }, propertiesThreshold);
+ continue;
+ }
+
+ var value = descriptor.value;
+ if (value === null) {
+ this._appendPropertyPreview(preview, { name: name, type: "object", subtype: "null", value: "null", __proto__: null }, propertiesThreshold);
+ continue;
+ }
+
+ var type = typeof value;
+ if (!descriptor.enumerable && type === "function")
+ continue;
+ if (type === "undefined" && injectedScript._isHTMLAllCollection(value))
+ type = "object";
+
+ var maxLength = 100;
+ if (InjectedScript.primitiveTypes[type]) {
+ if (type === "string" && value.length > maxLength) {
+ value = this._abbreviateString(value, maxLength, true);
+ preview.lossless = false;
+ }
+ this._appendPropertyPreview(preview, { name: name, type: type, value: toStringDescription(value), __proto__: null }, propertiesThreshold);
+ continue;
+ }
+
+ var property = { name: name, type: type, __proto__: null };
+ var subtype = injectedScript._subtype(value);
+ if (subtype)
+ property.subtype = subtype;
+
+ if (secondLevelKeys === null || secondLevelKeys) {
+ var subPreview = this._generatePreview(value, secondLevelKeys || undefined, undefined, isTable);
+ property.valuePreview = subPreview;
+ if (!subPreview.lossless)
+ preview.lossless = false;
+ if (subPreview.overflow)
+ preview.overflow = true;
+ } else {
+ var description = "";
+ if (type !== "function")
+ description = this._abbreviateString(/** @type {string} */ (injectedScript._describe(value)), maxLength, subtype === "regexp");
+ property.value = description;
+ preview.lossless = false;
+ }
+ this._appendPropertyPreview(preview, property, propertiesThreshold);
+ }
+ },
+
+ /**
+ * @param {!RuntimeAgent.ObjectPreview} preview
+ * @param {!Object} property
+ * @param {!Object} propertiesThreshold
+ */
+ _appendPropertyPreview: function(preview, property, propertiesThreshold)
+ {
+ if (toString(property.name >>> 0) === property.name)
+ propertiesThreshold.indexes--;
+ else
+ propertiesThreshold.properties--;
+ if (propertiesThreshold.indexes < 0 || propertiesThreshold.properties < 0) {
+ preview.overflow = true;
+ preview.lossless = false;
+ } else {
+ push(preview.properties, property);
+ }
+ },
+
+ /**
+ * @param {string} string
+ * @param {number} maxLength
+ * @param {boolean=} middle
+ * @return {string}
+ */
+ _abbreviateString: function(string, maxLength, middle)
+ {
+ if (string.length <= maxLength)
+ return string;
+ if (middle) {
+ var leftHalf = maxLength >> 1;
+ var rightHalf = maxLength - leftHalf - 1;
+ return string.substr(0, leftHalf) + "\u2026" + string.substr(string.length - rightHalf, rightHalf);
+ }
+ return string.substr(0, maxLength) + "\u2026";
+ },
+
+ __proto__: null
+}
+/**
+ * @constructor
+ * @param {number} ordinal
+ * @param {!Object} callFrame
+ * @param {number} asyncOrdinal
+ */
+InjectedScript.CallFrameProxy = function(ordinal, callFrame, asyncOrdinal)
+{
+ this.callFrameId = "{\"ordinal\":" + ordinal + ",\"injectedScriptId\":" + injectedScriptId + (asyncOrdinal ? ",\"asyncOrdinal\":" + asyncOrdinal : "") + "}";
+ this.functionName = (callFrame.type === "function" ? callFrame.functionName : "");
+ this.location = { scriptId: toString(callFrame.sourceID), lineNumber: callFrame.line, columnNumber: callFrame.column, __proto__: null };
+ this.scopeChain = this._wrapScopeChain(callFrame);
+ this.this = injectedScript._wrapObject(callFrame.thisObject, "backtrace");
+ if (callFrame.isAtReturn)
+ this.returnValue = injectedScript._wrapObject(callFrame.returnValue, "backtrace");
+}
+
+InjectedScript.CallFrameProxy.prototype = {
+ /**
+ * @param {!Object} callFrame
+ * @return {!Array.<!DebuggerAgent.Scope>}
+ */
+ _wrapScopeChain: function(callFrame)
+ {
+ var scopeChain = callFrame.scopeChain;
+ var scopeChainProxy = [];
+ for (var i = 0; i < scopeChain.length; ++i)
+ scopeChainProxy[i] = InjectedScript.CallFrameProxy._createScopeJson(callFrame.scopeType(i), scopeChain[i], "backtrace");
+ return scopeChainProxy;
+ },
+
+ __proto__: null
+}
+
+/**
+ * @param {number} scopeTypeCode
+ * @param {*} scopeObject
+ * @param {string} groupId
+ * @return {!DebuggerAgent.Scope}
+ */
+InjectedScript.CallFrameProxy._createScopeJson = function(scopeTypeCode, scopeObject, groupId)
+{
+ var GLOBAL_SCOPE = 0;
+ var LOCAL_SCOPE = 1;
+ var WITH_SCOPE = 2;
+ var CLOSURE_SCOPE = 3;
+ var CATCH_SCOPE = 4;
+
+ /** @type {!Object.<number, string>} */
+ var scopeTypeNames = { __proto__: null };
+ scopeTypeNames[GLOBAL_SCOPE] = "global";
+ scopeTypeNames[LOCAL_SCOPE] = "local";
+ scopeTypeNames[WITH_SCOPE] = "with";
+ scopeTypeNames[CLOSURE_SCOPE] = "closure";
+ scopeTypeNames[CATCH_SCOPE] = "catch";
+
+ return {
+ object: injectedScript._wrapObject(scopeObject, groupId),
+ type: /** @type {!DebuggerAgent.ScopeType} */ (scopeTypeNames[scopeTypeCode]),
+ __proto__: null
+ };
+}
+
+/**
+ * @constructor
+ * @param {!CommandLineAPIImpl} commandLineAPIImpl
+ * @param {?Object} callFrame
+ */
+function CommandLineAPI(commandLineAPIImpl, callFrame)
+{
+ /**
+ * @param {string} member
+ * @return {boolean}
+ */
+ function inScopeVariables(member)
+ {
+ if (!callFrame)
+ return false;
+
+ var scopeChain = callFrame.scopeChain;
+ for (var i = 0; i < scopeChain.length; ++i) {
+ if (member in scopeChain[i])
+ return true;
+ }
+ return false;
+ }
+
+ /**
+ * @param {string} name The name of the method for which a toString method should be generated.
+ * @return {function():string}
+ */
+ function customToStringMethod(name)
+ {
+ return function()
+ {
+ var funcArgsSyntax = "";
+ try {
+ var funcSyntax = "" + commandLineAPIImpl[name];
+ funcSyntax = funcSyntax.replace(/\n/g, " ");
+ funcSyntax = funcSyntax.replace(/^function[^\(]*\(([^\)]*)\).*$/, "$1");
+ funcSyntax = funcSyntax.replace(/\s*,\s*/g, ", ");
+ funcSyntax = funcSyntax.replace(/\bopt_(\w+)\b/g, "[$1]");
+ funcArgsSyntax = funcSyntax.trim();
+ } catch (e) {
+ }
+ return "function " + name + "(" + funcArgsSyntax + ") { [Command Line API] }";
+ };
+ }
+
+ for (var i = 0; i < CommandLineAPI.members_.length; ++i) {
+ var member = CommandLineAPI.members_[i];
+ if (member in inspectedWindow || inScopeVariables(member))
+ continue;
+
+ this[member] = bind(commandLineAPIImpl[member], commandLineAPIImpl);
+ this[member].toString = customToStringMethod(member);
+ }
+
+ for (var i = 0; i < 5; ++i) {
+ var member = "$" + i;
+ if (member in inspectedWindow || inScopeVariables(member))
+ continue;
+
+ this.__defineGetter__("$" + i, bind(commandLineAPIImpl._inspectedObject, commandLineAPIImpl, i));
+ }
+
+ this.$_ = injectedScript._lastResult;
+
+ this.__proto__ = null;
+}
+
+// NOTE: Please keep the list of API methods below snchronized to that in WebInspector.RuntimeModel!
+// NOTE: Argument names of these methods will be printed in the console, so use pretty names!
+/**
+ * @type {!Array.<string>}
+ * @const
+ */
+CommandLineAPI.members_ = [
+ "$", "$$", "$x", "dir", "dirxml", "keys", "values", "profile", "profileEnd",
+ "monitorEvents", "unmonitorEvents", "inspect", "copy", "clear", "getEventListeners",
+ "debug", "undebug", "monitor", "unmonitor", "table"
+];
+
+/**
+ * @constructor
+ */
+function CommandLineAPIImpl()
+{
+}
+
+CommandLineAPIImpl.prototype = {
+ /**
+ * @param {string} selector
+ * @param {!Node=} opt_startNode
+ * @return {*}
+ */
+ $: function (selector, opt_startNode)
+ {
+ if (this._canQuerySelectorOnNode(opt_startNode))
+ return opt_startNode.querySelector(selector);
+
+ return inspectedWindow.document.querySelector(selector);
+ },
+
+ /**
+ * @param {string} selector
+ * @param {!Node=} opt_startNode
+ * @return {*}
+ */
+ $$: function (selector, opt_startNode)
+ {
+ if (this._canQuerySelectorOnNode(opt_startNode))
+ return opt_startNode.querySelectorAll(selector);
+ return inspectedWindow.document.querySelectorAll(selector);
+ },
+
+ /**
+ * @param {!Node=} node
+ * @return {boolean}
+ */
+ _canQuerySelectorOnNode: function(node)
+ {
+ return !!node && InjectedScriptHost.subtype(node) === "node" && (node.nodeType === Node.ELEMENT_NODE || node.nodeType === Node.DOCUMENT_NODE || node.nodeType === Node.DOCUMENT_FRAGMENT_NODE);
+ },
+
+ /**
+ * @param {string} xpath
+ * @param {!Node=} opt_startNode
+ * @return {*}
+ */
+ $x: function(xpath, opt_startNode)
+ {
+ var doc = (opt_startNode && opt_startNode.ownerDocument) || inspectedWindow.document;
+ var result = doc.evaluate(xpath, opt_startNode || doc, null, XPathResult.ANY_TYPE, null);
+ switch (result.resultType) {
+ case XPathResult.NUMBER_TYPE:
+ return result.numberValue;
+ case XPathResult.STRING_TYPE:
+ return result.stringValue;
+ case XPathResult.BOOLEAN_TYPE:
+ return result.booleanValue;
+ default:
+ var nodes = [];
+ var node;
+ while (node = result.iterateNext())
+ push(nodes, node);
+ return nodes;
+ }
+ },
+
+ /**
+ * @return {*}
+ */
+ dir: function(var_args)
+ {
+ return InjectedScriptHost.callFunction(inspectedWindow.console.dir, inspectedWindow.console, slice(arguments));
+ },
+
+ /**
+ * @return {*}
+ */
+ dirxml: function(var_args)
+ {
+ return InjectedScriptHost.callFunction(inspectedWindow.console.dirxml, inspectedWindow.console, slice(arguments));
+ },
+
+ /**
+ * @return {!Array.<string>}
+ */
+ keys: function(object)
+ {
+ return Object.keys(object);
+ },
+
+ /**
+ * @return {!Array.<*>}
+ */
+ values: function(object)
+ {
+ var result = [];
+ for (var key in object)
+ push(result, object[key]);
+ return result;
+ },
+
+ /**
+ * @return {*}
+ */
+ profile: function(opt_title)
+ {
+ return InjectedScriptHost.callFunction(inspectedWindow.console.profile, inspectedWindow.console, slice(arguments));
+ },
+
+ /**
+ * @return {*}
+ */
+ profileEnd: function(opt_title)
+ {
+ return InjectedScriptHost.callFunction(inspectedWindow.console.profileEnd, inspectedWindow.console, slice(arguments));
+ },
+
+ /**
+ * @param {!Object} object
+ * @param {!Array.<string>|string=} opt_types
+ */
+ monitorEvents: function(object, opt_types)
+ {
+ if (!object || !object.addEventListener || !object.removeEventListener)
+ return;
+ var types = this._normalizeEventTypes(opt_types);
+ for (var i = 0; i < types.length; ++i) {
+ object.removeEventListener(types[i], this._logEvent, false);
+ object.addEventListener(types[i], this._logEvent, false);
+ }
+ },
+
+ /**
+ * @param {!Object} object
+ * @param {!Array.<string>|string=} opt_types
+ */
+ unmonitorEvents: function(object, opt_types)
+ {
+ if (!object || !object.addEventListener || !object.removeEventListener)
+ return;
+ var types = this._normalizeEventTypes(opt_types);
+ for (var i = 0; i < types.length; ++i)
+ object.removeEventListener(types[i], this._logEvent, false);
+ },
+
+ /**
+ * @param {*} object
+ * @return {*}
+ */
+ inspect: function(object)
+ {
+ return injectedScript._inspect(object);
+ },
+
+ copy: function(object)
+ {
+ var string;
+ if (injectedScript._subtype(object) === "node") {
+ string = object.outerHTML;
+ } else if (injectedScript.isPrimitiveValue(object)) {
+ string = toString(object);
+ } else {
+ try {
+ string = JSON.stringify(object, null, " ");
+ } catch (e) {
+ string = toString(object);
+ }
+ }
+
+ var hints = { copyToClipboard: true, __proto__: null };
+ var remoteObject = injectedScript._wrapObject(string, "")
+ InjectedScriptHost.inspect(remoteObject, hints);
+ },
+
+ clear: function()
+ {
+ InjectedScriptHost.clearConsoleMessages();
+ },
+
+ /**
+ * @param {!Node} node
+ * @return {!{type: string, listener: function(), useCapture: boolean, remove: function()}|undefined}
+ */
+ getEventListeners: function(node)
+ {
+ var result = nullifyObjectProto(InjectedScriptHost.getEventListeners(node));
+ if (!result)
+ return result;
+ /** @this {{type: string, listener: function(), useCapture: boolean}} */
+ var removeFunc = function()
+ {
+ node.removeEventListener(this.type, this.listener, this.useCapture);
+ }
+ for (var type in result) {
+ var listeners = result[type];
+ for (var i = 0, listener; listener = listeners[i]; ++i) {
+ listener["type"] = type;
+ listener["remove"] = removeFunc;
+ }
+ }
+ return result;
+ },
+
+ debug: function(fn)
+ {
+ InjectedScriptHost.debugFunction(fn);
+ },
+
+ undebug: function(fn)
+ {
+ InjectedScriptHost.undebugFunction(fn);
+ },
+
+ monitor: function(fn)
+ {
+ InjectedScriptHost.monitorFunction(fn);
+ },
+
+ unmonitor: function(fn)
+ {
+ InjectedScriptHost.unmonitorFunction(fn);
+ },
+
+ table: function(data, opt_columns)
+ {
+ InjectedScriptHost.callFunction(inspectedWindow.console.table, inspectedWindow.console, slice(arguments));
+ },
+
+ /**
+ * @param {number} num
+ */
+ _inspectedObject: function(num)
+ {
+ return InjectedScriptHost.inspectedObject(num);
+ },
+
+ /**
+ * @param {!Array.<string>|string=} types
+ * @return {!Array.<string>}
+ */
+ _normalizeEventTypes: function(types)
+ {
+ if (typeof types === "undefined")
+ types = ["mouse", "key", "touch", "control", "load", "unload", "abort", "error", "select", "change", "submit", "reset", "focus", "blur", "resize", "scroll", "search", "devicemotion", "deviceorientation"];
+ else if (typeof types === "string")
+ types = [types];
+
+ var result = [];
+ for (var i = 0; i < types.length; ++i) {
+ if (types[i] === "mouse")
+ push(result, "mousedown", "mouseup", "click", "dblclick", "mousemove", "mouseover", "mouseout", "mousewheel");
+ else if (types[i] === "key")
+ push(result, "keydown", "keyup", "keypress", "textInput");
+ else if (types[i] === "touch")
+ push(result, "touchstart", "touchmove", "touchend", "touchcancel");
+ else if (types[i] === "control")
+ push(result, "resize", "scroll", "zoom", "focus", "blur", "select", "change", "submit", "reset");
+ else
+ push(result, types[i]);
+ }
+ return result;
+ },
+
+ /**
+ * @param {!Event} event
+ */
+ _logEvent: function(event)
+ {
+ inspectedWindow.console.log(event.type, event);
+ }
+}
+
+injectedScript._commandLineAPIImpl = new CommandLineAPIImpl();
+return injectedScript;
+})

Powered by Google App Engine
This is Rietveld 408576698