/* * Copyright (C) 2010 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. */ #import "config.h" #import "WebInspectorProxy.h" #if ENABLE(INSPECTOR) #import "WKAPICast.h" #import "WebContext.h" #import "WKInspectorPrivateMac.h" #import "WKMutableArray.h" #import "WKOpenPanelParameters.h" #import "WKOpenPanelResultListener.h" #import "WKRetainPtr.h" #import "WKURLCF.h" #import "WKViewPrivate.h" #import "WebInspectorMessages.h" #import "WebPageGroup.h" #import "WebPageProxy.h" #import "WebPreferences.h" #import "WebProcessProxy.h" #import #import #import #import #import #import #import SOFT_LINK_STAGED_FRAMEWORK(WebInspectorUI, PrivateFrameworks, A) using namespace WebCore; using namespace WebKit; // The height needed to match a typical NSToolbar. static const CGFloat windowContentBorderThickness = 55; // The margin from the top and right of the dock button (same as the full screen button). static const CGFloat dockButtonMargin = 3; // The spacing between the dock buttons. static const CGFloat dockButtonSpacing = dockButtonMargin * 2; static const NSUInteger windowStyleMask = NSTitledWindowMask | NSClosableWindowMask | NSMiniaturizableWindowMask | NSResizableWindowMask | NSTexturedBackgroundWindowMask; // WKWebInspectorProxyObjCAdapter is a helper ObjC object used as a delegate or notification observer // for the sole purpose of getting back into the C++ code from an ObjC caller. @interface WKWebInspectorProxyObjCAdapter () - (id)initWithWebInspectorProxy:(WebInspectorProxy*)inspectorProxy; - (void)close; @end @implementation WKWebInspectorProxyObjCAdapter - (WKInspectorRef)inspectorRef { return toAPI(static_cast(_inspectorProxy)); } - (id)initWithWebInspectorProxy:(WebInspectorProxy*)inspectorProxy { ASSERT_ARG(inspectorProxy, inspectorProxy); if (!(self = [super init])) return nil; _inspectorProxy = static_cast(inspectorProxy); // Not retained to prevent cycles return self; } - (IBAction)attachRight:(id)sender { static_cast(_inspectorProxy)->attach(AttachmentSideRight); } - (IBAction)attachBottom:(id)sender { static_cast(_inspectorProxy)->attach(AttachmentSideBottom); } - (void)close { _inspectorProxy = 0; } - (void)windowDidMove:(NSNotification *)notification { static_cast(_inspectorProxy)->windowFrameDidChange(); } - (void)windowDidResize:(NSNotification *)notification { static_cast(_inspectorProxy)->windowFrameDidChange(); } - (void)windowWillClose:(NSNotification *)notification { static_cast(_inspectorProxy)->close(); } - (void)inspectedViewFrameDidChange:(NSNotification *)notification { // Resizing the views while inside this notification can lead to bad results when entering // or exiting full screen. To avoid that we need to perform the work after a delay. We only // depend on this for enforcing the height constraints, so a small delay isn't terrible. Most // of the time the views will already have the correct frames because of autoresizing masks. dispatch_after(DISPATCH_TIME_NOW, dispatch_get_main_queue(), ^{ if (!_inspectorProxy) return; static_cast(_inspectorProxy)->inspectedViewFrameDidChange(); }); } @end @interface WKWebInspectorWKView : WKView @end @implementation WKWebInspectorWKView - (NSInteger)tag { return WKInspectorViewTag; } @end @interface NSWindow (AppKitDetails) - (NSCursor *)_cursorForResizeDirection:(NSInteger)direction; - (NSRect)_customTitleFrame; @end @interface WKWebInspectorWindow : NSWindow { @public RetainPtr _dockBottomButton; RetainPtr _dockRightButton; } @end @implementation WKWebInspectorWindow - (NSCursor *)_cursorForResizeDirection:(NSInteger)direction { // Don't show a resize cursor for the northeast (top right) direction if the dock button is visible. // This matches what happens when the full screen button is visible. if (direction == 1 && ![_dockRightButton isHidden]) return nil; return [super _cursorForResizeDirection:direction]; } - (NSRect)_customTitleFrame { // Adjust the title frame if needed to prevent it from intersecting the dock button. NSRect titleFrame = [super _customTitleFrame]; NSRect dockButtonFrame = _dockBottomButton.get().frame; if (NSMaxX(titleFrame) > NSMinX(dockButtonFrame) - dockButtonMargin) titleFrame.size.width -= (NSMaxX(titleFrame) - NSMinX(dockButtonFrame)) + dockButtonMargin; return titleFrame; } @end namespace WebKit { static bool inspectorReallyUsesWebKitUserInterface(WebPreferences* preferences) { // This matches a similar check in WebInspectorMac.mm. Keep them in sync. // Call the soft link framework function to dlopen it, then [NSBundle bundleWithIdentifier:] will work. WebInspectorUILibrary(); if (![[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] pathForResource:@"Main" ofType:@"html"]) return true; if (![[NSBundle bundleWithIdentifier:@"com.apple.WebCore"] pathForResource:@"inspector" ofType:@"html" inDirectory:@"inspector"]) return false; return preferences->inspectorUsesWebKitUserInterface(); } static WKRect getWindowFrame(WKPageRef, const void* clientInfo) { WebInspectorProxy* webInspectorProxy = static_cast(const_cast(clientInfo)); ASSERT(webInspectorProxy); return webInspectorProxy->inspectorWindowFrame(); } static void setWindowFrame(WKPageRef, WKRect frame, const void* clientInfo) { WebInspectorProxy* webInspectorProxy = static_cast(const_cast(clientInfo)); ASSERT(webInspectorProxy); webInspectorProxy->setInspectorWindowFrame(frame); } static unsigned long long exceededDatabaseQuota(WKPageRef, WKFrameRef, WKSecurityOriginRef, WKStringRef, WKStringRef, unsigned long long, unsigned long long, unsigned long long currentDatabaseUsage, unsigned long long expectedUsage, const void*) { return std::max(expectedUsage, currentDatabaseUsage * 1.25); } static void runOpenPanel(WKPageRef page, WKFrameRef frame, WKOpenPanelParametersRef parameters, WKOpenPanelResultListenerRef listener, const void* clientInfo) { WebInspectorProxy* webInspectorProxy = static_cast(const_cast(clientInfo)); ASSERT(webInspectorProxy); NSOpenPanel *openPanel = [NSOpenPanel openPanel]; [openPanel setAllowsMultipleSelection:WKOpenPanelParametersGetAllowsMultipleFiles(parameters)]; WKRetain(listener); // If the inspector is detached, then openPanel will be window-modal; otherwise, openPanel is opened in a new window. [openPanel beginSheetModalForWindow:webInspectorProxy->inspectorWindow() completionHandler:^(NSInteger result) { if (result == NSFileHandlingPanelOKButton) { WKMutableArrayRef fileURLs = WKMutableArrayCreate(); for (NSURL* nsURL in [openPanel URLs]) { WKURLRef wkURL = WKURLCreateWithCFURL(reinterpret_cast(nsURL)); WKArrayAppendItem(fileURLs, wkURL); WKRelease(wkURL); } WKOpenPanelResultListenerChooseFiles(listener, fileURLs); WKRelease(fileURLs); } else WKOpenPanelResultListenerCancel(listener); WKRelease(listener); }]; } void WebInspectorProxy::setInspectorWindowFrame(WKRect& frame) { if (m_isAttached) return; [m_inspectorWindow setFrame:NSMakeRect(frame.origin.x, frame.origin.y, frame.size.width, frame.size.height) display:YES]; } WKRect WebInspectorProxy::inspectorWindowFrame() { if (m_isAttached) return WKRectMake(0, 0, 0, 0); NSRect frame = m_inspectorWindow.get().frame; return WKRectMake(frame.origin.x, frame.origin.y, frame.size.width, frame.size.height); } static NSButton *createDockButton(NSString *imageName) { // Create a full screen button so we can turn it into a dock button. NSButton *dockButton = [NSWindow standardWindowButton:NSWindowFullScreenButton forStyleMask:windowStyleMask]; // Set the autoresizing mask to keep the dock button pinned to the top right corner. dockButton.autoresizingMask = NSViewMinXMargin | NSViewMinYMargin; // Get the dock image and make it a template so the button cell effects will apply. NSImage *dockImage = [[NSBundle bundleForClass:[WKWebInspectorWKView class]] imageForResource:imageName]; [dockImage setTemplate:YES]; // Set the dock image on the button cell. NSCell *dockButtonCell = dockButton.cell; dockButtonCell.image = dockImage; return [dockButton retain]; } void WebInspectorProxy::createInspectorWindow() { ASSERT(!m_inspectorWindow); NSRect windowFrame = NSMakeRect(0, 0, initialWindowWidth, initialWindowHeight); // Restore the saved window frame, if there was one. NSString *savedWindowFrameString = page()->pageGroup()->preferences()->inspectorWindowFrame(); NSRect savedWindowFrame = NSRectFromString(savedWindowFrameString); if (!NSIsEmptyRect(savedWindowFrame)) windowFrame = savedWindowFrame; WKWebInspectorWindow *window = [[WKWebInspectorWindow alloc] initWithContentRect:windowFrame styleMask:windowStyleMask backing:NSBackingStoreBuffered defer:NO]; [window setDelegate:m_inspectorProxyObjCAdapter.get()]; [window setMinSize:NSMakeSize(minimumWindowWidth, minimumWindowHeight)]; [window setReleasedWhenClosed:NO]; [window setAutorecalculatesContentBorderThickness:NO forEdge:NSMaxYEdge]; [window setContentBorderThickness:windowContentBorderThickness forEdge:NSMaxYEdge]; WKNSWindowMakeBottomCornersSquare(window); m_inspectorWindow = adoptNS(window); NSView *contentView = [window contentView]; static const int32_t firstVersionOfSafariWithDockToRightSupport = 0x02181d0d; // 536.29.13 static bool supportsDockToRight = NSVersionOfLinkTimeLibrary("Safari") >= firstVersionOfSafariWithDockToRightSupport; m_dockBottomButton = adoptNS(createDockButton(@"DockBottom")); m_dockRightButton = adoptNS(createDockButton(@"DockRight")); m_dockBottomButton.get().target = m_inspectorProxyObjCAdapter.get(); m_dockBottomButton.get().action = @selector(attachBottom:); m_dockRightButton.get().target = m_inspectorProxyObjCAdapter.get(); m_dockRightButton.get().action = @selector(attachRight:); m_dockRightButton.get().enabled = supportsDockToRight; m_dockRightButton.get().alphaValue = supportsDockToRight ? 1 : 0.5; // Store the dock buttons on the window too so it can check its visibility. window->_dockBottomButton = m_dockBottomButton; window->_dockRightButton = m_dockRightButton; // Get the frame view, the superview of the content view, and its frame. // This will be the superview of the dock button too. NSView *frameView = contentView.superview; NSRect frameViewBounds = frameView.bounds; NSSize dockButtonSize = m_dockBottomButton.get().frame.size; ASSERT(!frameView.isFlipped); // Position the dock button in the corner to match where the full screen button is normally. NSPoint dockButtonOrigin; dockButtonOrigin.x = NSMaxX(frameViewBounds) - dockButtonSize.width - dockButtonMargin; dockButtonOrigin.y = NSMaxY(frameViewBounds) - dockButtonSize.height - dockButtonMargin; m_dockRightButton.get().frameOrigin = dockButtonOrigin; dockButtonOrigin.x -= dockButtonSize.width + dockButtonSpacing; m_dockBottomButton.get().frameOrigin = dockButtonOrigin; [frameView addSubview:m_dockBottomButton.get()]; [frameView addSubview:m_dockRightButton.get()]; // Hide the dock buttons if we can't attach. m_dockBottomButton.get().hidden = !canAttach(); m_dockRightButton.get().hidden = !canAttach(); [m_inspectorView.get() setFrame:[contentView bounds]]; [m_inspectorView.get() setAutoresizingMask:NSViewWidthSizable | NSViewHeightSizable]; [contentView addSubview:m_inspectorView.get()]; // Center the window if the saved frame was empty. if (NSIsEmptyRect(savedWindowFrame)) [window center]; updateInspectorWindowTitle(); } void WebInspectorProxy::updateInspectorWindowTitle() const { if (!m_inspectorWindow) return; NSString *title = [NSString stringWithFormat:WEB_UI_STRING("Web Inspector — %@", "Web Inspector window title"), (NSString *)m_urlString]; [m_inspectorWindow.get() setTitle:title]; } WebPageProxy* WebInspectorProxy::platformCreateInspectorPage() { ASSERT(m_page); ASSERT(!m_inspectorView); NSRect initialRect; if (m_isAttached) { NSRect inspectedViewFrame = m_page->wkView().frame; switch (m_attachmentSide) { case AttachmentSideBottom: initialRect = NSMakeRect(0, 0, NSWidth(inspectedViewFrame), inspectorPageGroup()->preferences()->inspectorAttachedHeight()); break; case AttachmentSideRight: initialRect = NSMakeRect(0, 0, inspectorPageGroup()->preferences()->inspectorAttachedWidth(), NSHeight(inspectedViewFrame)); break; } } else { initialRect = NSMakeRect(0, 0, initialWindowWidth, initialWindowHeight); NSString *windowFrameString = page()->pageGroup()->preferences()->inspectorWindowFrame(); NSRect windowFrame = NSRectFromString(windowFrameString); if (!NSIsEmptyRect(windowFrame)) initialRect = [NSWindow contentRectForFrameRect:windowFrame styleMask:windowStyleMask]; } m_inspectorView = adoptNS([[WKWebInspectorWKView alloc] initWithFrame:initialRect contextRef:toAPI(page()->process()->context()) pageGroupRef:toAPI(inspectorPageGroup()) relatedToPage:toAPI(m_page)]); ASSERT(m_inspectorView); [m_inspectorView.get() setDrawsBackground:NO]; m_inspectorProxyObjCAdapter = adoptNS([[WKWebInspectorProxyObjCAdapter alloc] initWithWebInspectorProxy:this]); WebPageProxy* inspectorPage = toImpl(m_inspectorView.get().pageRef); WKPageUIClient uiClient = { kWKPageUIClientCurrentVersion, this, /* clientInfo */ 0, // createNewPage_deprecatedForUseWithV0 0, // showPage 0, // closePage 0, // takeFocus 0, // focus 0, // unfocus 0, // runJavaScriptAlert 0, // runJavaScriptConfirm 0, // runJavaScriptPrompt 0, // setStatusText 0, // mouseDidMoveOverElement_deprecatedForUseWithV0 0, // missingPluginButtonClicked_deprecatedForUseWithV0 0, // didNotHandleKeyEvent 0, // didNotHandleWheelEvent 0, // areToolbarsVisible 0, // setToolbarsVisible 0, // isMenuBarVisible 0, // setMenuBarVisible 0, // isStatusBarVisible 0, // setStatusBarVisible 0, // isResizable 0, // setResizable getWindowFrame, setWindowFrame, 0, // runBeforeUnloadConfirmPanel 0, // didDraw 0, // pageDidScroll exceededDatabaseQuota, runOpenPanel, 0, // decidePolicyForGeolocationPermissionRequest 0, // headerHeight 0, // footerHeight 0, // drawHeader 0, // drawFooter 0, // printFrame 0, // runModal 0, // unused 0, // saveDataToFileInDownloadsFolder 0, // shouldInterruptJavaScript 0, // createPage 0, // mouseDidMoveOverElement 0, // decidePolicyForNotificationPermissionRequest 0, // unavailablePluginButtonClicked_deprecatedForUseWithV1 0, // showColorPicker 0, // hideColorPicker 0, // unavailablePluginButtonClicked }; inspectorPage->initializeUIClient(&uiClient); return inspectorPage; } void WebInspectorProxy::platformOpen() { if (m_isAttached) platformAttach(); else createInspectorWindow(); platformBringToFront(); } void WebInspectorProxy::platformDidClose() { if (m_inspectorWindow) { [m_inspectorWindow.get() setDelegate:nil]; [m_inspectorWindow.get() orderOut:nil]; m_inspectorWindow = 0; } m_inspectorView = 0; [m_inspectorProxyObjCAdapter.get() close]; m_inspectorProxyObjCAdapter = 0; } void WebInspectorProxy::platformHide() { if (m_isAttached) { platformDetach(); return; } if (m_inspectorWindow) { [m_inspectorWindow.get() setDelegate:nil]; [m_inspectorWindow.get() orderOut:nil]; m_inspectorWindow = 0; } } void WebInspectorProxy::platformBringToFront() { // If the Web Inspector is no longer in the same window as the inspected view, // then we need to reopen the Inspector to get it attached to the right window. // This can happen when dragging tabs to another window in Safari. if (m_isAttached && m_inspectorView.get().window != m_page->wkView().window) { platformOpen(); return; } // FIXME : this will not bring a background tab in Safari to the front, only its window. [m_inspectorView.get().window makeKeyAndOrderFront:nil]; [m_inspectorView.get().window makeFirstResponder:m_inspectorView.get()]; } bool WebInspectorProxy::platformIsFront() { // FIXME : this will not return false for a background tab in Safari, only a background window. return m_isVisible && [m_inspectorView.get().window isMainWindow]; } void WebInspectorProxy::platformAttachAvailabilityChanged(bool available) { m_dockBottomButton.get().hidden = !available; m_dockRightButton.get().hidden = !available; } void WebInspectorProxy::platformInspectedURLChanged(const String& urlString) { m_urlString = urlString; updateInspectorWindowTitle(); } void WebInspectorProxy::platformSave(const String& suggestedURL, const String& content, bool forceSaveDialog) { ASSERT(!suggestedURL.isEmpty()); NSURL *platformURL = m_suggestedToActualURLMap.get(suggestedURL).get(); if (!platformURL) { platformURL = [NSURL URLWithString:suggestedURL]; // The user must confirm new filenames before we can save to them. forceSaveDialog = true; } ASSERT(platformURL); if (!platformURL) return; // Necessary for the block below. String suggestedURLCopy = suggestedURL; String contentCopy = content; auto saveToURL = ^(NSURL *actualURL) { ASSERT(actualURL); m_suggestedToActualURLMap.set(suggestedURLCopy, actualURL); [contentCopy writeToURL:actualURL atomically:YES encoding:NSUTF8StringEncoding error:NULL]; m_page->process()->send(Messages::WebInspector::DidSave([actualURL absoluteString]), m_page->pageID()); }; if (!forceSaveDialog) { saveToURL(platformURL); return; } NSSavePanel *panel = [NSSavePanel savePanel]; panel.nameFieldStringValue = platformURL.lastPathComponent; panel.directoryURL = [platformURL URLByDeletingLastPathComponent]; [panel beginSheetModalForWindow:m_inspectorWindow.get() completionHandler:^(NSInteger result) { if (result == NSFileHandlingPanelCancelButton) return; ASSERT(result == NSFileHandlingPanelOKButton); saveToURL(panel.URL); }]; } void WebInspectorProxy::platformAppend(const String& suggestedURL, const String& content) { ASSERT(!suggestedURL.isEmpty()); RetainPtr actualURL = m_suggestedToActualURLMap.get(suggestedURL); // Do not append unless the user has already confirmed this filename in save(). if (!actualURL) return; NSFileHandle *handle = [NSFileHandle fileHandleForWritingToURL:actualURL.get() error:NULL]; [handle seekToEndOfFile]; [handle writeData:[content dataUsingEncoding:NSUTF8StringEncoding]]; [handle closeFile]; m_page->process()->send(Messages::WebInspector::DidAppend([actualURL absoluteString]), m_page->pageID()); } void WebInspectorProxy::windowFrameDidChange() { ASSERT(!m_isAttached); ASSERT(m_isVisible); ASSERT(m_inspectorWindow); if (m_isAttached || !m_isVisible || !m_inspectorWindow) return; NSString *frameString = NSStringFromRect([m_inspectorWindow frame]); page()->pageGroup()->preferences()->setInspectorWindowFrame(frameString); } void WebInspectorProxy::inspectedViewFrameDidChange(CGFloat currentDimension) { if (!m_isAttached || !m_isVisible) return; WKView *inspectedView = m_page->wkView(); NSRect inspectedViewFrame = [inspectedView frame]; NSRect inspectorFrame = NSZeroRect; NSRect parentBounds = [[inspectedView superview] bounds]; CGFloat inspectedViewTop = NSMaxY(inspectedViewFrame); switch (m_attachmentSide) { case AttachmentSideBottom: { if (!currentDimension) currentDimension = NSHeight([m_inspectorView.get() frame]); CGFloat parentHeight = NSHeight(parentBounds); CGFloat inspectorHeight = InspectorFrontendClientLocal::constrainedAttachedWindowHeight(currentDimension, parentHeight); // Preserve the top position of the inspected view so banners in Safari still work. inspectedViewFrame = NSMakeRect(0, inspectorHeight, NSWidth(parentBounds), inspectedViewTop - inspectorHeight); inspectorFrame = NSMakeRect(0, 0, NSWidth(inspectedViewFrame), inspectorHeight); break; } case AttachmentSideRight: { if (!currentDimension) currentDimension = NSWidth([m_inspectorView.get() frame]); CGFloat parentWidth = NSWidth(parentBounds); CGFloat inspectorWidth = InspectorFrontendClientLocal::constrainedAttachedWindowWidth(currentDimension, parentWidth); // Preserve the top position of the inspected view so banners in Safari still work. But don't use that // top position for the inspector view since the banners only stretch as wide as the the inspected view. inspectedViewFrame = NSMakeRect(0, 0, parentWidth - inspectorWidth, inspectedViewTop); inspectorFrame = NSMakeRect(parentWidth - inspectorWidth, 0, inspectorWidth, NSHeight(parentBounds)); break; } } // Disable screen updates to make sure the layers for both views resize in sync. [[m_inspectorView window] disableScreenUpdatesUntilFlush]; [m_inspectorView setFrame:inspectorFrame]; [inspectedView setFrame:inspectedViewFrame]; } unsigned WebInspectorProxy::platformInspectedWindowHeight() { WKView *inspectedView = m_page->wkView(); NSRect inspectedViewRect = [inspectedView frame]; return static_cast(inspectedViewRect.size.height); } unsigned WebInspectorProxy::platformInspectedWindowWidth() { WKView *inspectedView = m_page->wkView(); NSRect inspectedViewRect = [inspectedView frame]; return static_cast(inspectedViewRect.size.width); } void WebInspectorProxy::platformAttach() { WKView *inspectedView = m_page->wkView(); [[NSNotificationCenter defaultCenter] addObserver:m_inspectorProxyObjCAdapter.get() selector:@selector(inspectedViewFrameDidChange:) name:NSViewFrameDidChangeNotification object:inspectedView]; if (m_inspectorWindow) { [m_inspectorWindow.get() setDelegate:nil]; [m_inspectorWindow.get() orderOut:nil]; m_inspectorWindow = 0; } [m_inspectorView.get() removeFromSuperview]; [m_inspectorView.get() setAutoresizingMask:NSViewWidthSizable | NSViewMaxYMargin]; CGFloat currentDimension; switch (m_attachmentSide) { case AttachmentSideBottom: currentDimension = inspectorPageGroup()->preferences()->inspectorAttachedHeight(); break; case AttachmentSideRight: currentDimension = inspectorPageGroup()->preferences()->inspectorAttachedWidth(); break; } inspectedViewFrameDidChange(currentDimension); [[inspectedView superview] addSubview:m_inspectorView.get() positioned:NSWindowBelow relativeTo:inspectedView]; [[inspectedView window] makeFirstResponder:m_inspectorView.get()]; } void WebInspectorProxy::platformDetach() { WKView *inspectedView = m_page->wkView(); [[NSNotificationCenter defaultCenter] removeObserver:m_inspectorProxyObjCAdapter.get() name:NSViewFrameDidChangeNotification object:inspectedView]; [m_inspectorView.get() removeFromSuperview]; // Make sure that we size the inspected view's frame after detaching so that it takes up the space that the // attached inspector used to. Preserve the top position of the inspected view so banners in Safari still work. inspectedView.frame = NSMakeRect(0, 0, NSWidth(inspectedView.superview.bounds), NSMaxY(inspectedView.frame)); // Return early if we are not visible. This means the inspector was closed while attached // and we should not create and show the inspector window. if (!m_isVisible) return; createInspectorWindow(); platformBringToFront(); } void WebInspectorProxy::platformSetAttachedWindowHeight(unsigned height) { if (!m_isAttached) return; inspectedViewFrameDidChange(height); } void WebInspectorProxy::platformSetAttachedWindowWidth(unsigned width) { if (!m_isAttached) return; inspectedViewFrameDidChange(width); } void WebInspectorProxy::platformSetToolbarHeight(unsigned height) { [m_inspectorWindow setContentBorderThickness:height forEdge:NSMaxYEdge]; } String WebInspectorProxy::inspectorPageURL() const { NSString *path; if (inspectorReallyUsesWebKitUserInterface(page()->pageGroup()->preferences())) path = [[NSBundle bundleWithIdentifier:@"com.apple.WebCore"] pathForResource:@"inspector" ofType:@"html" inDirectory:@"inspector"]; else path = [[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] pathForResource:@"Main" ofType:@"html"]; ASSERT([path length]); return [[NSURL fileURLWithPath:path] absoluteString]; } String WebInspectorProxy::inspectorBaseURL() const { NSString *path; if (inspectorReallyUsesWebKitUserInterface(page()->pageGroup()->preferences())) path = [[NSBundle bundleWithIdentifier:@"com.apple.WebCore"] resourcePath]; else path = [[NSBundle bundleWithIdentifier:@"com.apple.WebInspectorUI"] resourcePath]; ASSERT([path length]); return [[NSURL fileURLWithPath:path] absoluteString]; } } // namespace WebKit #endif // ENABLE(INSPECTOR)