658f8cbc56e4b8a446285709f41794952f2b3d29
[WebKit-https.git] / Source / WebCore / layout / LayoutContext.cpp
1 /*
2  * Copyright (C) 2018 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 #include "config.h"
27 #include "LayoutContext.h"
28
29 #if ENABLE(LAYOUT_FORMATTING_CONTEXT)
30
31 #include "BlockFormattingContext.h"
32 #include "BlockFormattingState.h"
33 #include "BlockInvalidation.h"
34 #include "DisplayBox.h"
35 #include "InlineFormattingContext.h"
36 #include "InlineFormattingState.h"
37 #include "InlineInvalidation.h"
38 #include "Invalidation.h"
39 #include "LayoutBox.h"
40 #include "LayoutContainer.h"
41 #include <wtf/IsoMallocInlines.h>
42
43 namespace WebCore {
44 namespace Layout {
45
46 WTF_MAKE_ISO_ALLOCATED_IMPL(LayoutContext);
47
48 LayoutContext::LayoutContext(const Box& root)
49     : m_root(makeWeakPtr(const_cast<Box&>(root)))
50 {
51 }
52
53 void LayoutContext::updateLayout()
54 {
55     ASSERT(!m_formattingContextRootListForLayout.isEmpty());
56     for (auto* layoutRoot : m_formattingContextRootListForLayout) {
57         RELEASE_ASSERT(layoutRoot->establishesFormattingContext());
58         auto context = formattingContext(*layoutRoot);
59         auto& state = establishedFormattingState(*layoutRoot, *context);
60         context->layout(*this, state);
61     }
62     m_formattingContextRootListForLayout.clear();
63 }
64
65 Display::Box& LayoutContext::createDisplayBox(const Box& layoutBox)
66 {
67     std::unique_ptr<Display::Box> displayBox(new Display::Box());
68     auto* displayBoxPtr = displayBox.get();
69     m_layoutToDisplayBox.add(&layoutBox, WTFMove(displayBox));
70     return *displayBoxPtr;
71 }
72
73 void LayoutContext::styleChanged(const Box& layoutBox, StyleDiff styleDiff)
74 {
75     auto& formattingState = formattingStateForBox(layoutBox);
76     const Container* invalidationRoot = nullptr;
77     if (is<BlockFormattingState>(formattingState))
78         invalidationRoot = BlockInvalidation::invalidate(layoutBox, styleDiff, *this, downcast<BlockFormattingState>(formattingState)).root;
79     else if (is<InlineFormattingState>(formattingState))
80         invalidationRoot = InlineInvalidation::invalidate(layoutBox, styleDiff, *this, downcast<InlineFormattingState>(formattingState)).root;
81     else
82         ASSERT_NOT_REACHED();
83     ASSERT(invalidationRoot);
84     m_formattingContextRootListForLayout.addVoid(invalidationRoot);
85 }
86
87 void LayoutContext::markNeedsUpdate(const Box&, OptionSet<UpdateType>)
88 {
89 }
90
91 FormattingState& LayoutContext::formattingStateForBox(const Box& layoutBox) const
92 {
93     auto& root = layoutBox.formattingContextRoot();
94     RELEASE_ASSERT(m_formattingStates.contains(&root));
95     return *m_formattingStates.get(&root);
96 }
97
98 FormattingState& LayoutContext::establishedFormattingState(const Box& formattingContextRoot, const FormattingContext& context)
99 {
100     return *m_formattingStates.ensure(&formattingContextRoot, [this, &context] {
101         return context.createFormattingState(context.createOrFindFloatingState(*this));
102     }).iterator->value;
103 }
104
105 std::unique_ptr<FormattingContext> LayoutContext::formattingContext(const Box& formattingContextRoot)
106 {
107     if (formattingContextRoot.establishesBlockFormattingContext())
108         return std::make_unique<BlockFormattingContext>(formattingContextRoot);
109
110     if (formattingContextRoot.establishesInlineFormattingContext())
111         return std::make_unique<InlineFormattingContext>(formattingContextRoot);
112
113     ASSERT_NOT_REACHED();
114     return nullptr;
115 }
116
117 }
118 }
119
120 #endif