| OLD | NEW |
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "config.h" | 5 #include "config.h" |
| 6 #include "core/paint/ViewPainter.h" | 6 #include "core/paint/ViewPainter.h" |
| 7 | 7 |
| 8 #include "core/frame/FrameView.h" | 8 #include "core/frame/FrameView.h" |
| 9 #include "core/frame/Settings.h" | 9 #include "core/frame/Settings.h" |
| 10 #include "core/layout/LayoutBox.h" | 10 #include "core/layout/LayoutBox.h" |
| (...skipping 38 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 49 // canvas. None of the stacking context effects (except transformation) o
n the root element | 49 // canvas. None of the stacking context effects (except transformation) o
n the root element |
| 50 // affects the background. | 50 // affects the background. |
| 51 // 3. The main frame is also responsible for painting the user-agent-defined
base background | 51 // 3. The main frame is also responsible for painting the user-agent-defined
base background |
| 52 // color. Conceptually it should be painted by the embedder but painting
it here allows | 52 // color. Conceptually it should be painted by the embedder but painting
it here allows |
| 53 // culling and pre-blending optimization when possible. | 53 // culling and pre-blending optimization when possible. |
| 54 | 54 |
| 55 GraphicsContext& context = paintInfo.context; | 55 GraphicsContext& context = paintInfo.context; |
| 56 if (LayoutObjectDrawingRecorder::useCachedDrawingIfPossible(context, m_layou
tView, DisplayItem::BoxDecorationBackground, LayoutPoint())) | 56 if (LayoutObjectDrawingRecorder::useCachedDrawingIfPossible(context, m_layou
tView, DisplayItem::BoxDecorationBackground, LayoutPoint())) |
| 57 return; | 57 return; |
| 58 | 58 |
| 59 IntRect documentRect = m_layoutView.documentRect(); | 59 // When the view uses composited scrolling, the main GraphicsLayer (where th
e background is painted) |
| 60 // is sized to the layout viewport. Otherwise, it's the size of the document
. |
| 61 IntRect backgroundRect = m_layoutView.layer()->needsCompositedScrolling() ? |
| 62 pixelSnappedIntRect(m_layoutView.viewRect()) : m_layoutView.documentRect
(); |
| 63 |
| 60 const Document& document = m_layoutView.document(); | 64 const Document& document = m_layoutView.document(); |
| 61 const FrameView& frameView = *m_layoutView.frameView(); | 65 const FrameView& frameView = *m_layoutView.frameView(); |
| 62 bool isMainFrame = !document.ownerElement(); | 66 bool isMainFrame = !document.ownerElement(); |
| 63 bool paintsBaseBackground = isMainFrame && !frameView.isTransparent(); | 67 bool paintsBaseBackground = isMainFrame && !frameView.isTransparent(); |
| 64 bool shouldClearCanvas = paintsBaseBackground && (document.settings() && doc
ument.settings()->shouldClearDocumentBackground()); | 68 bool shouldClearCanvas = paintsBaseBackground && (document.settings() && doc
ument.settings()->shouldClearDocumentBackground()); |
| 65 Color baseBackgroundColor = paintsBaseBackground ? frameView.baseBackgroundC
olor() : Color(); | 69 Color baseBackgroundColor = paintsBaseBackground ? frameView.baseBackgroundC
olor() : Color(); |
| 66 Color rootBackgroundColor = m_layoutView.style()->visitedDependentColor(CSSP
ropertyBackgroundColor); | 70 Color rootBackgroundColor = m_layoutView.style()->visitedDependentColor(CSSP
ropertyBackgroundColor); |
| 67 const LayoutObject* rootObject = document.documentElement() ? document.docum
entElement()->layoutObject() : nullptr; | 71 const LayoutObject* rootObject = document.documentElement() ? document.docum
entElement()->layoutObject() : nullptr; |
| 68 | 72 |
| 69 LayoutObjectDrawingRecorder recorder(context, m_layoutView, DisplayItem::Box
DecorationBackground, documentRect, LayoutPoint()); | 73 LayoutObjectDrawingRecorder recorder(context, m_layoutView, DisplayItem::Box
DecorationBackground, backgroundRect, LayoutPoint()); |
| 70 | 74 |
| 71 // Special handling for print economy mode. | 75 // Special handling for print economy mode. |
| 72 bool forceBackgroundToWhite = BoxPainter::shouldForceWhiteBackgroundForPrint
Economy(m_layoutView.styleRef(), document); | 76 bool forceBackgroundToWhite = BoxPainter::shouldForceWhiteBackgroundForPrint
Economy(m_layoutView.styleRef(), document); |
| 73 if (forceBackgroundToWhite) { | 77 if (forceBackgroundToWhite) { |
| 74 // If for any reason the view background is not transparent, paint white
instead, otherwise keep transparent as is. | 78 // If for any reason the view background is not transparent, paint white
instead, otherwise keep transparent as is. |
| 75 if (paintsBaseBackground || rootBackgroundColor.alpha() || m_layoutView.
style()->backgroundLayers().image()) | 79 if (paintsBaseBackground || rootBackgroundColor.alpha() || m_layoutView.
style()->backgroundLayers().image()) |
| 76 context.fillRect(documentRect, Color::white, SkXfermode::kSrc_Mode); | 80 context.fillRect(backgroundRect, Color::white, SkXfermode::kSrc_Mode
); |
| 77 return; | 81 return; |
| 78 } | 82 } |
| 79 | 83 |
| 80 // Compute the enclosing rect of the view, in root element space. | 84 // Compute the enclosing rect of the view, in root element space. |
| 81 // | 85 // |
| 82 // For background colors we can simply paint the document rect in the defaul
t space. | 86 // For background colors we can simply paint the document rect in the defaul
t space. |
| 83 // However for background image, the root element transform applies. The str
ategy is to apply | 87 // However for background image, the root element transform applies. The str
ategy is to apply |
| 84 // root element transform on the context and issue draw commands in the loca
l space, therefore | 88 // root element transform on the context and issue draw commands in the loca
l space, therefore |
| 85 // we need to apply inverse transform on the document rect to get to the roo
t element space. | 89 // we need to apply inverse transform on the document rect to get to the roo
t element space. |
| 86 bool backgroundRenderable = true; | 90 bool backgroundRenderable = true; |
| 87 TransformationMatrix transform; | 91 TransformationMatrix transform; |
| 88 IntRect paintRect = documentRect; | 92 IntRect paintRect = backgroundRect; |
| 89 if (!rootObject || !rootObject->isBox()) { | 93 if (!rootObject || !rootObject->isBox()) { |
| 90 backgroundRenderable = false; | 94 backgroundRenderable = false; |
| 91 } else if (rootObject->hasLayer()) { | 95 } else if (rootObject->hasLayer()) { |
| 92 const PaintLayer& rootLayer = *toLayoutBoxModelObject(rootObject)->layer
(); | 96 const PaintLayer& rootLayer = *toLayoutBoxModelObject(rootObject)->layer
(); |
| 93 LayoutPoint offset; | 97 LayoutPoint offset; |
| 94 rootLayer.convertToLayerCoords(nullptr, offset); | 98 rootLayer.convertToLayerCoords(nullptr, offset); |
| 95 transform.translate(offset.x(), offset.y()); | 99 transform.translate(offset.x(), offset.y()); |
| 96 transform.multiply(rootLayer.renderableTransform(paintInfo.globalPaintFl
ags())); | 100 transform.multiply(rootLayer.renderableTransform(paintInfo.globalPaintFl
ags())); |
| 97 | 101 |
| 98 if (!transform.isInvertible()) { | 102 if (!transform.isInvertible()) { |
| 99 backgroundRenderable = false; | 103 backgroundRenderable = false; |
| 100 } else { | 104 } else { |
| 101 bool isClamped; | 105 bool isClamped; |
| 102 paintRect = transform.inverse().projectQuad(FloatQuad(documentRect),
&isClamped).enclosingBoundingBox(); | 106 paintRect = transform.inverse().projectQuad(FloatQuad(backgroundRect
), &isClamped).enclosingBoundingBox(); |
| 103 backgroundRenderable = !isClamped; | 107 backgroundRenderable = !isClamped; |
| 104 } | 108 } |
| 105 } | 109 } |
| 106 | 110 |
| 107 if (!backgroundRenderable) { | 111 if (!backgroundRenderable) { |
| 108 if (baseBackgroundColor.alpha()) | 112 if (baseBackgroundColor.alpha()) |
| 109 context.fillRect(documentRect, baseBackgroundColor, shouldClearCanva
s ? SkXfermode::kSrc_Mode : SkXfermode::kSrcOver_Mode); | 113 context.fillRect(backgroundRect, baseBackgroundColor, shouldClearCan
vas ? SkXfermode::kSrc_Mode : SkXfermode::kSrcOver_Mode); |
| 110 else if (shouldClearCanvas) | 114 else if (shouldClearCanvas) |
| 111 context.fillRect(documentRect, Color(), SkXfermode::kClear_Mode); | 115 context.fillRect(backgroundRect, Color(), SkXfermode::kClear_Mode); |
| 112 return; | 116 return; |
| 113 } | 117 } |
| 114 | 118 |
| 115 BoxPainter::FillLayerOcclusionOutputList reversedPaintList; | 119 BoxPainter::FillLayerOcclusionOutputList reversedPaintList; |
| 116 bool shouldDrawBackgroundInSeparateBuffer = BoxPainter(m_layoutView).calcula
teFillLayerOcclusionCulling(reversedPaintList, m_layoutView.style()->backgroundL
ayers()); | 120 bool shouldDrawBackgroundInSeparateBuffer = BoxPainter(m_layoutView).calcula
teFillLayerOcclusionCulling(reversedPaintList, m_layoutView.style()->backgroundL
ayers()); |
| 117 ASSERT(reversedPaintList.size()); | 121 ASSERT(reversedPaintList.size()); |
| 118 | 122 |
| 119 // If the root background color is opaque, isolation group can be skipped be
cause the canvas | 123 // If the root background color is opaque, isolation group can be skipped be
cause the canvas |
| 120 // will be cleared by root background color. | 124 // will be cleared by root background color. |
| 121 if (!rootBackgroundColor.hasAlpha()) | 125 if (!rootBackgroundColor.hasAlpha()) |
| 122 shouldDrawBackgroundInSeparateBuffer = false; | 126 shouldDrawBackgroundInSeparateBuffer = false; |
| 123 | 127 |
| 124 // We are going to clear the canvas with transparent pixels, isolation group
can be skipped. | 128 // We are going to clear the canvas with transparent pixels, isolation group
can be skipped. |
| 125 if (!baseBackgroundColor.alpha() && shouldClearCanvas) | 129 if (!baseBackgroundColor.alpha() && shouldClearCanvas) |
| 126 shouldDrawBackgroundInSeparateBuffer = false; | 130 shouldDrawBackgroundInSeparateBuffer = false; |
| 127 | 131 |
| 128 if (shouldDrawBackgroundInSeparateBuffer) { | 132 if (shouldDrawBackgroundInSeparateBuffer) { |
| 129 if (baseBackgroundColor.alpha()) | 133 if (baseBackgroundColor.alpha()) |
| 130 context.fillRect(documentRect, baseBackgroundColor, shouldClearCanva
s ? SkXfermode::kSrc_Mode : SkXfermode::kSrcOver_Mode); | 134 context.fillRect(backgroundRect, baseBackgroundColor, shouldClearCan
vas ? SkXfermode::kSrc_Mode : SkXfermode::kSrcOver_Mode); |
| 131 context.beginLayer(); | 135 context.beginLayer(); |
| 132 } | 136 } |
| 133 | 137 |
| 134 Color combinedBackgroundColor = shouldDrawBackgroundInSeparateBuffer ? rootB
ackgroundColor : baseBackgroundColor.blend(rootBackgroundColor); | 138 Color combinedBackgroundColor = shouldDrawBackgroundInSeparateBuffer ? rootB
ackgroundColor : baseBackgroundColor.blend(rootBackgroundColor); |
| 135 if (combinedBackgroundColor.alpha()) | 139 if (combinedBackgroundColor.alpha()) |
| 136 context.fillRect(documentRect, combinedBackgroundColor, (shouldDrawBackg
roundInSeparateBuffer || shouldClearCanvas) ? SkXfermode::kSrc_Mode : SkXfermode
::kSrcOver_Mode); | 140 context.fillRect(backgroundRect, combinedBackgroundColor, (shouldDrawBac
kgroundInSeparateBuffer || shouldClearCanvas) ? SkXfermode::kSrc_Mode : SkXfermo
de::kSrcOver_Mode); |
| 137 else if (shouldClearCanvas && !shouldDrawBackgroundInSeparateBuffer) | 141 else if (shouldClearCanvas && !shouldDrawBackgroundInSeparateBuffer) |
| 138 context.fillRect(documentRect, Color(), SkXfermode::kClear_Mode); | 142 context.fillRect(backgroundRect, Color(), SkXfermode::kClear_Mode); |
| 139 | 143 |
| 140 for (auto it = reversedPaintList.rbegin(); it != reversedPaintList.rend(); +
+it) { | 144 for (auto it = reversedPaintList.rbegin(); it != reversedPaintList.rend(); +
+it) { |
| 141 ASSERT((*it)->clip() == BorderFillBox); | 145 ASSERT((*it)->clip() == BorderFillBox); |
| 142 | 146 |
| 143 bool shouldPaintInViewportSpace = (*it)->attachment() == FixedBackground
Attachment; | 147 bool shouldPaintInViewportSpace = (*it)->attachment() == FixedBackground
Attachment; |
| 144 if (shouldPaintInViewportSpace) { | 148 if (shouldPaintInViewportSpace) { |
| 145 BoxPainter::paintFillLayerExtended(m_layoutView, paintInfo, Color(),
**it, LayoutRect(LayoutRect::infiniteIntRect()), BackgroundBleedNone); | 149 BoxPainter::paintFillLayerExtended(m_layoutView, paintInfo, Color(),
**it, LayoutRect(LayoutRect::infiniteIntRect()), BackgroundBleedNone); |
| 146 } else { | 150 } else { |
| 147 context.save(); | 151 context.save(); |
| 148 // TODO(trchen): We should be able to handle 3D-transformed root | 152 // TODO(trchen): We should be able to handle 3D-transformed root |
| 149 // background with slimming paint by using transform display items. | 153 // background with slimming paint by using transform display items. |
| 150 context.concatCTM(transform.toAffineTransform()); | 154 context.concatCTM(transform.toAffineTransform()); |
| 151 BoxPainter::paintFillLayerExtended(m_layoutView, paintInfo, Color(),
**it, LayoutRect(paintRect), BackgroundBleedNone); | 155 BoxPainter::paintFillLayerExtended(m_layoutView, paintInfo, Color(),
**it, LayoutRect(paintRect), BackgroundBleedNone); |
| 152 context.restore(); | 156 context.restore(); |
| 153 } | 157 } |
| 154 } | 158 } |
| 155 | 159 |
| 156 if (shouldDrawBackgroundInSeparateBuffer) | 160 if (shouldDrawBackgroundInSeparateBuffer) |
| 157 context.endLayer(); | 161 context.endLayer(); |
| 158 } | 162 } |
| 159 | 163 |
| 160 } // namespace blink | 164 } // namespace blink |
| OLD | NEW |