summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorpfeldman@chromium.org <pfeldman@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2009-03-30 08:19:40 +0000
committerpfeldman@chromium.org <pfeldman@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2009-03-30 08:19:40 +0000
commit46ce4c7d272138e15282ee102f9c6ff0508353c7 (patch)
treeb0e00ae8d60ae169aa27f7cb8a522a8b255c21d1
parent646dce40970192643ce533518888dfaf9c736815 (diff)
downloadchromium_src-46ce4c7d272138e15282ee102f9c6ff0508353c7.zip
chromium_src-46ce4c7d272138e15282ee102f9c6ff0508353c7.tar.gz
chromium_src-46ce4c7d272138e15282ee102f9c6ff0508353c7.tar.bz2
DevTools: Implement basic DOM edit operations.
Review URL: http://codereview.chromium.org/56005 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@12765 0039d316-1c4b-4281-b951-d872f2087c98
-rw-r--r--webkit/glue/devtools/debugger_agent.h6
-rw-r--r--webkit/glue/devtools/devtools_rpc.h72
-rw-r--r--webkit/glue/devtools/devtools_rpc_js.h447
-rw-r--r--webkit/glue/devtools/dom_agent.h17
-rw-r--r--webkit/glue/devtools/dom_agent_impl.cc28
-rw-r--r--webkit/glue/devtools/dom_agent_impl.h11
-rw-r--r--webkit/glue/devtools/js/devtools.js2
-rw-r--r--webkit/glue/devtools/js/devtools_host_stub.js29
-rw-r--r--webkit/glue/devtools/js/dom_agent.js168
-rw-r--r--webkit/glue/devtools/net_agent.h5
-rw-r--r--webkit/glue/devtools/tools_agent.h5
11 files changed, 537 insertions, 253 deletions
diff --git a/webkit/glue/devtools/debugger_agent.h b/webkit/glue/devtools/debugger_agent.h
index a3d6fe0..213cf28 100644
--- a/webkit/glue/devtools/debugger_agent.h
+++ b/webkit/glue/devtools/debugger_agent.h
@@ -7,13 +7,15 @@
#include "webkit/glue/devtools/devtools_rpc.h"
-#define DEBUGGER_AGENT_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3) \
+#define DEBUGGER_AGENT_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3, \
+ METHOD4) \
/* Stops v8 execution as soon as it gets control. */ \
METHOD0(DebugBreak)
DEFINE_RPC_CLASS(DebuggerAgent, DEBUGGER_AGENT_STRUCT)
-#define DEBUGGER_AGENT_DELEGATE_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3) \
+#define DEBUGGER_AGENT_DELEGATE_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3, \
+ METHOD4) \
METHOD1(DebuggerOutput, std::string /* output text */)
DEFINE_RPC_CLASS(DebuggerAgentDelegate, DEBUGGER_AGENT_DELEGATE_STRUCT)
diff --git a/webkit/glue/devtools/devtools_rpc.h b/webkit/glue/devtools/devtools_rpc.h
index 6f134ba..1fca50d 100644
--- a/webkit/glue/devtools/devtools_rpc.h
+++ b/webkit/glue/devtools/devtools_rpc.h
@@ -6,7 +6,7 @@
// implementation. The client is responsible for defining the Rpc-enabled
// interface in terms of its macros:
//
-// #define MYAPI_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3)
+// #define MYAPI_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3, METHOD4)
// METHOD0(Method1)
// METHOD3(Method2, int, String, Value)
// METHOD1(Method3, int)
@@ -122,10 +122,17 @@ struct RpcTypeTrait<std::string> {
RpcTypeTrait<T2>::ApiType t2, \
RpcTypeTrait<T3>::ApiType t3) = 0;
+#define TOOLS_RPC_API_METHOD4(Method, T1, T2, T3, T4) \
+ virtual void Method(RpcTypeTrait<T1>::ApiType t1, \
+ RpcTypeTrait<T2>::ApiType t2, \
+ RpcTypeTrait<T3>::ApiType t3, \
+ RpcTypeTrait<T4>::ApiType t4) = 0;
+
#define TOOLS_RPC_ENUM_LITERAL0(Method) METHOD_##Method,
#define TOOLS_RPC_ENUM_LITERAL1(Method, T1) METHOD_##Method,
#define TOOLS_RPC_ENUM_LITERAL2(Method, T1, T2) METHOD_##Method,
#define TOOLS_RPC_ENUM_LITERAL3(Method, T1, T2, T3) METHOD_##Method,
+#define TOOLS_RPC_ENUM_LITERAL4(Method, T1, T2, T3, T4) METHOD_##Method,
///////////////////////////////////////////////////////
// RPC stub method implementations
@@ -154,6 +161,15 @@ struct RpcTypeTrait<std::string> {
&t3); \
}
+#define TOOLS_RPC_STUB_METHOD4(Method, T1, T2, T3, T4) \
+ virtual void Method(RpcTypeTrait<T1>::ApiType t1, \
+ RpcTypeTrait<T2>::ApiType t2, \
+ RpcTypeTrait<T3>::ApiType t3, \
+ RpcTypeTrait<T4>::ApiType t4) { \
+ InvokeAsync(RpcTypeToNumber<CLASS>::number, METHOD_##Method, &t1, &t2, \
+ &t3, &t4); \
+ }
+
///////////////////////////////////////////////////////
// RPC dispatch method implementations
@@ -201,6 +217,25 @@ case CLASS::METHOD_##Method: { \
return true; \
}
+#define TOOLS_RPC_DISPATCH4(Method, T1, T2, T3, T4) \
+case CLASS::METHOD_##Method: { \
+ RpcTypeTrait<T1>::DispatchType t1; \
+ RpcTypeTrait<T2>::DispatchType t2; \
+ RpcTypeTrait<T3>::DispatchType t3; \
+ RpcTypeTrait<T4>::DispatchType t4; \
+ DevToolsRpc::GetListValue(message, 2, &t1); \
+ DevToolsRpc::GetListValue(message, 3, &t2); \
+ DevToolsRpc::GetListValue(message, 4, &t3); \
+ DevToolsRpc::GetListValue(message, 5, &t4); \
+ delegate->Method( \
+ RpcTypeTrait<T1>::Pass(t1), \
+ RpcTypeTrait<T2>::Pass(t2), \
+ RpcTypeTrait<T3>::Pass(t3), \
+ RpcTypeTrait<T4>::Pass(t4) \
+ ); \
+ return true; \
+}
+
#define TOOLS_END_RPC_DISPATCH() \
}
@@ -214,15 +249,20 @@ class Class {\
~Class() {} \
\
enum MethodNames { \
- STRUCT(TOOLS_RPC_ENUM_LITERAL0, TOOLS_RPC_ENUM_LITERAL1, \
- TOOLS_RPC_ENUM_LITERAL2, TOOLS_RPC_ENUM_LITERAL3) \
+ STRUCT( \
+ TOOLS_RPC_ENUM_LITERAL0, \
+ TOOLS_RPC_ENUM_LITERAL1, \
+ TOOLS_RPC_ENUM_LITERAL2, \
+ TOOLS_RPC_ENUM_LITERAL3, \
+ TOOLS_RPC_ENUM_LITERAL4) \
}; \
\
STRUCT( \
TOOLS_RPC_API_METHOD0, \
TOOLS_RPC_API_METHOD1, \
TOOLS_RPC_API_METHOD2, \
- TOOLS_RPC_API_METHOD3) \
+ TOOLS_RPC_API_METHOD3, \
+ TOOLS_RPC_API_METHOD4) \
private: \
DISALLOW_COPY_AND_ASSIGN(Class); \
}; \
@@ -233,10 +273,11 @@ class Class##Stub : public Class, public DevToolsRpc { \
virtual ~Class##Stub() {} \
typedef Class CLASS; \
STRUCT( \
- TOOLS_RPC_STUB_METHOD0, \
- TOOLS_RPC_STUB_METHOD1, \
- TOOLS_RPC_STUB_METHOD2, \
- TOOLS_RPC_STUB_METHOD3) \
+ TOOLS_RPC_STUB_METHOD0, \
+ TOOLS_RPC_STUB_METHOD1, \
+ TOOLS_RPC_STUB_METHOD2, \
+ TOOLS_RPC_STUB_METHOD3, \
+ TOOLS_RPC_STUB_METHOD4) \
private: \
DISALLOW_COPY_AND_ASSIGN(Class##Stub); \
}; \
@@ -266,7 +307,8 @@ class Class##Dispatch { \
TOOLS_RPC_DISPATCH0, \
TOOLS_RPC_DISPATCH1, \
TOOLS_RPC_DISPATCH2, \
- TOOLS_RPC_DISPATCH3) \
+ TOOLS_RPC_DISPATCH3, \
+ TOOLS_RPC_DISPATCH4) \
default: return false; \
} \
} \
@@ -345,6 +387,18 @@ class DevToolsRpc {
SendValueMessage(message);
}
+ template<class T1, class T2, class T3, class T4>
+ void InvokeAsync(int class_id, int method, T1 t1, T2 t2, T3 t3, T4 t4) {
+ ListValue message;
+ message.Append(CreateValue(&class_id));
+ message.Append(CreateValue(&method));
+ message.Append(CreateValue(t1));
+ message.Append(CreateValue(t2));
+ message.Append(CreateValue(t3));
+ message.Append(CreateValue(t4));
+ SendValueMessage(message);
+ }
+
static Value* ParseMessage(const std::string& raw_msg);
static std::string Serialize(const Value& value);
static void GetListValue(const ListValue& message, int index, bool* value);
diff --git a/webkit/glue/devtools/devtools_rpc_js.h b/webkit/glue/devtools/devtools_rpc_js.h
index 45b34c7..6dffeca 100644
--- a/webkit/glue/devtools/devtools_rpc_js.h
+++ b/webkit/glue/devtools/devtools_rpc_js.h
@@ -1,206 +1,241 @@
-// Copyright (c) 2009 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-// Additional set of macros for the JS RPC.
-
-#ifndef WEBKIT_GLUE_DEVTOOLS_DEVTOOLS_RPC_JS_H_
-#define WEBKIT_GLUE_DEVTOOLS_DEVTOOLS_RPC_JS_H_
-
-#include <string>
-
-#include <wtf/OwnPtr.h>
-
-#include "base/basictypes.h"
-#include "base/logging.h"
-#include "base/values.h"
-#include "webkit/glue/cpp_bound_class.h"
-#include "webkit/glue/devtools/devtools_rpc.h"
-#include "webkit/glue/glue_util.h"
-#include "webkit/glue/webframe.h"
-
-///////////////////////////////////////////////////////
-// JS RPC binds and stubs
-
-template<typename T>
-struct RpcJsTypeTrait {
-};
-
-template<>
-struct RpcJsTypeTrait<bool> {
- static bool Pass(const CppVariant& var) {
- return var.ToBoolean();
- }
-};
-
-template<>
-struct RpcJsTypeTrait<int> {
- static int Pass(const CppVariant& var) {
- return var.ToInt32();
- }
-};
-
-template<>
-struct RpcJsTypeTrait<String> {
- static String Pass(const CppVariant& var) {
- return webkit_glue::StdStringToString(var.ToString());
- }
-};
-
-template<>
-struct RpcJsTypeTrait<std::string> {
- static std::string Pass(const CppVariant& var) {
- return var.ToString();
- }
-};
-
-#define TOOLS_RPC_JS_BIND_METHOD0(Method) \
- BindMethod(#Method, &OCLASS::Js##Method);
-
-#define TOOLS_RPC_JS_BIND_METHOD1(Method, T1) \
- BindMethod(#Method, &OCLASS::Js##Method);
-
-#define TOOLS_RPC_JS_BIND_METHOD2(Method, T1, T2) \
- BindMethod(#Method, &OCLASS::Js##Method);
-
-#define TOOLS_RPC_JS_BIND_METHOD3(Method, T1, T2, T3) \
- BindMethod(#Method, &OCLASS::Js##Method);
-
-#define TOOLS_RPC_JS_STUB_METHOD0(Method) \
- void Js##Method(const CppArgumentList& args, CppVariant* result) { \
- InvokeAsync(RpcTypeToNumber<CLASS>::number, METHOD_##Method); \
- result->SetNull(); \
- }
-
-#define TOOLS_RPC_JS_STUB_METHOD1(Method, T1) \
- void Js##Method(const CppArgumentList& args, CppVariant* result) { \
- T1 t1 = RpcJsTypeTrait<T1>::Pass(args[0]); \
- InvokeAsync(RpcTypeToNumber<CLASS>::number, METHOD_##Method, &t1); \
- result->SetNull(); \
- }
-
-#define TOOLS_RPC_JS_STUB_METHOD2(Method, T1, T2) \
- void Js##Method(const CppArgumentList& args, CppVariant* result) { \
- T1 t1 = RpcJsTypeTrait<T1>::Pass(args[0]); \
- T2 t2 = RpcJsTypeTrait<T2>::Pass(args[1]); \
- InvokeAsync(RpcTypeToNumber<CLASS>::number, METHOD_##Method, &t1, &t2); \
- result->SetNull(); \
- }
-
-#define TOOLS_RPC_JS_STUB_METHOD3(Method, T1, T2, T3) \
- void Js##Method(const CppArgumentList& args, CppVariant* result) { \
- T1 t1 = RpcJsTypeTrait<T1>::Pass(args[0]); \
- T2 t2 = RpcJsTypeTrait<T2>::Pass(args[1]); \
- T3 t3 = RpcJsTypeTrait<T3>::Pass(args[2]); \
- InvokeAsync(RpcTypeToNumber<CLASS>::number, METHOD_##Method, &t1, &t2, \
- &t3); \
- result->SetNull(); \
- }
-
-///////////////////////////////////////////////////////
-// JS RPC dispatch method implementations
-
-#define TOOLS_RPC_JS_DISPATCH0(Method) \
-case CLASS::METHOD_##Method: { \
- *expr = StringPrintf("%s.%s()", js_obj.c_str(), #Method); \
- return true; \
-}
-
-#define TOOLS_RPC_JS_DISPATCH1(Method, T1) \
-case CLASS::METHOD_##Method: { \
- Value* t1; \
- message.Get(2, &t1); \
- *expr = StringPrintf("%s.%s(%s)", js_obj.c_str(), #Method, \
- DevToolsRpc::Serialize(*t1).c_str()); \
- return true; \
-}
-
-#define TOOLS_RPC_JS_DISPATCH2(Method, T1, T2) \
-case CLASS::METHOD_##Method: { \
- Value* t1; \
- Value* t2; \
- message.Get(2, &t1); \
- message.Get(3, &t2); \
- *expr = StringPrintf("%s.%s(%s, %s)", js_obj.c_str(), #Method, \
- DevToolsRpc::Serialize(*t1).c_str(), \
- DevToolsRpc::Serialize(*t2).c_str()); \
- return true; \
-}
-
-#define TOOLS_RPC_JS_DISPATCH3(Method, T1, T2, T3) \
-case CLASS::METHOD_##Method: { \
- Value* t1; \
- Value* t2; \
- Value* t3; \
- message.Get(2, &t1); \
- message.Get(3, &t2); \
- message.Get(4, &t3); \
- *expr = StringPrintf("%s.%s(%s, %s, %s)", js_obj.c_str(), #Method, \
- DevToolsRpc::Serialize(*t1).c_str(), \
- DevToolsRpc::Serialize(*t2).c_str(), \
- DevToolsRpc::Serialize(*t3).c_str()); \
- return true; \
-}
-
-#define DEFINE_RPC_JS_DISPATCH(Class, STRUCT) \
-class Js##Class##Dispatch { \
- public: \
- explicit Js##Class##Dispatch(const std::wstring& classname) \
- : js_obj(WideToUTF8(classname)) {} \
- virtual ~Js##Class##Dispatch() {} \
- \
- bool Dispatch(const ListValue& message, std::string* expr) { \
- int class_id; \
- message.GetInteger(0, &class_id); \
- if (class_id != RpcTypeToNumber<Class>::number) { \
- return false; \
- } \
- int method; \
- message.GetInteger(1, &method); \
- typedef Class CLASS; \
- switch (method) { \
- STRUCT( \
- TOOLS_RPC_JS_DISPATCH0, \
- TOOLS_RPC_JS_DISPATCH1, \
- TOOLS_RPC_JS_DISPATCH2, \
- TOOLS_RPC_JS_DISPATCH3) \
- default: return false; \
- } \
- } \
- private: \
- std::string js_obj; \
- DISALLOW_COPY_AND_ASSIGN(Js##Class##Dispatch); \
-};
-
-///////////////////////////////////////////////////////
-// JS RPC main obj macro
-
-#define DEFINE_RPC_JS_BOUND_OBJ(Class, STRUCT, DClass, DELEGATE_STRUCT) \
-DEFINE_RPC_JS_DISPATCH(DClass, DELEGATE_STRUCT) \
-class Js##Class##BoundObj : public Class##Stub, \
- public CppBoundClass, \
- public Js##DClass##Dispatch { \
- public: \
- Js##Class##BoundObj(Delegate* rpc_delegate, WebFrame* frame, \
- const std::wstring& classname) \
- : Class##Stub(rpc_delegate), \
- Js##DClass##Dispatch(classname) { \
- BindToJavascript(frame, classname); \
- STRUCT( \
- TOOLS_RPC_JS_BIND_METHOD0, \
- TOOLS_RPC_JS_BIND_METHOD1, \
- TOOLS_RPC_JS_BIND_METHOD2, \
- TOOLS_RPC_JS_BIND_METHOD3) \
- } \
- virtual ~Js##Class##BoundObj() {} \
- typedef Js##Class##BoundObj OCLASS; \
- STRUCT( \
- TOOLS_RPC_JS_STUB_METHOD0, \
- TOOLS_RPC_JS_STUB_METHOD1, \
- TOOLS_RPC_JS_STUB_METHOD2, \
- TOOLS_RPC_JS_STUB_METHOD3) \
- private: \
- DISALLOW_COPY_AND_ASSIGN(Js##Class##BoundObj); \
-};
-
-#endif // WEBKIT_GLUE_DEVTOOLS_DEVTOOLS_RPC_JS_H_
+// Copyright (c) 2009 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+// Additional set of macros for the JS RPC.
+
+#ifndef WEBKIT_GLUE_DEVTOOLS_DEVTOOLS_RPC_JS_H_
+#define WEBKIT_GLUE_DEVTOOLS_DEVTOOLS_RPC_JS_H_
+
+#include <string>
+
+#include <wtf/OwnPtr.h>
+
+#include "base/basictypes.h"
+#include "base/logging.h"
+#include "base/values.h"
+#include "webkit/glue/cpp_bound_class.h"
+#include "webkit/glue/devtools/devtools_rpc.h"
+#include "webkit/glue/glue_util.h"
+#include "webkit/glue/webframe.h"
+
+///////////////////////////////////////////////////////
+// JS RPC binds and stubs
+
+template<typename T>
+struct RpcJsTypeTrait {
+};
+
+template<>
+struct RpcJsTypeTrait<bool> {
+ static bool Pass(const CppVariant& var) {
+ return var.ToBoolean();
+ }
+};
+
+template<>
+struct RpcJsTypeTrait<int> {
+ static int Pass(const CppVariant& var) {
+ return var.ToInt32();
+ }
+};
+
+template<>
+struct RpcJsTypeTrait<String> {
+ static String Pass(const CppVariant& var) {
+ return webkit_glue::StdStringToString(var.ToString());
+ }
+};
+
+template<>
+struct RpcJsTypeTrait<std::string> {
+ static std::string Pass(const CppVariant& var) {
+ return var.ToString();
+ }
+};
+
+#define TOOLS_RPC_JS_BIND_METHOD0(Method) \
+ BindMethod(#Method, &OCLASS::Js##Method);
+
+#define TOOLS_RPC_JS_BIND_METHOD1(Method, T1) \
+ BindMethod(#Method, &OCLASS::Js##Method);
+
+#define TOOLS_RPC_JS_BIND_METHOD2(Method, T1, T2) \
+ BindMethod(#Method, &OCLASS::Js##Method);
+
+#define TOOLS_RPC_JS_BIND_METHOD3(Method, T1, T2, T3) \
+ BindMethod(#Method, &OCLASS::Js##Method);
+
+#define TOOLS_RPC_JS_BIND_METHOD4(Method, T1, T2, T3, T4) \
+ BindMethod(#Method, &OCLASS::Js##Method);
+
+#define TOOLS_RPC_JS_STUB_METHOD0(Method) \
+ void Js##Method(const CppArgumentList& args, CppVariant* result) { \
+ InvokeAsync(RpcTypeToNumber<CLASS>::number, METHOD_##Method); \
+ result->SetNull(); \
+ }
+
+#define TOOLS_RPC_JS_STUB_METHOD1(Method, T1) \
+ void Js##Method(const CppArgumentList& args, CppVariant* result) { \
+ T1 t1 = RpcJsTypeTrait<T1>::Pass(args[0]); \
+ InvokeAsync(RpcTypeToNumber<CLASS>::number, METHOD_##Method, &t1); \
+ result->SetNull(); \
+ }
+
+#define TOOLS_RPC_JS_STUB_METHOD2(Method, T1, T2) \
+ void Js##Method(const CppArgumentList& args, CppVariant* result) { \
+ T1 t1 = RpcJsTypeTrait<T1>::Pass(args[0]); \
+ T2 t2 = RpcJsTypeTrait<T2>::Pass(args[1]); \
+ InvokeAsync(RpcTypeToNumber<CLASS>::number, METHOD_##Method, &t1, &t2); \
+ result->SetNull(); \
+ }
+
+#define TOOLS_RPC_JS_STUB_METHOD3(Method, T1, T2, T3) \
+ void Js##Method(const CppArgumentList& args, CppVariant* result) { \
+ T1 t1 = RpcJsTypeTrait<T1>::Pass(args[0]); \
+ T2 t2 = RpcJsTypeTrait<T2>::Pass(args[1]); \
+ T3 t3 = RpcJsTypeTrait<T3>::Pass(args[2]); \
+ InvokeAsync(RpcTypeToNumber<CLASS>::number, METHOD_##Method, &t1, &t2, \
+ &t3); \
+ result->SetNull(); \
+ }
+
+#define TOOLS_RPC_JS_STUB_METHOD4(Method, T1, T2, T3, T4) \
+ void Js##Method(const CppArgumentList& args, CppVariant* result) { \
+ T1 t1 = RpcJsTypeTrait<T1>::Pass(args[0]); \
+ T2 t2 = RpcJsTypeTrait<T2>::Pass(args[1]); \
+ T3 t3 = RpcJsTypeTrait<T3>::Pass(args[2]); \
+ T4 t4 = RpcJsTypeTrait<T4>::Pass(args[3]); \
+ InvokeAsync(RpcTypeToNumber<CLASS>::number, METHOD_##Method, &t1, &t2, \
+ &t3, &t4); \
+ result->SetNull(); \
+ }
+
+///////////////////////////////////////////////////////
+// JS RPC dispatch method implementations
+
+#define TOOLS_RPC_JS_DISPATCH0(Method) \
+case CLASS::METHOD_##Method: { \
+ *expr = StringPrintf("%s.%s()", js_obj.c_str(), #Method); \
+ return true; \
+}
+
+#define TOOLS_RPC_JS_DISPATCH1(Method, T1) \
+case CLASS::METHOD_##Method: { \
+ Value* t1; \
+ message.Get(2, &t1); \
+ *expr = StringPrintf("%s.%s(%s)", js_obj.c_str(), #Method, \
+ DevToolsRpc::Serialize(*t1).c_str()); \
+ return true; \
+}
+
+#define TOOLS_RPC_JS_DISPATCH2(Method, T1, T2) \
+case CLASS::METHOD_##Method: { \
+ Value* t1; \
+ Value* t2; \
+ message.Get(2, &t1); \
+ message.Get(3, &t2); \
+ *expr = StringPrintf("%s.%s(%s, %s)", js_obj.c_str(), #Method, \
+ DevToolsRpc::Serialize(*t1).c_str(), \
+ DevToolsRpc::Serialize(*t2).c_str()); \
+ return true; \
+}
+
+#define TOOLS_RPC_JS_DISPATCH3(Method, T1, T2, T3) \
+case CLASS::METHOD_##Method: { \
+ Value* t1; \
+ Value* t2; \
+ Value* t3; \
+ message.Get(2, &t1); \
+ message.Get(3, &t2); \
+ message.Get(4, &t3); \
+ *expr = StringPrintf("%s.%s(%s, %s, %s)", js_obj.c_str(), #Method, \
+ DevToolsRpc::Serialize(*t1).c_str(), \
+ DevToolsRpc::Serialize(*t2).c_str(), \
+ DevToolsRpc::Serialize(*t3).c_str()); \
+ return true; \
+}
+
+#define TOOLS_RPC_JS_DISPATCH4(Method, T1, T2, T3, T4) \
+case CLASS::METHOD_##Method: { \
+ Value* t1; \
+ Value* t2; \
+ Value* t3; \
+ Value* t4; \
+ message.Get(2, &t1); \
+ message.Get(3, &t2); \
+ message.Get(4, &t3); \
+ message.Get(5, &t4); \
+ *expr = StringPrintf("%s.%s(%s, %s, %s, %s)", js_obj.c_str(), #Method, \
+ DevToolsRpc::Serialize(*t1).c_str(), \
+ DevToolsRpc::Serialize(*t2).c_str(), \
+ DevToolsRpc::Serialize(*t3).c_str(), \
+ DevToolsRpc::Serialize(*t4).c_str()); \
+ return true; \
+}
+
+#define DEFINE_RPC_JS_DISPATCH(Class, STRUCT) \
+class Js##Class##Dispatch { \
+ public: \
+ explicit Js##Class##Dispatch(const std::wstring& classname) \
+ : js_obj(WideToUTF8(classname)) {} \
+ virtual ~Js##Class##Dispatch() {} \
+ \
+ bool Dispatch(const ListValue& message, std::string* expr) { \
+ int class_id; \
+ message.GetInteger(0, &class_id); \
+ if (class_id != RpcTypeToNumber<Class>::number) { \
+ return false; \
+ } \
+ int method; \
+ message.GetInteger(1, &method); \
+ typedef Class CLASS; \
+ switch (method) { \
+ STRUCT( \
+ TOOLS_RPC_JS_DISPATCH0, \
+ TOOLS_RPC_JS_DISPATCH1, \
+ TOOLS_RPC_JS_DISPATCH2, \
+ TOOLS_RPC_JS_DISPATCH3, \
+ TOOLS_RPC_JS_DISPATCH4) \
+ default: return false; \
+ } \
+ } \
+ private: \
+ std::string js_obj; \
+ DISALLOW_COPY_AND_ASSIGN(Js##Class##Dispatch); \
+};
+
+///////////////////////////////////////////////////////
+// JS RPC main obj macro
+
+#define DEFINE_RPC_JS_BOUND_OBJ(Class, STRUCT, DClass, DELEGATE_STRUCT) \
+DEFINE_RPC_JS_DISPATCH(DClass, DELEGATE_STRUCT) \
+class Js##Class##BoundObj : public Class##Stub, \
+ public CppBoundClass, \
+ public Js##DClass##Dispatch { \
+ public: \
+ Js##Class##BoundObj(Delegate* rpc_delegate, WebFrame* frame, \
+ const std::wstring& classname) \
+ : Class##Stub(rpc_delegate), \
+ Js##DClass##Dispatch(classname) { \
+ BindToJavascript(frame, classname); \
+ STRUCT( \
+ TOOLS_RPC_JS_BIND_METHOD0, \
+ TOOLS_RPC_JS_BIND_METHOD1, \
+ TOOLS_RPC_JS_BIND_METHOD2, \
+ TOOLS_RPC_JS_BIND_METHOD3, \
+ TOOLS_RPC_JS_BIND_METHOD4) \
+ } \
+ virtual ~Js##Class##BoundObj() {} \
+ typedef Js##Class##BoundObj OCLASS; \
+ STRUCT( \
+ TOOLS_RPC_JS_STUB_METHOD0, \
+ TOOLS_RPC_JS_STUB_METHOD1, \
+ TOOLS_RPC_JS_STUB_METHOD2, \
+ TOOLS_RPC_JS_STUB_METHOD3, \
+ TOOLS_RPC_JS_STUB_METHOD4) \
+ private: \
+ DISALLOW_COPY_AND_ASSIGN(Js##Class##BoundObj); \
+};
+
+#endif // WEBKIT_GLUE_DEVTOOLS_DEVTOOLS_RPC_JS_H_
diff --git a/webkit/glue/devtools/dom_agent.h b/webkit/glue/devtools/dom_agent.h
index 33306de..205e4d9 100644
--- a/webkit/glue/devtools/dom_agent.h
+++ b/webkit/glue/devtools/dom_agent.h
@@ -11,7 +11,7 @@
// WebDevToolsAgent. It is capable of sending DOM tree to the client as well
// as providing DOM notifications for the nodes known to the client.
// DomAgent's environment is represented with the DomAgentDelegate interface.
-#define DOM_AGENT_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3) \
+#define DOM_AGENT_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3, METHOD4) \
/* Requests that the document root element is sent to the delegate. */ \
METHOD0(GetDocumentElement) \
\
@@ -19,13 +19,16 @@
METHOD2(GetChildNodes, int /* call_id */, int /* id */) \
\
/* Sets attribute value in the element with given id. */ \
- METHOD3(SetAttribute, int /* id */, String /* name */, String /* value */) \
+ METHOD4(SetAttribute, int /* call_id */, int /* id */, String /* name */, \
+ String /* value */) \
\
/* Removes attribute from the element with given id. */ \
- METHOD2(RemoveAttribute, int /* id */, String /* name */) \
+ METHOD3(RemoveAttribute, int /* call_id */, int /* id */, \
+ String /* name */) \
\
/* Sets text node value in the node with given id. */ \
- METHOD2(SetTextNodeValue, int /* id */, String /* text */) \
+ METHOD3(SetTextNodeValue, int /* call_id */, int /* id */, \
+ String /* text */) \
\
/* Perform search. */ \
METHOD2(PerformSearch, int /* call_id */, String /* query */) \
@@ -37,13 +40,17 @@
DEFINE_RPC_CLASS(DomAgent, DOM_AGENT_STRUCT)
-#define DOM_AGENT_DELEGATE_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3) \
+#define DOM_AGENT_DELEGATE_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3, \
+ METHOD4) \
/* Response to GetChildNodes. */ \
METHOD1(DidGetChildNodes, int /* call_id */) \
\
/* Perform search. */ \
METHOD2(DidPerformSearch, int /* call_id */, Value /* results */) \
\
+ /* Ack for the Set/RemoveAttribute, SetTextNodeValue. */ \
+ METHOD2(DidApplyDomChange, int /* call_id */, bool /* success */) \
+ \
/* Notifies the delegate that element's attributes are updated. */ \
METHOD2(AttributesUpdated, int /* id */, Value /* attributes */) \
\
diff --git a/webkit/glue/devtools/dom_agent_impl.cc b/webkit/glue/devtools/dom_agent_impl.cc
index 6d6471c..d946a7e 100644
--- a/webkit/glue/devtools/dom_agent_impl.cc
+++ b/webkit/glue/devtools/dom_agent_impl.cc
@@ -69,7 +69,9 @@ DomAgentImpl::~DomAgentImpl() {
}
void DomAgentImpl::SetDocument(Document* doc) {
- DiscardBindings();
+ if (documents_.size() && doc == documents_.begin()->get()) {
+ return;
+ }
ListHashSet<RefPtr<Document> > copy = documents_;
for (ListHashSet<RefPtr<Document> >::iterator it = copy.begin();
@@ -84,6 +86,8 @@ void DomAgentImpl::SetDocument(Document* doc) {
GetDocumentElement();
document_element_requested_ = false;
}
+ } else {
+ DiscardBindings();
}
}
@@ -289,33 +293,49 @@ int DomAgentImpl::PushNodePathToClient(Node* node_to_select) {
}
void DomAgentImpl::SetAttribute(
+ int call_id,
int element_id,
const String& name,
const String& value) {
Node* node = GetNodeForId(element_id);
if (node && (node->nodeType() == Node::ELEMENT_NODE)) {
Element* element = static_cast<Element*>(node);
- ExceptionCode ec = 0;
+ ExceptionCode ec = 0;
element->setAttribute(name, value, ec);
+ delegate_->DidApplyDomChange(call_id, ec == 0);
+ } else {
+ delegate_->DidApplyDomChange(call_id, false);
}
}
-void DomAgentImpl::RemoveAttribute(int element_id, const String& name) {
+void DomAgentImpl::RemoveAttribute(
+ int call_id,
+ int element_id,
+ const String& name) {
Node* node = GetNodeForId(element_id);
if (node && (node->nodeType() == Node::ELEMENT_NODE)) {
Element* element = static_cast<Element*>(node);
ExceptionCode ec = 0;
element->removeAttribute(name, ec);
+ delegate_->DidApplyDomChange(call_id, ec == 0);
+ } else {
+ delegate_->DidApplyDomChange(call_id, false);
}
}
-void DomAgentImpl::SetTextNodeValue(int element_id, const String& value) {
+void DomAgentImpl::SetTextNodeValue(
+ int call_id,
+ int element_id,
+ const String& value) {
Node* node = GetNodeForId(element_id);
if (node && (node->nodeType() == Node::TEXT_NODE)) {
Text* text_node = static_cast<Text*>(node);
ExceptionCode ec = 0;
// TODO(pfeldman): Add error handling
text_node->replaceWholeText(value, ec);
+ delegate_->DidApplyDomChange(call_id, ec == 0);
+ } else {
+ delegate_->DidApplyDomChange(call_id, false);
}
}
diff --git a/webkit/glue/devtools/dom_agent_impl.h b/webkit/glue/devtools/dom_agent_impl.h
index 3f02669..0a8eb31 100644
--- a/webkit/glue/devtools/dom_agent_impl.h
+++ b/webkit/glue/devtools/dom_agent_impl.h
@@ -35,11 +35,18 @@ class DomAgentImpl : public DomAgent {
void GetDocumentElement();
void GetChildNodes(int call_id, int element_id);
void SetAttribute(
+ int call_id,
int element_id,
const WebCore::String& name,
const WebCore::String& value);
- void RemoveAttribute(int element_id, const WebCore::String& name);
- void SetTextNodeValue(int element_id, const WebCore::String& value);
+ void RemoveAttribute(
+ int call_id,
+ int element_id,
+ const WebCore::String& name);
+ void SetTextNodeValue(
+ int call_id,
+ int element_id,
+ const WebCore::String& value);
void PerformSearch(int call_id, const String& query);
void DiscardBindings();
diff --git a/webkit/glue/devtools/js/devtools.js b/webkit/glue/devtools/js/devtools.js
index 39a81fc..31d7782 100644
--- a/webkit/glue/devtools/js/devtools.js
+++ b/webkit/glue/devtools/js/devtools.js
@@ -145,7 +145,7 @@ var webkitUpdateChildren =
WebInspector.ElementsTreeElement.prototype.updateChildren = function() {
var self = this;
- devtools.tools.getDomAgent().getChildNodesAsync(this.representedObject.id,
+ devtools.tools.getDomAgent().getChildNodesAsync(this.representedObject,
function() {
webkitUpdateChildren.call(self);
});
diff --git a/webkit/glue/devtools/js/devtools_host_stub.js b/webkit/glue/devtools/js/devtools_host_stub.js
index c820d67..17b49e8 100644
--- a/webkit/glue/devtools/js/devtools_host_stub.js
+++ b/webkit/glue/devtools/js/devtools_host_stub.js
@@ -95,15 +95,24 @@ RemoteDomAgentStub.prototype.GetChildNodes = function(callId, id) {
};
-RemoteDomAgentStub.prototype.SetAttribute = function() {
+RemoteDomAgentStub.prototype.SetAttribute = function(callId) {
+ setTimeout(function() {
+ RemoteDomAgent.DidApplyDomChange(callId, true);
+ }, 0);
};
-RemoteDomAgentStub.prototype.RemoveAttribute = function() {
+RemoteDomAgentStub.prototype.RemoveAttribute = function(callId) {
+ setTimeout(function() {
+ RemoteDomAgent.DidApplyDomChange(callId, true);
+ }, 0);
};
-RemoteDomAgentStub.prototype.SetTextNodeValue = function() {
+RemoteDomAgentStub.prototype.SetTextNodeValue = function(callId) {
+ setTimeout(function() {
+ RemoteDomAgent.DidApplyDomChange(callId, true);
+ }, 0);
};
@@ -144,6 +153,18 @@ RemoteToolsAgentStub.prototype.EvaluateJavaSctipt = function(callId, script) {
}, 0);
};
+
+/**
+ * @constructor
+ */
+RemoteDebuggerCommandExecutorStub = function() {
+};
+
+
+RemoteDebuggerCommandExecutorStub.prototype.DebuggerCommand = function() {
+};
+
+
/**
* @constructor
*/
@@ -168,6 +189,8 @@ DevToolsHostStub.prototype.loaded = function() {
if (!window['DevToolsHost']) {
window['RemoteDebuggerAgent'] = new RemoteDebuggerAgentStub();
+ window['RemoteDebuggerCommandExecutor'] =
+ new RemoteDebuggerCommandExecutorStub();
window['RemoteDomAgent'] = new RemoteDomAgentStub();
window['RemoteNetAgent'] = new RemoteNetAgentStub();
window['RemoteToolsAgent'] = new RemoteToolsAgentStub();
diff --git a/webkit/glue/devtools/js/dom_agent.js b/webkit/glue/devtools/js/dom_agent.js
index 0590e6b..ebcba55 100644
--- a/webkit/glue/devtools/js/dom_agent.js
+++ b/webkit/glue/devtools/js/dom_agent.js
@@ -36,10 +36,10 @@ devtools.PayloadIndex = {
devtools.DomNode = function(doc, payload) {
this.ownerDocument = doc;
- this.id = payload[devtools.PayloadIndex.ID];
+ this.id_ = payload[devtools.PayloadIndex.ID];
this.nodeType = payload[devtools.PayloadIndex.TYPE];
this.nodeName = payload[devtools.PayloadIndex.NAME];
- this.nodeValue = payload[devtools.PayloadIndex.VALUE];
+ this.nodeValue_ = payload[devtools.PayloadIndex.VALUE];
this.textContent = this.nodeValue;
this.attributes = [];
@@ -65,14 +65,37 @@ devtools.DomNode = function(doc, payload) {
/**
+ * Overrides for getters and setters.
+ */
+devtools.DomNode.prototype = {
+ get nodeValue() {
+ return this.nodeValue_;
+ },
+
+ set nodeValue(value) {
+ if (this.nodeType != Node.TEXT_NODE) {
+ return;
+ }
+ var self = this;
+ this.ownerDocument.domAgent_.setTextNodeValueAsync(this, value,
+ function() {
+ self.nodeValue_ = value;
+ self.textContent = value;
+ });
+ }
+};
+
+
+/**
* Sets attributes for a given node based on a given attrs payload.
* @param {Array.<string>} attrs Attribute key-value pairs to set.
* @private
*/
devtools.DomNode.prototype.setAttributesPayload_ = function(attrs) {
for (var i = 0; i < attrs.length; i += 2) {
- this.attributes.push({"name" : attrs[i], "value" : attrs[i+1]});
- this.attributesMap_[attrs[i]] = attrs[i+1];
+ var attr = {"name" : attrs[i], "value" : attrs[i+1]};
+ this.attributes.push(attr);
+ this.attributesMap_[attrs[i]] = attr;
}
};
@@ -164,18 +187,62 @@ devtools.DomNode.prototype.renumber_ = function() {
/**
* Returns attribute value.
* @param {string} name Attribute name to get value for.
- @ @return {string} Attribute value.
+ * @return {string} Attribute value.
*/
devtools.DomNode.prototype.getAttribute = function(name) {
- return this.attributesMap_[name];
+ var attr = this.attributesMap_[name];
+ return attr ? attr.value : undefined;
+};
+
+
+/**
+ * Sends 'set attribute' command to the remote agent.
+ * @param {string} name Attribute name to set value for.
+ * @param {string} value Attribute value to set.
+ */
+devtools.DomNode.prototype.setAttribute = function(name, value) {
+ var self = this;
+ this.ownerDocument.domAgent_.setAttributeAsync(this, name, value,
+ function() {
+ var attr = self.attributesMap_[name];
+ if (attr) {
+ attr.value = value;
+ } else {
+ attr = {"name" : name, "value" : value};
+ self.attributesMap_[name] = attr;
+ self.attributes_.push(attr);
+ }
+ });
+};
+
+
+/**
+ * Sends 'remove attribute' command to the remote agent.
+ * @param {string} name Attribute name to set value for.
+ */
+devtools.DomNode.prototype.removeAttribute = function(name) {
+ var self = this;
+ this.ownerDocument.domAgent_.removeAttributeAsync(this, name, function() {
+ if (!success) {
+ return;
+ }
+ delete self.attributesMap_[name];
+ for (var i = 0; i < self.attributes_.length; ++i) {
+ if (self.attributes_[i].name == name) {
+ self.attributes_.splice(i, 1);
+ break;
+ }
+ }
+ });
};
/**
* Remote Dom document abstraction.
+ * @param {devtools.DomAgent} domAgent owner agent.
* @constructor.
*/
-devtools.DomDocument = function() {
+devtools.DomDocument = function(domAgent) {
devtools.DomNode.call(this, null,
[
0, // id
@@ -190,6 +257,7 @@ devtools.DomDocument = function() {
getComputedStyle : function() {},
getMatchedCSSRules : function() {}
};
+ this.domAgent_ = domAgent;
};
goog.inherits(devtools.DomDocument, devtools.DomNode);
@@ -256,6 +324,12 @@ devtools.DomAgent = function() {
devtools.Callback.processCallback;
RemoteDomAgent.DidPerformSearch =
devtools.Callback.processCallback;
+ RemoteDomAgent.DidApplyDomChange =
+ devtools.Callback.processCallback;
+ RemoteDomAgent.DidRemoveAttribute =
+ devtools.Callback.processCallback;
+ RemoteDomAgent.DidSetTextNodeValue =
+ devtools.Callback.processCallback;
RemoteDomAgent.AttributesUpdated =
goog.bind(this.attributesUpdated, this);
RemoteDomAgent.SetDocumentElement =
@@ -297,8 +371,7 @@ devtools.DomAgent = function() {
* Rests dom agent to its initial state.
*/
devtools.DomAgent.prototype.reset = function() {
- RemoteDomAgent.DiscardBindings();
- this.document_ = new devtools.DomDocument();
+ this.document_ = new devtools.DomDocument(this);
this.idToDomNode_ = { 0 : this.document_ };
this.searchResults_ = [];
};
@@ -325,25 +398,86 @@ devtools.DomAgent.prototype.getDocumentElementAsync = function() {
/**
* Asynchronously fetches children from the element with given id.
- * @param {number} parentId Element to get children for.
+ * @param {devtools.DomNode} parent Element to get children for.
* @param {function(devtools.DomNode):undefined} opt_callback Callback with
* the result.
*/
-devtools.DomAgent.prototype.getChildNodesAsync = function(parentId,
+devtools.DomAgent.prototype.getChildNodesAsync = function(parent,
opt_callback) {
- var children = this.idToDomNode_[parentId].children;
+ var children = parent.children;
if (children && opt_callback) {
opt_callback(children);
return;
}
- var self = this;
var mycallback = function() {
if (opt_callback) {
- opt_callback(self.idToDomNode_[parentId].children);
+ opt_callback(parent.children);
}
};
var callId = devtools.Callback.wrap(mycallback);
- RemoteDomAgent.GetChildNodes(callId, parentId);
+ RemoteDomAgent.GetChildNodes(callId, parent.id_);
+};
+
+
+/**
+ * Sends 'set attribute' command to the remote agent.
+ * @param {devtools.DomNode} node Node to change.
+ * @param {string} name Attribute name to set value for.
+ * @param {string} value Attribute value to set.
+ * @param {function():undefined} opt_callback Callback on success.
+ */
+devtools.DomAgent.prototype.setAttributeAsync = function(node, name, value,
+ callback) {
+ var mycallback = goog.bind(this.didApplyDomChange_, this, node, callback);
+ RemoteDomAgent.SetAttribute(devtools.Callback.wrap(mycallback),
+ node.id_, name, value);
+};
+
+
+/**
+ * Sends 'remove attribute' command to the remote agent.
+ * @param {devtools.DomNode} node Node to change.
+ * @param {string} name Attribute name to set value for.
+ * @param {function():undefined} opt_callback Callback on success.
+ */
+devtools.DomAgent.prototype.removeAttributeAsync = function(node, name,
+ callback) {
+ var mycallback = goog.bind(this.didApplyDomChange_, this, node, callback);
+ RemoteDomAgent.RemoveAttribute(devtools.Callback.wrap(mycallback),
+ node.id_, name);
+};
+
+
+/**
+ * Sends 'set text node value' command to the remote agent.
+ * @param {devtools.DomNode} node Node to change.
+ * @param {string} text Text value to set.
+ * @param {function():undefined} opt_callback Callback on success.
+ */
+devtools.DomAgent.prototype.setTextNodeValueAsync = function(node, text,
+ callback) {
+ var mycallback = goog.bind(this.didApplyDomChange_, this, node, callback);
+ RemoteDomAgent.SetTextNodeValue(devtools.Callback.wrap(mycallback),
+ node.id_, text);
+};
+
+
+/**
+ * Universal callback wrapper for edit dom operations.
+ * @param {devtools.DomNode} node Node to apply local changes on.
+ * @param {Function} callback Post-operation call.
+ * @param {boolean} success True iff operation has completed successfully.
+ */
+devtools.DomAgent.prototype.didApplyDomChange_ = function(node,
+ callback, success) {
+ if (!success) {
+ return;
+ }
+ callback();
+ var elem = WebInspector.panels.elements.treeOutline.findTreeElement(node);
+ if (elem) {
+ elem._updateTitle();
+ }
};
@@ -403,7 +537,7 @@ devtools.DomAgent.prototype.setChildNodes = function(parentId, payloads) {
devtools.DomAgent.prototype.bindNodes_ = function(children) {
for (var i = 0; i < children.length; ++i) {
var child = children[i];
- this.idToDomNode_[child.id] = child;
+ this.idToDomNode_[child.id_] = child;
if (child.children) {
this.bindNodes_(child.children);
}
@@ -428,7 +562,7 @@ devtools.DomAgent.prototype.childNodeInserted = function(
var parent = this.idToDomNode_[parentId];
var prev = this.idToDomNode_[prevId];
var node = parent.insertChild_(prev, payload);
- this.idToDomNode_[node.id] = node;
+ this.idToDomNode_[node.id_] = node;
var event = { target : node, relatedNode : parent };
this.document_.fireDomEvent_("DOMNodeInserted", event);
};
diff --git a/webkit/glue/devtools/net_agent.h b/webkit/glue/devtools/net_agent.h
index 0921c26..87459f0 100644
--- a/webkit/glue/devtools/net_agent.h
+++ b/webkit/glue/devtools/net_agent.h
@@ -11,7 +11,7 @@
// WebDevToolsAgent. It is capable of sniffing network calls and passing the
// HTTPRequest-related data to the client.
// NetAgent's environment is represented with the NetAgentDelegate interface.
-#define NET_AGENT_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3) \
+#define NET_AGENT_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3, METHOD4) \
/* Requests that the agent sends content of the resource with given id to the
delegate. */ \
METHOD3(GetResourceContent, int /* call_id */, int /* identifier */, \
@@ -19,7 +19,8 @@
DEFINE_RPC_CLASS(NetAgent, NET_AGENT_STRUCT)
-#define NET_AGENT_DELEGATE_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3) \
+#define NET_AGENT_DELEGATE_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3, \
+ METHOD4) \
/* Notifies the delegate that a request is about to be sent out. */ \
METHOD2(WillSendRequest, int, Value) \
\
diff --git a/webkit/glue/devtools/tools_agent.h b/webkit/glue/devtools/tools_agent.h
index 0ca14d3..8db0de6 100644
--- a/webkit/glue/devtools/tools_agent.h
+++ b/webkit/glue/devtools/tools_agent.h
@@ -9,7 +9,7 @@
// Tools agent provides API for enabling / disabling other agents as well as
// API for auxiliary UI functions such as dom elements highlighting.
-#define TOOLS_AGENT_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3) \
+#define TOOLS_AGENT_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3, METHOD4) \
/* Highlights Dom node with given ID */ \
METHOD1(HighlightDOMNode, int /* node_id */) \
\
@@ -21,7 +21,8 @@
DEFINE_RPC_CLASS(ToolsAgent, TOOLS_AGENT_STRUCT)
-#define TOOLS_AGENT_DELEGATE_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3) \
+#define TOOLS_AGENT_DELEGATE_STRUCT(METHOD0, METHOD1, METHOD2, METHOD3, \
+ METHOD4) \
/* Updates focused node on the client. */ \
METHOD1(UpdateFocusedNode, int /* node_id */) \
\