/* * Copyright (C) 2008 Nokia Corporation and/or its subsidiary(-ies) * Copyright (C) 2009 Torch Mobile Inc. http://www.torchmobile.com/ * * 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 "LayoutTestControllerQt.h" #include "DumpRenderTreeQt.h" #include "WorkQueue.h" #include "WorkQueueItemQt.h" #include #include #include extern void qt_dump_editing_callbacks(bool b); extern void qt_dump_frame_loader(bool b); extern void qt_dump_resource_load_callbacks(bool b); extern void qt_drt_setFrameSetFlatteningEnabled(QWebPage*, bool); extern void qt_drt_setJavaScriptProfilingEnabled(QWebFrame*, bool enabled); extern bool qt_drt_pauseAnimation(QWebFrame*, const QString& name, double time, const QString& elementId); extern bool qt_drt_pauseTransitionOfProperty(QWebFrame*, const QString& name, double time, const QString& elementId); extern bool qt_drt_pauseSVGAnimation(QWebFrame*, const QString& animationId, double time, const QString& elementId); extern int qt_drt_numberOfActiveAnimations(QWebFrame*); extern void qt_drt_setDomainRelaxationForbiddenForURLScheme(bool forbidden, const QString& scheme); extern void qt_drt_whiteListAccessFromOrigin(const QString& sourceOrigin, const QString& destinationProtocol, const QString& destinationHost, bool allowDestinationSubdomains); extern QString qt_drt_counterValueForElementById(QWebFrame* qFrame, const QString& id); extern int qt_drt_workerThreadCount(); extern int qt_drt_pageNumberForElementById(QWebFrame* qFrame, const QString& id, float width, float height); LayoutTestController::LayoutTestController(WebCore::DumpRenderTree* drt) : QObject() , m_drt(drt) { reset(); } void LayoutTestController::reset() { m_hasDumped = false; m_loadFinished = false; m_textDump = false; m_dumpBackForwardList = false; m_dumpChildrenAsText = false; m_canOpenWindows = false; m_waitForDone = false; m_dumpTitleChanges = false; m_dumpDatabaseCallbacks = false; m_dumpStatusCallbacks = false; m_timeoutTimer.stop(); m_topLoadingFrame = 0; m_waitForPolicy = false; m_handleErrorPages = false; m_webHistory = 0; m_globalFlag = false; qt_dump_editing_callbacks(false); qt_dump_frame_loader(false); qt_dump_resource_load_callbacks(false); emit hidePage(); } void LayoutTestController::processWork() { // qDebug() << ">>>processWork"; // if we didn't start a new load, then we finished all the commands, so we're ready to dump state if (WorkQueue::shared()->processWork() && !shouldWaitUntilDone()) { emit done(); m_hasDumped = true; } } // Called on loadFinished on WebPage void LayoutTestController::maybeDump(bool success) { // This can happen on any of the http/tests/security/window-events-*.html tests, where the test opens // a new window, calls the unload and load event handlers on the window's page, and then immediately // issues a notifyDone. Needs investigation. if (!m_topLoadingFrame) return; // It is possible that we get called by windows created from the main page that have finished // loading, so we don't ASSERT here. At the moment we do not gather results from such windows, // but may need to in future. if (sender() != m_topLoadingFrame->page()) return; m_loadFinished = true; // as the function is called on loadFinished, the test might // already have dumped and thus no longer be active, thus // bail out here. if (m_hasDumped) return; WorkQueue::shared()->setFrozen(true); // first complete load freezes the queue for the rest of this test if (WorkQueue::shared()->count()) QTimer::singleShot(0, this, SLOT(processWork())); else if (!shouldWaitUntilDone()) { if (success) emit done(); m_hasDumped = true; } } void LayoutTestController::waitUntilDone() { //qDebug() << ">>>>waitForDone"; m_waitForDone = true; m_timeoutTimer.start(15000, this); } QString LayoutTestController::counterValueForElementById(const QString& id) { return qt_drt_counterValueForElementById(m_drt->webPage()->mainFrame(), id); } int LayoutTestController::webHistoryItemCount() { if (!m_webHistory) return -1; // Subtract one here as our QWebHistory::count() includes the actual page, // which is not considered in the DRT tests. return m_webHistory->count() - 1; } void LayoutTestController::keepWebHistory() { m_webHistory = m_drt->webPage()->history(); } void LayoutTestController::notifyDone() { qDebug() << ">>>>notifyDone"; if (!m_timeoutTimer.isActive()) return; m_timeoutTimer.stop(); m_waitForDone = false; // If the page has not finished loading (i.e. loadFinished() has not been emitted) then // content created by the likes of document.write() JS methods will not be available yet. // When the page has finished loading, maybeDump above will dump the results now that we have // just set shouldWaitUntilDone to false. if (!m_loadFinished) return; emit done(); // FIXME: investigate why always resetting these result in timeouts m_hasDumped = true; m_waitForPolicy = false; } int LayoutTestController::windowCount() { return m_drt->windowCount(); } void LayoutTestController::display() { emit showPage(); } void LayoutTestController::clearBackForwardList() { m_drt->webPage()->history()->clear(); } QString LayoutTestController::pathToLocalResource(const QString& url) { // Function introduced in r28690. return QDir::toNativeSeparators(url); } void LayoutTestController::dumpEditingCallbacks() { qDebug() << ">>>dumpEditingCallbacks"; qt_dump_editing_callbacks(true); } void LayoutTestController::dumpFrameLoadCallbacks() { qt_dump_frame_loader(true); } void LayoutTestController::dumpResourceLoadCallbacks() { qt_dump_resource_load_callbacks(true); } void LayoutTestController::queueBackNavigation(int howFarBackward) { //qDebug() << ">>>queueBackNavigation" << howFarBackward; WorkQueue::shared()->queue(new BackItem(howFarBackward, m_drt->webPage())); } void LayoutTestController::queueForwardNavigation(int howFarForward) { //qDebug() << ">>>queueForwardNavigation" << howFarForward; WorkQueue::shared()->queue(new ForwardItem(howFarForward, m_drt->webPage())); } void LayoutTestController::queueLoad(const QString& url, const QString& target) { //qDebug() << ">>>queueLoad" << url << target; QUrl mainResourceUrl = m_drt->webPage()->mainFrame()->url(); QString absoluteUrl = mainResourceUrl.resolved(QUrl(url)).toEncoded(); WorkQueue::shared()->queue(new LoadItem(absoluteUrl, target, m_drt->webPage())); } void LayoutTestController::queueReload() { //qDebug() << ">>>queueReload"; WorkQueue::shared()->queue(new ReloadItem(m_drt->webPage())); } void LayoutTestController::queueLoadingScript(const QString& script) { //qDebug() << ">>>queueLoadingScript" << script; WorkQueue::shared()->queue(new LoadingScriptItem(script, m_drt->webPage())); } void LayoutTestController::queueNonLoadingScript(const QString& script) { //qDebug() << ">>>queueNonLoadingScript" << script; WorkQueue::shared()->queue(new NonLoadingScriptItem(script, m_drt->webPage())); } void LayoutTestController::provisionalLoad() { QWebFrame* frame = qobject_cast(sender()); if (!m_topLoadingFrame && !m_hasDumped) m_topLoadingFrame = frame; } void LayoutTestController::timerEvent(QTimerEvent *ev) { if (ev->timerId() == m_timeoutTimer.timerId()) { const char* message = "FAIL: Timed out waiting for notifyDone to be called\n"; fprintf(stderr, "%s", message); fprintf(stdout, "%s", message); notifyDone(); } else QObject::timerEvent(ev); } QString LayoutTestController::encodeHostName(const QString& host) { QString encoded = QString::fromLatin1(QUrl::toAce(host + QLatin1String(".no"))); encoded.truncate(encoded.length() - 3); // strip .no return encoded; } QString LayoutTestController::decodeHostName(const QString& host) { QString decoded = QUrl::fromAce(host.toLatin1() + QByteArray(".no")); decoded.truncate(decoded.length() - 3); return decoded; } void LayoutTestController::showWebInspector() { m_drt->webPage()->settings()->setAttribute(QWebSettings::DeveloperExtrasEnabled, true); m_drt->webPage()->webInspector()->show(); } void LayoutTestController::hideWebInspector() { m_drt->webPage()->webInspector()->hide(); } void LayoutTestController::setFrameSetFlatteningEnabled(bool enabled) { qt_drt_setFrameSetFlatteningEnabled(m_drt->webPage(), enabled); } void LayoutTestController::setAllowUniversalAccessFromFileURLs(bool enabled) { m_drt->webPage()->settings()->setAttribute(QWebSettings::LocalContentCanAccessRemoteUrls, enabled); } void LayoutTestController::setJavaScriptProfilingEnabled(bool enable) { m_topLoadingFrame->page()->settings()->setAttribute(QWebSettings::DeveloperExtrasEnabled, true); qt_drt_setJavaScriptProfilingEnabled(m_topLoadingFrame, enable); } void LayoutTestController::setFixedContentsSize(int width, int height) { m_topLoadingFrame->page()->setPreferredContentsSize(QSize(width, height)); } void LayoutTestController::setPrivateBrowsingEnabled(bool enable) { m_drt->webPage()->settings()->setAttribute(QWebSettings::PrivateBrowsingEnabled, enable); } void LayoutTestController::setPopupBlockingEnabled(bool enable) { m_drt->webPage()->settings()->setAttribute(QWebSettings::JavascriptCanOpenWindows, !enable); } void LayoutTestController::setPOSIXLocale(const QString& locale) { QLocale qlocale(locale); QLocale::setDefault(qlocale); } void LayoutTestController::setWindowIsKey(bool isKey) { m_drt->switchFocus(isKey); } void LayoutTestController::setMainFrameIsFirstResponder(bool isFirst) { //FIXME: only need this for the moment: https://bugs.webkit.org/show_bug.cgi?id=32990 } void LayoutTestController::setXSSAuditorEnabled(bool enable) { // Set XSSAuditorEnabled globally so that windows created by the test inherit it too. // resetSettings() will call this to reset the page and global setting to false again. // Needed by http/tests/security/xssAuditor/link-opens-new-window.html QWebSettings* globalSettings = QWebSettings::globalSettings(); globalSettings->setAttribute(QWebSettings::XSSAuditorEnabled, enable); m_drt->webPage()->settings()->setAttribute(QWebSettings::XSSAuditorEnabled, enable); } bool LayoutTestController::pauseAnimationAtTimeOnElementWithId(const QString& animationName, double time, const QString& elementId) { QWebFrame* frame = m_drt->webPage()->mainFrame(); Q_ASSERT(frame); return qt_drt_pauseAnimation(frame, animationName, time, elementId); } bool LayoutTestController::pauseTransitionAtTimeOnElementWithId(const QString& propertyName, double time, const QString& elementId) { QWebFrame* frame = m_drt->webPage()->mainFrame(); Q_ASSERT(frame); return qt_drt_pauseTransitionOfProperty(frame, propertyName, time, elementId); } bool LayoutTestController::sampleSVGAnimationForElementAtTime(const QString& animationId, double time, const QString& elementId) { QWebFrame* frame = m_drt->webPage()->mainFrame(); Q_ASSERT(frame); return qt_drt_pauseSVGAnimation(frame, animationId, time, elementId); } unsigned LayoutTestController::numberOfActiveAnimations() const { QWebFrame* frame = m_drt->webPage()->mainFrame(); Q_ASSERT(frame); return qt_drt_numberOfActiveAnimations(frame); } void LayoutTestController::disableImageLoading() { // FIXME: Implement for testing fix for https://bugs.webkit.org/show_bug.cgi?id=27896 // Also need to make sure image loading is re-enabled for each new test. } void LayoutTestController::dispatchPendingLoadRequests() { // FIXME: Implement for testing fix for 6727495 } void LayoutTestController::setDatabaseQuota(int size) { if (!m_topLoadingFrame) return; m_topLoadingFrame->securityOrigin().setDatabaseQuota(size); } void LayoutTestController::clearAllDatabases() { QWebDatabase::removeAllDatabases(); } void LayoutTestController::whiteListAccessFromOrigin(const QString& sourceOrigin, const QString& destinationProtocol, const QString& destinationHost, bool allowDestinationSubdomains) { qt_drt_whiteListAccessFromOrigin(sourceOrigin, destinationProtocol, destinationHost, allowDestinationSubdomains); } void LayoutTestController::waitForPolicyDelegate() { m_waitForPolicy = true; waitUntilDone(); } void LayoutTestController::overridePreference(const QString& name, const QVariant& value) { QWebSettings* settings = m_topLoadingFrame->page()->settings(); if (name == "WebKitJavaScriptEnabled") settings->setAttribute(QWebSettings::JavascriptEnabled, value.toBool()); else if (name == "WebKitTabToLinksPreferenceKey") settings->setAttribute(QWebSettings::LinksIncludedInFocusChain, value.toBool()); else if (name == "WebKitOfflineWebApplicationCacheEnabled") settings->setAttribute(QWebSettings::OfflineWebApplicationCacheEnabled, value.toBool()); else if (name == "WebKitDefaultFontSize") settings->setFontSize(QWebSettings::DefaultFontSize, value.toInt()); else if (name == "WebKitUsesPageCachePreferenceKey") QWebSettings::setMaximumPagesInCache(value.toInt()); else printf("ERROR: LayoutTestController::overridePreference() does not support the '%s' preference\n", name.toLatin1().data()); } void LayoutTestController::setUserStyleSheetLocation(const QString& url) { m_userStyleSheetLocation = QUrl(url); } void LayoutTestController::setUserStyleSheetEnabled(bool enabled) { if (enabled) m_drt->webPage()->settings()->setUserStyleSheetUrl(m_userStyleSheetLocation); else m_drt->webPage()->settings()->setUserStyleSheetUrl(QUrl()); } void LayoutTestController::setDomainRelaxationForbiddenForURLScheme(bool forbidden, const QString& scheme) { qt_drt_setDomainRelaxationForbiddenForURLScheme(forbidden, scheme); } int LayoutTestController::workerThreadCount() { return qt_drt_workerThreadCount(); } int LayoutTestController::pageNumberForElementById(const QString& id, float width, float height) { // If no size specified, webpage viewport size is used if (!width && !height) { width = m_drt->webPage()->viewportSize().width(); height = m_drt->webPage()->viewportSize().height(); } return qt_drt_pageNumberForElementById(m_drt->webPage()->mainFrame(), id, width, height); }