diff options
author | saurabhb17 | 2020-02-26 16:01:28 +0530 |
---|---|---|
committer | GitHub | 2020-02-26 16:01:28 +0530 |
commit | d51317f0193609fb43e932730d78aa86a4984083 (patch) | |
tree | 6acee185a4dc19113fcbf0f9a3d6941085dedaf7 /pcbnew/pcbplot.cpp | |
parent | 0db48f6533517ecebfd9f0693f89deca28408b76 (diff) | |
parent | 886d9cb772e81d2e5262284bc3082664f084337f (diff) | |
download | KiCad-eSim-d51317f0193609fb43e932730d78aa86a4984083.tar.gz KiCad-eSim-d51317f0193609fb43e932730d78aa86a4984083.tar.bz2 KiCad-eSim-d51317f0193609fb43e932730d78aa86a4984083.zip |
Merge pull request #2 from FOSSEE/develop
Develop
Diffstat (limited to 'pcbnew/pcbplot.cpp')
-rw-r--r-- | pcbnew/pcbplot.cpp | 426 |
1 files changed, 426 insertions, 0 deletions
diff --git a/pcbnew/pcbplot.cpp b/pcbnew/pcbplot.cpp new file mode 100644 index 0000000..6d1b1c3 --- /dev/null +++ b/pcbnew/pcbplot.cpp @@ -0,0 +1,426 @@ +/* + * This program source code file is part of KiCad, a free EDA CAD application. + * + * Copyright (C) 2015 Jean-Pierre Charras, jp.charras at wanadoo.fr + * Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com> + * Copyright (C) 1992-2015 KiCad Developers, see AUTHORS.txt for contributors. + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, you may find one here: + * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html + * or you may search the http://www.gnu.org website for the version 2 license, + * or you may write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA + */ + +/** + * @file pcbnew/pcbplot.cpp + */ + +#include <fctsys.h> +#include <plot_common.h> +#include <confirm.h> +#include <wxPcbStruct.h> +#include <pcbplot.h> +#include <pcbstruct.h> +#include <base_units.h> +#include <reporter.h> +#include <class_board.h> +#include <pcbnew.h> +#include <plotcontroller.h> +#include <pcb_plot_params.h> +#include <wx/ffile.h> +#include <dialog_plot.h> +#include <macros.h> +#include <build_version.h> + + +const wxString GetGerberProtelExtension( LAYER_NUM aLayer ) +{ + if( IsCopperLayer( aLayer ) ) + { + if( aLayer == F_Cu ) + return wxT( "gtl" ); + else if( aLayer == B_Cu ) + return wxT( "gbl" ); + else + { + return wxString::Format( wxT( "g%d" ), aLayer+1 ); + } + } + else + { + switch( aLayer ) + { + case B_Adhes: return wxT( "gba" ); + case F_Adhes: return wxT( "gta" ); + + case B_Paste: return wxT( "gbp" ); + case F_Paste: return wxT( "gtp" ); + + case B_SilkS: return wxT( "gbo" ); + case F_SilkS: return wxT( "gto" ); + + case B_Mask: return wxT( "gbs" ); + case F_Mask: return wxT( "gts" ); + + case Edge_Cuts: return wxT( "gm1" ); + + case Dwgs_User: + case Cmts_User: + case Eco1_User: + case Eco2_User: + default: return wxT( "gbr" ); + } + } +} + + +wxString GetGerberFileFunctionAttribute( const BOARD *aBoard, + LAYER_NUM aLayer, bool aUseX1CompatibilityMode ) +{ + wxString attrib; + + switch( aLayer ) + { + case F_Adhes: + attrib = wxString( wxT( "Glue,Top" ) ); + break; + + case B_Adhes: + attrib = wxString( wxT( "Glue,Bot" ) ); + break; + + case F_SilkS: + attrib = wxString( wxT( "Legend,Top" ) ); + break; + + case B_SilkS: + attrib = wxString( wxT( "Legend,Bot" ) ); + break; + + case F_Mask: + attrib = wxString( wxT( "Soldermask,Top" ) ); + break; + + case B_Mask: + attrib = wxString( wxT( "Soldermask,Bot" ) ); + break; + + case F_Paste: + attrib = wxString( wxT( "Paste,Top" ) ); + break; + + case B_Paste: + attrib = wxString( wxT( "Paste,Bot" ) ); + break; + + case Edge_Cuts: + // Board outline. + // Can be "Profile,NP" (Not Plated: usual) or "Profile,P" + // This last is the exception (Plated) + attrib = wxString( wxT( "Profile,NP" ) ); + break; + + case Dwgs_User: + attrib = wxString( wxT( "Drawing" ) ); + break; + + case Cmts_User: + attrib = wxString( wxT( "Other,Comment" ) ); + break; + + case Eco1_User: + attrib = wxString( wxT( "Other,ECO1" ) ); + break; + + case Eco2_User: + attrib = wxString( wxT( "Other,ECO2" ) ); + break; + + case B_Fab: + attrib = wxString( wxT( "Other,Fab,Bot" ) ); + break; + + case F_Fab: + attrib = wxString( wxT( "Other,Fab,Top" ) ); + break; + + case B_Cu: + attrib = wxString::Format( wxT( "Copper,L%d,Bot" ), aBoard->GetCopperLayerCount() ); + break; + + case F_Cu: + attrib = wxString::Format( wxT( "Copper,L1,Top" ) ); + break; + + default: + if( IsCopperLayer( aLayer ) ) + attrib = wxString::Format( wxT( "Copper,L%d,Inr" ), aLayer+1 ); + else + attrib = wxString::Format( wxT( "Other,User" ), aLayer+1 ); + break; + } + + // Add the signal type of the layer, if relevant + if( IsCopperLayer( aLayer ) ) + { + LAYER_T type = aBoard->GetLayerType( ToLAYER_ID( aLayer ) ); + + switch( type ) + { + case LT_SIGNAL: + attrib += wxString( wxT( ",Signal" ) ); + break; + case LT_POWER: + attrib += wxString( wxT( ",Plane" ) ); + break; + case LT_MIXED: + attrib += wxString( wxT( ",Mixed" ) ); + break; + default: + break; // do nothing (but avoid a warning for unhandled LAYER_T values from GCC) + } + } + + wxString fileFct; + + if( aUseX1CompatibilityMode ) + fileFct.Printf( "G04 #@! TF.FileFunction,%s*", GetChars( attrib ) ); + else + fileFct.Printf( "%%TF.FileFunction,%s*%%", GetChars( attrib ) ); + + return fileFct; +} + +/* Add some X2 attributes to the file header, as defined in the + * Gerber file format specification J4 and "Revision 2015.06" + */ +#define USE_REVISION_2015_06_ATTR +void AddGerberX2Attribute( PLOTTER * aPlotter, + const BOARD *aBoard, LAYER_NUM aLayer ) +{ + wxString text; + +#ifdef USE_REVISION_2015_06_ATTR + // Creates the TF,.GenerationSoftware. Format is: + // %TF,.GenerationSoftware,<vendor>,<application name>[,<application version>]*% + text.Printf( wxT( "%%TF.GenerationSoftware,KiCad,Pcbnew,%s*%%" ), GetBuildVersion() ); + aPlotter->AddLineToHeader( text ); + + // creates the TF.CreationDate ext: + // The attribute value must conform to the full version of the ISO 8601 + // date and time format, including time and time zone. Note that this is + // the date the Gerber file was effectively created, + // not the time the project of PCB was started + wxDateTime date( wxDateTime::GetTimeNow() ); + // Date format: see http://www.cplusplus.com/reference/ctime/strftime + wxString msg = date.Format( wxT( "%z" ) ); // Extract the time zone offset + // The time zone offset format is + (or -) mm or hhmm (mm = number of minutes, hh = number of hours) + // we want +(or -) hh:mm + if( msg.Len() > 3 ) + msg.insert( 3, ":", 1 ), + text.Printf( wxT( "%%TF.CreationDate,%s%s*%%" ), GetChars( date.FormatISOCombined() ), GetChars( msg ) ); + aPlotter->AddLineToHeader( text ); + + // Creates the TF,.ProjectId. Format is (from Gerber file format doc): + // %TF.ProjectId,<project id>,<project GUID>,<revision id>*% + // <project id> is the name of the project, restricted to basic ASCII symbols only, + // and comma not accepted + // All illegal chars will be replaced by underscore + // <project GUID> is a 32 hexadecimal digits string which is an unique id of a project. + // This is a random 128-bit number expressed in 32 hexadecimal digits. + // See en.wikipedia.org/wiki/GUID for more information + // However Kicad does not handle such a project GUID, so it is built from the board name + // Rem: <project id> accepts only ASCII 7 code (only basic ASCII codes are allowed in gerber files). + wxFileName fn = aBoard->GetFileName(); + msg = fn.GetFullName(); + wxString guid; + + // Build a 32 digits GUID from the board name: + for( unsigned ii = 0; ii < msg.Len(); ii++ ) + { + int cc1 = int( msg[ii] ) & 0x0F; + int cc2 = ( int( msg[ii] ) >> 4) & 0x0F; + guid << wxString::Format( wxT( "%X%X" ), cc2, cc1 ); + + if( guid.Len() >= 32 ) + break; + } + + // guid has 32 digits, so add missing digits + int cnt = 32 - guid.Len(); + + if( cnt > 0 ) + guid.Append( '0', cnt ); + + // build the <project id> string: this is the board short filename (without ext) + // and all non ASCII chars and comma are replaced by '_' + msg = fn.GetName(); + msg.Replace( wxT( "," ), wxT( "_" ) ); + + // build the <rec> string. All non ASCII chars and comma are replaced by '_' + wxString rev = ((BOARD*)aBoard)->GetTitleBlock().GetRevision(); + rev.Replace( wxT( "," ), wxT( "_" ) ); + + if( rev.IsEmpty() ) + rev = wxT( "rev?" ); + + text.Printf( wxT( "%%TF.ProjectId,%s,%s,%s*%%" ), msg.ToAscii(), GetChars( guid ), rev.ToAscii() ); + aPlotter->AddLineToHeader( text ); +#endif + + // Add the TF.FileFunction + text = GetGerberFileFunctionAttribute( aBoard, aLayer, false ); + aPlotter->AddLineToHeader( text ); +} + + +void BuildPlotFileName( wxFileName* aFilename, const wxString& aOutputDir, + const wxString& aSuffix, const wxString& aExtension ) +{ + // aFilename contains the base filename only (without path and extension) + // when calling this function. + // It is expected to be a valid filename (this is usually the board filename) + aFilename->SetPath( aOutputDir ); + + // Set the file extension + aFilename->SetExt( aExtension ); + + // remove leading and trailing spaces if any from the suffix, if + // something survives add it to the name; + // also the suffix can contain some not allowed chars in filename (/ \ . :), + // so change them to underscore + // Remember it can be called from a python script, so the illegal chars + // have to be filtered here. + wxString suffix = aSuffix; + suffix.Trim( true ); + suffix.Trim( false ); + + wxString badchars = wxFileName::GetForbiddenChars(wxPATH_DOS); + badchars.Append( '%' ); + + for( unsigned ii = 0; ii < badchars.Len(); ii++ ) + suffix.Replace( badchars[ii], wxT("_") ); + + if( !suffix.IsEmpty() ) + aFilename->SetName( aFilename->GetName() + wxT( "-" ) + suffix ); +} + + +PLOT_CONTROLLER::PLOT_CONTROLLER( BOARD *aBoard ) +{ + m_plotter = NULL; + m_board = aBoard; + m_plotLayer = UNDEFINED_LAYER; +} + + +PLOT_CONTROLLER::~PLOT_CONTROLLER() +{ + ClosePlot(); +} + + +/* IMPORTANT THING TO KNOW: the locale during plots *MUST* be kept as + * C/POSIX using a LOCALE_IO object on the stack. This even when + * opening/closing the plotfile, since some drivers do I/O even then */ + +void PLOT_CONTROLLER::ClosePlot() +{ + LOCALE_IO toggle; + + if( m_plotter ) + { + m_plotter->EndPlot(); + delete m_plotter; + m_plotter = NULL; + } +} + + +bool PLOT_CONTROLLER::OpenPlotfile( const wxString &aSuffix, + PlotFormat aFormat, + const wxString &aSheetDesc ) +{ + LOCALE_IO toggle; + + /* Save the current format: sadly some plot routines depends on this + but the main reason is that the StartPlot method uses it to + dispatch the plotter creation */ + GetPlotOptions().SetFormat( aFormat ); + + // Ensure that the previous plot is closed + ClosePlot(); + + // Now compute the full filename for the output and start the plot + // (after ensuring the output directory is OK) + wxString outputDirName = GetPlotOptions().GetOutputDirectory() ; + wxFileName outputDir = wxFileName::DirName( outputDirName ); + wxString boardFilename = m_board->GetFileName(); + + if( EnsureFileDirectoryExists( &outputDir, boardFilename ) ) + { + // outputDir contains now the full path of plot files + m_plotFile = boardFilename; + m_plotFile.SetPath( outputDir.GetPath() ); + wxString fileExt = GetDefaultPlotExtension( aFormat ); + + // Gerber format can use specific file ext, depending on layers + // (now not a good practice, because the official file ext is .gbr) + if( GetPlotOptions().GetFormat() == PLOT_FORMAT_GERBER && + GetPlotOptions().GetUseGerberProtelExtensions() ) + fileExt = GetGerberProtelExtension( GetLayer() ); + + // Build plot filenames from the board name and layer names: + BuildPlotFileName( &m_plotFile, outputDir.GetPath(), aSuffix, fileExt ); + + m_plotter = StartPlotBoard( m_board, &GetPlotOptions(), ToLAYER_ID( GetLayer() ), + m_plotFile.GetFullPath(), aSheetDesc ); + } + + return( m_plotter != NULL ); +} + + +bool PLOT_CONTROLLER::PlotLayer() +{ + LOCALE_IO toggle; + + // No plot open, nothing to do... + if( !m_plotter ) + return false; + + // Fully delegated to the parent + PlotOneBoardLayer( m_board, m_plotter, ToLAYER_ID( GetLayer() ), GetPlotOptions() ); + + return true; +} + + +void PLOT_CONTROLLER::SetColorMode( bool aColorMode ) +{ + if( !m_plotter ) + return; + + m_plotter->SetColorMode( aColorMode ); +} + + +bool PLOT_CONTROLLER::GetColorMode() +{ + if( !m_plotter ) + return false; + + return m_plotter->GetColorMode(); +} |