• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2009 Apple Inc. All rights reserved.
3  * Copyright (C) 2011 Google Inc. All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  *
9  * 1.  Redistributions of source code must retain the above copyright
10  *     notice, this list of conditions and the following disclaimer.
11  * 2.  Redistributions in binary form must reproduce the above copyright
12  *     notice, this list of conditions and the following disclaimer in the
13  *     documentation and/or other materials provided with the distribution.
14  * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
15  *     its contributors may be used to endorse or promote products derived
16  *     from this software without specific prior written permission.
17  *
18  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
19  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
20  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
21  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
22  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
23  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
24  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
25  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28  */
29 
30 #ifndef InspectorDOMAgent_h
31 #define InspectorDOMAgent_h
32 
33 #include "core/InspectorFrontend.h"
34 #include "core/inspector/InjectedScript.h"
35 #include "core/inspector/InjectedScriptManager.h"
36 #include "core/inspector/InspectorBaseAgent.h"
37 #include "core/rendering/RenderLayer.h"
38 #include "platform/JSONValues.h"
39 
40 #include "wtf/HashMap.h"
41 #include "wtf/HashSet.h"
42 #include "wtf/OwnPtr.h"
43 #include "wtf/PassOwnPtr.h"
44 #include "wtf/RefPtr.h"
45 #include "wtf/Vector.h"
46 #include "wtf/text/AtomicString.h"
47 
48 namespace blink {
49 
50 class CharacterData;
51 class DOMEditor;
52 class Document;
53 class Element;
54 class EventTarget;
55 class ExceptionState;
56 class InspectorFrontend;
57 class InspectorHistory;
58 class InspectorOverlay;
59 class InspectorPageAgent;
60 class InspectorState;
61 class InstrumentingAgents;
62 class Node;
63 class PlatformGestureEvent;
64 class PlatformTouchEvent;
65 class RevalidateStyleAttributeTask;
66 class ShadowRoot;
67 
68 struct HighlightConfig;
69 
70 typedef String ErrorString;
71 
72 
73 struct EventListenerInfo {
EventListenerInfoEventListenerInfo74     EventListenerInfo(EventTarget* eventTarget, const AtomicString& eventType, const EventListenerVector& eventListenerVector)
75         : eventTarget(eventTarget)
76         , eventType(eventType)
77         , eventListenerVector(eventListenerVector)
78     {
79     }
80 
81     EventTarget* eventTarget;
82     const AtomicString eventType;
83     const EventListenerVector eventListenerVector;
84 };
85 
86 class InspectorDOMAgent FINAL : public InspectorBaseAgent<InspectorDOMAgent>, public InspectorBackendDispatcher::DOMCommandHandler {
87     WTF_MAKE_NONCOPYABLE(InspectorDOMAgent);
88 public:
89     struct DOMListener : public WillBeGarbageCollectedMixin {
~DOMListenerDOMListener90         virtual ~DOMListener()
91         {
92         }
93         virtual void didRemoveDocument(Document*) = 0;
94         virtual void didRemoveDOMNode(Node*) = 0;
95         virtual void didModifyDOMAttr(Element*) = 0;
96     };
97 
create(InspectorPageAgent * pageAgent,InjectedScriptManager * injectedScriptManager,InspectorOverlay * overlay)98     static PassOwnPtrWillBeRawPtr<InspectorDOMAgent> create(InspectorPageAgent* pageAgent, InjectedScriptManager* injectedScriptManager, InspectorOverlay* overlay)
99     {
100         return adoptPtrWillBeNoop(new InspectorDOMAgent(pageAgent, injectedScriptManager, overlay));
101     }
102 
103     static String toErrorString(ExceptionState&);
104 
105     virtual ~InspectorDOMAgent();
106     virtual void trace(Visitor*) OVERRIDE;
107 
108     virtual void setFrontend(InspectorFrontend*) OVERRIDE;
109     virtual void clearFrontend() OVERRIDE;
110     virtual void restore() OVERRIDE;
111 
112     WillBeHeapVector<RawPtrWillBeMember<Document> > documents();
113     void reset();
114 
115     // Methods called from the frontend for DOM nodes inspection.
116     virtual void enable(ErrorString*) OVERRIDE;
117     virtual void disable(ErrorString*) OVERRIDE;
118     virtual void querySelector(ErrorString*, int nodeId, const String& selectors, int* elementId) OVERRIDE;
119     virtual void querySelectorAll(ErrorString*, int nodeId, const String& selectors, RefPtr<TypeBuilder::Array<int> >& result) OVERRIDE;
120     virtual void getDocument(ErrorString*, RefPtr<TypeBuilder::DOM::Node>& root) OVERRIDE;
121     virtual void requestChildNodes(ErrorString*, int nodeId, const int* depth) OVERRIDE;
122     virtual void setAttributeValue(ErrorString*, int elementId, const String& name, const String& value) OVERRIDE;
123     virtual void setAttributesAsText(ErrorString*, int elementId, const String& text, const String* name) OVERRIDE;
124     virtual void removeAttribute(ErrorString*, int elementId, const String& name) OVERRIDE;
125     virtual void removeNode(ErrorString*, int nodeId) OVERRIDE;
126     virtual void setNodeName(ErrorString*, int nodeId, const String& name, int* newId) OVERRIDE;
127     virtual void getOuterHTML(ErrorString*, int nodeId, WTF::String* outerHTML) OVERRIDE;
128     virtual void setOuterHTML(ErrorString*, int nodeId, const String& outerHTML) OVERRIDE;
129     virtual void setNodeValue(ErrorString*, int nodeId, const String& value) OVERRIDE;
130     virtual void getEventListenersForNode(ErrorString*, int nodeId, const WTF::String* objectGroup, RefPtr<TypeBuilder::Array<TypeBuilder::DOM::EventListener> >& listenersArray) OVERRIDE;
131     virtual void performSearch(ErrorString*, const String& whitespaceTrimmedQuery, const bool* includeUserAgentShadowDOM, String* searchId, int* resultCount) OVERRIDE;
132     virtual void getSearchResults(ErrorString*, const String& searchId, int fromIndex, int toIndex, RefPtr<TypeBuilder::Array<int> >&) OVERRIDE;
133     virtual void discardSearchResults(ErrorString*, const String& searchId) OVERRIDE;
134     virtual void resolveNode(ErrorString*, int nodeId, const String* objectGroup, RefPtr<TypeBuilder::Runtime::RemoteObject>& result) OVERRIDE;
135     virtual void getAttributes(ErrorString*, int nodeId, RefPtr<TypeBuilder::Array<String> >& result) OVERRIDE;
136     virtual void setInspectModeEnabled(ErrorString*, bool enabled, const bool* inspectUAShadowDOM, const RefPtr<JSONObject>* highlightConfig) OVERRIDE;
137     virtual void requestNode(ErrorString*, const String& objectId, int* nodeId) OVERRIDE;
138     virtual void pushNodeByPathToFrontend(ErrorString*, const String& path, int* nodeId) OVERRIDE;
139     virtual void pushNodesByBackendIdsToFrontend(ErrorString*, const RefPtr<JSONArray>& nodeIds, RefPtr<TypeBuilder::Array<int> >&) OVERRIDE;
140     virtual void hideHighlight(ErrorString*) OVERRIDE;
141     virtual void highlightRect(ErrorString*, int x, int y, int width, int height, const RefPtr<JSONObject>* color, const RefPtr<JSONObject>* outlineColor) OVERRIDE;
142     virtual void highlightQuad(ErrorString*, const RefPtr<JSONArray>& quad, const RefPtr<JSONObject>* color, const RefPtr<JSONObject>* outlineColor) OVERRIDE;
143     virtual void highlightNode(ErrorString*, const RefPtr<JSONObject>& highlightConfig, const int* nodeId, const String* objectId) OVERRIDE;
144     virtual void highlightFrame(ErrorString*, const String& frameId, const RefPtr<JSONObject>* color, const RefPtr<JSONObject>* outlineColor) OVERRIDE;
145 
146     virtual void copyTo(ErrorString*, int nodeId, int targetElementId, const int* anchorNodeId, int* newNodeId) OVERRIDE;
147     virtual void moveTo(ErrorString*, int nodeId, int targetNodeId, const int* anchorNodeId, int* newNodeId) OVERRIDE;
148     virtual void undo(ErrorString*) OVERRIDE;
149     virtual void redo(ErrorString*) OVERRIDE;
150     virtual void markUndoableState(ErrorString*) OVERRIDE;
151     virtual void focus(ErrorString*, int nodeId) OVERRIDE;
152     virtual void setFileInputFiles(ErrorString*, int nodeId, const RefPtr<JSONArray>& files) OVERRIDE;
153     virtual void getBoxModel(ErrorString*, int nodeId, RefPtr<TypeBuilder::DOM::BoxModel>&) OVERRIDE;
154     virtual void getNodeForLocation(ErrorString*, int x, int y, int* nodeId) OVERRIDE;
155     virtual void getRelayoutBoundary(ErrorString*, int nodeId, int* relayoutBoundaryNodeId) OVERRIDE;
156 
157     static void getEventListeners(EventTarget*, Vector<EventListenerInfo>& listenersArray, bool includeAncestors);
158 
159     class Listener : public WillBeGarbageCollectedMixin {
160     public:
~Listener()161         virtual ~Listener() { }
162         virtual void domAgentWasEnabled() = 0;
163         virtual void domAgentWasDisabled() = 0;
164     };
setListener(Listener * listener)165     void setListener(Listener* listener) { m_listener = listener; }
166 
167     bool enabled() const;
168 
169     // Methods called from the InspectorInstrumentation.
170     void setDocument(Document*);
171     void releaseDanglingNodes();
172 
173     void domContentLoadedEventFired(LocalFrame*);
174     void didCommitLoad(LocalFrame*, DocumentLoader*);
175 
176     void didInsertDOMNode(Node*);
177     void willRemoveDOMNode(Node*);
178     void willModifyDOMAttr(Element*, const AtomicString& oldValue, const AtomicString& newValue);
179     void didModifyDOMAttr(Element*, const AtomicString& name, const AtomicString& value);
180     void didRemoveDOMAttr(Element*, const AtomicString& name);
181     void styleAttributeInvalidated(const WillBeHeapVector<RawPtrWillBeMember<Element> >& elements);
182     void characterDataModified(CharacterData*);
183     void didInvalidateStyleAttr(Node*);
184     void didPushShadowRoot(Element* host, ShadowRoot*);
185     void willPopShadowRoot(Element* host, ShadowRoot*);
186     void frameDocumentUpdated(LocalFrame*);
187     void pseudoElementCreated(PseudoElement*);
188     void pseudoElementDestroyed(PseudoElement*);
189 
190     Node* nodeForId(int nodeId);
191     int boundNodeId(Node*);
192     void setDOMListener(DOMListener*);
193 
194     static String documentURLString(Document*);
195 
196     PassRefPtr<TypeBuilder::Runtime::RemoteObject> resolveNode(Node*, const String& objectGroup);
197     bool handleMousePress();
198     bool handleGestureEvent(LocalFrame*, const PlatformGestureEvent&);
199     bool handleTouchEvent(LocalFrame*, const PlatformTouchEvent&);
200     bool handleMouseMove(LocalFrame*, const PlatformMouseEvent&);
201 
history()202     InspectorHistory* history() { return m_history.get(); }
203 
204     // We represent embedded doms as a part of the same hierarchy. Hence we treat children of frame owners differently.
205     // We also skip whitespace text nodes conditionally. Following methods encapsulate these specifics.
206     static Node* innerFirstChild(Node*);
207     static Node* innerNextSibling(Node*);
208     static Node* innerPreviousSibling(Node*);
209     static unsigned innerChildNodeCount(Node*);
210     static Node* innerParentNode(Node*);
211     static bool isWhitespace(Node*);
212 
213     Node* assertNode(ErrorString*, int nodeId);
214     Element* assertElement(ErrorString*, int nodeId);
215     Document* assertDocument(ErrorString*, int nodeId);
216 
217 private:
218     enum SearchMode { NotSearching, SearchingForNormal, SearchingForUAShadow };
219 
220     InspectorDOMAgent(InspectorPageAgent*, InjectedScriptManager*, InspectorOverlay*);
221 
222     void innerEnable();
223     void notifyDocumentUpdated();
224 
225     void setSearchingForNode(ErrorString*, SearchMode, JSONObject* highlightConfig);
226     PassOwnPtr<HighlightConfig> highlightConfigFromInspectorObject(ErrorString*, JSONObject* highlightInspectorObject);
227 
228     // Node-related methods.
229     typedef WillBeHeapHashMap<RefPtrWillBeMember<Node>, int> NodeToIdMap;
230     int bind(Node*, NodeToIdMap*);
231     void unbind(Node*, NodeToIdMap*);
232 
233     Node* assertEditableNode(ErrorString*, int nodeId);
234     Node* assertEditableChildNode(ErrorString*, Element* parentElement, int nodeId);
235     Element* assertEditableElement(ErrorString*, int nodeId);
236 
237     void inspect(Node*);
238 
239     int pushNodePathToFrontend(Node*);
240     void pushChildNodesToFrontend(int nodeId, int depth = 1);
241 
242     void invalidateFrameOwnerElement(LocalFrame*);
243 
244     PassRefPtr<TypeBuilder::DOM::Node> buildObjectForNode(Node*, int depth, NodeToIdMap*);
245     PassRefPtr<TypeBuilder::Array<String> > buildArrayForElementAttributes(Element*);
246     PassRefPtr<TypeBuilder::Array<TypeBuilder::DOM::Node> > buildArrayForContainerChildren(Node* container, int depth, NodeToIdMap* nodesMap);
247     PassRefPtr<TypeBuilder::DOM::EventListener> buildObjectForEventListener(const RegisteredEventListener&, const AtomicString& eventType, Node*, const String* objectGroupId);
248     PassRefPtr<TypeBuilder::Array<TypeBuilder::DOM::Node> > buildArrayForPseudoElements(Element*, NodeToIdMap* nodesMap);
249 
250     Node* nodeForPath(const String& path);
251 
252     void discardFrontendBindings();
253 
254     void innerHighlightQuad(PassOwnPtr<FloatQuad>, const RefPtr<JSONObject>* color, const RefPtr<JSONObject>* outlineColor);
255 
256     bool pushDocumentUponHandlelessOperation(ErrorString*);
257 
258     RawPtrWillBeMember<InspectorPageAgent> m_pageAgent;
259     RawPtrWillBeMember<InjectedScriptManager> m_injectedScriptManager;
260     InspectorOverlay* m_overlay;
261     InspectorFrontend::DOM* m_frontend;
262     RawPtrWillBeMember<DOMListener> m_domListener;
263     OwnPtrWillBeMember<NodeToIdMap> m_documentNodeToIdMap;
264     // Owns node mappings for dangling nodes.
265     WillBeHeapVector<OwnPtrWillBeMember<NodeToIdMap> > m_danglingNodeToIdMaps;
266     WillBeHeapHashMap<int, RawPtrWillBeMember<Node> > m_idToNode;
267     WillBeHeapHashMap<int, RawPtrWillBeMember<NodeToIdMap> > m_idToNodesMap;
268     HashSet<int> m_childrenRequested;
269     HashMap<int, int> m_cachedChildCount;
270     int m_lastNodeId;
271     RefPtrWillBeMember<Document> m_document;
272     typedef WillBeHeapHashMap<String, WillBeHeapVector<RefPtrWillBeMember<Node> > > SearchResults;
273     SearchResults m_searchResults;
274     OwnPtrWillBeMember<RevalidateStyleAttributeTask> m_revalidateStyleAttrTask;
275     SearchMode m_searchingForNode;
276     OwnPtr<HighlightConfig> m_inspectModeHighlightConfig;
277     OwnPtrWillBeMember<InspectorHistory> m_history;
278     OwnPtrWillBeMember<DOMEditor> m_domEditor;
279     bool m_suppressAttributeModifiedEvent;
280     RawPtrWillBeMember<Listener> m_listener;
281 };
282 
283 
284 } // namespace blink
285 
286 #endif // !defined(InspectorDOMAgent_h)
287