View scale changes are temporarily lost after restoring a page from the page cache
[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 "FrameView.h"
33 #include "MainFrame.h"
34 #include "Node.h"
35 #include "Page.h"
36 #include "Settings.h"
37 #include "VisitedLinkState.h"
38 #include <wtf/CurrentTime.h>
39 #include <wtf/RefCountedLeakCounter.h>
40 #include <wtf/StdLibExtras.h>
41
42 #if PLATFORM(IOS)
43 #include "FrameSelection.h"
44 #endif
45
46 using namespace JSC;
47
48 namespace WebCore {
49
50 DEFINE_DEBUG_ONLY_GLOBAL(WTF::RefCountedLeakCounter, cachedPageCounter, ("CachedPage"));
51
52 CachedPage::CachedPage(Page& page)
53     : m_expirationTime(monotonicallyIncreasingTime() + page.settings().backForwardCacheExpirationInterval())
54     , m_cachedMainFrame(std::make_unique<CachedFrame>(page.mainFrame()))
55 {
56 #ifndef NDEBUG
57     cachedPageCounter.increment();
58 #endif
59 }
60
61 CachedPage::~CachedPage()
62 {
63 #ifndef NDEBUG
64     cachedPageCounter.decrement();
65 #endif
66
67     if (m_cachedMainFrame)
68         m_cachedMainFrame->destroy();
69 }
70
71 void CachedPage::restore(Page& page)
72 {
73     ASSERT(m_cachedMainFrame);
74     ASSERT(m_cachedMainFrame->view()->frame().isMainFrame());
75     ASSERT(!page.subframeCount());
76
77     m_cachedMainFrame->open();
78     
79     // Restore the focus appearance for the focused element.
80     // 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.
81     Document* focusedDocument = page.focusController().focusedOrMainFrame().document();
82     if (Element* element = focusedDocument->focusedElement()) {
83 #if PLATFORM(IOS)
84         // We don't want focused nodes changing scroll position when restoring from the cache
85         // as it can cause ugly jumps before we manage to restore the cached position.
86         page.mainFrame().selection().suppressScrolling();
87
88         bool hadProhibitsScrolling = false;
89         FrameView* frameView = page.mainFrame().view();
90         if (frameView) {
91             hadProhibitsScrolling = frameView->prohibitsScrolling();
92             frameView->setProhibitsScrolling(true);
93         }
94 #endif
95         element->updateFocusAppearance(true);
96 #if PLATFORM(IOS)
97         if (frameView)
98             frameView->setProhibitsScrolling(hadProhibitsScrolling);
99         page.mainFrame().selection().restoreScrolling();
100 #endif
101     }
102
103     if (m_needStyleRecalcForVisitedLinks) {
104         for (Frame* frame = &page.mainFrame(); frame; frame = frame->tree().traverseNext())
105             frame->document()->visitedLinkState().invalidateStyleForAllLinks();
106     }
107
108     if (m_needsDeviceOrPageScaleChanged)
109         page.mainFrame().deviceOrPageScaleFactorChanged();
110
111     if (m_needsFullStyleRecalc)
112         page.setNeedsRecalcStyleInAllFrames();
113
114 #if ENABLE(VIDEO_TRACK)
115     if (m_needsCaptionPreferencesChanged)
116         page.captionPreferencesChanged();
117 #endif
118
119     clear();
120 }
121
122 void CachedPage::clear()
123 {
124     ASSERT(m_cachedMainFrame);
125     m_cachedMainFrame->clear();
126     m_cachedMainFrame = nullptr;
127     m_needStyleRecalcForVisitedLinks = false;
128     m_needsFullStyleRecalc = false;
129 #if ENABLE(VIDEO_TRACK)
130     m_needsCaptionPreferencesChanged = false;
131 #endif
132     m_needsDeviceOrPageScaleChanged = false;
133 }
134
135 bool CachedPage::hasExpired() const
136 {
137     return monotonicallyIncreasingTime() > m_expirationTime;
138 }
139
140 } // namespace WebCore