Apply vrml_layer_pth, from Cirilo Bernardo
This commit is contained in:
parent
52d2e7eb59
commit
a91eabb805
|
@ -25,37 +25,6 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
* NOTE:
|
|
||||||
* 1. for improved looks, create a DRILL layer for PTH drills.
|
|
||||||
* To render the improved board, render the vertical outline only
|
|
||||||
* for the board (no added drill holes), then render the
|
|
||||||
* outline only for PTH, and finally render the top and bottom
|
|
||||||
* of the board. NOTE: if we don't want extra eye-candy then
|
|
||||||
* we must maintain the current board export.
|
|
||||||
* Additional bits needed for improved eyecandy:
|
|
||||||
* + CalcOutline: calculates only the outline of a VRML_LAYER or
|
|
||||||
* a VERTICAL_HOLES
|
|
||||||
* + WriteVerticalIndices: writes the indices of only the vertical
|
|
||||||
* facets of a VRML_LAYER or a VRML_HOLES.
|
|
||||||
* + WriteVerticalVertices: writes only the outline vertices to
|
|
||||||
* form vertical walls; applies to VRML_LAYER and VRML_HOLES
|
|
||||||
*
|
|
||||||
* 2. How can we suppress fiducials such as those in the corners of the pic-programmer demo?
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
* KNOWN BUGS:
|
|
||||||
* 1. silk outlines are sometimes mangled; this is somehow due to
|
|
||||||
* many overlapping segments. This does not happen in 3Dviewer
|
|
||||||
* so it is worth inspecting that code to see what is different.
|
|
||||||
*
|
|
||||||
* These artefacts can be suppressed for exploratory purposes by
|
|
||||||
* removing the line width parameter from the length calculation in
|
|
||||||
* export_vrml_line()
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <fctsys.h>
|
#include <fctsys.h>
|
||||||
#include <kicad_string.h>
|
#include <kicad_string.h>
|
||||||
#include <wxPcbStruct.h>
|
#include <wxPcbStruct.h>
|
||||||
|
@ -88,7 +57,7 @@
|
||||||
#define MIN_VRML_LINEWIDTH 0.12
|
#define MIN_VRML_LINEWIDTH 0.12
|
||||||
|
|
||||||
// offset for art layers, mm (silk, paste, etc)
|
// offset for art layers, mm (silk, paste, etc)
|
||||||
#define ART_OFFSET 0.02
|
#define ART_OFFSET 0.025
|
||||||
|
|
||||||
/* helper function:
|
/* helper function:
|
||||||
* some characters cannot be used in names,
|
* some characters cannot be used in names,
|
||||||
|
@ -183,6 +152,7 @@ public:
|
||||||
VRML_LAYER bot_silk;
|
VRML_LAYER bot_silk;
|
||||||
VRML_LAYER top_tin;
|
VRML_LAYER top_tin;
|
||||||
VRML_LAYER bot_tin;
|
VRML_LAYER bot_tin;
|
||||||
|
VRML_LAYER plated_holes;
|
||||||
|
|
||||||
double scale; // board internal units to output scaling
|
double scale; // board internal units to output scaling
|
||||||
double minLineWidth; // minimum width of a VRML line segment
|
double minLineWidth; // minimum width of a VRML line segment
|
||||||
|
@ -229,8 +199,18 @@ public:
|
||||||
|
|
||||||
void SetOffset( double aXoff, double aYoff )
|
void SetOffset( double aXoff, double aYoff )
|
||||||
{
|
{
|
||||||
tx = aXoff;
|
tx = aXoff;
|
||||||
ty = aYoff;
|
ty = -aYoff;
|
||||||
|
|
||||||
|
holes.SetVertexOffsets( aXoff, aYoff );
|
||||||
|
board.SetVertexOffsets( aXoff, aYoff );
|
||||||
|
top_copper.SetVertexOffsets( aXoff, aYoff );
|
||||||
|
bot_copper.SetVertexOffsets( aXoff, aYoff );
|
||||||
|
top_silk.SetVertexOffsets( aXoff, aYoff );
|
||||||
|
bot_silk.SetVertexOffsets( aXoff, aYoff );
|
||||||
|
top_tin.SetVertexOffsets( aXoff, aYoff );
|
||||||
|
bot_tin.SetVertexOffsets( aXoff, aYoff );
|
||||||
|
plated_holes.SetVertexOffsets( aXoff, aYoff );
|
||||||
}
|
}
|
||||||
|
|
||||||
double GetLayerZ( LAYER_NUM aLayer )
|
double GetLayerZ( LAYER_NUM aLayer )
|
||||||
|
@ -278,6 +258,7 @@ public:
|
||||||
bot_silk.SetArcParams( iMaxSeg, smin, smax );
|
bot_silk.SetArcParams( iMaxSeg, smin, smax );
|
||||||
top_tin.SetArcParams( iMaxSeg, smin, smax );
|
top_tin.SetArcParams( iMaxSeg, smin, smax );
|
||||||
bot_tin.SetArcParams( iMaxSeg, smin, smax );
|
bot_tin.SetArcParams( iMaxSeg, smin, smax );
|
||||||
|
plated_holes.SetArcParams( iMaxSeg, smin, smax );
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -455,6 +436,16 @@ static void write_layers( MODEL_VRML& aModel, std::ofstream& output_file, BOARD*
|
||||||
- Millimeter2iu( ART_OFFSET / 2.0 ) * aModel.scale,
|
- Millimeter2iu( ART_OFFSET / 2.0 ) * aModel.scale,
|
||||||
0, aModel.precision );
|
0, aModel.precision );
|
||||||
|
|
||||||
|
// VRML_LAYER PTH;
|
||||||
|
aModel.plated_holes.Tesselate( NULL, true );
|
||||||
|
write_triangle_bag( output_file, aModel.GetColor( VRML_COLOR_TIN ),
|
||||||
|
&aModel.plated_holes, false, false,
|
||||||
|
aModel.GetLayerZ( LAST_COPPER_LAYER )
|
||||||
|
+ Millimeter2iu( ART_OFFSET / 2.0 ) * aModel.scale,
|
||||||
|
aModel.GetLayerZ( FIRST_COPPER_LAYER )
|
||||||
|
- Millimeter2iu( ART_OFFSET / 2.0 ) * aModel.scale,
|
||||||
|
aModel.precision );
|
||||||
|
|
||||||
// VRML_LAYER top_silk;
|
// VRML_LAYER top_silk;
|
||||||
aModel.top_silk.Tesselate( &aModel.holes );
|
aModel.top_silk.Tesselate( &aModel.holes );
|
||||||
write_triangle_bag( output_file, aModel.GetColor( VRML_COLOR_SILK ),
|
write_triangle_bag( output_file, aModel.GetColor( VRML_COLOR_SILK ),
|
||||||
|
@ -588,10 +579,10 @@ static void export_vrml_drawsegment( MODEL_VRML& aModel, DRAWSEGMENT* drawseg )
|
||||||
{
|
{
|
||||||
LAYER_NUM layer = drawseg->GetLayer();
|
LAYER_NUM layer = drawseg->GetLayer();
|
||||||
double w = drawseg->GetWidth() * aModel.scale;
|
double w = drawseg->GetWidth() * aModel.scale;
|
||||||
double x = drawseg->GetStart().x * aModel.scale + aModel.tx;
|
double x = drawseg->GetStart().x * aModel.scale;
|
||||||
double y = drawseg->GetStart().y * aModel.scale + aModel.ty;
|
double y = drawseg->GetStart().y * aModel.scale;
|
||||||
double xf = drawseg->GetEnd().x * aModel.scale + aModel.tx;
|
double xf = drawseg->GetEnd().x * aModel.scale;
|
||||||
double yf = drawseg->GetEnd().y * aModel.scale + aModel.ty;
|
double yf = drawseg->GetEnd().y * aModel.scale;
|
||||||
|
|
||||||
// Items on the edge layer are handled elsewhere; just return
|
// Items on the edge layer are handled elsewhere; just return
|
||||||
if( layer == EDGE_N )
|
if( layer == EDGE_N )
|
||||||
|
@ -626,12 +617,10 @@ static void vrml_text_callback( int x0, int y0, int xf, int yf )
|
||||||
LAYER_NUM s_text_layer = model_vrml->s_text_layer;
|
LAYER_NUM s_text_layer = model_vrml->s_text_layer;
|
||||||
int s_text_width = model_vrml->s_text_width;
|
int s_text_width = model_vrml->s_text_width;
|
||||||
double scale = model_vrml->scale;
|
double scale = model_vrml->scale;
|
||||||
double tx = model_vrml->tx;
|
|
||||||
double ty = model_vrml->ty;
|
|
||||||
|
|
||||||
export_vrml_line( *model_vrml, s_text_layer,
|
export_vrml_line( *model_vrml, s_text_layer,
|
||||||
x0 * scale + tx, y0 * scale + ty,
|
x0 * scale, y0 * scale,
|
||||||
xf * scale + tx, yf * scale + ty,
|
xf * scale, yf * scale,
|
||||||
s_text_width * scale );
|
s_text_width * scale );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -729,8 +718,6 @@ static void export_vrml_board( MODEL_VRML& aModel, BOARD* pcb )
|
||||||
}
|
}
|
||||||
|
|
||||||
double scale = aModel.scale;
|
double scale = aModel.scale;
|
||||||
double dx = aModel.tx;
|
|
||||||
double dy = aModel.ty;
|
|
||||||
|
|
||||||
int i = 0;
|
int i = 0;
|
||||||
int seg;
|
int seg;
|
||||||
|
@ -756,8 +743,8 @@ static void export_vrml_board( MODEL_VRML& aModel, BOARD* pcb )
|
||||||
if( bufferPcbOutlines[i].end_contour )
|
if( bufferPcbOutlines[i].end_contour )
|
||||||
break;
|
break;
|
||||||
|
|
||||||
aModel.board.AddVertex( seg, bufferPcbOutlines[i].x * scale + dx,
|
aModel.board.AddVertex( seg, bufferPcbOutlines[i].x * scale,
|
||||||
-(bufferPcbOutlines[i].y * scale + dy) );
|
-(bufferPcbOutlines[i].y * scale ) );
|
||||||
|
|
||||||
++i;
|
++i;
|
||||||
}
|
}
|
||||||
|
@ -788,8 +775,8 @@ static void export_vrml_board( MODEL_VRML& aModel, BOARD* pcb )
|
||||||
if( allLayerHoles[i].end_contour )
|
if( allLayerHoles[i].end_contour )
|
||||||
break;
|
break;
|
||||||
|
|
||||||
aModel.holes.AddVertex( seg, allLayerHoles[i].x * scale + dx,
|
aModel.holes.AddVertex( seg, allLayerHoles[i].x * scale,
|
||||||
-(allLayerHoles[i].y * scale + dy) );
|
-(allLayerHoles[i].y * scale ) );
|
||||||
|
|
||||||
++i;
|
++i;
|
||||||
}
|
}
|
||||||
|
@ -849,8 +836,8 @@ static void export_vrml_via( MODEL_VRML& aModel, BOARD* pcb, const VIA* via )
|
||||||
|
|
||||||
hole = via->GetDrillValue() * aModel.scale / 2.0;
|
hole = via->GetDrillValue() * aModel.scale / 2.0;
|
||||||
r = via->GetWidth() * aModel.scale / 2.0;
|
r = via->GetWidth() * aModel.scale / 2.0;
|
||||||
x = via->GetStart().x * aModel.scale + aModel.tx;
|
x = via->GetStart().x * aModel.scale;
|
||||||
y = via->GetStart().y * aModel.scale + aModel.ty;
|
y = via->GetStart().y * aModel.scale;
|
||||||
via->LayerPair( &top_layer, &bottom_layer );
|
via->LayerPair( &top_layer, &bottom_layer );
|
||||||
|
|
||||||
// do not render a buried via
|
// do not render a buried via
|
||||||
|
@ -873,10 +860,10 @@ static void export_vrml_tracks( MODEL_VRML& aModel, BOARD* pcb )
|
||||||
else if( track->GetLayer() == FIRST_COPPER_LAYER
|
else if( track->GetLayer() == FIRST_COPPER_LAYER
|
||||||
|| track->GetLayer() == LAST_COPPER_LAYER )
|
|| track->GetLayer() == LAST_COPPER_LAYER )
|
||||||
export_vrml_line( aModel, track->GetLayer(),
|
export_vrml_line( aModel, track->GetLayer(),
|
||||||
track->GetStart().x * aModel.scale + aModel.tx,
|
track->GetStart().x * aModel.scale,
|
||||||
track->GetStart().y * aModel.scale + aModel.ty,
|
track->GetStart().y * aModel.scale,
|
||||||
track->GetEnd().x * aModel.scale + aModel.tx,
|
track->GetEnd().x * aModel.scale,
|
||||||
track->GetEnd().y * aModel.scale + aModel.ty,
|
track->GetEnd().y * aModel.scale,
|
||||||
track->GetWidth() * aModel.scale );
|
track->GetWidth() * aModel.scale );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -886,9 +873,6 @@ static void export_vrml_zones( MODEL_VRML& aModel, BOARD* aPcb )
|
||||||
{
|
{
|
||||||
|
|
||||||
double scale = aModel.scale;
|
double scale = aModel.scale;
|
||||||
double dx = aModel.tx;
|
|
||||||
double dy = aModel.ty;
|
|
||||||
|
|
||||||
double x, y;
|
double x, y;
|
||||||
|
|
||||||
for( int ii = 0; ii < aPcb->GetAreaCount(); ii++ )
|
for( int ii = 0; ii < aPcb->GetAreaCount(); ii++ )
|
||||||
|
@ -920,8 +904,8 @@ static void export_vrml_zones( MODEL_VRML& aModel, BOARD* aPcb )
|
||||||
|
|
||||||
while( i < nvert )
|
while( i < nvert )
|
||||||
{
|
{
|
||||||
x = poly.GetX(i) * scale + dx;
|
x = poly.GetX(i) * scale;
|
||||||
y = -(poly.GetY(i) * scale + dy);
|
y = -(poly.GetY(i) * scale);
|
||||||
|
|
||||||
if( poly.IsEndContour(i) )
|
if( poly.IsEndContour(i) )
|
||||||
break;
|
break;
|
||||||
|
@ -971,10 +955,10 @@ static void export_vrml_edge_module( MODEL_VRML& aModel, EDGE_MODULE* aOutline,
|
||||||
double aOrientation )
|
double aOrientation )
|
||||||
{
|
{
|
||||||
LAYER_NUM layer = aOutline->GetLayer();
|
LAYER_NUM layer = aOutline->GetLayer();
|
||||||
double x = aOutline->GetStart().x * aModel.scale + aModel.tx;
|
double x = aOutline->GetStart().x * aModel.scale;
|
||||||
double y = aOutline->GetStart().y * aModel.scale + aModel.ty;
|
double y = aOutline->GetStart().y * aModel.scale;
|
||||||
double xf = aOutline->GetEnd().x * aModel.scale + aModel.tx;
|
double xf = aOutline->GetEnd().x * aModel.scale;
|
||||||
double yf = aOutline->GetEnd().y * aModel.scale + aModel.ty;
|
double yf = aOutline->GetEnd().y * aModel.scale;
|
||||||
double w = aOutline->GetWidth() * aModel.scale;
|
double w = aOutline->GetWidth() * aModel.scale;
|
||||||
|
|
||||||
switch( aOutline->GetShape() )
|
switch( aOutline->GetShape() )
|
||||||
|
@ -1015,8 +999,8 @@ static void export_vrml_edge_module( MODEL_VRML& aModel, EDGE_MODULE* aOutline,
|
||||||
corner.x += aOutline->GetPosition().x;
|
corner.x += aOutline->GetPosition().x;
|
||||||
corner.y += aOutline->GetPosition().y;
|
corner.y += aOutline->GetPosition().y;
|
||||||
|
|
||||||
x = corner.x * aModel.scale + aModel.tx;
|
x = corner.x * aModel.scale;
|
||||||
y = - ( corner.y * aModel.scale + aModel.ty );
|
y = - ( corner.y * aModel.scale );
|
||||||
|
|
||||||
if( !vl->AddVertex( seg, x, y ) )
|
if( !vl->AddVertex( seg, x, y ) )
|
||||||
throw( std::runtime_error( vl->GetError() ) );
|
throw( std::runtime_error( vl->GetError() ) );
|
||||||
|
@ -1037,8 +1021,8 @@ static void export_vrml_padshape( MODEL_VRML& aModel, VRML_LAYER* aTinLayer, D_P
|
||||||
{
|
{
|
||||||
// The (maybe offset) pad position
|
// The (maybe offset) pad position
|
||||||
wxPoint pad_pos = aPad->ShapePos();
|
wxPoint pad_pos = aPad->ShapePos();
|
||||||
double pad_x = pad_pos.x * aModel.scale + aModel.tx;
|
double pad_x = pad_pos.x * aModel.scale;
|
||||||
double pad_y = pad_pos.y * aModel.scale + aModel.ty;
|
double pad_y = pad_pos.y * aModel.scale;
|
||||||
wxSize pad_delta = aPad->GetDelta();
|
wxSize pad_delta = aPad->GetDelta();
|
||||||
|
|
||||||
double pad_dx = pad_delta.x * aModel.scale / 2.0;
|
double pad_dx = pad_delta.x * aModel.scale / 2.0;
|
||||||
|
@ -1121,22 +1105,36 @@ static void export_vrml_pad( MODEL_VRML& aModel, BOARD* pcb, D_PAD* aPad )
|
||||||
double hole_drill_w = (double) aPad->GetDrillSize().x * aModel.scale / 2.0;
|
double hole_drill_w = (double) aPad->GetDrillSize().x * aModel.scale / 2.0;
|
||||||
double hole_drill_h = (double) aPad->GetDrillSize().y * aModel.scale / 2.0;
|
double hole_drill_h = (double) aPad->GetDrillSize().y * aModel.scale / 2.0;
|
||||||
double hole_drill = std::min( hole_drill_w, hole_drill_h );
|
double hole_drill = std::min( hole_drill_w, hole_drill_h );
|
||||||
double hole_x = aPad->GetPosition().x * aModel.scale + aModel.tx;
|
double hole_x = aPad->GetPosition().x * aModel.scale;
|
||||||
double hole_y = aPad->GetPosition().y * aModel.scale + aModel.ty;
|
double hole_y = aPad->GetPosition().y * aModel.scale;
|
||||||
|
|
||||||
// Export the hole on the edge layer
|
// Export the hole on the edge layer
|
||||||
if( hole_drill > 0 )
|
if( hole_drill > 0 )
|
||||||
{
|
{
|
||||||
|
bool pth = false;
|
||||||
|
|
||||||
|
if( aPad->GetAttribute() != PAD_HOLE_NOT_PLATED )
|
||||||
|
pth = true;
|
||||||
|
|
||||||
if( aPad->GetDrillShape() == PAD_DRILL_OBLONG )
|
if( aPad->GetDrillShape() == PAD_DRILL_OBLONG )
|
||||||
{
|
{
|
||||||
// Oblong hole (slot)
|
// Oblong hole (slot)
|
||||||
aModel.holes.AddSlot( hole_x, -hole_y, hole_drill_w * 2.0, hole_drill_h * 2.0,
|
aModel.holes.AddSlot( hole_x, -hole_y, hole_drill_w * 2.0, hole_drill_h * 2.0,
|
||||||
aPad->GetOrientation()/10.0, true );
|
aPad->GetOrientation()/10.0, true, pth );
|
||||||
|
|
||||||
|
if( pth )
|
||||||
|
aModel.plated_holes.AddSlot( hole_x, -hole_y,
|
||||||
|
hole_drill_w * 2.0, hole_drill_h * 2.0,
|
||||||
|
aPad->GetOrientation()/10.0, true, false );
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Drill a round hole
|
// Drill a round hole
|
||||||
aModel.holes.AddCircle( hole_x, -hole_y, hole_drill, true );
|
aModel.holes.AddCircle( hole_x, -hole_y, hole_drill, true, pth );
|
||||||
|
|
||||||
|
if( pth )
|
||||||
|
aModel.plated_holes.AddCircle( hole_x, -hole_y, hole_drill, true, false );
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1390,7 +1388,7 @@ bool PCB_EDIT_FRAME::ExportVRML_File( const wxString& aFullFileName,
|
||||||
EDA_RECT bbbox = pcb->ComputeBoundingBox();
|
EDA_RECT bbbox = pcb->ComputeBoundingBox();
|
||||||
|
|
||||||
model3d.SetOffset( -model3d.scale * bbbox.Centre().x,
|
model3d.SetOffset( -model3d.scale * bbbox.Centre().x,
|
||||||
-model3d.scale * bbbox.Centre().y );
|
model3d.scale * bbbox.Centre().y );
|
||||||
|
|
||||||
output_file << " children [\n";
|
output_file << " children [\n";
|
||||||
|
|
||||||
|
|
|
@ -23,6 +23,14 @@
|
||||||
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
// Wishlist:
|
||||||
|
// 1. crop anything outside the board outline on PTH, silk, and copper layers
|
||||||
|
// 2. on the PTH layer, handle cropped holes differently from others;
|
||||||
|
// these are assumed to be castellated edges and the profile is not
|
||||||
|
// a closed loop as assumed for all other outlines.
|
||||||
|
// 3. a scheme is needed to tell a castellated edge from a plain board edge
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <iomanip>
|
#include <iomanip>
|
||||||
|
@ -137,12 +145,24 @@ static void CALLBACK vrml_tess_err( GLenum errorID, void* user_data )
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void CALLBACK vrml_tess_combine( GLdouble coords[3], void* vertex_data[4],
|
static void CALLBACK vrml_tess_combine( GLdouble coords[3], VERTEX_3D* vertex_data[4],
|
||||||
GLfloat weight[4], void** outData, void* user_data )
|
GLfloat weight[4], void** outData, void* user_data )
|
||||||
{
|
{
|
||||||
VRML_LAYER* lp = (VRML_LAYER*) user_data;
|
VRML_LAYER* lp = (VRML_LAYER*) user_data;
|
||||||
|
|
||||||
*outData = lp->AddExtraVertex( coords[0], coords[1] );
|
// the plating is set to true only if all are plated
|
||||||
|
bool plated = vertex_data[0]->pth;
|
||||||
|
|
||||||
|
if( !vertex_data[1]->pth )
|
||||||
|
plated = false;
|
||||||
|
|
||||||
|
if( vertex_data[2] && !vertex_data[2]->pth )
|
||||||
|
plated = false;
|
||||||
|
|
||||||
|
if( vertex_data[3] && !vertex_data[3]->pth )
|
||||||
|
plated = false;
|
||||||
|
|
||||||
|
*outData = lp->AddExtraVertex( coords[0], coords[1], plated );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -152,6 +172,8 @@ VRML_LAYER::VRML_LAYER()
|
||||||
maxArcSeg = 48;
|
maxArcSeg = 48;
|
||||||
minSegLength = 0.1;
|
minSegLength = 0.1;
|
||||||
maxSegLength = 0.5;
|
maxSegLength = 0.5;
|
||||||
|
offsetX = 0.0;
|
||||||
|
offsetY = 0.0;
|
||||||
|
|
||||||
fix = false;
|
fix = false;
|
||||||
Fault = false;
|
Fault = false;
|
||||||
|
@ -230,6 +252,8 @@ void VRML_LAYER::Clear( void )
|
||||||
contours.pop_back();
|
contours.pop_back();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pth.clear();
|
||||||
|
|
||||||
areas.clear();
|
areas.clear();
|
||||||
|
|
||||||
for( i = vertices.size(); i > 0; --i )
|
for( i = vertices.size(); i > 0; --i )
|
||||||
|
@ -254,6 +278,7 @@ void VRML_LAYER::clearTmp( void )
|
||||||
glcmd = 0;
|
glcmd = 0;
|
||||||
|
|
||||||
triplets.clear();
|
triplets.clear();
|
||||||
|
solid.clear();
|
||||||
|
|
||||||
for( i = outline.size(); i > 0; --i )
|
for( i = outline.size(); i > 0; --i )
|
||||||
{
|
{
|
||||||
|
@ -283,7 +308,7 @@ void VRML_LAYER::clearTmp( void )
|
||||||
|
|
||||||
// create a new contour to be populated; returns an index
|
// create a new contour to be populated; returns an index
|
||||||
// into the contour list or -1 if there are problems
|
// into the contour list or -1 if there are problems
|
||||||
int VRML_LAYER::NewContour( void )
|
int VRML_LAYER::NewContour( bool aPlatedHole )
|
||||||
{
|
{
|
||||||
if( fix )
|
if( fix )
|
||||||
return -1;
|
return -1;
|
||||||
|
@ -296,6 +321,8 @@ int VRML_LAYER::NewContour( void )
|
||||||
contours.push_back( contour );
|
contours.push_back( contour );
|
||||||
areas.push_back( 0.0 );
|
areas.push_back( 0.0 );
|
||||||
|
|
||||||
|
pth.push_back( aPlatedHole );
|
||||||
|
|
||||||
return contours.size() - 1;
|
return contours.size() - 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -329,6 +356,7 @@ bool VRML_LAYER::AddVertex( int aContourID, double aXpos, double aYpos )
|
||||||
vertex->y = aYpos;
|
vertex->y = aYpos;
|
||||||
vertex->i = idx++;
|
vertex->i = idx++;
|
||||||
vertex->o = -1;
|
vertex->o = -1;
|
||||||
|
vertex->pth = pth[ aContourID ];
|
||||||
|
|
||||||
VERTEX_3D* v2 = NULL;
|
VERTEX_3D* v2 = NULL;
|
||||||
|
|
||||||
|
@ -440,9 +468,15 @@ bool VRML_LAYER::AppendCircle( double aXpos, double aYpos,
|
||||||
|
|
||||||
// adds a circle the existing list; if 'hole' is true the contour is
|
// adds a circle the existing list; if 'hole' is true the contour is
|
||||||
// a hole. Returns true if OK.
|
// a hole. Returns true if OK.
|
||||||
bool VRML_LAYER::AddCircle( double aXpos, double aYpos, double aRadius, bool aHoleFlag )
|
bool VRML_LAYER::AddCircle( double aXpos, double aYpos, double aRadius,
|
||||||
|
bool aHoleFlag, bool aPlatedHole )
|
||||||
{
|
{
|
||||||
int pad = NewContour();
|
int pad;
|
||||||
|
|
||||||
|
if( aHoleFlag && aPlatedHole )
|
||||||
|
pad = NewContour( true );
|
||||||
|
else
|
||||||
|
pad = NewContour( false );
|
||||||
|
|
||||||
if( pad < 0 )
|
if( pad < 0 )
|
||||||
{
|
{
|
||||||
|
@ -458,7 +492,7 @@ bool VRML_LAYER::AddCircle( double aXpos, double aYpos, double aRadius, bool aHo
|
||||||
// contour is a hole. Returns true if OK.
|
// contour is a hole. Returns true if OK.
|
||||||
bool VRML_LAYER::AddSlot( double aCenterX, double aCenterY,
|
bool VRML_LAYER::AddSlot( double aCenterX, double aCenterY,
|
||||||
double aSlotLength, double aSlotWidth,
|
double aSlotLength, double aSlotWidth,
|
||||||
double aAngle, bool aHoleFlag )
|
double aAngle, bool aHoleFlag, bool aPlatedHole )
|
||||||
{
|
{
|
||||||
aAngle *= M_PI / 180.0;
|
aAngle *= M_PI / 180.0;
|
||||||
|
|
||||||
|
@ -480,7 +514,12 @@ bool VRML_LAYER::AddSlot( double aCenterX, double aCenterY,
|
||||||
|
|
||||||
double ang, da;
|
double ang, da;
|
||||||
int i;
|
int i;
|
||||||
int pad = NewContour();
|
int pad;
|
||||||
|
|
||||||
|
if( aHoleFlag && aPlatedHole )
|
||||||
|
pad = NewContour( true );
|
||||||
|
else
|
||||||
|
pad = NewContour( false );
|
||||||
|
|
||||||
if( pad < 0 )
|
if( pad < 0 )
|
||||||
{
|
{
|
||||||
|
@ -578,7 +617,7 @@ bool VRML_LAYER::AppendArc( double aCenterX, double aCenterY, double aRadius,
|
||||||
|
|
||||||
// adds an arc with the given center, start point, pen width, and angle (degrees).
|
// adds an arc with the given center, start point, pen width, and angle (degrees).
|
||||||
bool VRML_LAYER::AddArc( double aCenterX, double aCenterY, double aStartX, double aStartY,
|
bool VRML_LAYER::AddArc( double aCenterX, double aCenterY, double aStartX, double aStartY,
|
||||||
double aArcWidth, double aAngle, bool aHoleFlag )
|
double aArcWidth, double aAngle, bool aHoleFlag, bool aPlatedHole )
|
||||||
{
|
{
|
||||||
aAngle *= M_PI / 180.0;
|
aAngle *= M_PI / 180.0;
|
||||||
|
|
||||||
|
@ -636,7 +675,12 @@ bool VRML_LAYER::AddArc( double aCenterX, double aCenterY, double aStartX, doubl
|
||||||
std::swap( iendy, isty );
|
std::swap( iendy, isty );
|
||||||
}
|
}
|
||||||
|
|
||||||
int arc = NewContour();
|
int arc;
|
||||||
|
|
||||||
|
if( aHoleFlag && aPlatedHole )
|
||||||
|
arc = NewContour( true );
|
||||||
|
else
|
||||||
|
arc = NewContour( false );
|
||||||
|
|
||||||
if( arc < 0 )
|
if( arc < 0 )
|
||||||
{
|
{
|
||||||
|
@ -688,7 +732,7 @@ bool VRML_LAYER::AddArc( double aCenterX, double aCenterY, double aStartX, doubl
|
||||||
|
|
||||||
// tesselates the contours in preparation for a 3D output;
|
// tesselates the contours in preparation for a 3D output;
|
||||||
// returns true if all was fine, false otherwise
|
// returns true if all was fine, false otherwise
|
||||||
bool VRML_LAYER::Tesselate( VRML_LAYER* holes )
|
bool VRML_LAYER::Tesselate( VRML_LAYER* holes, bool aHolesOnly )
|
||||||
{
|
{
|
||||||
if( !tess )
|
if( !tess )
|
||||||
{
|
{
|
||||||
|
@ -699,6 +743,12 @@ bool VRML_LAYER::Tesselate( VRML_LAYER* holes )
|
||||||
pholes = holes;
|
pholes = holes;
|
||||||
Fault = false;
|
Fault = false;
|
||||||
|
|
||||||
|
if( aHolesOnly )
|
||||||
|
gluTessProperty( tess, GLU_TESS_WINDING_RULE, GLU_TESS_WINDING_NEGATIVE );
|
||||||
|
else
|
||||||
|
gluTessProperty( tess, GLU_TESS_WINDING_RULE, GLU_TESS_WINDING_POSITIVE );
|
||||||
|
|
||||||
|
|
||||||
if( contours.size() < 1 || vertices.size() < 3 )
|
if( contours.size() < 1 || vertices.size() < 3 )
|
||||||
{
|
{
|
||||||
error = "Tesselate(): not enough vertices";
|
error = "Tesselate(): not enough vertices";
|
||||||
|
@ -736,9 +786,37 @@ bool VRML_LAYER::Tesselate( VRML_LAYER* holes )
|
||||||
|
|
||||||
eidx = idx + hidx;
|
eidx = idx + hidx;
|
||||||
|
|
||||||
|
if( aHolesOnly && ( checkNContours( true ) == 0 ) )
|
||||||
|
{
|
||||||
|
error = "tesselate(): no hole contours";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
else if( !aHolesOnly && ( checkNContours( false ) == 0 ) )
|
||||||
|
{
|
||||||
|
error = "tesselate(): no solid contours";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
// open the polygon
|
// open the polygon
|
||||||
gluTessBeginPolygon( tess, this );
|
gluTessBeginPolygon( tess, this );
|
||||||
|
|
||||||
|
if( aHolesOnly )
|
||||||
|
{
|
||||||
|
pholes = NULL; // do not accept foreign holes
|
||||||
|
hidx = 0;
|
||||||
|
eidx = idx;
|
||||||
|
|
||||||
|
// add holes
|
||||||
|
pushVertices( true );
|
||||||
|
|
||||||
|
gluTessEndPolygon( tess );
|
||||||
|
|
||||||
|
if( Fault )
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// add solid outlines
|
// add solid outlines
|
||||||
pushVertices( false );
|
pushVertices( false );
|
||||||
|
|
||||||
|
@ -748,6 +826,13 @@ bool VRML_LAYER::Tesselate( VRML_LAYER* holes )
|
||||||
if( Fault )
|
if( Fault )
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
|
// if there are no outlines we cannot proceed
|
||||||
|
if( outline.empty() )
|
||||||
|
{
|
||||||
|
error = "tesselate(): no points in result";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
// at this point we have a solid outline; add it to the tesselator
|
// at this point we have a solid outline; add it to the tesselator
|
||||||
gluTessBeginPolygon( tess, this );
|
gluTessBeginPolygon( tess, this );
|
||||||
|
|
||||||
|
@ -793,6 +878,7 @@ bool VRML_LAYER::Tesselate( VRML_LAYER* holes )
|
||||||
|
|
||||||
// close the polygon; this creates the outline points
|
// close the polygon; this creates the outline points
|
||||||
// and the point ordering list 'ordmap'
|
// and the point ordering list 'ordmap'
|
||||||
|
solid.clear();
|
||||||
gluTessEndPolygon( tess );
|
gluTessEndPolygon( tess );
|
||||||
|
|
||||||
// repeat the last operation but request a tesselated surface
|
// repeat the last operation but request a tesselated surface
|
||||||
|
@ -825,6 +911,8 @@ bool VRML_LAYER::pushOutline( VRML_LAYER* holes )
|
||||||
std::list<std::list<int>*>::const_iterator obeg = outline.begin();
|
std::list<std::list<int>*>::const_iterator obeg = outline.begin();
|
||||||
std::list<std::list<int>*>::const_iterator oend = outline.end();
|
std::list<std::list<int>*>::const_iterator oend = outline.end();
|
||||||
|
|
||||||
|
int nc = 0; // number of contours pushed
|
||||||
|
|
||||||
int pi;
|
int pi;
|
||||||
std::list<int>::const_iterator begin;
|
std::list<int>::const_iterator begin;
|
||||||
std::list<int>::const_iterator end;
|
std::list<int>::const_iterator end;
|
||||||
|
@ -876,6 +964,13 @@ bool VRML_LAYER::pushOutline( VRML_LAYER* holes )
|
||||||
|
|
||||||
gluTessEndContour( tess );
|
gluTessEndContour( tess );
|
||||||
++obeg;
|
++obeg;
|
||||||
|
++nc;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( !nc )
|
||||||
|
{
|
||||||
|
error = "pushOutline():: no valid contours available";
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@ -902,7 +997,7 @@ bool VRML_LAYER::WriteVertices( double aZcoord, std::ofstream& aOutFile, int aPr
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
std::string strx, stry, strz;
|
std::string strx, stry, strz;
|
||||||
FormatDoublet( vp->x, vp->y, aPrecision, strx, stry );
|
FormatDoublet( vp->x + offsetX, vp->y + offsetY, aPrecision, strx, stry );
|
||||||
FormatSinglet( aZcoord, aPrecision, strz );
|
FormatSinglet( aZcoord, aPrecision, strz );
|
||||||
|
|
||||||
aOutFile << strx << " " << stry << " " << strz;
|
aOutFile << strx << " " << stry << " " << strz;
|
||||||
|
@ -914,7 +1009,7 @@ bool VRML_LAYER::WriteVertices( double aZcoord, std::ofstream& aOutFile, int aPr
|
||||||
if( !vp )
|
if( !vp )
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
FormatDoublet( vp->x, vp->y, aPrecision, strx, stry );
|
FormatDoublet( vp->x + offsetX, vp->y + offsetY, aPrecision, strx, stry );
|
||||||
|
|
||||||
if( i & 1 )
|
if( i & 1 )
|
||||||
aOutFile << ", " << strx << " " << stry << " " << strz;
|
aOutFile << ", " << strx << " " << stry << " " << strz;
|
||||||
|
@ -954,7 +1049,7 @@ bool VRML_LAYER::Write3DVertices( double aTopZ, double aBottomZ,
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
std::string strx, stry, strz;
|
std::string strx, stry, strz;
|
||||||
FormatDoublet( vp->x, vp->y, aPrecision, strx, stry );
|
FormatDoublet( vp->x + offsetX, vp->y + offsetY, aPrecision, strx, stry );
|
||||||
FormatSinglet( aTopZ, aPrecision, strz );
|
FormatSinglet( aTopZ, aPrecision, strz );
|
||||||
|
|
||||||
aOutFile << strx << " " << stry << " " << strz;
|
aOutFile << strx << " " << stry << " " << strz;
|
||||||
|
@ -966,7 +1061,7 @@ bool VRML_LAYER::Write3DVertices( double aTopZ, double aBottomZ,
|
||||||
if( !vp )
|
if( !vp )
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
FormatDoublet( vp->x, vp->y, aPrecision, strx, stry );
|
FormatDoublet( vp->x + offsetX, vp->y + offsetY, aPrecision, strx, stry );
|
||||||
|
|
||||||
if( i & 1 )
|
if( i & 1 )
|
||||||
aOutFile << ", " << strx << " " << stry << " " << strz;
|
aOutFile << ", " << strx << " " << stry << " " << strz;
|
||||||
|
@ -976,7 +1071,7 @@ bool VRML_LAYER::Write3DVertices( double aTopZ, double aBottomZ,
|
||||||
|
|
||||||
// repeat for the bottom layer
|
// repeat for the bottom layer
|
||||||
vp = getVertexByIndex( ordmap[0], pholes );
|
vp = getVertexByIndex( ordmap[0], pholes );
|
||||||
FormatDoublet( vp->x, vp->y, aPrecision, strx, stry );
|
FormatDoublet( vp->x + offsetX, vp->y + offsetY, aPrecision, strx, stry );
|
||||||
FormatSinglet( aBottomZ, aPrecision, strz );
|
FormatSinglet( aBottomZ, aPrecision, strz );
|
||||||
|
|
||||||
bool endl;
|
bool endl;
|
||||||
|
@ -995,7 +1090,7 @@ bool VRML_LAYER::Write3DVertices( double aTopZ, double aBottomZ,
|
||||||
for( i = 1, j = ordmap.size(); i < j; ++i )
|
for( i = 1, j = ordmap.size(); i < j; ++i )
|
||||||
{
|
{
|
||||||
vp = getVertexByIndex( ordmap[i], pholes );
|
vp = getVertexByIndex( ordmap[i], pholes );
|
||||||
FormatDoublet( vp->x, vp->y, aPrecision, strx, stry );
|
FormatDoublet( vp->x + offsetX, vp->y + offsetY, aPrecision, strx, stry );
|
||||||
|
|
||||||
if( endl )
|
if( endl )
|
||||||
{
|
{
|
||||||
|
@ -1064,68 +1159,73 @@ bool VRML_LAYER::WriteIndices( bool aTopFlag, std::ofstream& aOutFile )
|
||||||
|
|
||||||
|
|
||||||
// writes out the index list for a 3D feature
|
// writes out the index list for a 3D feature
|
||||||
bool VRML_LAYER::Write3DIndices( std::ofstream& aOutFile )
|
bool VRML_LAYER::Write3DIndices( std::ofstream& aOutFile, bool aIncludePlatedHoles )
|
||||||
{
|
{
|
||||||
if( triplets.empty() )
|
|
||||||
{
|
|
||||||
error = "Write3DIndices(): no triplets (triangular facets) to write";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if( outline.empty() )
|
if( outline.empty() )
|
||||||
{
|
{
|
||||||
error = "WriteIndices(): no outline available";
|
error = "WriteIndices(): no outline available";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// go through the triplet list and write out the indices based on order
|
char mark;
|
||||||
std::list<TRIPLET_3D>::const_iterator tbeg = triplets.begin();
|
bool holes_only = triplets.empty();
|
||||||
std::list<TRIPLET_3D>::const_iterator tend = triplets.end();
|
|
||||||
|
|
||||||
int i = 1;
|
int i = 1;
|
||||||
int idx2 = ordmap.size(); // index to the bottom vertices
|
int idx2 = ordmap.size(); // index to the bottom vertices
|
||||||
|
|
||||||
// print out the top vertices
|
if( !holes_only )
|
||||||
aOutFile << tbeg->i1 << ", " << tbeg->i2 << ", " << tbeg->i3 << ", -1";
|
|
||||||
++tbeg;
|
|
||||||
|
|
||||||
while( tbeg != tend )
|
|
||||||
{
|
{
|
||||||
if( (i++ & 7) == 4 )
|
mark = ',';
|
||||||
{
|
|
||||||
i = 1;
|
|
||||||
aOutFile << ",\n" << tbeg->i1 << ", " << tbeg->i2 << ", " << tbeg->i3 << ", -1";
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
aOutFile << ", " << tbeg->i1 << ", " << tbeg->i2 << ", " << tbeg->i3 << ", -1";
|
|
||||||
}
|
|
||||||
|
|
||||||
|
// go through the triplet list and write out the indices based on order
|
||||||
|
std::list<TRIPLET_3D>::const_iterator tbeg = triplets.begin();
|
||||||
|
std::list<TRIPLET_3D>::const_iterator tend = triplets.end();
|
||||||
|
|
||||||
|
// print out the top vertices
|
||||||
|
aOutFile << tbeg->i1 << ", " << tbeg->i2 << ", " << tbeg->i3 << ", -1";
|
||||||
++tbeg;
|
++tbeg;
|
||||||
}
|
|
||||||
|
|
||||||
// print out the bottom vertices
|
while( tbeg != tend )
|
||||||
tbeg = triplets.begin();
|
{
|
||||||
|
if( (i++ & 7) == 4 )
|
||||||
|
{
|
||||||
|
i = 1;
|
||||||
|
aOutFile << ",\n" << tbeg->i1 << ", " << tbeg->i2 << ", " << tbeg->i3 << ", -1";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
aOutFile << ", " << tbeg->i1 << ", " << tbeg->i2 << ", " << tbeg->i3 << ", -1";
|
||||||
|
}
|
||||||
|
|
||||||
while( tbeg != tend )
|
++tbeg;
|
||||||
{
|
|
||||||
if( (i++ & 7) == 4 )
|
|
||||||
{
|
|
||||||
i = 1;
|
|
||||||
aOutFile << ",\n" << (tbeg->i2 + idx2) << ", " << (tbeg->i1 + idx2) << ", " << (tbeg->i3 + idx2) << ", -1";
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
aOutFile << ", " << (tbeg->i2 + idx2) << ", " << (tbeg->i1 + idx2) << ", " << (tbeg->i3 + idx2) << ", -1";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
++tbeg;
|
// print out the bottom vertices
|
||||||
|
tbeg = triplets.begin();
|
||||||
|
|
||||||
|
while( tbeg != tend )
|
||||||
|
{
|
||||||
|
if( (i++ & 7) == 4 )
|
||||||
|
{
|
||||||
|
i = 1;
|
||||||
|
aOutFile << ",\n" << (tbeg->i2 + idx2) << ", " << (tbeg->i1 + idx2) << ", " << (tbeg->i3 + idx2) << ", -1";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
aOutFile << ", " << (tbeg->i2 + idx2) << ", " << (tbeg->i1 + idx2) << ", " << (tbeg->i3 + idx2) << ", -1";
|
||||||
|
}
|
||||||
|
|
||||||
|
++tbeg;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
mark = ' ';
|
||||||
|
|
||||||
|
|
||||||
// print out indices for the walls joining top to bottom
|
// print out indices for the walls joining top to bottom
|
||||||
int firstPoint;
|
|
||||||
int lastPoint;
|
int lastPoint;
|
||||||
int curPoint;
|
int curPoint;
|
||||||
|
int curContour = 0;
|
||||||
|
|
||||||
std::list<std::list<int>*>::const_iterator obeg = outline.begin();
|
std::list<std::list<int>*>::const_iterator obeg = outline.begin();
|
||||||
std::list<std::list<int>*>::const_iterator oend = outline.end();
|
std::list<std::list<int>*>::const_iterator oend = outline.end();
|
||||||
|
@ -1141,22 +1241,71 @@ bool VRML_LAYER::Write3DIndices( std::ofstream& aOutFile )
|
||||||
if( cp->size() < 3 )
|
if( cp->size() < 3 )
|
||||||
{
|
{
|
||||||
++obeg;
|
++obeg;
|
||||||
|
++curContour;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
cbeg = cp->begin();
|
cbeg = cp->begin();
|
||||||
cend = cp->end();
|
cend = cp->end();
|
||||||
|
lastPoint = *(cbeg++);
|
||||||
|
|
||||||
firstPoint = *(cbeg++);
|
// skip all PTH vertices which are not in a solid outline
|
||||||
lastPoint = firstPoint;
|
if( !aIncludePlatedHoles && !solid[curContour]
|
||||||
|
&& getVertexByIndex( ordmap[lastPoint], pholes )->pth )
|
||||||
|
{
|
||||||
|
++obeg;
|
||||||
|
++curContour;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
while( cbeg != cend )
|
while( cbeg != cend )
|
||||||
{
|
{
|
||||||
curPoint = *(cbeg++);
|
curPoint = *(cbeg++);
|
||||||
|
|
||||||
|
if( !holes_only )
|
||||||
|
{
|
||||||
|
if( (i++ & 3) == 2 )
|
||||||
|
{
|
||||||
|
i = 1;
|
||||||
|
aOutFile << mark << "\n" << curPoint << ", " << lastPoint << ", " << curPoint + idx2;
|
||||||
|
aOutFile << ", -1, " << curPoint + idx2 << ", " << lastPoint << ", " << lastPoint + idx2 << ", -1";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
aOutFile << mark << " " << curPoint << ", " << lastPoint << ", " << curPoint + idx2;
|
||||||
|
aOutFile << ", -1, " << curPoint + idx2 << ", " << lastPoint << ", " << lastPoint + idx2 << ", -1";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if( (i++ & 3) == 2 )
|
||||||
|
{
|
||||||
|
i = 1;
|
||||||
|
aOutFile << mark << "\n" << curPoint << ", " << curPoint + idx2 << ", " << lastPoint;
|
||||||
|
aOutFile << ", -1, " << curPoint + idx2 << ", " << lastPoint + idx2 << ", " << lastPoint << ", -1";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
aOutFile << mark << " " << curPoint << ", " << curPoint + idx2 << ", " << lastPoint;
|
||||||
|
aOutFile << ", -1, " << curPoint + idx2 << ", " << lastPoint + idx2 << ", " << lastPoint << ", -1";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
mark = ',';
|
||||||
|
lastPoint = curPoint;
|
||||||
|
}
|
||||||
|
|
||||||
|
// check if the loop needs to be closed
|
||||||
|
cbeg = cp->begin();
|
||||||
|
cend = --cp->end();
|
||||||
|
|
||||||
|
curPoint = *(cbeg);
|
||||||
|
lastPoint = *(cend);
|
||||||
|
|
||||||
|
if( !holes_only )
|
||||||
|
{
|
||||||
if( (i++ & 3) == 2 )
|
if( (i++ & 3) == 2 )
|
||||||
{
|
{
|
||||||
i = 1;
|
|
||||||
aOutFile << ",\n" << curPoint << ", " << lastPoint << ", " << curPoint + idx2;
|
aOutFile << ",\n" << curPoint << ", " << lastPoint << ", " << curPoint + idx2;
|
||||||
aOutFile << ", -1, " << curPoint + idx2 << ", " << lastPoint << ", " << lastPoint + idx2 << ", -1";
|
aOutFile << ", -1, " << curPoint + idx2 << ", " << lastPoint << ", " << lastPoint + idx2 << ", -1";
|
||||||
}
|
}
|
||||||
|
@ -1165,21 +1314,23 @@ bool VRML_LAYER::Write3DIndices( std::ofstream& aOutFile )
|
||||||
aOutFile << ", " << curPoint << ", " << lastPoint << ", " << curPoint + idx2;
|
aOutFile << ", " << curPoint << ", " << lastPoint << ", " << curPoint + idx2;
|
||||||
aOutFile << ", -1, " << curPoint + idx2 << ", " << lastPoint << ", " << lastPoint + idx2 << ", -1";
|
aOutFile << ", -1, " << curPoint + idx2 << ", " << lastPoint << ", " << lastPoint + idx2 << ", -1";
|
||||||
}
|
}
|
||||||
lastPoint = curPoint;
|
|
||||||
}
|
|
||||||
|
|
||||||
if( (i++ & 3) == 2 )
|
|
||||||
{
|
|
||||||
aOutFile << ",\n" << firstPoint << ", " << lastPoint << ", " << firstPoint + idx2;
|
|
||||||
aOutFile << ", -1, " << firstPoint + idx2 << ", " << lastPoint << ", " << lastPoint + idx2 << ", -1";
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
aOutFile << ", " << firstPoint << ", " << lastPoint << ", " << firstPoint + idx2;
|
if( (i++ & 3) == 2 )
|
||||||
aOutFile << ", -1, " << firstPoint + idx2 << ", " << lastPoint << ", " << lastPoint + idx2 << ", -1";
|
{
|
||||||
|
aOutFile << ",\n" << curPoint << ", " << curPoint + idx2 << ", " << lastPoint;
|
||||||
|
aOutFile << ", -1, " << curPoint + idx2 << ", " << lastPoint + idx2 << ", " << lastPoint << ", -1";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
aOutFile << ", " << curPoint << ", " << curPoint + idx2 << ", " << lastPoint;
|
||||||
|
aOutFile << ", -1, " << curPoint + idx2 << ", " << lastPoint + idx2 << ", " << lastPoint << ", -1";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
++obeg;
|
++obeg;
|
||||||
|
++curContour;
|
||||||
}
|
}
|
||||||
|
|
||||||
return !aOutFile.fail();
|
return !aOutFile.fail();
|
||||||
|
@ -1222,7 +1373,7 @@ bool VRML_LAYER::addTriplet( VERTEX_3D* p0, VERTEX_3D* p1, VERTEX_3D* p2 )
|
||||||
|
|
||||||
|
|
||||||
// add an extra vertex (to be called only by the COMBINE callback)
|
// add an extra vertex (to be called only by the COMBINE callback)
|
||||||
VERTEX_3D* VRML_LAYER::AddExtraVertex( double aXpos, double aYpos )
|
VERTEX_3D* VRML_LAYER::AddExtraVertex( double aXpos, double aYpos, bool aPlatedHole )
|
||||||
{
|
{
|
||||||
VERTEX_3D* vertex = new VERTEX_3D;
|
VERTEX_3D* vertex = new VERTEX_3D;
|
||||||
|
|
||||||
|
@ -1239,6 +1390,7 @@ VERTEX_3D* VRML_LAYER::AddExtraVertex( double aXpos, double aYpos )
|
||||||
vertex->y = aYpos;
|
vertex->y = aYpos;
|
||||||
vertex->i = eidx++;
|
vertex->i = eidx++;
|
||||||
vertex->o = -1;
|
vertex->o = -1;
|
||||||
|
vertex->pth = aPlatedHole;
|
||||||
|
|
||||||
extra_verts.push_back( vertex );
|
extra_verts.push_back( vertex );
|
||||||
|
|
||||||
|
@ -1282,12 +1434,39 @@ void VRML_LAYER::glEnd( void )
|
||||||
if( !loop )
|
if( !loop )
|
||||||
break;
|
break;
|
||||||
|
|
||||||
for( unsigned int i = 0; i < vlist.size(); ++i )
|
double firstX = 0.0;
|
||||||
|
double firstY = 0.0;
|
||||||
|
double lastX, lastY;
|
||||||
|
double curX, curY;
|
||||||
|
double area = 0.0;
|
||||||
|
|
||||||
|
if( vlist.size() > 0 )
|
||||||
{
|
{
|
||||||
loop->push_back( vlist[i]->o );
|
loop->push_back( vlist[0]->o );
|
||||||
|
firstX = vlist[0]->x;
|
||||||
|
firstY = vlist[0]->y;
|
||||||
|
lastX = firstX;
|
||||||
|
lastY = firstY;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for( size_t i = 1; i < vlist.size(); ++i )
|
||||||
|
{
|
||||||
|
loop->push_back( vlist[i]->o );
|
||||||
|
curX = vlist[i]->x;
|
||||||
|
curY = vlist[i]->y;
|
||||||
|
area += ( curX - lastX ) * ( curY + lastY );
|
||||||
|
lastX = curX;
|
||||||
|
lastY = curY;
|
||||||
|
}
|
||||||
|
|
||||||
|
area += ( firstX - lastX ) * ( firstY + lastY );
|
||||||
|
|
||||||
outline.push_back( loop );
|
outline.push_back( loop );
|
||||||
|
|
||||||
|
if( area <= 0.0 )
|
||||||
|
solid.push_back( true );
|
||||||
|
else
|
||||||
|
solid.push_back( false );
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -1398,6 +1577,31 @@ void VRML_LAYER::processTri( void )
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int VRML_LAYER::checkNContours( bool holes )
|
||||||
|
{
|
||||||
|
int nc = 0; // number of contours
|
||||||
|
|
||||||
|
if( contours.empty() )
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
std::list<int>::const_iterator begin;
|
||||||
|
std::list<int>::const_iterator end;
|
||||||
|
|
||||||
|
for( size_t i = 0; i < contours.size(); ++i )
|
||||||
|
{
|
||||||
|
if( contours[i]->size() < 3 )
|
||||||
|
continue;
|
||||||
|
|
||||||
|
if( ( holes && areas[i] <= 0.0 ) || ( !holes && areas[i] > 0.0 ) )
|
||||||
|
continue;
|
||||||
|
|
||||||
|
++nc;
|
||||||
|
}
|
||||||
|
|
||||||
|
return nc;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// push the internally held vertices
|
// push the internally held vertices
|
||||||
void VRML_LAYER::pushVertices( bool holes )
|
void VRML_LAYER::pushVertices( bool holes )
|
||||||
{
|
{
|
||||||
|
@ -1434,6 +1638,8 @@ void VRML_LAYER::pushVertices( bool holes )
|
||||||
|
|
||||||
gluTessEndContour( tess );
|
gluTessEndContour( tess );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1564,3 +1770,11 @@ const std::string& VRML_LAYER::GetError( void )
|
||||||
{
|
{
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void VRML_LAYER::SetVertexOffsets( double aXoffset, double aYoffset )
|
||||||
|
{
|
||||||
|
offsetX = aXoffset;
|
||||||
|
offsetY = aYoffset;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
|
@ -70,6 +70,7 @@ struct VERTEX_3D
|
||||||
double y;
|
double y;
|
||||||
int i; // vertex index
|
int i; // vertex index
|
||||||
int o; // vertex order
|
int o; // vertex order
|
||||||
|
bool pth; // true for plate-through hole
|
||||||
};
|
};
|
||||||
|
|
||||||
struct TRIPLET_3D
|
struct TRIPLET_3D
|
||||||
|
@ -93,12 +94,18 @@ private:
|
||||||
double minSegLength; // min. segment length
|
double minSegLength; // min. segment length
|
||||||
double maxSegLength; // max. segment length
|
double maxSegLength; // max. segment length
|
||||||
|
|
||||||
|
// Vertex offsets to work around a suspected GLU tesselator bug
|
||||||
|
double offsetX;
|
||||||
|
double offsetY;
|
||||||
|
|
||||||
bool fix; // when true, no more vertices may be added by the user
|
bool fix; // when true, no more vertices may be added by the user
|
||||||
int idx; // vertex index (number of contained vertices)
|
int idx; // vertex index (number of contained vertices)
|
||||||
int ord; // vertex order (number of ordered vertices)
|
int ord; // vertex order (number of ordered vertices)
|
||||||
unsigned int idxout; // outline index to first point in 3D outline
|
unsigned int idxout; // outline index to first point in 3D outline
|
||||||
std::vector<VERTEX_3D*> vertices; // vertices of all contours
|
std::vector<VERTEX_3D*> vertices; // vertices of all contours
|
||||||
std::vector<std::list<int>*> contours; // lists of vertices for each contour
|
std::vector<std::list<int>*> contours; // lists of vertices for each contour
|
||||||
|
std::vector<bool>pth; // indicates whether a 'contour' is a PTH or not
|
||||||
|
std::vector<bool>solid; // indicates whether a 'contour' is a solid or a hole
|
||||||
std::vector< double > areas; // area of the contours (positive if winding is CCW)
|
std::vector< double > areas; // area of the contours (positive if winding is CCW)
|
||||||
std::list<TRIPLET_3D> triplets; // output facet triplet list (triplet of ORDER values)
|
std::list<TRIPLET_3D> triplets; // output facet triplet list (triplet of ORDER values)
|
||||||
std::list<std::list<int>*> outline; // indices for outline outputs (index by ORDER values)
|
std::list<std::list<int>*> outline; // indices for outline outputs (index by ORDER values)
|
||||||
|
@ -136,6 +143,9 @@ private:
|
||||||
// calculate number of sides on an arc (angle is in radians)
|
// calculate number of sides on an arc (angle is in radians)
|
||||||
int calcNSides( double aRadius, double aAngle );
|
int calcNSides( double aRadius, double aAngle );
|
||||||
|
|
||||||
|
// returns the number of solid or hole contours
|
||||||
|
int checkNContours( bool holes );
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/// set to true when a fault is encountered during tesselation
|
/// set to true when a fault is encountered during tesselation
|
||||||
bool Fault;
|
bool Fault;
|
||||||
|
@ -191,9 +201,11 @@ public:
|
||||||
* Function NewContour
|
* Function NewContour
|
||||||
* creates a new list of vertices and returns an index to the list
|
* creates a new list of vertices and returns an index to the list
|
||||||
*
|
*
|
||||||
|
* @param aPlatedHole is true if the new contour will represent a plated hole
|
||||||
|
*
|
||||||
* @return int: index to the list or -1 if the operation failed
|
* @return int: index to the list or -1 if the operation failed
|
||||||
*/
|
*/
|
||||||
int NewContour( void );
|
int NewContour( bool aPlatedHole = false );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function AddVertex
|
* Function AddVertex
|
||||||
|
@ -231,7 +243,8 @@ public:
|
||||||
*
|
*
|
||||||
* @return bool: true if the new contour was successfully created
|
* @return bool: true if the new contour was successfully created
|
||||||
*/
|
*/
|
||||||
bool AppendCircle( double aXpos, double aYpos, double aRadius, int aContourID, bool aHoleFlag = false );
|
bool AppendCircle( double aXpos, double aYpos, double aRadius,
|
||||||
|
int aContourID, bool aHoleFlag = false );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function AddCircle
|
* Function AddCircle
|
||||||
|
@ -241,10 +254,12 @@ public:
|
||||||
* @param aYpos is the Y coordinate of the hole center
|
* @param aYpos is the Y coordinate of the hole center
|
||||||
* @param aRadius is the radius of the hole
|
* @param aRadius is the radius of the hole
|
||||||
* @param aHoleFlag determines if the contour to be created is a cutout
|
* @param aHoleFlag determines if the contour to be created is a cutout
|
||||||
|
* @param aPlatedHole is true if this is a plated hole
|
||||||
*
|
*
|
||||||
* @return bool: true if the new contour was successfully created
|
* @return bool: true if the new contour was successfully created
|
||||||
*/
|
*/
|
||||||
bool AddCircle( double aXpos, double aYpos, double aRadius, bool aHoleFlag = false );
|
bool AddCircle( double aXpos, double aYpos, double aRadius,
|
||||||
|
bool aHoleFlag = false, bool aPlatedHole = false );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function AddSlot
|
* Function AddSlot
|
||||||
|
@ -256,11 +271,12 @@ public:
|
||||||
* @param aSlotWidth is the width of the slot along the minor axis
|
* @param aSlotWidth is the width of the slot along the minor axis
|
||||||
* @param aAngle (degrees) is the orientation of the slot
|
* @param aAngle (degrees) is the orientation of the slot
|
||||||
* @param aHoleFlag determines whether the slot is a hole or a solid
|
* @param aHoleFlag determines whether the slot is a hole or a solid
|
||||||
|
* @param aPlatedHole is true if this is a plated slot
|
||||||
*
|
*
|
||||||
* @return bool: true if the slot was successfully created
|
* @return bool: true if the slot was successfully created
|
||||||
*/
|
*/
|
||||||
bool AddSlot( double aCenterX, double aCenterY, double aSlotLength, double aSlotWidth,
|
bool AddSlot( double aCenterX, double aCenterY, double aSlotLength, double aSlotWidth,
|
||||||
double aAngle, bool aHoleFlag = false );
|
double aAngle, bool aHoleFlag = false, bool aPlatedHole = false );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function AppendArc
|
* Function AppendArc
|
||||||
|
@ -289,11 +305,14 @@ public:
|
||||||
* @param aArcWidth is the width of the arc
|
* @param aArcWidth is the width of the arc
|
||||||
* @param aAngle is the included angle (degrees)
|
* @param aAngle is the included angle (degrees)
|
||||||
* @param aHoleFlag determines whether the arc is to be a hole or a solid
|
* @param aHoleFlag determines whether the arc is to be a hole or a solid
|
||||||
|
* @param aPlatedHole is true if this is a plated slotted arc
|
||||||
*
|
*
|
||||||
* @return bool: true if the feature was successfully created
|
* @return bool: true if the feature was successfully created
|
||||||
*/
|
*/
|
||||||
bool AddArc( double aCenterX, double aCenterY, double aStartX, double aStartY,
|
bool AddArc( double aCenterX, double aCenterY,
|
||||||
double aArcWidth, double aAngle, bool aHoleFlag = false );
|
double aStartX, double aStartY,
|
||||||
|
double aArcWidth, double aAngle,
|
||||||
|
bool aHoleFlag = false, bool aPlatedHole = false );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function Tesselate
|
* Function Tesselate
|
||||||
|
@ -301,11 +320,12 @@ public:
|
||||||
* vertex sets required to render the surface.
|
* vertex sets required to render the surface.
|
||||||
*
|
*
|
||||||
* @param holes is an optional pointer to cutouts to be imposed on the
|
* @param holes is an optional pointer to cutouts to be imposed on the
|
||||||
* surface.
|
* surface.
|
||||||
|
* @param aHolesOnly is true if the outline contains only holes
|
||||||
*
|
*
|
||||||
* @return bool: true if the operation succeeded
|
* @return bool: true if the operation succeeded
|
||||||
*/
|
*/
|
||||||
bool Tesselate( VRML_LAYER* holes = NULL );
|
bool Tesselate( VRML_LAYER* holes = NULL, bool aHolesOnly = false );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function WriteVertices
|
* Function WriteVertices
|
||||||
|
@ -351,19 +371,26 @@ public:
|
||||||
* writes out the vertex sets required to render an extruded solid
|
* writes out the vertex sets required to render an extruded solid
|
||||||
*
|
*
|
||||||
* @param aOutFile is the file to write to
|
* @param aOutFile is the file to write to
|
||||||
|
* @param aIncludePlatedHoles is true if holes marked as plated should
|
||||||
|
* be rendered. Default is false since the user will usually
|
||||||
|
* render these holes in a different color
|
||||||
*
|
*
|
||||||
* @return bool: true if the operation succeeded
|
* @return bool: true if the operation succeeded
|
||||||
*/
|
*/
|
||||||
bool Write3DIndices( std::ofstream& aOutFile );
|
bool Write3DIndices( std::ofstream& aOutFile, bool aIncludePlatedHoles = false );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function AddExtraVertex
|
* Function AddExtraVertex
|
||||||
* adds an extra vertex as required by the GLU tesselator
|
* adds an extra vertex as required by the GLU tesselator.
|
||||||
|
*
|
||||||
|
* @param aXpos is the X coordinate of the newly created point
|
||||||
|
* @param aYpos is the Y coordinate of the newly created point
|
||||||
|
* @param aPlatedHole is true if this point is part of a plated hole
|
||||||
*
|
*
|
||||||
* @return VERTEX_3D*: is the new vertex or NULL if a vertex
|
* @return VERTEX_3D*: is the new vertex or NULL if a vertex
|
||||||
* could not be created.
|
* could not be created.
|
||||||
*/
|
*/
|
||||||
VERTEX_3D* AddExtraVertex( double aXpos, double aYpos );
|
VERTEX_3D* AddExtraVertex( double aXpos, double aYpos, bool aPlatedHole );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function glStart
|
* Function glStart
|
||||||
|
@ -429,6 +456,8 @@ public:
|
||||||
* Returns the error message related to the last failed operation
|
* Returns the error message related to the last failed operation
|
||||||
*/
|
*/
|
||||||
const std::string& GetError( void );
|
const std::string& GetError( void );
|
||||||
|
|
||||||
|
void SetVertexOffsets( double aXoffset, double aYoffset );
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // VRML_LAYER_H
|
#endif // VRML_LAYER_H
|
||||||
|
|
Loading…
Reference in New Issue