Avoid repaints when changing transform on an element with multiple background images
[WebKit-https.git] / Source / WebCore / rendering / style / FillLayer.h
1 /*
2  * Copyright (C) 2000 Lars Knoll (knoll@kde.org)
3  *           (C) 2000 Antti Koivisto (koivisto@kde.org)
4  *           (C) 2000 Dirk Mueller (mueller@kde.org)
5  * Copyright (C) 2003, 2005, 2006, 2007, 2008, 2014 Apple Inc. All rights reserved.
6  * Copyright (C) 2006 Graham Dennis (graham.dennis@gmail.com)
7  *
8  * This library is free software; you can redistribute it and/or
9  * modify it under the terms of the GNU Library General Public
10  * License as published by the Free Software Foundation; either
11  * version 2 of the License, or (at your option) any later version.
12  *
13  * This library is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
16  * Library General Public License for more details.
17  *
18  * You should have received a copy of the GNU Library General Public License
19  * along with this library; see the file COPYING.LIB.  If not, write to
20  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
21  * Boston, MA 02110-1301, USA.
22  *
23  */
24
25 #ifndef FillLayer_h
26 #define FillLayer_h
27
28 #include "GraphicsTypes.h"
29 #include "LengthSize.h"
30 #include "MaskImageOperation.h"
31 #include "RenderStyleConstants.h"
32 #include "StyleImage.h"
33 #include <wtf/RefPtr.h>
34
35 namespace WebCore {
36
37 class RenderElement;
38
39 struct FillSize {
40     FillSize()
41         : type(SizeLength)
42     {
43     }
44
45     FillSize(EFillSizeType type, const LengthSize& size)
46         : type(type)
47         , size(size)
48     {
49     }
50
51     EFillSizeType type;
52     LengthSize size;
53 };
54
55 inline bool operator==(const FillSize& a, const FillSize& b)
56 {
57     return a.type == b.type && a.size == b.size;
58 }
59
60 inline bool operator!=(const FillSize& a, const FillSize& b)
61 {
62     return !(a == b);
63 }
64
65 class FillLayer {
66     WTF_MAKE_FAST_ALLOCATED;
67 public:
68     explicit FillLayer(EFillLayerType);
69     ~FillLayer();
70
71     StyleImage* image() const { return hasMaskImage() ? maskImage()->image() : m_image.get(); }
72     const RefPtr<MaskImageOperation>& maskImage() const { return m_maskImageOperation; }
73     const Length& xPosition() const { return m_xPosition; }
74     const Length& yPosition() const { return m_yPosition; }
75     BackgroundEdgeOrigin backgroundXOrigin() const { return static_cast<BackgroundEdgeOrigin>(m_backgroundXOrigin); }
76     BackgroundEdgeOrigin backgroundYOrigin() const { return static_cast<BackgroundEdgeOrigin>(m_backgroundYOrigin); }
77     EFillAttachment attachment() const { return static_cast<EFillAttachment>(m_attachment); }
78     EFillBox clip() const { return static_cast<EFillBox>(m_clip); }
79     EFillBox origin() const { return static_cast<EFillBox>(m_origin); }
80     EFillRepeat repeatX() const { return static_cast<EFillRepeat>(m_repeatX); }
81     EFillRepeat repeatY() const { return static_cast<EFillRepeat>(m_repeatY); }
82     CompositeOperator composite() const { return static_cast<CompositeOperator>(m_composite); }
83     BlendMode blendMode() const { return static_cast<BlendMode>(m_blendMode); }
84     const LengthSize& sizeLength() const { return m_sizeLength; }
85     EFillSizeType sizeType() const { return static_cast<EFillSizeType>(m_sizeType); }
86     FillSize size() const { return FillSize(static_cast<EFillSizeType>(m_sizeType), m_sizeLength); }
87     EMaskSourceType maskSourceType() const { return static_cast<EMaskSourceType>(m_maskSourceType); }
88
89     const FillLayer* next() const { return m_next.get(); }
90     FillLayer* next() { return m_next.get(); }
91
92     bool isImageSet() const { return m_imageSet; }
93     bool isXPositionSet() const { return m_xPosSet; }
94     bool isYPositionSet() const { return m_yPosSet; }
95     bool isBackgroundOriginSet() const { return m_backgroundOriginSet; }
96     bool isAttachmentSet() const { return m_attachmentSet; }
97     bool isClipSet() const { return m_clipSet; }
98     bool isOriginSet() const { return m_originSet; }
99     bool isRepeatXSet() const { return m_repeatXSet; }
100     bool isRepeatYSet() const { return m_repeatYSet; }
101     bool isCompositeSet() const { return m_compositeSet; }
102     bool isBlendModeSet() const { return m_blendModeSet; }
103     bool isSizeSet() const { return m_sizeType != SizeNone; }
104     bool isMaskSourceTypeSet() const { return m_maskSourceTypeSet; }
105
106     void setMaskImage(PassRefPtr<MaskImageOperation> maskImage) { m_maskImageOperation = maskImage; }
107     void setImage(PassRefPtr<StyleImage> image) { m_image = image; m_imageSet = true; }
108     void setXPosition(Length length) { m_xPosition = WTF::move(length); m_xPosSet = true; }
109     void setYPosition(Length length) { m_yPosition = WTF::move(length); m_yPosSet = true; }
110     void setBackgroundXOrigin(BackgroundEdgeOrigin o) { m_backgroundXOrigin = o; m_backgroundOriginSet = true; }
111     void setBackgroundYOrigin(BackgroundEdgeOrigin o) { m_backgroundYOrigin = o; m_backgroundOriginSet = true; }
112     void setAttachment(EFillAttachment attachment) { m_attachment = attachment; m_attachmentSet = true; }
113     void setClip(EFillBox b) { m_clip = b; m_clipSet = true; }
114     void setOrigin(EFillBox b) { m_origin = b; m_originSet = true; }
115     void setRepeatX(EFillRepeat r) { m_repeatX = r; m_repeatXSet = true; }
116     void setRepeatY(EFillRepeat r) { m_repeatY = r; m_repeatYSet = true; }
117     void setComposite(CompositeOperator c) { m_composite = c; m_compositeSet = true; }
118     void setBlendMode(BlendMode b) { m_blendMode = b; m_blendModeSet = true; }
119     void setSizeType(EFillSizeType b) { m_sizeType = b; }
120     void setSizeLength(LengthSize l) { m_sizeLength = l; }
121     void setSize(FillSize f) { m_sizeType = f.type; m_sizeLength = f.size; }
122     void setMaskSourceType(EMaskSourceType m) { m_maskSourceType = m; m_maskSourceTypeSet = true; }
123
124     void clearMaskImage() { m_maskImageOperation.clear(); }
125     void clearImage() { m_image.clear(); m_imageSet = false; }
126     void clearXPosition() { m_xPosSet = false; m_backgroundOriginSet = false; }
127     void clearYPosition() { m_yPosSet = false; m_backgroundOriginSet = false; }
128
129     void clearAttachment() { m_attachmentSet = false; }
130     void clearClip() { m_clipSet = false; }
131     void clearOrigin() { m_originSet = false; }
132     void clearRepeatX() { m_repeatXSet = false; }
133     void clearRepeatY() { m_repeatYSet = false; }
134     void clearComposite() { m_compositeSet = false; }
135     void clearBlendMode() { m_blendModeSet = false; }
136     void clearSize() { m_sizeType = SizeNone; }
137     void clearMaskSourceType() { m_maskSourceTypeSet = false; }
138
139     void setNext(std::unique_ptr<FillLayer> next) { m_next = WTF::move(next); }
140
141     FillLayer& operator=(const FillLayer&);
142     FillLayer(const FillLayer&);
143
144     bool operator==(const FillLayer&) const;
145     bool operator!=(const FillLayer& other) const { return !(*this == other); }
146
147     bool containsImage(StyleImage&) const;
148     bool imagesAreLoaded() const;
149     bool hasImage() const;
150     bool hasMaskImage() const { return maskImage().get(); }
151     bool hasNonEmptyMaskImage() const;
152     bool hasFixedImage() const;
153     bool hasOpaqueImage(const RenderElement&) const;
154     bool hasRepeatXY() const;
155     bool clipOccludesNextLayers(bool firstLayer) const;
156
157     EFillLayerType type() const { return static_cast<EFillLayerType>(m_type); }
158
159     void fillUnsetProperties();
160     void cullEmptyLayers();
161
162     static bool imagesIdentical(const FillLayer*, const FillLayer*);
163
164     static EFillAttachment initialFillAttachment(EFillLayerType) { return ScrollBackgroundAttachment; }
165     static EFillBox initialFillClip(EFillLayerType) { return BorderFillBox; }
166     static EFillBox initialFillOrigin(EFillLayerType type) { return type == BackgroundFillLayer ? PaddingFillBox : BorderFillBox; }
167     static EFillRepeat initialFillRepeatX(EFillLayerType) { return RepeatFill; }
168     static EFillRepeat initialFillRepeatY(EFillLayerType) { return RepeatFill; }
169     static CompositeOperator initialFillComposite(EFillLayerType) { return CompositeSourceOver; }
170     static BlendMode initialFillBlendMode(EFillLayerType) { return BlendModeNormal; }
171     static FillSize initialFillSize(EFillLayerType) { return FillSize(); }
172     static Length initialFillXPosition(EFillLayerType) { return Length(0.0f, Percent); }
173     static Length initialFillYPosition(EFillLayerType) { return Length(0.0f, Percent); }
174     static StyleImage* initialFillImage(EFillLayerType) { return nullptr; }
175     static EMaskSourceType initialFillMaskSourceType(EFillLayerType) { return MaskAlpha; }
176
177 private:
178     friend class RenderStyle;
179
180     void computeClipMax() const;
181
182     std::unique_ptr<FillLayer> m_next;
183
184     // FIXME: A FillLayer will always have at least one of these pointers null.
185     // Maybe we could group them together somehow and decrease the size of FillLayer.
186     RefPtr<MaskImageOperation> m_maskImageOperation;
187     RefPtr<StyleImage> m_image;
188
189     Length m_xPosition;
190     Length m_yPosition;
191
192     LengthSize m_sizeLength;
193
194     unsigned m_attachment : 2; // EFillAttachment
195     unsigned m_clip : 2; // EFillBox
196     unsigned m_origin : 2; // EFillBox
197     unsigned m_repeatX : 3; // EFillRepeat
198     unsigned m_repeatY : 3; // EFillRepeat
199     unsigned m_composite : 4; // CompositeOperator
200     unsigned m_sizeType : 2; // EFillSizeType
201     unsigned m_blendMode : 5; // BlendMode
202     unsigned m_maskSourceType : 1; // EMaskSourceType
203
204     unsigned m_imageSet : 1;
205     unsigned m_attachmentSet : 1;
206     unsigned m_clipSet : 1;
207     unsigned m_originSet : 1;
208     unsigned m_repeatXSet : 1;
209     unsigned m_repeatYSet : 1;
210     unsigned m_xPosSet : 1;
211     unsigned m_yPosSet : 1;
212     unsigned m_backgroundOriginSet : 1;
213     unsigned m_backgroundXOrigin : 2; // BackgroundEdgeOrigin
214     unsigned m_backgroundYOrigin : 2; // BackgroundEdgeOrigin
215     unsigned m_compositeSet : 1;
216     unsigned m_blendModeSet : 1;
217     unsigned m_maskSourceTypeSet : 1;
218
219     unsigned m_type : 1; // EFillLayerType
220
221     mutable unsigned m_clipMax : 2; // EFillBox, maximum m_clip value from this to bottom layer
222 };
223
224 } // namespace WebCore
225
226 #endif // FillLayer_h