From 4b38dfcd5a938ea8b8b25473d78184a431f8c82f Mon Sep 17 00:00:00 2001 From: Edouard DUPIN Date: Mon, 23 Nov 2015 21:50:18 +0100 Subject: [PATCH] [DEV] add generation of ellipse --- esvg/Base.cpp | 1 + esvg/Base.h | 1 + esvg/Ellipse.cpp | 85 ++++++++++++++++++++++++++++++++++-------------- 3 files changed, 63 insertions(+), 24 deletions(-) diff --git a/esvg/Base.cpp b/esvg/Base.cpp index db49219..b8f347d 100644 --- a/esvg/Base.cpp +++ b/esvg/Base.cpp @@ -11,6 +11,7 @@ #include #include +const float esvg::kappa90(0.5522847493f); #undef __class__ #define __class__ "PaintState" diff --git a/esvg/Base.h b/esvg/Base.h index 02c3c8c..37b5a1b 100644 --- a/esvg/Base.h +++ b/esvg/Base.h @@ -19,6 +19,7 @@ #include namespace esvg { + extern const float kappa90; //!< proportional lenght to the radius of a bezier handle for 90° arcs. /** * @brief Painting mode of the Object: */ diff --git a/esvg/Ellipse.cpp b/esvg/Ellipse.cpp index 6255114..f1b7bc6 100644 --- a/esvg/Ellipse.cpp +++ b/esvg/Ellipse.cpp @@ -8,6 +8,8 @@ #include #include +#include +#include #undef __class__ #define __class__ "Ellipse" @@ -67,35 +69,70 @@ void esvg::Ellipse::display(int32_t _spacing) { void esvg::Ellipse::draw(esvg::Renderer& _myRenderer, mat2& _basicTrans, int32_t _level) { SVG_VERBOSE(spacingDist(_level) << "DRAW esvg::Ellipse"); - /* - _myRenderer.m_renderArea->color(agg::rgba8(m_paint.fill.r, m_paint.fill.g, m_paint.fill.b, m_paint.fill.a)); - // Creating an ellipse - agg::ellipse myEllipse(m_c.x(), m_c.y(), m_r.x(), m_r.y(), 0); + if ( m_r.x()<=0.0f + || m_r.y()<=0.0f) { + SVG_VERBOSE(spacingDist(_level+1) << "Too small radius" << m_r); + return; + } + esvg::render::Path listElement; + listElement.clear(); + listElement.moveTo(false, m_c + vec2(m_r.x(), 0.0f)); + listElement.curveTo(false, + m_c + vec2(m_r.x(), m_r.y()*esvg::kappa90), + m_c + vec2(m_r.x()*esvg::kappa90, m_r.y()), + m_c + vec2(0.0f, m_r.y())); + listElement.curveTo(false, + m_c + vec2(-m_r.x()*esvg::kappa90, m_r.y()), + m_c + vec2(-m_r.x(), m_r.y()*esvg::kappa90), + m_c + vec2(-m_r.x(), 0.0f)); + listElement.curveTo(false, + m_c + vec2(-m_r.x(), -m_r.y()*esvg::kappa90), + m_c + vec2(-m_r.x()*esvg::kappa90, -m_r.y()), + m_c + vec2(0.0f, -m_r.y())); + listElement.curveTo(false, + m_c + vec2(m_r.x()*esvg::kappa90, -m_r.y()), + m_c + vec2(m_r.x(), -m_r.y()*esvg::kappa90), + m_c + vec2(m_r.x(), 0.0f)); + listElement.close(); - // Calculate transformation matrix ... - mat2 mtx = m_transformMatrix; + mat2 mtx = m_transformMatrix; mtx *= _basicTrans; - // set the filling mode : - _myRenderer.m_rasterizer.filling_rule((m_paint.flagEvenOdd)?agg::fill_even_odd:agg::fill_non_zero); - - if (m_paint.fill.a != 0x00) { - agg::conv_transform trans(myEllipse, mtx); - _myRenderer.m_rasterizer.add_path(trans); - agg::render_scanlines(_myRenderer.m_rasterizer, _myRenderer.m_scanLine, *_myRenderer.m_renderArea); + esvg::render::PointList listPoints; + listPoints = listElement.generateListPoints(_level, + _myRenderer.getInterpolationRecurtionMax(), + _myRenderer.getInterpolationThreshold()); + esvg::render::SegmentList listSegmentFill; + esvg::render::SegmentList listSegmentStroke; + esvg::render::Weight tmpFill; + esvg::render::Weight tmpStroke; + // Check if we need to display background + if (m_paint.fill.a() != 0x00) { + listSegmentFill.createSegmentList(listPoints); + // now, traverse the scanlines and find the intersections on each scanline, use non-zero rule + tmpFill.generate(_myRenderer.getSize(), + _myRenderer.getNumberSubScanLine(), + listSegmentFill); } - if (m_paint.strokeWidth > 0 && m_paint.stroke.a!=0x00 ) { - _myRenderer.m_renderArea->color(agg::rgba8(m_paint.stroke.r, m_paint.stroke.g, m_paint.stroke.b, m_paint.stroke.a)); - // drawing as an outline - agg::conv_stroke myEllipseStroke(myEllipse); - myEllipseStroke.width(m_paint.strokeWidth); - agg::conv_transform, mat2> transStroke(myEllipseStroke, mtx); - // set the filling mode : - _myRenderer.m_rasterizer.filling_rule(agg::fill_non_zero); - _myRenderer.m_rasterizer.add_path(transStroke); - agg::render_scanlines(_myRenderer.m_rasterizer, _myRenderer.m_scanLine, *_myRenderer.m_renderArea); + // check if we need to display stroke: + if ( m_paint.strokeWidth > 0 + && m_paint.stroke.a() != 0x00) { + listSegmentStroke.createSegmentListStroke(listPoints); + // now, traverse the scanlines and find the intersections on each scanline, use non-zero rule + tmpStroke.generate(_myRenderer.getSize(), + _myRenderer.getNumberSubScanLine(), + listSegmentStroke); } - */ + // add on images: + _myRenderer.print(tmpFill, + m_paint.fill, + tmpStroke, + m_paint.stroke, + m_paint.opacity); + #ifdef DEBUG + _myRenderer.addDebugSegment(listSegmentFill); + _myRenderer.addDebugSegment(listSegmentStroke); + #endif }