b023fd0d39f6ef85e7470b0c21c6a1b41666a30b
[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
226     auto window = adoptNS([[NSWindow alloc] initWithContentRect:windowFrame styleMask:windowStyleMask backing:NSBackingStoreBuffered defer:NO]);
227     // [window setDelegate:m_inspectorProxyObjCAdapter.get()];
228     [window setMinSize:NSMakeSize(minimumWindowWidth, minimumWindowHeight)];
229     [window setReleasedWhenClosed:NO];
230     [window setCollectionBehavior:([window collectionBehavior] | NSWindowCollectionBehaviorFullScreenPrimary)];
231
232     CGFloat approximatelyHalfScreenSize = ([window screen].frame.size.width / 2) - 4;
233     CGFloat minimumFullScreenWidth = std::max<CGFloat>(636, approximatelyHalfScreenSize);
234     [window setMinFullScreenContentSize:NSMakeSize(minimumFullScreenWidth, minimumWindowHeight)];
235     [window setCollectionBehavior:([window collectionBehavior] | NSWindowCollectionBehaviorFullScreenAllowsTiling)];
236
237     [window setTitlebarAppearsTransparent:YES];
238
239     // Center the window if the saved frame was empty.
240     if (NSIsEmptyRect(savedWindowFrame))
241         [window center];
242
243     return window;
244 }
245
246 void WebInspectorProxy::updateInspectorWindowTitle() const
247 {
248     if (!m_inspectorWindow)
249         return;
250
251     unsigned level = inspectionLevel();
252     if (level > 1) {
253         NSString *debugTitle = [NSString stringWithFormat:WEB_UI_STRING("Web Inspector [%d] — %@", "Web Inspector window title when inspecting Web Inspector"), level, (NSString *)m_urlString];
254         [m_inspectorWindow setTitle:debugTitle];
255     } else {
256         NSString *title = [NSString stringWithFormat:WEB_UI_STRING("Web Inspector — %@", "Web Inspector window title"), (NSString *)m_urlString];
257         [m_inspectorWindow setTitle:title];
258     }
259 }
260
261 WebPageProxy* WebInspectorProxy::platformCreateInspectorPage()
262 {
263     ASSERT(inspectedPage());
264
265     m_closeTimer.stop();
266
267     if (m_inspectorView) {
268         ASSERT(m_inspectorProxyObjCAdapter);
269         return m_inspectorView->_page.get();
270     }
271
272     ASSERT(!m_inspectorView);
273     ASSERT(!m_inspectorProxyObjCAdapter);
274
275     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
276
277     NSRect initialRect;
278     if (m_isAttached) {
279         NSRect inspectedViewFrame = inspectedView.frame;
280
281         switch (m_attachmentSide) {
282         case AttachmentSide::Bottom:
283             initialRect = NSMakeRect(0, 0, NSWidth(inspectedViewFrame), inspectorPagePreferences().inspectorAttachedHeight());
284             break;
285         case AttachmentSide::Right:
286             initialRect = NSMakeRect(0, 0, inspectorPagePreferences().inspectorAttachedWidth(), NSHeight(inspectedViewFrame));
287             break;
288         case AttachmentSide::Left:
289             initialRect = NSMakeRect(0, 0, NSWidth(inspectedViewFrame) - inspectorPagePreferences().inspectorAttachedWidth(), NSHeight(inspectedViewFrame));
290             break;
291         }
292     } else {
293         initialRect = NSMakeRect(0, 0, initialWindowWidth, initialWindowHeight);
294
295         NSString *windowFrameString = inspectedPage()->pageGroup().preferences().inspectorWindowFrame();
296         NSRect windowFrame = NSRectFromString(windowFrameString);
297         if (!NSIsEmptyRect(windowFrame))
298             initialRect = [NSWindow contentRectForFrameRect:windowFrame styleMask:windowStyleMask];
299     }
300
301     m_inspectorProxyObjCAdapter = adoptNS([[WKWebInspectorProxyObjCAdapter alloc] initWithWebInspectorProxy:this]);
302     ASSERT(m_inspectorProxyObjCAdapter);
303
304     [[NSNotificationCenter defaultCenter] addObserver:m_inspectorProxyObjCAdapter.get() selector:@selector(inspectedViewFrameDidChange:) name:NSViewFrameDidChangeNotification object:inspectedView];
305
306     auto configuration = WebInspectorProxy::createFrontendConfiguration(inspectedPage(), isUnderTest());
307     m_inspectorView = adoptNS([[WKWebInspectorWKWebView alloc] initWithFrame:initialRect configuration:configuration.get()]);
308
309     return m_inspectorView->_page.get();
310 }
311
312 bool WebInspectorProxy::platformCanAttach(bool webProcessCanAttach)
313 {
314     if ([m_inspectorWindow styleMask] & NSWindowStyleMaskFullScreen)
315         return false;
316
317     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
318     if ([inspectedView isKindOfClass:[WKWebInspectorWKWebView class]])
319         return webProcessCanAttach;
320
321     static const float minimumAttachedHeight = 250;
322     static const float maximumAttachedHeightRatio = 0.75;
323     static const float minimumAttachedWidth = 500;
324
325     NSRect inspectedViewFrame = inspectedView.frame;
326
327     float maximumAttachedHeight = NSHeight(inspectedViewFrame) * maximumAttachedHeightRatio;
328     return minimumAttachedHeight <= maximumAttachedHeight && minimumAttachedWidth <= NSWidth(inspectedViewFrame);
329 }
330
331 void WebInspectorProxy::platformOpen()
332 {
333     if (m_isAttached)
334         platformAttach();
335     else
336         createInspectorWindow();
337
338     platformBringToFront();
339 }
340
341 void WebInspectorProxy::platformDidClose()
342 {
343     if (m_inspectorWindow) {
344         [m_inspectorWindow setDelegate:nil];
345         [m_inspectorWindow close];
346         m_inspectorWindow = nil;
347     }
348
349     m_closeTimer.startOneShot(webViewCloseTimeout);
350 }
351
352 void WebInspectorProxy::platformDidCloseForCrash()
353 {
354     m_closeTimer.stop();
355
356     closeTimerFired();
357 }
358
359 void WebInspectorProxy::platformInvalidate()
360 {
361     m_closeTimer.stop();
362
363     closeTimerFired();
364 }
365
366 void WebInspectorProxy::platformHide()
367 {
368     if (m_isAttached) {
369         platformDetach();
370         return;
371     }
372
373     if (m_inspectorWindow) {
374         [m_inspectorWindow setDelegate:nil];
375         [m_inspectorWindow close];
376         m_inspectorWindow = nil;
377     }
378 }
379
380 void WebInspectorProxy::platformBringToFront()
381 {
382     // If the Web Inspector is no longer in the same window as the inspected view,
383     // then we need to reopen the Inspector to get it attached to the right window.
384     // This can happen when dragging tabs to another window in Safari.
385     if (m_isAttached && m_inspectorView.get().window != inspectedPage()->platformWindow()) {
386         platformOpen();
387         return;
388     }
389
390     // FIXME <rdar://problem/10937688>: this will not bring a background tab in Safari to the front, only its window.
391     [m_inspectorView.get().window makeKeyAndOrderFront:nil];
392     [m_inspectorView.get().window makeFirstResponder:m_inspectorView.get()];
393 }
394
395 void WebInspectorProxy::platformBringInspectedPageToFront()
396 {
397     [inspectedPage()->platformWindow() makeKeyAndOrderFront:nil];
398 }
399
400 bool WebInspectorProxy::platformIsFront()
401 {
402     // FIXME <rdar://problem/10937688>: this will not return false for a background tab in Safari, only a background window.
403     return m_isVisible && [m_inspectorView.get().window isMainWindow];
404 }
405
406 void WebInspectorProxy::platformAttachAvailabilityChanged(bool available)
407 {
408     // Do nothing.
409 }
410
411 void WebInspectorProxy::platformInspectedURLChanged(const String& urlString)
412 {
413     m_urlString = urlString;
414
415     updateInspectorWindowTitle();
416 }
417
418 void WebInspectorProxy::platformSave(const String& suggestedURL, const String& content, bool base64Encoded, bool forceSaveDialog)
419 {
420     ASSERT(!suggestedURL.isEmpty());
421     
422     NSURL *platformURL = m_suggestedToActualURLMap.get(suggestedURL).get();
423     if (!platformURL) {
424         platformURL = [NSURL URLWithString:suggestedURL];
425         // The user must confirm new filenames before we can save to them.
426         forceSaveDialog = true;
427     }
428     
429     ASSERT(platformURL);
430     if (!platformURL)
431         return;
432
433     // Necessary for the block below.
434     String suggestedURLCopy = suggestedURL;
435     String contentCopy = content;
436
437     auto saveToURL = ^(NSURL *actualURL) {
438         ASSERT(actualURL);
439
440         m_suggestedToActualURLMap.set(suggestedURLCopy, actualURL);
441
442         if (base64Encoded) {
443             Vector<char> out;
444             if (!base64Decode(contentCopy, out, Base64ValidatePadding))
445                 return;
446             RetainPtr<NSData> dataContent = adoptNS([[NSData alloc] initWithBytes:out.data() length:out.size()]);
447             [dataContent writeToURL:actualURL atomically:YES];
448         } else
449             [contentCopy writeToURL:actualURL atomically:YES encoding:NSUTF8StringEncoding error:NULL];
450
451         m_inspectorPage->process().send(Messages::WebInspectorUI::DidSave([actualURL absoluteString]), m_inspectorPage->pageID());
452     };
453
454     if (!forceSaveDialog) {
455         saveToURL(platformURL);
456         return;
457     }
458
459     NSSavePanel *panel = [NSSavePanel savePanel];
460     panel.nameFieldStringValue = platformURL.lastPathComponent;
461
462     // If we have a file URL we've already saved this file to a path and
463     // can provide a good directory to show. Otherwise, use the system's
464     // default behavior for the initial directory to show in the dialog.
465     if (platformURL.isFileURL)
466         panel.directoryURL = [platformURL URLByDeletingLastPathComponent];
467
468     auto completionHandler = ^(NSInteger result) {
469         if (result == NSModalResponseCancel)
470             return;
471         ASSERT(result == NSModalResponseOK);
472         saveToURL(panel.URL);
473     };
474
475     if (m_inspectorWindow)
476         [panel beginSheetModalForWindow:m_inspectorWindow.get() completionHandler:completionHandler];
477     else
478         completionHandler([panel runModal]);
479 }
480
481 void WebInspectorProxy::platformAppend(const String& suggestedURL, const String& content)
482 {
483     ASSERT(!suggestedURL.isEmpty());
484     
485     RetainPtr<NSURL> actualURL = m_suggestedToActualURLMap.get(suggestedURL);
486     // Do not append unless the user has already confirmed this filename in save().
487     if (!actualURL)
488         return;
489
490     NSFileHandle *handle = [NSFileHandle fileHandleForWritingToURL:actualURL.get() error:NULL];
491     [handle seekToEndOfFile];
492     [handle writeData:[content dataUsingEncoding:NSUTF8StringEncoding]];
493     [handle closeFile];
494
495     m_inspectorPage->process().send(Messages::WebInspectorUI::DidAppend([actualURL absoluteString]), m_inspectorPage->pageID());
496 }
497
498 void WebInspectorProxy::windowFrameDidChange()
499 {
500     ASSERT(!m_isAttached);
501     ASSERT(m_isVisible);
502     ASSERT(m_inspectorWindow);
503
504     if (m_isAttached || !m_isVisible || !m_inspectorWindow)
505         return;
506
507     NSString *frameString = NSStringFromRect([m_inspectorWindow frame]);
508     inspectedPage()->pageGroup().preferences().setInspectorWindowFrame(frameString);
509 }
510
511 void WebInspectorProxy::windowFullScreenDidChange()
512 {
513     ASSERT(!m_isAttached);
514     ASSERT(m_isVisible);
515     ASSERT(m_inspectorWindow);
516
517     if (m_isAttached || !m_isVisible || !m_inspectorWindow)
518         return;
519
520     attachAvailabilityChanged(platformCanAttach(canAttach()));    
521 }
522
523 void WebInspectorProxy::inspectedViewFrameDidChange(CGFloat currentDimension)
524 {
525     if (!m_isVisible)
526         return;
527
528     if (!m_isAttached) {
529         // Check if the attach availability changed. We need to do this here in case
530         // the attachment view is not the WKView.
531         attachAvailabilityChanged(platformCanAttach(canAttach()));
532         return;
533     }
534
535     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
536     NSRect inspectedViewFrame = [inspectedView frame];
537     NSRect inspectorFrame = NSZeroRect;
538     NSRect parentBounds = [[inspectedView superview] bounds];
539     CGFloat inspectedViewTop = NSMaxY(inspectedViewFrame);
540
541     switch (m_attachmentSide) {
542     case AttachmentSide::Bottom: {
543         if (!currentDimension)
544             currentDimension = NSHeight([m_inspectorView frame]);
545
546         CGFloat parentHeight = NSHeight(parentBounds);
547         CGFloat inspectorHeight = InspectorFrontendClientLocal::constrainedAttachedWindowHeight(currentDimension, parentHeight);
548
549         // Preserve the top position of the inspected view so banners in Safari still work.
550         inspectedViewFrame = NSMakeRect(0, inspectorHeight, NSWidth(parentBounds), inspectedViewTop - inspectorHeight);
551         inspectorFrame = NSMakeRect(0, 0, NSWidth(inspectedViewFrame), inspectorHeight);
552         break;
553     }
554
555     case AttachmentSide::Right: {
556         if (!currentDimension)
557             currentDimension = NSWidth([m_inspectorView frame]);
558
559         CGFloat parentWidth = NSWidth(parentBounds);
560         CGFloat inspectorWidth = InspectorFrontendClientLocal::constrainedAttachedWindowWidth(currentDimension, parentWidth);
561
562         // Preserve the top position of the inspected view so banners in Safari still work. But don't use that
563         // top position for the inspector view since the banners only stretch as wide as the inspected view.
564         inspectedViewFrame = NSMakeRect(0, 0, parentWidth - inspectorWidth, inspectedViewTop);
565         CGFloat insetExcludingBanners = 0;
566         if ([inspectedView isKindOfClass:[WKView class]])
567             insetExcludingBanners = ((WKView *)inspectedView)._topContentInset - ((WKView *)inspectedView)._totalHeightOfBanners;
568         inspectorFrame = NSMakeRect(parentWidth - inspectorWidth, 0, inspectorWidth, NSHeight(parentBounds) - insetExcludingBanners);
569         break;
570     }
571
572     case AttachmentSide::Left: {
573         if (!currentDimension)
574             currentDimension = NSWidth([m_inspectorView frame]);
575
576         CGFloat parentWidth = NSWidth(parentBounds);
577         CGFloat inspectorWidth = InspectorFrontendClientLocal::constrainedAttachedWindowWidth(currentDimension, parentWidth);
578
579         // Preserve the top position of the inspected view so banners in Safari still work. But don't use that
580         // top position for the inspector view since the banners only stretch as wide as the inspected view.
581         inspectedViewFrame = NSMakeRect(inspectorWidth, 0, parentWidth - inspectorWidth, inspectedViewTop);
582         CGFloat insetExcludingBanners = 0;
583         if ([inspectedView isKindOfClass:[WKView class]])
584             insetExcludingBanners = ((WKView *)inspectedView)._topContentInset - ((WKView *)inspectedView)._totalHeightOfBanners;
585         inspectorFrame = NSMakeRect(0, 0, inspectorWidth, NSHeight(parentBounds) - insetExcludingBanners);
586         break;
587     }
588     }
589
590     if (NSEqualRects([m_inspectorView frame], inspectorFrame) && NSEqualRects([inspectedView frame], inspectedViewFrame))
591         return;
592
593     // Disable screen updates to make sure the layers for both views resize in sync.
594     [[m_inspectorView window] disableScreenUpdatesUntilFlush];
595
596     [m_inspectorView setFrame:inspectorFrame];
597     [inspectedView setFrame:inspectedViewFrame];
598 }
599
600 unsigned WebInspectorProxy::platformInspectedWindowHeight()
601 {
602     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
603     NSRect inspectedViewRect = [inspectedView frame];
604     return static_cast<unsigned>(inspectedViewRect.size.height);
605 }
606
607 unsigned WebInspectorProxy::platformInspectedWindowWidth()
608 {
609     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
610     NSRect inspectedViewRect = [inspectedView frame];
611     return static_cast<unsigned>(inspectedViewRect.size.width);
612 }
613
614 void WebInspectorProxy::platformAttach()
615 {
616     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
617
618     if (m_inspectorWindow) {
619         [m_inspectorWindow setDelegate:nil];
620         [m_inspectorWindow close];
621         m_inspectorWindow = nil;
622     }
623
624     [m_inspectorView removeFromSuperview];
625
626     CGFloat currentDimension;
627
628     switch (m_attachmentSide) {
629     case AttachmentSide::Bottom:
630         [m_inspectorView setAutoresizingMask:NSViewWidthSizable | NSViewMaxYMargin];
631         currentDimension = inspectorPagePreferences().inspectorAttachedHeight();
632         break;
633     case AttachmentSide::Right:
634         [m_inspectorView setAutoresizingMask:NSViewHeightSizable | NSViewMinXMargin];
635         currentDimension = inspectorPagePreferences().inspectorAttachedWidth();
636         break;
637     case AttachmentSide::Left:
638         [m_inspectorView setAutoresizingMask:NSViewHeightSizable | NSViewMaxXMargin];
639         currentDimension = inspectorPagePreferences().inspectorAttachedWidth();
640         break;
641     }
642
643     inspectedViewFrameDidChange(currentDimension);
644
645     [[inspectedView superview] addSubview:m_inspectorView.get() positioned:NSWindowBelow relativeTo:inspectedView];
646     [m_inspectorView.get().window makeFirstResponder:m_inspectorView.get()];
647 }
648
649 void WebInspectorProxy::platformDetach()
650 {
651     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
652
653     [m_inspectorView removeFromSuperview];
654
655     // Make sure that we size the inspected view's frame after detaching so that it takes up the space that the
656     // attached inspector used to. Preserve the top position of the inspected view so banners in Safari still work.
657
658     inspectedView.frame = NSMakeRect(0, 0, NSWidth(inspectedView.superview.bounds), NSMaxY(inspectedView.frame));
659
660     // Return early if we are not visible. This means the inspector was closed while attached
661     // and we should not create and show the inspector window.
662     if (!m_isVisible)
663         return;
664
665     createInspectorWindow();
666
667     platformBringToFront();
668 }
669
670 void WebInspectorProxy::platformSetAttachedWindowHeight(unsigned height)
671 {
672     if (!m_isAttached)
673         return;
674
675     inspectedViewFrameDidChange(height);
676 }
677
678 void WebInspectorProxy::platformSetAttachedWindowWidth(unsigned width)
679 {
680     if (!m_isAttached)
681         return;
682
683     inspectedViewFrameDidChange(width);
684 }
685
686 void WebInspectorProxy::platformStartWindowDrag()
687 {
688     m_inspectorView->_page->startWindowDrag();
689 }
690
691 String WebInspectorProxy::inspectorPageURL()
692 {
693     // Call the soft link framework function to dlopen it, then [NSBundle bundleWithIdentifier:] will work.
694     WebInspectorUILibrary();
695
696     NSString *path = [[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] pathForResource:@"Main" ofType:@"html"];
697     ASSERT([path length]);
698
699     return [[NSURL fileURLWithPath:path] absoluteString];
700 }
701
702 String WebInspectorProxy::inspectorTestPageURL()
703 {
704     // Call the soft link framework function to dlopen it, then [NSBundle bundleWithIdentifier:] will work.
705     WebInspectorUILibrary();
706
707     NSString *path = [[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] pathForResource:@"Test" ofType:@"html"];
708
709     // We might not have a Test.html in Production builds.
710     if (!path)
711         return String();
712
713     return [[NSURL fileURLWithPath:path] absoluteString];
714 }
715
716 String WebInspectorProxy::inspectorBaseURL()
717 {
718     // Call the soft link framework function to dlopen it, then [NSBundle bundleWithIdentifier:] will work.
719     WebInspectorUILibrary();
720
721     NSString *path = [[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] resourcePath];
722     ASSERT([path length]);
723
724     return [[NSURL fileURLWithPath:path] absoluteString];
725 }
726
727 } // namespace WebKit
728
729 #endif // PLATFORM(MAC) && WK_API_ENABLED