/** * @file SVG_plotter.cpp * @brief KiCad: specialized plotter for SVG files format */ /* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2020 Jean-Pierre Charras, jp.charras at wanadoo.fr * Copyright (C) 1992-2021 KiCad Developers, see AUTHORS.txt for contributors. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, you may find one here: * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html * or you may search the http://www.gnu.org website for the version 2 license, * or you may write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ /* Some info on basic items SVG format, used here: * The root element of all SVG files is the element. * * The element is used to group SVG shapes together. * Once grouped you can transform the whole group of shapes as if it was a single shape. * This is an advantage compared to a nested element * which cannot be the target of transformation by itself. * * The element represents a rectangle. * Using this element you can draw rectangles of various width, height, * with different stroke (outline) and fill colors, with sharp or rounded corners etc. * * * * * * * * The element is used to draw circles. * * * The element is used to draw ellipses. * An ellipse is a circle that does not have equal height and width. * Its radius in the x and y directions are different, in other words. * * * The element is used to draw lines. * * * * * The element is used to draw multiple connected lines * Here is a simple example: * * * * The element is used to draw with multiple (3 or more) sides / edges. * Here is a simple example: * * * * The element is used to draw advanced shapes combined from lines and arcs, * with or without fill. * It is probably the most advanced and versatile SVG shape of them all. * It is probably also the hardest element to master. * * * Draw an elliptic arc: it is one of basic path command: * * flag_arc_large: 0 = small arc > 180 deg, 1 = large arc > 180 deg * flag_sweep : 0 = CCW, 1 = CW * The center of ellipse is automatically calculated. */ #include #include #include #include #include #include #include #include #include /** * Translates '<' to "<", '>' to ">" and so on, according to the spec: * http://www.w3.org/TR/2000/WD-xml-c14n-20000119.html#charescaping * May be moved to a library if needed generally, but not expecting that. */ static wxString XmlEsc( const wxString& aStr, bool isAttribute = false ) { wxString escaped; escaped.reserve( aStr.length() ); for( wxString::const_iterator it = aStr.begin(); it != aStr.end(); ++it ) { const wxChar c = *it; switch( c ) { case wxS( '<' ): escaped.append( wxS( "<" ) ); break; case wxS( '>' ): escaped.append( wxS( ">" ) ); break; case wxS( '&' ): escaped.append( wxS( "&" ) ); break; case wxS( '\r' ): escaped.append( wxS( " " ) ); break; default: if( isAttribute ) { switch( c ) { case wxS( '"' ): escaped.append( wxS( """ ) ); break; case wxS( '\t' ): escaped.append( wxS( " " ) ); break; case wxS( '\n' ): escaped.append( wxS( " " )); break; default: escaped.append(c); } } else escaped.append(c); } } return escaped; } SVG_PLOTTER::SVG_PLOTTER() { m_graphics_changed = true; SetTextMode( PLOT_TEXT_MODE::STROKE ); m_fillMode = FILL_T::NO_FILL; // or FILLED_SHAPE or FILLED_WITH_BG_BODYCOLOR m_pen_rgb_color = 0; // current color value (black) m_brush_rgb_color = 0; // current color value (black) m_dashed = PLOT_DASH_TYPE::SOLID; m_useInch = true; // decimils is the default m_precision = 4; // because there where used before it was changeable } void SVG_PLOTTER::SetViewport( const wxPoint& aOffset, double aIusPerDecimil, double aScale, bool aMirror ) { m_plotMirror = aMirror; m_yaxisReversed = true; // unlike other plotters, SVG has Y axis reversed m_plotOffset = aOffset; m_plotScale = aScale; m_IUsPerDecimil = aIusPerDecimil; /* Compute the paper size in IUs */ m_paperSize = m_pageInfo.GetSizeMils(); m_paperSize.x *= 10.0 * aIusPerDecimil; m_paperSize.y *= 10.0 * aIusPerDecimil; // set iuPerDeviceUnit, in 0.1mils ( 2.54um ) // this was used before the format was changeable, so we set is as default SetSvgCoordinatesFormat( 4, true ); } void SVG_PLOTTER::SetSvgCoordinatesFormat( unsigned aResolution, bool aUseInches ) { m_useInch = aUseInches; m_precision = aResolution; // gives now a default value to iuPerDeviceUnit (because the units of the caller is now known) double iusPerMM = m_IUsPerDecimil / 2.54 * 1000; m_iuPerDeviceUnit = pow( 10.0, m_precision ) / ( iusPerMM ); if( m_useInch ) m_iuPerDeviceUnit /= 25.4; // convert to inch } void SVG_PLOTTER::SetColor( const COLOR4D& color ) { PSLIKE_PLOTTER::SetColor( color ); if( m_graphics_changed ) setSVGPlotStyle(); } void SVG_PLOTTER::setFillMode( FILL_T fill ) { if( m_fillMode != fill ) { m_graphics_changed = true; m_fillMode = fill; } } void SVG_PLOTTER::setSVGPlotStyle( bool aIsGroup, const std::string& aExtraStyle ) { if( aIsGroup ) fputs( "\n", m_outputFile ); m_graphics_changed = false; } fputs( "\n", m_outputFile ); } void SVG_PLOTTER::SetCurrentLineWidth( int aWidth, void* aData ) { if( aWidth == DO_NOT_SET_LINE_WIDTH ) return; else if( aWidth == USE_DEFAULT_LINE_WIDTH ) aWidth = m_renderSettings->GetDefaultPenWidth(); else if( aWidth == 0 ) aWidth = 1; wxASSERT_MSG( aWidth > 0, "Plotter called to set negative pen width" ); if( aWidth != m_currentPenWidth ) { m_graphics_changed = true; m_currentPenWidth = aWidth; } if( m_graphics_changed ) setSVGPlotStyle(); } void SVG_PLOTTER::StartBlock( void* aData ) { std::string* idstr = reinterpret_cast( aData ); fputs( "c_str() ); fprintf( m_outputFile, ">\n" ); } void SVG_PLOTTER::EndBlock( void* aData ) { fprintf( m_outputFile, "\n" ); m_graphics_changed = true; } void SVG_PLOTTER::emitSetRGBColor( double r, double g, double b ) { int red = (int) ( 255.0 * r ); int green = (int) ( 255.0 * g ); int blue = (int) ( 255.0 * b ); long rgb_color = (red << 16) | (green << 8) | blue; if( m_pen_rgb_color != rgb_color ) { m_graphics_changed = true; m_pen_rgb_color = rgb_color; // Currently, use the same color for brush and pen (i.e. to draw and fill a contour). m_brush_rgb_color = rgb_color; } } void SVG_PLOTTER::SetDash( PLOT_DASH_TYPE dashed ) { if( m_dashed != dashed ) { m_graphics_changed = true; m_dashed = dashed; } if( m_graphics_changed ) setSVGPlotStyle(); } void SVG_PLOTTER::Rect( const wxPoint& p1, const wxPoint& p2, FILL_T fill, int width ) { EDA_RECT rect( p1, wxSize( p2.x -p1.x, p2.y -p1.y ) ); rect.Normalize(); DPOINT org_dev = userToDeviceCoordinates( rect.GetOrigin() ); DPOINT end_dev = userToDeviceCoordinates( rect.GetEnd() ); DSIZE size_dev = end_dev - org_dev; // Ensure size of rect in device coordinates is > 0 // I don't know if this is a SVG issue or a Inkscape issue, but // Inkscape has problems with negative or null values for width and/or height, so avoid them DBOX rect_dev( org_dev, size_dev); rect_dev.Normalize(); setFillMode( fill ); SetCurrentLineWidth( width ); // Rectangles having a 0 size value for height or width are just not drawn on Inkscape, // so use a line when happens. if( rect_dev.GetSize().x == 0.0 || rect_dev.GetSize().y == 0.0 ) // Draw a line { fprintf( m_outputFile, "\n", rect_dev.GetPosition().x, rect_dev.GetPosition().y, rect_dev.GetEnd().x, rect_dev.GetEnd().y ); } else { fprintf( m_outputFile, "\n", rect_dev.GetPosition().x, rect_dev.GetPosition().y, rect_dev.GetSize().x, rect_dev.GetSize().y, 0.0 /* radius of rounded corners */ ); } } void SVG_PLOTTER::Circle( const wxPoint& pos, int diametre, FILL_T fill, int width ) { DPOINT pos_dev = userToDeviceCoordinates( pos ); double radius = userToDeviceSize( diametre / 2.0 ); setFillMode( fill ); SetCurrentLineWidth( width ); // If diameter is less than width, switch to filled mode if( fill == FILL_T::NO_FILL && diametre < width ) { setFillMode( FILL_T::FILLED_SHAPE ); SetCurrentLineWidth( 0 ); radius = userToDeviceSize( ( diametre / 2.0 ) + ( width / 2.0 ) ); } fprintf( m_outputFile, " \n", pos_dev.x, pos_dev.y, radius ); } void SVG_PLOTTER::Arc( const wxPoint& centre, double StAngle, double EndAngle, int radius, FILL_T fill, int width ) { /* Draws an arc of a circle, centered on (xc,yc), with starting point * (x1, y1) and ending at (x2, y2). The current pen is used for the outline * and the current brush for filling the shape. * * The arc is drawn in an anticlockwise direction from the start point to * the end point */ if( radius <= 0 ) { Circle( centre, width, FILL_T::FILLED_SHAPE, 0 ); return; } if( StAngle > EndAngle ) std::swap( StAngle, EndAngle ); // Calculate start point. DPOINT centre_dev = userToDeviceCoordinates( centre ); double radius_dev = userToDeviceSize( radius ); if( !m_yaxisReversed ) // Should be never the case { double tmp = StAngle; StAngle = -EndAngle; EndAngle = -tmp; } if( m_plotMirror ) { if( m_mirrorIsHorizontal ) { StAngle = 1800.0 -StAngle; EndAngle = 1800.0 -EndAngle; std::swap( StAngle, EndAngle ); } else { StAngle = -StAngle; EndAngle = -EndAngle; } } DPOINT start; start.x = radius_dev; RotatePoint( &start.x, &start.y, StAngle ); DPOINT end; end.x = radius_dev; RotatePoint( &end.x, &end.y, EndAngle ); start += centre_dev; end += centre_dev; double theta1 = DECIDEG2RAD( StAngle ); if( theta1 < 0 ) theta1 = theta1 + M_PI * 2; double theta2 = DECIDEG2RAD( EndAngle ); if( theta2 < 0 ) theta2 = theta2 + M_PI * 2; if( theta2 < theta1 ) theta2 = theta2 + M_PI * 2; int flg_arc = 0; // flag for large or small arc. 0 means less than 180 degrees if( fabs( theta2 - theta1 ) > M_PI ) flg_arc = 1; int flg_sweep = 0; // flag for sweep always 0 // Draw a single arc: an arc is one of 3 curve commands (2 other are 2 bezier curves) // params are start point, radius1, radius2, X axe rotation, // flag arc size (0 = small arc > 180 deg, 1 = large arc > 180 deg), // sweep arc ( 0 = CCW, 1 = CW), // end point if( fill != FILL_T::NO_FILL ) { // Filled arcs (in Eeschema) consist of the pie wedge and a stroke only on the arc // This needs to be drawn in two steps. setFillMode( fill ); SetCurrentLineWidth( 0 ); fprintf( m_outputFile, "\n", start.x, start.y, radius_dev, radius_dev, flg_arc, flg_sweep, end.x, end.y, centre_dev.x, centre_dev.y ); } setFillMode( FILL_T::NO_FILL ); SetCurrentLineWidth( width ); fprintf( m_outputFile, "\n", start.x, start.y, radius_dev, radius_dev, flg_arc, flg_sweep, end.x, end.y ); } void SVG_PLOTTER::BezierCurve( const wxPoint& aStart, const wxPoint& aControl1, const wxPoint& aControl2, const wxPoint& aEnd, int aTolerance, int aLineThickness ) { #if 1 setFillMode( FILL_T::NO_FILL ); SetCurrentLineWidth( aLineThickness ); DPOINT start = userToDeviceCoordinates( aStart ); DPOINT ctrl1 = userToDeviceCoordinates( aControl1 ); DPOINT ctrl2 = userToDeviceCoordinates( aControl2 ); DPOINT end = userToDeviceCoordinates( aEnd ); // Generate a cubic curve: start point and 3 other control points. fprintf( m_outputFile, "\n", start.x, start.y, ctrl1.x, ctrl1.y, ctrl2.x, ctrl2.y, end.x, end.y ); #else PLOTTER::BezierCurve( aStart, aControl1, aControl2, aEnd, aTolerance, aLineThickness ); #endif } void SVG_PLOTTER::PlotPoly( const std::vector& aCornerList, FILL_T aFill, int aWidth, void* aData ) { if( aCornerList.size() <= 1 ) return; setFillMode( aFill ); SetCurrentLineWidth( aWidth ); fprintf( m_outputFile, " \n" ); } else { pos = userToDeviceCoordinates( aCornerList.back() ); fprintf( m_outputFile, "%f,%f\n\" /> \n", pos.x, pos.y ); } } void SVG_PLOTTER::PlotImage( const wxImage& aImage, const wxPoint& aPos, double aScaleFactor ) { wxSize pix_size( aImage.GetWidth(), aImage.GetHeight() ); // Requested size (in IUs) DPOINT drawsize( aScaleFactor * pix_size.x, aScaleFactor * pix_size.y ); // calculate the bitmap start position wxPoint start( aPos.x - drawsize.x / 2, aPos.y - drawsize.y / 2); // Rectangles having a 0 size value for height or width are just not drawn on Inkscape, // so use a line when happens. if( drawsize.x == 0.0 || drawsize.y == 0.0 ) // Draw a line { PLOTTER::PlotImage( aImage, aPos, aScaleFactor ); } else { wxMemoryOutputStream img_stream; aImage.SaveFile( img_stream, wxBITMAP_TYPE_PNG ); size_t input_len = img_stream.GetOutputStreamBuffer()->GetBufferSize(); std::vector buffer( input_len ); std::vector encoded; img_stream.CopyTo( buffer.data(), buffer.size() ); base64::encode( buffer, encoded ); fprintf( m_outputFile, "( encoded[i] ) ); if( ( i % 64 ) == 63 ) fprintf( m_outputFile, "\n" ); } fprintf( m_outputFile, "\"\npreserveAspectRatio=\"none\" width=\"%f\" height=\"%f\" />", userToDeviceSize( drawsize.x ), userToDeviceSize( drawsize.y ) ); } } void SVG_PLOTTER::PenTo( const wxPoint& pos, char plume ) { if( plume == 'Z' ) { if( m_penState != 'Z' ) { fputs( "\" />\n", m_outputFile ); m_penState = 'Z'; m_penLastpos.x = -1; m_penLastpos.y = -1; } return; } if( m_penState == 'Z' ) // here plume = 'D' or 'U' { DPOINT pos_dev = userToDeviceCoordinates( pos ); // Ensure we do not use a fill mode when moving the pen, // in SVG mode (i;e. we are plotting only basic lines, not a filled area if( m_fillMode != FILL_T::NO_FILL ) { setFillMode( FILL_T::NO_FILL ); setSVGPlotStyle(); } fprintf( m_outputFile, "\n", " \n", "\n", (double) m_paperSize.x / m_IUsPerDecimil * 2.54 / 10000, (double) m_paperSize.y / m_IUsPerDecimil * 2.54 / 10000, origin.x, origin.y, (int) ( m_paperSize.x * m_iuPerDeviceUnit ), (int) ( m_paperSize.y * m_iuPerDeviceUnit) ); // Write title char date_buf[250]; time_t ltime = time( nullptr ); strftime( date_buf, 250, "%Y/%m/%d %H:%M:%S", localtime( <ime ) ); fprintf( m_outputFile, "SVG Picture created as %s date %s \n", TO_UTF8( XmlEsc( wxFileName( m_filename ).GetFullName() ) ), date_buf ); // End of header fprintf( m_outputFile, " Picture generated by %s \n", TO_UTF8( XmlEsc( m_creator ) ) ); // output the pen and brush color (RVB values in hex) and opacity double opacity = 1.0; // 0.0 (transparent to 1.0 (solid) fprintf( m_outputFile, "\n", m_outputFile ); return true; } bool SVG_PLOTTER::EndPlot() { fputs( " \n\n", m_outputFile ); fclose( m_outputFile ); m_outputFile = nullptr; return true; } void SVG_PLOTTER::Text( const wxPoint& aPos, const COLOR4D& aColor, const wxString& aText, const EDA_ANGLE& aOrient, const wxSize& aSize, enum GR_TEXT_H_ALIGN_T aH_justify, enum GR_TEXT_V_ALIGN_T aV_justify, int aWidth, bool aItalic, bool aBold, bool aMultilineAllowed, void* aData ) { setFillMode( FILL_T::NO_FILL ); SetColor( aColor ); SetCurrentLineWidth( aWidth ); wxPoint text_pos = aPos; const char *hjust = "start"; switch( aH_justify ) { case GR_TEXT_H_ALIGN_CENTER: hjust = "middle"; break; case GR_TEXT_H_ALIGN_RIGHT: hjust = "end"; break; case GR_TEXT_H_ALIGN_LEFT: hjust = "start"; break; } switch( aV_justify ) { case GR_TEXT_V_ALIGN_CENTER: text_pos.y += aSize.y / 2; break; case GR_TEXT_V_ALIGN_TOP: text_pos.y += aSize.y; break; case GR_TEXT_V_ALIGN_BOTTOM: break; } wxSize text_size; // aSize.x or aSize.y is < 0 for mirrored texts. // The actual text size value is the absolute value text_size.x = std::abs( GraphicTextWidth( aText, aSize, aItalic, aWidth ) ); text_size.y = std::abs( aSize.x * 4/3 ); // Hershey font height to em size conversion DPOINT anchor_pos_dev = userToDeviceCoordinates( aPos ); DPOINT text_pos_dev = userToDeviceCoordinates( text_pos ); DPOINT sz_dev = userToDeviceSize( text_size ); if( aOrient != EDA_ANGLE::ANGLE_0 ) { fprintf( m_outputFile, "\n", - aOrient.AsDegrees(), anchor_pos_dev.x, anchor_pos_dev.y ); } fprintf( m_outputFile, "\n", sz_dev.x, sz_dev.y, hjust, TO_UTF8( XmlEsc( aText ) ) ); if( aOrient != EDA_ANGLE::ANGLE_0 ) fputs( "\n", m_outputFile ); fprintf( m_outputFile, "%s\n", TO_UTF8( XmlEsc( aText ) ) ); PLOTTER::Text( aPos, aColor, aText, aOrient, aSize, aH_justify, aV_justify, aWidth, aItalic, aBold, aMultilineAllowed ); fputs( "", m_outputFile ); }