/* * 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. */ #include "WindowedPluginTest.h" #include "PluginObject.h" using namespace std; // NPN_InvalidateRect should invalidate the plugin's HWND. static const wchar_t instancePointerProperty[] = L"org.webkit.TestNetscapePlugin.NPNInvalidateRectInvalidatesWindow.InstancePointer"; class TemporaryWindowMover { public: TemporaryWindowMover(HWND); ~TemporaryWindowMover(); bool moveSucceeded() const { return m_moveSucceeded; } private: static const UINT standardSetWindowPosFlags = SWP_NOACTIVATE | SWP_NOOWNERZORDER | SWP_NOSIZE | SWP_NOZORDER; bool m_moveSucceeded; HWND m_window; RECT m_savedWindowRect; }; TemporaryWindowMover::TemporaryWindowMover(HWND window) : m_window(window) { m_moveSucceeded = false; if (!::GetWindowRect(m_window, &m_savedWindowRect)) return; if (!::SetWindowPos(m_window, 0, 0, 0, 0, 0, SWP_SHOWWINDOW | standardSetWindowPosFlags)) return; m_moveSucceeded = true; }; TemporaryWindowMover::~TemporaryWindowMover() { if (!m_moveSucceeded) return; ::SetWindowPos(m_window, 0, m_savedWindowRect.left, m_savedWindowRect.top, 0, 0, SWP_HIDEWINDOW | standardSetWindowPosFlags); } class NPNInvalidateRectInvalidatesWindow : public WindowedPluginTest { public: NPNInvalidateRectInvalidatesWindow(NPP, const string& identifier); ~NPNInvalidateRectInvalidatesWindow(); private: virtual LRESULT wndProc(UINT message, WPARAM, LPARAM, bool& handled); void onPaint(); void testInvalidateRect(); virtual NPError NPP_SetWindow(NPP, NPWindow*); TemporaryWindowMover* m_windowMover; }; NPNInvalidateRectInvalidatesWindow::NPNInvalidateRectInvalidatesWindow(NPP npp, const string& identifier) : WindowedPluginTest(npp, identifier) , m_windowMover(0) { } NPNInvalidateRectInvalidatesWindow::~NPNInvalidateRectInvalidatesWindow() { delete m_windowMover; } NPError NPNInvalidateRectInvalidatesWindow::NPP_SetWindow(NPP instance, NPWindow* npWindow) { NPError error = WindowedPluginTest::NPP_SetWindow(instance, npWindow); if (error != NPERR_NO_ERROR) return error; if (!window()) return NPERR_NO_ERROR; // The test harness's window (the one that contains the WebView) is off-screen and hidden. // We need to move it on-screen and make it visible in order for the plugin's window to // accumulate an update region when the DWM is disabled. HWND testHarnessWindow = ::GetAncestor(window(), GA_ROOT); if (!testHarnessWindow) { pluginLog(instance, "Failed to get test harness window"); return NPERR_GENERIC_ERROR; } m_windowMover = new TemporaryWindowMover(testHarnessWindow); if (!m_windowMover->moveSucceeded()) { pluginLog(instance, "Failed to move test harness window on-screen"); return NPERR_GENERIC_ERROR; } // Wait until we receive a WM_PAINT message to ensure that the window is on-screen before we do // the NPN_InvalidateRect test. waitUntilDone(); return NPERR_NO_ERROR; } LRESULT NPNInvalidateRectInvalidatesWindow::wndProc(UINT message, WPARAM wParam, LPARAM lParam, bool& handled) { if (message == WM_PAINT) onPaint(); handled = false; return 0; } void NPNInvalidateRectInvalidatesWindow::onPaint() { testInvalidateRect(); notifyDone(); delete m_windowMover; m_windowMover = 0; } void NPNInvalidateRectInvalidatesWindow::testInvalidateRect() { RECT clientRect; if (!::GetClientRect(window(), &clientRect)) { pluginLog(m_npp, "::GetClientRect failed"); return; } if (::IsRectEmpty(&clientRect)) { pluginLog(m_npp, "Plugin's HWND has not been sized when NPP_SetWindow is called"); return; } // Clear the invalid region. if (!::ValidateRect(window(), 0)) { pluginLog(m_npp, "::ValidateRect failed"); return; } // Invalidate our lower-right quadrant. NPRect rectToInvalidate; rectToInvalidate.left = (clientRect.right - clientRect.left) / 2; rectToInvalidate.top = (clientRect.bottom - clientRect.top) / 2; rectToInvalidate.right = clientRect.right; rectToInvalidate.bottom = clientRect.bottom; NPN_InvalidateRect(&rectToInvalidate); RECT invalidRect; if (!::GetUpdateRect(window(), &invalidRect, FALSE)) { pluginLog(m_npp, "::GetUpdateRect failed"); return; } if (invalidRect.left != rectToInvalidate.left || invalidRect.top != rectToInvalidate.top || invalidRect.right != rectToInvalidate.right || invalidRect.bottom != rectToInvalidate.bottom) { pluginLog(m_npp, "Expected invalid rect {left=%u, top=%u, right=%u, bottom=%u}, but got {left=%d, top=%d, right=%d, bottom=%d}", rectToInvalidate.left, rectToInvalidate.top, rectToInvalidate.right, rectToInvalidate.bottom, invalidRect.left, invalidRect.top, invalidRect.right, invalidRect.bottom); return; } pluginLog(m_npp, "Plugin's HWND has been invalidated as expected"); } static PluginTest::Register<NPNInvalidateRectInvalidatesWindow> registrar("npn-invalidate-rect-invalidates-window");