KiCad PCB EDA Suite
Loading...
Searching...
No Matches
dialog_footprint_properties_fp_editor.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2018 Jean-Pierre Charras, jp.charras at wanadoo.fr
5 * Copyright (C) 2015 Dick Hollenbeck, [email protected]
6 * Copyright (C) 2008 Wayne Stambaugh <[email protected]>
7 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
8 *
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version 2
12 * of the License, or (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, you may find one here:
21 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
22 * or you may search the http://www.gnu.org website for the version 2 license,
23 * or you may write to the Free Software Foundation, Inc.,
24 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
25 */
26
28
29#include <wx/debug.h>
30#include <wx/tokenzr.h>
31
34#include <bitmaps.h>
35#include <board_commit.h>
37#include <confirm.h>
38
41#include <embedded_files.h>
42#include <filename_resolver.h>
43#include <footprint.h>
45#include <pad.h>
48#include <layer_utils.h>
49#include <kiplatform/ui.h>
52#include <pgm_base.h>
54#include <tool/tool_manager.h>
55#include <tools/pcb_actions.h>
57#include <validators.h>
61#include <widgets/wx_grid.h>
62
64#include <project_pcb.h>
65#include <kidialog.h>
66
67
68class LAYERS_GRID_TABLE : public WX_GRID_TABLE_BASE, public std::vector<PCB_LAYER_ID>
69{
70public:
71 LAYERS_GRID_TABLE( PCB_BASE_FRAME* aFrame, const LSET& aForbiddenLayers );
73
74 int GetNumberRows() override { return (int) size(); }
75 int GetNumberCols() override { return 1; }
76
77 bool CanGetValueAs( int aRow, int aCol, const wxString& aTypeName ) override;
78 bool CanSetValueAs( int aRow, int aCol, const wxString& aTypeName ) override;
79 wxGridCellAttr* GetAttr( int aRow, int aCol, wxGridCellAttr::wxAttrKind aKind ) override;
80
81 wxString GetValue( int aRow, int aCol ) override;
82 long GetValueAsLong( int aRow, int aCol ) override;
83
84 void SetValue( int aRow, int aCol, const wxString& aValue ) override;
85 void SetValueAsLong( int aRow, int aCol, long aValue ) override;
86
87private:
89 wxGridCellAttr* m_layerColAttr;
90};
91
92
93LAYERS_GRID_TABLE::LAYERS_GRID_TABLE( PCB_BASE_FRAME* aFrame, const LSET& aForbiddenLayers ) :
94 m_frame( aFrame )
95{
96 m_layerColAttr = new wxGridCellAttr;
97 m_layerColAttr->SetRenderer( new GRID_CELL_LAYER_RENDERER( m_frame ) );
98
99 m_layerColAttr->SetEditor( new GRID_CELL_LAYER_SELECTOR( m_frame, aForbiddenLayers, true ) );
100}
101
102
107
108
109bool LAYERS_GRID_TABLE::CanGetValueAs( int aRow, int aCol, const wxString& aTypeName )
110{
111 return aTypeName == wxGRID_VALUE_NUMBER;
112}
113
114
115bool LAYERS_GRID_TABLE::CanSetValueAs( int aRow, int aCol, const wxString& aTypeName )
116{
117 return aTypeName == wxGRID_VALUE_NUMBER;
118}
119
120
121wxGridCellAttr* LAYERS_GRID_TABLE::GetAttr( int aRow, int aCol, wxGridCellAttr::wxAttrKind aKind )
122{
123 m_layerColAttr->IncRef();
124 return enhanceAttr( m_layerColAttr, aRow, aCol, aKind );
125}
126
127
128wxString LAYERS_GRID_TABLE::GetValue( int aRow, int aCol )
129{
130 return m_frame->GetBoard()->GetLayerName( this->at( (size_t) aRow ) );
131}
132
133
134long LAYERS_GRID_TABLE::GetValueAsLong( int aRow, int aCol )
135{
136 return this->at( (size_t) aRow );
137}
138
139
140void LAYERS_GRID_TABLE::SetValue( int aRow, int aCol, const wxString& aValue )
141{
142 wxFAIL_MSG( wxString::Format( wxT( "column %d doesn't hold a string value" ), aCol ) );
143}
144
145
146void LAYERS_GRID_TABLE::SetValueAsLong( int aRow, int aCol, long aValue )
147{
148 this->at( (size_t) aRow ) = ToLAYER_ID( (int) aValue );
149}
150
151
152// Remember the last open page during session.
153
155
156
158 FOOTPRINT* aFootprint ) :
160 m_frame( aParent ),
161 m_footprint( aFootprint ),
162 m_initialized( false ),
166{
167 SetEvtHandlerEnabled( false );
168
169 // Create the extra panels.
172
173 m_NoteBook->AddPage( m_3dPanel, _("3D Models"), false );
174 m_NoteBook->AddPage( m_embeddedFiles, _( "Embedded Files" ) );
175
176 m_fields = new PCB_FIELDS_GRID_TABLE( m_frame, this, { m_embeddedFiles->GetLocalFiles() } );
177
178 {
179 LSET forbiddenLayers = LSET::AllCuMask() | LSET::AllTechMask();
180 forbiddenLayers.set( Edge_Cuts );
181 forbiddenLayers.set( Margin );
182
183 m_privateLayers = new LAYERS_GRID_TABLE( m_frame, forbiddenLayers );
184 }
185
186 {
187 LSET forbiddenLayers = LSET::AllLayersMask() & ~LSET::UserDefinedLayersMask();
188 m_customUserLayers = new LAYERS_GRID_TABLE( m_frame, forbiddenLayers );
189 }
190
191 m_delayedErrorMessage = wxEmptyString;
192 m_delayedFocusCtrl = nullptr;
193 m_delayedFocusGrid = nullptr;
197
198 // Give an icon
199 wxIcon icon;
200 icon.CopyFromBitmap( KiBitmap( BITMAPS::icon_modedit ) );
201 SetIcon( icon );
202
203 m_itemsGrid->SetTable( m_fields );
206
207 m_itemsGrid->PushEventHandler( new GRID_TRICKS( m_itemsGrid ) );
209 [this]( wxCommandEvent& aEvent )
210 {
211 OnAddPrivateLayer( aEvent );
212 } ) );
213 m_nettieGroupsGrid->PushEventHandler( new GRID_TRICKS( m_nettieGroupsGrid,
214 [this]( wxCommandEvent& aEvent )
215 {
216 OnAddNettieGroup( aEvent );
217 } ) );
218 m_jumperGroupsGrid->PushEventHandler( new GRID_TRICKS( m_jumperGroupsGrid,
219 [this]( wxCommandEvent& aEvent )
220 {
221 OnAddJumperGroup( aEvent );
222 } ) );
224 [this]( wxCommandEvent& aEvent )
225 {
226 OnAddCustomLayer( aEvent );
227 } ) );
228
229 m_itemsGrid->SetupColumnAutosizer( PFC_VALUE );
230 m_privateLayersGrid->SetupColumnAutosizer( 0 );
231 m_nettieGroupsGrid->SetupColumnAutosizer( 0 );
232 m_jumperGroupsGrid->SetupColumnAutosizer( 0 );
233 m_customUserLayersGrid->SetupColumnAutosizer( 0 );
234
235 m_itemsGrid->SetSelectionMode( wxGrid::wxGridSelectRows );
236 m_privateLayersGrid->SetSelectionMode( wxGrid::wxGridSelectRows );
237 m_nettieGroupsGrid->SetSelectionMode( wxGrid::wxGridSelectRows );
238 m_jumperGroupsGrid->SetSelectionMode( wxGrid::wxGridSelectRows );
239 m_customUserLayersGrid->SetSelectionMode( wxGrid::wxGridSelectRows );
240
241 m_itemsGrid->ShowHideColumns( "0 1 2 3 4 5 7" );
242
244
245 // Set font sizes
246 wxFont infoFont = KIUI::GetInfoFont( this ).Italic();
247 m_staticTextInfoCopper->SetFont( infoFont );
248 m_staticTextInfoPaste->SetFont( infoFont );
249
250 if( static_cast<int>( m_page ) >= 0 )
251 m_NoteBook->SetSelection( (unsigned) m_page );
252
254 {
259 }
261 {
263 }
264
265 // Update label text and tooltip for combined offset + ratio field
266 m_SolderPasteMarginLabel->SetLabel( _( "Solder paste clearance:" ) );
267 m_SolderPasteMarginLabel->SetToolTip( _( "Local solder paste clearance for this footprint.\n"
268 "Enter an absolute value (e.g., -0.1mm), a percentage "
269 "(e.g., -5%), or both (e.g., -0.1mm - 5%).\n"
270 "If blank, the global value is used." ) );
271
272 // Hide the old ratio controls - they're no longer needed
273 m_PasteMarginRatioLabel->Show( false );
274 m_PasteMarginRatioCtrl->Show( false );
275 m_PasteMarginRatioUnits->Show( false );
276
277 // Configure button logos
288
290
292 SetEvtHandlerEnabled( true );
293}
294
295
297{
298 // Prevents crash bug in wxGrid's d'tor
299 m_itemsGrid->DestroyTable( m_fields );
300 m_privateLayersGrid->DestroyTable( m_privateLayers );
302
303 // Delete the GRID_TRICKS.
304 m_itemsGrid->PopEventHandler( true );
305 m_privateLayersGrid->PopEventHandler( true );
306 m_nettieGroupsGrid->PopEventHandler( true );
307 m_jumperGroupsGrid->PopEventHandler( true );
308 m_customUserLayersGrid->PopEventHandler( true );
309
310 m_page = static_cast<NOTEBOOK_PAGES>( m_NoteBook->GetSelection() );
311
312 // the GL canvas on the 3D models page has to be visible before it is destroyed
313 m_NoteBook->SetSelection( static_cast<int>( NOTEBOOK_PAGES::PAGE_3D_MODELS ) );
314}
315
316
318{
319 LIB_ID fpID = m_footprint->GetFPID();
320 wxString footprintName = fpID.GetLibItemName();
321
322 m_FootprintNameCtrl->ChangeValue( footprintName );
323
324 m_DocCtrl->SetValue( EscapeString( m_footprint->GetLibDescription(), CTX_LINE ) );
325 m_KeywordCtrl->SetValue( m_footprint->GetKeywords() );
326
327 if( !wxDialog::TransferDataToWindow() )
328 return false;
329
330 if( !m_PanelGeneral->TransferDataToWindow() )
331 return false;
332
333 // Add the models to the panel
334 if( !m_3dPanel->TransferDataToWindow() )
335 return false;
336
337 if( !m_embeddedFiles->TransferDataToWindow() )
338 return false;
339
340 // Footprint Fields
341 for( PCB_FIELD* field : m_footprint->GetFields() )
342 {
343 wxCHECK2( field, continue );
344
345 m_fields->push_back( *field );
346 }
347
348 // Notify the grid
349 wxGridTableMessage tmsg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, m_fields->GetNumberRows() );
350 m_itemsGrid->ProcessTableMessage( tmsg );
351
352 if( m_footprint->GetAttributes() & FP_THROUGH_HOLE )
353 m_componentType->SetSelection( 0 );
354 else if( m_footprint->GetAttributes() & FP_SMD )
355 m_componentType->SetSelection( 1 );
356 else
357 m_componentType->SetSelection( 2 );
358
359 // Private layers
360 for( PCB_LAYER_ID privateLayer : m_footprint->GetPrivateLayers().UIOrder() )
361 m_privateLayers->push_back( privateLayer );
362
363 // Notify the grid
364 wxGridTableMessage gridTableMessagesg( m_privateLayers, wxGRIDTABLE_NOTIFY_ROWS_APPENDED,
365 m_privateLayers->GetNumberRows() );
366 m_privateLayersGrid->ProcessTableMessage( gridTableMessagesg );
367
368 switch( m_footprint->GetStackupMode() )
369 {
371 {
372 m_cbCustomLayers->SetValue( false );
373
374 m_copperLayerCount->SetSelection( 0 );
375 break;
376 }
378 {
379 m_cbCustomLayers->SetValue( true );
380
381 const LSET& customFpLayers = m_footprint->GetStackupLayers();
382 const LSET customUserLayers = customFpLayers & LSET::UserDefinedLayersMask();
383
384 for( PCB_LAYER_ID customUserLayer : customUserLayers )
385 {
386 m_customUserLayers->push_back( customUserLayer );
387 }
388
389 // Set the number of copper layers
390 m_copperLayerCount->SetSelection( ( customFpLayers & LSET::AllCuMask() ).count() / 2 - 1 );
391 break;
392 }
393 }
395
396 // Notify the grid
397 {
398 wxGridTableMessage gridTableMessagesCustom( m_customUserLayers, wxGRIDTABLE_NOTIFY_ROWS_APPENDED,
399 m_customUserLayers->GetNumberRows() );
400 m_customUserLayersGrid->ProcessTableMessage( gridTableMessagesCustom );
401 }
402
403 m_boardOnly->SetValue( m_footprint->GetAttributes() & FP_BOARD_ONLY );
404 m_excludeFromPosFiles->SetValue( m_footprint->GetAttributes() & FP_EXCLUDE_FROM_POS_FILES );
405 m_excludeFromBOM->SetValue( m_footprint->GetAttributes() & FP_EXCLUDE_FROM_BOM );
406 m_cbDNP->SetValue( m_footprint->GetAttributes() & FP_DNP );
407
408 // Local Clearances
409
410 if( m_footprint->GetLocalClearance().has_value() )
411 m_netClearance.SetValue( m_footprint->GetLocalClearance().value() );
412 else
413 m_netClearance.SetValue( wxEmptyString );
414
415 if( m_footprint->GetLocalSolderMaskMargin().has_value() )
416 m_solderMask.SetValue( m_footprint->GetLocalSolderMaskMargin().value() );
417 else
418 m_solderMask.SetValue( wxEmptyString );
419
420 m_solderPaste.SetOffsetValue( m_footprint->GetLocalSolderPasteMargin() );
421 m_solderPaste.SetRatioValue( m_footprint->GetLocalSolderPasteMarginRatio() );
422
423 m_noCourtyards->SetValue( m_footprint->AllowMissingCourtyard() );
424 m_allowBridges->SetValue( m_footprint->AllowSolderMaskBridges() );
425
426 switch( m_footprint->GetLocalZoneConnection() )
427 {
428 default:
429 case ZONE_CONNECTION::INHERITED: m_ZoneConnectionChoice->SetSelection( 0 ); break;
430 case ZONE_CONNECTION::FULL: m_ZoneConnectionChoice->SetSelection( 1 ); break;
431 case ZONE_CONNECTION::THERMAL: m_ZoneConnectionChoice->SetSelection( 2 ); break;
432 case ZONE_CONNECTION::NONE: m_ZoneConnectionChoice->SetSelection( 3 ); break;
433 }
434
435 for( const wxString& group : m_footprint->GetNetTiePadGroups() )
436 {
437 if( !group.IsEmpty() )
438 {
439 m_nettieGroupsGrid->AppendRows( 1 );
440 m_nettieGroupsGrid->SetCellValue( m_nettieGroupsGrid->GetNumberRows() - 1, 0, group );
441 }
442 }
443
444 m_cbDuplicatePadsAreJumpers->SetValue( m_footprint->GetDuplicatePadNumbersAreJumpers() );
445
446 for( const std::set<wxString>& group : m_footprint->JumperPadGroups() )
447 {
448 wxString groupTxt;
449
450 for( const wxString& pinNumber : group )
451 {
452 if( !groupTxt.IsEmpty() )
453 groupTxt << ", ";
454
455 groupTxt << pinNumber;
456 }
457
458 m_jumperGroupsGrid->AppendRows( 1 );
459 m_jumperGroupsGrid->SetCellValue( m_jumperGroupsGrid->GetNumberRows() - 1, 0, groupTxt );
460 }
461
462 // Items grid
463 for( int col = 0; col < m_itemsGrid->GetNumberCols(); col++ )
464 {
465 // Adjust min size to the column label size
466 m_itemsGrid->SetColMinimalWidth( col, m_itemsGrid->GetVisibleWidth( col, true, false ) );
467 // Adjust the column size.
468 int col_size = m_itemsGrid->GetVisibleWidth( col );
469
470 if( col == PFC_LAYER ) // This one's a drop-down. Check all possible values.
471 {
472 BOARD* board = m_footprint->GetBoard();
473
474 for( PCB_LAYER_ID layer : board->GetEnabledLayers() )
475 col_size = std::max( col_size, GetTextExtent( board->GetLayerName( layer ) ).x );
476
477 // Swatch and gaps:
478 col_size += KiROUND( 14 * GetDPIScaleFactor() ) + 12;
479 }
480
481 if( m_itemsGrid->IsColShown( col ) )
482 m_itemsGrid->SetColSize( col, col_size );
483 }
484
485 m_itemsGrid->SetRowLabelSize( 0 );
486
487 Layout();
488 m_initialized = true;
489
490 return true;
491}
492
493
495 LIB_ID* doOverwrite )
496{
497 if( aFootprintName.IsEmpty() )
498 {
499 m_delayedErrorMessage = _( "Footprint must have a name." );
500 return false;
501 }
502 else if( !FOOTPRINT::IsLibNameValid( aFootprintName ) )
503 {
504 m_delayedErrorMessage.Printf( _( "Footprint name may not contain '%s'." ),
506 return false;
507 }
508
509 LIB_ID fpID = m_footprint->GetFPID();
510 wxString libraryName = fpID.GetLibNickname();
511 wxString originalFPName = fpID.GetLibItemName();
512
514
515 if( aFootprintName != originalFPName && adapter->FootprintExists( libraryName, aFootprintName ) )
516 {
517 wxString msg = wxString::Format( _( "Footprint '%s' already exists in library '%s'." ),
518 aFootprintName, libraryName );
519
520 KIDIALOG errorDlg( m_frame, msg, _( "Confirmation" ), wxOK | wxCANCEL | wxICON_WARNING );
521 errorDlg.SetOKLabel( _( "Overwrite" ) );
522
523 if( errorDlg.ShowModal() == wxID_OK )
524 {
525 doOverwrite->SetLibNickname( libraryName );
526 doOverwrite->SetLibItemName( aFootprintName );
527 return true;
528 }
529 }
530
531 return true;
532}
533
534
536{
537 LSET userLayers;
538 if( m_cbCustomLayers->GetValue() )
539 {
540 userLayers |= LSET::AllCuMask( ( m_copperLayerCount->GetSelection() + 1 ) * 2 );
541
542 for( PCB_LAYER_ID layer : *m_customUserLayers )
543 {
544 userLayers.set( layer );
545 }
546 }
547 else
548 {
549 userLayers |= LSET{ F_Cu, B_Cu };
550 userLayers |= LSET::InternalCuMask();
551 userLayers |= LSET::UserDefinedLayersMask( 4 );
552 }
553
554 return userLayers;
555}
556
557
558static LSET GetAllUsedFootprintLayers( const FOOTPRINT& aFootprint )
559{
560 LSET usedLayers{};
561 aFootprint.RunOnChildren(
562 [&]( BOARD_ITEM* aSubItem )
563 {
564 wxCHECK2( aSubItem, /*void*/ );
565
566 switch( aSubItem->Type() )
567 {
568 case PCB_ZONE_T:
569 {
570 ZONE& zone = static_cast<ZONE&>( *aSubItem );
571 usedLayers |= zone.GetLayerSet();
572 break;
573 }
574 default:
575 {
576 usedLayers.set( aSubItem->GetLayer() );
577 break;
578 }
579 }
580 },
582 return usedLayers;
583}
584
585
587{
588 if( !m_itemsGrid->CommitPendingChanges() )
589 return false;
590
591 if( !DIALOG_SHIM::Validate() )
592 return false;
593
594 // First, test for invalid chars in footprint name
595 wxString footprintName = m_FootprintNameCtrl->GetValue();
596 LIB_ID overwrite;
597
598 if( !checkFootprintName( footprintName, &overwrite ) )
599 {
600 if( m_NoteBook->GetSelection() != 0 )
601 m_NoteBook->SetSelection( 0 );
602
605
606 return false;
607 }
608
609 // Check for valid field text properties
610 for( int i = 0; i < (int) m_fields->size(); ++i )
611 {
612 PCB_FIELD& field = m_fields->at( i );
613
614 // Check for missing field names.
615 if( field.GetName( false ).IsEmpty() )
616 {
618 m_delayedErrorMessage = wxString::Format( _( "Fields must have a name." ) );
621
622 return false;
623 }
624
625 int minSize = pcbIUScale.mmToIU( TEXT_MIN_SIZE_MM );
626 int maxSize = pcbIUScale.mmToIU( TEXT_MAX_SIZE_MM );
627
628 if( field.GetTextWidth() < minSize || field.GetTextWidth() > maxSize )
629 {
631 m_delayedErrorMessage = wxString::Format( _( "The text width must be between %s and %s." ),
632 m_frame->StringFromValue( minSize, true ),
633 m_frame->StringFromValue( maxSize, true ) );
636
637 return false;
638 }
639
640 if( field.GetTextHeight() < minSize || field.GetTextHeight() > maxSize )
641 {
643 m_delayedErrorMessage = wxString::Format( _( "The text height must be between %s and %s." ),
644 m_frame->StringFromValue( minSize, true ),
645 m_frame->StringFromValue( maxSize, true ) );
648
649 return false;
650 }
651
652 // Test for acceptable values for thickness and size and clamp if fails
653 int maxPenWidth = ClampTextPenSize( field.GetTextThickness(), field.GetTextSize() );
654
655 if( field.GetTextThickness() > maxPenWidth )
656 {
657 m_itemsGrid->SetCellValue( i, PFC_THICKNESS, m_frame->StringFromValue( maxPenWidth, true ) );
658
660 m_delayedErrorMessage = _( "The text thickness is too large for the text size.\n"
661 "It will be clamped." );
664
665 return false;
666 }
667 }
668
669 if( !m_netClearance.Validate( 0, INT_MAX ) )
670 return false;
671
672 if( overwrite.IsValid() )
673 {
674 if( m_frame->DeleteFootprintFromLibrary( overwrite, false /* already confirmed */ ) )
675 m_frame->SyncLibraryTree( true );
676 }
677
678 // See if there is an object in the footprint that uses a layer that is not in that list
680
681 // Check that the user isn't trying to remove a layer that is used by the footprint
682 usedLayers &= ~getCustomLayersFromControls();
683 usedLayers &= ~LSET::AllTechMask();
684 usedLayers &= ~LSET::UserMask();
685
686 if( usedLayers.any() )
687 {
689 wxString::Format( _( "You are trying to remove layers that are used by the footprint: %s.\n"
690 "Please remove the objects that use these layers first." ),
691 LAYER_UTILS::AccumulateNames( usedLayers, m_frame->GetBoard() ) );
696 return false;
697 }
698
699 return true;
700}
701
702
704{
705 if( !m_itemsGrid->CommitPendingChanges()
706 || !m_privateLayersGrid->CommitPendingChanges()
707 || !m_nettieGroupsGrid->CommitPendingChanges()
708 || !m_jumperGroupsGrid->CommitPendingChanges()
709 || !m_customUserLayersGrid->CommitPendingChanges() )
710 {
711 return false;
712 }
713
714 KIGFX::PCB_VIEW* view = m_frame->GetCanvas()->GetView();
715 PCB_SELECTION_TOOL* selectionTool = m_frame->GetToolManager()->GetTool<PCB_SELECTION_TOOL>();
716 BOARD_COMMIT commit( m_frame );
717 commit.Modify( m_footprint );
718
719 // Must be done inside the commit to capture the undo state
720 // This will call TransferDataToWindow() on the 3D panel and
721 // the embedded files panel.
722 if( !DIALOG_SHIM::TransferDataFromWindow() )
723 return false;
724
725 // Clear out embedded files that are no longer in use
726 std::set<wxString> files;
727 std::set<wxString> files_to_delete;
728
729 // Get the new files from the footprint fields
730 for( const PCB_FIELD& field : *m_fields )
731 {
732 if( field.GetText().StartsWith( FILEEXT::KiCadUriPrefix ) )
733 files.insert( field.GetText() );
734 }
735
736 // Find any files referenced in the old fields that are not in the new fields
737 for( PCB_FIELD* field : m_footprint->GetFields() )
738 {
739 wxCHECK2( field, continue );
740
741 if( field->GetText().StartsWith( FILEEXT::KiCadUriPrefix ) )
742 {
743 if( files.find( field->GetText() ) == files.end() )
744 files_to_delete.insert( field->GetText() );
745 }
746 }
747
748 for( const wxString& file : files_to_delete )
749 {
750 wxString name = file.Mid( FILEEXT::KiCadUriPrefix.size() + 3 ); // Skip "kicad-embed://"
751 m_footprint->RemoveFile( name );
752 }
753
754 LIB_ID fpID = m_footprint->GetFPID();
755 fpID.SetLibItemName( m_FootprintNameCtrl->GetValue() );
756 m_footprint->SetFPID( fpID );
757
758 m_footprint->SetLibDescription( UnescapeString( m_DocCtrl->GetValue() ) );
759 m_footprint->SetKeywords( m_KeywordCtrl->GetValue() );
760
761 // Update fields
762 m_frame->GetToolManager()->RunAction( ACTIONS::selectionClear );
763
764 while( !m_footprint->GetFields().empty() )
765 {
766 PCB_FIELD* existing = m_footprint->GetFields().front();
767 view->Remove( existing );
768 m_footprint->Remove( existing );
769 delete existing;
770 }
771
772 for( PCB_FIELD& field : *m_fields )
773 {
774 PCB_FIELD* newField = field.CloneField();
775 m_footprint->Add( newField );
776 view->Add( newField );
777
778 if( newField->IsSelected() )
779 {
780 // The old copy was in the selection list, but this one is not. Remove the
781 // out-of-sync selection flag so we can re-add the field to the selection.
782 newField->ClearSelected();
783 selectionTool->AddItemToSel( newField, true );
784 }
785 }
786
787 LSET privateLayers;
788
789 for( PCB_LAYER_ID layer : *m_privateLayers )
790 privateLayers.set( layer );
791
792 m_footprint->SetPrivateLayers( privateLayers );
793
794 if( m_cbCustomLayers->GetValue() )
795 {
796 const LSET customLayers = getCustomLayersFromControls();
797
799 m_footprint->SetStackupLayers( std::move( customLayers ) );
800 }
801 else
802 {
803 // Just use the default stackup mode
805 }
806
807 int attributes = 0;
808
809 switch( m_componentType->GetSelection() )
810 {
811 case 0: attributes |= FP_THROUGH_HOLE; break;
812 case 1: attributes |= FP_SMD; break;
813 default: break;
814 }
815
816 if( m_boardOnly->GetValue() )
817 attributes |= FP_BOARD_ONLY;
818
819 if( m_excludeFromPosFiles->GetValue() )
820 attributes |= FP_EXCLUDE_FROM_POS_FILES;
821
822 if( m_excludeFromBOM->GetValue() )
823 attributes |= FP_EXCLUDE_FROM_BOM;
824
825 if( m_cbDNP->GetValue() )
826 attributes |= FP_DNP;
827
828 m_footprint->SetAttributes( attributes );
829
830 m_footprint->SetAllowMissingCourtyard( m_noCourtyards->GetValue() );
831 m_footprint->SetAllowSolderMaskBridges( m_allowBridges->GetValue() );
832
833 // Initialize mask clearances
834 if( m_netClearance.IsNull() )
835 m_footprint->SetLocalClearance( {} );
836 else
837 m_footprint->SetLocalClearance( m_netClearance.GetValue() );
838
839 if( m_solderMask.IsNull() )
840 m_footprint->SetLocalSolderMaskMargin( {} );
841 else
842 m_footprint->SetLocalSolderMaskMargin( m_solderMask.GetValue() );
843
844 m_footprint->SetLocalSolderPasteMargin( m_solderPaste.GetOffsetValue() );
845 m_footprint->SetLocalSolderPasteMarginRatio( m_solderPaste.GetRatioValue() );
846
847 switch( m_ZoneConnectionChoice->GetSelection() )
848 {
849 default:
850 case 0: m_footprint->SetLocalZoneConnection( ZONE_CONNECTION::INHERITED ); break;
851 case 1: m_footprint->SetLocalZoneConnection( ZONE_CONNECTION::FULL ); break;
852 case 2: m_footprint->SetLocalZoneConnection( ZONE_CONNECTION::THERMAL ); break;
853 case 3: m_footprint->SetLocalZoneConnection( ZONE_CONNECTION::NONE ); break;
854 }
855
856 m_footprint->ClearNetTiePadGroups();
857
858 for( int ii = 0; ii < m_nettieGroupsGrid->GetNumberRows(); ++ii )
859 {
860 wxString group = m_nettieGroupsGrid->GetCellValue( ii, 0 );
861
862 if( !group.IsEmpty() )
863 m_footprint->AddNetTiePadGroup( group );
864 }
865
866 m_footprint->SetDuplicatePadNumbersAreJumpers( m_cbDuplicatePadsAreJumpers->GetValue() );
867
868 std::set<wxString> availablePads;
869
870 for( const PAD* pad : m_footprint->Pads() )
871 availablePads.insert( pad->GetNumber() );
872
873 std::vector<std::set<wxString>> newJumpers;
874
875 for( int ii = 0; ii < m_jumperGroupsGrid->GetNumberRows(); ++ii )
876 {
877 wxStringTokenizer tokenizer( m_jumperGroupsGrid->GetCellValue( ii, 0 ), ", \t\r\n", wxTOKEN_STRTOK );
878 std::set<wxString>& group = newJumpers.emplace_back();
879
880 while( tokenizer.HasMoreTokens() )
881 {
882 wxString token = tokenizer.GetNextToken();
883
884 if( token.IsEmpty() )
885 continue;
886
887 if( !availablePads.count( token ) )
888 {
889 wxString msg;
890 msg.Printf( _( "Pad '%s' in jumper pad group %d does not exist in this footprint." ),
891 token, ii + 1 );
892 DisplayErrorMessage( this, msg );
893 return false;
894 }
895
896 group.insert( token );
897 }
898 }
899
900 m_footprint->JumperPadGroups() = std::move( newJumpers );
901
902 // Copy the models from the panel to the footprint
903 std::vector<FP_3DMODEL>& panelList = m_3dPanel->GetModelList();
904 std::vector<FP_3DMODEL>* fpList = &m_footprint->Models();
905 fpList->clear();
906 fpList->insert( fpList->end(), panelList.begin(), panelList.end() );
907
908 commit.Push( _( "Edit Footprint Properties" ) );
909
910 return true;
911}
912
913
915{
916 m_itemsGrid->OnAddRow(
917 [&]() -> std::pair<int, int>
918 {
919 const BOARD_DESIGN_SETTINGS& dsnSettings = m_frame->GetDesignSettings();
920
922
923 // Set active layer if legal; otherwise copy layer from previous text item
924 if( LSET::AllTechMask().test( m_frame->GetActiveLayer() ) )
925 newField.SetLayer( m_frame->GetActiveLayer() );
926 else
927 newField.SetLayer( m_fields->at( m_fields->size() - 1 ).GetLayer() );
928
929 newField.SetTextSize( dsnSettings.GetTextSize( newField.GetLayer() ) );
930 newField.SetTextThickness( dsnSettings.GetTextThickness( newField.GetLayer() ) );
931 newField.SetItalic( dsnSettings.GetTextItalic( newField.GetLayer() ) );
932
933 m_fields->push_back( newField );
934
935 // notify the grid
936 wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, 1 );
937 m_itemsGrid->ProcessTableMessage( msg );
938 OnModify();
939
940 return { m_fields->size() - 1, PFC_NAME };
941 } );
942}
943
944
946{
947 m_itemsGrid->OnDeleteRows(
948 [&]( int row )
949 {
950 if( row < m_fields->GetMandatoryRowCount() )
951 {
952 DisplayError( this, wxString::Format( _( "The first %d fields are mandatory." ),
953 m_fields->GetMandatoryRowCount() ) );
954 return false;
955 }
956
957 return true;
958 },
959 [&]( int row )
960 {
961 m_fields->erase( m_fields->begin() + row );
962
963 // notify the grid
964 wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_DELETED, row, 1 );
965 m_itemsGrid->ProcessTableMessage( msg );
966 } );
967
968 OnModify();
969}
970
971
973 int aRow )
974{
975 aLayerTable.erase( aLayerTable.begin() + aRow );
976
977 // notify the grid
978 wxGridTableMessage msg( &aLayerTable, wxGRIDTABLE_NOTIFY_ROWS_DELETED, aRow, 1 );
979 aGrid.ProcessTableMessage( msg );
980
981 OnModify();
982}
983
984
986 LAYERS_GRID_TABLE& aGridTable )
987{
988 PCB_LAYER_ID nextLayer = User_1;
989
990 while( alg::contains( aGridTable, nextLayer ) && nextLayer < User_45 )
991 nextLayer = ToLAYER_ID( nextLayer + 2 );
992
993 aGridTable.push_back( nextLayer );
994
995 // notify the grid
996 wxGridTableMessage msg( &aGridTable, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, 1 );
997 aGrid.ProcessTableMessage( msg );
998 OnModify();
999
1000 return { aGridTable.size() - 1, -1 };
1001}
1002
1003
1005{
1006 m_privateLayersGrid->OnAddRow(
1007 [&]()
1008 {
1010 } );
1011}
1012
1013
1015{
1016 m_privateLayersGrid->OnDeleteRows(
1017 [&]( int row )
1018 {
1020 } );
1021}
1022
1023
1028
1029
1031{
1032 m_customUserLayersGrid->OnAddRow(
1033 [&]()
1034 {
1036 } );
1037}
1038
1039
1041{
1042 m_customUserLayersGrid->OnDeleteRows(
1043 [&]( int row )
1044 {
1046 } );
1047}
1048
1049
1054
1055
1060
1061
1066
1067
1072
1073
1075{
1076 aGrid->OnAddRow(
1077 [&]() -> std::pair<int, int>
1078 {
1079 aGrid->AppendRows( 1 );
1080 OnModify();
1081
1082 return { aGrid->GetNumberRows() - 1, 0 };
1083 } );
1084}
1085
1086
1088{
1089 aGrid->OnDeleteRows(
1090 [&]( int row )
1091 {
1092 aGrid->DeleteRows( row, 1 );
1093 } );
1094
1095 OnModify();
1096}
1097
1098
1100{
1101 // Handle a delayed focus. The delay allows us to:
1102 // a) change focus when the error was triggered from within a killFocus handler
1103 // b) show the correct notebook page in the background before the error dialog comes up
1104 // when triggered from an OK or a notebook page change
1105
1106 if( static_cast<int>( m_delayedFocusPage ) >= 0 )
1107 {
1108 if( m_NoteBook->GetSelection() != static_cast<int>( m_delayedFocusPage ) )
1109 m_NoteBook->ChangeSelection( static_cast<int>( m_delayedFocusPage ) );
1110
1112 }
1113
1114 if( !m_delayedErrorMessage.IsEmpty() )
1115 {
1116 // We will re-enter this routine when the error dialog is displayed, so make
1117 // sure we don't keep putting up more dialogs.
1118 wxString msg = m_delayedErrorMessage;
1119 m_delayedErrorMessage = wxEmptyString;
1120
1121 // Do not use DisplayErrorMessage(); it screws up window order on Mac
1122 DisplayError( nullptr, msg );
1123 }
1124
1125 if( m_delayedFocusCtrl )
1126 {
1127 m_delayedFocusCtrl->SetFocus();
1128
1129 if( wxTextEntry* textEntry = dynamic_cast<wxTextEntry*>( m_delayedFocusCtrl ) )
1130 textEntry->SelectAll();
1131
1132 m_delayedFocusCtrl = nullptr;
1133 }
1134 else if( m_delayedFocusGrid )
1135 {
1136 m_delayedFocusGrid->SetFocus();
1139
1140 if( !( m_delayedFocusColumn == 0 && m_delayedFocusRow < m_fields->GetMandatoryRowCount() ) )
1141 m_delayedFocusGrid->EnableCellEditControl( true );
1142
1143 m_delayedFocusGrid->ShowCellEditControl();
1144
1145 m_delayedFocusGrid = nullptr;
1146 m_delayedFocusRow = -1;
1148 }
1149}
1150
1151
1153{
1154 bool enableCustomCtrls = m_cbCustomLayers->GetValue();
1155
1156 m_copperLayerCount->Enable( enableCustomCtrls );
1157 m_customUserLayersGrid->Enable( enableCustomCtrls );
1158 m_bpAddCustomLayer->Enable( enableCustomCtrls );
1159 m_bpDeleteCustomLayer->Enable( enableCustomCtrls );
1160}
1161
1162
1164{
1165 if( !m_itemsGrid->CommitPendingChanges() )
1166 aEvent.Veto();
1167
1168 if( !m_privateLayersGrid->CommitPendingChanges() )
1169 aEvent.Veto();
1170
1171 if( !m_customUserLayersGrid->CommitPendingChanges() )
1172 aEvent.Veto();
1173}
1174
1175
1177{
1178 if( m_initialized )
1179 OnModify();
1180}
1181
1182
1184{
1185 if( m_initialized )
1186 OnModify();
1187}
1188
1189
1191{
1192 if( m_initialized )
1193 OnModify();
1194}
const char * name
constexpr EDA_IU_SCALE pcbIUScale
Definition base_units.h:112
wxBitmap KiBitmap(BITMAPS aBitmap, int aHeightTag)
Construct a wxBitmap from an image identifier Returns the image from the active theme if the image ha...
Definition bitmap.cpp:104
wxBitmapBundle KiBitmapBundle(BITMAPS aBitmap, int aMinHeight)
Definition bitmap.cpp:110
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition box2.h:990
static TOOL_ACTION selectionClear
Clear the current selection.
Definition actions.h:224
BASE_SET & set(size_t pos)
Definition base_set.h:116
Container for design settings for a BOARD object.
int GetTextThickness(PCB_LAYER_ID aLayer) const
Return the default text thickness from the layer class for the given layer.
bool GetTextItalic(PCB_LAYER_ID aLayer) const
VECTOR2I GetTextSize(PCB_LAYER_ID aLayer) const
Return the default text size from the layer class for the given layer.
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition board_item.h:84
virtual PCB_LAYER_ID GetLayer() const
Return the primary layer this item is on.
Definition board_item.h:237
virtual void SetLayer(PCB_LAYER_ID aLayer)
Set the layer this item is on.
Definition board_item.h:285
Information pertinent to a Pcbnew printed circuit board.
Definition board.h:322
const wxString GetLayerName(PCB_LAYER_ID aLayer) const
Return the name of a aLayer.
Definition board.cpp:728
const LSET & GetEnabledLayers() const
A proxy function that calls the corresponding function in m_BoardSettings.
Definition board.cpp:967
DIALOG_FOOTPRINT_PROPERTIES_FP_EDITOR_BASE(wxWindow *parent, wxWindowID id=wxID_ANY, const wxString &title=_("Footprint Properties"), const wxPoint &pos=wxDefaultPosition, const wxSize &size=wxSize(-1,-1), long style=wxDEFAULT_DIALOG_STYLE|wxRESIZE_BORDER)
DIALOG_FOOTPRINT_PROPERTIES_FP_EDITOR(FOOTPRINT_EDIT_FRAME *aParent, FOOTPRINT *aFootprint)
LSET getCustomLayersFromControls() const
Get the layers for the footprint from the controls that can be affected by the stackup.
bool checkFootprintName(const wxString &aFootprintName, LIB_ID *doOverwrite)
void onLayerGridRowDelete(WX_GRID &aGrid, LAYERS_GRID_TABLE &aLayerTable, int aRow)
std::pair< int, int > onLayerGridRowAddUserLayer(WX_GRID &aGrid, LAYERS_GRID_TABLE &aLayerTable)
void SetInitialFocus(wxWindow *aWindow)
Sets the window (usually a wxTextCtrl) that should be focused when the dialog is shown.
Definition dialog_shim.h:82
void SetupStandardButtons(std::map< int, wxString > aLabels={})
void finishDialogSettings()
In all dialogs, we must call the same functions to fix minimal dlg size, the default position and per...
KICAD_T Type() const
Returns the type of object.
Definition eda_item.h:111
void ClearSelected()
Definition eda_item.h:143
bool IsSelected() const
Definition eda_item.h:128
int GetTextHeight() const
Definition eda_text.h:267
void SetTextSize(VECTOR2I aNewSize, bool aEnforceMinTextSize=true)
Definition eda_text.cpp:546
int GetTextWidth() const
Definition eda_text.h:264
void SetTextThickness(int aWidth)
The TextThickness is that set by the user.
Definition eda_text.cpp:298
int GetTextThickness() const
Definition eda_text.h:128
void SetItalic(bool aItalic)
Set the text to be italic - this will also update the font if needed.
Definition eda_text.cpp:321
VECTOR2I GetTextSize() const
Definition eda_text.h:261
An interface to the global shared library manager that is schematic-specific and linked to one projec...
bool FootprintExists(const wxString &aNickname, const wxString &aName)
Provide a custom wxValidator object for limiting the allowable characters when defining footprint nam...
Definition validators.h:53
static bool IsLibNameValid(const wxString &aName)
Test for validity of a name of a footprint to be used in a footprint library ( no spaces,...
void RunOnChildren(const std::function< void(BOARD_ITEM *)> &aFunction, RECURSE_MODE aMode) const override
Invoke a function on all children.
static const wxChar * StringLibNameInvalidChars(bool aUserReadable)
Test for validity of the name in a library of the footprint ( no spaces, dir separators ....
Add mouse and command handling (such as cut, copy, and paste) to a WX_GRID instance.
Definition grid_tricks.h:61
Helper class to create more flexible dialogs, including 'do not show again' checkbox handling.
Definition kidialog.h:42
int ShowModal() override
Definition kidialog.cpp:93
virtual void Add(VIEW_ITEM *aItem, int aDrawPriority=-1) override
Add a VIEW_ITEM to the view.
Definition pcb_view.cpp:57
virtual void Remove(VIEW_ITEM *aItem) override
Remove a VIEW_ITEM from the view.
Definition pcb_view.cpp:74
LAYERS_GRID_TABLE(PCB_BASE_FRAME *aFrame, const LSET &aForbiddenLayers)
wxString GetValue(int aRow, int aCol) override
bool CanGetValueAs(int aRow, int aCol, const wxString &aTypeName) override
void SetValueAsLong(int aRow, int aCol, long aValue) override
long GetValueAsLong(int aRow, int aCol) override
bool CanSetValueAs(int aRow, int aCol, const wxString &aTypeName) override
void SetValue(int aRow, int aCol, const wxString &aValue) override
wxGridCellAttr * GetAttr(int aRow, int aCol, wxGridCellAttr::wxAttrKind aKind) override
A logical library item identifier and consists of various portions much like a URI.
Definition lib_id.h:49
int SetLibItemName(const UTF8 &aLibItemName)
Override the library item name portion of the LIB_ID to aLibItemName.
Definition lib_id.cpp:111
bool IsValid() const
Check if this LID_ID is valid.
Definition lib_id.h:172
int SetLibNickname(const UTF8 &aLibNickname)
Override the logical library name portion of the LIB_ID to aLibNickname.
Definition lib_id.cpp:100
const UTF8 & GetLibItemName() const
Definition lib_id.h:102
const UTF8 & GetLibNickname() const
Return the logical library name portion of a LIB_ID.
Definition lib_id.h:87
LSET is a set of PCB_LAYER_IDs.
Definition lset.h:37
static const LSET & AllTechMask()
Return a mask holding all technical layers (no CU layer) on both side.
Definition lset.cpp:676
static LSET AllCuMask(int aCuLayerCount)
Return a mask holding the requested number of Cu PCB_LAYER_IDs.
Definition lset.cpp:599
static const LSET & AllLayersMask()
Definition lset.cpp:641
static LSET UserDefinedLayersMask(int aUserDefinedLayerCount=MAX_USER_DEFINED_LAYERS)
Return a mask with the requested number of user defined layers.
Definition lset.cpp:704
static LSET AllCuMask()
return AllCuMask( MAX_CU_LAYERS );
Definition lset.cpp:608
static const LSET & InternalCuMask()
Return a complete set of internal copper layers which is all Cu layers except F_Cu and B_Cu.
Definition lset.cpp:577
Definition pad.h:55
Base PCB main window class for Pcbnew, Gerbview, and CvPcb footprint viewer.
wxString GetName(bool aUseDefaultName=true) const
Return the field name (not translated).
PCB_FIELD * CloneField() const
Same as Clone, but returns a PCB_FIELD item.
Definition pcb_field.h:92
The selection tool: currently supports:
static FOOTPRINT_LIBRARY_ADAPTER * FootprintLibAdapter(PROJECT *aProject)
int AddItemToSel(const TOOL_EVENT &aEvent)
wxGridCellAttr * enhanceAttr(wxGridCellAttr *aInputAttr, int aRow, int aCol, wxGridCellAttr::wxAttrKind aKind)
Definition wx_grid.cpp:47
void OnDeleteRows(const std::function< void(int row)> &aDeleter)
Handles a row deletion event.
Definition wx_grid.cpp:742
void OnAddRow(const std::function< std::pair< int, int >()> &aAdder)
Definition wx_grid.cpp:722
Handle a list of polygons defining a copper zone.
Definition zone.h:73
virtual LSET GetLayerSet() const override
Return a std::bitset of all layers on which the item physically resides.
Definition zone.h:136
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
Definition confirm.cpp:202
void DisplayError(wxWindow *aParent, const wxString &aText)
Display an error or warning message box with aMessage.
Definition confirm.cpp:177
This file is part of the common library.
static LSET GetAllUsedFootprintLayers(const FOOTPRINT &aFootprint)
const int minSize
Push and Shove router track width and via size dialog.
#define _(s)
Declaration of the eda_3d_viewer class.
@ RECURSE
Definition eda_item.h:52
#define TEXT_MIN_SIZE_MM
Minimum text size (1 micron).
Definition eda_text.h:47
#define TEXT_MAX_SIZE_MM
Maximum text size in mm (~10 inches)
Definition eda_text.h:48
@ FP_SMD
Definition footprint.h:85
@ FP_DNP
Definition footprint.h:90
@ FP_EXCLUDE_FROM_POS_FILES
Definition footprint.h:86
@ FP_BOARD_ONLY
Definition footprint.h:88
@ FP_EXCLUDE_FROM_BOM
Definition footprint.h:87
@ FP_THROUGH_HOLE
Definition footprint.h:84
@ EXPAND_INNER_LAYERS
The 'normal' stackup handling, where there is a single inner layer (In1) and rule areas using it expa...
Definition footprint.h:99
@ CUSTOM_LAYERS
Stackup handling where the footprint can have any number of copper layers, and objects on those layer...
Definition footprint.h:104
int ClampTextPenSize(int aPenSize, int aSize, bool aStrict)
Pen width should not allow characters to become cluttered up in their own fatness.
Definition gr_text.cpp:73
static const std::string KiCadUriPrefix
PCB_LAYER_ID
A quick note on layer IDs:
Definition layer_ids.h:60
@ User_45
Definition layer_ids.h:168
@ Edge_Cuts
Definition layer_ids.h:112
@ B_Cu
Definition layer_ids.h:65
@ Margin
Definition layer_ids.h:113
@ User_1
Definition layer_ids.h:124
@ F_Cu
Definition layer_ids.h:64
PCB_LAYER_ID ToLAYER_ID(int aLayer)
Definition lset.cpp:754
KICOMMON_API wxFont GetInfoFont(wxWindow *aWindow)
wxString AccumulateNames(const LSEQ &aLayers, const BOARD *aBoard)
Accumulate layer names from a layer set into a comma separated string.
bool contains(const _Container &__container, _Value __value)
Returns true if the container contains the given value.
Definition kicad_algo.h:100
@ PFC_THICKNESS
see class PGM_BASE
wxString UnescapeString(const wxString &aSource)
wxString EscapeString(const wxString &aSource, ESCAPE_CONTEXT aContext)
The Escape/Unescape routines use HTML-entity-reference-style encoding to handle characters which are:...
@ CTX_LINE
wxString GetUserFieldName(int aFieldNdx, bool aTranslateForHI)
#define DO_TRANSLATE
@ USER
The field ID hasn't been set yet; field is invalid.
@ PCB_ZONE_T
class ZONE, a copper pour area
Definition typeinfo.h:108
Custom text control validator definitions.
@ THERMAL
Use thermal relief for pads.
Definition zones.h:50
@ NONE
Pads are not covered.
Definition zones.h:49
@ FULL
pads are covered by copper
Definition zones.h:51