762 lines
24 KiB
C++
762 lines
24 KiB
C++
/**
|
|
* @file rs274d.cpp
|
|
* @brief functions to read the rs274d commands from a rs274d/rs274x file
|
|
*/
|
|
|
|
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 1992-2023 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
|
|
*/
|
|
|
|
#include <gerbview.h>
|
|
#include <gerbview_frame.h>
|
|
#include <trigo.h>
|
|
#include <gerber_file_image.h>
|
|
#include <X2_gerber_attributes.h>
|
|
|
|
#include <cmath>
|
|
|
|
/* Gerber: NOTES about some important commands found in RS274D and RS274X (G codes).
|
|
* Some are now deprecated, but deprecated commands must be known by the Gerber reader
|
|
* Gn =
|
|
* G01 linear interpolation (linear trace)
|
|
* G02, G20, G21 Circular interpolation, clockwise
|
|
* G03, G30, G31 Circular interpolation, counterclockwise
|
|
* G04 = comment. Since Sept 2014, file attributes and other X2 attributes can be found here
|
|
* if the line starts by G04 #@!
|
|
* G06 parabolic interpolation
|
|
* G07 Cubic Interpolation
|
|
* G10 linear interpolation (scale x10)
|
|
* G11 linear interpolation (0.1x range)
|
|
* G12 linear interpolation (0.01x scale)
|
|
* G36 Start polygon mode (called a region, because the "polygon" can include arcs)
|
|
* G37 Stop polygon mode (and close it)
|
|
* G54 Selection Tool (outdated)
|
|
* G60 linear interpolation (scale x100)
|
|
* G70 Select Units = Inches
|
|
* G71 Select Units = Millimeters
|
|
* G74 enable 90 deg mode for arcs (CW or CCW)
|
|
* G75 enable 360 degrees for arcs (CW or CCW)
|
|
* G90 mode absolute coordinates
|
|
*
|
|
* X, Y
|
|
* X and Y are followed by + or - and m + n digits (not separated)
|
|
* m = integer part
|
|
* n = part after the comma
|
|
*ic formats: m = 2, n = 3 (size 2.3)
|
|
* m = 3, n = 4 (size 3.4)
|
|
* eg
|
|
* GxxX00345Y-06123*
|
|
*
|
|
* Tools and D_CODES
|
|
* Tool number (identification of shapes)
|
|
* 10 to 999
|
|
* D_CODES:
|
|
* D01 ... D9 = command codes:
|
|
* D01 = activating light (pen down) when placement
|
|
* D02 = light extinction (pen up) when placement
|
|
* D03 = Flash
|
|
* D09 = VAPE Flash (I never see this command in Gerber file)
|
|
* D51 = G54 preceded by -> Select VAPE
|
|
*
|
|
* D10 ... D999 = Identification Tool: tool selection
|
|
*/
|
|
|
|
|
|
/* Local Functions (are lower case since they are private to this source file)
|
|
**/
|
|
|
|
|
|
/**
|
|
* Initializes a given GBRITEM so that it can draw a circle which is filled and
|
|
* has no pen border.
|
|
*
|
|
* @param aGbrItem The GBRITEM to fill in.
|
|
* @param aAperture the associated type of aperture.
|
|
* @param Dcode_index The DCODE value, like D14.
|
|
* @param aPos The center point of the flash.
|
|
* @param aSize The diameter of the round flash.
|
|
* @param aLayerNegative set to true if the current layer is negative.
|
|
*/
|
|
void fillFlashedGBRITEM( GERBER_DRAW_ITEM* aGbrItem,
|
|
APERTURE_T aAperture,
|
|
int Dcode_index,
|
|
const VECTOR2I& aPos,
|
|
wxSize aSize,
|
|
bool aLayerNegative )
|
|
{
|
|
aGbrItem->m_Size = aSize;
|
|
aGbrItem->m_Start = aPos;
|
|
aGbrItem->m_End = aGbrItem->m_Start;
|
|
aGbrItem->m_DCode = Dcode_index;
|
|
aGbrItem->SetLayerPolarity( aLayerNegative );
|
|
aGbrItem->m_Flashed = true;
|
|
aGbrItem->SetNetAttributes( aGbrItem->m_GerberImageFile->m_NetAttributeDict );
|
|
|
|
switch( aAperture )
|
|
{
|
|
case APT_POLYGON: // flashed regular polygon
|
|
aGbrItem->m_ShapeType = GBR_SPOT_POLY;
|
|
break;
|
|
|
|
case APT_CIRCLE:
|
|
aGbrItem->m_ShapeType = GBR_SPOT_CIRCLE;
|
|
aGbrItem->m_Size.y = aGbrItem->m_Size.x;
|
|
break;
|
|
|
|
case APT_OVAL:
|
|
aGbrItem->m_ShapeType = GBR_SPOT_OVAL;
|
|
break;
|
|
|
|
case APT_RECT:
|
|
aGbrItem->m_ShapeType = GBR_SPOT_RECT;
|
|
break;
|
|
|
|
case APT_MACRO:
|
|
aGbrItem->m_ShapeType = GBR_SPOT_MACRO;
|
|
|
|
// Cache the bounding box for aperture macros
|
|
aGbrItem->GetDcodeDescr()->GetMacro()->GetApertureMacroShape( aGbrItem, aPos );
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Initialize a given GBRITEM so that it can draw a linear D code.
|
|
*
|
|
* @param aGbrItem The GERBER_DRAW_ITEM to fill in.
|
|
* @param Dcode_index The DCODE value, like D14.
|
|
* @param aStart The starting point of the line.
|
|
* @param aEnd The ending point of the line.
|
|
* @param aPenSize The size of the flash. Note rectangular shapes are legal.
|
|
* @param aLayerNegative set to true if the current layer is negative.
|
|
*/
|
|
void fillLineGBRITEM( GERBER_DRAW_ITEM* aGbrItem,
|
|
int Dcode_index,
|
|
const VECTOR2I& aStart,
|
|
const VECTOR2I& aEnd,
|
|
wxSize aPenSize,
|
|
bool aLayerNegative )
|
|
{
|
|
aGbrItem->m_Flashed = false;
|
|
|
|
aGbrItem->m_Size = aPenSize;
|
|
|
|
aGbrItem->m_Start = aStart;
|
|
aGbrItem->m_End = aEnd;
|
|
|
|
aGbrItem->m_DCode = Dcode_index;
|
|
aGbrItem->SetLayerPolarity( aLayerNegative );
|
|
|
|
aGbrItem->SetNetAttributes( aGbrItem->m_GerberImageFile->m_NetAttributeDict );
|
|
}
|
|
|
|
|
|
/**
|
|
* Initialize a given GBRITEM so that it can draw an arc G code.
|
|
*
|
|
* If multiquadrant == true : arc can be 0 to 360 degrees
|
|
* and \a rel_center is the center coordinate relative to start point.
|
|
*
|
|
* If multiquadrant == false arc can be only 0 to 90 deg,
|
|
* and only in the same quadrant :
|
|
* <ul>
|
|
* <li> absolute angle 0 to 90 (quadrant 1) or
|
|
* <li> absolute angle 90 to 180 (quadrant 2) or
|
|
* <li> absolute angle 180 to 270 (quadrant 3) or
|
|
* <li> absolute angle 270 to 0 (quadrant 4)
|
|
* </ul>
|
|
*
|
|
* @param aGbrItem is the GBRITEM to fill in.
|
|
* @param Dcode_index is the DCODE value, like D14.
|
|
* @param aStart is the starting point.
|
|
* @param aEnd is the ending point.
|
|
* @param aRelCenter is the center coordinate relative to start point,
|
|
* given in ABSOLUTE VALUE and the sign of values x et y de rel_center
|
|
* must be calculated from the previously given constraint: arc only in the same quadrant.
|
|
* @param aClockwise true if arc must be created clockwise
|
|
* @param aPenSize The size of the flash. Note rectangular shapes are legal.
|
|
* @param aMultiquadrant set to true to create arcs up to 360 degrees,
|
|
* false when arc is inside one quadrant
|
|
* @param aLayerNegative set to true if the current layer is negative.
|
|
*/
|
|
void fillArcGBRITEM( GERBER_DRAW_ITEM* aGbrItem, int Dcode_index, const VECTOR2I& aStart,
|
|
const VECTOR2I& aEnd, const VECTOR2I& aRelCenter, wxSize aPenSize,
|
|
bool aClockwise, bool aMultiquadrant, bool aLayerNegative )
|
|
{
|
|
VECTOR2I center, delta;
|
|
|
|
aGbrItem->m_ShapeType = GBR_ARC;
|
|
aGbrItem->m_Size = aPenSize;
|
|
aGbrItem->m_Flashed = false;
|
|
|
|
if( aGbrItem->m_GerberImageFile )
|
|
aGbrItem->SetNetAttributes( aGbrItem->m_GerberImageFile->m_NetAttributeDict );
|
|
|
|
if( aMultiquadrant )
|
|
{
|
|
center = aStart + aRelCenter;
|
|
}
|
|
else
|
|
{
|
|
// in single quadrant mode the relative coordinate aRelCenter is always >= 0
|
|
// So we must recalculate the actual sign of aRelCenter.x and aRelCenter.y
|
|
center = aRelCenter;
|
|
|
|
// calculate arc end coordinate relative to the starting point,
|
|
// because center is relative to the center point
|
|
delta = aEnd - aStart;
|
|
|
|
// now calculate the relative to aStart center position, for a draw function
|
|
// that use trigonometric arc angle (or counter-clockwise)
|
|
/* Quadrants:
|
|
* Y
|
|
* 2 | 1
|
|
* -------X
|
|
* 3 | 4
|
|
* C = actual relative arc center, S = arc start (axis origin) E = relative arc end
|
|
*/
|
|
if( (delta.x >= 0) && (delta.y >= 0) )
|
|
{
|
|
/* Quadrant 1 (trigo or cclockwise):
|
|
* C | E
|
|
* ---S---
|
|
* 3 | 4
|
|
*/
|
|
center.x = -center.x;
|
|
}
|
|
else if( (delta.x >= 0) && (delta.y < 0) )
|
|
{
|
|
/* Quadrant 4 (trigo or cclockwise):
|
|
* 2 | C
|
|
* ---S---
|
|
* 3 | E
|
|
*/
|
|
// Nothing to do
|
|
}
|
|
else if( (delta.x < 0) && (delta.y >= 0) )
|
|
{
|
|
/* Quadrant 2 (trigo or cclockwise):
|
|
* E | 1
|
|
* ---S---
|
|
* C | 4
|
|
*/
|
|
center.x = -center.x;
|
|
center.y = -center.y;
|
|
}
|
|
else
|
|
{
|
|
/* Quadrant 3 (trigo or cclockwise):
|
|
* 2 | 1
|
|
* ---S---
|
|
* E | C
|
|
*/
|
|
center.y = -center.y;
|
|
}
|
|
|
|
// Due to your draw arc function, we need this:
|
|
if( !aClockwise )
|
|
center = - center;
|
|
|
|
// Calculate actual arc center coordinate:
|
|
center += aStart;
|
|
}
|
|
|
|
if( aClockwise )
|
|
{
|
|
aGbrItem->m_Start = aStart;
|
|
aGbrItem->m_End = aEnd;
|
|
}
|
|
else
|
|
{
|
|
aGbrItem->m_Start = aEnd;
|
|
aGbrItem->m_End = aStart;
|
|
}
|
|
|
|
aGbrItem->m_ArcCentre = center;
|
|
|
|
aGbrItem->m_DCode = Dcode_index;
|
|
aGbrItem->SetLayerPolarity( aLayerNegative );
|
|
}
|
|
|
|
|
|
/**
|
|
* Create an arc G code when found in polygon outlines.
|
|
*
|
|
* If multiquadrant == true : arc can be 0 to 360 degrees and \a rel_center is the center
|
|
* coordinate relative to start point. If not multiquadrant, the arc can be only 0 to 90 deg,
|
|
* and only in the same quadrant:
|
|
*
|
|
* <ul>
|
|
* <li> absolute angle 0 to 90 (quadrant 1) or
|
|
* <li> absolute angle 90 to 180 (quadrant 2) or
|
|
* <li> absolute angle 180 to 270 (quadrant 3) or
|
|
* <li> absolute angle 270 to 0 (quadrant 4)
|
|
* </ul>
|
|
*
|
|
* @param aGbrItem is the GBRITEM to fill in.
|
|
* @param aStart is the starting point.
|
|
* @param aEnd is the ending point.
|
|
* @param rel_center is the center coordinate relative to start point,
|
|
* given in ABSOLUTE VALUE and the sign of values x et y de rel_center
|
|
* must be calculated from the previously given constraint: arc only in the
|
|
* same quadrant.
|
|
* @param aClockwise true if arc must be created clockwise
|
|
* @param aMultiquadrant set to true to create arcs up to 360 deg or
|
|
* false when arc is inside one quadrant
|
|
* @param aLayerNegative set to true if the current layer is negative
|
|
*/
|
|
static void fillArcPOLY( GERBER_DRAW_ITEM* aGbrItem, const VECTOR2I& aStart, const VECTOR2I& aEnd,
|
|
const VECTOR2I& rel_center, bool aClockwise, bool aMultiquadrant,
|
|
bool aLayerNegative )
|
|
{
|
|
/* in order to calculate arc parameters, we use fillArcGBRITEM
|
|
* so we muse create a dummy track and use its geometric parameters
|
|
*/
|
|
static GERBER_DRAW_ITEM dummyGbrItem( nullptr );
|
|
|
|
aGbrItem->SetLayerPolarity( aLayerNegative );
|
|
|
|
fillArcGBRITEM( &dummyGbrItem, 0, aStart, aEnd, rel_center, wxSize( 0, 0 ),
|
|
aClockwise, aMultiquadrant, aLayerNegative );
|
|
|
|
aGbrItem->SetNetAttributes( aGbrItem->m_GerberImageFile->m_NetAttributeDict );
|
|
|
|
VECTOR2I center;
|
|
center = dummyGbrItem.m_ArcCentre;
|
|
|
|
// Calculate coordinates relative to arc center;
|
|
VECTOR2I start = dummyGbrItem.m_Start - center;
|
|
VECTOR2I end = dummyGbrItem.m_End - center;
|
|
|
|
/* Calculate angle arc
|
|
* angle is trigonometrical (counter-clockwise),
|
|
* and axis is the X,Y gerber coordinates
|
|
*/
|
|
EDA_ANGLE start_angle( start );
|
|
EDA_ANGLE end_angle( end );
|
|
|
|
// dummyTrack has right geometric parameters, but
|
|
// fillArcGBRITEM calculates arc parameters for a draw function that expects
|
|
// start_angle < end_angle. So ensure this is the case here:
|
|
// Due to the fact atan2 returns angles between -180 to + 180 degrees,
|
|
// this is not always the case ( a modulo 360.0 degrees can be lost )
|
|
//
|
|
// Note also an arc with same start and end angle is a circle (360 deg arc)
|
|
// in gerber files
|
|
if( start_angle >= end_angle )
|
|
end_angle += ANGLE_360;
|
|
|
|
EDA_ANGLE arc_angle = start_angle - end_angle;
|
|
|
|
// Approximate arc by 36 segments per 360 degree
|
|
EDA_ANGLE increment_angle = ANGLE_360 / 36;
|
|
int count = std::abs( arc_angle.AsDegrees() / increment_angle.AsDegrees() );
|
|
|
|
if( aGbrItem->m_ShapeAsPolygon.OutlineCount() == 0 )
|
|
aGbrItem->m_ShapeAsPolygon.NewOutline();
|
|
|
|
// calculate polygon corners
|
|
// when arc is counter-clockwise, dummyGbrItem arc goes from end to start
|
|
// and we must always create a polygon from start to end.
|
|
for( int ii = 0; ii <= count; ii++ )
|
|
{
|
|
EDA_ANGLE rot;
|
|
VECTOR2I end_arc = start;
|
|
|
|
if( aClockwise )
|
|
rot = increment_angle * ii;
|
|
else
|
|
rot = increment_angle * ( count - ii );
|
|
|
|
if( ii < count )
|
|
RotatePoint( end_arc, -rot );
|
|
else // last point
|
|
end_arc = aClockwise ? end : start;
|
|
|
|
aGbrItem->m_ShapeAsPolygon.Append( end_arc + center );
|
|
}
|
|
}
|
|
|
|
|
|
int GERBER_FILE_IMAGE::CodeNumber( char*& aText )
|
|
{
|
|
int retval;
|
|
char* endptr;
|
|
|
|
errno = 0;
|
|
|
|
retval = strtol( aText + 1, &endptr, 10 );
|
|
|
|
if( endptr == aText || errno != 0 )
|
|
return 0;
|
|
|
|
wxCHECK_MSG( retval < std::numeric_limits<int>::max(), 0, _( "Invalid Code Number" ) );
|
|
|
|
aText = endptr;
|
|
|
|
return static_cast<int>( retval );
|
|
}
|
|
|
|
|
|
bool GERBER_FILE_IMAGE::Execute_G_Command( char*& text, int G_command )
|
|
{
|
|
switch( G_command )
|
|
{
|
|
case GC_PHOTO_MODE: // can starts a D03 flash command: redundant, can be safely ignored.
|
|
break;
|
|
|
|
case GC_LINEAR_INTERPOL_1X:
|
|
m_Iterpolation = GERB_INTERPOL_LINEAR_1X;
|
|
break;
|
|
|
|
case GC_CIRCLE_NEG_INTERPOL:
|
|
m_Iterpolation = GERB_INTERPOL_ARC_NEG;
|
|
break;
|
|
|
|
case GC_CIRCLE_POS_INTERPOL:
|
|
m_Iterpolation = GERB_INTERPOL_ARC_POS;
|
|
break;
|
|
|
|
case GC_COMMENT:
|
|
// Skip comment, but only if the line does not start by "G04 #@! "
|
|
// which is a metadata, i.e. a X2 command inside the comment.
|
|
// this comment is called a "structured comment"
|
|
if( strncmp( text, " #@! ", 5 ) == 0 )
|
|
{
|
|
text += 5;
|
|
|
|
// The string starting at text is the same as the X2 attribute,
|
|
// but a X2 attribute ends by '%'. So we build the X2 attribute string
|
|
std::string x2buf;
|
|
|
|
while( *text && (*text != '*') )
|
|
{
|
|
x2buf += *text;
|
|
text++;
|
|
}
|
|
|
|
// add the end of X2 attribute string
|
|
x2buf += "*%";
|
|
x2buf += '\0';
|
|
|
|
char* cptr = (char*)x2buf.data();
|
|
int code_command = ReadXCommandID( cptr );
|
|
ExecuteRS274XCommand( code_command, nullptr, 0, cptr );
|
|
}
|
|
|
|
while( *text && (*text != '*') )
|
|
text++;
|
|
|
|
break;
|
|
|
|
case GC_SELECT_TOOL:
|
|
{
|
|
int D_commande = CodeNumber( text );
|
|
|
|
if( D_commande < FIRST_DCODE )
|
|
return false;
|
|
|
|
if( D_commande > (TOOLS_MAX_COUNT - 1) )
|
|
D_commande = TOOLS_MAX_COUNT - 1;
|
|
|
|
m_Current_Tool = D_commande;
|
|
D_CODE* pt_Dcode = GetDCODE( D_commande );
|
|
|
|
if( pt_Dcode )
|
|
pt_Dcode->m_InUse = true;
|
|
|
|
break;
|
|
}
|
|
|
|
case GC_SPECIFY_INCHES:
|
|
m_GerbMetric = false; // false = Inches, true = metric
|
|
break;
|
|
|
|
case GC_SPECIFY_MILLIMETERS:
|
|
m_GerbMetric = true; // false = Inches, true = metric
|
|
break;
|
|
|
|
case GC_TURN_OFF_360_INTERPOL: // disable Multi cadran arc and Arc interpol
|
|
m_360Arc_enbl = false;
|
|
m_Iterpolation = GERB_INTERPOL_LINEAR_1X; // not sure it should be done
|
|
m_AsArcG74G75Cmd = true;
|
|
break;
|
|
|
|
case GC_TURN_ON_360_INTERPOL:
|
|
m_360Arc_enbl = true;
|
|
m_AsArcG74G75Cmd = true;
|
|
break;
|
|
|
|
case GC_SPECIFY_ABSOLUES_COORD:
|
|
m_Relative = false; // false = absolute Coord, true = relative
|
|
// Coord
|
|
break;
|
|
|
|
case GC_SPECIFY_RELATIVEES_COORD:
|
|
m_Relative = true; // false = absolute Coord, true = relative
|
|
// Coord
|
|
break;
|
|
|
|
case GC_TURN_ON_POLY_FILL:
|
|
m_PolygonFillMode = true;
|
|
m_Exposure = false;
|
|
break;
|
|
|
|
case GC_TURN_OFF_POLY_FILL:
|
|
if( m_Exposure && GetLastItemInList() ) // End of polygon
|
|
{
|
|
GERBER_DRAW_ITEM * gbritem = GetLastItemInList();
|
|
|
|
if( gbritem->m_ShapeAsPolygon.VertexCount() )
|
|
gbritem->m_ShapeAsPolygon.Append( gbritem->m_ShapeAsPolygon.CVertex( 0 ) );
|
|
|
|
StepAndRepeatItem( *gbritem );
|
|
}
|
|
|
|
m_Exposure = false;
|
|
m_PolygonFillMode = false;
|
|
m_PolygonFillModeState = 0;
|
|
m_Iterpolation = GERB_INTERPOL_LINEAR_1X; // not sure it should be done
|
|
break;
|
|
|
|
case GC_MOVE: // Non existent
|
|
default:
|
|
{
|
|
wxString msg;
|
|
msg.Printf( wxT( "G%0.2d command not handled" ), G_command );
|
|
AddMessageToList( msg );
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
bool GERBER_FILE_IMAGE::Execute_DCODE_Command( char*& text, int D_commande )
|
|
{
|
|
wxSize size( 15, 15 );
|
|
|
|
APERTURE_T aperture = APT_CIRCLE;
|
|
GERBER_DRAW_ITEM* gbritem;
|
|
|
|
int dcode = 0;
|
|
D_CODE* tool = nullptr;
|
|
wxString msg;
|
|
|
|
if( D_commande >= FIRST_DCODE ) // This is a "Set tool" command
|
|
{
|
|
if( D_commande > (TOOLS_MAX_COUNT - 1) )
|
|
D_commande = TOOLS_MAX_COUNT - 1;
|
|
|
|
// remember which tool is selected, nothing is done with it in this
|
|
// call
|
|
m_Current_Tool = D_commande;
|
|
|
|
D_CODE* pt_Dcode = GetDCODE( D_commande );
|
|
|
|
if( pt_Dcode )
|
|
pt_Dcode->m_InUse = true;
|
|
else
|
|
m_Has_MissingDCode = true;
|
|
|
|
return true;
|
|
}
|
|
else // D_commande = 0..9: this is a pen command (usually D1, D2 or D3)
|
|
{
|
|
m_Last_Pen_Command = D_commande;
|
|
}
|
|
|
|
if( m_PolygonFillMode ) // Enter a polygon description:
|
|
{
|
|
switch( D_commande )
|
|
{
|
|
case 1: // code D01 Draw line, exposure ON
|
|
if( !m_Exposure ) // Start a new polygon outline:
|
|
{
|
|
m_Exposure = true;
|
|
gbritem = new GERBER_DRAW_ITEM( this );
|
|
AddItemToList( gbritem );
|
|
gbritem->m_ShapeType = GBR_POLYGON;
|
|
gbritem->m_Flashed = false;
|
|
gbritem->m_DCode = 0; // No DCode for a Polygon (Region in Gerber dialect)
|
|
|
|
|
|
if( gbritem->m_GerberImageFile )
|
|
{
|
|
gbritem->SetNetAttributes( gbritem->m_GerberImageFile->m_NetAttributeDict );
|
|
gbritem->m_AperFunction = gbritem->m_GerberImageFile->m_AperFunction;
|
|
}
|
|
}
|
|
|
|
switch( m_Iterpolation )
|
|
{
|
|
case GERB_INTERPOL_ARC_NEG:
|
|
case GERB_INTERPOL_ARC_POS:
|
|
// Before any arc command, a G74 or G75 command must be set.
|
|
// Otherwise the Gerber file is invalid
|
|
if( !m_AsArcG74G75Cmd )
|
|
{
|
|
AddMessageToList( _( "Invalid Gerber file: missing G74 or G75 arc command" ) );
|
|
|
|
// Disable further warning messages:
|
|
m_AsArcG74G75Cmd = true;
|
|
}
|
|
|
|
gbritem = GetLastItemInList();
|
|
|
|
fillArcPOLY( gbritem, m_PreviousPos,
|
|
m_CurrentPos, m_IJPos,
|
|
( m_Iterpolation == GERB_INTERPOL_ARC_NEG ) ? false : true,
|
|
m_360Arc_enbl, GetLayerParams().m_LayerNegative );
|
|
break;
|
|
|
|
default:
|
|
gbritem = GetLastItemInList();
|
|
|
|
gbritem->m_Start = m_PreviousPos; // m_Start is used as temporary storage
|
|
|
|
if( gbritem->m_ShapeAsPolygon.OutlineCount() == 0 )
|
|
{
|
|
gbritem->m_ShapeAsPolygon.NewOutline();
|
|
gbritem->m_ShapeAsPolygon.Append( VECTOR2I( gbritem->m_Start ) );
|
|
}
|
|
|
|
gbritem->m_End = m_CurrentPos; // m_End is used as temporary storage
|
|
gbritem->m_ShapeAsPolygon.Append( VECTOR2I( gbritem->m_End ) );
|
|
break;
|
|
}
|
|
|
|
m_PreviousPos = m_CurrentPos;
|
|
m_PolygonFillModeState = 1;
|
|
break;
|
|
|
|
case 2: // code D2: exposure OFF (i.e. "move to")
|
|
if( m_Exposure && GetLastItemInList() ) // End of polygon
|
|
{
|
|
gbritem = GetLastItemInList();
|
|
gbritem->m_ShapeAsPolygon.Append( gbritem->m_ShapeAsPolygon.CVertex( 0 ) );
|
|
StepAndRepeatItem( *gbritem );
|
|
}
|
|
|
|
m_Exposure = false;
|
|
m_PreviousPos = m_CurrentPos;
|
|
m_PolygonFillModeState = 0;
|
|
break;
|
|
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
switch( D_commande )
|
|
{
|
|
case 1: // code D01 Draw line, exposure ON
|
|
m_Exposure = true;
|
|
|
|
tool = GetDCODE( m_Current_Tool );
|
|
|
|
if( tool )
|
|
{
|
|
size = tool->m_Size;
|
|
dcode = tool->m_Num_Dcode;
|
|
aperture = tool->m_ApertType;
|
|
}
|
|
|
|
switch( m_Iterpolation )
|
|
{
|
|
case GERB_INTERPOL_LINEAR_1X:
|
|
gbritem = new GERBER_DRAW_ITEM( this );
|
|
AddItemToList( gbritem );
|
|
|
|
fillLineGBRITEM( gbritem, dcode, m_PreviousPos,
|
|
m_CurrentPos, size, GetLayerParams().m_LayerNegative );
|
|
StepAndRepeatItem( *gbritem );
|
|
break;
|
|
|
|
case GERB_INTERPOL_ARC_NEG:
|
|
case GERB_INTERPOL_ARC_POS:
|
|
gbritem = new GERBER_DRAW_ITEM( this );
|
|
AddItemToList( gbritem );
|
|
|
|
if( m_LastCoordIsIJPos )
|
|
{
|
|
fillArcGBRITEM( gbritem, dcode, m_PreviousPos,
|
|
m_CurrentPos, m_IJPos, size,
|
|
( m_Iterpolation == GERB_INTERPOL_ARC_NEG ) ?
|
|
false : true, m_360Arc_enbl, GetLayerParams().m_LayerNegative );
|
|
m_LastCoordIsIJPos = false;
|
|
}
|
|
else
|
|
{
|
|
fillLineGBRITEM( gbritem, dcode, m_PreviousPos,
|
|
m_CurrentPos, size, GetLayerParams().m_LayerNegative );
|
|
}
|
|
|
|
StepAndRepeatItem( *gbritem );
|
|
|
|
break;
|
|
|
|
default:
|
|
msg.Printf( wxT( "RS274D: DCODE Command: interpol error (type %X)" ),
|
|
m_Iterpolation );
|
|
AddMessageToList( msg );
|
|
break;
|
|
}
|
|
|
|
m_PreviousPos = m_CurrentPos;
|
|
break;
|
|
|
|
case 2: // code D2: exposure OFF (i.e. "move to")
|
|
m_Exposure = false;
|
|
m_PreviousPos = m_CurrentPos;
|
|
break;
|
|
|
|
case 3: // code D3: flash aperture
|
|
tool = GetDCODE( m_Current_Tool );
|
|
|
|
if( tool )
|
|
{
|
|
size = tool->m_Size;
|
|
dcode = tool->m_Num_Dcode;
|
|
aperture = tool->m_ApertType;
|
|
}
|
|
|
|
gbritem = new GERBER_DRAW_ITEM( this );
|
|
AddItemToList( gbritem );
|
|
fillFlashedGBRITEM( gbritem, aperture, dcode, m_CurrentPos,
|
|
size, GetLayerParams().m_LayerNegative );
|
|
StepAndRepeatItem( *gbritem );
|
|
m_PreviousPos = m_CurrentPos;
|
|
break;
|
|
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|