KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
pcb_net_inspector_panel.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 The KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software: you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the
8 * Free Software Foundation, either version 3 of the License, or (at your
9 * option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License along
17 * with this program. If not, see <http://www.gnu.org/licenses/>.
18 */
19
22
23#include <advanced_config.h>
25#include <confirm.h>
28#include <footprint.h>
29#include <pad.h>
30#include <pcb_edit_frame.h>
31#include <pcb_painter.h>
32#include <pgm_base.h>
34#include <validators.h>
36#include <eda_pattern_match.h>
37
38#include <wx/wupdlock.h>
39#include <wx/filedlg.h>
40
41#include <algorithm>
42#include <thread_pool.h>
43
45 NET_INSPECTOR_PANEL( parent, aFrame ), m_frame( aFrame ), m_dataModel( new DATA_MODEL( *this ) )
46{
48
49 m_netsList->AssociateModel( &*m_dataModel );
50
51 // Rebuild nets list
52 buildNetsList( true );
53
54 // Register the panel to receive board change notifications
55 if( m_board != nullptr )
56 {
58 m_board->AddListener( this );
59 }
60
61 // Connect to board events
62 m_frame->Bind( EDA_EVT_UNITS_CHANGED, &PCB_NET_INSPECTOR_PANEL::onUnitsChanged, this );
63
64 // Connect to wxDataViewCtrl events
65 m_netsList->Bind( wxEVT_DATAVIEW_ITEM_EXPANDED, &PCB_NET_INSPECTOR_PANEL::OnExpandCollapseRow,
66 this );
67 m_netsList->Bind( wxEVT_DATAVIEW_ITEM_COLLAPSED, &PCB_NET_INSPECTOR_PANEL::OnExpandCollapseRow,
68 this );
69 m_netsList->Bind( wxEVT_DATAVIEW_COLUMN_HEADER_RIGHT_CLICK,
71 m_netsList->Bind( wxEVT_DATAVIEW_ITEM_CONTEXT_MENU,
73 m_netsList->Bind( wxEVT_DATAVIEW_ITEM_ACTIVATED,
75 m_netsList->Bind( wxEVT_DATAVIEW_COLUMN_SORTED,
77}
78
80{
82
83 m_netsList->AssociateModel( nullptr );
84
85 // Disconnect from board events
86 m_frame->Unbind( EDA_EVT_UNITS_CHANGED, &PCB_NET_INSPECTOR_PANEL::onUnitsChanged, this );
87
88 // Connect to wxDataViewCtrl events
89 m_netsList->Unbind( wxEVT_DATAVIEW_ITEM_EXPANDED, &PCB_NET_INSPECTOR_PANEL::OnExpandCollapseRow,
90 this );
91 m_netsList->Unbind( wxEVT_DATAVIEW_ITEM_COLLAPSED,
93 m_netsList->Unbind( wxEVT_DATAVIEW_COLUMN_HEADER_RIGHT_CLICK,
95 m_netsList->Unbind( wxEVT_DATAVIEW_ITEM_CONTEXT_MENU,
97 m_netsList->Unbind( wxEVT_DATAVIEW_ITEM_ACTIVATED,
99 m_netsList->Unbind( wxEVT_DATAVIEW_COLUMN_SORTED,
101}
102
103
104/*****************************************************************************************
105 *
106 * Grid / model columns configuration
107 *
108 * ***************************************************************************************/
109
110
112{
113 m_columns.clear();
114
115 // Set up the column display vector
116 m_columns.emplace_back( 0u, UNDEFINED_LAYER, _( "Name" ), _( "Net Name" ),
118 m_columns.emplace_back( 1u, UNDEFINED_LAYER, _( "Netclass" ), _( "Netclass" ),
120 m_columns.emplace_back( 2u, UNDEFINED_LAYER, _( "Total Length" ), _( "Net Length" ),
122 m_columns.emplace_back( 3u, UNDEFINED_LAYER, _( "Via Count" ), _( "Via Count" ),
124 m_columns.emplace_back( 4u, UNDEFINED_LAYER, _( "Via Length" ), _( "Via Length" ),
126 m_columns.emplace_back( 5u, UNDEFINED_LAYER, _( "Track Length" ), _( "Track Length" ),
128 m_columns.emplace_back( 6u, UNDEFINED_LAYER, _( "Die Length" ), _( "Die Length" ),
130 m_columns.emplace_back( 7u, UNDEFINED_LAYER, _( "Pad Count" ), _( "Pad Count" ), CSV_COLUMN_DESC::CSV_NONE, false );
131
132 const std::vector<std::function<void( void )>> add_col{
133 [&]()
134 {
135 m_netsList->AppendTextColumn( m_columns[COLUMN_NAME].display_name,
136 m_columns[COLUMN_NAME], wxDATAVIEW_CELL_INERT, -1,
137 wxALIGN_LEFT,
138 wxDATAVIEW_COL_RESIZABLE | wxDATAVIEW_COL_SORTABLE );
139 },
140 [&]()
141 {
142 m_netsList->AppendTextColumn( m_columns[COLUMN_NETCLASS].display_name,
143 m_columns[COLUMN_NETCLASS], wxDATAVIEW_CELL_INERT, -1,
144 wxALIGN_LEFT,
145 wxDATAVIEW_COL_RESIZABLE | wxDATAVIEW_COL_REORDERABLE
146 | wxDATAVIEW_COL_SORTABLE );
147 },
148 [&]()
149 {
150 m_netsList->AppendTextColumn( m_columns[COLUMN_TOTAL_LENGTH].display_name,
151 m_columns[COLUMN_TOTAL_LENGTH], wxDATAVIEW_CELL_INERT, -1,
152 wxALIGN_CENTER,
153 wxDATAVIEW_COL_RESIZABLE | wxDATAVIEW_COL_REORDERABLE
154 | wxDATAVIEW_COL_SORTABLE );
155 },
156 [&]()
157 {
158 m_netsList->AppendTextColumn( m_columns[COLUMN_VIA_COUNT].display_name,
159 m_columns[COLUMN_VIA_COUNT], wxDATAVIEW_CELL_INERT, -1,
160 wxALIGN_CENTER,
161 wxDATAVIEW_COL_RESIZABLE | wxDATAVIEW_COL_REORDERABLE
162 | wxDATAVIEW_COL_SORTABLE );
163 },
164 [&]()
165 {
166 m_netsList->AppendTextColumn( m_columns[COLUMN_VIA_LENGTH].display_name,
167 m_columns[COLUMN_VIA_LENGTH], wxDATAVIEW_CELL_INERT, -1,
168 wxALIGN_CENTER,
169 wxDATAVIEW_COL_RESIZABLE | wxDATAVIEW_COL_REORDERABLE
170 | wxDATAVIEW_COL_SORTABLE );
171 },
172 [&]()
173 {
174 m_netsList->AppendTextColumn( m_columns[COLUMN_BOARD_LENGTH].display_name,
175 m_columns[COLUMN_BOARD_LENGTH], wxDATAVIEW_CELL_INERT, -1,
176 wxALIGN_CENTER,
177 wxDATAVIEW_COL_RESIZABLE | wxDATAVIEW_COL_REORDERABLE
178 | wxDATAVIEW_COL_SORTABLE );
179 },
180 [&]()
181 {
182 m_netsList->AppendTextColumn( m_columns[COLUMN_PAD_DIE_LENGTH].display_name,
183 m_columns[COLUMN_PAD_DIE_LENGTH], wxDATAVIEW_CELL_INERT,
184 -1, wxALIGN_CENTER,
185 wxDATAVIEW_COL_RESIZABLE | wxDATAVIEW_COL_REORDERABLE
186 | wxDATAVIEW_COL_SORTABLE );
187 },
188 [&]()
189 {
190 m_netsList->AppendTextColumn( m_columns[COLUMN_PAD_COUNT].display_name,
191 m_columns[COLUMN_PAD_COUNT], wxDATAVIEW_CELL_INERT, -1,
192 wxALIGN_CENTER,
193 wxDATAVIEW_COL_RESIZABLE | wxDATAVIEW_COL_REORDERABLE
194 | wxDATAVIEW_COL_SORTABLE );
195 }
196 };
197
198 // If we have not yet loaded the first board, use a dummy local settings object to ensure we
199 // don't over-write existing board settings (note that PCB_EDIT_FRAME loads the local settings
200 // object prior to loading the board; the two are not synced and we need to account for that)
201 PANEL_NET_INSPECTOR_SETTINGS* cfg = nullptr;
202
203 if( m_boardLoaded )
204 {
206 cfg = &localSettings.m_NetInspectorPanel;
207 }
208 else
209 {
211 }
212
213 // Reset the column display settings if column count doesn't match
214 const int totalNumColumns = add_col.size() + m_board->GetCopperLayerCount();
215
216 if( (int) cfg->col_order.size() != totalNumColumns
217 || (int) cfg->col_hidden.size() != totalNumColumns )
218 {
219 cfg->col_order.resize( totalNumColumns );
220 cfg->col_hidden.resize( totalNumColumns );
221
222 for( int i = 0; i < totalNumColumns; ++i )
223 {
224 cfg->col_order[i] = i;
225 cfg->col_hidden[i] = false;
226 }
227 }
228
229 // Check that all rows are unique to protect against corrupted settings data
230 std::set<int> col_order_set( cfg->col_order.begin(), cfg->col_order.end() );
231 if( col_order_set.size() != cfg->col_order.size() )
232 {
233 for( std::size_t i = 0; i < cfg->col_order.size(); ++i )
234 cfg->col_order[i] = static_cast<int>( i );
235 }
236
237 // Add column records for copper layers
238 for( PCB_LAYER_ID layer : m_board->GetEnabledLayers().Seq() )
239 {
240 if( !IsCopperLayer( layer ) )
241 continue;
242
243 m_columns.emplace_back( m_columns.size(), layer, m_board->GetLayerName( layer ), m_board->GetLayerName( layer ),
245 }
246
247 // Add display columns in settings order
248 for( const int i : cfg->col_order )
249 {
250 const int addModelColumn = i;
251
252 if( addModelColumn >= (int) add_col.size() )
253 {
254 m_netsList->AppendTextColumn( m_board->GetLayerName( m_columns[addModelColumn].layer ),
255 m_columns[addModelColumn], wxDATAVIEW_CELL_INERT, -1, wxALIGN_CENTER,
256 wxDATAVIEW_COL_RESIZABLE | wxDATAVIEW_COL_REORDERABLE
257 | wxDATAVIEW_COL_SORTABLE );
258 }
259 else
260 {
261 add_col.at( i )();
262 }
263 }
264
265 // Set the name column as the expander row
266 if( wxDataViewColumn* col = getDisplayedColumnForModelField( COLUMN_NAME ) )
267 {
268 m_netsList->SetExpanderColumn( col );
269 }
270
272
273 // Delete the temporary config if used
274 if( !m_boardLoaded )
275 {
276 delete cfg;
277 }
278}
279
280
282{
283 wxWindowUpdateLocker locker( m_netsList );
284
285 if( cfg->col_widths.size() != m_columns.size() )
286 {
287 int minValueWidth = GetTextExtent( wxT( "00000,000 mm" ) ).x;
288 int minNumberWidth = GetTextExtent( wxT( "000" ) ).x;
289 int minNameWidth = GetTextExtent( wxT( "MMMMMMMMMMMM" ) ).x;
290
291 // Considering left and right margins.
292 // For wxRenderGeneric it is 5px.
293 // Also account for the sorting arrow in the column header.
294 // Column 0 also needs space for any potential expander icons.
295 constexpr int margins = 15;
296 constexpr int extra_width = 30;
297
298 auto getTargetWidth =
299 [&]( int columnID )
300 {
301 switch( columnID )
302 {
303 case COLUMN_NAME: return minNameWidth + extra_width;
304 case COLUMN_NETCLASS: return minNameWidth + margins;
305 case COLUMN_VIA_COUNT: return minNumberWidth + margins;
306 case COLUMN_PAD_COUNT: return minNumberWidth + margins;
307 default: return minValueWidth + margins;
308 }
309 };
310
311 wxASSERT( m_columns.size() == cfg->col_order.size() );
312
313 for( size_t i = 0; i < m_columns.size(); ++i )
314 {
315 const int modelColumn = cfg->col_order[i];
316 int titleSize = GetTextExtent( m_columns[modelColumn].display_name ).x;
317 titleSize = modelColumn == COLUMN_NAME ? titleSize + extra_width : titleSize + margins;
318 const int valSize = getTargetWidth( modelColumn );
319 m_netsList->GetColumn( i )->SetWidth( std::max( titleSize, valSize ) );
320 }
321 }
322 else
323 {
324 wxASSERT( m_columns.size() == cfg->col_hidden.size() );
325 wxASSERT( m_columns.size() == cfg->col_widths.size() );
326
327 for( size_t ii = 0; ii < m_columns.size(); ++ii )
328 {
329 const int newWidth = cfg->col_widths[ii];
330 // Make sure we end up with something non-zero so we can resize it
331 m_netsList->GetColumn( ii )->SetWidth( std::max( newWidth, 10 ) );
332 m_netsList->GetColumn( ii )->SetHidden( cfg->col_hidden[ii] );
333 }
334 }
335
336 m_netsList->Refresh();
337}
338
339
340bool PCB_NET_INSPECTOR_PANEL::restoreSortColumn( const int sortingColumnId, const bool sortOrderAsc ) const
341{
342 if( sortingColumnId != -1 )
343 {
344 if( wxDataViewColumn* col = getDisplayedColumnForModelField( sortingColumnId ) )
345 {
346 col->SetSortOrder( sortOrderAsc );
347 m_dataModel->Resort();
348 return true;
349 }
350 }
351
352 return false;
353}
354
355
356wxDataViewColumn* PCB_NET_INSPECTOR_PANEL::getDisplayedColumnForModelField( const int columnId ) const
357{
358 for( unsigned int i = 0; i < m_netsList->GetColumnCount(); ++i )
359 {
360 wxDataViewColumn* col = m_netsList->GetColumn( i );
361
362 if( static_cast<int>( col->GetModelColumn() ) == columnId )
363 {
364 return col;
365 }
366 }
367
368 return nullptr;
369}
370
371
372/*****************************************************************************************
373 *
374 * Nets list generation
375 *
376 * ***************************************************************************************/
377
378
379void PCB_NET_INSPECTOR_PANEL::buildNetsList( const bool rebuildColumns )
380{
381 // Only build the list of nets if there is a board present
382 if( !m_board )
383 return;
384
385 m_inBuildNetsList = true;
386
387 m_netsList->Freeze();
388
391
392 // Refresh all filtering / grouping settings
398
399 // Attempt to keep any expanded groups open
401 {
402 cfg->expanded_rows.clear();
403 DATA_MODEL* model = static_cast<DATA_MODEL*>( m_netsList->GetModel() );
404
405 for( const auto& groupItems = model->getGroupDataViewItems();
406 auto& [groupName, groupItem] : groupItems )
407 {
408 if( m_netsList->IsExpanded( groupItem ) )
409 cfg->expanded_rows.push_back( groupName );
410 }
411 }
412
413 // When rebuilding the netlist, try to keep the row selection
414 wxDataViewItemArray sel;
415 m_netsList->GetSelections( sel );
416
417 std::vector<int> prev_selected_netcodes;
418 prev_selected_netcodes.reserve( sel.GetCount() );
419
420 for( unsigned int i = 0; i < sel.GetCount(); ++i )
421 {
422 const LIST_ITEM* item = static_cast<const LIST_ITEM*>( sel.Item( i ).GetID() );
423 prev_selected_netcodes.push_back( item->GetNetCode() );
424 }
425
426 int sorting_column_id = cfg->sorting_column;
427 bool sort_order_asc = cfg->sort_order_asc;
428
429 if( wxDataViewColumn* sorting_column = m_netsList->GetSortingColumn() )
430 {
431 if( !m_boardLoading )
432 {
433 sorting_column_id = static_cast<int>( sorting_column->GetModelColumn() );
434 sort_order_asc = sorting_column->IsSortOrderAscending();
435 }
436
437 // On GTK, wxDVC will crash if we rebuild with a sorting column set.
438 sorting_column->UnsetAsSortKey();
439 }
440
441 if( rebuildColumns )
442 {
443 m_netsList->ClearColumns();
444 buildColumns();
445 }
446
447 m_dataModel->deleteAllItems();
448 m_custom_group_rules.clear();
449
450 for( const wxString& rule : cfg->custom_group_rules )
451 m_custom_group_rules.push_back( std::make_unique<EDA_COMBINED_MATCHER>( rule, CTX_NET ) );
452
453 m_dataModel->addCustomGroups();
454
455 std::vector<NETINFO_ITEM*> netCodes;
456
457 for( NETINFO_ITEM* ni : m_board->GetNetInfo() )
458 {
459 if( netFilterMatches( ni, cfg ) )
460 netCodes.emplace_back( ni );
461 }
462
463 std::ranges::sort( netCodes,
464 []( const NETINFO_ITEM* a, const NETINFO_ITEM* b )
465 {
466 return a->GetNetCode() < b->GetNetCode();
467 } );
468
469 std::vector<std::unique_ptr<LIST_ITEM>> lengths = calculateNets( netCodes, m_showZeroPadNets );
470
471 m_dataModel->addItems( lengths );
472
473 // Try to re-enable the sorting column
474 if( !restoreSortColumn( sorting_column_id, sort_order_asc ))
475 {
476 // By default, sort by Name column
478 }
479
480 // Try to restore the expanded groups
481 if( m_boardLoaded )
482 {
483 m_rowExpanding = true;
484
485 std::vector<std::pair<wxString, wxDataViewItem>> groupItems = m_dataModel->getGroupDataViewItems();
486
487 for( wxString& groupName : cfg->expanded_rows )
488 {
489 auto pred =
490 [&groupName]( const std::pair<wxString, wxDataViewItem>& item )
491 {
492 return groupName == item.first;
493 };
494
495 auto tableItem = std::ranges::find_if( groupItems, pred );
496
497 if( tableItem != groupItems.end() )
498 m_netsList->Expand( tableItem->second );
499 }
500
501 m_rowExpanding = false;
502 }
503
504 // Try to restore the selected rows
505 sel.Clear();
506
507 for( const int& nc : prev_selected_netcodes )
508 {
509 if( std::optional<LIST_ITEM_ITER> r = m_dataModel->findItem( nc ) )
510 {
511 const std::unique_ptr<LIST_ITEM>& list_item = *r.value();
512 sel.Add( wxDataViewItem( list_item.get() ) );
513 }
514 }
515
516 m_netsList->Thaw(); // Must thaw before reselecting to avoid windows selection bug
517
518 if( !sel.IsEmpty() )
519 {
520 m_netsList->SetSelections( sel );
521 m_netsList->EnsureVisible( sel.Item( 0 ) );
522 }
523 else
524 {
525 m_netsList->UnselectAll();
526 }
527
528 m_inBuildNetsList = false;
529}
530
531
534{
535 if( cfg == nullptr )
536 {
538 cfg = &localSettings.m_NetInspectorPanel;
539 }
540
541 // Never show an unconnected net
542 if( aNet->GetNetCode() <= 0 )
543 return false;
544
545 const wxString filterString = UnescapeString( m_searchCtrl->GetValue() ).Upper();
546 const wxString netName = UnescapeString( aNet->GetNetname() ).Upper();
547 const NETCLASS* netClass = aNet->GetNetClass();
548 const wxString netClassName = UnescapeString( netClass->GetName() ).Upper();
549
550 bool matched = false;
551
552 // No filter - match all
553 if( filterString.Length() == 0 )
554 matched = true;
555
556 // Search on net class
557 if( !matched && cfg->filter_by_netclass && netClassName.Find( filterString ) != wxNOT_FOUND )
558 matched = true;
559
560 // Search on net name
561 if( !matched && cfg->filter_by_net_name && netName.Find( filterString ) != wxNOT_FOUND )
562 matched = true;
563
564 // Remove unconnected nets if required
565 if( matched )
566 {
568 matched = !netName.StartsWith( wxT( "UNCONNECTED-(" ) );
569 }
570
571 return matched;
572}
573
574
576{
577 bool operator()( const CN_ITEM* a, const CN_ITEM* b ) const { return a->Net() < b->Net(); }
578
579 bool operator()( const CN_ITEM* a, int b ) const { return a->Net() < b; }
580
581 bool operator()( int a, const CN_ITEM* b ) const { return a < b->Net(); }
582};
583
584
586{
587 // Pre-filter the connectivity items and sort them by netcode. This avoids quadratic runtime when building the whole
588 // net list.
589 const auto type_bits = std::bitset<MAX_STRUCT_TYPE_ID>()
590 .set( PCB_TRACE_T )
591 .set( PCB_ARC_T )
592 .set( PCB_VIA_T )
593 .set( PCB_PAD_T );
594
595 std::vector<CN_ITEM*> cn_items;
596 cn_items.reserve( 1024 );
597
598 for( CN_ITEM* cn_item : m_board->GetConnectivity()->GetConnectivityAlgo()->ItemList() )
599 {
600 if( cn_item->Valid() && type_bits[cn_item->Parent()->Type()] )
601 cn_items.push_back( cn_item );
602 }
603
604 std::ranges::sort( cn_items, NETCODE_CMP_LESS() );
605
606 return cn_items;
607}
608
609
610std::vector<std::unique_ptr<PCB_NET_INSPECTOR_PANEL::LIST_ITEM>>
611PCB_NET_INSPECTOR_PANEL::calculateNets( const std::vector<NETINFO_ITEM*>& aNetCodes, bool aIncludeZeroPadNets ) const
612{
613 std::vector<std::unique_ptr<LIST_ITEM>> results;
614
616 const std::vector<CN_ITEM*> conItems = relevantConnectivityItems();
617
618 // First assemble the LENGTH_CALCULATION_ITEMs for board items which match the nets we need to recompute
619 // Precondition: conItems and aNetCodes are sorted in increasing netcode value
620 // Functionality: This extracts any items from conItems which have a netcode which is present in aNetCodes
621 std::unordered_map<int, std::vector<LENGTH_CALCULATION_ITEM>> netItemsMap;
622 std::vector<NETINFO_ITEM*> foundNets;
623
624 auto itemItr = conItems.begin();
625 auto netCodeItr = aNetCodes.begin();
626
627 while( itemItr != conItems.end() && netCodeItr != aNetCodes.end() )
628 {
629 const int curNetCode = ( *netCodeItr )->GetNetCode();
630 const int curItemNetCode = ( *itemItr )->Net();
631
632 if( curItemNetCode == curNetCode )
633 {
634 if( foundNets.empty() || foundNets.back() != *netCodeItr )
635 foundNets.emplace_back( *netCodeItr );
636
637 // Take the item
638 LENGTH_CALCULATION_ITEM lengthItem = calc->GetLengthCalculationItem( ( *itemItr )->Parent() );
639 netItemsMap[curItemNetCode].emplace_back( std::move( lengthItem ) );
640 ++itemItr;
641 }
642 else if( curItemNetCode < curNetCode )
643 {
644 // Fast-forward through items
645 while( itemItr != conItems.end() && ( *itemItr )->Net() < curNetCode )
646 ++itemItr;
647 }
648 else if( curItemNetCode > curNetCode )
649 {
650 // Fast-forward through required net codes
651 while( netCodeItr != aNetCodes.end() && curItemNetCode > ( *netCodeItr )->GetNetCode() )
652 ++netCodeItr;
653 }
654 }
655
656 // Now calculate the length statistics for each net. This includes potentially expensive path optimisations, so
657 // parallelize this work.
658 std::mutex resultsMutex;
660
661 auto resultsFuture = tp.parallelize_loop(
662 0, foundNets.size(),
663 [&, this, calc]( const int start, const int end )
664 {
665 for( int i = start; i < end; ++i )
666 {
667 int netCode = foundNets[i]->GetNetCode();
668
669 constexpr PATH_OPTIMISATIONS opts = { .OptimiseViaLayers = true,
670 .MergeTracks = true,
671 .OptimiseTracesInPads = true,
672 .InferViaInPad = false };
673 LENGTH_DETAILS lengthDetails =
674 calc->CalculateLengthDetails( netItemsMap[netCode], opts, nullptr, nullptr, true );
675
676 if( aIncludeZeroPadNets || lengthDetails.NumPads > 0 )
677 {
678 std::unique_ptr<LIST_ITEM> new_item = std::make_unique<LIST_ITEM>( foundNets[i] );
679
680 new_item->SetPadCount( lengthDetails.NumPads );
681 new_item->SetLayerCount( m_board->GetCopperLayerCount() );
682 new_item->SetPadDieLength( lengthDetails.PadToDieLength );
683 new_item->SetViaCount( lengthDetails.NumVias );
684 new_item->SetViaLength( lengthDetails.ViaLength );
685 new_item->SetLayerWireLengths( *lengthDetails.LayerLengths );
686
687 std::scoped_lock lock( resultsMutex );
688 results.emplace_back( std::move( new_item ) );
689 }
690 }
691 } );
692
693 resultsFuture.get();
694
695 return results;
696}
697
698
699/*****************************************************************************************
700 *
701 * Formatting helpers
702 *
703 * ***************************************************************************************/
704
705
707{
708 return wxString::Format( wxT( "%.3d" ), aNet->GetNetCode() );
709}
710
711
713{
714 return UnescapeString( aNet->GetNetname() );
715}
716
717
718wxString PCB_NET_INSPECTOR_PANEL::formatCount( const unsigned int aValue )
719{
720 return wxString::Format( wxT( "%u" ), aValue );
721}
722
723
724wxString PCB_NET_INSPECTOR_PANEL::formatLength( const int64_t aValue ) const
725{
726 return m_frame->MessageTextFromValue( aValue,
727 // don't include unit label in the string when reporting
728 !m_inReporting );
729}
730
731
732void PCB_NET_INSPECTOR_PANEL::updateDisplayedRowValues( const std::optional<LIST_ITEM_ITER>& aRow ) const
733{
734 if( !aRow )
735 return;
736
737 wxDataViewItemArray sel;
738 m_netsList->GetSelections( sel );
739
740 m_dataModel->updateItem( aRow );
741
742 if( !sel.IsEmpty() )
743 {
744 m_netsList->SetSelections( sel );
745 m_netsList->EnsureVisible( sel.Item( 0 ) );
746 }
747}
748
749
750/*****************************************************************************************
751 *
752 * BOARD_LISTENER event handling
753 *
754 * ***************************************************************************************/
755
756
758{
760
761 if( m_board )
762 m_board->AddListener( this );
763
764 m_boardLoaded = true;
765 m_boardLoading = true;
766
768 auto& cfg = localSettings.m_NetInspectorPanel;
769 m_searchCtrl->SetValue( cfg.filter_text );
770
771 buildNetsList( true );
772
773 m_boardLoading = false;
774}
775
776
778{
779 const std::vector<BOARD_ITEM*> item{ aBoardItem };
780 updateBoardItems( item );
781}
782
783
784void PCB_NET_INSPECTOR_PANEL::OnBoardItemsAdded( BOARD& aBoard, std::vector<BOARD_ITEM*>& aBoardItems )
785{
786 updateBoardItems( aBoardItems );
787}
788
789
790void PCB_NET_INSPECTOR_PANEL::updateBoardItems( const std::vector<BOARD_ITEM*>& aBoardItems )
791{
792 if( !IsShownOnScreen() )
793 return;
794
795 // Rebuild full list for large changes
796 if( aBoardItems.size()
798 {
800 }
801 else
802 {
803 std::vector<NETINFO_ITEM*> changedNets;
804
805 for( BOARD_ITEM* boardItem : aBoardItems )
806 {
807 if( NETINFO_ITEM* net = dynamic_cast<NETINFO_ITEM*>( boardItem ) )
808 {
809 // A new net has been added to the board. Add it to our list if it passes the netname filter test.
810 if( netFilterMatches( net ) )
811 changedNets.emplace_back( net );
812 }
813 else if( BOARD_CONNECTED_ITEM* i = dynamic_cast<BOARD_CONNECTED_ITEM*>( boardItem ) )
814 {
815 changedNets.emplace_back( i->GetNet() );
816 }
817 else if( FOOTPRINT* footprint = dynamic_cast<FOOTPRINT*>( boardItem ) )
818 {
819 for( const PAD* pad : footprint->Pads() )
820 {
821 if( netFilterMatches( pad->GetNet() ) )
822 changedNets.emplace_back( pad->GetNet() );
823 }
824 }
825 }
826
827 std::ranges::sort( changedNets,
828 []( const NETINFO_ITEM* a, const NETINFO_ITEM* b )
829 {
830 return a->GetNetCode() < b->GetNetCode();
831 } );
832
833 updateNets( changedNets );
834 }
835
836 m_netsList->Refresh();
837}
838
839
840void PCB_NET_INSPECTOR_PANEL::updateNets( const std::vector<NETINFO_ITEM*>& aNets ) const
841{
842 std::vector<NETINFO_ITEM*> netsToUpdate;
843 std::unordered_set<NETINFO_ITEM*> netsToDelete;
844
845 for( NETINFO_ITEM* net : aNets )
846 {
847 // Add all nets to the deletion list - we will prune this later to only contain unhandled nets
848 netsToDelete.insert( net );
849
850 // Only calculate nets that match the current filter
851 if( netFilterMatches( net ) )
852 netsToUpdate.emplace_back( net );
853 }
854
855 m_netsList->Freeze();
856
857 std::vector<std::unique_ptr<LIST_ITEM>> newListItems = calculateNets( aNets, true );
858
859 for( std::unique_ptr<LIST_ITEM>& newListItem : newListItems )
860 {
861 // Remove the handled net from the deletion list
862 netsToDelete.erase( newListItem->GetNet() );
863
864 std::optional<LIST_ITEM_ITER> curNetRow = m_dataModel->findItem( newListItem->GetNetCode() );
865
866 if( !m_showZeroPadNets && newListItem->GetPadCount() == 0 )
867 {
868 m_dataModel->deleteItem( curNetRow );
869 continue;
870 }
871
872 if( !curNetRow )
873 {
874 m_dataModel->addItem( std::move( newListItem ) );
875 continue;
876 }
877
878 const std::unique_ptr<LIST_ITEM>& curListItem = *curNetRow.value();
879
880 if( curListItem->GetNetName() != newListItem->GetNetName() )
881 {
882 // If the name has changed, it might require re-grouping. It's easier to remove and re-insert it.
883 m_dataModel->deleteItem( curNetRow );
884 m_dataModel->addItem( std::move( newListItem ) );
885 }
886 else
887 {
888 curListItem->SetPadCount( newListItem->GetPadCount() );
889 curListItem->SetPadDieLength( newListItem->GetPadDieLength() );
890 curListItem->SetViaCount( newListItem->GetViaCount() );
891 curListItem->SetViaLength( newListItem->GetViaLength() );
892 curListItem->SetLayerWireLengths( newListItem->GetLayerWireLengths() );
893
894 updateDisplayedRowValues( curNetRow );
895 }
896 }
897
898 // Delete any nets we have not yet handled
899 for( const NETINFO_ITEM* netToDelete : netsToDelete )
900 m_dataModel->deleteItem( m_dataModel->findItem( netToDelete->GetNetCode() ) );
901
902 m_netsList->Thaw();
903}
904
905
907{
908 const std::vector<BOARD_ITEM*> item{ aBoardItem };
909 updateBoardItems( item );
910}
911
912
913void PCB_NET_INSPECTOR_PANEL::OnBoardItemsRemoved( BOARD& aBoard, std::vector<BOARD_ITEM*>& aBoardItems )
914{
915 updateBoardItems( aBoardItems );
916}
917
918
920{
921 if( !IsShownOnScreen() )
922 return;
923
925}
926
927
929{
930 const std::vector<BOARD_ITEM*> item{ aBoardItem };
931 updateBoardItems( item );
932}
933
934
936 std::vector<BOARD_ITEM*>& aBoardItems )
937{
938 updateBoardItems( aBoardItems );
939}
940
941
943 std::vector<BOARD_ITEM*>& aAddedItems,
944 std::vector<BOARD_ITEM*>& aRemovedItems,
945 std::vector<BOARD_ITEM*>& aChangedItems )
946{
947 if( !IsShownOnScreen() )
948 return;
949
950 std::vector<BOARD_ITEM*> allItems{ aAddedItems.begin(), aAddedItems.end() };
951 allItems.insert( allItems.end(), aRemovedItems.begin(), aRemovedItems.end() );
952 allItems.insert( allItems.end(), aChangedItems.begin(), aChangedItems.end() );
953 updateBoardItems( allItems );
954}
955
956
958{
959 if( m_highlightingNets || !IsShownOnScreen() )
960 return;
961
962 if( !m_board->IsHighLightNetON() )
963 {
964 m_netsList->UnselectAll();
965 }
966 else
967 {
968 const std::set<int>& selected_codes = m_board->GetHighLightNetCodes();
969
970 wxDataViewItemArray new_selection;
971 new_selection.Alloc( selected_codes.size() );
972
973 for( const int code : selected_codes )
974 {
975 if( std::optional<LIST_ITEM_ITER> r = m_dataModel->findItem( code ) )
976 new_selection.Add( wxDataViewItem( &***r ) );
977 }
978
979 m_netsList->SetSelections( new_selection );
980
981 if( !new_selection.IsEmpty() )
982 m_netsList->EnsureVisible( new_selection.Item( 0 ) );
983 }
984}
985
986
987/*****************************************************************************************
988 *
989 * UI-generated event handling
990 *
991 * ***************************************************************************************/
992
994{
997}
998
999
1001{
1002 bool multipleSelections = false;
1003 const LIST_ITEM* selItem = nullptr;
1004
1005 if( m_netsList->GetSelectedItemsCount() == 1 )
1006 {
1007 selItem = static_cast<const LIST_ITEM*>( m_netsList->GetSelection().GetID() );
1008 }
1009 else
1010 {
1011 if( m_netsList->GetSelectedItemsCount() > 1 )
1012 multipleSelections = true;
1013 }
1014
1015 wxMenu menu;
1016
1017 // Net edit menu items
1018 wxMenuItem* highlightNet = new wxMenuItem( &menu, ID_HIGHLIGHT_SELECTED_NETS,
1019 _( "Highlight Selected Net" ),
1020 wxEmptyString, wxITEM_NORMAL );
1021 menu.Append( highlightNet );
1022
1023 wxMenuItem* clearHighlighting = new wxMenuItem( &menu, ID_CLEAR_HIGHLIGHTING,
1024 _( "Clear Net Highlighting" ),
1025 wxEmptyString, wxITEM_NORMAL );
1026 menu.Append( clearHighlighting );
1027
1028 RENDER_SETTINGS* renderSettings = m_frame->GetCanvas()->GetView()->GetPainter()->GetSettings();
1029 const std::set<int>& selected_codes = renderSettings->GetHighlightNetCodes();
1030
1031 if( selected_codes.size() == 0 )
1032 clearHighlighting->Enable( false );
1033
1034 menu.AppendSeparator();
1035
1036 wxMenuItem* renameNet = new wxMenuItem( &menu, ID_RENAME_NET, _( "Rename Selected Net..." ),
1037 wxEmptyString, wxITEM_NORMAL );
1038 menu.Append( renameNet );
1039
1040 wxMenuItem* deleteNet = new wxMenuItem( &menu, ID_DELETE_NET, _( "Delete Selected Net" ),
1041 wxEmptyString, wxITEM_NORMAL );
1042 menu.Append( deleteNet );
1043
1044 menu.AppendSeparator();
1045
1046 wxMenuItem* addNet = new wxMenuItem( &menu, ID_ADD_NET, _( "Add Net..." ),
1047 wxEmptyString, wxITEM_NORMAL );
1048 menu.Append( addNet );
1049
1050 if( !selItem && !multipleSelections )
1051 {
1052 highlightNet->Enable( false );
1053 deleteNet->Enable( false );
1054 renameNet->Enable( false );
1055 }
1056 else
1057 {
1058 if( multipleSelections || selItem->GetIsGroup() )
1059 {
1060 highlightNet->SetItemLabel( _( "Highlight Selected Nets" ) );
1061 renameNet->Enable( false );
1062 deleteNet->SetItemLabel( _( "Delete Selected Nets" ) );
1063 }
1064 }
1065
1066 menu.AppendSeparator();
1067
1068 wxMenuItem* removeSelectedGroup = new wxMenuItem( &menu, ID_REMOVE_SELECTED_GROUP,
1069 _( "Remove Selected Custom Group" ),
1070 wxEmptyString, wxITEM_NORMAL );
1071 menu.Append( removeSelectedGroup );
1072
1073 if( !selItem || !selItem->GetIsGroup() )
1074 removeSelectedGroup->Enable( false );
1075
1076 menu.Bind( wxEVT_COMMAND_MENU_SELECTED, &PCB_NET_INSPECTOR_PANEL::onContextMenuSelection, this );
1077
1078 PopupMenu( &menu );
1079}
1080
1081
1083{
1084 SaveSettings();
1085 buildNetsList();
1086}
1087
1088
1090{
1091 wxString newGroupName;
1092 NETNAME_VALIDATOR validator( &newGroupName );
1093
1094 WX_TEXT_ENTRY_DIALOG dlg( this, _( "Group name / pattern:" ), _( "New Group" ), newGroupName );
1095 wxStaticText* help = new wxStaticText( &dlg, wxID_ANY,
1096 _( "(Use /.../ to indicate a regular expression.)" ) );
1097 help->SetFont( KIUI::GetInfoFont( this ).Italic() );
1098 dlg.m_ContentSizer->Add( help, 0, wxALL|wxEXPAND, 5 );
1099 dlg.SetTextValidator( validator );
1100 dlg.GetSizer()->SetSizeHints( &dlg );
1101
1102 if( dlg.ShowModal() != wxID_OK || dlg.GetValue().IsEmpty() )
1103 return; //Aborted by user
1104
1105 newGroupName = UnescapeString( dlg.GetValue() );
1106
1107 if( newGroupName == "" )
1108 return;
1109
1110 if( std::ranges::find_if( m_custom_group_rules,
1111 [&]( std::unique_ptr<EDA_COMBINED_MATCHER>& rule )
1112 {
1113 return rule->GetPattern() == newGroupName;
1114 } )
1115 == m_custom_group_rules.end() )
1116 {
1117 m_custom_group_rules.push_back( std::make_unique<EDA_COMBINED_MATCHER>( newGroupName,
1118 CTX_NET ) );
1119 SaveSettings();
1120 }
1121
1122 buildNetsList();
1123}
1124
1125
1127{
1128 m_highlightingNets = true;
1129
1133
1134 m_highlightingNets = false;
1135}
1136
1137
1139{
1140 if( !m_rowExpanding )
1141 SaveSettings();
1142}
1143
1144
1146{
1147 wxMenu menu;
1149 menu.Bind( wxEVT_COMMAND_MENU_SELECTED, &PCB_NET_INSPECTOR_PANEL::onContextMenuSelection, this );
1150 PopupMenu( &menu );
1151}
1152
1153
1154void PCB_NET_INSPECTOR_PANEL::OnConfigButton( wxCommandEvent& event )
1155{
1157 auto& cfg = localSettings.m_NetInspectorPanel;
1158
1159 const LIST_ITEM* selItem = nullptr;
1160
1161 if( m_netsList->GetSelectedItemsCount() == 1 )
1162 {
1163 selItem = static_cast<const LIST_ITEM*>( m_netsList->GetSelection().GetID() );
1164 }
1165
1166 wxMenu menu;
1167
1168 // Filtering menu items
1169 wxMenuItem* filterByNetName = new wxMenuItem( &menu, ID_FILTER_BY_NET_NAME,
1170 _( "Filter by Net Name" ),
1171 wxEmptyString, wxITEM_CHECK );
1172 menu.Append( filterByNetName );
1173 filterByNetName->Check( cfg.filter_by_net_name );
1174
1175 wxMenuItem* filterByNetclass = new wxMenuItem( &menu, ID_FILTER_BY_NETCLASS,
1176 _( "Filter by Netclass" ),
1177 wxEmptyString, wxITEM_CHECK );
1178 menu.Append( filterByNetclass );
1179 filterByNetclass->Check( cfg.filter_by_netclass );
1180
1181 menu.AppendSeparator();
1182
1183 // Grouping menu items
1184 //wxMenuItem* groupConstraint =
1185 // new wxMenuItem( &menu, ID_GROUP_BY_CONSTRAINT, _( "Group by DRC Constraint" ),
1186 // wxEmptyString, wxITEM_CHECK );
1187 //groupConstraint->Check( m_group_by_constraint );
1188 //menu.Append( groupConstraint );
1189
1190 wxMenuItem* groupNetclass = new wxMenuItem( &menu, ID_GROUP_BY_NETCLASS,
1191 _( "Group by Netclass" ),
1192 wxEmptyString, wxITEM_CHECK );
1193 menu.Append( groupNetclass );
1194 groupNetclass->Check( m_groupByNetclass );
1195
1196 menu.AppendSeparator();
1197
1198 wxMenuItem* addGroup = new wxMenuItem( &menu, ID_ADD_GROUP, _( "Add Custom Group..." ),
1199 wxEmptyString, wxITEM_NORMAL );
1200 menu.Append( addGroup );
1201
1202 wxMenuItem* removeSelectedGroup = new wxMenuItem( &menu, ID_REMOVE_SELECTED_GROUP,
1203 _( "Remove Selected Custom Group" ),
1204 wxEmptyString, wxITEM_NORMAL );
1205 menu.Append( removeSelectedGroup );
1206
1207 if( !selItem || !selItem->GetIsGroup() )
1208 removeSelectedGroup->Enable( false );
1209
1210 wxMenuItem* removeCustomGroups = new wxMenuItem( &menu, ID_REMOVE_GROUPS,
1211 _( "Remove All Custom Groups" ),
1212 wxEmptyString, wxITEM_NORMAL );
1213 menu.Append( removeCustomGroups );
1214 removeCustomGroups->Enable( m_custom_group_rules.size() != 0 );
1215
1216 menu.AppendSeparator();
1217
1218 wxMenuItem* showZeroNetPads = new wxMenuItem( &menu, ID_SHOW_ZERO_NET_PADS,
1219 _( "Show Zero Pad Nets" ),
1220 wxEmptyString, wxITEM_CHECK );
1221 menu.Append( showZeroNetPads );
1222 showZeroNetPads->Check( m_showZeroPadNets );
1223
1224 wxMenuItem* showUnconnectedNets = new wxMenuItem( &menu, ID_SHOW_UNCONNECTED_NETS,
1225 _( "Show Unconnected Nets" ),
1226 wxEmptyString, wxITEM_CHECK );
1227 menu.Append( showUnconnectedNets );
1228 showUnconnectedNets->Check( m_showUnconnectedNets );
1229
1230 menu.AppendSeparator();
1231
1232 // Report generation
1233 wxMenuItem* generateReport = new wxMenuItem( &menu, ID_GENERATE_REPORT,
1234 _( "Save Net Inspector Report..." ),
1235 wxEmptyString, wxITEM_NORMAL );
1236 menu.Append( generateReport );
1237
1238 menu.AppendSeparator();
1239
1240 // Show / hide columns menu items
1241 wxMenu* colsMenu = new wxMenu();
1242 generateShowHideColumnMenu( colsMenu );
1243 menu.AppendSubMenu( colsMenu, _( "Show / Hide Columns" ) );
1244
1245 menu.Bind( wxEVT_COMMAND_MENU_SELECTED, &PCB_NET_INSPECTOR_PANEL::onContextMenuSelection, this );
1246
1247 PopupMenu( &menu );
1248}
1249
1250
1252{
1253 for( int i = 1; i <= COLUMN_LAST_STATIC_COL; ++i )
1254 {
1255 wxMenuItem* opt = new wxMenuItem( target, ID_HIDE_COLUMN + i, m_columns[i].display_name,
1256 wxEmptyString, wxITEM_CHECK );
1257 wxDataViewColumn* col = getDisplayedColumnForModelField( i );
1258 target->Append( opt );
1259 opt->Check( !col->IsHidden() );
1260 }
1261
1262 target->AppendSeparator();
1263
1264 for( std::size_t i = COLUMN_LAST_STATIC_COL + 1; i < m_columns.size(); ++i )
1265 {
1266 wxMenuItem* opt = new wxMenuItem( target, ID_HIDE_COLUMN + i, m_columns[i].display_name,
1267 wxEmptyString, wxITEM_CHECK );
1268 wxDataViewColumn* col = getDisplayedColumnForModelField( i );
1269 target->Append( opt );
1270 opt->Check( !col->IsHidden() );
1271 }
1272}
1273
1274
1276{
1277 bool saveAndRebuild = true;
1278
1279 switch( event.GetId() )
1280 {
1281 case ID_ADD_NET:
1282 onAddNet();
1283 break;
1284
1285 case ID_RENAME_NET:
1287 break;
1288
1289 case ID_DELETE_NET:
1291 break;
1292
1293 case ID_ADD_GROUP:
1294 onAddGroup();
1295 break;
1296
1298
1300
1302
1304
1307 break;
1308
1309 case ID_REMOVE_GROUPS:
1310 m_custom_group_rules.clear();
1311 break;
1312
1314
1316
1317 case ID_GENERATE_REPORT:
1319 saveAndRebuild = false;
1320 break;
1321
1324 saveAndRebuild = false;
1325 break;
1326
1329 saveAndRebuild = false;
1330 break;
1331
1332 default:
1333 if( event.GetId() >= ID_HIDE_COLUMN )
1334 {
1335 const int columnId = event.GetId() - ID_HIDE_COLUMN;
1336 wxDataViewColumn* col = getDisplayedColumnForModelField( columnId );
1337 // Make sure we end up with something non-zero so we can resize it
1338 col->SetWidth( std::max( col->GetWidth(), 10 ) );
1339 col->SetHidden( !col->IsHidden() );
1340 }
1341 break;
1342 }
1343
1344 if( saveAndRebuild )
1345 {
1346 SaveSettings();
1347 buildNetsList();
1348 }
1349}
1350
1351
1353{
1354 if( m_netsList->GetSelectedItemsCount() == 1 )
1355 {
1356 auto* selItem = static_cast<const LIST_ITEM*>( m_netsList->GetSelection().GetID() );
1357
1358 if( selItem->GetIsGroup() )
1359 {
1360 const wxString groupName = selItem->GetGroupName();
1361 const auto groupIter = std::ranges::find_if( m_custom_group_rules,
1362 [&]( std::unique_ptr<EDA_COMBINED_MATCHER>& rule )
1363 {
1364 return rule->GetPattern() == groupName;
1365 } );
1366
1367 if( groupIter != m_custom_group_rules.end() )
1368 {
1369 m_custom_group_rules.erase( groupIter );
1370 SaveSettings();
1371 buildNetsList();
1372 }
1373 }
1374 }
1375}
1376
1377
1379{
1380 wxFileDialog dlg( this, _( "Save Net Inspector Report File" ), "", "",
1381 _( "Report file" ) + AddFileExtListToFilter( { "csv" } ),
1382 wxFD_SAVE | wxFD_OVERWRITE_PROMPT );
1383
1384 if( dlg.ShowModal() == wxID_CANCEL )
1385 return;
1386
1387 wxTextFile f( dlg.GetPath() );
1388
1389 f.Create();
1390
1391 wxString txt;
1392
1393 m_inReporting = true;
1394
1395 // Print Header:
1396 for( auto&& col : m_columns )
1397 {
1398 txt += '"';
1399
1400 if( col.has_units )
1401 {
1402 txt += wxString::Format( _( "%s (%s)" ),
1403 col.csv_name,
1405 }
1406 else
1407 {
1408 txt += col.csv_name;
1409 }
1410
1411 txt += wxT( "\";" );
1412 }
1413
1414 f.AddLine( txt );
1415
1416 // Print list of nets:
1417 const unsigned int num_rows = m_dataModel->itemCount();
1418
1419 for( unsigned int row = 0; row < num_rows; row++ )
1420 {
1421 auto& i = m_dataModel->itemAt( row );
1422
1423 if( i.GetIsGroup() || i.GetNetCode() == 0 )
1424 continue;
1425
1426 txt = "";
1427
1428 for( auto&& col : m_columns )
1429 {
1430 if( static_cast<int>( col.csv_flags ) & static_cast<int>( CSV_COLUMN_DESC::CSV_QUOTE ) )
1431 txt += '"' + m_dataModel->valueAt( col.num, row ).GetString() + wxT( "\";" );
1432 else
1433 txt += m_dataModel->valueAt( col.num, row ).GetString() + ';';
1434 }
1435
1436 f.AddLine( txt );
1437 }
1438
1439 m_inReporting = false;
1440
1441 f.Write();
1442 f.Close();
1443}
1444
1445
1447{
1449}
1450
1451
1453{
1454 // ignore selection changes while the whole list is being rebuilt.
1455 if( m_inBuildNetsList )
1456 return;
1457
1458 m_highlightingNets = true;
1459
1460 RENDER_SETTINGS* renderSettings = m_frame->GetCanvas()->GetView()->GetPainter()->GetSettings();
1461
1462 if( m_netsList->HasSelection() )
1463 {
1464 wxDataViewItemArray sel;
1465 m_netsList->GetSelections( sel );
1466
1467 renderSettings->SetHighlight( false );
1468
1469 for( unsigned int i = 0; i < sel.GetCount(); ++i )
1470 {
1471 const LIST_ITEM* ii = static_cast<const LIST_ITEM*>( sel.Item( i ).GetID() );
1472
1473 if( ii->GetIsGroup() )
1474 {
1475 for( auto c = ii->ChildrenBegin(), end = ii->ChildrenEnd(); c != end; ++c )
1476 renderSettings->SetHighlight( true, ( *c )->GetNetCode(), true );
1477 }
1478 else
1479 {
1480 renderSettings->SetHighlight( true, ii->GetNetCode(), true );
1481 }
1482 }
1483 }
1484 else
1485 {
1486 renderSettings->SetHighlight( false );
1487 }
1488
1491
1492 m_highlightingNets = false;
1493}
1494
1495
1496void PCB_NET_INSPECTOR_PANEL::OnColumnSorted( wxDataViewEvent& event )
1497{
1498 if( !m_inBuildNetsList )
1499 SaveSettings();
1500}
1501
1502
1504{
1505 // Rebuilt the nets list, and force rebuild of columns in case the stackup has changed
1506 buildNetsList( true );
1507}
1508
1509
1511{
1512 wxString newNetName;
1513 NETNAME_VALIDATOR validator( &newNetName );
1514
1515 WX_TEXT_ENTRY_DIALOG dlg( this, _( "Net name:" ), _( "New Net" ), newNetName );
1516 dlg.SetTextValidator( validator );
1517
1518 while( true )
1519 {
1520 if( dlg.ShowModal() != wxID_OK || dlg.GetValue().IsEmpty() )
1521 return; //Aborted by user
1522
1523 newNetName = dlg.GetValue();
1524
1525 if( m_board->FindNet( newNetName ) )
1526 {
1527 DisplayError( this,
1528 wxString::Format( _( "Net name '%s' is already in use." ), newNetName ) );
1529 newNetName = wxEmptyString;
1530 }
1531 else
1532 {
1533 break;
1534 }
1535 }
1536
1537 NETINFO_ITEM* newnet = new NETINFO_ITEM( m_board, dlg.GetValue(), 0 );
1538
1539 m_board->Add( newnet );
1540
1541 // We'll get an OnBoardItemAdded callback from this to update our listbox
1542 m_frame->OnModify();
1543}
1544
1545
1547{
1548 if( m_netsList->GetSelectedItemsCount() == 1 )
1549 {
1550 const LIST_ITEM* sel = static_cast<const LIST_ITEM*>( m_netsList->GetSelection().GetID() );
1551
1552 if( sel->GetIsGroup() )
1553 return;
1554
1555 NETINFO_ITEM* net = sel->GetNet();
1556 wxString fullNetName = net->GetNetname();
1557 wxString netPath;
1558 wxString shortNetName;
1559
1560 if( fullNetName.Contains( wxT( "/" ) ) )
1561 {
1562 netPath = fullNetName.BeforeLast( '/' ) + '/';
1563 shortNetName = fullNetName.AfterLast( '/' );
1564 }
1565 else
1566 {
1567 shortNetName = fullNetName;
1568 }
1569
1570 wxString unescapedShortName = UnescapeString( shortNetName );
1571
1572 WX_TEXT_ENTRY_DIALOG dlg( this, _( "Net name:" ), _( "Rename Net" ), unescapedShortName );
1573 NETNAME_VALIDATOR validator( &unescapedShortName );
1574 dlg.SetTextValidator( validator );
1575
1576 while( true )
1577 {
1578 if( dlg.ShowModal() != wxID_OK || dlg.GetValue() == unescapedShortName )
1579 return;
1580
1581 unescapedShortName = dlg.GetValue();
1582
1583 if( unescapedShortName.IsEmpty() )
1584 {
1585 DisplayError( this, wxString::Format( _( "Net name cannot be empty." ),
1586 unescapedShortName ) );
1587 continue;
1588 }
1589
1590 shortNetName = EscapeString( unescapedShortName, CTX_NETNAME );
1591 fullNetName = netPath + shortNetName;
1592
1593 if( m_board->FindNet( shortNetName ) || m_board->FindNet( fullNetName ) )
1594 {
1595 DisplayError( this, wxString::Format( _( "Net name '%s' is already in use." ),
1596 unescapedShortName ) );
1597 unescapedShortName = wxEmptyString;
1598 }
1599 else
1600 {
1601 break;
1602 }
1603 }
1604
1605 for( BOARD_CONNECTED_ITEM* boardItem : m_frame->GetBoard()->AllConnectedItems() )
1606 {
1607 if( boardItem->GetNet() == net )
1608 boardItem->SetFlags( CANDIDATE );
1609 else
1610 boardItem->ClearFlags( CANDIDATE );
1611 }
1612
1613 // the changed name might require re-grouping. remove/re-insert is easier.
1614 auto removed_item = m_dataModel->deleteItem( m_dataModel->findItem( net ) );
1615
1616 m_board->Remove( net );
1617 net->SetNetname( fullNetName );
1618 m_board->Add( net );
1619
1620 for( BOARD_CONNECTED_ITEM* boardItem : m_frame->GetBoard()->AllConnectedItems() )
1621 {
1622 if( boardItem->GetFlags() & CANDIDATE )
1623 boardItem->SetNet( net );
1624 }
1625
1626 buildNetsList();
1627
1628 if( std::optional<LIST_ITEM_ITER> r = m_dataModel->findItem( net ) )
1629 m_netsList->Select( wxDataViewItem( r.value()->get() ) );
1630
1631 m_frame->OnModify();
1632
1633 // Currently only tracks and pads have netname annotations and need to be redrawn,
1634 // but zones are likely to follow. Since we don't have a way to ask what is current,
1635 // just refresh all items.
1638 }
1639}
1640
1641
1643{
1644 if( !m_netsList->HasSelection() )
1645 return;
1646
1647 wxDataViewItemArray sel;
1648 m_netsList->GetSelections( sel );
1649
1650 auto delete_one = [this]( const LIST_ITEM* i )
1651 {
1652 if( i->GetPadCount() == 0
1653 || IsOK( this, wxString::Format( _( "Net '%s' is in use. Delete anyway?" ),
1654 i->GetNetName() ) ) )
1655 {
1656 // This is a bit hacky, but it will do for now, since this is the only path
1657 // outside the netlist updater where you can remove a net from a BOARD.
1658 int removedCode = i->GetNetCode();
1659
1661 [removedCode]( KIGFX::VIEW_ITEM* aItem ) -> int
1662 {
1663 auto boardItem = dynamic_cast<BOARD_CONNECTED_ITEM*>( aItem );
1664
1665 if( boardItem && boardItem->GetNetCode() == removedCode )
1666 return KIGFX::REPAINT;
1667
1668 EDA_TEXT* text = dynamic_cast<EDA_TEXT*>( aItem );
1669
1670 if( text && text->HasTextVars() )
1671 {
1672 text->ClearRenderCache();
1673 text->ClearBoundingBoxCache();
1675 }
1676
1677 return 0;
1678 } );
1679
1680 m_board->Remove( i->GetNet() );
1681 m_frame->OnModify();
1682
1683 // We'll get an OnBoardItemRemoved callback from this to update our listbox
1684 }
1685 };
1686
1687 for( unsigned int i = 0; i < sel.GetCount(); ++i )
1688 {
1689 const LIST_ITEM* ii = static_cast<const LIST_ITEM*>( sel.Item( i ).GetID() );
1690
1691 if( ii->GetIsGroup() )
1692 {
1693 if( ii->ChildrenCount() != 0
1694 && IsOK( this, wxString::Format( _( "Delete all nets in group '%s'?" ),
1695 ii->GetGroupName() ) ) )
1696 {
1697 // we can't be iterating the children container and deleting items from
1698 // it at the same time. thus take a copy of it first.
1699 std::vector<const LIST_ITEM*> children;
1700 children.reserve( ii->ChildrenCount() );
1701 std::copy( ii->ChildrenBegin(), ii->ChildrenEnd(), std::back_inserter( children ) );
1702
1703 for( const LIST_ITEM* c : children )
1704 delete_one( c );
1705 }
1706 }
1707 else
1708 {
1709 delete_one( ii );
1710 }
1711 }
1712}
1713
1714/*****************************************************************************************
1715 *
1716 * Application-generated event handling
1717 *
1718 * ***************************************************************************************/
1719
1720
1722{
1723 SaveSettings();
1724 buildNetsList( true );
1725 m_dataModel->updateAllItems();
1726}
1727
1728
1729void PCB_NET_INSPECTOR_PANEL::onUnitsChanged( wxCommandEvent& event )
1730{
1731 m_dataModel->updateAllItems();
1732 event.Skip();
1733}
1734
1735
1736/*****************************************************************************************
1737 *
1738 * Settings persistence
1739 *
1740 * ***************************************************************************************/
1741
1742
1744{
1745 // Don't save settings if a board has not yet been loaded or the panel hasn't been displayed.
1746 // Events fire while we set up the panel which overwrite the settings we haven't yet loaded.
1747 bool displayed = false;
1748
1749 for( unsigned int ii = 0; ii < m_dataModel->columnCount() && !displayed; ++ii )
1750 {
1751 if( m_netsList->GetColumn( ii )->GetWidth() > 0 )
1752 displayed = true;
1753 }
1754
1755 if( !displayed || !m_boardLoaded || m_boardLoading )
1756 return;
1757
1759 auto& cfg = localSettings.m_NetInspectorPanel;
1760
1761 // User-defined filters / grouping
1762 cfg.filter_text = m_searchCtrl->GetValue();
1763 cfg.filter_by_net_name = m_filterByNetName;
1764 cfg.filter_by_netclass = m_filterByNetclass;
1765 cfg.group_by_netclass = m_groupByNetclass;
1766 cfg.group_by_constraint = m_groupByConstraint;
1767 cfg.show_zero_pad_nets = m_showZeroPadNets;
1768 cfg.show_unconnected_nets = m_showUnconnectedNets;
1769
1770 // Grid sorting
1771 wxDataViewColumn* sortingCol = m_netsList->GetSortingColumn();
1772 cfg.sorting_column = sortingCol ? static_cast<int>( sortingCol->GetModelColumn() ) : -1;
1773 cfg.sort_order_asc = sortingCol ? sortingCol->IsSortOrderAscending() : true;
1774
1775 // Column arrangement / sizes
1776 cfg.col_order.resize( m_dataModel->columnCount() );
1777 cfg.col_widths.resize( m_dataModel->columnCount() );
1778 cfg.col_hidden.resize( m_dataModel->columnCount() );
1779
1780 for( unsigned int ii = 0; ii < m_dataModel->columnCount(); ++ii )
1781 {
1782 cfg.col_order[ii] = (int) m_netsList->GetColumn( ii )->GetModelColumn();
1783 cfg.col_widths[ii] = m_netsList->GetColumn( ii )->GetWidth();
1784 cfg.col_hidden[ii] = m_netsList->GetColumn( ii )->IsHidden();
1785 }
1786
1787 // Expanded rows
1788 cfg.expanded_rows.clear();
1789 std::vector<std::pair<wxString, wxDataViewItem>> groupItems = m_dataModel->getGroupDataViewItems();
1790
1791 for( std::pair<wxString, wxDataViewItem>& item : groupItems )
1792 {
1793 if( m_netsList->IsExpanded( item.second ) )
1794 cfg.expanded_rows.push_back( item.first );
1795 }
1796
1797 // Customer group rules
1798 cfg.custom_group_rules.clear();
1799
1800 for( const std::unique_ptr<EDA_COMBINED_MATCHER>& rule : m_custom_group_rules )
1801 cfg.custom_group_rules.push_back( rule->GetPattern() );
1802}
static const ADVANCED_CFG & GetCfg()
Get the singleton instance's config, which is shared by all consumers.
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition: board_item.h:78
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:297
const NETINFO_LIST & GetNetInfo() const
Definition: board.h:897
LSET GetEnabledLayers() const
A proxy function that calls the corresponding function in m_BoardSettings.
Definition: board.cpp:829
void Add(BOARD_ITEM *aItem, ADD_MODE aMode=ADD_MODE::INSERT, bool aSkipConnectivity=false) override
Removes an item from the container.
Definition: board.cpp:1060
const std::set< int > & GetHighLightNetCodes() const
Definition: board.h:552
const std::vector< BOARD_CONNECTED_ITEM * > AllConnectedItems()
Definition: board.cpp:2656
void AddListener(BOARD_LISTENER *aListener)
Add a listener to the board to receive calls whenever something on the board has been modified.
Definition: board.cpp:2711
NETINFO_ITEM * FindNet(int aNetcode) const
Search for a net with the given netcode.
Definition: board.cpp:2011
int GetCopperLayerCount() const
Definition: board.cpp:781
bool IsHighLightNetON() const
Definition: board.h:568
const wxString GetLayerName(PCB_LAYER_ID aLayer) const
Return the name of a aLayer.
Definition: board.cpp:614
LENGTH_CALCULATION * GetLengthCalculation() const
Returns the track length calculator.
Definition: board.h:1311
void Remove(BOARD_ITEM *aBoardItem, REMOVE_MODE aMode=REMOVE_MODE::NORMAL) override
Removes an item from the container.
Definition: board.cpp:1213
std::shared_ptr< CONNECTIVITY_DATA > GetConnectivity() const
Return a list of missing connections between components/tracks.
Definition: board.h:495
CN_ITEM represents a BOARD_CONNETED_ITEM in the connectivity system (ie: a pad, track/arc/via,...
int Net() const
int ShowModal() override
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
A mix-in class (via multiple inheritance) that handles texts such as labels, parts,...
Definition: eda_text.h:80
virtual RENDER_SETTINGS * GetSettings()=0
Return a pointer to current settings that are going to be used when drawing items.
Container for all the knowledge about how graphical objects are drawn on any output surface/device.
const std::set< int > & GetHighlightNetCodes() const
Return the netcode of currently highlighted net.
void SetHighlight(bool aEnabled, int aNetcode=-1, bool aMulti=false)
Turns on/off highlighting.
An abstract base class for deriving all objects that can be added to a VIEW.
Definition: view_item.h:86
void UpdateAllLayersColor()
Apply the new coloring scheme to all layers.
Definition: view.cpp:765
void UpdateAllItems(int aUpdateFlags)
Update all items in the view according to the given flags.
Definition: view.cpp:1549
PAINTER * GetPainter() const
Return the painter object used by the view for drawing #VIEW_ITEMS.
Definition: view.h:216
void UpdateAllItemsConditionally(int aUpdateFlags, std::function< bool(VIEW_ITEM *)> aCondition)
Update items in the view according to the given flags and condition.
Definition: view.cpp:1559
Lightweight class which holds a pad, via, or a routed trace outline.
Class which calculates lengths (and associated routing statistics) in a BOARD context.
LENGTH_CALCULATION_ITEM GetLengthCalculationItem(BOARD_CONNECTED_ITEM *aBoardItem) const
Return a LENGTH_CALCULATION_ITEM constructed from the given BOARD_CONNECTED_ITEM.
LSEQ Seq(const LSEQ &aSequence) const
Return an LSEQ from the union of this LSET and a desired sequence.
Definition: lset.cpp:297
A collection of nets and the parameters used to route or test these nets.
Definition: netclass.h:45
const wxString GetName() const
Gets the name of this (maybe aggregate) netclass in a format for internal usage or for export to exte...
Definition: netclass.cpp:314
Handle the data for a net.
Definition: netinfo.h:56
void SetNetname(const wxString &aNewName)
Set the long netname to aNetName, the short netname to the last token in the long netname's path,...
Definition: netinfo.h:139
const wxString & GetNetname() const
Definition: netinfo.h:114
NETCLASS * GetNetClass()
Definition: netinfo.h:101
int GetNetCode() const
Definition: netinfo.h:108
A base class used to implement docking net inspector panels.
wxDataViewCtrl * m_netsList
wxSearchCtrl * m_searchCtrl
Definition: pad.h:54
PCB_DRAW_PANEL_GAL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
BOARD * GetBoard() const
virtual KIGFX::PCB_VIEW * GetView() const override
Return a pointer to the #VIEW instance used in the panel.
The main frame for Pcbnew.
void OnModify() override
Must be called after a board change to set the modified flag.
Data model for display in the Net Inspector panel.
std::vector< std::pair< wxString, wxDataViewItem > > getGroupDataViewItems()
Primary data item for entries in the Net Inspector list.
void OnBoardItemRemoved(BOARD &aBoard, BOARD_ITEM *aBoardItem) override
void OnBoardHighlightNetChanged(BOARD &aBoard) override
void generateReport()
Generates a CSV report from currently disaplyed data.
PCB_NET_INSPECTOR_PANEL(wxWindow *parent, PCB_EDIT_FRAME *aFrame)
void onDeleteSelectedNet()
Deletes a selected net.
bool restoreSortColumn(int sortingColumnId, bool sortOrderAsc) const
Sets the sort column in the grid to that showing the given model ID column.
void OnBoardItemsChanged(BOARD &aBoard, std::vector< BOARD_ITEM * > &aBoardItems) override
void adjustListColumnSizes(PANEL_NET_INSPECTOR_SETTINGS *cfg) const
Adjust the sizing of list columns.
void OnSearchTextChanged(wxCommandEvent &event) override
void updateDisplayedRowValues(const std::optional< LIST_ITEM_ITER > &aRow) const
Refreshes displayed data for the given rows.
void onAddNet()
Adds a new user-specified net to the board.
PCB_EDIT_FRAME * m_frame
Owning edit frame.
void OnBoardItemsRemoved(BOARD &aBoard, std::vector< BOARD_ITEM * > &aBoardItems) override
void onClearHighlighting()
Clears highlighting from nets.
void onContextMenuSelection(wxCommandEvent &event)
Handle a net row(s) context menu selection.
static wxString formatNetCode(const NETINFO_ITEM *aNet)
void OnBoardChanged() override
Update panel when board is changed.
void generateShowHideColumnMenu(wxMenu *target)
Generates a sub-menu for the show / hide columns submenu.
std::vector< std::unique_ptr< LIST_ITEM > > calculateNets(const std::vector< NETINFO_ITEM * > &aNetCodes, bool aIncludeZeroPadNets) const
Calculates the length statistics for each given netcode.
static wxString formatCount(unsigned int aValue)
std::vector< std::unique_ptr< EDA_COMBINED_MATCHER > > m_custom_group_rules
Custom net grouping rules.
std::vector< CN_ITEM * > relevantConnectivityItems() const
Fetches an ordered (by NetCode) list of all board connectivity items.
void highlightSelectedNets()
Highlight the currently selected net.
void updateNets(const std::vector< NETINFO_ITEM * > &aNets) const
Updates displayed statistics for the given nets.
void OnBoardItemsAdded(BOARD &aBoard, std::vector< BOARD_ITEM * > &aBoardItems) override
void OnLanguageChangedImpl() override
Reloads strings on an application language change.
void onAddGroup()
Adds a custom display grouping of nets.
void OnNetsListContextMenu(wxDataViewEvent &event)
void OnBoardNetSettingsChanged(BOARD &aBoard) override
void OnBoardCompositeUpdate(BOARD &aBoard, std::vector< BOARD_ITEM * > &aAddedItems, std::vector< BOARD_ITEM * > &aRemovedItems, std::vector< BOARD_ITEM * > &aChangedItems) override
void OnNetsListItemActivated(wxDataViewEvent &event)
void OnColumnSorted(wxDataViewEvent &event)
void onRemoveSelectedGroup()
Removes a custom display grouping.
void buildColumns()
Build the required columns in the net inspector grid.
void OnBoardItemAdded(BOARD &aBoard, BOARD_ITEM *aBoardItem) override
void OnParentSetupChanged() override
Updates the netlist based on global board changes (e.g.
void OnHeaderContextMenu(wxCommandEvent &event)
bool netFilterMatches(NETINFO_ITEM *aNet, PANEL_NET_INSPECTOR_SETTINGS *cfg=nullptr) const
Filter to determine whether a board net should be included in the net inspector.
void OnConfigButton(wxCommandEvent &event) override
void buildNetsList(bool rebuildColumns=false)
Rebuilds the net inspector list, removing all previous entries.
wxString formatLength(int64_t aValue) const
void SaveSettings() override
Persist the net inspector configuration to project / global settings.
void onRenameSelectedNet()
Renames a selected net.
void onUnitsChanged(wxCommandEvent &event)
Handle an application-level change of units.
void OnExpandCollapseRow(wxCommandEvent &event)
void updateBoardItems(const std::vector< BOARD_ITEM * > &aBoardItems)
Unified handling of added / deleted / modified board items.
wxObjectDataPtr< DATA_MODEL > m_dataModel
The bound data model to display.
wxDataViewColumn * getDisplayedColumnForModelField(int columnId) const
Fetches the displayed grid view column for the given model column ID.
static wxString formatNetName(const NETINFO_ITEM *aNet)
void OnBoardItemChanged(BOARD &aBoard, BOARD_ITEM *aBoardItem) override
std::vector< COLUMN_DESC > m_columns
All displayed (or hidden) columns.
void OnShowPanel() override
Prepare the panel when shown in the editor.
virtual SETTINGS_MANAGER & GetSettingsManager() const
Definition: pgm_base.h:125
The project local settings are things that are attached to a particular project, but also might be pa...
PANEL_NET_INSPECTOR_SETTINGS m_NetInspectorPanel
The state of the net inspector panel.
virtual PROJECT_LOCAL_SETTINGS & GetLocalSettings() const
Definition: project.h:209
PROJECT & Prj() const
A helper while we are not MDI-capable – return the one and only project.
wxString MessageTextFromValue(double aValue, bool aAddUnitLabel=true, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
A lower-precision version of StringFromValue().
EDA_UNITS GetUserUnits() const
wxBoxSizer * m_ContentSizer
A KICAD version of wxTextEntryDialog which supports the various improvements/work-arounds from DIALOG...
wxString GetValue() const
void SetTextValidator(wxTextValidatorStyle style)
bool IsOK(wxWindow *aParent, const wxString &aMessage)
Display a yes/no dialog with aMessage and returns the user response.
Definition: confirm.cpp:249
void DisplayError(wxWindow *aParent, const wxString &aText)
Display an error or warning message box with aMessage.
Definition: confirm.cpp:170
This file is part of the common library.
#define _(s)
#define CANDIDATE
flag indicating that the structure is connected
Abstract pattern-matching tool and implementations.
@ CTX_NET
int m_NetInspectorBulkUpdateOptimisationThreshold
When updating the net inspector, it either recalculates all nets or iterates through items one-by-one...
bool IsCopperLayer(int aLayerId)
Test whether a layer is a copper layer.
Definition: layer_ids.h:663
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ UNDEFINED_LAYER
Definition: layer_ids.h:61
KICOMMON_API wxString GetLabel(EDA_UNITS aUnits, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
Get the units string for a given units type.
Definition: eda_units.cpp:156
@ REPAINT
Item needs to be redrawn.
Definition: view_item.h:58
@ GEOMETRY
Position or shape has changed.
Definition: view_item.h:55
KICOMMON_API wxFont GetInfoFont(wxWindow *aWindow)
Definition: ui_common.cpp:156
PGM_BASE & Pgm()
The global program "get" accessor.
Definition: pgm_base.cpp:1071
see class PGM_BASE
static bool highlightNet(TOOL_MANAGER *aToolMgr, const VECTOR2D &aPosition)
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_NETNAME
Definition: string_utils.h:53
bool operator()(const CN_ITEM *a, const CN_ITEM *b) const
bool operator()(int a, const CN_ITEM *b) const
bool operator()(const CN_ITEM *a, int b) const
Persisted state for the net inspector panel.
std::vector< wxString > expanded_rows
std::vector< wxString > custom_group_rules
VECTOR2I end
thread_pool & GetKiCadThreadPool()
Get a reference to the current thread pool.
Definition: thread_pool.cpp:30
static thread_pool * tp
Definition: thread_pool.cpp:28
BS::thread_pool thread_pool
Definition: thread_pool.h:31
@ PCB_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
Definition: typeinfo.h:97
@ PCB_PAD_T
class PAD, a pad in a footprint
Definition: typeinfo.h:87
@ PCB_ARC_T
class PCB_ARC, an arc track segment on a copper layer
Definition: typeinfo.h:98
@ PCB_TRACE_T
class PCB_TRACK, a track segment (segment on a copper layer)
Definition: typeinfo.h:96
Custom text control validator definitions.
wxString AddFileExtListToFilter(const std::vector< std::string > &aExts)
Build the wildcard extension file dialog wildcard filter to add to the base message dialog.
Definition of file extensions used in Kicad.