ec482265770d175efae8e9cd2db62d651453a27c
[WebKit-https.git] / Source / WebCore / platform / graphics / CustomPaintImage.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 "CustomPaintImage.h"
28
29 #if ENABLE(CSS_PAINTING_API)
30
31 #include "CSSComputedStyleDeclaration.h"
32 #include "CSSPrimitiveValue.h"
33 #include "CSSPropertyParser.h"
34 #include "CSSUnitValue.h"
35 #include "CSSUnparsedValue.h"
36 #include "CustomPaintCanvas.h"
37 #include "GraphicsContext.h"
38 #include "ImageBitmap.h"
39 #include "ImageBuffer.h"
40 #include "JSCSSPaintCallback.h"
41 #include "PaintRenderingContext2D.h"
42 #include "RenderElement.h"
43 #include <JavaScriptCore/ConstructData.h>
44
45 namespace WebCore {
46
47 CustomPaintImage::CustomPaintImage(PaintWorkletGlobalScope::PaintDefinition& definition, const FloatSize& size, RenderElement& element, const Vector<String>& arguments)
48     : m_paintDefinition(makeWeakPtr(definition))
49     , m_inputProperties(definition.inputProperties)
50     , m_element(makeWeakPtr(element))
51     , m_arguments(arguments)
52 {
53     setContainerSize(size);
54 }
55
56 CustomPaintImage::~CustomPaintImage() = default;
57
58 ImageDrawResult CustomPaintImage::doCustomPaint(GraphicsContext& destContext, const FloatSize& destSize)
59 {
60     if (!m_element || !m_element->element() || !m_paintDefinition)
61         return ImageDrawResult::DidNothing;
62
63     JSC::JSValue paintConstructor(m_paintDefinition->paintConstructor);
64
65     if (!paintConstructor)
66         return ImageDrawResult::DidNothing;
67
68     auto& paintCallback = m_paintDefinition->paintCallback.get();
69
70     ASSERT(!m_element->needsLayout());
71     ASSERT(!m_element->element()->document().needsStyleRecalc());
72
73     JSCSSPaintCallback& callback = static_cast<JSCSSPaintCallback&>(paintCallback);
74     auto* scriptExecutionContext = callback.scriptExecutionContext();
75     if (!scriptExecutionContext)
76         return ImageDrawResult::DidNothing;
77
78     auto canvas = CustomPaintCanvas::create(*scriptExecutionContext, destSize.width(), destSize.height());
79     ExceptionOr<RefPtr<PaintRenderingContext2D>> contextOrException = canvas->getContext();
80
81     if (contextOrException.hasException())
82         return ImageDrawResult::DidNothing;
83     auto context = contextOrException.releaseReturnValue();
84
85     HashMap<String, Ref<CSSStyleValue>> propertyValues;
86     ComputedStyleExtractor extractor(m_element->element());
87
88     for (auto& name : m_inputProperties) {
89         RefPtr<CSSValue> value;
90         if (isCustomPropertyName(name))
91             value = extractor.customPropertyValue(name);
92         else {
93             CSSPropertyID propertyID = cssPropertyID(name);
94             if (!propertyID)
95                 return ImageDrawResult::DidNothing;
96             value = extractor.propertyValue(propertyID, DoNotUpdateLayout);
97         }
98
99         if (!value) {
100             propertyValues.add(name, CSSUnparsedValue::create(emptyString()));
101             continue;
102         }
103
104         // FIXME: Properly reify all length values.
105         if (is<CSSPrimitiveValue>(*value) && downcast<CSSPrimitiveValue>(*value).primitiveType() == CSSPrimitiveValue::CSS_PX)
106             propertyValues.add(name, CSSUnitValue::create(downcast<CSSPrimitiveValue>(*value).doubleValue(), "px"));
107         else
108             propertyValues.add(name, CSSUnparsedValue::create(value->cssText()));
109     }
110
111     auto size = CSSPaintSize::create(destSize.width(), destSize.height());
112     auto propertyMap = StylePropertyMapReadOnly::create(WTFMove(propertyValues));
113
114     auto& vm = *paintConstructor.getObject()->vm();
115     JSC::JSLockHolder lock(vm);
116     auto scope = DECLARE_THROW_SCOPE(vm);
117     auto& globalObject = *paintConstructor.getObject()->globalObject();
118
119     auto& state = *globalObject.globalExec();
120     JSC::ArgList noArgs;
121     JSC::JSValue thisObject = { JSC::construct(&state, WTFMove(paintConstructor), noArgs, "Failed to construct paint class") };
122
123     if (UNLIKELY(scope.exception())) {
124         reportException(&state, scope.exception());
125         return ImageDrawResult::DidNothing;
126     }
127
128     auto result = paintCallback.handleEvent(WTFMove(thisObject), *context, size, propertyMap, m_arguments);
129     if (result.type() != CallbackResultType::Success)
130         return ImageDrawResult::DidNothing;
131
132     canvas->replayDisplayList(&destContext);
133
134     return ImageDrawResult::DidDraw;
135 }
136
137 ImageDrawResult CustomPaintImage::draw(GraphicsContext& destContext, const FloatRect& destRect, const FloatRect& srcRect, CompositeOperator compositeOp, BlendMode blendMode, DecodingMode, ImageOrientationDescription)
138 {
139     GraphicsContextStateSaver stateSaver(destContext);
140     destContext.setCompositeOperation(compositeOp, blendMode);
141     destContext.clip(destRect);
142     destContext.translate(destRect.location());
143     if (destRect.size() != srcRect.size())
144         destContext.scale(destRect.size() / srcRect.size());
145     destContext.translate(-srcRect.location());
146     return doCustomPaint(destContext, size());
147 }
148
149 void CustomPaintImage::drawPattern(GraphicsContext& destContext, const FloatRect& destRect, const FloatRect& srcRect, const AffineTransform& patternTransform,
150     const FloatPoint& phase, const FloatSize& spacing, CompositeOperator compositeOp, BlendMode blendMode)
151 {
152     // Allow the generator to provide visually-equivalent tiling parameters for better performance.
153     FloatSize adjustedSize = size();
154     FloatRect adjustedSrcRect = srcRect;
155
156     // Factor in the destination context's scale to generate at the best resolution
157     AffineTransform destContextCTM = destContext.getCTM(GraphicsContext::DefinitelyIncludeDeviceScale);
158     double xScale = fabs(destContextCTM.xScale());
159     double yScale = fabs(destContextCTM.yScale());
160     AffineTransform adjustedPatternCTM = patternTransform;
161     adjustedPatternCTM.scale(1.0 / xScale, 1.0 / yScale);
162     adjustedSrcRect.scale(xScale, yScale);
163
164     auto buffer = ImageBuffer::createCompatibleBuffer(adjustedSize, ColorSpaceSRGB, destContext);
165     if (!buffer)
166         return;
167     doCustomPaint(buffer->context(), adjustedSize);
168
169     if (destContext.drawLuminanceMask())
170         buffer->convertToLuminanceMask();
171
172     buffer->drawPattern(destContext, destRect, adjustedSrcRect, adjustedPatternCTM, phase, spacing, compositeOp, blendMode);
173     destContext.setDrawLuminanceMask(false);
174 }
175
176 }
177 #endif