| /* |
| * Copyright (C) 2006, 2007, 2009, 2010, 2011, 2012, 2017 Apple Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR |
| * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
| * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
| * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
| * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY |
| * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #pragma once |
| |
| #include "AffineTransform.h" |
| #include "CanvasDirection.h" |
| #include "CanvasFillRule.h" |
| #include "CanvasLineCap.h" |
| #include "CanvasLineJoin.h" |
| #include "CanvasPath.h" |
| #include "CanvasRenderingContext.h" |
| #include "CanvasStyle.h" |
| #include "CanvasTextAlign.h" |
| #include "CanvasTextBaseline.h" |
| #include "Color.h" |
| #include "FloatSize.h" |
| #include "FontCascade.h" |
| #include "FontSelectorClient.h" |
| #include "GraphicsContext.h" |
| #include "GraphicsTypes.h" |
| #include "ImageBuffer.h" |
| #include "ImageSmoothingQuality.h" |
| #include "Path.h" |
| #include "PlatformLayer.h" |
| #include <wtf/Vector.h> |
| #include <wtf/text/WTFString.h> |
| |
| namespace WebCore { |
| |
| class TypedOMCSSImageValue; |
| class CachedImage; |
| class CanvasBase; |
| class CanvasGradient; |
| class CanvasPattern; |
| class DOMMatrix; |
| class FloatRect; |
| class GraphicsContext; |
| class HTMLImageElement; |
| class HTMLVideoElement; |
| class ImageBitmap; |
| class ImageData; |
| class Path2D; |
| class RenderStyle; |
| class RenderObject; |
| class TextMetrics; |
| |
| struct DOMMatrix2DInit; |
| |
| #if ENABLE(VIDEO) && ENABLE(CSS_TYPED_OM) |
| using CanvasImageSource = Variant<RefPtr<HTMLImageElement>, RefPtr<HTMLVideoElement>, RefPtr<HTMLCanvasElement>, RefPtr<ImageBitmap>, RefPtr<TypedOMCSSImageValue>>; |
| #elif ENABLE(VIDEO) |
| using CanvasImageSource = Variant<RefPtr<HTMLImageElement>, RefPtr<HTMLVideoElement>, RefPtr<HTMLCanvasElement>, RefPtr<ImageBitmap>>; |
| #else |
| using CanvasImageSource = Variant<RefPtr<HTMLImageElement>, RefPtr<HTMLCanvasElement>, RefPtr<ImageBitmap>>; |
| #endif |
| |
| class CanvasRenderingContext2DBase : public CanvasRenderingContext, public CanvasPath { |
| WTF_MAKE_ISO_ALLOCATED(CanvasRenderingContext2DBase); |
| public: |
| CanvasRenderingContext2DBase(CanvasBase&, bool usesCSSCompatibilityParseMode); |
| virtual ~CanvasRenderingContext2DBase(); |
| |
| float lineWidth() const; |
| void setLineWidth(float); |
| |
| CanvasLineCap lineCap() const; |
| void setLineCap(CanvasLineCap); |
| void setLineCap(const String&); |
| |
| CanvasLineJoin lineJoin() const; |
| void setLineJoin(CanvasLineJoin); |
| void setLineJoin(const String&); |
| |
| float miterLimit() const; |
| void setMiterLimit(float); |
| |
| const Vector<float>& getLineDash() const; |
| void setLineDash(const Vector<float>&); |
| const Vector<float>& webkitLineDash() const { return getLineDash(); } |
| void setWebkitLineDash(const Vector<float>&); |
| |
| float lineDashOffset() const; |
| void setLineDashOffset(float); |
| |
| float shadowOffsetX() const; |
| void setShadowOffsetX(float); |
| |
| float shadowOffsetY() const; |
| void setShadowOffsetY(float); |
| |
| float shadowBlur() const; |
| void setShadowBlur(float); |
| |
| String shadowColor() const; |
| void setShadowColor(const String&); |
| |
| float globalAlpha() const; |
| void setGlobalAlpha(float); |
| |
| String globalCompositeOperation() const; |
| void setGlobalCompositeOperation(const String&); |
| |
| void save() { ++m_unrealizedSaveCount; } |
| void restore(); |
| |
| void scale(float sx, float sy); |
| void rotate(float angleInRadians); |
| void translate(float tx, float ty); |
| void transform(float m11, float m12, float m21, float m22, float dx, float dy); |
| |
| Ref<DOMMatrix> getTransform() const; |
| void setTransform(float m11, float m12, float m21, float m22, float dx, float dy); |
| ExceptionOr<void> setTransform(DOMMatrix2DInit&&); |
| void resetTransform(); |
| |
| void setStrokeColor(const String& color, Optional<float> alpha = WTF::nullopt); |
| void setStrokeColor(float grayLevel, float alpha = 1.0); |
| void setStrokeColor(float r, float g, float b, float a); |
| void setStrokeColor(float c, float m, float y, float k, float a); |
| |
| void setFillColor(const String& color, Optional<float> alpha = WTF::nullopt); |
| void setFillColor(float grayLevel, float alpha = 1.0f); |
| void setFillColor(float r, float g, float b, float a); |
| void setFillColor(float c, float m, float y, float k, float a); |
| |
| void beginPath(); |
| |
| void fill(CanvasFillRule = CanvasFillRule::Nonzero); |
| void stroke(); |
| void clip(CanvasFillRule = CanvasFillRule::Nonzero); |
| |
| void fill(Path2D&, CanvasFillRule = CanvasFillRule::Nonzero); |
| void stroke(Path2D&); |
| void clip(Path2D&, CanvasFillRule = CanvasFillRule::Nonzero); |
| |
| bool isPointInPath(float x, float y, CanvasFillRule = CanvasFillRule::Nonzero); |
| bool isPointInStroke(float x, float y); |
| |
| bool isPointInPath(Path2D&, float x, float y, CanvasFillRule = CanvasFillRule::Nonzero); |
| bool isPointInStroke(Path2D&, float x, float y); |
| |
| void clearRect(float x, float y, float width, float height); |
| void fillRect(float x, float y, float width, float height); |
| void strokeRect(float x, float y, float width, float height); |
| |
| void setShadow(float width, float height, float blur, const String& color = String(), Optional<float> alpha = WTF::nullopt); |
| void setShadow(float width, float height, float blur, float grayLevel, float alpha = 1.0); |
| void setShadow(float width, float height, float blur, float r, float g, float b, float a); |
| void setShadow(float width, float height, float blur, float c, float m, float y, float k, float a); |
| |
| void clearShadow(); |
| |
| ExceptionOr<void> drawImage(CanvasImageSource&&, float dx, float dy); |
| ExceptionOr<void> drawImage(CanvasImageSource&&, float dx, float dy, float dw, float dh); |
| ExceptionOr<void> drawImage(CanvasImageSource&&, float sx, float sy, float sw, float sh, float dx, float dy, float dw, float dh); |
| |
| void drawImageFromRect(HTMLImageElement&, float sx = 0, float sy = 0, float sw = 0, float sh = 0, float dx = 0, float dy = 0, float dw = 0, float dh = 0, const String& compositeOperation = emptyString()); |
| |
| using Style = Variant<String, RefPtr<CanvasGradient>, RefPtr<CanvasPattern>>; |
| Style strokeStyle() const; |
| void setStrokeStyle(Style&&); |
| Style fillStyle() const; |
| void setFillStyle(Style&&); |
| |
| ExceptionOr<Ref<CanvasGradient>> createLinearGradient(float x0, float y0, float x1, float y1); |
| ExceptionOr<Ref<CanvasGradient>> createRadialGradient(float x0, float y0, float r0, float x1, float y1, float r1); |
| ExceptionOr<RefPtr<CanvasPattern>> createPattern(CanvasImageSource&&, const String& repetition); |
| |
| RefPtr<ImageData> createImageData(ImageData&) const; |
| ExceptionOr<RefPtr<ImageData>> createImageData(float width, float height) const; |
| ExceptionOr<RefPtr<ImageData>> getImageData(float sx, float sy, float sw, float sh) const; |
| void putImageData(ImageData&, float dx, float dy); |
| void putImageData(ImageData&, float dx, float dy, float dirtyX, float dirtyY, float dirtyWidth, float dirtyHeight); |
| |
| float webkitBackingStorePixelRatio() const { return 1; } |
| |
| void reset(); |
| |
| LineCap getLineCap() const { return state().lineCap; } |
| LineJoin getLineJoin() const { return state().lineJoin; } |
| |
| bool imageSmoothingEnabled() const; |
| void setImageSmoothingEnabled(bool); |
| |
| ImageSmoothingQuality imageSmoothingQuality() const; |
| void setImageSmoothingQuality(ImageSmoothingQuality); |
| |
| void setPath(Path2D&); |
| Ref<Path2D> getPath() const; |
| |
| bool usesDisplayListDrawing() const { return m_usesDisplayListDrawing; }; |
| void setUsesDisplayListDrawing(bool flag) { m_usesDisplayListDrawing = flag; }; |
| |
| bool tracksDisplayListReplay() const { return m_tracksDisplayListReplay; } |
| void setTracksDisplayListReplay(bool); |
| |
| String displayListAsText(DisplayList::AsTextFlags) const; |
| String replayDisplayListAsText(DisplayList::AsTextFlags) const; |
| |
| using Direction = CanvasDirection; |
| |
| class FontProxy : public FontSelectorClient { |
| public: |
| FontProxy() = default; |
| virtual ~FontProxy(); |
| FontProxy(const FontProxy&); |
| FontProxy& operator=(const FontProxy&); |
| |
| bool realized() const { return m_font.fontSelector(); } |
| void initialize(FontSelector&, const RenderStyle&); |
| const FontMetrics& fontMetrics() const; |
| const FontCascadeDescription& fontDescription() const; |
| float width(const TextRun&, GlyphOverflow* = 0) const; |
| void drawBidiText(GraphicsContext&, const TextRun&, const FloatPoint&, FontCascade::CustomFontNotReadyAction) const; |
| |
| private: |
| void update(FontSelector&); |
| void fontsNeedUpdate(FontSelector&) override; |
| |
| FontCascade m_font; |
| }; |
| |
| struct State final { |
| State(); |
| |
| State(const State&); |
| State& operator=(const State&); |
| |
| String unparsedStrokeColor; |
| String unparsedFillColor; |
| CanvasStyle strokeStyle; |
| CanvasStyle fillStyle; |
| float lineWidth; |
| LineCap lineCap; |
| LineJoin lineJoin; |
| float miterLimit; |
| FloatSize shadowOffset; |
| float shadowBlur; |
| Color shadowColor; |
| float globalAlpha; |
| CompositeOperator globalComposite; |
| BlendMode globalBlend; |
| AffineTransform transform; |
| bool hasInvertibleTransform; |
| Vector<float> lineDash; |
| float lineDashOffset; |
| bool imageSmoothingEnabled; |
| ImageSmoothingQuality imageSmoothingQuality; |
| |
| // Text state. |
| TextAlign textAlign; |
| TextBaseline textBaseline; |
| Direction direction; |
| |
| String unparsedFont; |
| FontProxy font; |
| }; |
| |
| const State& state() const { return m_stateStack.last(); } |
| const Vector<State, 1>& stateStack(); |
| |
| protected: |
| static const int DefaultFontSize; |
| static const char* const DefaultFontFamily; |
| static const char* const DefaultFont; |
| |
| enum CanvasDidDrawOption { |
| CanvasDidDrawApplyNone = 0, |
| CanvasDidDrawApplyTransform = 1, |
| CanvasDidDrawApplyShadow = 1 << 1, |
| CanvasDidDrawApplyClip = 1 << 2, |
| CanvasDidDrawApplyAll = 0xffffffff |
| }; |
| |
| bool isFullCanvasCompositeMode(CompositeOperator); |
| |
| State& modifiableState() { ASSERT(!m_unrealizedSaveCount || m_stateStack.size() >= MaxSaveCount); return m_stateStack.last(); } |
| |
| void applyLineDash() const; |
| void setShadow(const FloatSize& offset, float blur, const Color&); |
| void applyShadow(); |
| bool shouldDrawShadows() const; |
| |
| void didDraw(const FloatRect&, unsigned options = CanvasDidDrawApplyAll); |
| void didDrawEntireCanvas(); |
| |
| void paintRenderingResultsToCanvas() override; |
| |
| GraphicsContext* drawingContext() const; |
| |
| void unwindStateStack(); |
| void realizeSaves(); |
| void realizeSavesLoop(); |
| |
| void applyStrokePattern(); |
| void applyFillPattern(); |
| |
| void setStrokeStyle(CanvasStyle); |
| void setFillStyle(CanvasStyle); |
| |
| ExceptionOr<RefPtr<CanvasPattern>> createPattern(HTMLImageElement&, bool repeatX, bool repeatY); |
| ExceptionOr<RefPtr<CanvasPattern>> createPattern(CanvasBase&, bool repeatX, bool repeatY); |
| #if ENABLE(VIDEO) |
| ExceptionOr<RefPtr<CanvasPattern>> createPattern(HTMLVideoElement&, bool repeatX, bool repeatY); |
| #endif |
| ExceptionOr<RefPtr<CanvasPattern>> createPattern(ImageBitmap&, bool repeatX, bool repeatY); |
| #if ENABLE(CSS_TYPED_OM) |
| ExceptionOr<RefPtr<CanvasPattern>> createPattern(TypedOMCSSImageValue&, bool repeatX, bool repeatY); |
| #endif |
| |
| ExceptionOr<void> drawImage(HTMLImageElement&, const FloatRect& srcRect, const FloatRect& dstRect); |
| ExceptionOr<void> drawImage(HTMLImageElement&, const FloatRect& srcRect, const FloatRect& dstRect, const CompositeOperator&, const BlendMode&); |
| ExceptionOr<void> drawImage(HTMLCanvasElement&, const FloatRect& srcRect, const FloatRect& dstRect); |
| ExceptionOr<void> drawImage(Document&, CachedImage*, const RenderObject*, const FloatRect& imageRect, const FloatRect& srcRect, const FloatRect& dstRect, const CompositeOperator&, const BlendMode&); |
| #if ENABLE(VIDEO) |
| ExceptionOr<void> drawImage(HTMLVideoElement&, const FloatRect& srcRect, const FloatRect& dstRect); |
| #endif |
| #if ENABLE(CSS_TYPED_OM) |
| ExceptionOr<void> drawImage(TypedOMCSSImageValue&, const FloatRect& srcRect, const FloatRect& dstRect); |
| #endif |
| ExceptionOr<void> drawImage(ImageBitmap&, const FloatRect& srcRect, const FloatRect& dstRect); |
| |
| void beginCompositeLayer(); |
| void endCompositeLayer(); |
| |
| void fillInternal(const Path&, CanvasFillRule); |
| void strokeInternal(const Path&); |
| void clipInternal(const Path&, CanvasFillRule); |
| |
| bool isPointInPathInternal(const Path&, float x, float y, CanvasFillRule); |
| bool isPointInStrokeInternal(const Path&, float x, float y); |
| |
| void clearCanvas(); |
| Path transformAreaToDevice(const Path&) const; |
| Path transformAreaToDevice(const FloatRect&) const; |
| bool rectContainsCanvas(const FloatRect&) const; |
| |
| template<class T> IntRect calculateCompositingBufferRect(const T&, IntSize*); |
| std::unique_ptr<ImageBuffer> createCompositingBuffer(const IntRect&); |
| void compositeBuffer(ImageBuffer&, const IntRect&, CompositeOperator); |
| |
| void inflateStrokeRect(FloatRect&) const; |
| |
| template<class T> void fullCanvasCompositedDrawImage(T&, const FloatRect&, const FloatRect&, CompositeOperator); |
| |
| ExceptionOr<RefPtr<ImageData>> getImageData(ImageBuffer::CoordinateSystem, float sx, float sy, float sw, float sh) const; |
| void putImageData(ImageData&, ImageBuffer::CoordinateSystem, float dx, float dy, float dirtyX, float dirtyY, float dirtyWidth, float dirtyHeight); |
| |
| bool isAccelerated() const override; |
| |
| bool hasInvertibleTransform() const override { return state().hasInvertibleTransform; } |
| |
| #if ENABLE(ACCELERATED_2D_CANVAS) |
| PlatformLayer* platformLayer() const override; |
| #endif |
| |
| static const unsigned MaxSaveCount = 1024 * 16; |
| Vector<State, 1> m_stateStack; |
| unsigned m_unrealizedSaveCount { 0 }; |
| bool m_usesCSSCompatibilityParseMode; |
| bool m_usesDisplayListDrawing { false }; |
| bool m_tracksDisplayListReplay { false }; |
| mutable std::unique_ptr<struct DisplayListDrawingContext> m_recordingContext; |
| }; |
| |
| } // namespace WebCore |
| |