/* * Copyright (C) 2006, 2007, 2008, 2009, 2011 Apple Inc. All rights reserved. * Copyright (C) 2008, 2009 Torch Mobile Inc. All rights reserved. (http://www.torchmobile.com/) * Copyright (C) Research In Motion Limited 2009. 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. */ #ifndef FrameLoader_h #define FrameLoader_h #include "CachePolicy.h" #include "FrameLoaderStateMachine.h" #include "FrameLoaderTypes.h" #include "HistoryController.h" #include "IconDatabaseBase.h" #include "PolicyChecker.h" #include "ResourceLoadNotifier.h" #include "ScriptValue.h" #include "SubframeLoader.h" #include "ThreadableLoader.h" #include "Timer.h" #include #include namespace WebCore { class Archive; class AuthenticationChallenge; class CachedFrameBase; class CachedPage; class CachedResource; class Chrome; class DOMWrapperWorld; class Document; class DocumentLoader; class Event; class FormData; class FormState; class FormSubmission; class Frame; class FrameLoaderClient; class FrameNetworkingContext; class HistoryItem; class HTMLFormElement; class IconLoader; class NavigationAction; class NetworkingContext; class Page; class ProtectionSpace; class ResourceError; class ResourceLoader; class ResourceRequest; class ResourceResponse; class ScriptSourceCode; class ScriptValue; class SecurityOrigin; class SerializedScriptValue; class SharedBuffer; class StringWithDirection; class SubstituteData; class TextResourceDecoder; struct FrameLoadRequest; struct WindowFeatures; bool isBackForwardLoadType(FrameLoadType); class FrameLoader { WTF_MAKE_NONCOPYABLE(FrameLoader); public: FrameLoader(Frame*, FrameLoaderClient*); ~FrameLoader(); void init(); Frame* frame() const { return m_frame; } PolicyChecker* policyChecker() const { return &m_policyChecker; } HistoryController* history() const { return &m_history; } ResourceLoadNotifier* notifier() const { return &m_notifer; } SubframeLoader* subframeLoader() const { return &m_subframeLoader; } // FIXME: This is not cool, people. There are too many different functions that all start loads. // We should aim to consolidate these into a smaller set of functions, and try to reuse more of // the logic by extracting common code paths. void prepareForLoadStart(); void setupForReplace(); void setupForReplaceByMIMEType(const String& newMIMEType); void loadURLIntoChildFrame(const KURL&, const String& referer, Frame*); void loadFrameRequest(const FrameLoadRequest&, bool lockHistory, bool lockBackForwardList, // Called by submitForm, calls loadPostRequest and loadURL. PassRefPtr, PassRefPtr, ReferrerPolicy); void load(const ResourceRequest&, bool lockHistory); // Called by WebFrame, calls load(ResourceRequest, SubstituteData). void load(const ResourceRequest&, const SubstituteData&, bool lockHistory); // Called both by WebFrame and internally, calls load(DocumentLoader*). void load(const ResourceRequest&, const String& frameName, bool lockHistory); // Called by WebPluginController. #if ENABLE(WEB_ARCHIVE) void loadArchive(PassRefPtr); #endif static void reportLocalLoadFailed(Frame*, const String& url); unsigned long loadResourceSynchronously(const ResourceRequest&, StoredCredentials, ResourceError&, ResourceResponse&, Vector& data); bool canHandleRequest(const ResourceRequest&); // Also not cool. void stopAllLoaders(ClearProvisionalItemPolicy = ShouldClearProvisionalItem); void stopForUserCancel(bool deferCheckLoadComplete = false); bool isLoadingMainResource() const { return m_isLoadingMainResource; } bool isLoading() const; bool frameHasLoaded() const; void transferLoadingResourcesFromPage(Page*); void dispatchTransferLoadingResourceFromPage(unsigned long, DocumentLoader*, const ResourceRequest&, Page*); int numPendingOrLoadingRequests(bool recurse) const; String referrer() const; String outgoingReferrer() const; String outgoingOrigin() const; DocumentLoader* activeDocumentLoader() const; DocumentLoader* documentLoader() const { return m_documentLoader.get(); } DocumentLoader* policyDocumentLoader() const { return m_policyDocumentLoader.get(); } DocumentLoader* provisionalDocumentLoader() const { return m_provisionalDocumentLoader.get(); } FrameState state() const { return m_state; } static double timeOfLastCompletedLoad(); bool shouldUseCredentialStorage(ResourceLoader*); #if USE(PROTECTION_SPACE_AUTH_CALLBACK) bool canAuthenticateAgainstProtectionSpace(ResourceLoader* loader, const ProtectionSpace& protectionSpace); #endif const ResourceRequest& originalRequest() const; const ResourceRequest& initialRequest() const; void receivedMainResourceError(const ResourceError&, bool isComplete); bool willLoadMediaElementURL(KURL&); void handleFallbackContent(); bool isStopping() const; void finishedLoading(); ResourceError cancelledError(const ResourceRequest&) const; ResourceError fileDoesNotExistError(const ResourceResponse&) const; ResourceError blockedError(const ResourceRequest&) const; ResourceError cannotShowURLError(const ResourceRequest&) const; ResourceError interruptionForPolicyChangeError(const ResourceRequest&) const; bool isHostedByObjectElement() const; bool isLoadingMainFrame() const; bool canShowMIMEType(const String& MIMEType) const; bool representationExistsForURLScheme(const String& URLScheme); String generatedMIMETypeForURLScheme(const String& URLScheme); void reload(bool endToEndReload = false); void reloadWithOverrideEncoding(const String& overrideEncoding); void didReceiveServerRedirectForProvisionalLoadForFrame(); void finishedLoadingDocument(DocumentLoader*); bool isReplacing() const; void setReplacing(); void revertToProvisional(DocumentLoader*); void setMainDocumentError(DocumentLoader*, const ResourceError&); void mainReceivedCompleteError(DocumentLoader*, const ResourceError&); bool subframeIsLoading() const; void willChangeTitle(DocumentLoader*); void didChangeTitle(DocumentLoader*); void didChangeIcons(DocumentLoader*); FrameLoadType loadType() const; CachePolicy subresourceCachePolicy() const; void didFirstLayout(); void didFirstVisuallyNonEmptyLayout(); void loadedResourceFromMemoryCache(const CachedResource*); void tellClientAboutPastMemoryCacheLoads(); void checkLoadComplete(); void detachFromParent(); void detachViewsAndDocumentLoader(); void addExtraFieldsToSubresourceRequest(ResourceRequest&); void addExtraFieldsToMainResourceRequest(ResourceRequest&); static void addHTTPOriginIfNeeded(ResourceRequest&, String origin); FrameLoaderClient* client() const { return m_client; } void setDefersLoading(bool); void changeLocation(PassRefPtr, const KURL&, const String& referrer, bool lockHistory = true, bool lockBackForwardList = true, bool refresh = false); void urlSelected(const KURL&, const String& target, PassRefPtr, bool lockHistory, bool lockBackForwardList, ReferrerPolicy); void submitForm(PassRefPtr); void stop(); void stopLoading(UnloadEventPolicy); bool closeURL(); void didExplicitOpen(); // Callbacks from DocumentWriter void didBeginDocument(bool dispatchWindowObjectAvailable); void didEndDocument(); void willSetEncoding(); KURL iconURL(); void commitIconURLToIconDatabase(const KURL&); KURL baseURL() const; void handledOnloadEvents(); String userAgent(const KURL&) const; void dispatchDidClearWindowObjectInWorld(DOMWrapperWorld*); void dispatchDidClearWindowObjectsInAllWorlds(); void dispatchDocumentElementAvailable(); void ownerElementSandboxFlagsChanged() { updateSandboxFlags(); } bool isSandboxed(SandboxFlags mask) const { return m_sandboxFlags & mask; } SandboxFlags sandboxFlags() const { return m_sandboxFlags; } // The following sandbox flags will be forced, regardless of changes to // the sandbox attribute of any parent frames. void setForcedSandboxFlags(SandboxFlags flags) { m_forcedSandboxFlags = flags; m_sandboxFlags |= flags; } // Mixed content related functions. static bool isMixedContent(SecurityOrigin* context, const KURL&); void checkIfDisplayInsecureContent(SecurityOrigin* context, const KURL&); void checkIfRunInsecureContent(SecurityOrigin* context, const KURL&); Frame* opener(); void setOpener(Frame*); bool isProcessingUserGesture(); void resetMultipleFormSubmissionProtection(); void checkCallImplicitClose(); void frameDetached(); void setOutgoingReferrer(const KURL&); void loadDone(); void finishedParsing(); void checkCompleted(); void checkDidPerformFirstNavigation(); bool isComplete() const; KURL completeURL(const String& url); void cancelAndClear(); void setTitle(const StringWithDirection&); void setIconURL(const String&); void commitProvisionalLoad(); bool isLoadingFromCachedPage() const { return m_loadingFromCachedPage; } FrameLoaderStateMachine* stateMachine() const { return &m_stateMachine; } void startIconLoader(); void iconLoadDecisionReceived(IconLoadDecision); void continueIconLoadWithDecision(IconLoadDecision); bool shouldAllowNavigation(Frame* targetFrame) const; Frame* findFrameForNavigation(const AtomicString& name); void applyUserAgent(ResourceRequest& request); bool shouldInterruptLoadForXFrameOptions(const String&, const KURL&); void open(CachedFrameBase&); #if ENABLE(PLUGIN_PROXY_FOR_VIDEO) void hideMediaPlayerProxyPlugin(Widget*); void showMediaPlayerProxyPlugin(Widget*); #endif // FIXME: Should these really be public? void completed(); bool allAncestorsAreComplete() const; // including this bool allChildrenAreComplete() const; // immediate children, not all descendants void clientRedirected(const KURL&, double delay, double fireDate, bool lockBackForwardList); void clientRedirectCancelledOrFinished(bool cancelWithLoadInProgress); void loadItem(HistoryItem*, FrameLoadType); // FIXME: This is public because this asynchronous callback from the FrameLoaderClient // uses the policy machinery (and therefore is called via the PolicyChecker). Once we // introduce a proper callback type for this function, we should make it private again. void continueLoadAfterWillSubmitForm(); bool suppressOpenerInNewFrame() const { return m_suppressOpenerInNewFrame; } static ObjectContentType defaultObjectContentType(const KURL&, const String& mimeType, bool shouldPreferPlugInsForImages); void clear(bool clearWindowProperties = true, bool clearScriptObjects = true, bool clearFrameView = true); bool quickRedirectComing() const { return m_quickRedirectComing; } bool shouldClose(); void started(); bool pageDismissalEventBeingDispatched() const { return m_pageDismissalEventBeingDispatched; } NetworkingContext* networkingContext() const; private: void checkTimerFired(Timer*); void loadSameDocumentItem(HistoryItem*); void loadDifferentDocumentItem(HistoryItem*, FrameLoadType); void loadProvisionalItemFromCachedPage(); void receivedFirstData(); void updateFirstPartyForCookies(); void setFirstPartyForCookies(const KURL&); void addExtraFieldsToRequest(ResourceRequest&, FrameLoadType loadType, bool isMainResource, bool cookiePolicyURLFromRequest); // Also not cool. void stopLoadingSubframes(ClearProvisionalItemPolicy); void clearProvisionalLoad(); void markLoadComplete(); void transitionToCommitted(PassRefPtr); void frameLoadCompleted(); void mainReceivedError(const ResourceError&, bool isComplete); static void callContinueLoadAfterNavigationPolicy(void*, const ResourceRequest&, PassRefPtr, bool shouldContinue); static void callContinueLoadAfterNewWindowPolicy(void*, const ResourceRequest&, PassRefPtr, const String& frameName, const NavigationAction&, bool shouldContinue); static void callContinueFragmentScrollAfterNavigationPolicy(void*, const ResourceRequest&, PassRefPtr, bool shouldContinue); bool fireBeforeUnloadEvent(Chrome*); void continueLoadAfterNavigationPolicy(const ResourceRequest&, PassRefPtr, bool shouldContinue); void continueLoadAfterNewWindowPolicy(const ResourceRequest&, PassRefPtr, const String& frameName, const NavigationAction&, bool shouldContinue); void continueFragmentScrollAfterNavigationPolicy(const ResourceRequest&, bool shouldContinue); bool shouldScrollToAnchor(bool isFormSubmission, const String& httpMethod, FrameLoadType, const KURL&); void checkLoadCompleteForThisFrame(); void setDocumentLoader(DocumentLoader*); void setPolicyDocumentLoader(DocumentLoader*); void setProvisionalDocumentLoader(DocumentLoader*); void setState(FrameState); void closeOldDataSources(); void prepareForCachedPageRestore(); bool shouldReloadToHandleUnreachableURL(DocumentLoader*); void dispatchDidCommitLoad(); void urlSelected(const FrameLoadRequest&, PassRefPtr, bool lockHistory, bool lockBackForwardList, ReferrerPolicy, ShouldReplaceDocumentIfJavaScriptURL); void loadWithDocumentLoader(DocumentLoader*, FrameLoadType, PassRefPtr); // Calls continueLoadAfterNavigationPolicy void load(DocumentLoader*); // Calls loadWithDocumentLoader void loadWithNavigationAction(const ResourceRequest&, const NavigationAction&, // Calls loadWithDocumentLoader bool lockHistory, FrameLoadType, PassRefPtr); void loadPostRequest(const ResourceRequest&, const String& referrer, // Called by loadFrameRequest, calls loadWithNavigationAction const String& frameName, bool lockHistory, FrameLoadType, PassRefPtr, PassRefPtr); void loadURL(const KURL&, const String& referrer, const String& frameName, // Called by loadFrameRequest, calls loadWithNavigationAction or dispatches to navigation policy delegate bool lockHistory, FrameLoadType, PassRefPtr, PassRefPtr); bool shouldReload(const KURL& currentURL, const KURL& destinationURL); void requestFromDelegate(ResourceRequest&, unsigned long& identifier, ResourceError&); void recursiveCheckLoadComplete(); void detachChildren(); void closeAndRemoveChild(Frame*); void loadInSameDocument(const KURL&, SerializedScriptValue* stateObject, bool isNewNavigation); void provisionalLoadStarted(); bool didOpenURL(const KURL&); void scheduleCheckCompleted(); void scheduleCheckLoadComplete(); void startCheckCompleteTimer(); KURL originalRequestURL() const; bool shouldTreatURLAsSameAsCurrent(const KURL&) const; void updateSandboxFlags(); Frame* m_frame; FrameLoaderClient* m_client; mutable PolicyChecker m_policyChecker; mutable HistoryController m_history; mutable ResourceLoadNotifier m_notifer; mutable SubframeLoader m_subframeLoader; mutable FrameLoaderStateMachine m_stateMachine; FrameState m_state; FrameLoadType m_loadType; // Document loaders for the three phases of frame loading. Note that while // a new request is being loaded, the old document loader may still be referenced. // E.g. while a new request is in the "policy" state, the old document loader may // be consulted in particular as it makes sense to imply certain settings on the new loader. RefPtr m_documentLoader; RefPtr m_provisionalDocumentLoader; RefPtr m_policyDocumentLoader; bool m_delegateIsHandlingProvisionalLoadError; bool m_quickRedirectComing; bool m_sentRedirectNotification; bool m_inStopAllLoaders; String m_outgoingReferrer; bool m_isExecutingJavaScriptFormAction; bool m_didCallImplicitClose; bool m_wasUnloadEventEmitted; bool m_pageDismissalEventBeingDispatched; bool m_isComplete; bool m_isLoadingMainResource; RefPtr m_pendingStateObject; KURL m_workingURL; OwnPtr m_iconLoader; bool m_mayLoadIconLater; bool m_needsClear; KURL m_submittedFormURL; Timer m_checkTimer; bool m_shouldCallCheckCompleted; bool m_shouldCallCheckLoadComplete; Frame* m_opener; HashSet m_openedFrames; bool m_didPerformFirstNavigation; bool m_loadingFromCachedPage; bool m_suppressOpenerInNewFrame; SandboxFlags m_sandboxFlags; SandboxFlags m_forcedSandboxFlags; RefPtr m_networkingContext; KURL m_previousUrl; }; // This function is called by createWindow() in JSDOMWindowBase.cpp, for example, for // modal dialog creation. The lookupFrame is for looking up the frame name in case // the frame name references a frame different from the openerFrame, e.g. when it is // "_self" or "_parent". // // FIXME: Consider making this function part of an appropriate class (not FrameLoader) // and moving it to a more appropriate location. Frame* createWindow(Frame* openerFrame, Frame* lookupFrame, const FrameLoadRequest&, const WindowFeatures&, bool& created); } // namespace WebCore #endif // FrameLoader_h