Upstream version 9.38.198.0
[platform/framework/web/crosswalk.git] / src / third_party / WebKit / Source / web / WebDevToolsAgentImpl.cpp
1 /*
2  * Copyright (C) 2010-2011 Google Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  *
8  *     * Redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer.
10  *     * Redistributions in binary form must reproduce the above
11  * copyright notice, this list of conditions and the following disclaimer
12  * in the documentation and/or other materials provided with the
13  * distribution.
14  *     * Neither the name of Google Inc. nor the names of its
15  * contributors may be used to endorse or promote products derived from
16  * this software without specific prior written permission.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29  */
30
31 #include "config.h"
32 #include "web/WebDevToolsAgentImpl.h"
33
34 #include "bindings/core/v8/PageScriptDebugServer.h"
35 #include "bindings/core/v8/ScriptController.h"
36 #include "bindings/core/v8/V8Binding.h"
37 #include "core/InspectorBackendDispatcher.h"
38 #include "core/InspectorFrontend.h"
39 #include "core/dom/ExceptionCode.h"
40 #include "core/fetch/MemoryCache.h"
41 #include "core/frame/FrameView.h"
42 #include "core/frame/LocalFrame.h"
43 #include "core/frame/Settings.h"
44 #include "core/inspector/InjectedScriptHost.h"
45 #include "core/inspector/InspectorController.h"
46 #include "core/page/FocusController.h"
47 #include "core/page/Page.h"
48 #include "core/rendering/RenderView.h"
49 #include "platform/JSONValues.h"
50 #include "platform/RuntimeEnabledFeatures.h"
51 #include "platform/TraceEvent.h"
52 #include "platform/graphics/GraphicsContext.h"
53 #include "platform/network/ResourceError.h"
54 #include "platform/network/ResourceRequest.h"
55 #include "platform/network/ResourceResponse.h"
56 #include "public/platform/Platform.h"
57 #include "public/platform/WebRect.h"
58 #include "public/platform/WebString.h"
59 #include "public/platform/WebURL.h"
60 #include "public/platform/WebURLError.h"
61 #include "public/platform/WebURLRequest.h"
62 #include "public/platform/WebURLResponse.h"
63 #include "public/web/WebDataSource.h"
64 #include "public/web/WebDevToolsAgentClient.h"
65 #include "public/web/WebDeviceEmulationParams.h"
66 #include "public/web/WebMemoryUsageInfo.h"
67 #include "public/web/WebSettings.h"
68 #include "public/web/WebViewClient.h"
69 #include "web/WebInputEventConversion.h"
70 #include "web/WebLocalFrameImpl.h"
71 #include "web/WebViewImpl.h"
72 #include "wtf/CurrentTime.h"
73 #include "wtf/MathExtras.h"
74 #include "wtf/Noncopyable.h"
75 #include "wtf/text/WTFString.h"
76
77 using namespace blink;
78
79 namespace OverlayZOrders {
80 // Use 99 as a big z-order number so that highlight is above other overlays.
81 static const int highlight = 99;
82 }
83
84 namespace blink {
85
86 class ClientMessageLoopAdapter : public PageScriptDebugServer::ClientMessageLoop {
87 public:
88     static void ensureClientMessageLoopCreated(WebDevToolsAgentClient* client)
89     {
90         if (s_instance)
91             return;
92         OwnPtr<ClientMessageLoopAdapter> instance = adoptPtr(new ClientMessageLoopAdapter(adoptPtr(client->createClientMessageLoop())));
93         s_instance = instance.get();
94         PageScriptDebugServer::shared().setClientMessageLoop(instance.release());
95     }
96
97     static void inspectedViewClosed(WebViewImpl* view)
98     {
99         if (s_instance)
100             s_instance->m_frozenViews.remove(view);
101     }
102
103     static void didNavigate()
104     {
105         // Release render thread if necessary.
106         if (s_instance && s_instance->m_running)
107             PageScriptDebugServer::shared().continueProgram();
108     }
109
110 private:
111     ClientMessageLoopAdapter(PassOwnPtr<blink::WebDevToolsAgentClient::WebKitClientMessageLoop> messageLoop)
112         : m_running(false)
113         , m_messageLoop(messageLoop) { }
114
115
116     virtual void run(Page* page)
117     {
118         if (m_running)
119             return;
120         m_running = true;
121
122         // 0. Flush pending frontend messages.
123         WebViewImpl* viewImpl = WebViewImpl::fromPage(page);
124         WebDevToolsAgentImpl* agent = static_cast<WebDevToolsAgentImpl*>(viewImpl->devToolsAgent());
125         agent->flushPendingFrontendMessages();
126
127         Vector<WebViewImpl*> views;
128
129         // 1. Disable input events.
130         const HashSet<Page*>& pages = Page::ordinaryPages();
131         HashSet<Page*>::const_iterator end = pages.end();
132         for (HashSet<Page*>::const_iterator it =  pages.begin(); it != end; ++it) {
133             WebViewImpl* view = WebViewImpl::fromPage(*it);
134             if (!view)
135                 continue;
136             m_frozenViews.add(view);
137             views.append(view);
138             view->setIgnoreInputEvents(true);
139         }
140         // Notify embedder about pausing.
141         agent->client()->willEnterDebugLoop();
142
143         // 2. Disable active objects
144         WebView::willEnterModalLoop();
145
146         // 3. Process messages until quitNow is called.
147         m_messageLoop->run();
148
149         // 4. Resume active objects
150         WebView::didExitModalLoop();
151
152         // 5. Resume input events.
153         for (Vector<WebViewImpl*>::iterator it = views.begin(); it != views.end(); ++it) {
154             if (m_frozenViews.contains(*it)) {
155                 // The view was not closed during the dispatch.
156                 (*it)->setIgnoreInputEvents(false);
157             }
158         }
159         agent->client()->didExitDebugLoop();
160
161         // 6. All views have been resumed, clear the set.
162         m_frozenViews.clear();
163
164         m_running = false;
165     }
166
167     virtual void quitNow()
168     {
169         m_messageLoop->quitNow();
170     }
171
172     bool m_running;
173     OwnPtr<blink::WebDevToolsAgentClient::WebKitClientMessageLoop> m_messageLoop;
174     typedef HashSet<WebViewImpl*> FrozenViewsSet;
175     FrozenViewsSet m_frozenViews;
176     // FIXME: The ownership model for s_instance is somewhat complicated. Can we make this simpler?
177     static ClientMessageLoopAdapter* s_instance;
178 };
179
180 ClientMessageLoopAdapter* ClientMessageLoopAdapter::s_instance = 0;
181
182 class DebuggerTask : public PageScriptDebugServer::Task {
183 public:
184     DebuggerTask(PassOwnPtr<WebDevToolsAgent::MessageDescriptor> descriptor)
185         : m_descriptor(descriptor)
186     {
187     }
188
189     virtual ~DebuggerTask() { }
190     virtual void run()
191     {
192         if (WebDevToolsAgent* webagent = m_descriptor->agent())
193             webagent->dispatchOnInspectorBackend(m_descriptor->message());
194     }
195
196 private:
197     OwnPtr<WebDevToolsAgent::MessageDescriptor> m_descriptor;
198 };
199
200 WebDevToolsAgentImpl::WebDevToolsAgentImpl(
201     WebViewImpl* webViewImpl,
202     WebDevToolsAgentClient* client)
203     : m_debuggerId(client->debuggerId())
204     , m_layerTreeId(0)
205     , m_client(client)
206     , m_webViewImpl(webViewImpl)
207     , m_attached(false)
208     , m_generatingEvent(false)
209     , m_webViewDidLayoutOnceAfterLoad(false)
210     , m_deviceMetricsEnabled(false)
211     , m_emulateMobileEnabled(false)
212     , m_originalViewportEnabled(false)
213     , m_isOverlayScrollbarsEnabled(false)
214     , m_originalMinimumPageScaleFactor(0)
215     , m_originalMaximumPageScaleFactor(0)
216     , m_pageScaleLimitsOverriden(false)
217     , m_touchEventEmulationEnabled(false)
218 {
219     long processId = client->processId();
220     ASSERT(processId > 0);
221     inspectorController()->setProcessId(processId);
222
223     ASSERT(m_debuggerId > 0);
224     ClientMessageLoopAdapter::ensureClientMessageLoopCreated(m_client);
225 }
226
227 WebDevToolsAgentImpl::~WebDevToolsAgentImpl()
228 {
229     ClientMessageLoopAdapter::inspectedViewClosed(m_webViewImpl);
230     if (m_attached)
231         blink::Platform::current()->currentThread()->removeTaskObserver(this);
232 }
233
234 void WebDevToolsAgentImpl::attach(const WebString& hostId)
235 {
236     if (m_attached)
237         return;
238
239     inspectorController()->connectFrontend(hostId, this);
240     blink::Platform::current()->currentThread()->addTaskObserver(this);
241     m_attached = true;
242 }
243
244 void WebDevToolsAgentImpl::reattach(const WebString& hostId, const WebString& savedState)
245 {
246     if (m_attached)
247         return;
248
249     inspectorController()->reuseFrontend(hostId, this, savedState);
250     blink::Platform::current()->currentThread()->addTaskObserver(this);
251     m_attached = true;
252 }
253
254 void WebDevToolsAgentImpl::detach()
255 {
256     blink::Platform::current()->currentThread()->removeTaskObserver(this);
257
258     // Prevent controller from sending messages to the frontend.
259     InspectorController* ic = inspectorController();
260     ic->disconnectFrontend();
261     m_attached = false;
262 }
263
264 void WebDevToolsAgentImpl::continueProgram()
265 {
266     ClientMessageLoopAdapter::didNavigate();
267 }
268
269 void WebDevToolsAgentImpl::didBeginFrame(int frameId)
270 {
271     TRACE_EVENT_INSTANT1(TRACE_DISABLED_BY_DEFAULT("devtools.timeline"), "BeginMainThreadFrame", "layerTreeId", m_layerTreeId);
272     if (InspectorController* ic = inspectorController())
273         ic->didBeginFrame(frameId);
274 }
275
276 void WebDevToolsAgentImpl::didCancelFrame()
277 {
278     if (InspectorController* ic = inspectorController())
279         ic->didCancelFrame();
280 }
281
282 void WebDevToolsAgentImpl::willComposite()
283 {
284     TRACE_EVENT_BEGIN1(TRACE_DISABLED_BY_DEFAULT("devtools.timeline"), "CompositeLayers", "layerTreeId", m_layerTreeId);
285     if (InspectorController* ic = inspectorController())
286         ic->willComposite();
287 }
288
289 void WebDevToolsAgentImpl::didComposite()
290 {
291     TRACE_EVENT_END0(TRACE_DISABLED_BY_DEFAULT("devtools.timeline"), "CompositeLayers");
292     if (InspectorController* ic = inspectorController())
293         ic->didComposite();
294 }
295
296 void WebDevToolsAgentImpl::didCreateScriptContext(WebLocalFrameImpl* webframe, int worldId)
297 {
298     if (blink::LocalFrame* frame = webframe->frame())
299         frame->script().setWorldDebugId(worldId, m_debuggerId);
300     // Skip non main world contexts.
301     if (worldId)
302         return;
303     m_webViewDidLayoutOnceAfterLoad = false;
304 }
305
306 bool WebDevToolsAgentImpl::handleInputEvent(blink::Page* page, const WebInputEvent& inputEvent)
307 {
308     if (!m_attached && !m_generatingEvent)
309         return false;
310
311     // FIXME: This workaround is required for touch emulation on Mac, where
312     // compositor-side pinch handling is not enabled. See http://crbug.com/138003.
313     bool isPinch = inputEvent.type == WebInputEvent::GesturePinchBegin || inputEvent.type == WebInputEvent::GesturePinchUpdate || inputEvent.type == WebInputEvent::GesturePinchEnd;
314     if (isPinch && m_touchEventEmulationEnabled) {
315         FrameView* frameView = page->deprecatedLocalMainFrame()->view();
316         PlatformGestureEventBuilder gestureEvent(frameView, static_cast<const WebGestureEvent&>(inputEvent));
317         float pageScaleFactor = page->pageScaleFactor();
318         if (gestureEvent.type() == PlatformEvent::GesturePinchBegin) {
319             m_lastPinchAnchorCss = adoptPtr(new blink::IntPoint(frameView->scrollPosition() + gestureEvent.position()));
320             m_lastPinchAnchorDip = adoptPtr(new blink::IntPoint(gestureEvent.position()));
321             m_lastPinchAnchorDip->scale(pageScaleFactor, pageScaleFactor);
322         }
323         if (gestureEvent.type() == PlatformEvent::GesturePinchUpdate && m_lastPinchAnchorCss) {
324             float newPageScaleFactor = pageScaleFactor * gestureEvent.scale();
325             blink::IntPoint anchorCss(*m_lastPinchAnchorDip.get());
326             anchorCss.scale(1.f / newPageScaleFactor, 1.f / newPageScaleFactor);
327             m_webViewImpl->setPageScaleFactor(newPageScaleFactor);
328             m_webViewImpl->setMainFrameScrollOffset(*m_lastPinchAnchorCss.get() - toIntSize(anchorCss));
329         }
330         if (gestureEvent.type() == PlatformEvent::GesturePinchEnd) {
331             m_lastPinchAnchorCss.clear();
332             m_lastPinchAnchorDip.clear();
333         }
334         return true;
335     }
336
337     InspectorController* ic = inspectorController();
338     if (!ic)
339         return false;
340
341     if (WebInputEvent::isGestureEventType(inputEvent.type) && inputEvent.type == WebInputEvent::GestureTap) {
342         // Only let GestureTab in (we only need it and we know PlatformGestureEventBuilder supports it).
343         PlatformGestureEvent gestureEvent = PlatformGestureEventBuilder(page->deprecatedLocalMainFrame()->view(), static_cast<const WebGestureEvent&>(inputEvent));
344         return ic->handleGestureEvent(toLocalFrame(page->mainFrame()), gestureEvent);
345     }
346     if (WebInputEvent::isMouseEventType(inputEvent.type) && inputEvent.type != WebInputEvent::MouseEnter) {
347         // PlatformMouseEventBuilder does not work with MouseEnter type, so we filter it out manually.
348         PlatformMouseEvent mouseEvent = PlatformMouseEventBuilder(page->deprecatedLocalMainFrame()->view(), static_cast<const WebMouseEvent&>(inputEvent));
349         return ic->handleMouseEvent(toLocalFrame(page->mainFrame()), mouseEvent);
350     }
351     if (WebInputEvent::isTouchEventType(inputEvent.type)) {
352         PlatformTouchEvent touchEvent = PlatformTouchEventBuilder(page->deprecatedLocalMainFrame()->view(), static_cast<const WebTouchEvent&>(inputEvent));
353         return ic->handleTouchEvent(toLocalFrame(page->mainFrame()), touchEvent);
354     }
355     if (WebInputEvent::isKeyboardEventType(inputEvent.type)) {
356         PlatformKeyboardEvent keyboardEvent = PlatformKeyboardEventBuilder(static_cast<const WebKeyboardEvent&>(inputEvent));
357         return ic->handleKeyboardEvent(page->deprecatedLocalMainFrame(), keyboardEvent);
358     }
359     return false;
360 }
361
362 void WebDevToolsAgentImpl::didLayout()
363 {
364     m_webViewDidLayoutOnceAfterLoad = true;
365 }
366
367 void WebDevToolsAgentImpl::setDeviceMetricsOverride(int width, int height, float deviceScaleFactor, bool mobile, bool fitWindow, float scale, float offsetX, float offsetY)
368 {
369     if (!m_deviceMetricsEnabled) {
370         m_deviceMetricsEnabled = true;
371         m_webViewImpl->setBackgroundColorOverride(Color::darkGray);
372     }
373     if (mobile)
374         enableMobileEmulation();
375     else
376         disableMobileEmulation();
377
378     WebDeviceEmulationParams params;
379     params.screenPosition = mobile ? WebDeviceEmulationParams::Mobile : WebDeviceEmulationParams::Desktop;
380     params.deviceScaleFactor = deviceScaleFactor;
381     params.viewSize = WebSize(width, height);
382     params.fitToView = fitWindow;
383     params.scale = scale;
384     params.offset = WebFloatPoint(offsetX, offsetY);
385     m_client->enableDeviceEmulation(params);
386 }
387
388 void WebDevToolsAgentImpl::clearDeviceMetricsOverride()
389 {
390     if (m_deviceMetricsEnabled) {
391         m_deviceMetricsEnabled = false;
392         m_webViewImpl->setBackgroundColorOverride(Color::transparent);
393         disableMobileEmulation();
394         m_client->disableDeviceEmulation();
395     }
396 }
397
398 void WebDevToolsAgentImpl::setTouchEventEmulationEnabled(bool enabled)
399 {
400     m_client->setTouchEventEmulationEnabled(enabled, enabled);
401     m_touchEventEmulationEnabled = enabled;
402     updatePageScaleFactorLimits();
403 }
404
405 void WebDevToolsAgentImpl::enableMobileEmulation()
406 {
407     if (m_emulateMobileEnabled)
408         return;
409     m_emulateMobileEnabled = true;
410     m_isOverlayScrollbarsEnabled = RuntimeEnabledFeatures::overlayScrollbarsEnabled();
411     RuntimeEnabledFeatures::setOverlayScrollbarsEnabled(true);
412     m_originalViewportEnabled = RuntimeEnabledFeatures::cssViewportEnabled();
413     RuntimeEnabledFeatures::setCSSViewportEnabled(true);
414     m_webViewImpl->settings()->setViewportEnabled(true);
415     m_webViewImpl->settings()->setViewportMetaEnabled(true);
416     m_webViewImpl->settings()->setShrinksViewportContentToFit(true);
417     m_webViewImpl->setIgnoreViewportTagScaleLimits(true);
418     m_webViewImpl->setZoomFactorOverride(1);
419     updatePageScaleFactorLimits();
420 }
421
422 void WebDevToolsAgentImpl::disableMobileEmulation()
423 {
424     if (!m_emulateMobileEnabled)
425         return;
426     RuntimeEnabledFeatures::setOverlayScrollbarsEnabled(m_isOverlayScrollbarsEnabled);
427     RuntimeEnabledFeatures::setCSSViewportEnabled(m_originalViewportEnabled);
428     m_webViewImpl->settings()->setViewportEnabled(false);
429     m_webViewImpl->settings()->setViewportMetaEnabled(false);
430     m_webViewImpl->settings()->setShrinksViewportContentToFit(false);
431     m_webViewImpl->setIgnoreViewportTagScaleLimits(false);
432     m_webViewImpl->setZoomFactorOverride(0);
433     m_emulateMobileEnabled = false;
434     updatePageScaleFactorLimits();
435 }
436
437 void WebDevToolsAgentImpl::updatePageScaleFactorLimits()
438 {
439     if (m_touchEventEmulationEnabled || m_emulateMobileEnabled) {
440         if (!m_pageScaleLimitsOverriden) {
441             m_originalMinimumPageScaleFactor = m_webViewImpl->minimumPageScaleFactor();
442             m_originalMaximumPageScaleFactor = m_webViewImpl->maximumPageScaleFactor();
443             m_pageScaleLimitsOverriden = true;
444         }
445         if (m_emulateMobileEnabled) {
446             m_webViewImpl->setPageScaleFactorLimits(-1, -1);
447             m_webViewImpl->setInitialPageScaleOverride(-1);
448         } else {
449             m_webViewImpl->setPageScaleFactorLimits(1, 4);
450             m_webViewImpl->setInitialPageScaleOverride(1);
451         }
452     } else {
453         if (m_pageScaleLimitsOverriden) {
454             m_pageScaleLimitsOverriden = false;
455             m_webViewImpl->setPageScaleFactorLimits(m_originalMinimumPageScaleFactor, m_originalMaximumPageScaleFactor);
456             m_webViewImpl->setInitialPageScaleOverride(1);
457         }
458     }
459 }
460
461 void WebDevToolsAgentImpl::getAllocatedObjects(HashSet<const void*>& set)
462 {
463     class CountingVisitor : public WebDevToolsAgentClient::AllocatedObjectVisitor {
464     public:
465         CountingVisitor() : m_totalObjectsCount(0)
466         {
467         }
468
469         virtual bool visitObject(const void* ptr)
470         {
471             ++m_totalObjectsCount;
472             return true;
473         }
474         size_t totalObjectsCount() const
475         {
476             return m_totalObjectsCount;
477         }
478
479     private:
480         size_t m_totalObjectsCount;
481     };
482
483     CountingVisitor counter;
484     m_client->visitAllocatedObjects(&counter);
485
486     class PointerCollector : public WebDevToolsAgentClient::AllocatedObjectVisitor {
487     public:
488         explicit PointerCollector(size_t maxObjectsCount)
489             : m_maxObjectsCount(maxObjectsCount)
490             , m_index(0)
491             , m_success(true)
492             , m_pointers(new const void*[maxObjectsCount])
493         {
494         }
495         virtual ~PointerCollector()
496         {
497             delete[] m_pointers;
498         }
499         virtual bool visitObject(const void* ptr)
500         {
501             if (m_index == m_maxObjectsCount) {
502                 m_success = false;
503                 return false;
504             }
505             m_pointers[m_index++] = ptr;
506             return true;
507         }
508
509         bool success() const { return m_success; }
510
511         void copyTo(HashSet<const void*>& set)
512         {
513             for (size_t i = 0; i < m_index; i++)
514                 set.add(m_pointers[i]);
515         }
516
517     private:
518         const size_t m_maxObjectsCount;
519         size_t m_index;
520         bool m_success;
521         const void** m_pointers;
522     };
523
524     // Double size to allow room for all objects that may have been allocated
525     // since we counted them.
526     size_t estimatedMaxObjectsCount = counter.totalObjectsCount() * 2;
527     while (true) {
528         PointerCollector collector(estimatedMaxObjectsCount);
529         m_client->visitAllocatedObjects(&collector);
530         if (collector.success()) {
531             collector.copyTo(set);
532             break;
533         }
534         estimatedMaxObjectsCount *= 2;
535     }
536 }
537
538 void WebDevToolsAgentImpl::dumpUncountedAllocatedObjects(const HashMap<const void*, size_t>& map)
539 {
540     class InstrumentedObjectSizeProvider : public WebDevToolsAgentClient::InstrumentedObjectSizeProvider {
541     public:
542         InstrumentedObjectSizeProvider(const HashMap<const void*, size_t>& map) : m_map(map) { }
543         virtual size_t objectSize(const void* ptr) const
544         {
545             HashMap<const void*, size_t>::const_iterator i = m_map.find(ptr);
546             return i == m_map.end() ? 0 : i->value;
547         }
548
549     private:
550         const HashMap<const void*, size_t>& m_map;
551     };
552
553     InstrumentedObjectSizeProvider provider(map);
554     m_client->dumpUncountedAllocatedObjects(&provider);
555 }
556
557 void WebDevToolsAgentImpl::setTraceEventCallback(const String& categoryFilter, TraceEventCallback callback)
558 {
559     m_client->setTraceEventCallback(categoryFilter, callback);
560 }
561
562 void WebDevToolsAgentImpl::resetTraceEventCallback()
563 {
564     m_client->resetTraceEventCallback();
565 }
566
567 void WebDevToolsAgentImpl::enableTracing(const String& categoryFilter)
568 {
569     m_client->enableTracing(categoryFilter);
570 }
571
572 void WebDevToolsAgentImpl::disableTracing()
573 {
574     m_client->disableTracing();
575 }
576
577 void WebDevToolsAgentImpl::startGPUEventsRecording()
578 {
579     m_client->startGPUEventsRecording();
580 }
581
582 void WebDevToolsAgentImpl::stopGPUEventsRecording()
583 {
584     m_client->stopGPUEventsRecording();
585 }
586
587 void WebDevToolsAgentImpl::processGPUEvent(const GPUEvent& event)
588 {
589     if (InspectorController* ic = inspectorController())
590         ic->processGPUEvent(event.timestamp, event.phase, event.foreign, event.usedGPUMemoryBytes, event.limitGPUMemoryBytes);
591 }
592
593 void WebDevToolsAgentImpl::dispatchKeyEvent(const PlatformKeyboardEvent& event)
594 {
595     if (!m_webViewImpl->page()->focusController().isFocused())
596         m_webViewImpl->setFocus(true);
597
598     m_generatingEvent = true;
599     WebKeyboardEvent webEvent = WebKeyboardEventBuilder(event);
600     if (!webEvent.keyIdentifier[0] && webEvent.type != WebInputEvent::Char)
601         webEvent.setKeyIdentifierFromWindowsKeyCode();
602     m_webViewImpl->handleInputEvent(webEvent);
603     m_generatingEvent = false;
604 }
605
606 void WebDevToolsAgentImpl::dispatchMouseEvent(const PlatformMouseEvent& event)
607 {
608     if (!m_webViewImpl->page()->focusController().isFocused())
609         m_webViewImpl->setFocus(true);
610
611     m_generatingEvent = true;
612     WebMouseEvent webEvent = WebMouseEventBuilder(m_webViewImpl->mainFrameImpl()->frameView(), event);
613     m_webViewImpl->handleInputEvent(webEvent);
614     m_generatingEvent = false;
615 }
616
617 void WebDevToolsAgentImpl::dispatchOnInspectorBackend(const WebString& message)
618 {
619     inspectorController()->dispatchMessageFromFrontend(message);
620 }
621
622 void WebDevToolsAgentImpl::inspectElementAt(const WebPoint& point)
623 {
624     m_webViewImpl->inspectElementAt(point);
625 }
626
627 InspectorController* WebDevToolsAgentImpl::inspectorController()
628 {
629     if (Page* page = m_webViewImpl->page())
630         return &page->inspectorController();
631     return 0;
632 }
633
634 LocalFrame* WebDevToolsAgentImpl::mainFrame()
635 {
636     if (Page* page = m_webViewImpl->page())
637         return page->deprecatedLocalMainFrame();
638     return 0;
639 }
640
641 // WebPageOverlay
642 void WebDevToolsAgentImpl::paintPageOverlay(WebCanvas* canvas)
643 {
644     InspectorController* ic = inspectorController();
645     if (ic) {
646         GraphicsContext context(canvas);
647         context.setCertainlyOpaque(false);
648         ic->drawHighlight(context);
649     }
650 }
651
652 void WebDevToolsAgentImpl::highlight()
653 {
654     if (!m_webViewDidLayoutOnceAfterLoad) {
655         m_webViewDidLayoutOnceAfterLoad = true;
656         m_webViewImpl->layout();
657     }
658     m_webViewImpl->addPageOverlay(this, OverlayZOrders::highlight);
659 }
660
661 void WebDevToolsAgentImpl::hideHighlight()
662 {
663     m_webViewImpl->removePageOverlay(this);
664 }
665
666 void WebDevToolsAgentImpl::sendMessageToFrontend(PassRefPtr<blink::JSONObject> message)
667 {
668     m_frontendMessageQueue.append(message);
669 }
670
671 void WebDevToolsAgentImpl::flush()
672 {
673     flushPendingFrontendMessages();
674 }
675
676 void WebDevToolsAgentImpl::updateInspectorStateCookie(const String& state)
677 {
678     m_client->saveAgentRuntimeState(state);
679 }
680
681 void WebDevToolsAgentImpl::setLayerTreeId(int layerTreeId)
682 {
683     m_layerTreeId = layerTreeId;
684     inspectorController()->setLayerTreeId(layerTreeId);
685 }
686
687 void WebDevToolsAgentImpl::evaluateInWebInspector(long callId, const WebString& script)
688 {
689     InspectorController* ic = inspectorController();
690     ic->evaluateForTestInFrontend(callId, script);
691 }
692
693 void WebDevToolsAgentImpl::flushPendingFrontendMessages()
694 {
695     InspectorController* ic = inspectorController();
696     ic->flushPendingFrontendMessages();
697
698     for (size_t i = 0; i < m_frontendMessageQueue.size(); ++i)
699         m_client->sendMessageToInspectorFrontend(m_frontendMessageQueue[i]->toJSONString());
700     m_frontendMessageQueue.clear();
701 }
702
703 void WebDevToolsAgentImpl::willProcessTask()
704 {
705     if (!m_attached)
706         return;
707     if (InspectorController* ic = inspectorController())
708         ic->willProcessTask();
709     TRACE_EVENT_BEGIN0(TRACE_DISABLED_BY_DEFAULT("devtools.timeline"), "Program");
710 }
711
712 void WebDevToolsAgentImpl::didProcessTask()
713 {
714     if (!m_attached)
715         return;
716     if (InspectorController* ic = inspectorController())
717         ic->didProcessTask();
718     TRACE_EVENT_END0(TRACE_DISABLED_BY_DEFAULT("devtools.timeline"), "Program");
719     flushPendingFrontendMessages();
720 }
721
722 void WebDevToolsAgent::interruptAndDispatch(MessageDescriptor* rawDescriptor)
723 {
724     // rawDescriptor can't be a PassOwnPtr because interruptAndDispatch is a WebKit API function.
725     OwnPtr<MessageDescriptor> descriptor = adoptPtr(rawDescriptor);
726     OwnPtr<DebuggerTask> task = adoptPtr(new DebuggerTask(descriptor.release()));
727     PageScriptDebugServer::interruptAndRun(task.release());
728 }
729
730 bool WebDevToolsAgent::shouldInterruptForMessage(const WebString& message)
731 {
732     String commandName;
733     if (!InspectorBackendDispatcher::getCommandName(message, &commandName))
734         return false;
735     return commandName == InspectorBackendDispatcher::commandName(InspectorBackendDispatcher::kDebugger_pauseCmd)
736         || commandName == InspectorBackendDispatcher::commandName(InspectorBackendDispatcher::kDebugger_setBreakpointCmd)
737         || commandName == InspectorBackendDispatcher::commandName(InspectorBackendDispatcher::kDebugger_setBreakpointByUrlCmd)
738         || commandName == InspectorBackendDispatcher::commandName(InspectorBackendDispatcher::kDebugger_removeBreakpointCmd)
739         || commandName == InspectorBackendDispatcher::commandName(InspectorBackendDispatcher::kDebugger_setBreakpointsActiveCmd);
740 }
741
742 void WebDevToolsAgent::processPendingMessages()
743 {
744     PageScriptDebugServer::shared().runPendingTasks();
745 }
746
747 } // namespace blink