2011-09-23 13:57:12 +00:00
|
|
|
/**
|
|
|
|
* @file plothpgl.cpp
|
|
|
|
*/
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2012-06-09 09:38:58 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2012 KiCad Developers, see change_log.txt for contributors.
|
|
|
|
* Copyright (C) 2012 Dick Hollenbeck, dick@softplc.com
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <fctsys.h>
|
|
|
|
#include <common.h>
|
|
|
|
#include <plot_common.h>
|
|
|
|
#include <wxBasePcbFrame.h>
|
|
|
|
#include <class_board.h>
|
|
|
|
#include <pcbnew.h>
|
|
|
|
#include <pcbplot.h>
|
2012-06-09 09:38:58 +00:00
|
|
|
#include <convert_to_biu.h>
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2009-11-20 14:55:20 +00:00
|
|
|
|
2011-09-29 16:49:40 +00:00
|
|
|
bool PCB_BASE_FRAME::ExportToHpglFile( const wxString& aFullFileName, int aLayer,
|
2012-01-03 17:14:17 +00:00
|
|
|
EDA_DRAW_MODE_T aTraceMode )
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2011-12-22 21:57:50 +00:00
|
|
|
wxSize boardSize;
|
|
|
|
wxPoint boardCenter;
|
|
|
|
bool center = false;
|
|
|
|
double scale;
|
|
|
|
wxPoint offset;
|
|
|
|
LOCALE_IO toggle;
|
|
|
|
FILE* output_file = wxFopen( aFullFileName, wxT( "wt" ) );
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2010-01-01 13:30:39 +00:00
|
|
|
if( output_file == NULL )
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2012-04-05 18:27:56 +00:00
|
|
|
PCB_PLOT_PARAMS plot_opts = GetPlotSettings();
|
|
|
|
|
2012-06-09 09:38:58 +00:00
|
|
|
// Compute pen_dim (from m_HPGLPenDiam in mils) in pcb units,
|
|
|
|
// with plot scale (if Scale is 2, pen diameter value is always m_HPGLPenDiam
|
2007-08-23 04:28:46 +00:00
|
|
|
// so apparent pen diam is real pen diam / Scale
|
2012-06-09 09:38:58 +00:00
|
|
|
int pen_diam = KiROUND( plot_opts.m_HPGLPenDiam * IU_PER_MILS /
|
2012-04-05 18:27:56 +00:00
|
|
|
plot_opts.m_PlotScale );
|
2007-08-23 04:28:46 +00:00
|
|
|
|
2012-06-09 09:38:58 +00:00
|
|
|
// compute pen_overlay (from m_HPGLPenOvr in mils) in pcb units
|
|
|
|
// with plot scale
|
2012-04-05 18:27:56 +00:00
|
|
|
if( plot_opts.m_HPGLPenOvr < 0 )
|
|
|
|
plot_opts.m_HPGLPenOvr = 0;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2012-04-05 18:27:56 +00:00
|
|
|
if( plot_opts.m_HPGLPenOvr >= plot_opts.m_HPGLPenDiam )
|
|
|
|
plot_opts.m_HPGLPenOvr = plot_opts.m_HPGLPenDiam - 1;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2012-06-09 09:38:58 +00:00
|
|
|
int pen_overlay = KiROUND( plot_opts.m_HPGLPenOvr * IU_PER_MILS /
|
2012-04-05 18:27:56 +00:00
|
|
|
plot_opts.m_PlotScale );
|
2007-08-23 04:28:46 +00:00
|
|
|
|
|
|
|
|
2012-04-05 18:27:56 +00:00
|
|
|
if( plot_opts.m_PlotScale != 1.0 || plot_opts.m_AutoScale )
|
2011-12-22 21:57:50 +00:00
|
|
|
{
|
|
|
|
// when scale != 1.0 we must calculate the position in page
|
|
|
|
// because actual position has no meaning
|
|
|
|
center = true;
|
|
|
|
}
|
2009-06-28 16:50:42 +00:00
|
|
|
|
2011-12-22 21:57:50 +00:00
|
|
|
wxSize pageSizeIU = GetPageSizeIU();
|
2007-08-23 04:28:46 +00:00
|
|
|
|
++PCBNew
* Removed Pcb_Frame argument from BOARD() constructor, since it precludes
having a BOARD being edited by more than one editor, it was a bad design.
And this meant removing m_PcbFrame from BOARD.
* removed BOARD::SetWindowFrame(), and BOARD::m_PcbFrame
* Removed the global BOARD_DESIGN_SETTINGS which was in class_board.cpp
* added BOARD_DESIGN_SETTINGS to the BOARD class, a full instance
* a couple dialogs now only change BOARD_DESIGN_SETTINGS when OK is pressed,
such as dialog_mask_clearance, dialog_drc, etc.
* Removed common/pcbcommon.cpp's int g_CurrentVersionPCB = 1 and replaced it
with build_version.h's #define BOARD_FILE_VERSION, although there may be a
better place for this constant.
* Made the public functions in PARAM_CFG_ARRAY be type const.
void SaveParam(..) const and void ReadParam(..) const
* PARAM_CFG_BASE now has virtual destructor since we have various way of
destroying the derived class and boost::ptr_vector must be told about this.
* Pass const PARAM_CFG_ARRAY& instead of PARAM_CFG_ARRAY so that we can use
an automatic PARAM_CFG_ARRAY which is on the stack.\
* PCB_EDIT_FRAME::GetProjectFileParameters() may no longer cache the array,
since it has to access the current BOARD and the BOARD can change.
Remember BOARD_DESIGN_SETTINGS are now in the BOARD.
* Made the m_BoundingBox member private, this was a brutally hard task,
and indicative of the lack of commitment to accessors and object oriented
design on the part of KiCad developers. We must do better.
Added BOARD::GetBoundingBox, SetBoundingBox(), ComputeBoundingBox().
* Added PCB_BASE_FRAME::GetBoardBoundingBox() which calls BOARD::ComputeBoundingBox()
2011-12-05 06:15:33 +00:00
|
|
|
// Calculate the center of the PCB
|
|
|
|
EDA_RECT bbbox = GetBoardBoundingBox();
|
|
|
|
|
2011-12-22 21:57:50 +00:00
|
|
|
boardSize = bbbox.GetSize();
|
|
|
|
boardCenter = bbbox.Centre();
|
2007-08-23 04:28:46 +00:00
|
|
|
|
2012-04-05 18:27:56 +00:00
|
|
|
if( plot_opts.m_AutoScale ) // Optimum scale
|
2007-08-23 04:28:46 +00:00
|
|
|
{
|
2009-06-29 05:30:08 +00:00
|
|
|
// Fit to 80% of the page
|
2011-12-22 21:57:50 +00:00
|
|
|
double Xscale = ( ( pageSizeIU.x * 0.8 ) / boardSize.x );
|
|
|
|
double Yscale = ( ( pageSizeIU.y * 0.8 ) / boardSize.y );
|
2012-06-09 09:38:58 +00:00
|
|
|
scale = std::min( Xscale, Yscale );
|
2007-08-23 04:28:46 +00:00
|
|
|
}
|
2009-06-28 16:50:42 +00:00
|
|
|
else
|
2011-09-07 19:41:04 +00:00
|
|
|
{
|
2012-04-05 18:27:56 +00:00
|
|
|
scale = plot_opts.m_PlotScale;
|
2011-09-07 19:41:04 +00:00
|
|
|
}
|
2007-08-23 04:28:46 +00:00
|
|
|
|
2009-11-20 14:55:20 +00:00
|
|
|
// Calculate the page size offset.
|
2011-12-22 21:57:50 +00:00
|
|
|
if( center )
|
2009-06-29 05:30:08 +00:00
|
|
|
{
|
// Dick Hollenbeck's KiROUND R&D
// This provides better project control over rounding to int from double
// than wxRound() did. This scheme provides better logging in Debug builds
// and it provides for compile time calculation of constants.
#include <stdio.h>
#include <assert.h>
#include <limits.h>
//-----<KiROUND KIT>------------------------------------------------------------
/**
* KiROUND
* rounds a floating point number to an int using
* "round halfway cases away from zero".
* In Debug build an assert fires if will not fit into an int.
*/
#if defined( DEBUG )
// DEBUG: a macro to capture line and file, then calls this inline
static inline int KiRound( double v, int line, const char* filename )
{
v = v < 0 ? v - 0.5 : v + 0.5;
if( v > INT_MAX + 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' > 0 ' for int\n", __FUNCTION__, filename, line, v );
}
else if( v < INT_MIN - 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' < 0 ' for int\n", __FUNCTION__, filename, line, v );
}
return int( v );
}
#define KiROUND( v ) KiRound( v, __LINE__, __FILE__ )
#else
// RELEASE: a macro so compile can pre-compute constants.
#define KiROUND( v ) int( (v) < 0 ? (v) - 0.5 : (v) + 0.5 )
#endif
//-----</KiROUND KIT>-----------------------------------------------------------
// Only a macro is compile time calculated, an inline function causes a static constructor
// in a situation like this.
// Therefore the Release build is best done with a MACRO not an inline function.
int Computed = KiROUND( 14.3 * 8 );
int main( int argc, char** argv )
{
for( double d = double(INT_MAX)-1; d < double(INT_MAX)+8; d += 2.0 )
{
int i = KiROUND( d );
printf( "t: %d %.16g\n", i, d );
}
return 0;
}
2012-04-19 06:55:45 +00:00
|
|
|
offset.x = KiROUND( (double) boardCenter.x -
|
2011-12-22 21:57:50 +00:00
|
|
|
( (double) pageSizeIU.x / 2.0 ) / scale );
|
// Dick Hollenbeck's KiROUND R&D
// This provides better project control over rounding to int from double
// than wxRound() did. This scheme provides better logging in Debug builds
// and it provides for compile time calculation of constants.
#include <stdio.h>
#include <assert.h>
#include <limits.h>
//-----<KiROUND KIT>------------------------------------------------------------
/**
* KiROUND
* rounds a floating point number to an int using
* "round halfway cases away from zero".
* In Debug build an assert fires if will not fit into an int.
*/
#if defined( DEBUG )
// DEBUG: a macro to capture line and file, then calls this inline
static inline int KiRound( double v, int line, const char* filename )
{
v = v < 0 ? v - 0.5 : v + 0.5;
if( v > INT_MAX + 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' > 0 ' for int\n", __FUNCTION__, filename, line, v );
}
else if( v < INT_MIN - 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' < 0 ' for int\n", __FUNCTION__, filename, line, v );
}
return int( v );
}
#define KiROUND( v ) KiRound( v, __LINE__, __FILE__ )
#else
// RELEASE: a macro so compile can pre-compute constants.
#define KiROUND( v ) int( (v) < 0 ? (v) - 0.5 : (v) + 0.5 )
#endif
//-----</KiROUND KIT>-----------------------------------------------------------
// Only a macro is compile time calculated, an inline function causes a static constructor
// in a situation like this.
// Therefore the Release build is best done with a MACRO not an inline function.
int Computed = KiROUND( 14.3 * 8 );
int main( int argc, char** argv )
{
for( double d = double(INT_MAX)-1; d < double(INT_MAX)+8; d += 2.0 )
{
int i = KiROUND( d );
printf( "t: %d %.16g\n", i, d );
}
return 0;
}
2012-04-19 06:55:45 +00:00
|
|
|
offset.y = KiROUND( (double) boardCenter.y -
|
2011-12-22 21:57:50 +00:00
|
|
|
( (double) pageSizeIU.y / 2.0 ) / scale );
|
2009-06-29 05:30:08 +00:00
|
|
|
}
|
|
|
|
else
|
2007-08-23 04:28:46 +00:00
|
|
|
{
|
2009-06-29 05:30:08 +00:00
|
|
|
offset.x = 0;
|
|
|
|
offset.y = 0;
|
2009-06-28 16:50:42 +00:00
|
|
|
}
|
|
|
|
|
2009-08-29 10:20:48 +00:00
|
|
|
HPGL_PLOTTER* plotter = new HPGL_PLOTTER();
|
2011-12-22 21:57:50 +00:00
|
|
|
|
|
|
|
plotter->SetPageSettings( GetPageSettings() );
|
|
|
|
|
2012-04-05 18:27:56 +00:00
|
|
|
// why did we have to change these settings above?
|
|
|
|
SetPlotSettings( plot_opts );
|
|
|
|
|
2012-06-09 09:38:58 +00:00
|
|
|
plotter->SetViewport( offset, IU_PER_DECIMILS, scale,
|
2012-05-03 18:37:56 +00:00
|
|
|
plot_opts.m_PlotMirror );
|
|
|
|
plotter->SetDefaultLineWidth( plot_opts.m_PlotLineWidth );
|
|
|
|
plotter->SetCreator( wxT( "PCBNEW-HPGL" ) );
|
|
|
|
plotter->SetFilename( aFullFileName );
|
|
|
|
plotter->SetPenSpeed( plot_opts.m_HPGLPenSpeed );
|
|
|
|
plotter->SetPenNumber( plot_opts.m_HPGLPenNum );
|
|
|
|
plotter->SetPenOverlap( pen_overlay );
|
|
|
|
plotter->SetPenDiameter( pen_diam );
|
|
|
|
plotter->StartPlot( output_file );
|
2009-06-28 16:50:42 +00:00
|
|
|
|
2011-12-22 21:57:50 +00:00
|
|
|
// The worksheet is not significant with scale!=1... It is with paperscale!=1, anyway
|
2012-04-05 18:27:56 +00:00
|
|
|
if( plot_opts.m_PlotFrameRef && !center )
|
2012-06-09 09:38:58 +00:00
|
|
|
PlotWorkSheet( plotter, GetScreen(), plot_opts.GetPlotLineWidth() );
|
2009-06-28 16:50:42 +00:00
|
|
|
|
2011-09-29 16:49:40 +00:00
|
|
|
Plot_Layer( plotter, aLayer, aTraceMode );
|
2012-05-03 18:37:56 +00:00
|
|
|
plotter->EndPlot();
|
2009-06-28 16:50:42 +00:00
|
|
|
delete plotter;
|
2010-01-01 13:30:39 +00:00
|
|
|
|
|
|
|
return true;
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|