954 lines
28 KiB
C++
954 lines
28 KiB
C++
/****************************************/
|
|
/**** Routine de trace GERBER RS274X ****/
|
|
/****************************************/
|
|
|
|
#include "fctsys.h"
|
|
|
|
#include "common.h"
|
|
#include "plot_common.h"
|
|
#include "pcbnew.h"
|
|
#include "pcbplot.h"
|
|
#include "trigo.h"
|
|
#include "plotgerb.h"
|
|
|
|
#include "protos.h"
|
|
|
|
|
|
/* Variables locales : */
|
|
static int s_Last_D_code;
|
|
static float Gerb_scale_plot; // Coeff de conversion d'unites des traces
|
|
static D_CODE* s_DCodeList; // Pointeur sur la zone de stockage des D_CODES
|
|
wxString GerberFullFileName;
|
|
static double scale_x, scale_y; // echelles de convertion en X et Y (compte tenu
|
|
// des unites relatives du PCB et des traceurs
|
|
static bool ShowDcodeError = TRUE;
|
|
|
|
/* Routines Locales */
|
|
|
|
static void Init_Trace_GERBER( WinEDA_BasePcbFrame* frame, FILE* gerbfile );
|
|
static void Init_ApertureList();
|
|
static void Fin_Trace_GERBER( WinEDA_BasePcbFrame* frame, FILE* gerbfile );
|
|
static void Plot_1_CIRCLE_pad_GERBER( wxPoint pos, int diametre );
|
|
static void trace_1_pastille_OVALE_GERBER( wxPoint pos, wxSize size, int orient );
|
|
static void PlotRectangularPad_GERBER( wxPoint pos, wxSize size, int orient );
|
|
|
|
static D_CODE* get_D_code( int dx, int dy, int type, int drill );
|
|
static void trace_1_pad_TRAPEZE_GERBER( wxPoint pos, wxSize size, wxSize delta,
|
|
int orient, int modetrace );
|
|
|
|
|
|
/********************************************************************************/
|
|
void WinEDA_BasePcbFrame::Genere_GERBER( const wxString& FullFileName, int Layer,
|
|
bool PlotOriginIsAuxAxis )
|
|
/********************************************************************************/
|
|
|
|
/* Creates the output files, one per board layer:
|
|
* filenames are like xxxc.PHO and use the RS274X format
|
|
* Units = inches
|
|
* format 3.4, Leading zero omitted, Abs format
|
|
* format 3.4 uses the native pcbnew units (1/10000 inch).
|
|
*/
|
|
{
|
|
int tracevia = 1;
|
|
|
|
EraseMsgBox();
|
|
GerberFullFileName = FullFileName;
|
|
|
|
g_PlotOrient = 0;
|
|
if( Plot_Set_MIROIR )
|
|
g_PlotOrient |= PLOT_MIROIR;
|
|
|
|
/* Calcul des echelles de conversion */
|
|
Gerb_scale_plot = 1.0; /* pour unites gerber en 0,1 Mils, format 3.4 */
|
|
scale_x = Scale_X * Gerb_scale_plot;
|
|
scale_y = Scale_Y * Gerb_scale_plot;
|
|
g_PlotOffset.x = 0;
|
|
g_PlotOffset.y = 0;
|
|
if( PlotOriginIsAuxAxis )
|
|
g_PlotOffset = m_Auxiliary_Axis_Position;
|
|
|
|
dest = wxFopen( FullFileName, wxT( "wt" ) );
|
|
if( dest == NULL )
|
|
{
|
|
wxString msg = _( "unable to create file " ) + FullFileName;
|
|
DisplayError( this, msg );
|
|
return;
|
|
}
|
|
|
|
SetLocaleTo_C_standard( );
|
|
|
|
InitPlotParametresGERBER( g_PlotOffset, scale_x, scale_y );
|
|
|
|
/* Clear the memory used for handle the D_CODE (aperture) list */
|
|
Init_ApertureList();
|
|
|
|
Affiche_1_Parametre( this, 0, _( "File" ), FullFileName, CYAN );
|
|
|
|
Init_Trace_GERBER( this, dest );
|
|
|
|
nb_plot_erreur = 0;
|
|
|
|
int layer_mask = g_TabOneLayerMask[Layer];
|
|
|
|
// Specify that the contents of the "Edges Pcb" layer are also to be
|
|
// plotted, unless the option of excluding that layer has been selected.
|
|
if( !g_Exclude_Edges_Pcb )
|
|
layer_mask |= EDGE_LAYER;
|
|
|
|
switch( Layer )
|
|
{
|
|
case FIRST_COPPER_LAYER:
|
|
case LAYER_N_2:
|
|
case LAYER_N_3:
|
|
case LAYER_N_4:
|
|
case LAYER_N_5:
|
|
case LAYER_N_6:
|
|
case LAYER_N_7:
|
|
case LAYER_N_8:
|
|
case LAYER_N_9:
|
|
case LAYER_N_10:
|
|
case LAYER_N_11:
|
|
case LAYER_N_12:
|
|
case LAYER_N_13:
|
|
case LAYER_N_14:
|
|
case LAYER_N_15:
|
|
case LAST_COPPER_LAYER:
|
|
Plot_Layer_GERBER( dest, layer_mask, 0, 1 );
|
|
break;
|
|
|
|
case SOLDERMASK_N_CU:
|
|
case SOLDERMASK_N_CMP: /* Trace du vernis epargne */
|
|
if( g_DrawViaOnMaskLayer )
|
|
tracevia = 1;
|
|
else
|
|
tracevia = 0;
|
|
Plot_Layer_GERBER( dest, layer_mask, g_DesignSettings.m_MaskMargin, tracevia );
|
|
break;
|
|
|
|
case SOLDERPASTE_N_CU:
|
|
case SOLDERPASTE_N_CMP: /* Trace du masque de pate de soudure */
|
|
Plot_Layer_GERBER( dest, layer_mask, 0, 0 );
|
|
break;
|
|
|
|
default:
|
|
Plot_Serigraphie( PLOT_FORMAT_GERBER, dest, layer_mask );
|
|
break;
|
|
}
|
|
|
|
Fin_Trace_GERBER( this, dest );
|
|
SetLocaleTo_Default( );
|
|
}
|
|
|
|
|
|
/***********************************************************************/
|
|
void WinEDA_BasePcbFrame::Plot_Layer_GERBER( FILE* File, int masque_layer,
|
|
int garde, int tracevia )
|
|
/***********************************************************************/
|
|
|
|
/* Creates one GERBER file for a copper layer or a technical layer
|
|
* the silkscreen layers are plotted by Plot_Serigraphie() because they have special features
|
|
*/
|
|
{
|
|
wxPoint pos;
|
|
wxSize size;
|
|
MODULE* Module;
|
|
D_PAD* PtPad;
|
|
TRACK* track;
|
|
EDA_BaseStruct* PtStruct;
|
|
wxString msg;
|
|
|
|
// (Following command has been superceded by new command on lines 93 and 94.)
|
|
// masque_layer |= EDGE_LAYER; /* Les elements de la couche EDGE sont tj traces */
|
|
|
|
/* Draw items type Drawings Pcb : */
|
|
PtStruct = m_Pcb->m_Drawings;
|
|
for( ; PtStruct != NULL; PtStruct = PtStruct->Pnext )
|
|
{
|
|
switch( PtStruct->Type() )
|
|
{
|
|
case TYPEDRAWSEGMENT:
|
|
PlotDrawSegment( (DRAWSEGMENT*) PtStruct, PLOT_FORMAT_GERBER,
|
|
masque_layer );
|
|
break;
|
|
|
|
case TYPETEXTE:
|
|
PlotTextePcb( (TEXTE_PCB*) PtStruct, PLOT_FORMAT_GERBER,
|
|
masque_layer );
|
|
break;
|
|
|
|
case TYPECOTATION:
|
|
PlotCotation( (COTATION*) PtStruct, PLOT_FORMAT_GERBER,
|
|
masque_layer );
|
|
break;
|
|
|
|
case TYPEMIRE:
|
|
PlotMirePcb( (MIREPCB*) PtStruct, PLOT_FORMAT_GERBER,
|
|
masque_layer );
|
|
break;
|
|
|
|
case TYPEMARKER:
|
|
break;
|
|
|
|
default:
|
|
DisplayError( this, wxT( "Type Draw non gere" ) );
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* Draw footprints shapes without pads (pads will plotted later) */
|
|
nb_items = 0;
|
|
Affiche_1_Parametre( this, 38, wxT( "DrawMod" ), wxEmptyString, GREEN );
|
|
Module = m_Pcb->m_Modules;
|
|
for( ; Module != NULL; Module = (MODULE*) Module->Pnext )
|
|
{
|
|
PtStruct = Module->m_Drawings;
|
|
for( ; PtStruct != NULL; PtStruct = PtStruct->Pnext )
|
|
{
|
|
switch( PtStruct->Type() )
|
|
{
|
|
case TYPEEDGEMODULE:
|
|
if( masque_layer & g_TabOneLayerMask[( (EDGE_MODULE*) PtStruct )->GetLayer()] )
|
|
Plot_1_EdgeModule( PLOT_FORMAT_GERBER, (EDGE_MODULE*) PtStruct );
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Plot footprint pads */
|
|
nb_items = 0;
|
|
Affiche_1_Parametre( this, 48, wxT( "Pads" ), wxEmptyString, GREEN );
|
|
Module = m_Pcb->m_Modules;
|
|
for( ; Module != NULL; Module = (MODULE*) Module->Pnext )
|
|
{
|
|
PtPad = (D_PAD*) Module->m_Pads;
|
|
for( ; PtPad != NULL; PtPad = (D_PAD*) PtPad->Pnext )
|
|
{
|
|
wxPoint shape_pos;
|
|
if( (PtPad->m_Masque_Layer & masque_layer) == 0 )
|
|
continue;
|
|
shape_pos = PtPad->ReturnShapePos();
|
|
pos = shape_pos;
|
|
|
|
size.x = PtPad->m_Size.x + 2 * garde;
|
|
size.y = PtPad->m_Size.y + 2 * garde;
|
|
|
|
/* Don't draw a null size item : */
|
|
if( size.x <= 0 || size.y <= 0 )
|
|
continue;
|
|
|
|
nb_items++;
|
|
|
|
switch( PtPad->m_PadShape )
|
|
{
|
|
case PAD_CIRCLE:
|
|
Plot_1_CIRCLE_pad_GERBER( pos, size.x );
|
|
break;
|
|
|
|
case PAD_OVAL:
|
|
|
|
// Check whether the pad really has a circular shape instead
|
|
if( size.x == size.y )
|
|
Plot_1_CIRCLE_pad_GERBER( pos, size.x );
|
|
else
|
|
trace_1_pastille_OVALE_GERBER( pos, size, PtPad->m_Orient );
|
|
break;
|
|
|
|
case PAD_TRAPEZOID:
|
|
{
|
|
wxSize delta = PtPad->m_DeltaSize;
|
|
trace_1_pad_TRAPEZE_GERBER( pos, size,
|
|
delta, PtPad->m_Orient, FILLED );
|
|
}
|
|
break;
|
|
|
|
case PAD_RECT:
|
|
default:
|
|
PlotRectangularPad_GERBER( pos, size, PtPad->m_Orient );
|
|
break;
|
|
}
|
|
|
|
msg.Printf( wxT( "%d" ), nb_items );
|
|
Affiche_1_Parametre( this, 48, wxEmptyString, msg, GREEN );
|
|
}
|
|
}
|
|
|
|
/* Plot vias : */
|
|
if( tracevia )
|
|
{
|
|
nb_items = 0;
|
|
Affiche_1_Parametre( this, 56, wxT( "Vias" ), wxEmptyString, RED );
|
|
for( track = m_Pcb->m_Track; track != NULL; track = (TRACK*) track->Pnext )
|
|
{
|
|
if( track->Type() != TYPEVIA )
|
|
continue;
|
|
SEGVIA* Via = (SEGVIA*) track;
|
|
|
|
// vias not plotted if not on selected layer, but if layer
|
|
// == SOLDERMASK_LAYER_CU or SOLDERMASK_LAYER_CMP, vias are drawn,
|
|
// if they are on a external copper layer
|
|
int via_mask_layer = Via->ReturnMaskLayer();
|
|
if( via_mask_layer & CUIVRE_LAYER )
|
|
via_mask_layer |= SOLDERMASK_LAYER_CU;
|
|
if( via_mask_layer & CMP_LAYER )
|
|
via_mask_layer |= SOLDERMASK_LAYER_CMP;
|
|
if( ( via_mask_layer & masque_layer) == 0 )
|
|
continue;
|
|
|
|
pos = Via->m_Start;
|
|
size.x = size.y = Via->m_Width + 2 * garde;
|
|
/* Don't draw a null size item : */
|
|
if( size.x <= 0 )
|
|
continue;
|
|
Plot_1_CIRCLE_pad_GERBER( pos, size.x );
|
|
nb_items++;
|
|
msg.Printf( wxT( "%d" ), nb_items );
|
|
Affiche_1_Parametre( this, 56, wxEmptyString, msg, RED );
|
|
}
|
|
}
|
|
/* Plot tracks (not vias) : */
|
|
nb_items = 0;
|
|
Affiche_1_Parametre( this, 64, wxT( "Tracks" ), wxEmptyString, YELLOW );
|
|
|
|
for( track = m_Pcb->m_Track; track != NULL; track = (TRACK*) track->Pnext )
|
|
{
|
|
wxPoint end;
|
|
|
|
if( track->Type() == TYPEVIA )
|
|
continue;
|
|
if( (g_TabOneLayerMask[track->GetLayer()] & masque_layer) == 0 )
|
|
continue;
|
|
|
|
size.x = size.y = track->m_Width;
|
|
pos = track->m_Start;
|
|
end = track->m_End;
|
|
|
|
PlotGERBERLine( pos, end, size.x );
|
|
|
|
nb_items++;
|
|
msg.Printf( wxT( "%d" ), nb_items );
|
|
Affiche_1_Parametre( this, 64, wxEmptyString, msg, YELLOW );
|
|
}
|
|
|
|
/* Plot zones: */
|
|
nb_items = 0;
|
|
if( m_Pcb->m_Zone )
|
|
Affiche_1_Parametre( this, 72, wxT( "Zones " ), wxEmptyString, YELLOW );
|
|
|
|
for( track = m_Pcb->m_Zone; track != NULL; track = (TRACK*) track->Pnext )
|
|
{
|
|
wxPoint end;
|
|
|
|
if( (g_TabOneLayerMask[track->GetLayer()] & masque_layer) == 0 )
|
|
continue;
|
|
|
|
size.x = size.y = track->m_Width;
|
|
pos = track->m_Start;
|
|
end = track->m_End;
|
|
|
|
PlotGERBERLine( pos, end, size.x );
|
|
|
|
nb_items++;
|
|
msg.Printf( wxT( "%d" ), nb_items );
|
|
Affiche_1_Parametre( this, 72, wxEmptyString, msg, YELLOW );
|
|
}
|
|
|
|
/* Plot filled ares */
|
|
for( int ii = 0; ii < m_Pcb->GetAreaCount(); ii++ )
|
|
{
|
|
ZONE_CONTAINER* edge_zone = m_Pcb->GetArea(ii);
|
|
if( ( (1 << edge_zone->GetLayer()) & masque_layer ) == 0 )
|
|
continue;
|
|
PlotFilledAreas(edge_zone, PLOT_FORMAT_GERBER);
|
|
}
|
|
}
|
|
|
|
|
|
/**********************************************************************/
|
|
void trace_1_pastille_OVALE_GERBER( wxPoint pos, wxSize size, int orient )
|
|
/**********************************************************************/
|
|
|
|
/* Trace 1 pastille PAD_OVAL en position pos_X,Y:
|
|
* dimensions dx, dy,
|
|
* orientation orient
|
|
* Pour une orientation verticale ou horizontale, la forme est flashee
|
|
* Pour une orientation quelconque la forme est tracee comme un segment
|
|
*/
|
|
{
|
|
D_CODE* dcode_ptr;
|
|
char cbuf[256];
|
|
int x0, y0, x1, y1, delta;
|
|
|
|
if( orient == 900 || orient == 2700 ) /* orient tournee de 90 deg */
|
|
EXCHG( size.x, size.y );
|
|
|
|
/* Trace de la forme flashee */
|
|
if( orient == 0 || orient == 900 || orient == 1800 || orient == 2700 )
|
|
{
|
|
UserToDeviceCoordinate( pos );
|
|
UserToDeviceSize( size );
|
|
|
|
dcode_ptr = get_D_code( size.x, size.y, GERB_OVALE, 0 );
|
|
if( dcode_ptr->m_NumDcode != s_Last_D_code )
|
|
{
|
|
sprintf( cbuf, "G54D%d*\n", dcode_ptr->m_NumDcode );
|
|
fputs( cbuf, dest );
|
|
s_Last_D_code = dcode_ptr->m_NumDcode;
|
|
}
|
|
sprintf( cbuf, "X%5.5dY%5.5dD03*\n", pos.x, pos.y );
|
|
fputs( cbuf, dest );
|
|
}
|
|
else /* Forme tracee comme un segment */
|
|
{
|
|
if( size.x > size.y )
|
|
{
|
|
EXCHG( size.x, size.y );
|
|
if( orient < 2700 )
|
|
orient += 900;
|
|
else
|
|
orient -= 2700;
|
|
}
|
|
/* la pastille est ramenee a une pastille ovale avec dy > dx */
|
|
delta = size.y - size.x;
|
|
x0 = 0;
|
|
y0 = -delta / 2;
|
|
x1 = 0;
|
|
y1 = delta / 2;
|
|
RotatePoint( &x0, &y0, orient );
|
|
RotatePoint( &x1, &y1, orient );
|
|
PlotGERBERLine( wxPoint( pos.x + x0, pos.y + y0 ),
|
|
wxPoint( pos.x + x1, pos.y + y1 ), size.x );
|
|
}
|
|
}
|
|
|
|
|
|
/******************************************************************/
|
|
void Plot_1_CIRCLE_pad_GERBER( wxPoint pos, int diametre )
|
|
/******************************************************************/
|
|
|
|
/* Plot a circular pad or via at the user position pos
|
|
*/
|
|
{
|
|
D_CODE* dcode_ptr;
|
|
char cbuf[256];
|
|
|
|
wxSize size( diametre, diametre );
|
|
|
|
UserToDeviceCoordinate( pos );
|
|
UserToDeviceSize( size );
|
|
|
|
dcode_ptr = get_D_code( size.x, size.x, GERB_CIRCLE, 0 );
|
|
if( dcode_ptr->m_NumDcode != s_Last_D_code )
|
|
{
|
|
sprintf( cbuf, "G54D%d*\n", dcode_ptr->m_NumDcode );
|
|
fputs( cbuf, dest );
|
|
s_Last_D_code = dcode_ptr->m_NumDcode;
|
|
}
|
|
|
|
sprintf( cbuf, "X%5.5dY%5.5dD03*\n", pos.x, pos.y );
|
|
fputs( cbuf, dest );
|
|
}
|
|
|
|
|
|
/**************************************************************************/
|
|
void PlotRectangularPad_GERBER( wxPoint pos, wxSize size, int orient )
|
|
/**************************************************************************/
|
|
|
|
/* Plot 1 rectangular pad
|
|
* donne par son centre, ses dimensions, et son orientation
|
|
* For a vertical or horizontal shape, the shape is an aperture (Dcode) and it is flashed
|
|
* For others orientations the shape is plotted as a polygon
|
|
*/
|
|
{
|
|
D_CODE* dcode_ptr;
|
|
char cbuf[256];
|
|
|
|
/* Trace de la forme flashee */
|
|
switch( orient )
|
|
{
|
|
case 900:
|
|
case 2700: /* la rotation de 90 ou 270 degres revient a permutter des dimensions */
|
|
EXCHG( size.x, size.y );
|
|
|
|
// Pass through
|
|
|
|
case 0:
|
|
case 1800:
|
|
UserToDeviceCoordinate( pos );
|
|
UserToDeviceSize( size );
|
|
|
|
dcode_ptr = get_D_code( size.x, size.y, GERB_RECT, 0 );
|
|
if( dcode_ptr->m_NumDcode != s_Last_D_code )
|
|
{
|
|
sprintf( cbuf, "G54D%d*\n", dcode_ptr->m_NumDcode );
|
|
fputs( cbuf, dest );
|
|
s_Last_D_code = dcode_ptr->m_NumDcode;
|
|
}
|
|
sprintf( cbuf, "X%5.5dY%5.5dD03*\n", pos.x, pos.y );
|
|
fputs( cbuf, dest );
|
|
break;
|
|
|
|
default: /* plot pad shape as polygon */
|
|
trace_1_pad_TRAPEZE_GERBER( pos, size, wxSize( 0, 0 ), orient, FILLED );
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
/*****************************************************************/
|
|
void trace_1_contour_GERBER( wxPoint pos, wxSize size, wxSize delta,
|
|
int penwidth, int orient )
|
|
/*****************************************************************/
|
|
|
|
/* Trace 1 contour rectangulaire ou trapezoidal d'orientation quelconque
|
|
* donne par son centre,
|
|
* ses dimensions ,
|
|
* ses variations ,
|
|
* l'epaisseur du trait,
|
|
* et son orientation orient
|
|
*/
|
|
{
|
|
int ii;
|
|
wxPoint coord[4];
|
|
|
|
size.x /= 2;
|
|
size.y /= 2;
|
|
delta.x /= 2;
|
|
delta.y /= 2; /* demi dim dx et dy */
|
|
|
|
coord[0].x = pos.x - size.x - delta.y;
|
|
coord[0].y = pos.y + size.y + delta.x;
|
|
|
|
coord[1].x = pos.x - size.x + delta.y;
|
|
coord[1].y = pos.y - size.y - delta.x;
|
|
|
|
coord[2].x = pos.x + size.x - delta.y;
|
|
coord[2].y = pos.y - size.y + delta.x;
|
|
|
|
coord[3].x = pos.x + size.x + delta.y;
|
|
coord[3].y = pos.y + size.y - delta.x;
|
|
|
|
for( ii = 0; ii < 4; ii++ )
|
|
{
|
|
RotatePoint( &coord[ii].x, &coord[ii].y, pos.x, pos.y, orient );
|
|
}
|
|
|
|
PlotGERBERLine( coord[0], coord[1], penwidth );
|
|
PlotGERBERLine( coord[1], coord[2], penwidth );
|
|
PlotGERBERLine( coord[2], coord[3], penwidth );
|
|
PlotGERBERLine( coord[3], coord[0], penwidth );
|
|
}
|
|
|
|
|
|
/*******************************************************************/
|
|
void trace_1_pad_TRAPEZE_GERBER( wxPoint pos, wxSize size, wxSize delta,
|
|
int orient, int modetrace )
|
|
/*******************************************************************/
|
|
|
|
/* Trace 1 pad trapezoidal donne par :
|
|
* son centre pos.x,pos.y
|
|
* ses dimensions size.x et size.y
|
|
* les variations delta.x et delta.y ( 1 des deux au moins doit etre nulle)
|
|
* son orientation orient en 0.1 degres
|
|
* le mode de trace (FILLED, SKETCH, FILAIRE)
|
|
*
|
|
* Le trace n'est fait que pour un trapeze, c.a.d que delta.x ou delta.y
|
|
* = 0.
|
|
*
|
|
* les notation des sommets sont ( vis a vis de la table tracante )
|
|
*
|
|
* " 0 ------------- 3 "
|
|
* " . . "
|
|
* " . O . "
|
|
* " . . "
|
|
* " 1 ---- 2 "
|
|
*
|
|
*
|
|
* exemple de Disposition pour delta.y > 0, delta.x = 0
|
|
* " 1 ---- 2 "
|
|
* " . . "
|
|
* " . O . "
|
|
* " . . "
|
|
* " 0 ------------- 3 "
|
|
*
|
|
*
|
|
* exemple de Disposition pour delta.y = 0, delta.x > 0
|
|
* " 0 "
|
|
* " . . "
|
|
* " . . "
|
|
* " . 3 "
|
|
* " . . "
|
|
* " . O . "
|
|
* " . . "
|
|
* " . 2 "
|
|
* " . . "
|
|
* " . . "
|
|
* " 1 "
|
|
*/
|
|
{
|
|
int ii, jj;
|
|
int dx, dy;
|
|
wxPoint polygon[4]; /* polygon corners */
|
|
int coord[8];
|
|
int ddx, ddy;
|
|
|
|
/* calcul des dimensions optimales du spot choisi = 1/4 plus petite dim */
|
|
dx = size.x - abs( delta.y );
|
|
dy = size.y - abs( delta.x );
|
|
|
|
dx = size.x / 2;
|
|
dy = size.y / 2;
|
|
ddx = delta.x / 2;
|
|
ddy = delta.y / 2;
|
|
|
|
polygon[0].x = -dx - ddy;
|
|
polygon[0].y = +dy + ddx;
|
|
polygon[1].x = -dx + ddy;
|
|
polygon[1].y = -dy - ddx;
|
|
polygon[2].x = +dx - ddy;
|
|
polygon[2].y = -dy + ddx;
|
|
polygon[3].x = +dx + ddy;
|
|
polygon[3].y = +dy - ddx;
|
|
|
|
/* Dessin du polygone et Remplissage eventuel de l'interieur */
|
|
|
|
for( ii = 0, jj = 0; ii < 4; ii++ )
|
|
{
|
|
RotatePoint( &polygon[ii].x, &polygon[ii].y, orient );
|
|
coord[jj] = polygon[ii].x += pos.x;
|
|
jj++;
|
|
coord[jj] = polygon[ii].y += pos.y;
|
|
jj++;
|
|
}
|
|
|
|
if( modetrace != FILLED )
|
|
{
|
|
int plotLine_width = (int) (10 * g_PlotLine_Width * Gerb_scale_plot);
|
|
PlotGERBERLine( polygon[0], polygon[1], plotLine_width );
|
|
PlotGERBERLine( polygon[1], polygon[2], plotLine_width );
|
|
PlotGERBERLine( polygon[2], polygon[3], plotLine_width );
|
|
PlotGERBERLine( polygon[3], polygon[0], plotLine_width );
|
|
}
|
|
else
|
|
PlotFilledPolygon_GERBER( 4, coord );
|
|
}
|
|
|
|
|
|
/**********************************************************/
|
|
void PlotGERBERLine( wxPoint start, wxPoint end, int width )
|
|
/**********************************************************/
|
|
|
|
/* Trace 1 segment de piste :
|
|
*/
|
|
{
|
|
D_CODE* dcode_ptr;
|
|
char cbuf[256];
|
|
|
|
UserToDeviceCoordinate( start );
|
|
UserToDeviceCoordinate( end );
|
|
|
|
dcode_ptr = get_D_code( width, width, GERB_LINE, 0 );
|
|
if( dcode_ptr->m_NumDcode != s_Last_D_code )
|
|
{
|
|
sprintf( cbuf, "G54D%d*\n", dcode_ptr->m_NumDcode );
|
|
fputs( cbuf, dest );
|
|
s_Last_D_code = dcode_ptr->m_NumDcode;
|
|
}
|
|
sprintf( cbuf, "X%5.5dY%5.5dD02*\n", start.x, start.y );
|
|
fputs( cbuf, dest );
|
|
sprintf( cbuf, "X%5.5dY%5.5dD01*\n", end.x, end.y );
|
|
fputs( cbuf, dest );
|
|
}
|
|
|
|
|
|
/********************************************************************/
|
|
void PlotCircle_GERBER( wxPoint centre, int rayon, int epaisseur )
|
|
/********************************************************************/
|
|
|
|
/* Plot one cercle as segments (6 to 16 depending on its radius
|
|
*/
|
|
{
|
|
int ii;
|
|
int ox, oy, fx, fy;
|
|
int delta; /* increment (en 0.1 degres) angulaire pour trace de cercles */
|
|
|
|
delta = 120; /* un cercle sera trace en 3600/delta segments */
|
|
/* Correction pour petits cercles par rapport a l'epaisseur du trait */
|
|
if( rayon < (epaisseur * 10) )
|
|
delta = 225; /* 16 segm pour 360 deg */
|
|
if( rayon < (epaisseur * 5) )
|
|
delta = 300; /* 12 segm pour 360 deg */
|
|
if( rayon < (epaisseur * 2) )
|
|
delta = 600; /* 6 segm pour 360 deg */
|
|
|
|
ox = centre.x + rayon;
|
|
oy = centre.y;
|
|
for( ii = delta; ii < 3600; ii += delta )
|
|
{
|
|
fx = centre.x + (int) (rayon * fcosinus[ii]);
|
|
fy = centre.y + (int) (rayon * fsinus[ii]);
|
|
PlotGERBERLine( wxPoint( ox, oy ), wxPoint( fx, fy ), epaisseur );
|
|
ox = fx;
|
|
oy = fy;
|
|
}
|
|
|
|
fx = centre.x + rayon;
|
|
fy = centre.y;
|
|
PlotGERBERLine( wxPoint( ox, oy ), wxPoint( fx, fy ), epaisseur );
|
|
}
|
|
|
|
|
|
/***************************************************************/
|
|
void PlotFilledPolygon_GERBER( int nb_segm, int* coord )
|
|
/***************************************************************/
|
|
{
|
|
int ii;
|
|
wxPoint pos, startpos;
|
|
|
|
fputs( "G36*\n", dest );
|
|
pos.x = *coord;
|
|
coord++;
|
|
pos.y = *coord;
|
|
coord++;
|
|
UserToDeviceCoordinate( pos );
|
|
startpos = pos;
|
|
fprintf( dest, "X%5.5dY%5.5dD02*\n", pos.x, pos.y );
|
|
for( ii = 1; ii < nb_segm; ii++ )
|
|
{
|
|
pos.x = *coord;
|
|
coord++;
|
|
pos.y = *coord;
|
|
coord++;
|
|
UserToDeviceCoordinate( pos );
|
|
fprintf( dest, "X%5.5dY%5.5dD01*\n", pos.x, pos.y );
|
|
}
|
|
|
|
fprintf( dest, "X%5.5dY%5.5dD01*\n", startpos.x, startpos.y );
|
|
fputs( "G37*\n", dest );
|
|
}
|
|
|
|
|
|
/***************************************************************/
|
|
void PlotPolygon_GERBER( int nb_segm, int* coord, int width )
|
|
/***************************************************************/
|
|
{
|
|
wxPoint start, end, startpoint;
|
|
startpoint.x = *coord++;
|
|
startpoint.y = *coord++;
|
|
start = startpoint;
|
|
for( int ii = 0; ii < nb_segm-1; ii++ )
|
|
{
|
|
end.x = *coord;
|
|
coord++;
|
|
end.y = *coord;
|
|
coord++;
|
|
PlotGERBERLine(start, end, width );
|
|
start = end;
|
|
}
|
|
|
|
if ( startpoint != end ) // Close poly
|
|
PlotGERBERLine(end, startpoint, width );
|
|
|
|
}
|
|
|
|
|
|
/*******************************************************/
|
|
D_CODE* get_D_code( int dx, int dy, int type, int drill )
|
|
/*******************************************************/
|
|
|
|
/* Fonction Recherchant et Creant eventuellement la description
|
|
* du D_CODE du type et dimensions demandees
|
|
*/
|
|
{
|
|
D_CODE* ptr_tool, * last_dcode_ptr;
|
|
int num_new_D_code = FIRST_DCODE_VALUE;
|
|
|
|
|
|
ptr_tool = last_dcode_ptr = s_DCodeList;
|
|
|
|
while( ptr_tool && ptr_tool->m_Type >= 0 )
|
|
{
|
|
if( ( ptr_tool->m_Size.x == dx )
|
|
&& ( ptr_tool->m_Size.y == dy )
|
|
&& ( ptr_tool->m_Type == type ) )
|
|
return ptr_tool; /* D_code deja existant */
|
|
last_dcode_ptr = ptr_tool;
|
|
ptr_tool = ptr_tool->m_Pnext;
|
|
num_new_D_code++;
|
|
}
|
|
|
|
/* At this point, the requested D_CODE does not exist: It will be created */
|
|
if( ptr_tool == NULL ) /* We must create a new data */
|
|
{
|
|
ptr_tool = new D_CODE();
|
|
|
|
ptr_tool->m_NumDcode = num_new_D_code;
|
|
if( last_dcode_ptr )
|
|
{
|
|
ptr_tool->m_Pback = last_dcode_ptr;
|
|
last_dcode_ptr->m_Pnext = ptr_tool;
|
|
}
|
|
else
|
|
s_DCodeList = ptr_tool;
|
|
}
|
|
ptr_tool->m_Size.x = dx;
|
|
ptr_tool->m_Size.y = dy;
|
|
ptr_tool->m_Type = type;
|
|
return ptr_tool;
|
|
}
|
|
|
|
|
|
/******************************************************************/
|
|
void Init_Trace_GERBER( WinEDA_BasePcbFrame* frame, FILE* gerbfile )
|
|
/******************************************************************/
|
|
{
|
|
char Line[1024];
|
|
|
|
s_Last_D_code = 0;
|
|
|
|
DateAndTime( Line );
|
|
wxString Title = g_Main_Title + wxT( " " ) + GetBuildVersion();
|
|
fprintf( gerbfile, "G04 (Genere par %s) le %s*\n", CONV_TO_UTF8( Title ), Line );
|
|
|
|
// Specify linear interpol (G01), unit = INCH (G70), abs format (G90):
|
|
fputs( "G01*\nG70*\nG90*\n", gerbfile );
|
|
fputs( "%MOIN*%\n", gerbfile ); // set unites = INCHES
|
|
|
|
/* Set gerber format to 3.4 */
|
|
strcpy( Line, "G04 Gerber Fmt 3.4, Leading zero omitted, Abs format*\n%FSLAX34Y34*%\n" );
|
|
|
|
fputs( Line, gerbfile );
|
|
|
|
fputs( "G04 APERTURE LIST*\n", gerbfile );
|
|
}
|
|
|
|
|
|
/***********************************/
|
|
static void Init_ApertureList()
|
|
/***********************************/
|
|
|
|
/* Init the memory to handle the aperture list:
|
|
* the member .m_Type is used by get_D_code() to handle the end of list:
|
|
* .m_Type < 0 is the first free aperture descr
|
|
*/
|
|
{
|
|
D_CODE* ptr_tool;
|
|
|
|
ptr_tool = s_DCodeList;
|
|
while( ptr_tool )
|
|
{
|
|
s_DCodeList->m_Type = -1;
|
|
ptr_tool = ptr_tool->m_Pnext;
|
|
}
|
|
|
|
ShowDcodeError = TRUE;
|
|
}
|
|
|
|
|
|
/*****************************************************************/
|
|
void Fin_Trace_GERBER( WinEDA_BasePcbFrame* frame, FILE* gerbfile )
|
|
/*****************************************************************/
|
|
{
|
|
char line[1024];
|
|
wxString TmpFileName, msg;
|
|
FILE* outfile;
|
|
|
|
fputs( "M02*\n", gerbfile );
|
|
fclose( gerbfile );
|
|
|
|
// Reouverture gerbfile pour ajout des Apertures
|
|
gerbfile = wxFopen( GerberFullFileName, wxT( "rt" ) );
|
|
if( gerbfile == NULL )
|
|
{
|
|
msg.Printf( _( "unable to reopen file <%s>" ), GerberFullFileName.GetData() );
|
|
DisplayError( frame, msg );
|
|
return;
|
|
}
|
|
|
|
// Ouverture tmpfile
|
|
TmpFileName = GerberFullFileName + wxT( ".$$$" );
|
|
outfile = wxFopen( TmpFileName, wxT( "wt" ) );
|
|
if( outfile == NULL )
|
|
{
|
|
fclose( gerbfile );
|
|
DisplayError( frame, wxT( "Fin_Trace_GERBER(): Can't Open tmp file" ) );
|
|
return;
|
|
}
|
|
|
|
// Placement des Apertures en RS274X
|
|
rewind( gerbfile );
|
|
while( fgets( line, 1024, gerbfile ) )
|
|
{
|
|
fputs( line, outfile );
|
|
if( strcmp( strtok( line, "\n\r" ), "G04 APERTURE LIST*" ) == 0 )
|
|
{
|
|
frame->Gen_D_CODE_File( outfile );
|
|
fputs( "G04 APERTURE END LIST*\n", outfile );
|
|
}
|
|
}
|
|
|
|
fclose( outfile );
|
|
fclose( gerbfile );
|
|
wxRemoveFile( GerberFullFileName );
|
|
wxRenameFile( TmpFileName, GerberFullFileName );
|
|
}
|
|
|
|
|
|
/******************************************************/
|
|
int WinEDA_BasePcbFrame::Gen_D_CODE_File( FILE* penfile )
|
|
/******************************************************/
|
|
|
|
/* Genere la liste courante des D_CODES
|
|
* Retourne le nombre de D_Codes utilises
|
|
* Genere une sequence RS274X
|
|
*/
|
|
{
|
|
D_CODE* ptr_tool;
|
|
char cbuf[1024];
|
|
int nb_dcodes = 0;
|
|
|
|
/* Init : */
|
|
ptr_tool = s_DCodeList;
|
|
|
|
while( ptr_tool && ( ptr_tool->m_Type >= 0 ) )
|
|
{
|
|
float fscale = 0.0001; // For 3.4 format
|
|
char* text;
|
|
sprintf( cbuf, "%%ADD%d", ptr_tool->m_NumDcode );
|
|
text = cbuf + strlen( cbuf );
|
|
|
|
switch( ptr_tool->m_Type )
|
|
{
|
|
case 1: // Circle (flash )
|
|
sprintf( text, "C,%f*%%\n", ptr_tool->m_Size.x * fscale );
|
|
break;
|
|
|
|
case 2: // PAD_RECT
|
|
sprintf( text, "R,%fX%f*%%\n", ptr_tool->m_Size.x * fscale,
|
|
ptr_tool->m_Size.y * fscale );
|
|
break;
|
|
|
|
case 3: // Circle ( lines )
|
|
sprintf( text, "C,%f*%%\n", ptr_tool->m_Size.x * fscale );
|
|
break;
|
|
|
|
case 4: // PAD_OVAL
|
|
sprintf( text, "O,%fX%f*%%\n", ptr_tool->m_Size.x * fscale,
|
|
ptr_tool->m_Size.y * fscale );
|
|
break;
|
|
|
|
default:
|
|
DisplayError( this, wxT( "Gen_D_CODE_File(): Dcode Type err" ) );
|
|
break;
|
|
}
|
|
|
|
fputs( cbuf, penfile );
|
|
ptr_tool = ptr_tool->m_Pnext;
|
|
nb_dcodes++;
|
|
}
|
|
|
|
return nb_dcodes;
|
|
}
|