| /* |
| * Copyright (C) 2005, 2006, 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 "FrameLoadDelegate.h" |
| |
| #include "AccessibilityController.h" |
| #include "DumpRenderTree.h" |
| #include "EventSender.h" |
| #include "GCController.h" |
| #include "LayoutTestController.h" |
| #include "WorkQueueItem.h" |
| #include "WorkQueue.h" |
| #include <WebCore/COMPtr.h> |
| #include <JavaScriptCore/Assertions.h> |
| #include <JavaScriptCore/JavaScriptCore.h> |
| #include <WebKit/WebKit.h> |
| #include <wtf/Vector.h> |
| #include <stdio.h> |
| #include <string> |
| |
| using std::string; |
| |
| static FrameLoadDelegate* g_delegateWaitingOnTimer; |
| |
| string BSTRtoString(BSTR bstr) |
| { |
| int result = WideCharToMultiByte(CP_UTF8, 0, bstr, SysStringLen(bstr) + 1, 0, 0, 0, 0); |
| Vector<char> utf8Vector(result); |
| result = WideCharToMultiByte(CP_UTF8, 0, bstr, SysStringLen(bstr) + 1, utf8Vector.data(), result, 0, 0); |
| if (!result) |
| return string(); |
| |
| return string(utf8Vector.data(), utf8Vector.size() - 1); |
| } |
| |
| string descriptionSuitableForTestResult(IWebFrame* webFrame) |
| { |
| COMPtr<IWebView> webView; |
| if (FAILED(webFrame->webView(&webView))) |
| return string(); |
| |
| COMPtr<IWebFrame> mainFrame; |
| if (FAILED(webView->mainFrame(&mainFrame))) |
| return string(); |
| |
| BSTR frameNameBSTR; |
| if (FAILED(webFrame->name(&frameNameBSTR)) || BSTRtoString(frameNameBSTR).empty() ) |
| return (webFrame == mainFrame) ? "main frame" : string(); |
| |
| string frameName = (webFrame == mainFrame) ? "main frame" : "frame"; |
| frameName += " \"" + BSTRtoString(frameNameBSTR) + "\""; |
| |
| SysFreeString(frameNameBSTR); |
| |
| return frameName; |
| } |
| |
| FrameLoadDelegate::FrameLoadDelegate() |
| : m_refCount(1) |
| , m_gcController(new GCController) |
| , m_accessibilityController(new AccessibilityController) |
| { |
| } |
| |
| FrameLoadDelegate::~FrameLoadDelegate() |
| { |
| } |
| |
| HRESULT STDMETHODCALLTYPE FrameLoadDelegate::QueryInterface(REFIID riid, void** ppvObject) |
| { |
| *ppvObject = 0; |
| if (IsEqualGUID(riid, IID_IUnknown)) |
| *ppvObject = static_cast<IWebFrameLoadDelegate*>(this); |
| else if (IsEqualGUID(riid, IID_IWebFrameLoadDelegate)) |
| *ppvObject = static_cast<IWebFrameLoadDelegate*>(this); |
| else if (IsEqualGUID(riid, IID_IWebFrameLoadDelegate2)) |
| *ppvObject = static_cast<IWebFrameLoadDelegate*>(this); |
| else if (IsEqualGUID(riid, IID_IWebFrameLoadDelegatePrivate)) |
| *ppvObject = static_cast<IWebFrameLoadDelegatePrivate*>(this); |
| else |
| return E_NOINTERFACE; |
| |
| AddRef(); |
| return S_OK; |
| } |
| |
| ULONG STDMETHODCALLTYPE FrameLoadDelegate::AddRef(void) |
| { |
| return ++m_refCount; |
| } |
| |
| ULONG STDMETHODCALLTYPE FrameLoadDelegate::Release(void) |
| { |
| ULONG newRef = --m_refCount; |
| if (!newRef) |
| delete(this); |
| |
| return newRef; |
| } |
| |
| |
| HRESULT STDMETHODCALLTYPE FrameLoadDelegate::didStartProvisionalLoadForFrame( |
| /* [in] */ IWebView* webView, |
| /* [in] */ IWebFrame* frame) |
| { |
| if (!done && gLayoutTestController->dumpFrameLoadCallbacks()) |
| printf("%s - didStartProvisionalLoadForFrame\n", |
| descriptionSuitableForTestResult(frame).c_str()); |
| |
| // Make sure we only set this once per test. If it gets cleared, and then set again, we might |
| // end up doing two dumps for one test. |
| if (!topLoadingFrame && !done) |
| topLoadingFrame = frame; |
| |
| return S_OK; |
| } |
| |
| HRESULT STDMETHODCALLTYPE FrameLoadDelegate::didFailProvisionalLoadWithError( |
| /* [in] */ IWebView *webView, |
| /* [in] */ IWebError *error, |
| /* [in] */ IWebFrame *frame) |
| { |
| if (!done && gLayoutTestController->dumpFrameLoadCallbacks()) |
| printf("%s - didFailProvisionalLoadWithError\n", |
| descriptionSuitableForTestResult(frame).c_str()); |
| |
| return S_OK; |
| } |
| |
| HRESULT STDMETHODCALLTYPE FrameLoadDelegate::didCommitLoadForFrame( |
| /* [in] */ IWebView *webView, |
| /* [in] */ IWebFrame *frame) |
| { |
| COMPtr<IWebViewPrivate> webViewPrivate; |
| HRESULT hr = webView->QueryInterface(&webViewPrivate); |
| if (FAILED(hr)) |
| return hr; |
| webViewPrivate->updateFocusedAndActiveState(); |
| |
| if (!done && gLayoutTestController->dumpFrameLoadCallbacks()) |
| printf("%s - didCommitLoadForFrame\n", |
| descriptionSuitableForTestResult(frame).c_str()); |
| |
| |
| return S_OK; |
| } |
| |
| HRESULT STDMETHODCALLTYPE FrameLoadDelegate::didReceiveTitle( |
| /* [in] */ IWebView *webView, |
| /* [in] */ BSTR title, |
| /* [in] */ IWebFrame *frame) |
| { |
| if (::gLayoutTestController->dumpTitleChanges() && !done) |
| printf("TITLE CHANGED: %S\n", title ? title : L""); |
| return S_OK; |
| } |
| |
| void FrameLoadDelegate::processWork() |
| { |
| // quit doing work once a load is in progress |
| while (!topLoadingFrame && WorkQueue::shared()->count()) { |
| WorkQueueItem* item = WorkQueue::shared()->dequeue(); |
| ASSERT(item); |
| item->invoke(); |
| } |
| |
| // if we didn't start a new load, then we finished all the commands, so we're ready to dump state |
| if (!topLoadingFrame && !::gLayoutTestController->waitToDump()) |
| dump(); |
| } |
| |
| static void CALLBACK processWorkTimer(HWND, UINT, UINT_PTR id, DWORD) |
| { |
| ::KillTimer(0, id); |
| FrameLoadDelegate* d = g_delegateWaitingOnTimer; |
| g_delegateWaitingOnTimer = 0; |
| d->processWork(); |
| } |
| |
| void FrameLoadDelegate::locationChangeDone(IWebError*, IWebFrame* frame) |
| { |
| if (frame != topLoadingFrame) |
| return; |
| |
| topLoadingFrame = 0; |
| WorkQueue::shared()->setFrozen(true); |
| |
| if (::gLayoutTestController->waitToDump()) |
| return; |
| |
| if (WorkQueue::shared()->count()) { |
| ASSERT(!g_delegateWaitingOnTimer); |
| g_delegateWaitingOnTimer = this; |
| ::SetTimer(0, 0, 0, processWorkTimer); |
| return; |
| } |
| |
| dump(); |
| } |
| |
| HRESULT STDMETHODCALLTYPE FrameLoadDelegate::didFinishLoadForFrame( |
| /* [in] */ IWebView* webView, |
| /* [in] */ IWebFrame* frame) |
| { |
| if (!done && gLayoutTestController->dumpFrameLoadCallbacks()) |
| printf("%s - didFinishLoadForFrame\n", |
| descriptionSuitableForTestResult(frame).c_str()); |
| |
| locationChangeDone(0, frame); |
| return S_OK; |
| } |
| |
| HRESULT STDMETHODCALLTYPE FrameLoadDelegate::didFailLoadWithError( |
| /* [in] */ IWebView* webView, |
| /* [in] */ IWebError* error, |
| /* [in] */ IWebFrame* forFrame) |
| { |
| locationChangeDone(error, forFrame); |
| return S_OK; |
| } |
| |
| HRESULT STDMETHODCALLTYPE FrameLoadDelegate::willCloseFrame( |
| /* [in] */ IWebView *webView, |
| /* [in] */ IWebFrame *frame) |
| { |
| return E_NOTIMPL; |
| } |
| |
| HRESULT STDMETHODCALLTYPE FrameLoadDelegate::didClearWindowObject( |
| /* [in] */ IWebView*webView, |
| /* [in] */ JSContextRef context, |
| /* [in] */ JSObjectRef windowObject, |
| /* [in] */ IWebFrame* frame) |
| { |
| JSValueRef exception = 0; |
| |
| ::gLayoutTestController->makeWindowObject(context, windowObject, &exception); |
| ASSERT(!exception); |
| |
| m_gcController->makeWindowObject(context, windowObject, &exception); |
| ASSERT(!exception); |
| |
| m_accessibilityController->makeWindowObject(context, windowObject, &exception); |
| ASSERT(!exception); |
| |
| JSStringRef eventSenderStr = JSStringCreateWithUTF8CString("eventSender"); |
| JSValueRef eventSender = makeEventSender(context); |
| JSObjectSetProperty(context, windowObject, eventSenderStr, eventSender, kJSPropertyAttributeReadOnly | kJSPropertyAttributeDontDelete, 0); |
| JSStringRelease(eventSenderStr); |
| |
| return S_OK; |
| } |
| |
| HRESULT STDMETHODCALLTYPE FrameLoadDelegate::didFinishDocumentLoadForFrame( |
| /* [in] */ IWebView *sender, |
| /* [in] */ IWebFrame *frame) |
| { |
| if (!done && gLayoutTestController->dumpFrameLoadCallbacks()) |
| printf("%s - didFinishDocumentLoadForFrame\n", |
| descriptionSuitableForTestResult(frame).c_str()); |
| if (!done) { |
| COMPtr<IWebFramePrivate> webFramePrivate; |
| HRESULT hr = frame->QueryInterface(&webFramePrivate); |
| if (FAILED(hr)) |
| return hr; |
| unsigned pendingFrameUnloadEvents; |
| hr = webFramePrivate->pendingFrameUnloadEventCount(&pendingFrameUnloadEvents); |
| if (FAILED(hr)) |
| return hr; |
| if (pendingFrameUnloadEvents) |
| printf("%s - has %u onunload handler(s)\n", |
| descriptionSuitableForTestResult(frame).c_str(), pendingFrameUnloadEvents); |
| } |
| |
| return S_OK; |
| } |
| |
| HRESULT STDMETHODCALLTYPE FrameLoadDelegate::didHandleOnloadEventsForFrame( |
| /* [in] */ IWebView *sender, |
| /* [in] */ IWebFrame *frame) |
| { |
| if (!done && gLayoutTestController->dumpFrameLoadCallbacks()) |
| printf("%s - didHandleOnloadEventsForFrame\n", |
| descriptionSuitableForTestResult(frame).c_str()); |
| |
| return S_OK; |
| } |
| |