d96f5dd9b0df0e5890c2b10b737e8f5b0b8d7a3b
[WebKit-https.git] / Source / WebCore / history / CachedPage.cpp
1 /*
2  * Copyright (C) 2006, 2007, 2008, 2014 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25
26 #include "config.h"
27 #include "CachedPage.h"
28
29 #include "Document.h"
30 #include "Element.h"
31 #include "FocusController.h"
32 #include "FrameLoader.h"
33 #include "FrameView.h"
34 #include "HistoryController.h"
35 #include "HistoryItem.h"
36 #include "MainFrame.h"
37 #include "Node.h"
38 #include "Page.h"
39 #include "PageTransitionEvent.h"
40 #include "ScriptDisallowedScope.h"
41 #include "Settings.h"
42 #include "VisitedLinkState.h"
43 #include <wtf/CurrentTime.h>
44 #include <wtf/RefCountedLeakCounter.h>
45 #include <wtf/StdLibExtras.h>
46
47 #if PLATFORM(IOS)
48 #include "FrameSelection.h"
49 #endif
50
51
52 namespace WebCore {
53 using namespace JSC;
54
55 DEFINE_DEBUG_ONLY_GLOBAL(WTF::RefCountedLeakCounter, cachedPageCounter, ("CachedPage"));
56
57 CachedPage::CachedPage(Page& page)
58     : m_page(page)
59     , m_expirationTime(MonotonicTime::now() + Seconds(page.settings().backForwardCacheExpirationInterval()))
60     , m_cachedMainFrame(std::make_unique<CachedFrame>(page.mainFrame()))
61 {
62 #ifndef NDEBUG
63     cachedPageCounter.increment();
64 #endif
65 }
66
67 CachedPage::~CachedPage()
68 {
69 #ifndef NDEBUG
70     cachedPageCounter.decrement();
71 #endif
72
73     if (m_cachedMainFrame)
74         m_cachedMainFrame->destroy();
75 }
76
77 static void firePageShowAndPopStateEvents(Page& page)
78 {
79     // Dispatching JavaScript events can cause frame destruction.
80     auto& mainFrame = page.mainFrame();
81     Vector<Ref<Frame>> childFrames;
82     for (auto* child = mainFrame.tree().traverseNextInPostOrder(CanWrap::Yes); child; child = child->tree().traverseNextInPostOrder(CanWrap::No))
83         childFrames.append(*child);
84
85     for (auto& child : childFrames) {
86         if (!child->tree().isDescendantOf(&mainFrame))
87             continue;
88         auto* document = child->document();
89         if (!document)
90             continue;
91
92         // FIXME: Update Page Visibility state here.
93         // https://bugs.webkit.org/show_bug.cgi?id=116770
94         document->dispatchPageshowEvent(PageshowEventPersisted);
95
96         auto* historyItem = child->loader().history().currentItem();
97         if (historyItem && historyItem->stateObject())
98             document->dispatchPopstateEvent(historyItem->stateObject());
99     }
100 }
101
102 class CachedPageRestorationScope {
103 public:
104     CachedPageRestorationScope(Page& page)
105         : m_page(page)
106     {
107         m_page.setIsRestoringCachedPage(true);
108     }
109
110     ~CachedPageRestorationScope()
111     {
112         m_page.setIsRestoringCachedPage(false);
113     }
114
115 private:
116     Page& m_page;
117 };
118
119 void CachedPage::restore(Page& page)
120 {
121     ASSERT(m_cachedMainFrame);
122     ASSERT(m_cachedMainFrame->view()->frame().isMainFrame());
123     ASSERT(!page.subframeCount());
124
125     CachedPageRestorationScope restorationScope(page);
126     m_cachedMainFrame->open();
127
128     // Restore the focus appearance for the focused element.
129     // FIXME: Right now we don't support pages w/ frames in the b/f cache.  This may need to be tweaked when we add support for that.
130     Document* focusedDocument = page.focusController().focusedOrMainFrame().document();
131     if (Element* element = focusedDocument->focusedElement()) {
132 #if PLATFORM(IOS)
133         // We don't want focused nodes changing scroll position when restoring from the cache
134         // as it can cause ugly jumps before we manage to restore the cached position.
135         page.mainFrame().selection().suppressScrolling();
136
137         bool hadProhibitsScrolling = false;
138         FrameView* frameView = page.mainFrame().view();
139         if (frameView) {
140             hadProhibitsScrolling = frameView->prohibitsScrolling();
141             frameView->setProhibitsScrolling(true);
142         }
143 #endif
144         element->updateFocusAppearance(SelectionRestorationMode::Restore);
145 #if PLATFORM(IOS)
146         if (frameView)
147             frameView->setProhibitsScrolling(hadProhibitsScrolling);
148         page.mainFrame().selection().restoreScrolling();
149 #endif
150     }
151
152     if (m_needsDeviceOrPageScaleChanged)
153         page.mainFrame().deviceOrPageScaleFactorChanged();
154
155     page.setNeedsRecalcStyleInAllFrames();
156
157 #if ENABLE(VIDEO_TRACK)
158     if (m_needsCaptionPreferencesChanged)
159         page.captionPreferencesChanged();
160 #endif
161
162     if (m_needsUpdateContentsSize) {
163         if (FrameView* frameView = page.mainFrame().view())
164             frameView->updateContentsSize();
165     }
166
167     firePageShowAndPopStateEvents(page);
168
169     clear();
170 }
171
172 void CachedPage::clear()
173 {
174     ASSERT(m_cachedMainFrame);
175     m_cachedMainFrame->clear();
176     m_cachedMainFrame = nullptr;
177 #if ENABLE(VIDEO_TRACK)
178     m_needsCaptionPreferencesChanged = false;
179 #endif
180     m_needsDeviceOrPageScaleChanged = false;
181     m_needsUpdateContentsSize = false;
182 }
183
184 bool CachedPage::hasExpired() const
185 {
186     return MonotonicTime::now() > m_expirationTime;
187 }
188
189 } // namespace WebCore