CachedFrame.cpp 9.38 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
/*
 * Copyright (C) 2009 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 COMPUTER, INC. ``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 COMPUTER, INC. OR
 * 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 "CachedPage.h"

#include "CachedFramePlatformData.h"
30
#include "Document.h"
31
#include "DocumentLoader.h"
32
#include "ExceptionCode.h"
ap@apple.com's avatar
ap@apple.com committed
33
#include "EventNames.h"
34
#include "FocusController.h"
35
#include "Frame.h"
36
#include "FrameLoaderClient.h"
37
#include "FrameView.h"
38
#include "HistoryItem.h"
39
#include "Logging.h"
40
#include "PageTransitionEvent.h"
41
#include "SerializedScriptValue.h"
42
#include <wtf/text/CString.h>
43 44 45 46 47 48
#include <wtf/RefCountedLeakCounter.h>

#if ENABLE(SVG)
#include "SVGDocumentExtensions.h"
#endif

49 50 51 52 53 54
#if ENABLE(TOUCH_EVENTS)
#include "Chrome.h"
#include "ChromeClient.h"
#include "Page.h"
#endif

55 56 57 58 59 60 61 62 63 64
namespace WebCore {

#ifndef NDEBUG
static WTF::RefCountedLeakCounter& cachedFrameCounter()
{
    DEFINE_STATIC_LOCAL(WTF::RefCountedLeakCounter, counter, ("CachedFrame"));
    return counter;
}
#endif

65
CachedFrameBase::CachedFrameBase(Frame* frame)
66 67 68 69
    : m_document(frame->document())
    , m_documentLoader(frame->loader()->documentLoader())
    , m_view(frame->view())
    , m_mousePressNode(frame->eventHandler()->mousePressNode())
70
    , m_url(frame->document()->url())
71
    , m_isMainFrame(!frame->tree()->parent())
72
{
73 74 75 76 77 78 79 80 81 82 83 84 85 86
}

CachedFrameBase::~CachedFrameBase()
{
#ifndef NDEBUG
    cachedFrameCounter().decrement();
#endif
    // CachedFrames should always have had destroy() called by their parent CachedPage
    ASSERT(!m_document);
}

void CachedFrameBase::restore()
{
    ASSERT(m_document->view() == m_view);
87

88 89 90
    if (m_isMainFrame)
        m_view->setParentVisible(true);

91 92 93 94 95 96 97 98
    Frame* frame = m_view->frame();
    m_cachedFrameScriptData->restore(frame);

#if ENABLE(SVG)
    if (m_document->svgExtensions())
        m_document->accessSVGExtensions()->unpauseAnimations();
#endif

99
    frame->animation()->resumeAnimationsForDocument(m_document.get());
100 101
    frame->eventHandler()->setMousePressNode(m_mousePressNode.get());
    m_document->resumeActiveDOMObjects();
102
    m_document->resumeScriptedAnimationControllerCallbacks();
103 104 105 106 107

    // It is necessary to update any platform script objects after restoring the
    // cached page.
    frame->script()->updatePlatformScriptObjects();

108 109
    frame->loader()->client()->didRestoreFromPageCache();

110 111 112
    // Reconstruct the FrameTree
    for (unsigned i = 0; i < m_childFrames.size(); ++i)
        frame->tree()->appendChild(m_childFrames[i]->view()->frame());
113 114 115

    // Open the child CachedFrames in their respective FrameLoaders.
    for (unsigned i = 0; i < m_childFrames.size(); ++i)
116
        m_childFrames[i]->open();
117

118
    m_document->enqueuePageshowEvent(PageshowEventPersisted);
119 120 121 122
    
    HistoryItem* historyItem = frame->loader()->history()->currentItem();
    m_document->enqueuePopstateEvent(historyItem && historyItem->stateObject() ? historyItem->stateObject() : SerializedScriptValue::nullValue());
    
123 124 125 126
#if ENABLE(TOUCH_EVENTS)
    if (m_document->hasListenerType(Document::TOUCH_LISTENER))
        m_document->page()->chrome()->client()->needTouchEvents(true);
#endif
127 128

    m_document->documentDidBecomeActive();
129 130 131 132 133
}

CachedFrame::CachedFrame(Frame* frame)
    : CachedFrameBase(frame)
{
134 135 136
#ifndef NDEBUG
    cachedFrameCounter().increment();
#endif
beidson@apple.com's avatar
beidson@apple.com committed
137
    ASSERT(m_document);
138 139
    ASSERT(m_documentLoader);
    ASSERT(m_view);
beidson@apple.com's avatar
beidson@apple.com committed
140

141 142 143
    if (frame->page()->focusController()->focusedFrame() == frame)
        frame->page()->focusController()->setFocusedFrame(frame->page()->mainFrame());

144 145 146
    // Custom scrollbar renderers will get reattached when the document comes out of the page cache
    m_view->detachCustomScrollbars();

ap@apple.com's avatar
ap@apple.com committed
147
    m_document->documentWillBecomeInactive();
148 149
    frame->clearTimers();
    m_document->setInPageCache(true);
ap@apple.com's avatar
ap@apple.com committed
150
    frame->loader()->stopLoading(UnloadEventPolicyUnloadAndPageHide);
151

152
    // Create the CachedFrames for all Frames in the FrameTree.
153 154 155
    for (Frame* child = frame->tree()->firstChild(); child; child = child->tree()->nextSibling())
        m_childFrames.append(CachedFrame::create(child));

156 157 158 159 160
    // Active DOM objects must be suspended before we cache the frame script data,
    // but after we've fired the pagehide event, in case that creates more objects.
    // Suspending must also happen after we've recursed over child frames, in case
    // those create more objects.
    // FIXME: It's still possible to have objects created after suspending in some cases, see http://webkit.org/b/53733 for more details.
161
    m_document->suspendScriptedAnimationControllerCallbacks();
162 163 164 165 166
    m_document->suspendActiveDOMObjects(ActiveDOMObject::DocumentWillBecomeInactive);
    m_cachedFrameScriptData = adoptPtr(new ScriptCachedFrameData(frame));

    frame->loader()->client()->savePlatformDataToCachedFrame(this);

167 168 169 170 171 172
    // Deconstruct the FrameTree, to restore it later.
    // We do this for two reasons:
    // 1 - We reuse the main frame, so when it navigates to a new page load it needs to start with a blank FrameTree.
    // 2 - It's much easier to destroy a CachedFrame while it resides in the PageCache if it is disconnected from its parent.
    for (unsigned i = 0; i < m_childFrames.size(); ++i)
        frame->tree()->removeChild(m_childFrames[i]->view()->frame());
173

ap@apple.com's avatar
ap@apple.com committed
174 175 176
    if (!m_isMainFrame)
        frame->page()->decrementFrameCount();

177 178
    frame->loader()->client()->didSaveToPageCache();

179 180 181 182 183 184
#ifndef NDEBUG
    if (m_isMainFrame)
        LOG(PageCache, "Finished creating CachedFrame for main frame url '%s' and DocumentLoader %p\n", m_url.string().utf8().data(), m_documentLoader.get());
    else
        LOG(PageCache, "Finished creating CachedFrame for child frame with url '%s' and DocumentLoader %p\n", m_url.string().utf8().data(), m_documentLoader.get());
#endif
185 186 187 188 189

#if ENABLE(TOUCH_EVENTS)
    if (m_document->hasListenerType(Document::TOUCH_LISTENER))
        m_document->page()->chrome()->client()->needTouchEvents(false);
#endif
190 191
}

192
void CachedFrame::open()
193
{
194 195
    ASSERT(m_view);
    m_view->frame()->loader()->open(*this);
ap@apple.com's avatar
ap@apple.com committed
196 197 198

    if (!m_isMainFrame)
        m_view->frame()->page()->incrementFrameCount();
199 200 201 202 203 204 205
}

void CachedFrame::clear()
{
    if (!m_document)
        return;

206 207 208 209 210
    // clear() should only be called for Frames representing documents that are no longer in the page cache.
    // This means the CachedFrame has been:
    // 1 - Successfully restore()'d by going back/forward.
    // 2 - destroy()'ed because the PageCache is pruning or the WebView was closed.
    ASSERT(!m_document->inPageCache());
211 212 213
    ASSERT(m_view);
    ASSERT(m_document->frame() == m_view->frame());

214 215 216
    for (int i = m_childFrames.size() - 1; i >= 0; --i)
        m_childFrames[i]->clear();

217 218 219
    m_document = 0;
    m_view = 0;
    m_mousePressNode = 0;
220
    m_url = KURL();
221 222 223 224 225

    m_cachedFramePlatformData.clear();
    m_cachedFrameScriptData.clear();
}

226 227 228 229 230 231 232 233 234 235
void CachedFrame::destroy()
{
    if (!m_document)
        return;
    
    // Only CachedFrames that are still in the PageCache should be destroyed in this manner
    ASSERT(m_document->inPageCache());
    ASSERT(m_view);
    ASSERT(m_document->frame() == m_view->frame());

236 237 238 239 240
    if (!m_isMainFrame) {
        m_view->frame()->detachFromPage();
        m_view->frame()->loader()->detachViewsAndDocumentLoader();
    }
    
241 242 243
    for (int i = m_childFrames.size() - 1; i >= 0; --i)
        m_childFrames[i]->destroy();

244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260
    if (m_cachedFramePlatformData)
        m_cachedFramePlatformData->clear();

    Frame::clearTimers(m_view.get(), m_document.get());

    // FIXME: Why do we need to call removeAllEventListeners here? When the document is in page cache, this method won't work
    // fully anyway, because the document won't be able to access its DOMWindow object (due to being frameless).
    m_document->removeAllEventListeners();

    m_document->setInPageCache(false);
    // FIXME: We don't call willRemove here. Why is that OK?
    m_document->detach();
    m_view->clearFrame();

    clear();
}

261
void CachedFrame::setCachedFramePlatformData(PassOwnPtr<CachedFramePlatformData> data)
262
{
263
    m_cachedFramePlatformData = data;
264 265 266 267 268 269 270
}

CachedFramePlatformData* CachedFrame::cachedFramePlatformData()
{
    return m_cachedFramePlatformData.get();
}

beidson@apple.com's avatar
beidson@apple.com committed
271 272 273 274 275 276 277 278 279
int CachedFrame::descendantFrameCount() const
{
    int count = m_childFrames.size();
    for (size_t i = 0; i < m_childFrames.size(); ++i)
        count += m_childFrames[i]->descendantFrameCount();
    
    return count;
}

280
} // namespace WebCore