Web Inspector: consolidate code that hosts the Inspector page inside a WKWebView
[WebKit-https.git] / Source / WebKit / UIProcess / mac / WebInspectorProxyMac.mm
1 /*
2  * Copyright (C) 2010, 2014 Apple 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
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #import "config.h"
27 #import "WebInspectorProxy.h"
28
29 #if PLATFORM(MAC) && WK_API_ENABLED
30
31 #import "WKInspectorPrivateMac.h"
32 #import "WKPreferencesInternal.h"
33 #import "WKProcessPoolInternal.h"
34 #import "WKViewInternal.h"
35 #import "WKWebInspectorWKWebView.h"
36 #import "WKWebViewConfigurationPrivate.h"
37 #import "WKWebViewInternal.h"
38 #import "WebInspectorUIMessages.h"
39 #import "WebPageGroup.h"
40 #import "WebPageProxy.h"
41 #import <WebCore/InspectorFrontendClientLocal.h>
42 #import <WebCore/LocalizedStrings.h>
43 #import <wtf/SoftLinking.h>
44 #import <wtf/text/Base64.h>
45
46 SOFT_LINK_STAGED_FRAMEWORK(WebInspectorUI, PrivateFrameworks, A)
47
48 using namespace WebCore;
49 using namespace WebKit;
50
51 static const NSUInteger windowStyleMask = NSWindowStyleMaskTitled | NSWindowStyleMaskClosable | NSWindowStyleMaskMiniaturizable | NSWindowStyleMaskResizable | NSWindowStyleMaskFullSizeContentView;
52
53 // The time we keep our WebView alive before closing it and its process.
54 // Reusing the WebView improves start up time for people that jump in and out of the Inspector.
55 static const Seconds webViewCloseTimeout { 1_min };
56
57 // WKWebInspectorProxyObjCAdapter is a helper ObjC object used as a delegate or notification observer
58 // for the sole purpose of getting back into the C++ code from an ObjC caller.
59
60 @interface WKWebInspectorProxyObjCAdapter ()
61
62 - (id)initWithWebInspectorProxy:(WebInspectorProxy*)inspectorProxy;
63 - (void)close;
64
65 @end
66
67 @implementation WKWebInspectorProxyObjCAdapter
68
69 - (WKInspectorRef)inspectorRef
70 {
71     return toAPI(static_cast<WebInspectorProxy*>(_inspectorProxy));
72 }
73
74 - (id)initWithWebInspectorProxy:(WebInspectorProxy*)inspectorProxy
75 {
76     ASSERT_ARG(inspectorProxy, inspectorProxy);
77
78     if (!(self = [super init]))
79         return nil;
80
81     _inspectorProxy = static_cast<void*>(inspectorProxy); // Not retained to prevent cycles
82
83     return self;
84 }
85
86 - (void)close
87 {
88     _inspectorProxy = nullptr;
89 }
90
91 - (void)windowDidMove:(NSNotification *)notification
92 {
93     static_cast<WebInspectorProxy*>(_inspectorProxy)->windowFrameDidChange();
94 }
95
96 - (void)windowDidResize:(NSNotification *)notification
97 {
98     static_cast<WebInspectorProxy*>(_inspectorProxy)->windowFrameDidChange();
99 }
100
101 - (void)windowWillClose:(NSNotification *)notification
102 {
103     static_cast<WebInspectorProxy*>(_inspectorProxy)->close();
104 }
105
106 - (void)windowDidEnterFullScreen:(NSNotification *)notification
107 {
108     static_cast<WebInspectorProxy*>(_inspectorProxy)->windowFullScreenDidChange();
109 }
110
111 - (void)windowDidExitFullScreen:(NSNotification *)notification
112 {
113     static_cast<WebInspectorProxy*>(_inspectorProxy)->windowFullScreenDidChange();
114 }
115
116 - (void)inspectedViewFrameDidChange:(NSNotification *)notification
117 {
118     // Resizing the views while inside this notification can lead to bad results when entering
119     // or exiting full screen. To avoid that we need to perform the work after a delay. We only
120     // depend on this for enforcing the height constraints, so a small delay isn't terrible. Most
121     // of the time the views will already have the correct frames because of autoresizing masks.
122
123     dispatch_after(DISPATCH_TIME_NOW, dispatch_get_main_queue(), ^{
124         if (!_inspectorProxy)
125             return;
126         static_cast<WebInspectorProxy*>(_inspectorProxy)->inspectedViewFrameDidChange();
127     });
128 }
129
130 @end
131
132 namespace WebKit {
133
134 void WebInspectorProxy::attachmentViewDidChange(NSView *oldView, NSView *newView)
135 {
136     [[NSNotificationCenter defaultCenter] removeObserver:m_inspectorProxyObjCAdapter.get() name:NSViewFrameDidChangeNotification object:oldView];
137     [[NSNotificationCenter defaultCenter] addObserver:m_inspectorProxyObjCAdapter.get() selector:@selector(inspectedViewFrameDidChange:) name:NSViewFrameDidChangeNotification object:newView];
138
139     if (m_isAttached)
140         attach(m_attachmentSide);
141 }
142
143 void WebInspectorProxy::setInspectorWindowFrame(WKRect& frame)
144 {
145     if (m_isAttached)
146         return;
147     [m_inspectorWindow setFrame:NSMakeRect(frame.origin.x, frame.origin.y, frame.size.width, frame.size.height) display:YES];
148 }
149
150 WKRect WebInspectorProxy::inspectorWindowFrame()
151 {
152     if (m_isAttached)
153         return WKRectMake(0, 0, 0, 0);
154
155     NSRect frame = m_inspectorWindow.get().frame;
156     return WKRectMake(frame.origin.x, frame.origin.y, frame.size.width, frame.size.height);
157 }
158
159 void WebInspectorProxy::closeTimerFired()
160 {
161     ASSERT(!m_isAttached || !m_inspectorWindow);
162
163     if (m_inspectorView) {
164         m_inspectorView->_page->close();
165         m_inspectorView = nil;
166     }
167
168     if (m_inspectorProxyObjCAdapter) {
169         [[NSNotificationCenter defaultCenter] removeObserver:m_inspectorProxyObjCAdapter.get()];
170
171         [m_inspectorProxyObjCAdapter close];
172         m_inspectorProxyObjCAdapter = nil;
173     }
174 }
175
176 void WebInspectorProxy::createInspectorWindow()
177 {
178     ASSERT(!m_inspectorWindow);
179
180     NSString *savedWindowFrameString = inspectedPage()->pageGroup().preferences().inspectorWindowFrame();
181     NSRect savedWindowFrame = NSRectFromString(savedWindowFrameString);
182
183     m_inspectorWindow = WebInspectorProxy::createFrontendWindow(savedWindowFrame);
184     [m_inspectorWindow setDelegate:m_inspectorProxyObjCAdapter.get()];
185
186     NSView *contentView = [m_inspectorWindow contentView];
187     [m_inspectorView setFrame:[contentView bounds]];
188     [m_inspectorView setAutoresizingMask:NSViewWidthSizable | NSViewHeightSizable];
189     [contentView addSubview:m_inspectorView.get()];
190
191     updateInspectorWindowTitle();
192 }
193
194 RetainPtr<WKWebViewConfiguration> WebInspectorProxy::createFrontendConfiguration(WebPageProxy* page, bool underTest)
195 {
196     auto configuration = adoptNS([[WKWebViewConfiguration alloc] init]);
197
198     WKPreferences *preferences = [configuration preferences];
199     preferences._allowFileAccessFromFileURLs = YES;
200     [configuration _setAllowUniversalAccessFromFileURLs:YES];
201     preferences._storageBlockingPolicy = _WKStorageBlockingPolicyAllowAll;
202     preferences._javaScriptRuntimeFlags = 0;
203
204 #ifndef NDEBUG
205     // Allow developers to inspect the Web Inspector in debug builds without changing settings.
206     preferences._developerExtrasEnabled = YES;
207     preferences._logsPageMessagesToSystemConsoleEnabled = YES;
208 #endif
209
210     if (underTest) {
211         preferences._hiddenPageDOMTimerThrottlingEnabled = NO;
212         preferences._pageVisibilityBasedProcessSuppressionEnabled = NO;
213     }
214
215     unsigned inspectorLevel = inspectorLevelForPage(page);
216     [configuration setProcessPool: ::WebKit::wrapper(inspectorProcessPool(inspectorLevel))];
217     [configuration _setGroupIdentifier:inspectorPageGroupIdentifierForPage(page)];
218
219     return configuration;
220 }
221
222 RetainPtr<NSWindow> WebInspectorProxy::createFrontendWindow(NSRect savedWindowFrame)
223 {
224     NSRect windowFrame = !NSIsEmptyRect(savedWindowFrame) ? savedWindowFrame : NSMakeRect(0, 0, initialWindowWidth, initialWindowHeight);
225     auto window = adoptNS([[NSWindow alloc] initWithContentRect:windowFrame styleMask:windowStyleMask backing:NSBackingStoreBuffered defer:NO]);
226     [window setMinSize:NSMakeSize(minimumWindowWidth, minimumWindowHeight)];
227     [window setReleasedWhenClosed:NO];
228     [window setCollectionBehavior:([window collectionBehavior] | NSWindowCollectionBehaviorFullScreenPrimary)];
229
230     CGFloat approximatelyHalfScreenSize = ([window screen].frame.size.width / 2) - 4;
231     CGFloat minimumFullScreenWidth = std::max<CGFloat>(636, approximatelyHalfScreenSize);
232     [window setMinFullScreenContentSize:NSMakeSize(minimumFullScreenWidth, minimumWindowHeight)];
233     [window setCollectionBehavior:([window collectionBehavior] | NSWindowCollectionBehaviorFullScreenAllowsTiling)];
234
235     [window setTitlebarAppearsTransparent:YES];
236
237     // Center the window if the saved frame was empty.
238     if (NSIsEmptyRect(savedWindowFrame))
239         [window center];
240
241     return window;
242 }
243
244 void WebInspectorProxy::updateInspectorWindowTitle() const
245 {
246     if (!m_inspectorWindow)
247         return;
248
249     unsigned level = inspectionLevel();
250     if (level > 1) {
251         NSString *debugTitle = [NSString stringWithFormat:WEB_UI_STRING("Web Inspector [%d] — %@", "Web Inspector window title when inspecting Web Inspector"), level, (NSString *)m_urlString];
252         [m_inspectorWindow setTitle:debugTitle];
253     } else {
254         NSString *title = [NSString stringWithFormat:WEB_UI_STRING("Web Inspector — %@", "Web Inspector window title"), (NSString *)m_urlString];
255         [m_inspectorWindow setTitle:title];
256     }
257 }
258
259 WebPageProxy* WebInspectorProxy::platformCreateInspectorPage()
260 {
261     ASSERT(inspectedPage());
262
263     m_closeTimer.stop();
264
265     if (m_inspectorView) {
266         ASSERT(m_inspectorProxyObjCAdapter);
267         return m_inspectorView->_page.get();
268     }
269
270     ASSERT(!m_inspectorView);
271     ASSERT(!m_inspectorProxyObjCAdapter);
272
273     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
274
275     NSRect initialRect;
276     if (m_isAttached) {
277         NSRect inspectedViewFrame = inspectedView.frame;
278
279         switch (m_attachmentSide) {
280         case AttachmentSide::Bottom:
281             initialRect = NSMakeRect(0, 0, NSWidth(inspectedViewFrame), inspectorPagePreferences().inspectorAttachedHeight());
282             break;
283         case AttachmentSide::Right:
284             initialRect = NSMakeRect(0, 0, inspectorPagePreferences().inspectorAttachedWidth(), NSHeight(inspectedViewFrame));
285             break;
286         case AttachmentSide::Left:
287             initialRect = NSMakeRect(0, 0, NSWidth(inspectedViewFrame) - inspectorPagePreferences().inspectorAttachedWidth(), NSHeight(inspectedViewFrame));
288             break;
289         }
290     } else {
291         initialRect = NSMakeRect(0, 0, initialWindowWidth, initialWindowHeight);
292
293         NSString *windowFrameString = inspectedPage()->pageGroup().preferences().inspectorWindowFrame();
294         NSRect windowFrame = NSRectFromString(windowFrameString);
295         if (!NSIsEmptyRect(windowFrame))
296             initialRect = [NSWindow contentRectForFrameRect:windowFrame styleMask:windowStyleMask];
297     }
298
299     m_inspectorProxyObjCAdapter = adoptNS([[WKWebInspectorProxyObjCAdapter alloc] initWithWebInspectorProxy:this]);
300     ASSERT(m_inspectorProxyObjCAdapter);
301
302     [[NSNotificationCenter defaultCenter] addObserver:m_inspectorProxyObjCAdapter.get() selector:@selector(inspectedViewFrameDidChange:) name:NSViewFrameDidChangeNotification object:inspectedView];
303
304     auto configuration = WebInspectorProxy::createFrontendConfiguration(inspectedPage(), isUnderTest());
305     m_inspectorView = adoptNS([[WKWebInspectorWKWebView alloc] initWithFrame:initialRect configuration:configuration.get()]);
306
307     return m_inspectorView->_page.get();
308 }
309
310 bool WebInspectorProxy::platformCanAttach(bool webProcessCanAttach)
311 {
312     if ([m_inspectorWindow styleMask] & NSWindowStyleMaskFullScreen)
313         return false;
314
315     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
316     if ([inspectedView isKindOfClass:[WKWebInspectorWKWebView class]])
317         return webProcessCanAttach;
318
319     static const float minimumAttachedHeight = 250;
320     static const float maximumAttachedHeightRatio = 0.75;
321     static const float minimumAttachedWidth = 500;
322
323     NSRect inspectedViewFrame = inspectedView.frame;
324
325     float maximumAttachedHeight = NSHeight(inspectedViewFrame) * maximumAttachedHeightRatio;
326     return minimumAttachedHeight <= maximumAttachedHeight && minimumAttachedWidth <= NSWidth(inspectedViewFrame);
327 }
328
329 void WebInspectorProxy::platformOpen()
330 {
331     if (m_isAttached)
332         platformAttach();
333     else
334         createInspectorWindow();
335
336     platformBringToFront();
337 }
338
339 void WebInspectorProxy::platformDidClose()
340 {
341     if (m_inspectorWindow) {
342         [m_inspectorWindow setDelegate:nil];
343         [m_inspectorWindow close];
344         m_inspectorWindow = nil;
345     }
346
347     m_closeTimer.startOneShot(webViewCloseTimeout);
348 }
349
350 void WebInspectorProxy::platformDidCloseForCrash()
351 {
352     m_closeTimer.stop();
353
354     closeTimerFired();
355 }
356
357 void WebInspectorProxy::platformInvalidate()
358 {
359     m_closeTimer.stop();
360
361     closeTimerFired();
362 }
363
364 void WebInspectorProxy::platformHide()
365 {
366     if (m_isAttached) {
367         platformDetach();
368         return;
369     }
370
371     if (m_inspectorWindow) {
372         [m_inspectorWindow setDelegate:nil];
373         [m_inspectorWindow close];
374         m_inspectorWindow = nil;
375     }
376 }
377
378 void WebInspectorProxy::platformBringToFront()
379 {
380     // If the Web Inspector is no longer in the same window as the inspected view,
381     // then we need to reopen the Inspector to get it attached to the right window.
382     // This can happen when dragging tabs to another window in Safari.
383     if (m_isAttached && m_inspectorView.get().window != inspectedPage()->platformWindow()) {
384         platformOpen();
385         return;
386     }
387
388     // FIXME <rdar://problem/10937688>: this will not bring a background tab in Safari to the front, only its window.
389     [m_inspectorView.get().window makeKeyAndOrderFront:nil];
390     [m_inspectorView.get().window makeFirstResponder:m_inspectorView.get()];
391 }
392
393 void WebInspectorProxy::platformBringInspectedPageToFront()
394 {
395     [inspectedPage()->platformWindow() makeKeyAndOrderFront:nil];
396 }
397
398 bool WebInspectorProxy::platformIsFront()
399 {
400     // FIXME <rdar://problem/10937688>: this will not return false for a background tab in Safari, only a background window.
401     return m_isVisible && [m_inspectorView.get().window isMainWindow];
402 }
403
404 void WebInspectorProxy::platformAttachAvailabilityChanged(bool available)
405 {
406     // Do nothing.
407 }
408
409 void WebInspectorProxy::platformInspectedURLChanged(const String& urlString)
410 {
411     m_urlString = urlString;
412
413     updateInspectorWindowTitle();
414 }
415
416 void WebInspectorProxy::platformSave(const String& suggestedURL, const String& content, bool base64Encoded, bool forceSaveDialog)
417 {
418     ASSERT(!suggestedURL.isEmpty());
419     
420     NSURL *platformURL = m_suggestedToActualURLMap.get(suggestedURL).get();
421     if (!platformURL) {
422         platformURL = [NSURL URLWithString:suggestedURL];
423         // The user must confirm new filenames before we can save to them.
424         forceSaveDialog = true;
425     }
426     
427     ASSERT(platformURL);
428     if (!platformURL)
429         return;
430
431     // Necessary for the block below.
432     String suggestedURLCopy = suggestedURL;
433     String contentCopy = content;
434
435     auto saveToURL = ^(NSURL *actualURL) {
436         ASSERT(actualURL);
437
438         m_suggestedToActualURLMap.set(suggestedURLCopy, actualURL);
439
440         if (base64Encoded) {
441             Vector<char> out;
442             if (!base64Decode(contentCopy, out, Base64ValidatePadding))
443                 return;
444             RetainPtr<NSData> dataContent = adoptNS([[NSData alloc] initWithBytes:out.data() length:out.size()]);
445             [dataContent writeToURL:actualURL atomically:YES];
446         } else
447             [contentCopy writeToURL:actualURL atomically:YES encoding:NSUTF8StringEncoding error:NULL];
448
449         m_inspectorPage->process().send(Messages::WebInspectorUI::DidSave([actualURL absoluteString]), m_inspectorPage->pageID());
450     };
451
452     if (!forceSaveDialog) {
453         saveToURL(platformURL);
454         return;
455     }
456
457     NSSavePanel *panel = [NSSavePanel savePanel];
458     panel.nameFieldStringValue = platformURL.lastPathComponent;
459
460     // If we have a file URL we've already saved this file to a path and
461     // can provide a good directory to show. Otherwise, use the system's
462     // default behavior for the initial directory to show in the dialog.
463     if (platformURL.isFileURL)
464         panel.directoryURL = [platformURL URLByDeletingLastPathComponent];
465
466     auto completionHandler = ^(NSInteger result) {
467         if (result == NSModalResponseCancel)
468             return;
469         ASSERT(result == NSModalResponseOK);
470         saveToURL(panel.URL);
471     };
472
473     if (m_inspectorWindow)
474         [panel beginSheetModalForWindow:m_inspectorWindow.get() completionHandler:completionHandler];
475     else
476         completionHandler([panel runModal]);
477 }
478
479 void WebInspectorProxy::platformAppend(const String& suggestedURL, const String& content)
480 {
481     ASSERT(!suggestedURL.isEmpty());
482     
483     RetainPtr<NSURL> actualURL = m_suggestedToActualURLMap.get(suggestedURL);
484     // Do not append unless the user has already confirmed this filename in save().
485     if (!actualURL)
486         return;
487
488     NSFileHandle *handle = [NSFileHandle fileHandleForWritingToURL:actualURL.get() error:NULL];
489     [handle seekToEndOfFile];
490     [handle writeData:[content dataUsingEncoding:NSUTF8StringEncoding]];
491     [handle closeFile];
492
493     m_inspectorPage->process().send(Messages::WebInspectorUI::DidAppend([actualURL absoluteString]), m_inspectorPage->pageID());
494 }
495
496 void WebInspectorProxy::windowFrameDidChange()
497 {
498     ASSERT(!m_isAttached);
499     ASSERT(m_isVisible);
500     ASSERT(m_inspectorWindow);
501
502     if (m_isAttached || !m_isVisible || !m_inspectorWindow)
503         return;
504
505     NSString *frameString = NSStringFromRect([m_inspectorWindow frame]);
506     inspectedPage()->pageGroup().preferences().setInspectorWindowFrame(frameString);
507 }
508
509 void WebInspectorProxy::windowFullScreenDidChange()
510 {
511     ASSERT(!m_isAttached);
512     ASSERT(m_isVisible);
513     ASSERT(m_inspectorWindow);
514
515     if (m_isAttached || !m_isVisible || !m_inspectorWindow)
516         return;
517
518     attachAvailabilityChanged(platformCanAttach(canAttach()));    
519 }
520
521 void WebInspectorProxy::inspectedViewFrameDidChange(CGFloat currentDimension)
522 {
523     if (!m_isVisible)
524         return;
525
526     if (!m_isAttached) {
527         // Check if the attach availability changed. We need to do this here in case
528         // the attachment view is not the WKView.
529         attachAvailabilityChanged(platformCanAttach(canAttach()));
530         return;
531     }
532
533     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
534     NSRect inspectedViewFrame = [inspectedView frame];
535     NSRect inspectorFrame = NSZeroRect;
536     NSRect parentBounds = [[inspectedView superview] bounds];
537     CGFloat inspectedViewTop = NSMaxY(inspectedViewFrame);
538
539     switch (m_attachmentSide) {
540     case AttachmentSide::Bottom: {
541         if (!currentDimension)
542             currentDimension = NSHeight([m_inspectorView frame]);
543
544         CGFloat parentHeight = NSHeight(parentBounds);
545         CGFloat inspectorHeight = InspectorFrontendClientLocal::constrainedAttachedWindowHeight(currentDimension, parentHeight);
546
547         // Preserve the top position of the inspected view so banners in Safari still work.
548         inspectedViewFrame = NSMakeRect(0, inspectorHeight, NSWidth(parentBounds), inspectedViewTop - inspectorHeight);
549         inspectorFrame = NSMakeRect(0, 0, NSWidth(inspectedViewFrame), inspectorHeight);
550         break;
551     }
552
553     case AttachmentSide::Right: {
554         if (!currentDimension)
555             currentDimension = NSWidth([m_inspectorView frame]);
556
557         CGFloat parentWidth = NSWidth(parentBounds);
558         CGFloat inspectorWidth = InspectorFrontendClientLocal::constrainedAttachedWindowWidth(currentDimension, parentWidth);
559
560         // Preserve the top position of the inspected view so banners in Safari still work. But don't use that
561         // top position for the inspector view since the banners only stretch as wide as the inspected view.
562         inspectedViewFrame = NSMakeRect(0, 0, parentWidth - inspectorWidth, inspectedViewTop);
563         CGFloat insetExcludingBanners = 0;
564         if ([inspectedView isKindOfClass:[WKView class]])
565             insetExcludingBanners = ((WKView *)inspectedView)._topContentInset - ((WKView *)inspectedView)._totalHeightOfBanners;
566         inspectorFrame = NSMakeRect(parentWidth - inspectorWidth, 0, inspectorWidth, NSHeight(parentBounds) - insetExcludingBanners);
567         break;
568     }
569
570     case AttachmentSide::Left: {
571         if (!currentDimension)
572             currentDimension = NSWidth([m_inspectorView frame]);
573
574         CGFloat parentWidth = NSWidth(parentBounds);
575         CGFloat inspectorWidth = InspectorFrontendClientLocal::constrainedAttachedWindowWidth(currentDimension, parentWidth);
576
577         // Preserve the top position of the inspected view so banners in Safari still work. But don't use that
578         // top position for the inspector view since the banners only stretch as wide as the inspected view.
579         inspectedViewFrame = NSMakeRect(inspectorWidth, 0, parentWidth - inspectorWidth, inspectedViewTop);
580         CGFloat insetExcludingBanners = 0;
581         if ([inspectedView isKindOfClass:[WKView class]])
582             insetExcludingBanners = ((WKView *)inspectedView)._topContentInset - ((WKView *)inspectedView)._totalHeightOfBanners;
583         inspectorFrame = NSMakeRect(0, 0, inspectorWidth, NSHeight(parentBounds) - insetExcludingBanners);
584         break;
585     }
586     }
587
588     if (NSEqualRects([m_inspectorView frame], inspectorFrame) && NSEqualRects([inspectedView frame], inspectedViewFrame))
589         return;
590
591     // Disable screen updates to make sure the layers for both views resize in sync.
592     [[m_inspectorView window] disableScreenUpdatesUntilFlush];
593
594     [m_inspectorView setFrame:inspectorFrame];
595     [inspectedView setFrame:inspectedViewFrame];
596 }
597
598 unsigned WebInspectorProxy::platformInspectedWindowHeight()
599 {
600     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
601     NSRect inspectedViewRect = [inspectedView frame];
602     return static_cast<unsigned>(inspectedViewRect.size.height);
603 }
604
605 unsigned WebInspectorProxy::platformInspectedWindowWidth()
606 {
607     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
608     NSRect inspectedViewRect = [inspectedView frame];
609     return static_cast<unsigned>(inspectedViewRect.size.width);
610 }
611
612 void WebInspectorProxy::platformAttach()
613 {
614     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
615
616     if (m_inspectorWindow) {
617         [m_inspectorWindow setDelegate:nil];
618         [m_inspectorWindow close];
619         m_inspectorWindow = nil;
620     }
621
622     [m_inspectorView removeFromSuperview];
623
624     CGFloat currentDimension;
625
626     switch (m_attachmentSide) {
627     case AttachmentSide::Bottom:
628         [m_inspectorView setAutoresizingMask:NSViewWidthSizable | NSViewMaxYMargin];
629         currentDimension = inspectorPagePreferences().inspectorAttachedHeight();
630         break;
631     case AttachmentSide::Right:
632         [m_inspectorView setAutoresizingMask:NSViewHeightSizable | NSViewMinXMargin];
633         currentDimension = inspectorPagePreferences().inspectorAttachedWidth();
634         break;
635     case AttachmentSide::Left:
636         [m_inspectorView setAutoresizingMask:NSViewHeightSizable | NSViewMaxXMargin];
637         currentDimension = inspectorPagePreferences().inspectorAttachedWidth();
638         break;
639     }
640
641     inspectedViewFrameDidChange(currentDimension);
642
643     [[inspectedView superview] addSubview:m_inspectorView.get() positioned:NSWindowBelow relativeTo:inspectedView];
644     [m_inspectorView.get().window makeFirstResponder:m_inspectorView.get()];
645 }
646
647 void WebInspectorProxy::platformDetach()
648 {
649     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
650
651     [m_inspectorView removeFromSuperview];
652
653     // Make sure that we size the inspected view's frame after detaching so that it takes up the space that the
654     // attached inspector used to. Preserve the top position of the inspected view so banners in Safari still work.
655
656     inspectedView.frame = NSMakeRect(0, 0, NSWidth(inspectedView.superview.bounds), NSMaxY(inspectedView.frame));
657
658     // Return early if we are not visible. This means the inspector was closed while attached
659     // and we should not create and show the inspector window.
660     if (!m_isVisible)
661         return;
662
663     createInspectorWindow();
664
665     platformBringToFront();
666 }
667
668 void WebInspectorProxy::platformSetAttachedWindowHeight(unsigned height)
669 {
670     if (!m_isAttached)
671         return;
672
673     inspectedViewFrameDidChange(height);
674 }
675
676 void WebInspectorProxy::platformSetAttachedWindowWidth(unsigned width)
677 {
678     if (!m_isAttached)
679         return;
680
681     inspectedViewFrameDidChange(width);
682 }
683
684 void WebInspectorProxy::platformStartWindowDrag()
685 {
686     m_inspectorView->_page->startWindowDrag();
687 }
688
689 String WebInspectorProxy::inspectorPageURL()
690 {
691     // Call the soft link framework function to dlopen it, then [NSBundle bundleWithIdentifier:] will work.
692     WebInspectorUILibrary();
693
694     NSString *path = [[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] pathForResource:@"Main" ofType:@"html"];
695     ASSERT([path length]);
696
697     return [[NSURL fileURLWithPath:path] absoluteString];
698 }
699
700 String WebInspectorProxy::inspectorTestPageURL()
701 {
702     // Call the soft link framework function to dlopen it, then [NSBundle bundleWithIdentifier:] will work.
703     WebInspectorUILibrary();
704
705     NSString *path = [[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] pathForResource:@"Test" ofType:@"html"];
706
707     // We might not have a Test.html in Production builds.
708     if (!path)
709         return String();
710
711     return [[NSURL fileURLWithPath:path] absoluteString];
712 }
713
714 String WebInspectorProxy::inspectorBaseURL()
715 {
716     // Call the soft link framework function to dlopen it, then [NSBundle bundleWithIdentifier:] will work.
717     WebInspectorUILibrary();
718
719     NSString *path = [[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] resourcePath];
720     ASSERT([path length]);
721
722     return [[NSURL fileURLWithPath:path] absoluteString];
723 }
724
725 } // namespace WebKit
726
727 #endif // PLATFORM(MAC) && WK_API_ENABLED