|  | // Copyright 2015 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #ifndef CanvasRenderingContext2DState_h | 
|  | #define CanvasRenderingContext2DState_h | 
|  |  | 
|  | #include "core/css/CSSFontSelectorClient.h" | 
|  | #include "modules/canvas2d/ClipList.h" | 
|  | #include "platform/fonts/Font.h" | 
|  | #include "platform/transforms/AffineTransform.h" | 
|  | #include "wtf/Vector.h" | 
|  |  | 
|  | namespace blink { | 
|  |  | 
|  | class CanvasStyle; | 
|  | class CSSValue; | 
|  | class Element; | 
|  |  | 
|  | class CanvasRenderingContext2DState final : public NoBaseWillBeGarbageCollectedFinalized<CanvasRenderingContext2DState>, public CSSFontSelectorClient { | 
|  | WILL_BE_USING_GARBAGE_COLLECTED_MIXIN(CanvasRenderingContext2DState); | 
|  | WTF_MAKE_FAST_ALLOCATED_WILL_BE_REMOVED(CanvasRenderingContext2DState); | 
|  | public: | 
|  | static PassOwnPtrWillBeRawPtr<CanvasRenderingContext2DState> create() | 
|  | { | 
|  | return adoptPtrWillBeNoop(new CanvasRenderingContext2DState); | 
|  | } | 
|  |  | 
|  | ~CanvasRenderingContext2DState() override; | 
|  |  | 
|  | DECLARE_VIRTUAL_TRACE(); | 
|  |  | 
|  | enum ClipListCopyMode { | 
|  | CopyClipList, | 
|  | DontCopyClipList | 
|  | }; | 
|  |  | 
|  | enum PaintType { | 
|  | FillPaintType, | 
|  | StrokePaintType, | 
|  | ImagePaintType, | 
|  | }; | 
|  |  | 
|  | static PassOwnPtrWillBeRawPtr<CanvasRenderingContext2DState> create(const CanvasRenderingContext2DState& other, ClipListCopyMode mode) | 
|  | { | 
|  | return adoptPtrWillBeNoop(new CanvasRenderingContext2DState(other, mode)); | 
|  | } | 
|  | CanvasRenderingContext2DState& operator=(const CanvasRenderingContext2DState&); | 
|  |  | 
|  | // CSSFontSelectorClient implementation | 
|  | void fontsNeedUpdate(CSSFontSelector*) override; | 
|  |  | 
|  | bool hasUnrealizedSaves() const { return m_unrealizedSaveCount; } | 
|  | void save() { ++m_unrealizedSaveCount; } | 
|  | void restore() { --m_unrealizedSaveCount; } | 
|  | void resetUnrealizedSaveCount() { m_unrealizedSaveCount = 0; } | 
|  |  | 
|  | void setLineDash(const Vector<float>&); | 
|  | const Vector<float>& lineDash() const { return m_lineDash; } | 
|  |  | 
|  | void setShouldAntialias(bool); | 
|  | bool shouldAntialias() const; | 
|  |  | 
|  | void setLineDashOffset(float); | 
|  | float lineDashOffset() const { return m_lineDashOffset; } | 
|  |  | 
|  | // setTransform returns true iff transform is invertible; | 
|  | void setTransform(const AffineTransform&); | 
|  | void resetTransform(); | 
|  | AffineTransform transform() const { return m_transform; } | 
|  | bool isTransformInvertible() const { return m_isTransformInvertible; } | 
|  |  | 
|  | void clipPath(const SkPath&, AntiAliasingMode); | 
|  | bool hasClip() const { return m_hasClip; } | 
|  | bool hasComplexClip() const { return m_hasComplexClip; } | 
|  | void playbackClips(SkCanvas* canvas) const { m_clipList.playback(canvas); } | 
|  |  | 
|  | void setFont(const Font&, CSSFontSelector*); | 
|  | const Font& font() const; | 
|  | bool hasRealizedFont() const { return m_realizedFont; } | 
|  | void setUnparsedFont(const String& font) { m_unparsedFont = font; } | 
|  | const String& unparsedFont() const { return m_unparsedFont; } | 
|  |  | 
|  | void setFilter(PassRefPtrWillBeRawPtr<CSSValue>); | 
|  | void setUnparsedFilter(const String& filterString) { m_unparsedFilter = filterString; } | 
|  | const String& unparsedFilter() const { return m_unparsedFilter; } | 
|  | SkImageFilter* getFilter(Element*, const Font&) const; | 
|  | bool hasFilter() const { return m_filterValue; } | 
|  |  | 
|  | void setStrokeStyle(CanvasStyle*); | 
|  | CanvasStyle* strokeStyle() const { return m_strokeStyle.get(); } | 
|  |  | 
|  | void setFillStyle(CanvasStyle*); | 
|  | CanvasStyle* fillStyle() const { return m_fillStyle.get(); } | 
|  |  | 
|  | CanvasStyle* style(PaintType) const; | 
|  |  | 
|  | enum Direction { | 
|  | DirectionInherit, | 
|  | DirectionRTL, | 
|  | DirectionLTR | 
|  | }; | 
|  |  | 
|  | void setDirection(Direction direction) { m_direction = direction; } | 
|  | Direction direction() const { return m_direction; } | 
|  |  | 
|  | void setTextAlign(TextAlign align) { m_textAlign = align; } | 
|  | TextAlign textAlign() const { return m_textAlign; } | 
|  |  | 
|  | void setTextBaseline(TextBaseline baseline) { m_textBaseline = baseline; } | 
|  | TextBaseline textBaseline() const { return m_textBaseline; } | 
|  |  | 
|  | void setLineWidth(float lineWidth) { m_strokePaint.setStrokeWidth(lineWidth); } | 
|  | float lineWidth() const { return m_strokePaint.getStrokeWidth(); } | 
|  |  | 
|  | void setLineCap(LineCap lineCap) { m_strokePaint.setStrokeCap(static_cast<SkPaint::Cap>(lineCap)); } | 
|  | LineCap lineCap() const { return static_cast<LineCap>(m_strokePaint.getStrokeCap()); } | 
|  |  | 
|  | void setLineJoin(LineJoin lineJoin) { m_strokePaint.setStrokeJoin(static_cast<SkPaint::Join>(lineJoin)); } | 
|  | LineJoin lineJoin() const { return static_cast<LineJoin>(m_strokePaint.getStrokeJoin()); } | 
|  |  | 
|  | void setMiterLimit(float miterLimit) { m_strokePaint.setStrokeMiter(miterLimit); } | 
|  | float miterLimit() const { return m_strokePaint.getStrokeMiter(); } | 
|  |  | 
|  | void setShadowOffsetX(float); | 
|  | void setShadowOffsetY(float); | 
|  | const FloatSize& shadowOffset() const { return m_shadowOffset; } | 
|  |  | 
|  | void setShadowBlur(float); | 
|  | float shadowBlur() const { return m_shadowBlur; } | 
|  |  | 
|  | void setShadowColor(SkColor); | 
|  | SkColor shadowColor() const { return m_shadowColor; } | 
|  |  | 
|  | void setGlobalAlpha(float); | 
|  | float globalAlpha() const { return m_globalAlpha; } | 
|  |  | 
|  | void setGlobalComposite(SkXfermode::Mode); | 
|  | SkXfermode::Mode globalComposite() const; | 
|  |  | 
|  | void setImageSmoothingEnabled(bool); | 
|  | bool imageSmoothingEnabled() const; | 
|  |  | 
|  | void setUnparsedStrokeColor(const String& color) { m_unparsedStrokeColor = color; } | 
|  | const String& unparsedStrokeColor() const { return m_unparsedStrokeColor; } | 
|  |  | 
|  | void setUnparsedFillColor(const String& color) { m_unparsedFillColor = color; } | 
|  | const String& unparsedFillColor() const { return m_unparsedFillColor; } | 
|  |  | 
|  | bool shouldDrawShadows() const; | 
|  |  | 
|  | enum ImageType { | 
|  | NoImage, | 
|  | OpaqueImage, | 
|  | NonOpaqueImage | 
|  | }; | 
|  |  | 
|  | // If paint will not be used for painting a bitmap, set bitmapOpacity to Opaque | 
|  | const SkPaint* getPaint(PaintType, ShadowMode, ImageType = NoImage) const; | 
|  |  | 
|  | private: | 
|  | CanvasRenderingContext2DState(); | 
|  | CanvasRenderingContext2DState(const CanvasRenderingContext2DState&, ClipListCopyMode = CopyClipList); | 
|  |  | 
|  | void updateLineDash() const; | 
|  | void updateStrokeStyle() const; | 
|  | void updateFillStyle() const; | 
|  | void shadowParameterChanged(); | 
|  | SkDrawLooper* emptyDrawLooper() const; | 
|  | SkDrawLooper* shadowOnlyDrawLooper() const; | 
|  | SkDrawLooper* shadowAndForegroundDrawLooper() const; | 
|  | SkImageFilter* shadowOnlyImageFilter() const; | 
|  | SkImageFilter* shadowAndForegroundImageFilter() const; | 
|  |  | 
|  | unsigned m_unrealizedSaveCount; | 
|  |  | 
|  | String m_unparsedStrokeColor; | 
|  | String m_unparsedFillColor; | 
|  | PersistentWillBeMember<CanvasStyle> m_strokeStyle; | 
|  | PersistentWillBeMember<CanvasStyle> m_fillStyle; | 
|  |  | 
|  | mutable SkPaint m_strokePaint; | 
|  | mutable SkPaint m_fillPaint; | 
|  | mutable SkPaint m_imagePaint; | 
|  |  | 
|  | FloatSize m_shadowOffset; | 
|  | float m_shadowBlur; | 
|  | SkColor m_shadowColor; | 
|  | mutable RefPtr<SkDrawLooper> m_emptyDrawLooper; | 
|  | mutable RefPtr<SkDrawLooper> m_shadowOnlyDrawLooper; | 
|  | mutable RefPtr<SkDrawLooper> m_shadowAndForegroundDrawLooper; | 
|  | mutable RefPtr<SkImageFilter> m_shadowOnlyImageFilter; | 
|  | mutable RefPtr<SkImageFilter> m_shadowAndForegroundImageFilter; | 
|  |  | 
|  | float m_globalAlpha; | 
|  | AffineTransform m_transform; | 
|  | Vector<float> m_lineDash; | 
|  | float m_lineDashOffset; | 
|  |  | 
|  | String m_unparsedFont; | 
|  | Font m_font; | 
|  |  | 
|  | String m_unparsedFilter; | 
|  | RefPtrWillBeMember<CSSValue> m_filterValue; | 
|  | mutable RefPtr<SkImageFilter> m_resolvedFilter; | 
|  |  | 
|  | // Text state. | 
|  | TextAlign m_textAlign; | 
|  | TextBaseline m_textBaseline; | 
|  | Direction m_direction; | 
|  |  | 
|  | bool m_realizedFont : 1; | 
|  | bool m_isTransformInvertible : 1; | 
|  | bool m_hasClip : 1; | 
|  | bool m_hasComplexClip : 1; | 
|  | mutable bool m_fillStyleDirty : 1; | 
|  | mutable bool m_strokeStyleDirty : 1; | 
|  | mutable bool m_lineDashDirty : 1; | 
|  |  | 
|  | ClipList m_clipList; | 
|  | }; | 
|  |  | 
|  | } // blink | 
|  |  | 
|  | #endif |