diff -r 000000000000 -r 4f2f89ce4247 WebKit/win/WebNodeHighlight.cpp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/WebKit/win/WebNodeHighlight.cpp Fri Sep 17 09:02:29 2010 +0300 @@ -0,0 +1,293 @@ +/* + * Copyright (C) 2007 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. + * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of + * its contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY APPLE 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 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. + */ + +#include "config.h" +#include "WebNodeHighlight.h" + +#include "WebView.h" +#pragma warning(push, 0) +#include +#include +#include +#include +#include +#include +#pragma warning(pop) +#include +#include + +using namespace WebCore; + +static LPCTSTR kOverlayWindowClassName = TEXT("WebNodeHighlightWindowClass"); +static ATOM registerOverlayClass(); +static LPCTSTR kWebNodeHighlightPointerProp = TEXT("WebNodeHighlightPointer"); + +WebNodeHighlight::WebNodeHighlight(WebView* webView) + : m_inspectedWebView(webView) + , m_overlay(0) + , m_observedWindow(0) + , m_showsWhileWebViewIsVisible(false) +{ + m_inspectedWebView->viewWindow(reinterpret_cast(&m_inspectedWebViewWindow)); +} + +WebNodeHighlight::~WebNodeHighlight() +{ + if (m_observedWindow) + WindowMessageBroadcaster::removeListener(m_observedWindow, this); + if (m_inspectedWebViewWindow) + WindowMessageBroadcaster::removeListener(m_inspectedWebViewWindow, this); + + if (m_overlay) + ::DestroyWindow(m_overlay); +} + +void WebNodeHighlight::setShowsWhileWebViewIsVisible(bool shows) +{ + if (m_showsWhileWebViewIsVisible == shows) + return; + m_showsWhileWebViewIsVisible = shows; + + if (!m_showsWhileWebViewIsVisible) { + hide(); + return; + } + + bool webViewVisible = isWebViewVisible(); + + if (isShowing() == webViewVisible) + return; + + if (webViewVisible) + show(); + else + hide(); +} + +bool WebNodeHighlight::isWebViewVisible() const +{ + if (!m_inspectedWebViewWindow) + return false; + + return IsWindowVisible(m_inspectedWebViewWindow); +} + +void WebNodeHighlight::show() +{ + if (!m_overlay) { + registerOverlayClass(); + + m_overlay = ::CreateWindowEx(WS_EX_LAYERED | WS_EX_TOOLWINDOW | WS_EX_TRANSPARENT, kOverlayWindowClassName, 0, WS_POPUP, + 0, 0, 0, 0, + m_inspectedWebViewWindow, 0, 0, 0); + if (!m_overlay) + return; + + ::SetProp(m_overlay, kWebNodeHighlightPointerProp, reinterpret_cast(this)); + + m_observedWindow = GetAncestor(m_inspectedWebViewWindow, GA_ROOT); + WindowMessageBroadcaster::addListener(m_observedWindow, this); + WindowMessageBroadcaster::addListener(m_inspectedWebViewWindow, this); + } + + ASSERT(m_showsWhileWebViewIsVisible); + + update(); + SetWindowPos(m_overlay, 0, 0, 0, 0, 0, SWP_SHOWWINDOW | SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_NOACTIVATE); +} + +void WebNodeHighlight::hide() +{ + if (m_overlay) + ::ShowWindow(m_overlay, SW_HIDE); +} + +bool WebNodeHighlight::isShowing() const +{ + return m_overlay && ::IsWindowVisible(m_overlay); +} + +void WebNodeHighlight::update() +{ + ASSERT(m_overlay); + + HDC hdc = ::CreateCompatibleDC(::GetDC(m_overlay)); + if (!hdc) + return; + + RECT webViewRect; + ::GetWindowRect(m_inspectedWebViewWindow, &webViewRect); + + SIZE size; + size.cx = webViewRect.right - webViewRect.left; + size.cy = webViewRect.bottom - webViewRect.top; + + BitmapInfo bitmapInfo = BitmapInfo::createBottomUp(IntSize(size)); + + void* pixels = 0; + OwnPtr hbmp(::CreateDIBSection(hdc, &bitmapInfo, DIB_RGB_COLORS, &pixels, 0, 0)); + + ::SelectObject(hdc, hbmp.get()); + + GraphicsContext context(hdc); + + m_inspectedWebView->page()->inspectorController()->drawNodeHighlight(context); + + BLENDFUNCTION bf; + bf.BlendOp = AC_SRC_OVER; + bf.BlendFlags = 0; + bf.SourceConstantAlpha = 255; + bf.AlphaFormat = AC_SRC_ALPHA; + + POINT srcPoint; + srcPoint.x = 0; + srcPoint.y = 0; + + POINT dstPoint; + dstPoint.x = webViewRect.left; + dstPoint.y = webViewRect.top; + + ::UpdateLayeredWindow(m_overlay, ::GetDC(0), &dstPoint, &size, hdc, &srcPoint, 0, &bf, ULW_ALPHA); + + ::DeleteDC(hdc); +} + +void WebNodeHighlight::placeBehindWindow(HWND window) +{ + ASSERT(m_overlay); + SetWindowPos(m_overlay, window, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE); +} + +static ATOM registerOverlayClass() +{ + static bool haveRegisteredWindowClass = false; + + if (haveRegisteredWindowClass) + return true; + + WNDCLASSEX wcex; + + wcex.cbSize = sizeof(WNDCLASSEX); + + wcex.style = 0; + wcex.lpfnWndProc = OverlayWndProc; + wcex.cbClsExtra = 0; + wcex.cbWndExtra = 0; + wcex.hInstance = 0; + wcex.hIcon = 0; + wcex.hCursor = LoadCursor(0, IDC_ARROW); + wcex.hbrBackground = 0; + wcex.lpszMenuName = 0; + wcex.lpszClassName = kOverlayWindowClassName; + wcex.hIconSm = 0; + + haveRegisteredWindowClass = true; + + return ::RegisterClassEx(&wcex); +} + +LRESULT CALLBACK OverlayWndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) +{ + WebNodeHighlight* highlight = reinterpret_cast(::GetProp(hwnd, kWebNodeHighlightPointerProp)); + if (!highlight) + return ::DefWindowProc(hwnd, msg, wParam, lParam); + + return ::DefWindowProc(hwnd, msg, wParam, lParam); +} + +void WebNodeHighlight::onWebViewShowWindow(bool showing) +{ + if (!m_showsWhileWebViewIsVisible) + return; + + if (isShowing() == showing) + return; + + if (showing) + show(); + else + hide(); +} + +void WebNodeHighlight::onWebViewWindowPosChanged(WINDOWPOS* windowPos) +{ + bool sizing = !(windowPos->flags & SWP_NOSIZE); + + if (!sizing) + return; + + if (!isShowing()) + return; + + update(); +} + +void WebNodeHighlight::onRootWindowPosChanged(WINDOWPOS* windowPos) +{ + bool moving = !(windowPos->flags & SWP_NOMOVE); + bool sizing = !(windowPos->flags & SWP_NOSIZE); + + if (!moving) + return; + + // Size changes are handled by onWebViewWindowPosChanged. + if (sizing) + return; + + if (!isShowing()) + return; + + update(); +} + +void WebNodeHighlight::windowReceivedMessage(HWND window, UINT msg, WPARAM wParam, LPARAM lParam) +{ + if (window == m_inspectedWebViewWindow) { + switch (msg) { + case WM_SHOWWINDOW: + onWebViewShowWindow(wParam); + break; + case WM_WINDOWPOSCHANGED: + onWebViewWindowPosChanged(reinterpret_cast(lParam)); + break; + default: + break; + } + + return; + } + + ASSERT(window == m_observedWindow); + switch (msg) { + case WM_WINDOWPOSCHANGED: + onRootWindowPosChanged(reinterpret_cast(lParam)); + break; + default: + break; + } +}