Allow fixed background layers to be moved by the ScrollingCoordinator
[WebKit-https.git] / Source / WebCore / page / scrolling / ScrollingCoordinator.h
1 /*
2  * Copyright (C) 2011 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. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #ifndef ScrollingCoordinator_h
27 #define ScrollingCoordinator_h
28
29 #include "IntRect.h"
30 #include "LayoutRect.h"
31 #include "PlatformWheelEvent.h"
32 #include "RenderObject.h"
33 #include "ScrollTypes.h"
34 #include "Timer.h"
35 #include <wtf/Forward.h>
36
37 #if ENABLE(THREADED_SCROLLING)
38 #include <wtf/HashMap.h>
39 #include <wtf/ThreadSafeRefCounted.h>
40 #include <wtf/Threading.h>
41 #endif
42
43 #if PLATFORM(MAC)
44 #include <wtf/RetainPtr.h>
45 #endif
46
47 namespace WebCore {
48
49 typedef unsigned MainThreadScrollingReasons;
50 typedef uint64_t ScrollingNodeID;
51
52 enum ScrollingNodeType { ScrollingNode, FixedNode, StickyNode };
53
54 class Document;
55 class Frame;
56 class FrameView;
57 class GraphicsLayer;
58 class Page;
59 class Region;
60 class ScrollableArea;
61 class ViewportConstraints;
62
63 #if ENABLE(THREADED_SCROLLING)
64 class ScrollingTree;
65 #endif
66
67 IntSize scrollOffsetForFixedPosition(const IntRect& visibleContentRect, const IntSize& contentsSize, const IntPoint& scrollPosition,
68     const IntPoint& scrollOrigin, float frameScaleFactor, bool fixedElementsLayoutRelativeToFrame);
69
70 enum SetOrSyncScrollingLayerPosition {
71     SetScrollingLayerPosition,
72     SyncScrollingLayerPosition
73 };
74
75 class ScrollingCoordinator : public ThreadSafeRefCounted<ScrollingCoordinator> {
76 public:
77     static PassRefPtr<ScrollingCoordinator> create(Page*);
78     virtual ~ScrollingCoordinator();
79
80     virtual void pageDestroyed();
81
82 #if ENABLE(THREADED_SCROLLING)
83     virtual ScrollingTree* scrollingTree() const { return 0; }
84 #endif
85
86     // Return whether this scrolling coordinator handles scrolling for the given frame view.
87     bool coordinatesScrollingForFrameView(FrameView*) const;
88
89     // Should be called whenever the given frame view has been laid out.
90     virtual void frameViewLayoutUpdated(FrameView*) { }
91
92     // Should be called whenever a wheel event handler is added or removed in the 
93     // frame view's underlying document.
94     void frameViewWheelEventHandlerCountChanged(FrameView*);
95
96     // Should be called whenever the slow repaint objects counter changes between zero and one.
97     void frameViewHasSlowRepaintObjectsDidChange(FrameView*);
98
99     // Should be called whenever the set of fixed objects changes.
100     void frameViewFixedObjectsDidChange(FrameView*);
101
102     // Should be called whenever the root layer for the given frame view changes.
103     virtual void frameViewRootLayerDidChange(FrameView*);
104
105     // Return whether this scrolling coordinator can keep fixed position layers fixed to their
106     // containers while scrolling.
107     virtual bool supportsFixedPositionLayers() const { return false; }
108
109 #if PLATFORM(MAC) || (PLATFORM(CHROMIUM) && OS(DARWIN))
110     // Dispatched by the scrolling tree during handleWheelEvent. This is required as long as scrollbars are painted on the main thread.
111     void handleWheelEventPhase(PlatformWheelEventPhase);
112 #endif
113
114     // Force all scroll layer position updates to happen on the main thread.
115     void setForceMainThreadScrollLayerPositionUpdates(bool);
116
117     // These virtual functions are currently unique to the threaded scrolling architecture. 
118     // Their meaningful implementations are in ScrollingCoordinatorMac.
119     virtual void commitTreeStateIfNeeded() { }
120     virtual bool requestScrollPositionUpdate(FrameView*, const IntPoint&) { return false; }
121     virtual bool handleWheelEvent(FrameView*, const PlatformWheelEvent&) { return true; }
122     virtual ScrollingNodeID attachToStateTree(ScrollingNodeType, ScrollingNodeID newNodeID, ScrollingNodeID /*parentID*/) { return newNodeID; }
123     virtual void detachFromStateTree(ScrollingNodeID) { }
124     virtual void clearStateTree() { }
125     virtual void updateViewportConstrainedNode(ScrollingNodeID, const ViewportConstraints&, GraphicsLayer*) { }
126     virtual void updateScrollingNode(ScrollingNodeID, GraphicsLayer* /*scrollLayer*/, GraphicsLayer* /*counterScrollingLayer*/) { }
127     virtual void syncChildPositions(const LayoutRect&) { }
128     virtual String scrollingStateTreeAsText() const;
129
130     // Generated a unique id for scroll layers.
131     ScrollingNodeID uniqueScrollLayerID();
132
133     // Dispatched by the scrolling tree whenever the main frame scroll position changes.
134     void scheduleUpdateMainFrameScrollPosition(const IntPoint&, bool programmaticScroll, SetOrSyncScrollingLayerPosition);
135     void updateMainFrameScrollPosition(const IntPoint&, bool programmaticScroll, SetOrSyncScrollingLayerPosition);
136
137     enum MainThreadScrollingReasonFlags {
138         ForcedOnMainThread = 1 << 0,
139         HasSlowRepaintObjects = 1 << 1,
140         HasViewportConstrainedObjectsWithoutSupportingFixedLayers = 1 << 2,
141         HasNonLayerViewportConstrainedObjects = 1 << 3,
142         IsImageDocument = 1 << 4
143     };
144
145     MainThreadScrollingReasons mainThreadScrollingReasons() const;
146     bool shouldUpdateScrollLayerPositionOnMainThread() const { return mainThreadScrollingReasons() != 0; }
147
148     // These virtual functions are currently unique to Chromium's WebLayer approach. Their meaningful
149     // implementations are in ScrollingCoordinatorChromium.
150     virtual void frameViewHorizontalScrollbarLayerDidChange(FrameView*, GraphicsLayer*) { }
151     virtual void frameViewVerticalScrollbarLayerDidChange(FrameView*, GraphicsLayer*) { }
152     virtual void scrollableAreaScrollLayerDidChange(ScrollableArea*, GraphicsLayer*) { }
153     virtual void setLayerIsContainerForFixedPositionLayers(GraphicsLayer*, bool) { }
154     virtual void setLayerIsFixedToContainerLayer(GraphicsLayer*, bool) { }
155     virtual void touchEventTargetRectsDidChange(const Document*) { }
156
157 #if ENABLE(TOUCH_EVENT_TRACKING)
158     void computeAbsoluteTouchEventTargetRects(const Document*, Vector<IntRect>&);
159 #endif
160
161     static String mainThreadScrollingReasonsAsText(MainThreadScrollingReasons);
162     String mainThreadScrollingReasonsAsText() const;
163
164     Region computeNonFastScrollableRegion(const Frame*, const IntPoint& frameLocation) const;
165
166 protected:
167     explicit ScrollingCoordinator(Page*);
168
169     unsigned computeCurrentWheelEventHandlerCount();
170     GraphicsLayer* scrollLayerForFrameView(FrameView*);
171     GraphicsLayer* counterScrollingLayerForFrameView(FrameView*);
172
173     Page* m_page;
174
175 private:
176     virtual void recomputeWheelEventHandlerCountForFrameView(FrameView*) { }
177     virtual void setShouldUpdateScrollLayerPositionOnMainThread(MainThreadScrollingReasons) { }
178
179     virtual bool hasVisibleSlowRepaintViewportConstrainedObjects(FrameView*) const;
180     void updateShouldUpdateScrollLayerPositionOnMainThread();
181     
182     void updateMainFrameScrollPositionTimerFired(Timer<ScrollingCoordinator>*);
183
184     Timer<ScrollingCoordinator> m_updateMainFrameScrollPositionTimer;
185     IntPoint m_scheduledUpdateScrollPosition;
186     bool m_scheduledUpdateIsProgrammaticScroll;
187     SetOrSyncScrollingLayerPosition m_scheduledScrollingLayerPositionAction;
188
189     bool m_forceMainThreadScrollLayerPositionUpdates;
190 };
191
192 } // namespace WebCore
193
194 #endif // ScrollingCoordinator_h