Use unified build for UIProcess
[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 "WKInspectorViewController.h"
33 #import "WKInspectorWindow.h"
34 #import "WKViewInternal.h"
35 #import "WKWebViewInternal.h"
36 #import "WebInspectorUIMessages.h"
37 #import "WebPageGroup.h"
38 #import "WebPageProxy.h"
39 #import <WebCore/InspectorFrontendClientLocal.h>
40 #import <WebCore/LocalizedStrings.h>
41 #import <wtf/SoftLinking.h>
42 #import <wtf/text/Base64.h>
43
44 SOFT_LINK_STAGED_FRAMEWORK(WebInspectorUI, PrivateFrameworks, A)
45
46 static const NSUInteger windowStyleMask = NSWindowStyleMaskTitled | NSWindowStyleMaskClosable | NSWindowStyleMaskMiniaturizable | NSWindowStyleMaskResizable | NSWindowStyleMaskFullSizeContentView;
47
48 // The time we keep our WebView alive before closing it and its process.
49 // Reusing the WebView improves start up time for people that jump in and out of the Inspector.
50 static const Seconds webViewCloseTimeout { 1_min };
51
52 @interface WKWebInspectorProxyObjCAdapter () <WKInspectorViewControllerDelegate>
53
54 - (instancetype)initWithWebInspectorProxy:(WebKit::WebInspectorProxy*)inspectorProxy;
55 - (void)invalidate;
56
57 @end
58
59 @implementation WKWebInspectorProxyObjCAdapter {
60     WebKit::WebInspectorProxy* _inspectorProxy;
61 }
62
63 - (WKInspectorRef)inspectorRef
64 {
65     return toAPI(_inspectorProxy);
66 }
67
68 - (instancetype)initWithWebInspectorProxy:(WebKit::WebInspectorProxy*)inspectorProxy
69 {
70     ASSERT_ARG(inspectorProxy, inspectorProxy);
71
72     if (!(self = [super init]))
73         return nil;
74
75     // Unretained to avoid a reference cycle.
76     _inspectorProxy = inspectorProxy;
77
78     return self;
79 }
80
81 - (void)invalidate
82 {
83     _inspectorProxy = nullptr;
84 }
85
86 - (void)windowDidMove:(NSNotification *)notification
87 {
88     if (_inspectorProxy)
89         _inspectorProxy->windowFrameDidChange();
90 }
91
92 - (void)windowDidResize:(NSNotification *)notification
93 {
94     if (_inspectorProxy)
95         _inspectorProxy->windowFrameDidChange();
96 }
97
98 - (void)windowWillClose:(NSNotification *)notification
99 {
100     if (_inspectorProxy)
101         _inspectorProxy->close();
102 }
103
104 - (void)windowDidEnterFullScreen:(NSNotification *)notification
105 {
106     if (_inspectorProxy)
107         _inspectorProxy->windowFullScreenDidChange();
108 }
109
110 - (void)windowDidExitFullScreen:(NSNotification *)notification
111 {
112     if (_inspectorProxy)
113         _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             _inspectorProxy->inspectedViewFrameDidChange();
126     });
127 }
128
129 // MARK: WKInspectorViewControllerDelegate methods
130
131 - (void)inspectorViewControllerInspectorDidCrash:(WKInspectorViewController *)inspectorViewController
132 {
133     if (_inspectorProxy)
134         _inspectorProxy->closeForCrash();
135 }
136
137 - (BOOL)inspectorViewControllerInspectorIsUnderTest:(WKInspectorViewController *)inspectorViewController
138 {
139     return _inspectorProxy ? _inspectorProxy->isUnderTest() : false;
140 }
141
142 @end
143
144 namespace WebKit {
145 using namespace WebCore;
146
147 void WebInspectorProxy::attachmentViewDidChange(NSView *oldView, NSView *newView)
148 {
149     [[NSNotificationCenter defaultCenter] removeObserver:m_objCAdapter.get() name:NSViewFrameDidChangeNotification object:oldView];
150     [[NSNotificationCenter defaultCenter] addObserver:m_objCAdapter.get() selector:@selector(inspectedViewFrameDidChange:) name:NSViewFrameDidChangeNotification object:newView];
151
152     if (m_isAttached)
153         attach(m_attachmentSide);
154 }
155
156 void WebInspectorProxy::setInspectorWindowFrame(WKRect& frame)
157 {
158     if (m_isAttached)
159         return;
160
161     [m_inspectorWindow setFrame:NSMakeRect(frame.origin.x, frame.origin.y, frame.size.width, frame.size.height) display:YES];
162 }
163
164 WKRect WebInspectorProxy::inspectorWindowFrame()
165 {
166     if (m_isAttached)
167         return WKRectMake(0, 0, 0, 0);
168
169     NSRect frame = m_inspectorWindow.get().frame;
170     return WKRectMake(frame.origin.x, frame.origin.y, frame.size.width, frame.size.height);
171 }
172
173 void WebInspectorProxy::updateInspectorWindowTitle() const
174 {
175     if (!m_inspectorWindow)
176         return;
177
178     unsigned level = inspectionLevel();
179     if (level > 1) {
180         NSString *debugTitle = [NSString stringWithFormat:WEB_UI_STRING("Web Inspector [%d] — %@", "Web Inspector window title when inspecting Web Inspector"), level, (NSString *)m_urlString];
181         [m_inspectorWindow setTitle:debugTitle];
182     } else {
183         NSString *title = [NSString stringWithFormat:WEB_UI_STRING("Web Inspector — %@", "Web Inspector window title"), (NSString *)m_urlString];
184         [m_inspectorWindow setTitle:title];
185     }
186 }
187
188 RetainPtr<NSWindow> WebInspectorProxy::createFrontendWindow(NSRect savedWindowFrame)
189 {
190     NSRect windowFrame = !NSIsEmptyRect(savedWindowFrame) ? savedWindowFrame : NSMakeRect(0, 0, initialWindowWidth, initialWindowHeight);
191     auto window = adoptNS([[WKInspectorWindow alloc] initWithContentRect:windowFrame styleMask:windowStyleMask backing:NSBackingStoreBuffered defer:NO]);
192     [window setMinSize:NSMakeSize(minimumWindowWidth, minimumWindowHeight)];
193     [window setReleasedWhenClosed:NO];
194     [window setCollectionBehavior:([window collectionBehavior] | NSWindowCollectionBehaviorFullScreenPrimary)];
195
196     CGFloat approximatelyHalfScreenSize = ([window screen].frame.size.width / 2) - 4;
197     CGFloat minimumFullScreenWidth = std::max<CGFloat>(636, approximatelyHalfScreenSize);
198     [window setMinFullScreenContentSize:NSMakeSize(minimumFullScreenWidth, minimumWindowHeight)];
199     [window setCollectionBehavior:([window collectionBehavior] | NSWindowCollectionBehaviorFullScreenAllowsTiling)];
200
201     [window setTitlebarAppearsTransparent:YES];
202
203     // Center the window if the saved frame was empty.
204     if (NSIsEmptyRect(savedWindowFrame))
205         [window center];
206
207     return window;
208 }
209
210 WebPageProxy* WebInspectorProxy::platformCreateFrontendPage()
211 {
212     ASSERT(inspectedPage());
213     ASSERT(!m_inspectorPage);
214
215     m_closeFrontendAfterInactivityTimer.stop();
216
217     if (m_inspectorViewController) {
218         ASSERT(m_objCAdapter);
219         return [m_inspectorViewController webView]->_page.get();
220     }
221
222     m_objCAdapter = adoptNS([[WKWebInspectorProxyObjCAdapter alloc] initWithWebInspectorProxy:this]);
223     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
224     [[NSNotificationCenter defaultCenter] addObserver:m_objCAdapter.get() selector:@selector(inspectedViewFrameDidChange:) name:NSViewFrameDidChangeNotification object:inspectedView];
225
226     m_inspectorViewController = adoptNS([[WKInspectorViewController alloc] initWithInspectedPage:inspectedPage()]);
227     [m_inspectorViewController.get() setDelegate:m_objCAdapter.get()];
228
229     WebPageProxy *inspectorPage = [m_inspectorViewController webView]->_page.get();
230     ASSERT(inspectorPage);
231
232     return inspectorPage;
233 }
234
235 void WebInspectorProxy::platformCreateFrontendWindow()
236 {
237     ASSERT(!m_inspectorWindow);
238
239     NSString *savedWindowFrameString = inspectedPage()->pageGroup().preferences().inspectorWindowFrame();
240     NSRect savedWindowFrame = NSRectFromString(savedWindowFrameString);
241
242     m_inspectorWindow = WebInspectorProxy::createFrontendWindow(savedWindowFrame);
243     [m_inspectorWindow setDelegate:m_objCAdapter.get()];
244
245     WKWebView *inspectorView = [m_inspectorViewController webView];
246     NSView *contentView = [m_inspectorWindow contentView];
247     inspectorView.frame = [contentView bounds];
248     [contentView addSubview:inspectorView];
249
250     updateInspectorWindowTitle();
251 }
252
253 void WebInspectorProxy::closeFrontendPage()
254 {
255     ASSERT(!m_isAttached || !m_inspectorWindow);
256
257     if (m_inspectorViewController) {
258         [m_inspectorViewController.get() setDelegate:nil];
259         m_inspectorViewController = nil;
260     }
261
262     if (m_objCAdapter) {
263         [[NSNotificationCenter defaultCenter] removeObserver:m_objCAdapter.get()];
264
265         [m_objCAdapter invalidate];
266         m_objCAdapter = nil;
267     }
268 }
269
270 void WebInspectorProxy::closeFrontendAfterInactivityTimerFired()
271 {
272     closeFrontendPage();
273 }
274
275 void WebInspectorProxy::platformCloseFrontendPageAndWindow()
276 {
277     if (m_inspectorWindow) {
278         [m_inspectorWindow setDelegate:nil];
279         [m_inspectorWindow close];
280         m_inspectorWindow = nil;
281     }
282
283     m_closeFrontendAfterInactivityTimer.startOneShot(webViewCloseTimeout);
284 }
285
286 void WebInspectorProxy::platformDidCloseForCrash()
287 {
288     m_closeFrontendAfterInactivityTimer.stop();
289
290     closeFrontendPage();
291 }
292
293 void WebInspectorProxy::platformInvalidate()
294 {
295     m_closeFrontendAfterInactivityTimer.stop();
296
297     closeFrontendPage();
298 }
299
300 void WebInspectorProxy::platformHide()
301 {
302     if (m_isAttached) {
303         platformDetach();
304         return;
305     }
306
307     if (m_inspectorWindow) {
308         [m_inspectorWindow setDelegate:nil];
309         [m_inspectorWindow close];
310         m_inspectorWindow = nil;
311     }
312 }
313
314 void WebInspectorProxy::platformBringToFront()
315 {
316     // If the Web Inspector is no longer in the same window as the inspected view,
317     // then we need to reopen the Inspector to get it attached to the right window.
318     // This can happen when dragging tabs to another window in Safari.
319     if (m_isAttached && [m_inspectorViewController webView].window != inspectedPage()->platformWindow()) {
320         open();
321         return;
322     }
323
324     // FIXME <rdar://problem/10937688>: this will not bring a background tab in Safari to the front, only its window.
325     [[m_inspectorViewController webView].window makeKeyAndOrderFront:nil];
326     [[m_inspectorViewController webView].window makeFirstResponder:[m_inspectorViewController webView]];
327 }
328
329 void WebInspectorProxy::platformBringInspectedPageToFront()
330 {
331     [inspectedPage()->platformWindow() makeKeyAndOrderFront:nil];
332 }
333
334 bool WebInspectorProxy::platformIsFront()
335 {
336     // FIXME <rdar://problem/10937688>: this will not return false for a background tab in Safari, only a background window.
337     return m_isVisible && [m_inspectorViewController webView].window.isMainWindow;
338 }
339
340 bool WebInspectorProxy::platformCanAttach(bool webProcessCanAttach)
341 {
342     if ([m_inspectorWindow styleMask] & NSWindowStyleMaskFullScreen)
343         return false;
344
345     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
346     if ([WKInspectorViewController viewIsInspectorWebView:inspectedView])
347         return webProcessCanAttach;
348
349     static const float minimumAttachedHeight = 250;
350     static const float maximumAttachedHeightRatio = 0.75;
351     static const float minimumAttachedWidth = 500;
352
353     NSRect inspectedViewFrame = inspectedView.frame;
354
355     float maximumAttachedHeight = NSHeight(inspectedViewFrame) * maximumAttachedHeightRatio;
356     return minimumAttachedHeight <= maximumAttachedHeight && minimumAttachedWidth <= NSWidth(inspectedViewFrame);
357 }
358
359 void WebInspectorProxy::platformAttachAvailabilityChanged(bool available)
360 {
361     // Do nothing.
362 }
363
364 void WebInspectorProxy::platformInspectedURLChanged(const String& urlString)
365 {
366     m_urlString = urlString;
367
368     updateInspectorWindowTitle();
369 }
370
371 void WebInspectorProxy::platformSave(const String& suggestedURL, const String& content, bool base64Encoded, bool forceSaveDialog)
372 {
373     ASSERT(!suggestedURL.isEmpty());
374     
375     NSURL *platformURL = m_suggestedToActualURLMap.get(suggestedURL).get();
376     if (!platformURL) {
377         platformURL = [NSURL URLWithString:suggestedURL];
378         // The user must confirm new filenames before we can save to them.
379         forceSaveDialog = true;
380     }
381     
382     ASSERT(platformURL);
383     if (!platformURL)
384         return;
385
386     // Necessary for the block below.
387     String suggestedURLCopy = suggestedURL;
388     String contentCopy = content;
389
390     auto saveToURL = ^(NSURL *actualURL) {
391         ASSERT(actualURL);
392
393         m_suggestedToActualURLMap.set(suggestedURLCopy, actualURL);
394
395         if (base64Encoded) {
396             Vector<char> out;
397             if (!base64Decode(contentCopy, out, Base64ValidatePadding))
398                 return;
399             RetainPtr<NSData> dataContent = adoptNS([[NSData alloc] initWithBytes:out.data() length:out.size()]);
400             [dataContent writeToURL:actualURL atomically:YES];
401         } else
402             [contentCopy writeToURL:actualURL atomically:YES encoding:NSUTF8StringEncoding error:NULL];
403
404         m_inspectorPage->process().send(Messages::WebInspectorUI::DidSave([actualURL absoluteString]), m_inspectorPage->pageID());
405     };
406
407     if (!forceSaveDialog) {
408         saveToURL(platformURL);
409         return;
410     }
411
412     NSSavePanel *panel = [NSSavePanel savePanel];
413     panel.nameFieldStringValue = platformURL.lastPathComponent;
414
415     // If we have a file URL we've already saved this file to a path and
416     // can provide a good directory to show. Otherwise, use the system's
417     // default behavior for the initial directory to show in the dialog.
418     if (platformURL.isFileURL)
419         panel.directoryURL = [platformURL URLByDeletingLastPathComponent];
420
421     auto completionHandler = ^(NSInteger result) {
422         if (result == NSModalResponseCancel)
423             return;
424         ASSERT(result == NSModalResponseOK);
425         saveToURL(panel.URL);
426     };
427
428     if (m_inspectorWindow)
429         [panel beginSheetModalForWindow:m_inspectorWindow.get() completionHandler:completionHandler];
430     else
431         completionHandler([panel runModal]);
432 }
433
434 void WebInspectorProxy::platformAppend(const String& suggestedURL, const String& content)
435 {
436     ASSERT(!suggestedURL.isEmpty());
437     
438     RetainPtr<NSURL> actualURL = m_suggestedToActualURLMap.get(suggestedURL);
439     // Do not append unless the user has already confirmed this filename in save().
440     if (!actualURL)
441         return;
442
443     NSFileHandle *handle = [NSFileHandle fileHandleForWritingToURL:actualURL.get() error:NULL];
444     [handle seekToEndOfFile];
445     [handle writeData:[content dataUsingEncoding:NSUTF8StringEncoding]];
446     [handle closeFile];
447
448     m_inspectorPage->process().send(Messages::WebInspectorUI::DidAppend([actualURL absoluteString]), m_inspectorPage->pageID());
449 }
450
451 void WebInspectorProxy::windowFrameDidChange()
452 {
453     ASSERT(!m_isAttached);
454     ASSERT(m_isVisible);
455     ASSERT(m_inspectorWindow);
456
457     if (m_isAttached || !m_isVisible || !m_inspectorWindow)
458         return;
459
460     NSString *frameString = NSStringFromRect([m_inspectorWindow frame]);
461     inspectedPage()->pageGroup().preferences().setInspectorWindowFrame(frameString);
462 }
463
464 void WebInspectorProxy::windowFullScreenDidChange()
465 {
466     ASSERT(!m_isAttached);
467     ASSERT(m_isVisible);
468     ASSERT(m_inspectorWindow);
469
470     if (m_isAttached || !m_isVisible || !m_inspectorWindow)
471         return;
472
473     attachAvailabilityChanged(platformCanAttach(canAttach()));    
474 }
475
476 void WebInspectorProxy::inspectedViewFrameDidChange(CGFloat currentDimension)
477 {
478     if (!m_isVisible)
479         return;
480
481     if (!m_isAttached) {
482         // Check if the attach availability changed. We need to do this here in case
483         // the attachment view is not the WKView.
484         attachAvailabilityChanged(platformCanAttach(canAttach()));
485         return;
486     }
487
488     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
489     WKWebView *inspectorView = [m_inspectorViewController webView];
490
491     NSRect inspectedViewFrame = inspectedView.frame;
492     NSRect oldInspectorViewFrame = inspectorView.frame;
493     NSRect newInspectorViewFrame = NSZeroRect;
494     NSRect parentBounds = inspectedView.superview.bounds;
495     CGFloat inspectedViewTop = NSMaxY(inspectedViewFrame);
496
497     switch (m_attachmentSide) {
498     case AttachmentSide::Bottom: {
499         if (!currentDimension)
500             currentDimension = NSHeight(oldInspectorViewFrame);
501
502         CGFloat parentHeight = NSHeight(parentBounds);
503         CGFloat inspectorHeight = InspectorFrontendClientLocal::constrainedAttachedWindowHeight(currentDimension, parentHeight);
504
505         // Preserve the top position of the inspected view so banners in Safari still work.
506         inspectedViewFrame = NSMakeRect(0, inspectorHeight, NSWidth(parentBounds), inspectedViewTop - inspectorHeight);
507         newInspectorViewFrame = NSMakeRect(0, 0, NSWidth(inspectedViewFrame), inspectorHeight);
508         break;
509     }
510
511     case AttachmentSide::Right: {
512         if (!currentDimension)
513             currentDimension = NSWidth(oldInspectorViewFrame);
514
515         CGFloat parentWidth = NSWidth(parentBounds);
516         CGFloat inspectorWidth = InspectorFrontendClientLocal::constrainedAttachedWindowWidth(currentDimension, parentWidth);
517
518         // Preserve the top position of the inspected view so banners in Safari still work. But don't use that
519         // top position for the inspector view since the banners only stretch as wide as the inspected view.
520         inspectedViewFrame = NSMakeRect(0, 0, parentWidth - inspectorWidth, inspectedViewTop);
521         CGFloat insetExcludingBanners = 0;
522         if ([inspectedView isKindOfClass:[WKView class]])
523             insetExcludingBanners = ((WKView *)inspectedView)._topContentInset - ((WKView *)inspectedView)._totalHeightOfBanners;
524         newInspectorViewFrame = NSMakeRect(parentWidth - inspectorWidth, 0, inspectorWidth, NSHeight(parentBounds) - insetExcludingBanners);
525         break;
526     }
527
528     case AttachmentSide::Left: {
529         if (!currentDimension)
530             currentDimension = NSWidth(oldInspectorViewFrame);
531
532         CGFloat parentWidth = NSWidth(parentBounds);
533         CGFloat inspectorWidth = InspectorFrontendClientLocal::constrainedAttachedWindowWidth(currentDimension, parentWidth);
534
535         // Preserve the top position of the inspected view so banners in Safari still work. But don't use that
536         // top position for the inspector view since the banners only stretch as wide as the inspected view.
537         inspectedViewFrame = NSMakeRect(inspectorWidth, 0, parentWidth - inspectorWidth, inspectedViewTop);
538         CGFloat insetExcludingBanners = 0;
539         if ([inspectedView isKindOfClass:[WKView class]])
540             insetExcludingBanners = ((WKView *)inspectedView)._topContentInset - ((WKView *)inspectedView)._totalHeightOfBanners;
541         newInspectorViewFrame = NSMakeRect(0, 0, inspectorWidth, NSHeight(parentBounds) - insetExcludingBanners);
542         break;
543     }
544     }
545
546     if (NSEqualRects(oldInspectorViewFrame, newInspectorViewFrame) && NSEqualRects([inspectedView frame], inspectedViewFrame))
547         return;
548
549     // Disable screen updates to make sure the layers for both views resize in sync.
550     [inspectorView.window disableScreenUpdatesUntilFlush];
551
552     [inspectorView setFrame:newInspectorViewFrame];
553     [inspectedView setFrame:inspectedViewFrame];
554 }
555
556 unsigned WebInspectorProxy::platformInspectedWindowHeight()
557 {
558     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
559     NSRect inspectedViewRect = [inspectedView frame];
560     return static_cast<unsigned>(inspectedViewRect.size.height);
561 }
562
563 unsigned WebInspectorProxy::platformInspectedWindowWidth()
564 {
565     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
566     NSRect inspectedViewRect = [inspectedView frame];
567     return static_cast<unsigned>(inspectedViewRect.size.width);
568 }
569
570 void WebInspectorProxy::platformAttach()
571 {
572     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
573     WKWebView *inspectorView = [m_inspectorViewController webView];
574
575     if (m_inspectorWindow) {
576         [m_inspectorWindow setDelegate:nil];
577         [m_inspectorWindow close];
578         m_inspectorWindow = nil;
579     }
580
581     [inspectorView removeFromSuperview];
582
583     CGFloat currentDimension;
584     switch (m_attachmentSide) {
585     case AttachmentSide::Bottom:
586         [inspectorView setAutoresizingMask:NSViewWidthSizable | NSViewMaxYMargin];
587         currentDimension = inspectorPagePreferences().inspectorAttachedHeight();
588         break;
589     case AttachmentSide::Right:
590         [inspectorView setAutoresizingMask:NSViewHeightSizable | NSViewMinXMargin];
591         currentDimension = inspectorPagePreferences().inspectorAttachedWidth();
592         break;
593     case AttachmentSide::Left:
594         [inspectorView setAutoresizingMask:NSViewHeightSizable | NSViewMaxXMargin];
595         currentDimension = inspectorPagePreferences().inspectorAttachedWidth();
596         break;
597     }
598
599     inspectedViewFrameDidChange(currentDimension);
600
601     [inspectedView.superview addSubview:inspectorView positioned:NSWindowBelow relativeTo:inspectedView];
602     [inspectorView.window makeFirstResponder:inspectorView];
603 }
604
605 void WebInspectorProxy::platformDetach()
606 {
607     NSView *inspectedView = inspectedPage()->inspectorAttachmentView();
608     WKWebView *inspectorView = [m_inspectorViewController webView];
609
610     [inspectorView removeFromSuperview];
611
612     [inspectorView setAutoresizingMask:NSViewWidthSizable | NSViewHeightSizable];
613
614     // Make sure that we size the inspected view's frame after detaching so that it takes up the space that the
615     // attached inspector used to. Preserve the top position of the inspected view so banners in Safari still work.
616
617     inspectedView.frame = NSMakeRect(0, 0, NSWidth(inspectedView.superview.bounds), NSMaxY(inspectedView.frame));
618
619     // Return early if we are not visible. This means the inspector was closed while attached
620     // and we should not create and show the inspector window.
621     if (!m_isVisible)
622         return;
623
624     open();
625 }
626
627 void WebInspectorProxy::platformSetAttachedWindowHeight(unsigned height)
628 {
629     if (!m_isAttached)
630         return;
631
632     inspectedViewFrameDidChange(height);
633 }
634
635 void WebInspectorProxy::platformSetAttachedWindowWidth(unsigned width)
636 {
637     if (!m_isAttached)
638         return;
639
640     inspectedViewFrameDidChange(width);
641 }
642
643 void WebInspectorProxy::platformStartWindowDrag()
644 {
645     [m_inspectorViewController webView]->_page->startWindowDrag();
646 }
647
648 String WebInspectorProxy::inspectorPageURL()
649 {
650     // Call the soft link framework function to dlopen it, then [NSBundle bundleWithIdentifier:] will work.
651     WebInspectorUILibrary();
652
653     NSString *path = [[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] pathForResource:@"Main" ofType:@"html"];
654     ASSERT([path length]);
655
656     return [[NSURL fileURLWithPath:path isDirectory:NO] absoluteString];
657 }
658
659 String WebInspectorProxy::inspectorTestPageURL()
660 {
661     // Call the soft link framework function to dlopen it, then [NSBundle bundleWithIdentifier:] will work.
662     WebInspectorUILibrary();
663
664     NSString *path = [[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] pathForResource:@"Test" ofType:@"html"];
665
666     // We might not have a Test.html in Production builds.
667     if (!path)
668         return String();
669
670     return [[NSURL fileURLWithPath:path isDirectory:NO] absoluteString];
671 }
672
673 String WebInspectorProxy::inspectorBaseURL()
674 {
675     // Call the soft link framework function to dlopen it, then [NSBundle bundleWithIdentifier:] will work.
676     WebInspectorUILibrary();
677
678     NSString *path = [[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] resourcePath];
679     ASSERT([path length]);
680
681     return [[NSURL fileURLWithPath:path isDirectory:YES] absoluteString];
682 }
683
684 } // namespace WebKit
685
686 #endif // PLATFORM(MAC) && WK_API_ENABLED