/* * Copyright 2005-2007, Stephan Aßmus . * Copyright 2008, Andrej Spielmann . * Copyright 2015, Julian Harnath * All rights reserved. Distributed under the terms of the MIT License. * * API to the Anti-Grain Geometry based "Painter" drawing backend. Manages * rendering pipe-lines for stroke, fills, bitmap and text rendering. * */ #ifndef PAINTER_H #define PAINTER_H #include "AGGTextRenderer.h" #include "FontManager.h" #include "PainterAggInterface.h" #include "PatternHandler.h" #include "ServerFont.h" #include "Transformable.h" #include "defines.h" #include #include #include #include class BBitmap; class BRegion; class BGradient; class BGradientLinear; class BGradientRadial; class BGradientRadialFocus; class BGradientDiamond; class BGradientConic; class DrawState; class FontCacheReference; class RenderingBuffer; class ServerBitmap; class ServerFont; // Defines for SIMD support. #define APPSERVER_SIMD_MMX (1 << 0) #define APPSERVER_SIMD_SSE (1 << 1) class Painter { public: Painter(); virtual ~Painter(); // frame buffer stuff void AttachToBuffer(RenderingBuffer* buffer); void DetachFromBuffer(); BRect Bounds() const; void SetDrawState(const DrawState* data, int32 xOffset = 0, int32 yOffset = 0); void ConstrainClipping(const BRegion* region); const BRegion* ClippingRegion() const { return fClippingRegion; } // object settings void SetTransform(BAffineTransform transform, int32 xOffset, int32 yOffset); inline bool IsIdentityTransform() const { return fIdentityTransform; } const Transformable& Transform() const { return fTransform; } void SetHighColor(const rgb_color& color); inline rgb_color HighColor() const { return fPatternHandler.HighColor(); } void SetLowColor(const rgb_color& color); inline rgb_color LowColor() const { return fPatternHandler.LowColor(); } void SetPenSize(float size); inline float PenSize() const { return fPenSize; } void SetStrokeMode(cap_mode lineCap, join_mode joinMode, float miterLimit); void SetFillRule(int32 fillRule); void SetPattern(const pattern& p); inline pattern Pattern() const { return *fPatternHandler.GetR5Pattern(); } void SetDrawingMode(drawing_mode mode); inline drawing_mode DrawingMode() const { return fDrawingMode; } void SetBlendingMode(source_alpha srcAlpha, alpha_function alphaFunc); void SetFont(const ServerFont& font); void SetFont(const DrawState* state); inline const ServerFont& Font() const { return fTextRenderer.Font(); } // painting functions // lines void StrokeLine(BPoint a, BPoint b); // returns true if the line was either vertical or horizontal // draws a solid one pixel wide line of color c, no blending bool StraightLine(BPoint a, BPoint b, const rgb_color& c) const; // triangles BRect StrokeTriangle(BPoint pt1, BPoint pt2, BPoint pt3) const; BRect FillTriangle(BPoint pt1, BPoint pt2, BPoint pt3) const; BRect FillTriangle(BPoint pt1, BPoint pt2, BPoint pt3, const BGradient& gradient) const; // polygons BRect DrawPolygon(BPoint* ptArray, int32 numPts, bool filled, bool closed) const; BRect FillPolygon(BPoint* ptArray, int32 numPts, const BGradient& gradient, bool closed) const; // bezier curves BRect DrawBezier(BPoint* controlPoints, bool filled) const; BRect FillBezier(BPoint* controlPoints, const BGradient& gradient) const; // shapes BRect DrawShape(const int32& opCount, const uint32* opList, const int32& ptCount, const BPoint* ptList, bool filled, const BPoint& viewToScreenOffset, float viewScale) const; BRect FillShape(const int32& opCount, const uint32* opList, const int32& ptCount, const BPoint* ptList, const BGradient& gradient, const BPoint& viewToScreenOffset, float viewScale) const; // rects BRect StrokeRect(const BRect& r) const; // strokes a one pixel wide solid rect, no blending void StrokeRect(const BRect& r, const rgb_color& c) const; BRect FillRect(const BRect& r) const; BRect FillRect(const BRect& r, const BGradient& gradient) const; // fills a solid rect with color c, no blending void FillRect(const BRect& r, const rgb_color& c) const; // fills a rect with a linear gradient, the caller should be // sure that the gradient is indeed vertical. The start point of // the gradient should be above the end point, or this function // will not draw anything. void FillRectVerticalGradient(BRect r, const BGradientLinear& gradient) const; // fills a solid rect with color c, no blending, no clipping void FillRectNoClipping(const clipping_rect& r, const rgb_color& c) const; // round rects BRect StrokeRoundRect(const BRect& r, float xRadius, float yRadius) const; BRect FillRoundRect(const BRect& r, float xRadius, float yRadius) const; BRect FillRoundRect(const BRect& r, float xRadius, float yRadius, const BGradient& gradient) const; // ellipses void AlignEllipseRect(BRect* rect, bool filled) const; BRect DrawEllipse(BRect r, bool filled) const; BRect FillEllipse(BRect r, const BGradient& gradient) const; // arcs BRect StrokeArc(BPoint center, float xRadius, float yRadius, float angle, float span) const; BRect FillArc(BPoint center, float xRadius, float yRadius, float angle, float span) const; BRect FillArc(BPoint center, float xRadius, float yRadius, float angle, float span, const BGradient& gradient) const; // strings BRect DrawString(const char* utf8String, uint32 length, BPoint baseLine, const escapement_delta* delta, FontCacheReference* cacheReference = NULL); BRect DrawString(const char* utf8String, uint32 length, const BPoint* offsets, FontCacheReference* cacheReference = NULL); BRect BoundingBox(const char* utf8String, uint32 length, BPoint baseLine, BPoint* penLocation, const escapement_delta* delta, FontCacheReference* cacheReference = NULL) const; BRect BoundingBox(const char* utf8String, uint32 length, const BPoint* offsets, BPoint* penLocation, FontCacheReference* cacheReference = NULL) const; float StringWidth(const char* utf8String, uint32 length, const escapement_delta* delta = NULL); // bitmaps BRect DrawBitmap(const ServerBitmap* bitmap, BRect bitmapRect, BRect viewRect, uint32 options) const; // some convenience stuff BRect FillRegion(const BRegion* region) const; BRect FillRegion(const BRegion* region, const BGradient& gradient) const; BRect InvertRect(const BRect& r) const; inline BRect TransformAndClipRect(BRect rect) const; inline BRect ClipRect(BRect rect) const; inline BRect TransformAlignAndClipRect(BRect rect) const; inline BRect AlignAndClipRect(BRect rect) const; inline BRect AlignRect(BRect rect) const; void SetRendererOffset(int32 offsetX, int32 offsetY); private: float _Align(float coord, bool round, bool centerOffset) const; void _Align(BPoint* point, bool round, bool centerOffset) const; void _Align(BPoint* point, bool centerOffset = true) const; BPoint _Align(const BPoint& point, bool centerOffset = true) const; BRect _Clipped(const BRect& rect) const; void _UpdateFont() const; void _UpdateLineWidth(); void _UpdateDrawingMode(); void _SetRendererColor(const rgb_color& color) const; // drawing functions stroke/fill BRect _DrawTriangle(BPoint pt1, BPoint pt2, BPoint pt3, bool fill) const; void _IterateShapeData(const int32& opCount, const uint32* opList, const int32& ptCount, const BPoint* points, const BPoint& viewToScreenOffset, float viewScale) const; void _InvertRect32(BRect r) const; void _BlendRect32(const BRect& r, const rgb_color& c) const; template BRect _BoundingBox(VertexSource& path) const; template BRect _StrokePath(VertexSource& path) const; template BRect _StrokePath(VertexSource& path, cap_mode capMode) const; template BRect _FillPath(VertexSource& path) const; template BRect _RasterizePath(VertexSource& path) const; template BRect _FillPath(VertexSource& path, const BGradient& gradient) const; template BRect _RasterizePath(VertexSource& path, const BGradient& gradient) const; void _CalcLinearGradientTransform(BPoint startPoint, BPoint endPoint, agg::trans_affine& mtx, float gradient_d2 = 100.0f) const; void _CalcRadialGradientTransform(BPoint center, agg::trans_affine& mtx, float gradient_d2 = 100.0f) const; void _MakeGradient(const BGradient& gradient, int32 colorCount, uint32* colors, int32 arrayOffset, int32 arraySize) const; template void _MakeGradient(Array& array, const BGradient& gradient) const; template void _RasterizePath(VertexSource& path, const BGradient& gradient, GradientFunction function, agg::trans_affine& gradientTransform, int gradientStop = 100) const; private: class BitmapPainter; friend class BitmapPainter; // needed only for gcc2 private: mutable PainterAggInterface fInternal; // for internal coordinate rounding/transformation bool fSubpixelPrecise : 1; bool fValidClipping : 1; bool fAttached : 1; bool fIdentityTransform : 1; Transformable fTransform; float fPenSize; const BRegion* fClippingRegion; drawing_mode fDrawingMode; source_alpha fAlphaSrcMode; alpha_function fAlphaFncMode; cap_mode fLineCapMode; join_mode fLineJoinMode; float fMiterLimit; PatternHandler fPatternHandler; // a class handling rendering and caching of glyphs // it is setup to load from a specific Freetype supported // font file which it gets from ServerFont mutable AGGTextRenderer fTextRenderer; }; inline BRect Painter::TransformAndClipRect(BRect rect) const { rect.left = floorf(rect.left); rect.top = floorf(rect.top); rect.right = ceilf(rect.right); rect.bottom = ceilf(rect.bottom); if (!fIdentityTransform) rect = fTransform.TransformBounds(rect); return _Clipped(rect); } inline BRect Painter::ClipRect(BRect rect) const { rect.left = floorf(rect.left); rect.top = floorf(rect.top); rect.right = ceilf(rect.right); rect.bottom = ceilf(rect.bottom); return _Clipped(rect); } inline BRect Painter::AlignAndClipRect(BRect rect) const { return _Clipped(AlignRect(rect)); } inline BRect Painter::TransformAlignAndClipRect(BRect rect) const { rect = AlignRect(rect); if (!fIdentityTransform) rect = fTransform.TransformBounds(rect); return _Clipped(rect); } inline BRect Painter::AlignRect(BRect rect) const { rect.left = floorf(rect.left); rect.top = floorf(rect.top); if (fSubpixelPrecise) { rect.right = ceilf(rect.right); rect.bottom = ceilf(rect.bottom); } else { rect.right = floorf(rect.right); rect.bottom = floorf(rect.bottom); } return rect; } #endif // PAINTER_H