summaryrefslogtreecommitdiff
path: root/pcbnew/class_pcb_layer_widget.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'pcbnew/class_pcb_layer_widget.cpp')
-rw-r--r--pcbnew/class_pcb_layer_widget.cpp528
1 files changed, 528 insertions, 0 deletions
diff --git a/pcbnew/class_pcb_layer_widget.cpp b/pcbnew/class_pcb_layer_widget.cpp
new file mode 100644
index 0000000..17d041f
--- /dev/null
+++ b/pcbnew/class_pcb_layer_widget.cpp
@@ -0,0 +1,528 @@
+/*
+ * This program source code file is part of KiCad, a free EDA CAD application.
+ *
+ * Copyright (C) 2004-2015 Jean-Pierre Charras, jean-pierre.charras@gpisa-lab.inpg.fr
+ * Copyright (C) 2010-2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
+ * Copyright (C) 2010-2015 KiCad Developers, see change_log.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
+ */
+
+
+/******************************************************/
+/* class_pcb_layer_widget.cpp - Pcbnew layers manager */
+/******************************************************/
+
+#include <fctsys.h>
+#include <pgm_base.h>
+#include <class_drawpanel.h>
+#include <class_draw_panel_gal.h>
+#include <view/view.h>
+#include <painter.h>
+
+#include <confirm.h>
+#include <wxPcbStruct.h>
+#include <pcbstruct.h> // enum PCB_VISIBLE
+#include <layer_widget.h>
+#include <macros.h>
+
+#include <class_board.h>
+#include <class_pcb_layer_widget.h>
+
+#include <pcbnew.h>
+#include <collectors.h>
+#include <pcbnew_id.h>
+
+#include <gal/graphics_abstraction_layer.h>
+
+
+/// This is a read only template that is copied and modified before adding to LAYER_WIDGET
+const LAYER_WIDGET::ROW PCB_LAYER_WIDGET::s_render_rows[] = {
+
+#define RR LAYER_WIDGET::ROW // Render Row abbreviation to reduce source width
+
+ // text id color tooltip
+ RR( _( "Through Via" ), VIA_THROUGH_VISIBLE, WHITE, _( "Show through vias" ) ),
+ RR( _( "Bl/Buried Via" ), VIA_BBLIND_VISIBLE, WHITE, _( "Show blind or buried vias" ) ),
+ RR( _( "Micro Via" ), VIA_MICROVIA_VISIBLE, WHITE, _( "Show micro vias") ),
+ RR( _( "Non Plated" ), NON_PLATED_VISIBLE, WHITE, _( "Show non plated holes") ),
+ RR( _( "Ratsnest" ), RATSNEST_VISIBLE, WHITE, _( "Show unconnected nets as a ratsnest") ),
+
+ RR( _( "Pads Front" ), PAD_FR_VISIBLE, WHITE, _( "Show footprint pads on board's front" ) ),
+ RR( _( "Pads Back" ), PAD_BK_VISIBLE, WHITE, _( "Show footprint pads on board's back" ) ),
+
+ RR( _( "Text Front" ), MOD_TEXT_FR_VISIBLE, UNSPECIFIED_COLOR, _( "Show footprint text on board's front" ) ),
+ RR( _( "Text Back" ), MOD_TEXT_BK_VISIBLE, UNSPECIFIED_COLOR, _( "Show footprint text on board's back" ) ),
+ RR( _( "Hidden Text" ), MOD_TEXT_INVISIBLE, WHITE, _( "Show footprint text marked as invisible" ) ),
+
+ RR( _( "Anchors" ), ANCHOR_VISIBLE, WHITE, _( "Show footprint and text origins as a cross" ) ),
+ RR( _( "Grid" ), GRID_VISIBLE, WHITE, _( "Show the (x,y) grid dots" ) ),
+ RR( _( "No-Connects" ), NO_CONNECTS_VISIBLE, UNSPECIFIED_COLOR, _( "Show a marker on pads which have no net connected" ) ),
+ RR( _( "Footprints Front" ), MOD_FR_VISIBLE, UNSPECIFIED_COLOR, _( "Show footprints that are on board's front") ),
+ RR( _( "Footprints Back" ), MOD_BK_VISIBLE, UNSPECIFIED_COLOR, _( "Show footprints that are on board's back") ),
+ RR( _( "Values" ), MOD_VALUES_VISIBLE, UNSPECIFIED_COLOR, _( "Show footprint's values") ),
+ RR( _( "References" ), MOD_REFERENCES_VISIBLE, UNSPECIFIED_COLOR, _( "Show footprint's references") ),
+};
+
+static int s_allowed_in_FpEditor[] =
+{
+ MOD_TEXT_INVISIBLE, PAD_FR_VISIBLE, PAD_BK_VISIBLE,
+ GRID_VISIBLE, MOD_VALUES_VISIBLE, MOD_REFERENCES_VISIBLE
+};
+
+
+PCB_LAYER_WIDGET::PCB_LAYER_WIDGET( PCB_BASE_FRAME* aParent, wxWindow* aFocusOwner,
+ int aPointSize, bool aFpEditorMode ) :
+ LAYER_WIDGET( aParent, aFocusOwner, aPointSize ),
+ myframe( aParent )
+{
+ m_alwaysShowActiveCopperLayer = false;
+ m_fp_editor_mode = aFpEditorMode;
+ ReFillRender();
+
+ // Update default tabs labels
+ SetLayersManagerTabsText();
+
+ //-----<Popup menu>-------------------------------------------------
+ // handle the popup menu over the layer window.
+ m_LayerScrolledWindow->Connect( wxEVT_RIGHT_DOWN,
+ wxMouseEventHandler( PCB_LAYER_WIDGET::onRightDownLayers ), NULL, this );
+
+ // since Popupmenu() calls this->ProcessEvent() we must call this->Connect()
+ // and not m_LayerScrolledWindow->Connect()
+ Connect( ID_SHOW_ALL_COPPERS, ID_ALWAYS_SHOW_NO_COPPERS_BUT_ACTIVE,
+ wxEVT_COMMAND_MENU_SELECTED,
+ wxCommandEventHandler( PCB_LAYER_WIDGET::onPopupSelection ), NULL, this );
+
+ // install the right click handler into each control at end of ReFill()
+ // using installRightLayerClickHandler
+}
+
+
+bool PCB_LAYER_WIDGET::isAllowedInFpMode( int aId )
+{
+ for( unsigned ii = 0; ii < DIM( s_allowed_in_FpEditor ); ii++ )
+ if( s_allowed_in_FpEditor[ii] == aId )
+ return true;
+
+ return false;
+}
+
+
+bool PCB_LAYER_WIDGET::isLayerAllowedInFpMode( LAYER_ID aLayer )
+{
+ static LSET allowed = LSET::AllTechMask();
+ // Currently not in use because putting a graphic item on a copper layer
+ // is not currently supported by DRC.
+ // allowed.set( F_Cu ).set( B_Cu );
+ return allowed.test( aLayer );
+}
+
+
+void PCB_LAYER_WIDGET::installRightLayerClickHandler()
+{
+ int rowCount = GetLayerRowCount();
+
+ for( int row=0; row < rowCount; ++row )
+ {
+ for( int col=0; col<LYR_COLUMN_COUNT; ++col )
+ {
+ wxWindow* w = getLayerComp( row, col );
+
+ w->Connect( wxEVT_RIGHT_DOWN, wxMouseEventHandler(
+ PCB_LAYER_WIDGET::onRightDownLayers ), NULL, this );
+ }
+ }
+}
+
+
+void PCB_LAYER_WIDGET::onRightDownLayers( wxMouseEvent& event )
+{
+ wxMenu menu;
+
+ // menu text is capitalized:
+ // http://library.gnome.org/devel/hig-book/2.20/design-text-labels.html.en#layout-capitalization
+ menu.Append( new wxMenuItem( &menu, ID_SHOW_ALL_COPPERS,
+ _( "Show All Copper Layers" ) ) );
+ menu.Append( new wxMenuItem( &menu, ID_SHOW_NO_COPPERS_BUT_ACTIVE,
+ _( "Hide All Copper Layers But Active" ) ) );
+ menu.Append( new wxMenuItem( &menu, ID_ALWAYS_SHOW_NO_COPPERS_BUT_ACTIVE,
+ _( "Always Hide All Copper Layers But Active" ) ) );
+ menu.Append( new wxMenuItem( &menu, ID_SHOW_NO_COPPERS,
+ _( "Hide All Copper Layers" ) ) );
+
+ PopupMenu( &menu );
+
+ passOnFocus();
+}
+
+
+void PCB_LAYER_WIDGET::onPopupSelection( wxCommandEvent& event )
+{
+ int rowCount;
+ int menuId = event.GetId();
+ bool visible;
+ bool force_active_layer_visible;
+
+ visible = menuId == ID_SHOW_ALL_COPPERS;
+ m_alwaysShowActiveCopperLayer = ( menuId == ID_ALWAYS_SHOW_NO_COPPERS_BUT_ACTIVE );
+ force_active_layer_visible = ( menuId == ID_SHOW_NO_COPPERS_BUT_ACTIVE ||
+ menuId == ID_ALWAYS_SHOW_NO_COPPERS_BUT_ACTIVE );
+
+ switch( menuId )
+ {
+ case ID_SHOW_ALL_COPPERS:
+ case ID_ALWAYS_SHOW_NO_COPPERS_BUT_ACTIVE:
+ case ID_SHOW_NO_COPPERS_BUT_ACTIVE:
+ case ID_SHOW_NO_COPPERS:
+ // Search the last copper layer row index:
+ int lastCu = -1;
+ rowCount = GetLayerRowCount();
+ for( int row = rowCount-1; row>=0; --row )
+ {
+ wxCheckBox* cb = (wxCheckBox*) getLayerComp( row, COLUMN_COLOR_LYR_CB );
+ LAYER_ID layer = ToLAYER_ID( getDecodedId( cb->GetId() ) );
+
+ if( IsCopperLayer( layer ) )
+ {
+ lastCu = row;
+ break;
+ }
+ }
+
+ // Enable/disable the copper layers visibility:
+ for( int row=0; row<rowCount; ++row )
+ {
+ wxCheckBox* cb = (wxCheckBox*) getLayerComp( row, COLUMN_COLOR_LYR_CB );
+ LAYER_ID layer = ToLAYER_ID( getDecodedId( cb->GetId() ) );
+
+ if( IsCopperLayer( layer ) )
+ {
+ bool loc_visible = visible;
+
+ if( force_active_layer_visible && (layer == myframe->GetActiveLayer() ) )
+ loc_visible = true;
+
+ cb->SetValue( loc_visible );
+
+ bool isLastCopperLayer = (row==lastCu);
+ OnLayerVisible( layer, loc_visible, isLastCopperLayer );
+
+ if( isLastCopperLayer )
+ break;
+ }
+ }
+ break;
+ }
+}
+
+
+void PCB_LAYER_WIDGET::SetLayersManagerTabsText()
+{
+ m_notebook->SetPageText( 0, _( "Layer" ) );
+ m_notebook->SetPageText( 1, _( "Render" ) );
+}
+
+
+void PCB_LAYER_WIDGET::ReFillRender()
+{
+ BOARD* board = myframe->GetBoard();
+ ClearRenderRows();
+
+ // Add "Render" tab rows to LAYER_WIDGET, after setting color and checkbox state.
+ // Because s_render_rows is created static, we must explicitly call
+ // wxGetTranslation for texts which are internationalized (tool tips
+ // and item names)
+ for( unsigned row=0; row<DIM(s_render_rows); ++row )
+ {
+ LAYER_WIDGET::ROW renderRow = s_render_rows[row];
+
+ if( m_fp_editor_mode && !isAllowedInFpMode( renderRow.id ) )
+ continue;
+
+ renderRow.tooltip = wxGetTranslation( s_render_rows[row].tooltip );
+ renderRow.rowName = wxGetTranslation( s_render_rows[row].rowName );
+
+ if( renderRow.color != -1 ) // does this row show a color?
+ {
+ // this window frame must have an established BOARD, i.e. after SetBoard()
+ renderRow.color = board->GetVisibleElementColor( renderRow.id );
+ }
+
+ renderRow.state = board->IsElementVisible( renderRow.id );
+
+ AppendRenderRow( renderRow );
+ }
+}
+
+
+void PCB_LAYER_WIDGET::SyncRenderStates()
+{
+ BOARD* board = myframe->GetBoard();
+
+ for( unsigned row=0; row<DIM(s_render_rows); ++row )
+ {
+ int rowId = s_render_rows[row].id;
+
+ if( m_fp_editor_mode && !isAllowedInFpMode( rowId ) )
+ continue;
+
+ // this does not fire a UI event
+ SetRenderState( rowId, board->IsElementVisible( rowId ) );
+ }
+}
+
+
+void PCB_LAYER_WIDGET::SyncLayerVisibilities()
+{
+ BOARD* board = myframe->GetBoard();
+ int count = GetLayerRowCount();
+
+ for( int row=0; row<count; ++row )
+ {
+ // this utilizes more implementation knowledge than ideal, eventually
+ // add member ROW getRow() or similar to base LAYER_WIDGET.
+
+ wxWindow* w = getLayerComp( row, COLUMN_ICON_ACTIVE );
+
+ LAYER_ID layerId = ToLAYER_ID( getDecodedId( w->GetId() ) );
+
+ // this does not fire a UI event
+ SetLayerVisible( layerId, board->IsLayerVisible( layerId ) );
+ }
+}
+
+
+void PCB_LAYER_WIDGET::ReFill()
+{
+ BOARD* brd = myframe->GetBoard();
+ LSET enabled = brd->GetEnabledLayers();
+
+ ClearLayerRows();
+
+ wxString dsc;
+
+ // show all coppers first, with front on top, back on bottom, then technical layers
+ for( LSEQ cu_stack = enabled.CuStack(); cu_stack; ++cu_stack )
+ {
+ LAYER_ID layer = *cu_stack;
+
+ switch( layer )
+ {
+ case F_Cu:
+ dsc = _( "Front copper layer" );
+ break;
+
+ case B_Cu:
+ dsc = _( "Back copper layer" );
+ break;
+
+ default:
+ dsc = _( "Inner copper layer" );
+ break;
+ }
+
+ AppendLayerRow( LAYER_WIDGET::ROW(
+ brd->GetLayerName( layer ), layer, brd->GetLayerColor( layer ),
+ dsc, true ) );
+
+ if( m_fp_editor_mode && !isLayerAllowedInFpMode( layer ) )
+ {
+ getLayerComp( GetLayerRowCount()-1, COLUMN_COLOR_LYRNAME )->Enable( false );
+ getLayerComp( GetLayerRowCount()-1,
+ COLUMN_COLORBM )->SetToolTip( wxEmptyString );
+ }
+ }
+
+
+ // technical layers are shown in this order:
+ // Because they are static, wxGetTranslation must be explicitly
+ // called for tooltips.
+ static const struct {
+ LAYER_ID layerId;
+ wxString tooltip;
+ } non_cu_seq[] = {
+ { F_Adhes, _( "Adhesive on board's front" ) },
+ { B_Adhes, _( "Adhesive on board's back" ) },
+ { F_Paste, _( "Solder paste on board's front" ) },
+ { B_Paste, _( "Solder paste on board's back" ) },
+ { F_SilkS, _( "Silkscreen on board's front" ) },
+ { B_SilkS, _( "Silkscreen on board's back" ) },
+ { F_Mask, _( "Solder mask on board's front" ) },
+ { B_Mask, _( "Solder mask on board's back" ) },
+ { Dwgs_User, _( "Explanatory drawings" ) },
+ { Cmts_User, _( "Explanatory comments" ) },
+ { Eco1_User, _( "User defined meaning" ) },
+ { Eco2_User, _( "User defined meaning" ) },
+ { Edge_Cuts, _( "Board's perimeter definition" ) },
+ { Margin, _( "Board's edge setback outline" ) },
+ { F_CrtYd, _( "Footprint courtyards on board's front" ) },
+ { B_CrtYd, _( "Footprint courtyards on board's back" ) },
+ { F_Fab, _( "Footprint assembly on board's front" ) },
+ { B_Fab, _( "Footprint assembly on board's back" ) }
+ };
+
+ for( unsigned i=0; i<DIM( non_cu_seq ); ++i )
+ {
+ LAYER_ID layer = non_cu_seq[i].layerId;
+
+ if( !enabled[layer] )
+ continue;
+
+ AppendLayerRow( LAYER_WIDGET::ROW(
+ brd->GetLayerName( layer ), layer, brd->GetLayerColor( layer ),
+ wxGetTranslation( non_cu_seq[i].tooltip ), true ) );
+
+ if( m_fp_editor_mode && !isLayerAllowedInFpMode( layer ) )
+ {
+ getLayerComp( GetLayerRowCount()-1, COLUMN_COLOR_LYRNAME )->Enable( false );
+ getLayerComp( GetLayerRowCount()-1,
+ COLUMN_COLORBM )->SetToolTip( wxEmptyString );
+ }
+ }
+
+ installRightLayerClickHandler();
+}
+
+
+//-----<LAYER_WIDGET callbacks>-------------------------------------------
+
+void PCB_LAYER_WIDGET::OnLayerColorChange( int aLayer, EDA_COLOR_T aColor )
+{
+ myframe->GetBoard()->SetLayerColor( ToLAYER_ID( aLayer ), aColor );
+
+ if( myframe->IsGalCanvasActive() )
+ {
+ KIGFX::VIEW* view = myframe->GetGalCanvas()->GetView();
+ view->GetPainter()->GetSettings()->ImportLegacyColors( myframe->GetBoard()->GetColorsSettings() );
+ view->UpdateLayerColor( aLayer );
+ }
+
+ myframe->GetCanvas()->Refresh();
+}
+
+
+bool PCB_LAYER_WIDGET::OnLayerSelect( int aLayer )
+{
+ // the layer change from the PCB_LAYER_WIDGET can be denied by returning
+ // false from this function.
+ LAYER_ID layer = ToLAYER_ID( aLayer );
+
+ if( m_fp_editor_mode && !isLayerAllowedInFpMode( layer ) )
+ return false;
+
+ myframe->SetActiveLayer( layer );
+ DISPLAY_OPTIONS* displ_opts = (DISPLAY_OPTIONS*)myframe->GetDisplayOptions();
+
+ if( m_alwaysShowActiveCopperLayer )
+ OnLayerSelected();
+ else if( displ_opts->m_ContrastModeDisplay )
+ myframe->GetCanvas()->Refresh();
+
+ return true;
+}
+
+
+bool PCB_LAYER_WIDGET::OnLayerSelected()
+{
+ if( !m_alwaysShowActiveCopperLayer )
+ return false;
+
+ // postprocess after an active layer selection
+ // ensure active layer visible
+ wxCommandEvent event;
+ event.SetId( ID_ALWAYS_SHOW_NO_COPPERS_BUT_ACTIVE );
+ onPopupSelection( event );
+
+ return true;
+}
+
+
+void PCB_LAYER_WIDGET::OnLayerVisible( int aLayer, bool isVisible, bool isFinal )
+{
+ BOARD* brd = myframe->GetBoard();
+
+ LSET visibleLayers = brd->GetVisibleLayers();
+
+ visibleLayers.set( aLayer, isVisible );
+
+ brd->SetVisibleLayers( visibleLayers );
+
+ EDA_DRAW_PANEL_GAL* galCanvas = myframe->GetGalCanvas();
+
+ if( galCanvas )
+ {
+ KIGFX::VIEW* view = galCanvas->GetView();
+ view->SetLayerVisible( aLayer, isVisible );
+ view->RecacheAllItems( true );
+ }
+
+ if( isFinal )
+ myframe->GetCanvas()->Refresh();
+}
+
+
+void PCB_LAYER_WIDGET::OnRenderColorChange( int aId, EDA_COLOR_T aColor )
+{
+ myframe->GetBoard()->SetVisibleElementColor( aId, aColor );
+
+ if( myframe->GetGalCanvas() )
+ {
+ KIGFX::VIEW* view = myframe->GetGalCanvas()->GetView();
+ view->GetPainter()->GetSettings()->ImportLegacyColors( myframe->GetBoard()->GetColorsSettings() );
+ view->UpdateLayerColor( aId );
+ }
+
+ myframe->GetCanvas()->Refresh();
+}
+
+
+void PCB_LAYER_WIDGET::OnRenderEnable( int aId, bool isEnabled )
+{
+ BOARD* brd = myframe->GetBoard();
+
+ LSET visibleLayers = brd->GetVisibleLayers();
+ visibleLayers.set( aId, isEnabled );
+
+ // The layer visibility status is saved in the board file so set the board modified
+ // state so the user has the option to save the changes.
+ if( brd->IsElementVisible( aId ) != isEnabled )
+ myframe->OnModify();
+
+ brd->SetElementVisibility( aId, isEnabled );
+
+ EDA_DRAW_PANEL_GAL* galCanvas = myframe->GetGalCanvas();
+
+ if( galCanvas )
+ {
+ if( aId == GRID_VISIBLE )
+ {
+ galCanvas->GetGAL()->SetGridVisibility( myframe->IsGridVisible() );
+ galCanvas->GetView()->MarkTargetDirty( KIGFX::TARGET_NONCACHED );
+ }
+ else
+ galCanvas->GetView()->SetLayerVisible( ITEM_GAL_LAYER( aId ), isEnabled );
+ }
+
+ if( galCanvas && myframe->IsGalCanvasActive() )
+ galCanvas->Refresh();
+ else
+ myframe->GetCanvas()->Refresh();
+}
+
+//-----</LAYER_WIDGET callbacks>------------------------------------------