/*
* 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 "config.h"
#include "HostWindow.h"
#include "Test.h"
#include <WebCore/COMPtr.h>
#include <WebKit/WebKit.h>
#include <WebKit/WebKitCOMAPI.h>
#include <wtf/PassOwnPtr.h>
namespace WebKitAPITest {
template <typename T>
static HRESULT WebKitCreateInstance(REFCLSID clsid, T** object)
{
return WebKitCreateInstance(clsid, 0, __uuidof(T), reinterpret_cast<void**>(object));
}
static int webViewCount()
{
COMPtr<IWebKitStatistics> statistics;
if (FAILED(WebKitCreateInstance(__uuidof(WebKitStatistics), &statistics)))
return -1;
int count;
if (FAILED(statistics->webViewCount(&count)))
return -1;
return count;
}
static void createAndInitializeWebView(COMPtr<IWebView>& outWebView, HostWindow& window, HWND& viewWindow)
{
COMPtr<IWebView> webView;
TEST_ASSERT(SUCCEEDED(WebKitCreateInstance(__uuidof(WebView), &webView)));
TEST_ASSERT(window.initialize());
TEST_ASSERT(SUCCEEDED(webView->setHostWindow(reinterpret_cast<OLE_HANDLE>(window.window()))));
TEST_ASSERT(SUCCEEDED(webView->initWithFrame(window.clientRect(), 0, 0)));
COMPtr<IWebViewPrivate> viewPrivate(Query, webView);
TEST_ASSERT(viewPrivate);
TEST_ASSERT(SUCCEEDED(viewPrivate->viewWindow(reinterpret_cast<OLE_HANDLE*>(&viewWindow))));
TEST_ASSERT(IsWindow(viewWindow));
outWebView.adoptRef(webView.releaseRef());
}
static void runMessagePump(DWORD timeoutMilliseconds)
{
DWORD startTickCount = GetTickCount();
MSG msg;
BOOL result;
while ((result = PeekMessageW(&msg, 0, 0, 0, PM_REMOVE)) && GetTickCount() - startTickCount <= timeoutMilliseconds) {
if (result == -1)
break;
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
static void finishWebViewDestructionTest(COMPtr<IWebView>& webView, HWND viewWindow)
{
// Allow window messages to be processed, because in some cases that would trigger a crash (e.g., <http://webkit.org/b/32827>).
runMessagePump(50);
// We haven't crashed. Release the WebView and ensure that its view window has been destroyed and the WebView doesn't leak.
int currentWebViewCount = webViewCount();
TEST_ASSERT(currentWebViewCount > 0);
webView = 0;
TEST_ASSERT(webViewCount() == currentWebViewCount - 1);
TEST_ASSERT(!IsWindow(viewWindow));
}
// Tests that releasing a WebView without calling IWebView::initWithFrame works.
TEST(WebViewDestruction, NoInitWithFrame)
{
COMPtr<IWebView> webView;
TEST_ASSERT(SUCCEEDED(WebKitCreateInstance(__uuidof(WebView), &webView)));
finishWebViewDestructionTest(webView, 0);
}
TEST(WebViewDestruction, CloseWithoutInitWithFrame)
{
COMPtr<IWebView> webView;
TEST_ASSERT(SUCCEEDED(WebKitCreateInstance(__uuidof(WebView), &webView)));
TEST_ASSERT(SUCCEEDED(webView->close()));
finishWebViewDestructionTest(webView, 0);
}
// Tests that releasing a WebView without calling IWebView::close or DestroyWindow doesn't leak. <http://webkit.org/b/33162>
TEST(WebViewDestruction, NoCloseOrDestroyViewWindow)
{
COMPtr<IWebView> webView;
HostWindow window;
HWND viewWindow;
createAndInitializeWebView(webView, window, viewWindow);
finishWebViewDestructionTest(webView, viewWindow);
}
// Tests that calling IWebView::close without calling DestroyWindow, then releasing a WebView doesn't crash. <http://webkit.org/b/32827>
TEST(WebViewDestruction, CloseWithoutDestroyViewWindow)
{
COMPtr<IWebView> webView;
HostWindow window;
HWND viewWindow;
createAndInitializeWebView(webView, window, viewWindow);
TEST_ASSERT(SUCCEEDED(webView->close()));
finishWebViewDestructionTest(webView, viewWindow);
}
TEST(WebViewDestruction, DestroyViewWindowWithoutClose)
{
COMPtr<IWebView> webView;
HostWindow window;
HWND viewWindow;
createAndInitializeWebView(webView, window, viewWindow);
DestroyWindow(viewWindow);
finishWebViewDestructionTest(webView, viewWindow);
}
TEST(WebViewDestruction, CloseThenDestroyViewWindow)
{
COMPtr<IWebView> webView;
HostWindow window;
HWND viewWindow;
createAndInitializeWebView(webView, window, viewWindow);
TEST_ASSERT(SUCCEEDED(webView->close()));
DestroyWindow(viewWindow);
finishWebViewDestructionTest(webView, viewWindow);
}
TEST(WebViewDestruction, DestroyViewWindowThenClose)
{
COMPtr<IWebView> webView;
HostWindow window;
HWND viewWindow;
createAndInitializeWebView(webView, window, viewWindow);
DestroyWindow(viewWindow);
TEST_ASSERT(SUCCEEDED(webView->close()));
finishWebViewDestructionTest(webView, viewWindow);
}
TEST(WebViewDestruction, DestroyHostWindow)
{
COMPtr<IWebView> webView;
HostWindow window;
HWND viewWindow;
createAndInitializeWebView(webView, window, viewWindow);
DestroyWindow(window.window());
finishWebViewDestructionTest(webView, viewWindow);
}
TEST(WebViewDestruction, DestroyHostWindowThenClose)
{
COMPtr<IWebView> webView;
HostWindow window;
HWND viewWindow;
createAndInitializeWebView(webView, window, viewWindow);
DestroyWindow(window.window());
TEST_ASSERT(SUCCEEDED(webView->close()));
finishWebViewDestructionTest(webView, viewWindow);
}
TEST(WebViewDestruction, CloseThenDestroyHostWindow)
{
COMPtr<IWebView> webView;
HostWindow window;
HWND viewWindow;
createAndInitializeWebView(webView, window, viewWindow);
TEST_ASSERT(SUCCEEDED(webView->close()));
DestroyWindow(window.window());
finishWebViewDestructionTest(webView, viewWindow);
}
// Tests that calling IWebView::mainFrame after calling IWebView::close doesn't crash. <http://webkit.org/b/32868>
TEST(WebViewDestruction, MainFrameAfterClose)
{
COMPtr<IWebView> webView;
HostWindow window;
HWND viewWindow;
createAndInitializeWebView(webView, window, viewWindow);
TEST_ASSERT(SUCCEEDED(webView->close()));
COMPtr<IWebFrame> mainFrame;
TEST_ASSERT(SUCCEEDED(webView->mainFrame(&mainFrame)));
finishWebViewDestructionTest(webView, viewWindow);
}
} // namespace WebKitAPITest