2009-04-05 20:49:15 +00:00
|
|
|
/*
|
2011-09-30 18:15:37 +00:00
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
2009-04-05 20:49:15 +00:00
|
|
|
*
|
|
|
|
* Copyright (C) 2008 Wayne Stambaugh <stambaughw@verizon.net>
|
2011-09-30 18:15:37 +00:00
|
|
|
* Copyright (C) 1992-2008 KiCad Developers, see change_log.txt for contributors.
|
2009-04-05 20:49:15 +00:00
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This file contains the global constants and variables used in the PCB
|
2011-09-30 18:15:37 +00:00
|
|
|
* applications Pcbnew, CvPcb, and GervView. The goal of this was to
|
2009-04-05 20:49:15 +00:00
|
|
|
* unobfuscate the original header file design that made it very difficult
|
|
|
|
* to figure out where these variables lived. Ideally, they should be pushed
|
|
|
|
* back into the application layer.
|
|
|
|
*/
|
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <fctsys.h>
|
|
|
|
#include <pcbcommon.h>
|
|
|
|
#include <plot_common.h>
|
2011-09-23 13:57:12 +00:00
|
|
|
|
2013-04-07 11:55:18 +00:00
|
|
|
#include <class_board.h>
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <class_pad.h>
|
2012-02-06 05:44:19 +00:00
|
|
|
#include <class_zone_settings.h>
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <class_board_design_settings.h>
|
2009-04-05 20:49:15 +00:00
|
|
|
|
2011-09-23 13:57:12 +00:00
|
|
|
|
|
|
|
class MODULE;
|
|
|
|
|
|
|
|
|
2009-04-05 20:49:15 +00:00
|
|
|
/* Look up Table for conversion copper layer count -> general copper layer
|
|
|
|
* mask: */
|
2013-03-30 17:24:04 +00:00
|
|
|
LAYER_MSK g_TabAllCopperLayerMask[NB_COPPER_LAYERS] = {
|
2009-04-05 20:49:15 +00:00
|
|
|
0x0001, 0x8001, 0x8003, 0x8007,
|
|
|
|
0x800F, 0x801F, 0x803F, 0x807F,
|
|
|
|
0x80FF, 0x81FF, 0x83FF, 0x87FF,
|
|
|
|
0x8FFF, 0x9FFF, 0xCFFF, 0xFFFF
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-04-07 11:55:18 +00:00
|
|
|
DISPLAY_OPTIONS DisplayOpt; // Display options for board items
|
2009-04-05 20:49:15 +00:00
|
|
|
|
|
|
|
int g_AnchorColor = BLUE;
|
|
|
|
int g_ModuleTextCMPColor = LIGHTGRAY;
|
|
|
|
int g_ModuleTextCUColor = MAGENTA;
|
|
|
|
int g_ModuleTextNOVColor = DARKGRAY;
|
|
|
|
int g_PadCUColor = GREEN;
|
|
|
|
int g_PadCMPColor = RED;
|
|
|
|
|
2009-11-23 15:16:50 +00:00
|
|
|
|
2009-04-05 20:49:15 +00:00
|
|
|
/**
|
|
|
|
* Used in track creation, a list of track segments currently being created,
|
|
|
|
* with the newest track at the end of the list, sorted by new-ness. e.g. use
|
|
|
|
* TRACK->Back() to get the next older track, TRACK->Next() to get the next
|
|
|
|
* newer track.
|
|
|
|
*/
|
|
|
|
DLIST<TRACK> g_CurrentTrackList;
|
|
|
|
|
2013-04-05 19:04:58 +00:00
|
|
|
LAYER_NUM FlipLayer( LAYER_NUM oldlayer )
|
|
|
|
{
|
|
|
|
switch( oldlayer )
|
|
|
|
{
|
|
|
|
case LAYER_N_BACK:
|
|
|
|
return LAYER_N_FRONT;
|
|
|
|
|
|
|
|
case LAYER_N_FRONT:
|
|
|
|
return LAYER_N_BACK;
|
|
|
|
|
|
|
|
case SILKSCREEN_N_BACK:
|
|
|
|
return SILKSCREEN_N_FRONT;
|
|
|
|
|
|
|
|
case SILKSCREEN_N_FRONT:
|
|
|
|
return SILKSCREEN_N_BACK;
|
|
|
|
|
|
|
|
case ADHESIVE_N_BACK:
|
|
|
|
return ADHESIVE_N_FRONT;
|
|
|
|
|
|
|
|
case ADHESIVE_N_FRONT:
|
|
|
|
return ADHESIVE_N_BACK;
|
|
|
|
|
|
|
|
case SOLDERMASK_N_BACK:
|
|
|
|
return SOLDERMASK_N_FRONT;
|
|
|
|
|
|
|
|
case SOLDERMASK_N_FRONT:
|
|
|
|
return SOLDERMASK_N_BACK;
|
|
|
|
|
|
|
|
case SOLDERPASTE_N_BACK:
|
|
|
|
return SOLDERPASTE_N_FRONT;
|
|
|
|
|
|
|
|
case SOLDERPASTE_N_FRONT:
|
|
|
|
return SOLDERPASTE_N_BACK;
|
|
|
|
|
|
|
|
// No change for the other layers
|
|
|
|
default:
|
|
|
|
return oldlayer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
LAYER_MSK FlipLayerMask( LAYER_MSK aMask )
|
|
|
|
{
|
|
|
|
LAYER_MSK newMask;
|
|
|
|
|
|
|
|
newMask = aMask & ~(LAYER_BACK | LAYER_FRONT |
|
|
|
|
SILKSCREEN_LAYER_BACK | SILKSCREEN_LAYER_FRONT |
|
|
|
|
ADHESIVE_LAYER_BACK | ADHESIVE_LAYER_FRONT |
|
|
|
|
SOLDERMASK_LAYER_BACK | SOLDERMASK_LAYER_FRONT |
|
|
|
|
SOLDERPASTE_LAYER_BACK | SOLDERPASTE_LAYER_FRONT |
|
|
|
|
ADHESIVE_LAYER_BACK | ADHESIVE_LAYER_FRONT);
|
|
|
|
|
|
|
|
if( aMask & LAYER_BACK )
|
|
|
|
newMask |= LAYER_FRONT;
|
|
|
|
|
|
|
|
if( aMask & LAYER_FRONT )
|
|
|
|
newMask |= LAYER_BACK;
|
|
|
|
|
|
|
|
if( aMask & SILKSCREEN_LAYER_BACK )
|
|
|
|
newMask |= SILKSCREEN_LAYER_FRONT;
|
|
|
|
|
|
|
|
if( aMask & SILKSCREEN_LAYER_FRONT )
|
|
|
|
newMask |= SILKSCREEN_LAYER_BACK;
|
|
|
|
|
|
|
|
if( aMask & ADHESIVE_LAYER_BACK )
|
|
|
|
newMask |= ADHESIVE_LAYER_FRONT;
|
|
|
|
|
|
|
|
if( aMask & ADHESIVE_LAYER_FRONT )
|
|
|
|
newMask |= ADHESIVE_LAYER_BACK;
|
|
|
|
|
|
|
|
if( aMask & SOLDERMASK_LAYER_BACK )
|
|
|
|
newMask |= SOLDERMASK_LAYER_FRONT;
|
|
|
|
|
|
|
|
if( aMask & SOLDERMASK_LAYER_FRONT )
|
|
|
|
newMask |= SOLDERMASK_LAYER_BACK;
|
|
|
|
|
|
|
|
if( aMask & SOLDERPASTE_LAYER_BACK )
|
|
|
|
newMask |= SOLDERPASTE_LAYER_FRONT;
|
|
|
|
|
|
|
|
if( aMask & SOLDERPASTE_LAYER_FRONT )
|
|
|
|
newMask |= SOLDERPASTE_LAYER_BACK;
|
|
|
|
|
|
|
|
if( aMask & ADHESIVE_LAYER_BACK )
|
|
|
|
newMask |= ADHESIVE_LAYER_FRONT;
|
|
|
|
|
|
|
|
if( aMask & ADHESIVE_LAYER_FRONT )
|
|
|
|
newMask |= ADHESIVE_LAYER_BACK;
|
|
|
|
|
|
|
|
return newMask;
|
|
|
|
}
|
|
|
|
|
|
|
|
LAYER_NUM ExtractLayer( LAYER_MSK aMask )
|
|
|
|
{
|
|
|
|
if( aMask == NO_LAYERS )
|
|
|
|
return UNSELECTED_LAYER;
|
|
|
|
|
|
|
|
LAYER_NUM candidate = UNDEFINED_LAYER;
|
|
|
|
|
|
|
|
// Scan all the layers and take note of the first set; if other are
|
|
|
|
// then found return UNDEFINED_LAYER
|
|
|
|
for( LAYER_NUM i = FIRST_LAYER; i < NB_LAYERS; ++i )
|
|
|
|
{
|
|
|
|
if( aMask & GetLayerMask( i ) )
|
|
|
|
{
|
|
|
|
if( candidate == UNDEFINED_LAYER )
|
|
|
|
candidate = i;
|
|
|
|
else
|
|
|
|
return UNDEFINED_LAYER;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return candidate;
|
|
|
|
}
|
2013-04-07 11:55:18 +00:00
|
|
|
|
|
|
|
wxString LayerMaskDescribe( const BOARD *aBoard, LAYER_MSK aMask )
|
|
|
|
{
|
|
|
|
// Try the single or no- layer case (easy)
|
|
|
|
LAYER_NUM layer = ExtractLayer( aMask );
|
|
|
|
switch( layer )
|
|
|
|
{
|
|
|
|
case UNSELECTED_LAYER:
|
|
|
|
return _( "No layers" );
|
|
|
|
|
|
|
|
case UNDEFINED_LAYER:
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return aBoard->GetLayerName( layer );
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to be smart and useful, starting with outer copper
|
|
|
|
// (which are more important than internal ones)
|
|
|
|
wxString layerInfo;
|
|
|
|
if( aMask & LAYER_FRONT )
|
|
|
|
AccumulateDescription( layerInfo, aBoard->GetLayerName( LAYER_N_FRONT ) );
|
|
|
|
|
|
|
|
if( aMask & LAYER_BACK )
|
|
|
|
AccumulateDescription( layerInfo, aBoard->GetLayerName( LAYER_N_BACK ) );
|
|
|
|
|
|
|
|
if( aMask & INTERNAL_CU_LAYERS )
|
|
|
|
AccumulateDescription( layerInfo, _("Internal" ) );
|
|
|
|
|
|
|
|
if( aMask & ALL_NO_CU_LAYERS )
|
|
|
|
AccumulateDescription( layerInfo, _("Non-copper" ) );
|
|
|
|
|
|
|
|
return layerInfo;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AccumulateDescription( wxString &aDesc, const wxString &aItem )
|
|
|
|
{
|
|
|
|
if( !aDesc.IsEmpty() )
|
|
|
|
aDesc << wxT(", ");
|
|
|
|
aDesc << aItem;
|
|
|
|
}
|