blob: 3e48ade6552196e9bea71523b7376266f84355e5 [file] [log] [blame]
/*
* Copyright (C) 2006, 2007, 2008, 2009, 2010, 2012 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 "TestRunner.h"
#include "DumpRenderTree.h"
#include "EditingDelegate.h"
#include "PolicyDelegate.h"
#include "WorkQueue.h"
#include "WorkQueueItem.h"
#include <CoreFoundation/CoreFoundation.h>
#include <JavaScriptCore/JSRetainPtr.h>
#include <JavaScriptCore/JSStringRefBSTR.h>
#include <JavaScriptCore/JavaScriptCore.h>
#include <WebCore/COMPtr.h>
#include <WebKit/WebKit.h>
#include <WebKit/WebKitCOMAPI.h>
#include <comutil.h>
#include <shlguid.h>
#include <shlwapi.h>
#include <shobjidl.h>
#include <string>
#include <wtf/Assertions.h>
#include <wtf/Platform.h>
#include <wtf/RetainPtr.h>
#include <wtf/Vector.h>
using std::string;
using std::wstring;
static bool resolveCygwinPath(const wstring& cygwinPath, wstring& windowsPath);
TestRunner::~TestRunner()
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
// reset webview-related states back to default values in preparation for next test
COMPtr<IWebViewPrivate> viewPrivate;
if (SUCCEEDED(webView->QueryInterface(&viewPrivate)))
viewPrivate->setTabKeyCyclesThroughElements(TRUE);
COMPtr<IWebViewEditing> viewEditing;
if (FAILED(webView->QueryInterface(&viewEditing)))
return;
COMPtr<IWebEditingDelegate> delegate;
if (FAILED(viewEditing->editingDelegate(&delegate)))
return;
COMPtr<EditingDelegate> editingDelegate(Query, viewEditing.get());
if (editingDelegate)
editingDelegate->setAcceptsEditing(TRUE);
}
void TestRunner::addDisallowedURL(JSStringRef url)
{
// FIXME: Implement!
}
void TestRunner::clearBackForwardList()
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebBackForwardList> backForwardList;
if (FAILED(webView->backForwardList(&backForwardList)))
return;
COMPtr<IWebHistoryItem> item;
if (FAILED(backForwardList->currentItem(&item)))
return;
// We clear the history by setting the back/forward list's capacity to 0
// then restoring it back and adding back the current item.
int capacity;
if (FAILED(backForwardList->capacity(&capacity)))
return;
backForwardList->setCapacity(0);
backForwardList->setCapacity(capacity);
backForwardList->addItem(item.get());
backForwardList->goToItem(item.get());
}
bool TestRunner::callShouldCloseOnWebView()
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return false;
COMPtr<IWebViewPrivate> viewPrivate;
if (FAILED(webView->QueryInterface(&viewPrivate)))
return false;
BOOL result;
viewPrivate->shouldClose(&result);
return result;
}
JSStringRef TestRunner::copyDecodedHostName(JSStringRef name)
{
// FIXME: Implement!
return 0;
}
JSStringRef TestRunner::copyEncodedHostName(JSStringRef name)
{
// FIXME: Implement!
return 0;
}
void TestRunner::disableImageLoading()
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
preferences->setLoadsImagesAutomatically(FALSE);
}
void TestRunner::dispatchPendingLoadRequests()
{
// FIXME: Implement for testing fix for 6727495
}
void TestRunner::display()
{
displayWebView();
}
void TestRunner::keepWebHistory()
{
COMPtr<IWebHistory> history;
if (FAILED(WebKitCreateInstance(CLSID_WebHistory, 0, __uuidof(history), reinterpret_cast<void**>(&history))))
return;
COMPtr<IWebHistory> sharedHistory;
if (FAILED(WebKitCreateInstance(CLSID_WebHistory, 0, __uuidof(sharedHistory), reinterpret_cast<void**>(&sharedHistory))))
return;
history->setOptionalSharedHistory(sharedHistory.get());
}
JSValueRef TestRunner::computedStyleIncludingVisitedInfo(JSContextRef context, JSValueRef value)
{
// FIXME: Implement this.
return JSValueMakeUndefined(context);
}
JSRetainPtr<JSStringRef> TestRunner::markerTextForListItem(JSContextRef context, JSValueRef nodeObject) const
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return 0;
COMPtr<IWebViewPrivate> webViewPrivate(Query, webView);
if (!webViewPrivate)
return 0;
COMPtr<IDOMElement> element;
if (FAILED(webViewPrivate->elementFromJS(context, nodeObject, &element)))
return 0;
COMPtr<IDOMElementPrivate> elementPrivate(Query, element);
if (!elementPrivate)
return 0;
BSTR textBSTR = 0;
if (FAILED(elementPrivate->markerTextForListItem(&textBSTR)))
return 0;
JSRetainPtr<JSStringRef> markerText(Adopt, JSStringCreateWithBSTR(textBSTR));
SysFreeString(textBSTR);
return markerText;
}
void TestRunner::waitForPolicyDelegate()
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
setWaitToDump(true);
policyDelegate->setControllerToNotifyDone(this);
webView->setPolicyDelegate(policyDelegate);
}
size_t TestRunner::webHistoryItemCount()
{
COMPtr<IWebHistory> history;
if (FAILED(WebKitCreateInstance(CLSID_WebHistory, 0, __uuidof(history), reinterpret_cast<void**>(&history))))
return 0;
COMPtr<IWebHistory> sharedHistory;
if (FAILED(history->optionalSharedHistory(&sharedHistory)) || !sharedHistory)
return 0;
COMPtr<IWebHistoryPrivate> sharedHistoryPrivate;
if (FAILED(sharedHistory->QueryInterface(&sharedHistoryPrivate)))
return 0;
int count;
if (FAILED(sharedHistoryPrivate->allItems(&count, 0)))
return 0;
return count;
}
unsigned TestRunner::workerThreadCount() const
{
COMPtr<IWebWorkersPrivate> workers;
if (FAILED(WebKitCreateInstance(CLSID_WebWorkersPrivate, 0, __uuidof(workers), reinterpret_cast<void**>(&workers))))
return 0;
unsigned count;
if (FAILED(workers->workerThreadCount(&count)))
return 0;
return count;
}
JSRetainPtr<JSStringRef> TestRunner::platformName() const
{
JSRetainPtr<JSStringRef> platformName(Adopt, JSStringCreateWithUTF8CString("win"));
return platformName;
}
void TestRunner::notifyDone()
{
// Same as on mac. This can be shared.
if (m_waitToDump && !topLoadingFrame && !WorkQueue::shared()->count())
dump();
m_waitToDump = false;
}
JSStringRef TestRunner::pathToLocalResource(JSContextRef context, JSStringRef url)
{
wstring input(JSStringGetCharactersPtr(url), JSStringGetLength(url));
wstring localPath;
if (!resolveCygwinPath(input, localPath)) {
printf("ERROR: Failed to resolve Cygwin path %S\n", input.c_str());
return 0;
}
return JSStringCreateWithCharacters(localPath.c_str(), localPath.length());
}
static wstring jsStringRefToWString(JSStringRef jsStr)
{
size_t length = JSStringGetLength(jsStr);
Vector<WCHAR> buffer(length + 1);
memcpy(buffer.data(), JSStringGetCharactersPtr(jsStr), length * sizeof(WCHAR));
buffer[length] = '\0';
return buffer.data();
}
void TestRunner::queueLoad(JSStringRef url, JSStringRef target)
{
COMPtr<IWebDataSource> dataSource;
if (FAILED(frame->dataSource(&dataSource)))
return;
COMPtr<IWebURLResponse> response;
if (FAILED(dataSource->response(&response)) || !response)
return;
BSTR responseURLBSTR;
if (FAILED(response->URL(&responseURLBSTR)))
return;
wstring responseURL(responseURLBSTR, SysStringLen(responseURLBSTR));
SysFreeString(responseURLBSTR);
// FIXME: We should do real relative URL resolution here.
int lastSlash = responseURL.rfind('/');
if (lastSlash != -1)
responseURL = responseURL.substr(0, lastSlash);
wstring wURL = jsStringRefToWString(url);
wstring wAbsoluteURL = responseURL + TEXT("/") + wURL;
JSRetainPtr<JSStringRef> jsAbsoluteURL(Adopt, JSStringCreateWithCharacters(wAbsoluteURL.data(), wAbsoluteURL.length()));
WorkQueue::shared()->queue(new LoadItem(jsAbsoluteURL.get(), target));
}
void TestRunner::setAcceptsEditing(bool acceptsEditing)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewEditing> viewEditing;
if (FAILED(webView->QueryInterface(&viewEditing)))
return;
COMPtr<IWebEditingDelegate> delegate;
if (FAILED(viewEditing->editingDelegate(&delegate)))
return;
EditingDelegate* editingDelegate = (EditingDelegate*)(IWebEditingDelegate*)delegate.get();
editingDelegate->setAcceptsEditing(acceptsEditing);
}
void TestRunner::setAlwaysAcceptCookies(bool alwaysAcceptCookies)
{
if (alwaysAcceptCookies == m_alwaysAcceptCookies)
return;
if (!::setAlwaysAcceptCookies(alwaysAcceptCookies))
return;
m_alwaysAcceptCookies = alwaysAcceptCookies;
}
void TestRunner::setAuthorAndUserStylesEnabled(bool flag)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
COMPtr<IWebPreferencesPrivate> prefsPrivate(Query, preferences);
if (!prefsPrivate)
return;
prefsPrivate->setAuthorAndUserStylesEnabled(flag);
}
void TestRunner::setAutofilled(JSContextRef context, JSValueRef nodeObject, bool autofilled)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewPrivate> webViewPrivate(Query, webView);
if (!webViewPrivate)
return;
COMPtr<IDOMElement> element;
if (FAILED(webViewPrivate->elementFromJS(context, nodeObject, &element)))
return;
COMPtr<IFormsAutoFillTransition> autofillElement(Query, element);
if (!autofillElement)
return;
autofillElement->setAutofilled(autofilled);
}
void TestRunner::setCustomPolicyDelegate(bool setDelegate, bool permissive)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
if (setDelegate) {
policyDelegate->setPermissive(permissive);
webView->setPolicyDelegate(policyDelegate);
} else
webView->setPolicyDelegate(0);
}
void TestRunner::setMockDeviceOrientation(bool canProvideAlpha, double alpha, bool canProvideBeta, double beta, bool canProvideGamma, double gamma)
{
// FIXME: Implement for DeviceOrientation layout tests.
// See https://bugs.webkit.org/show_bug.cgi?id=30335.
}
void TestRunner::setMockGeolocationPosition(double latitude, double longitude, double accuracy, bool providesAltitude, double altitude, bool providesAltitudeAccuracy, double altitudeAccuracy, bool providesHeading, double heading, bool providesSpeed, double speed)
{
// FIXME: Implement for Geolocation layout tests.
// See https://bugs.webkit.org/show_bug.cgi?id=28264.
}
void TestRunner::setMockGeolocationPositionUnavailableError(JSStringRef message)
{
// FIXME: Implement for Geolocation layout tests.
// See https://bugs.webkit.org/show_bug.cgi?id=28264.
}
void TestRunner::setGeolocationPermission(bool allow)
{
// FIXME: Implement for Geolocation layout tests.
setGeolocationPermissionCommon(allow);
}
int TestRunner::numberOfPendingGeolocationPermissionRequests()
{
// FIXME: Implement for Geolocation layout tests.
return -1;
}
void TestRunner::addMockSpeechInputResult(JSStringRef result, double confidence, JSStringRef language)
{
// FIXME: Implement for speech input layout tests.
// See https://bugs.webkit.org/show_bug.cgi?id=39485.
}
void TestRunner::setMockSpeechInputDumpRect(bool flag)
{
// FIXME: Implement for speech input layout tests.
// See https://bugs.webkit.org/show_bug.cgi?id=39485.
}
void TestRunner::startSpeechInput(JSContextRef inputElement)
{
// FIXME: Implement for speech input layout tests.
// See https://bugs.webkit.org/show_bug.cgi?id=39485.
}
void TestRunner::setIconDatabaseEnabled(bool iconDatabaseEnabled)
{
// See also <rdar://problem/6480108>
COMPtr<IWebIconDatabase> iconDatabase;
COMPtr<IWebIconDatabase> tmpIconDatabase;
if (FAILED(WebKitCreateInstance(CLSID_WebIconDatabase, 0, IID_IWebIconDatabase, (void**)&tmpIconDatabase)))
return;
if (FAILED(tmpIconDatabase->sharedIconDatabase(&iconDatabase)))
return;
iconDatabase->setEnabled(iconDatabaseEnabled);
}
void TestRunner::setMainFrameIsFirstResponder(bool flag)
{
// FIXME: Implement!
}
void TestRunner::setPrivateBrowsingEnabled(bool privateBrowsingEnabled)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
preferences->setPrivateBrowsingEnabled(privateBrowsingEnabled);
}
void TestRunner::setXSSAuditorEnabled(bool enabled)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
COMPtr<IWebPreferencesPrivate> prefsPrivate(Query, preferences);
if (!prefsPrivate)
return;
prefsPrivate->setXSSAuditorEnabled(enabled);
}
void TestRunner::setFrameFlatteningEnabled(bool enabled)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
COMPtr<IWebPreferencesPrivate> prefsPrivate(Query, preferences);
if (!prefsPrivate)
return;
prefsPrivate->setFrameFlatteningEnabled(enabled);
}
void TestRunner::setSpatialNavigationEnabled(bool enabled)
{
// FIXME: Implement for SpatialNavigation layout tests.
}
void TestRunner::setAllowUniversalAccessFromFileURLs(bool enabled)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
COMPtr<IWebPreferencesPrivate> prefsPrivate(Query, preferences);
if (!prefsPrivate)
return;
prefsPrivate->setAllowUniversalAccessFromFileURLs(enabled);
}
void TestRunner::setAllowFileAccessFromFileURLs(bool enabled)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
COMPtr<IWebPreferencesPrivate> prefsPrivate(Query, preferences);
if (!prefsPrivate)
return;
prefsPrivate->setAllowFileAccessFromFileURLs(enabled);
}
void TestRunner::setPopupBlockingEnabled(bool enabled)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
preferences->setJavaScriptCanOpenWindowsAutomatically(!enabled);
}
void TestRunner::setPluginsEnabled(bool flag)
{
// FIXME: Implement
}
void TestRunner::setJavaScriptCanAccessClipboard(bool enabled)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
COMPtr<IWebPreferencesPrivate> prefsPrivate(Query, preferences);
if (!prefsPrivate)
return;
prefsPrivate->setJavaScriptCanAccessClipboard(enabled);
}
void TestRunner::setTabKeyCyclesThroughElements(bool shouldCycle)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewPrivate> viewPrivate;
if (FAILED(webView->QueryInterface(&viewPrivate)))
return;
viewPrivate->setTabKeyCyclesThroughElements(shouldCycle ? TRUE : FALSE);
}
void TestRunner::setUseDashboardCompatibilityMode(bool flag)
{
// FIXME: Implement!
}
void TestRunner::setUserStyleSheetEnabled(bool flag)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
preferences->setUserStyleSheetEnabled(flag);
}
bool appendComponentToPath(wstring& path, const wstring& component)
{
WCHAR buffer[MAX_PATH];
if (path.size() + 1 > MAX_PATH)
return false;
memcpy(buffer, path.data(), path.size() * sizeof(WCHAR));
buffer[path.size()] = '\0';
if (!PathAppendW(buffer, component.c_str()))
return false;
path = wstring(buffer);
return true;
}
static bool followShortcuts(wstring& path)
{
if (PathFileExists(path.c_str()))
return true;
// Do we have a shortcut?
wstring linkPath = path;
linkPath.append(TEXT(".lnk"));
if (!PathFileExists(linkPath.c_str()))
return true;
// We have a shortcut, find its target.
COMPtr<IShellLink> shortcut(Create, CLSID_ShellLink);
if (!shortcut)
return false;
COMPtr<IPersistFile> persistFile(Query, shortcut);
if (!shortcut)
return false;
if (FAILED(persistFile->Load(linkPath.c_str(), STGM_READ)))
return false;
if (FAILED(shortcut->Resolve(0, 0)))
return false;
WCHAR targetPath[MAX_PATH];
DWORD targetPathLen = _countof(targetPath);
if (FAILED(shortcut->GetPath(targetPath, targetPathLen, 0, 0)))
return false;
if (!PathFileExists(targetPath))
return false;
// Use the target path as the result path instead.
path = wstring(targetPath);
return true;
}
static bool resolveCygwinPath(const wstring& cygwinPath, wstring& windowsPath)
{
wstring fileProtocol = L"file://";
bool isFileProtocol = cygwinPath.find(fileProtocol) != string::npos;
if (cygwinPath[isFileProtocol ? 7 : 0] != '/') // ensure path is absolute
return false;
// Get the Root path.
WCHAR rootPath[MAX_PATH];
DWORD rootPathSize = _countof(rootPath);
DWORD keyType;
DWORD result = ::SHGetValueW(HKEY_LOCAL_MACHINE, TEXT("SOFTWARE\\Cygnus Solutions\\Cygwin\\mounts v2\\/"), TEXT("native"), &keyType, &rootPath, &rootPathSize);
if (result != ERROR_SUCCESS || keyType != REG_SZ) {
// Cygwin 1.7 doesn't store Cygwin's root as a mount point anymore, because mount points are now stored in /etc/fstab.
// However, /etc/fstab doesn't contain any information about where / is located as a Windows path, so we need to use Cygwin's
// new registry key that has the root.
result = ::SHGetValueW(HKEY_LOCAL_MACHINE, TEXT("SOFTWARE\\Cygwin\\setup"), TEXT("rootdir"), &keyType, &rootPath, &rootPathSize);
if (result != ERROR_SUCCESS || keyType != REG_SZ)
return false;
}
windowsPath = wstring(rootPath, rootPathSize);
int oldPos = isFileProtocol ? 8 : 1;
while (1) {
int newPos = cygwinPath.find('/', oldPos);
if (newPos == -1) {
wstring pathComponent = cygwinPath.substr(oldPos);
if (!appendComponentToPath(windowsPath, pathComponent))
return false;
if (!followShortcuts(windowsPath))
return false;
break;
}
wstring pathComponent = cygwinPath.substr(oldPos, newPos - oldPos);
if (!appendComponentToPath(windowsPath, pathComponent))
return false;
if (!followShortcuts(windowsPath))
return false;
oldPos = newPos + 1;
}
if (isFileProtocol)
windowsPath = fileProtocol + windowsPath;
return true;
}
void TestRunner::setUserStyleSheetLocation(JSStringRef jsURL)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
RetainPtr<CFStringRef> urlString(AdoptCF, JSStringCopyCFString(0, jsURL));
RetainPtr<CFURLRef> url(AdoptCF, CFURLCreateWithString(0, urlString.get(), 0));
if (!url)
return;
// Now copy the file system path, POSIX style.
RetainPtr<CFStringRef> pathCF(AdoptCF, CFURLCopyFileSystemPath(url.get(), kCFURLPOSIXPathStyle));
if (!pathCF)
return;
wstring path = cfStringRefToWString(pathCF.get());
wstring resultPath;
if (!resolveCygwinPath(path, resultPath))
return;
// The path has been resolved, now convert it back to a CFURL.
int result = WideCharToMultiByte(CP_UTF8, 0, resultPath.c_str(), resultPath.size() + 1, 0, 0, 0, 0);
Vector<char> utf8Vector(result);
result = WideCharToMultiByte(CP_UTF8, 0, resultPath.c_str(), resultPath.size() + 1, utf8Vector.data(), result, 0, 0);
if (!result)
return;
url = CFURLCreateFromFileSystemRepresentation(0, (const UInt8*)utf8Vector.data(), utf8Vector.size() - 1, false);
if (!url)
return;
resultPath = cfStringRefToWString(CFURLGetString(url.get()));
BSTR resultPathBSTR = SysAllocStringLen(resultPath.data(), resultPath.size());
preferences->setUserStyleSheetLocation(resultPathBSTR);
SysFreeString(resultPathBSTR);
}
void TestRunner::setValueForUser(JSContextRef context, JSValueRef element, JSStringRef value)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewPrivate> webViewPrivate(Query, webView);
if (!webViewPrivate)
return;
COMPtr<IDOMElement> domElement;
if (FAILED(webViewPrivate->elementFromJS(context, element, &domElement)))
return;
COMPtr<IDOMHTMLInputElement> domInputElement;
if (FAILED(domElement->QueryInterface(IID_IDOMHTMLInputElement, reinterpret_cast<void**>(&domInputElement))))
return;
_bstr_t valueBSTR(JSStringCopyBSTR(value), false);
domInputElement->setValueForUser(valueBSTR);
}
void TestRunner::setViewModeMediaFeature(JSStringRef mode)
{
// FIXME: implement
}
void TestRunner::setPersistentUserStyleSheetLocation(JSStringRef jsURL)
{
RetainPtr<CFStringRef> urlString(AdoptCF, JSStringCopyCFString(0, jsURL));
::setPersistentUserStyleSheetLocation(urlString.get());
}
void TestRunner::clearPersistentUserStyleSheet()
{
::setPersistentUserStyleSheetLocation(0);
}
void TestRunner::setWindowIsKey(bool flag)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewPrivate> viewPrivate;
if (FAILED(webView->QueryInterface(&viewPrivate)))
return;
HWND webViewWindow;
if (FAILED(viewPrivate->viewWindow((OLE_HANDLE*)&webViewWindow)))
return;
::SendMessage(webViewWindow, flag ? WM_SETFOCUS : WM_KILLFOCUS, (WPARAM)::GetDesktopWindow(), 0);
}
void TestRunner::setSmartInsertDeleteEnabled(bool flag)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewEditing> viewEditing;
if (FAILED(webView->QueryInterface(&viewEditing)))
return;
viewEditing->setSmartInsertDeleteEnabled(flag ? TRUE : FALSE);
}
void TestRunner::setSelectTrailingWhitespaceEnabled(bool flag)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewEditing> viewEditing;
if (FAILED(webView->QueryInterface(&viewEditing)))
return;
viewEditing->setSelectTrailingWhitespaceEnabled(flag ? TRUE : FALSE);
}
static const CFTimeInterval waitToDumpWatchdogInterval = 30.0;
static void CALLBACK waitUntilDoneWatchdogFired(HWND, UINT, UINT_PTR, DWORD)
{
gTestRunner->waitToDumpWatchdogTimerFired();
}
void TestRunner::setWaitToDump(bool waitUntilDone)
{
m_waitToDump = waitUntilDone;
if (m_waitToDump && !waitToDumpWatchdog)
waitToDumpWatchdog = SetTimer(0, 0, waitToDumpWatchdogInterval * 1000, waitUntilDoneWatchdogFired);
}
int TestRunner::windowCount()
{
return openWindows().size();
}
bool TestRunner::elementDoesAutoCompleteForElementWithId(JSStringRef id)
{
COMPtr<IDOMDocument> document;
if (FAILED(frame->DOMDocument(&document)))
return false;
wstring idWstring = jsStringRefToWString(id);
BSTR idBSTR = SysAllocStringLen((OLECHAR*)idWstring.c_str(), idWstring.length());
COMPtr<IDOMElement> element;
HRESULT result = document->getElementById(idBSTR, &element);
SysFreeString(idBSTR);
if (FAILED(result))
return false;
COMPtr<IWebFramePrivate> framePrivate(Query, frame);
if (!framePrivate)
return false;
BOOL autoCompletes;
if (FAILED(framePrivate->elementDoesAutoComplete(element.get(), &autoCompletes)))
return false;
return autoCompletes;
}
void TestRunner::execCommand(JSStringRef name, JSStringRef value)
{
wstring wName = jsStringRefToWString(name);
wstring wValue = jsStringRefToWString(value);
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewPrivate> viewPrivate;
if (FAILED(webView->QueryInterface(&viewPrivate)))
return;
BSTR nameBSTR = SysAllocStringLen((OLECHAR*)wName.c_str(), wName.length());
BSTR valueBSTR = SysAllocStringLen((OLECHAR*)wValue.c_str(), wValue.length());
viewPrivate->executeCoreCommandByName(nameBSTR, valueBSTR);
SysFreeString(nameBSTR);
SysFreeString(valueBSTR);
}
bool TestRunner::findString(JSContextRef /* context */, JSStringRef /* target */, JSObjectRef /* optionsArray */)
{
// FIXME: Implement
return false;
}
void TestRunner::setCacheModel(int)
{
// FIXME: Implement
}
bool TestRunner::isCommandEnabled(JSStringRef /*name*/)
{
printf("ERROR: TestRunner::isCommandEnabled() not implemented\n");
return false;
}
void TestRunner::clearAllApplicationCaches()
{
// FIXME: Implement to support application cache quotas.
}
void TestRunner::clearApplicationCacheForOrigin(JSStringRef origin)
{
// FIXME: Implement to support deleting all application cache for an origin.
}
void TestRunner::setApplicationCacheOriginQuota(unsigned long long quota)
{
// FIXME: Implement to support application cache quotas.
}
JSValueRef TestRunner::originsWithApplicationCache(JSContextRef context)
{
// FIXME: Implement to get origins that have application caches.
return JSValueMakeUndefined(context);
}
long long TestRunner::applicationCacheDiskUsageForOrigin(JSStringRef name)
{
// FIXME: Implement to get disk usage by all application caches for an origin.
return 0;
}
void TestRunner::clearAllDatabases()
{
COMPtr<IWebDatabaseManager> databaseManager;
COMPtr<IWebDatabaseManager> tmpDatabaseManager;
if (FAILED(WebKitCreateInstance(CLSID_WebDatabaseManager, 0, IID_IWebDatabaseManager, (void**)&tmpDatabaseManager)))
return;
if (FAILED(tmpDatabaseManager->sharedWebDatabaseManager(&databaseManager)))
return;
databaseManager->deleteAllDatabases();
}
void TestRunner::overridePreference(JSStringRef key, JSStringRef value)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
COMPtr<IWebPreferencesPrivate> prefsPrivate(Query, preferences);
if (!prefsPrivate)
return;
BSTR keyBSTR = JSStringCopyBSTR(key);
BSTR valueBSTR = JSStringCopyBSTR(value);
prefsPrivate->setPreferenceForTest(keyBSTR, valueBSTR);
SysFreeString(keyBSTR);
SysFreeString(valueBSTR);
}
void TestRunner::setDatabaseQuota(unsigned long long quota)
{
COMPtr<IWebDatabaseManager> databaseManager;
COMPtr<IWebDatabaseManager> tmpDatabaseManager;
if (FAILED(WebKitCreateInstance(CLSID_WebDatabaseManager, 0, IID_IWebDatabaseManager, (void**)&tmpDatabaseManager)))
return;
if (FAILED(tmpDatabaseManager->sharedWebDatabaseManager(&databaseManager)))
return;
databaseManager->setQuota(TEXT("file:///"), quota);
}
void TestRunner::goBack()
{
// FIXME: implement to enable loader/navigation-while-deferring-loads.html
}
void TestRunner::setDefersLoading(bool)
{
// FIXME: implement to enable loader/navigation-while-deferring-loads.html
}
void TestRunner::setDomainRelaxationForbiddenForURLScheme(bool forbidden, JSStringRef scheme)
{
COMPtr<IWebViewPrivate> webView;
if (FAILED(WebKitCreateInstance(__uuidof(WebView), 0, __uuidof(webView), reinterpret_cast<void**>(&webView))))
return;
BSTR schemeBSTR = JSStringCopyBSTR(scheme);
webView->setDomainRelaxationForbiddenForURLScheme(forbidden, schemeBSTR);
SysFreeString(schemeBSTR);
}
void TestRunner::setAppCacheMaximumSize(unsigned long long size)
{
printf("ERROR: TestRunner::setAppCacheMaximumSize() not implemented\n");
}
bool TestRunner::pauseAnimationAtTimeOnElementWithId(JSStringRef animationName, double time, JSStringRef elementId)
{
COMPtr<IDOMDocument> document;
if (FAILED(frame->DOMDocument(&document)))
return false;
BSTR idBSTR = JSStringCopyBSTR(elementId);
COMPtr<IDOMElement> element;
HRESULT hr = document->getElementById(idBSTR, &element);
SysFreeString(idBSTR);
if (FAILED(hr))
return false;
COMPtr<IWebFramePrivate> framePrivate(Query, frame);
if (!framePrivate)
return false;
BSTR nameBSTR = JSStringCopyBSTR(animationName);
BOOL wasRunning = FALSE;
hr = framePrivate->pauseAnimation(nameBSTR, element.get(), time, &wasRunning);
SysFreeString(nameBSTR);
return SUCCEEDED(hr) && wasRunning;
}
bool TestRunner::pauseTransitionAtTimeOnElementWithId(JSStringRef propertyName, double time, JSStringRef elementId)
{
COMPtr<IDOMDocument> document;
if (FAILED(frame->DOMDocument(&document)))
return false;
BSTR idBSTR = JSStringCopyBSTR(elementId);
COMPtr<IDOMElement> element;
HRESULT hr = document->getElementById(idBSTR, &element);
SysFreeString(idBSTR);
if (FAILED(hr))
return false;
COMPtr<IWebFramePrivate> framePrivate(Query, frame);
if (!framePrivate)
return false;
BSTR nameBSTR = JSStringCopyBSTR(propertyName);
BOOL wasRunning = FALSE;
hr = framePrivate->pauseTransition(nameBSTR, element.get(), time, &wasRunning);
SysFreeString(nameBSTR);
return SUCCEEDED(hr) && wasRunning;
}
unsigned TestRunner::numberOfActiveAnimations() const
{
COMPtr<IWebFramePrivate> framePrivate(Query, frame);
if (!framePrivate)
return 0;
UINT number = 0;
if (FAILED(framePrivate->numberOfActiveAnimations(&number)))
return 0;
return number;
}
static _bstr_t bstrT(JSStringRef jsString)
{
// The false parameter tells the _bstr_t constructor to adopt the BSTR we pass it.
return _bstr_t(JSStringCopyBSTR(jsString), false);
}
void TestRunner::addOriginAccessWhitelistEntry(JSStringRef sourceOrigin, JSStringRef destinationProtocol, JSStringRef destinationHost, bool allowDestinationSubdomains)
{
COMPtr<IWebViewPrivate> webView;
if (FAILED(WebKitCreateInstance(__uuidof(WebView), 0, __uuidof(webView), reinterpret_cast<void**>(&webView))))
return;
webView->addOriginAccessWhitelistEntry(bstrT(sourceOrigin).GetBSTR(), bstrT(destinationProtocol).GetBSTR(), bstrT(destinationHost).GetBSTR(), allowDestinationSubdomains);
}
void TestRunner::removeOriginAccessWhitelistEntry(JSStringRef sourceOrigin, JSStringRef destinationProtocol, JSStringRef destinationHost, bool allowDestinationSubdomains)
{
COMPtr<IWebViewPrivate> webView;
if (FAILED(WebKitCreateInstance(__uuidof(WebView), 0, __uuidof(webView), reinterpret_cast<void**>(&webView))))
return;
webView->removeOriginAccessWhitelistEntry(bstrT(sourceOrigin).GetBSTR(), bstrT(destinationProtocol).GetBSTR(), bstrT(destinationHost).GetBSTR(), allowDestinationSubdomains);
}
void TestRunner::setScrollbarPolicy(JSStringRef orientation, JSStringRef policy)
{
// FIXME: implement
}
void TestRunner::addUserScript(JSStringRef source, bool runAtStart, bool allFrames)
{
COMPtr<IWebViewPrivate> webView;
if (FAILED(WebKitCreateInstance(__uuidof(WebView), 0, __uuidof(webView), reinterpret_cast<void**>(&webView))))
return;
COMPtr<IWebScriptWorld> world;
if (FAILED(WebKitCreateInstance(__uuidof(WebScriptWorld), 0, __uuidof(world), reinterpret_cast<void**>(&world))))
return;
webView->addUserScriptToGroup(_bstr_t(L"org.webkit.DumpRenderTree").GetBSTR(), world.get(), bstrT(source).GetBSTR(), 0, 0, 0, 0, 0, runAtStart ? WebInjectAtDocumentStart : WebInjectAtDocumentEnd);
}
void TestRunner::addUserStyleSheet(JSStringRef source, bool allFrames)
{
COMPtr<IWebViewPrivate> webView;
if (FAILED(WebKitCreateInstance(__uuidof(WebView), 0, __uuidof(webView), reinterpret_cast<void**>(&webView))))
return;
COMPtr<IWebScriptWorld> world;
if (FAILED(WebKitCreateInstance(__uuidof(WebScriptWorld), 0, __uuidof(world), reinterpret_cast<void**>(&world))))
return;
webView->addUserStyleSheetToGroup(_bstr_t(L"org.webkit.DumpRenderTree").GetBSTR(), world.get(), bstrT(source).GetBSTR(), 0, 0, 0, 0, 0);
}
void TestRunner::setDeveloperExtrasEnabled(bool enabled)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebPreferences> preferences;
if (FAILED(webView->preferences(&preferences)))
return;
COMPtr<IWebPreferencesPrivate> prefsPrivate(Query, preferences);
if (!prefsPrivate)
return;
prefsPrivate->setDeveloperExtrasEnabled(enabled);
}
void TestRunner::setAsynchronousSpellCheckingEnabled(bool)
{
// FIXME: Implement this.
}
void TestRunner::showWebInspector()
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewPrivate> viewPrivate(Query, webView);
if (!viewPrivate)
return;
COMPtr<IWebInspector> inspector;
if (SUCCEEDED(viewPrivate->inspector(&inspector)))
inspector->show();
}
void TestRunner::closeWebInspector()
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewPrivate> viewPrivate(Query, webView);
if (!viewPrivate)
return;
COMPtr<IWebInspector> inspector;
if (FAILED(viewPrivate->inspector(&inspector)))
return;
inspector->close();
}
void TestRunner::evaluateInWebInspector(long callId, JSStringRef script)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewPrivate> viewPrivate(Query, webView);
if (!viewPrivate)
return;
COMPtr<IWebInspector> inspector;
if (FAILED(viewPrivate->inspector(&inspector)))
return;
COMPtr<IWebInspectorPrivate> inspectorPrivate(Query, inspector);
if (!inspectorPrivate)
return;
inspectorPrivate->evaluateInFrontend(callId, bstrT(script).GetBSTR());
}
typedef HashMap<unsigned, COMPtr<IWebScriptWorld> > WorldMap;
static WorldMap& worldMap()
{
static WorldMap& map = *new WorldMap;
return map;
}
unsigned worldIDForWorld(IWebScriptWorld* world)
{
WorldMap::const_iterator end = worldMap().end();
for (WorldMap::const_iterator it = worldMap().begin(); it != end; ++it) {
if (it->value == world)
return it->key;
}
return 0;
}
void TestRunner::evaluateScriptInIsolatedWorldAndReturnValue(unsigned worldID, JSObjectRef globalObject, JSStringRef script)
{
// FIXME: Implement this.
}
void TestRunner::evaluateScriptInIsolatedWorld(unsigned worldID, JSObjectRef globalObject, JSStringRef script)
{
COMPtr<IWebFramePrivate> framePrivate(Query, frame);
if (!framePrivate)
return;
// A worldID of 0 always corresponds to a new world. Any other worldID corresponds to a world
// that is created once and cached forever.
COMPtr<IWebScriptWorld> world;
if (!worldID) {
if (FAILED(WebKitCreateInstance(__uuidof(WebScriptWorld), 0, __uuidof(world), reinterpret_cast<void**>(&world))))
return;
} else {
COMPtr<IWebScriptWorld>& worldSlot = worldMap().add(worldID, 0).iterator->value;
if (!worldSlot && FAILED(WebKitCreateInstance(__uuidof(WebScriptWorld), 0, __uuidof(worldSlot), reinterpret_cast<void**>(&worldSlot))))
return;
world = worldSlot;
}
BSTR result;
if (FAILED(framePrivate->stringByEvaluatingJavaScriptInScriptWorld(world.get(), globalObject, bstrT(script).GetBSTR(), &result)))
return;
SysFreeString(result);
}
void TestRunner::removeAllVisitedLinks()
{
COMPtr<IWebHistory> history;
if (FAILED(WebKitCreateInstance(CLSID_WebHistory, 0, __uuidof(history), reinterpret_cast<void**>(&history))))
return;
COMPtr<IWebHistory> sharedHistory;
if (FAILED(history->optionalSharedHistory(&sharedHistory)) || !sharedHistory)
return;
COMPtr<IWebHistoryPrivate> sharedHistoryPrivate;
if (FAILED(sharedHistory->QueryInterface(&sharedHistoryPrivate)))
return;
sharedHistoryPrivate->removeAllVisitedLinks();
}
void TestRunner::apiTestNewWindowDataLoadBaseURL(JSStringRef utf8Data, JSStringRef baseURL)
{
}
void TestRunner::apiTestGoToCurrentBackForwardItem()
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebBackForwardList> backForwardList;
if (FAILED(webView->backForwardList(&backForwardList)))
return;
COMPtr<IWebHistoryItem> item;
if (FAILED(backForwardList->currentItem(&item)))
return;
BOOL success;
webView->goToBackForwardItem(item.get(), &success);
}
void TestRunner::setWebViewEditable(bool)
{
}
void TestRunner::authenticateSession(JSStringRef, JSStringRef, JSStringRef)
{
}
void TestRunner::abortModal()
{
}
void TestRunner::setSerializeHTTPLoads(bool)
{
// FIXME: Implement.
}
void TestRunner::syncLocalStorage()
{
// FIXME: Implement.
}
void TestRunner::observeStorageTrackerNotifications(unsigned number)
{
// FIXME: Implement.
}
void TestRunner::deleteAllLocalStorage()
{
// FIXME: Implement.
}
JSValueRef TestRunner::originsWithLocalStorage(JSContextRef context)
{
// FIXME: Implement.
return JSValueMakeUndefined(context);
}
long long TestRunner::localStorageDiskUsageForOrigin(JSStringRef originIdentifier)
{
// FIXME: Implement to support getting local storage disk usage for an origin.
return 0;
}
void TestRunner::deleteLocalStorageForOrigin(JSStringRef URL)
{
// FIXME: Implement.
}
void TestRunner::setMinimumTimerInterval(double minimumTimerInterval)
{
COMPtr<IWebView> webView;
if (FAILED(frame->webView(&webView)))
return;
COMPtr<IWebViewPrivate> viewPrivate(Query, webView);
if (!viewPrivate)
return;
viewPrivate->setMinimumTimerInterval(minimumTimerInterval);
}
void TestRunner::setTextDirection(JSStringRef direction)
{
COMPtr<IWebFramePrivate> framePrivate(Query, frame);
if (!framePrivate)
return;
framePrivate->setTextDirection(bstrT(direction).GetBSTR());
}
void TestRunner::addChromeInputField()
{
}
void TestRunner::removeChromeInputField()
{
}
void TestRunner::focusWebView()
{
}
void TestRunner::setBackingScaleFactor(double)
{
}
void TestRunner::grantWebNotificationPermission(JSStringRef origin)
{
}
void TestRunner::denyWebNotificationPermission(JSStringRef jsOrigin)
{
}
void TestRunner::removeAllWebNotificationPermissions()
{
}
void TestRunner::simulateWebNotificationClick(JSValueRef jsNotification)
{
}
void TestRunner::simulateLegacyWebNotificationClick(JSStringRef title)
{
// FIXME: Implement.
}
void TestRunner::resetPageVisibility()
{
// FIXME: Implement this.
}
void TestRunner::setPageVisibility(const char*)
{
// FIXME: Implement this.
}
void TestRunner::setAutomaticLinkDetectionEnabled(bool)
{
// FIXME: Implement this.
}
void TestRunner::sendWebIntentResponse(JSStringRef)
{
// FIXME: Implement this.
}
void TestRunner::deliverWebIntent(JSStringRef, JSStringRef, JSStringRef)
{
// FIXME: Implement this.
}
void TestRunner::setStorageDatabaseIdleInterval(double)
{
// FIXME: Implement this.
}