Use NeverDestroyed instead of DEPRECATED_DEFINE_STATIC_LOCAL
[WebKit-https.git] / Source / WebCore / rendering / shapes / ShapeOutsideInfo.h
1 /*
2  * Copyright (C) 2012 Adobe Systems Incorporated. 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  *
8  * 1. Redistributions of source code must retain the above
9  *    copyright notice, this list of conditions and the following
10  *    disclaimer.
11  * 2. Redistributions in binary form must reproduce the above
12  *    copyright notice, this list of conditions and the following
13  *    disclaimer in the documentation and/or other materials
14  *    provided with the distribution.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER “AS IS” AND ANY
17  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE
20  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,
21  * OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
22  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
23  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
25  * TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
26  * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27  * SUCH DAMAGE.
28  */
29
30 #ifndef ShapeOutsideInfo_h
31 #define ShapeOutsideInfo_h
32
33 #if ENABLE(CSS_SHAPES)
34
35 #include "LayoutSize.h"
36 #include "Shape.h"
37 #include <wtf/HashMap.h>
38 #include <wtf/NeverDestroyed.h>
39
40 namespace WebCore {
41
42 class RenderBlockFlow;
43 class RenderBox;
44 class StyleImage;
45 class FloatingObject;
46
47 class ShapeOutsideDeltas final {
48 public:
49     ShapeOutsideDeltas()
50         : m_lineOverlapsShape(false)
51         , m_isValid(false)
52     {
53     }
54
55     ShapeOutsideDeltas(LayoutUnit leftMarginBoxDelta, LayoutUnit rightMarginBoxDelta, bool lineOverlapsShape, LayoutUnit borderBoxLineTop, LayoutUnit lineHeight)
56         : m_leftMarginBoxDelta(leftMarginBoxDelta)
57         , m_rightMarginBoxDelta(rightMarginBoxDelta)
58         , m_borderBoxLineTop(borderBoxLineTop)
59         , m_lineHeight(lineHeight)
60         , m_lineOverlapsShape(lineOverlapsShape)
61         , m_isValid(true)
62     {
63     }
64
65     bool isForLine(LayoutUnit borderBoxLineTop, LayoutUnit lineHeight)
66     {
67         return m_isValid && m_borderBoxLineTop == borderBoxLineTop && m_lineHeight == lineHeight;
68     }
69
70     bool isValid() { return m_isValid; }
71     LayoutUnit leftMarginBoxDelta() { ASSERT(m_isValid); return m_leftMarginBoxDelta; }
72     LayoutUnit rightMarginBoxDelta() { ASSERT(m_isValid); return m_rightMarginBoxDelta; }
73     bool lineOverlapsShape() { ASSERT(m_isValid); return m_lineOverlapsShape; }
74
75 private:
76     LayoutUnit m_leftMarginBoxDelta;
77     LayoutUnit m_rightMarginBoxDelta;
78     LayoutUnit m_borderBoxLineTop;
79     LayoutUnit m_lineHeight;
80     unsigned m_lineOverlapsShape : 1;
81     unsigned m_isValid : 1;
82 };
83
84 class ShapeOutsideInfo final {
85     WTF_MAKE_FAST_ALLOCATED;
86 public:
87     ShapeOutsideInfo(const RenderBox& renderer)
88         : m_renderer(renderer)
89     {
90     }
91
92     static bool isEnabledFor(const RenderBox&);
93
94     ShapeOutsideDeltas computeDeltasForContainingBlockLine(const RenderBlockFlow&, const FloatingObject&, LayoutUnit lineTop, LayoutUnit lineHeight);
95
96     void setReferenceBoxLogicalSize(LayoutSize);
97
98     LayoutUnit shapeLogicalTop() const { return computedShape().shapeMarginLogicalBoundingBox().y() + logicalTopOffset(); }
99     LayoutUnit shapeLogicalBottom() const { return computedShape().shapeMarginLogicalBoundingBox().maxY() + logicalTopOffset(); }
100     LayoutUnit shapeLogicalLeft() const { return computedShape().shapeMarginLogicalBoundingBox().x() + logicalLeftOffset(); }
101     LayoutUnit shapeLogicalRight() const { return computedShape().shapeMarginLogicalBoundingBox().maxX() + logicalLeftOffset(); }
102     LayoutUnit shapeLogicalWidth() const { return computedShape().shapeMarginLogicalBoundingBox().width(); }
103     LayoutUnit shapeLogicalHeight() const { return computedShape().shapeMarginLogicalBoundingBox().height(); }
104
105     void markShapeAsDirty() { m_shape = nullptr; }
106     bool isShapeDirty() { return !m_shape; }
107
108     LayoutRect computedShapePhysicalBoundingBox() const;
109     FloatPoint shapeToRendererPoint(const FloatPoint&) const;
110     FloatSize shapeToRendererSize(const FloatSize&) const;
111
112     const Shape& computedShape() const;
113
114     static ShapeOutsideInfo& ensureInfo(const RenderBox& key)
115     {
116         InfoMap& infoMap = ShapeOutsideInfo::infoMap();
117         if (ShapeOutsideInfo* info = infoMap.get(&key))
118             return *info;
119         auto result = infoMap.add(&key, std::make_unique<ShapeOutsideInfo>(key));
120         return *result.iterator->value;
121     }
122     static void removeInfo(const RenderBox& key) { infoMap().remove(&key); }
123     static ShapeOutsideInfo* info(const RenderBox& key) { return infoMap().get(&key); }
124
125 private:
126     std::unique_ptr<Shape> createShapeForImage(StyleImage*, float shapeImageThreshold, WritingMode, float margin) const;
127
128     LayoutUnit logicalTopOffset() const;
129     LayoutUnit logicalLeftOffset() const;
130
131     typedef HashMap<const RenderBox*, std::unique_ptr<ShapeOutsideInfo>> InfoMap;
132     static InfoMap& infoMap()
133     {
134         static NeverDestroyed<InfoMap> staticInfoMap;
135         return staticInfoMap;
136     }
137
138     const RenderBox& m_renderer;
139
140     mutable std::unique_ptr<Shape> m_shape;
141     LayoutSize m_referenceBoxLogicalSize;
142
143     ShapeOutsideDeltas m_shapeOutsideDeltas;
144 };
145
146 }
147 #endif
148 #endif