Index: third_party/WebKit/Source/modules/csspaint/CSSPaintDefinition.cpp |
diff --git a/third_party/WebKit/Source/modules/csspaint/CSSPaintDefinition.cpp b/third_party/WebKit/Source/modules/csspaint/CSSPaintDefinition.cpp |
index 339962f6cd6147786d9d8736a493e3331167b866..35a00874d06f8093852828ad542c1061ce7f12dd 100644 |
--- a/third_party/WebKit/Source/modules/csspaint/CSSPaintDefinition.cpp |
+++ b/third_party/WebKit/Source/modules/csspaint/CSSPaintDefinition.cpp |
@@ -5,9 +5,42 @@ |
#include "modules/csspaint/CSSPaintDefinition.h" |
#include "bindings/core/v8/ScriptState.h" |
+#include "bindings/core/v8/V8Binding.h" |
+#include "bindings/core/v8/V8BindingMacros.h" |
+#include "core/dom/ExecutionContext.h" |
+#include "core/frame/ConsoleTypes.h" |
+#include "core/inspector/ConsoleMessage.h" |
+#include "modules/csspaint/Geometry.h" |
+#include "modules/csspaint/PaintRenderingContext2D.h" |
+#include "platform/ScriptForbiddenScope.h" |
+#include "platform/graphics/ImageBuffer.h" |
+#include "platform/graphics/PaintGeneratedImage.h" |
+#include "platform/graphics/RecordingImageBufferSurface.h" |
+#include "platform/graphics/UnacceleratedImageBufferSurface.h" |
namespace blink { |
+namespace { |
+ |
+// TODO(ikilpatrick): Modify RecordingImageBufferSurface so that it cannot |
+// fallback to a UnacceleratedImageBufferSurface. |
+class UnacceleratedSurfaceFactory : public RecordingImageBufferFallbackSurfaceFactory { |
+public: |
+ virtual PassOwnPtr<ImageBufferSurface> createSurface(const IntSize& size, OpacityMode opacityMode) |
+ { |
+ return adoptPtr(new UnacceleratedImageBufferSurface(size, opacityMode)); |
+ } |
+ |
+ virtual ~UnacceleratedSurfaceFactory() { } |
+}; |
+ |
+static void clearHandle(const v8::WeakCallbackInfo<ScopedPersistent<v8::Object>>& data) |
+{ |
+ data.GetParameter()->clear(); |
+} |
+ |
+} // namespace |
+ |
CSSPaintDefinition* CSSPaintDefinition::create(ScriptState* scriptState, v8::Local<v8::Function> constructor, v8::Local<v8::Function> paint) |
{ |
return new CSSPaintDefinition(scriptState, constructor, paint); |
@@ -24,4 +57,74 @@ CSSPaintDefinition::~CSSPaintDefinition() |
{ |
} |
+PassRefPtr<Image> CSSPaintDefinition::paint(const IntSize& size) |
+{ |
+ v8::Isolate* isolate = m_scriptState->isolate(); |
+ |
+ ScriptState::Scope scope(m_scriptState.get()); |
+ v8::MicrotasksScope microtasksScope(isolate, v8::MicrotasksScope::kRunMicrotasks); |
+ |
+ v8::Local<v8::Object> instance = m_instance.newLocal(m_scriptState->isolate()); |
+ |
+ // We may have failed to create an instance class, in which case produce an |
+ // invalid image. |
+ if (isUndefinedOrNull(instance)) |
+ return nullptr; |
+ |
+ PaintRenderingContext2D* renderingContext = PaintRenderingContext2D::create( |
+ ImageBuffer::create(adoptPtr(new RecordingImageBufferSurface(size, adoptPtr(new UnacceleratedSurfaceFactory()))))); |
+ Geometry* geom = Geometry::create(size); |
+ |
+ v8::Local<v8::Value> argv[] = { |
+ toV8(renderingContext, m_scriptState->context()->Global(), isolate), |
+ toV8(geom, m_scriptState->context()->Global(), isolate) |
+ }; |
+ |
+ v8::Local<v8::Function> paint = m_paint.newLocal(isolate); |
+ |
+ ScriptForbiddenScope::AllowUserAgentScript allowScripting; |
+ v8::TryCatch block(isolate); |
+ |
+ V8ScriptRunner::callFunction(paint, m_scriptState->getExecutionContext(), instance, 2, argv, isolate); |
+ |
+ // The paint function may have produced an error, in which case produce an |
+ // invalid image. |
+ if (block.HasCaught()) { |
+ // TODO(ikilpatrick): messageHandlerInMainThread doesn't currently know |
+ // about main-thread worklets. |
+ m_scriptState->getExecutionContext()->addConsoleMessage(ConsoleMessage::create(JSMessageSource, ErrorMessageLevel, "Failed to invoke the paint method.")); |
+ return nullptr; |
+ } |
+ |
+ return PaintGeneratedImage::create(renderingContext->imageBuffer()->getPicture(), size); |
+} |
+ |
+v8::Local<v8::Object> CSSPaintDefinition::paintInstance() |
+{ |
+ v8::Isolate* isolate = m_scriptState->isolate(); |
+ |
+ ScriptState::Scope scope(m_scriptState.get()); |
+ v8::MicrotasksScope microtasksScope(isolate, v8::MicrotasksScope::kDoNotRunMicrotasks); |
+ |
+ if (m_instance.isEmpty()) { |
+ v8::Local<v8::Function> constructor = m_constructor.newLocal(isolate); |
+ ASSERT(!isUndefinedOrNull(constructor)); |
+ |
+ v8::Local<v8::Object> paintInstance; |
+ v8::TryCatch block(isolate); |
+ if (!v8Call(constructor->NewInstance(m_scriptState->context()), paintInstance, block)) { |
+ // TODO(ikilpatrick): messageHandlerInMainThread doesn't currently |
+ // know about main-thread worklets. |
+ m_scriptState->getExecutionContext()->addConsoleMessage(ConsoleMessage::create(JSMessageSource, ErrorMessageLevel, "Failed to create instance of class.")); |
+ } else { |
+ m_instance.set(isolate, paintInstance); |
+ m_instance.setWeak(&m_instance, clearHandle); |
+ } |
+ |
+ return paintInstance; |
+ } |
+ |
+ return m_instance.newLocal(isolate); |
+} |
+ |
} // namespace blink |