KiCad PCB EDA Suite
Loading...
Searching...
No Matches
hierarchy_pane.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) 2004 Jean-Pierre Charras, jp.charras at wanadoo.fr
5 * Copyright (C) 2008 Wayne Stambaugh <[email protected]>
6 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version 2
11 * of the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program. If not, see <https://www.gnu.org/licenses/>.
20 */
21
22#include <bitmaps.h>
23#include <sch_edit_frame.h>
24#include <sch_commit.h>
25#include <connection_graph.h>
26#include <schematic.h>
27#include <gal/color4d.h>
28#include <layer_ids.h>
29#include <tool/tool_manager.h>
30#include <tools/sch_actions.h>
31#include <hierarchy_pane.h>
33#include <kiface_base.h>
34#include <wx/object.h>
35#include <wx/generic/textdlgg.h>
36#include <wx/menu.h>
37#include <wx/wupdlock.h>
38#include <wx/msgdlg.h>
39
43class TREE_ITEM_DATA : public wxTreeItemData
44{
45public:
47
48 TREE_ITEM_DATA( SCH_SHEET_PATH& sheet ) : wxTreeItemData(), m_SheetPath( sheet ) {}
49};
50
51
52// Need to use wxRTTI macros in order for OnCompareItems to work properly
53// See: https://docs.wxwidgets.org/3.1/classwx_tree_ctrl.html#ab90a465793c291ca7aa827a576b7d146
55
56
57int HIERARCHY_TREE::OnCompareItems( const wxTreeItemId& item1, const wxTreeItemId& item2 )
58{
59 SCH_SHEET_PATH* item1Path = &static_cast<TREE_ITEM_DATA*>( GetItemData( item1 ) )->m_SheetPath;
60 SCH_SHEET_PATH* item2Path = &static_cast<TREE_ITEM_DATA*>( GetItemData( item2 ) )->m_SheetPath;
61
62 return item1Path->ComparePageNum( *item2Path );
63}
64
65
67 WX_PANEL( aParent )
68{
69 wxASSERT( dynamic_cast<SCH_EDIT_FRAME*>( aParent ) );
70
71 m_frame = aParent;
72
73 wxBoxSizer* sizer = new wxBoxSizer( wxVERTICAL );
74 SetSizer( sizer );
75 m_tree = new HIERARCHY_TREE( this );
76
77 wxVector<wxBitmapBundle> images;
78 images.push_back( KiBitmapBundle( BITMAPS::tree_nosel ) );
79 images.push_back( KiBitmapBundle( BITMAPS::tree_sel ) );
80 m_tree->SetImages( images );
81
82 sizer->Add( m_tree, 1, wxEXPAND, wxBORDER_NONE );
83
84 m_events_bound = false;
85 m_contextMenuOpen = false;
86
87 PROJECT_LOCAL_SETTINGS& localSettings = m_frame->Prj().GetLocalSettings();
88
89 for( const wxString& path : localSettings.m_SchHierarchyCollapsed )
90 m_collapsedPaths.insert( path );
91
93
94 // Enable selection events
95 Bind( wxEVT_TREE_ITEM_ACTIVATED, &HIERARCHY_PANE::onSelectSheetPath, this );
96 Bind( wxEVT_TREE_SEL_CHANGED, &HIERARCHY_PANE::onSelectSheetPath, this );
97 Bind( wxEVT_TREE_ITEM_RIGHT_CLICK, &HIERARCHY_PANE::onTreeItemRightClick, this );
98 Bind( wxEVT_CHAR_HOOK, &HIERARCHY_PANE::onCharHook, this );
99 m_tree->Bind( wxEVT_TREE_END_LABEL_EDIT, &HIERARCHY_PANE::onTreeEditFinished, this );
100 m_tree->Bind( wxEVT_CONTEXT_MENU, &HIERARCHY_PANE::onContextMenu, this );
101 m_events_bound = true;
102}
103
104
106{
107 // Cancel any in-progress label edit before unbinding. Destroying the tree while
108 // a text control is open causes a focus-loss event that fires onTreeEditFinished
109 // after the schematic has been torn down.
110 if( m_tree->GetEditControl() )
111 m_tree->EndEditLabel( m_tree->GetSelection(), true );
112
113 Unbind( wxEVT_TREE_ITEM_ACTIVATED, &HIERARCHY_PANE::onSelectSheetPath, this );
114 Unbind( wxEVT_TREE_SEL_CHANGED, &HIERARCHY_PANE::onSelectSheetPath, this );
115 Unbind( wxEVT_TREE_ITEM_RIGHT_CLICK, &HIERARCHY_PANE::onTreeItemRightClick, this );
116 Unbind( wxEVT_CHAR_HOOK, &HIERARCHY_PANE::onCharHook, this );
117 m_tree->Unbind( wxEVT_TREE_END_LABEL_EDIT, &HIERARCHY_PANE::onTreeEditFinished, this );
118 m_tree->Unbind( wxEVT_CONTEXT_MENU, &HIERARCHY_PANE::onContextMenu, this );
119}
120
121
122void HIERARCHY_PANE::buildHierarchyTree( SCH_SHEET_PATH* aList, const wxTreeItemId& aParent )
123{
124 std::vector<SCH_ITEM*> sheetChildren;
125 aList->LastScreen()->GetSheets( &sheetChildren );
126
127 for( SCH_ITEM* aItem : sheetChildren )
128 {
129 SCH_SHEET* sheet = static_cast<SCH_SHEET*>( aItem );
130 aList->push_back( sheet );
131
132 wxString sheetNameBase = sheet->GetField( FIELD_T::SHEET_NAME )->GetShownText( false );
133
134 // If the sheet name is empty, use the filename (without extension) as fallback
135 if( sheetNameBase.IsEmpty() )
136 {
137 wxFileName fn( sheet->GetFileName() );
138 sheetNameBase = fn.GetName();
139 }
140
141 wxString sheetName = formatPageString( sheetNameBase, aList->GetPageNumber() );
142 wxTreeItemId child = m_tree->AppendItem( aParent, sheetName, 0, 1 );
143 m_tree->SetItemData( child, new TREE_ITEM_DATA( *aList ) );
144
145 buildHierarchyTree( aList, child );
146 aList->pop_back();
147 }
148
149 m_tree->SortChildren( aParent );
150}
151
152
154{
155 bool eventsWereBound = m_events_bound;
156
157 if( eventsWereBound )
158 {
159 // Disable selection events
160 Unbind( wxEVT_TREE_ITEM_ACTIVATED, &HIERARCHY_PANE::onSelectSheetPath, this );
161 Unbind( wxEVT_TREE_SEL_CHANGED, &HIERARCHY_PANE::onSelectSheetPath, this );
162 Unbind( wxEVT_TREE_ITEM_RIGHT_CLICK, &HIERARCHY_PANE::onTreeItemRightClick, this );
163 m_tree->Unbind( wxEVT_CONTEXT_MENU, &HIERARCHY_PANE::onContextMenu, this );
164
165 m_events_bound = false;
166 }
167
168 std::function<void( const wxTreeItemId& )> recursiveDescent =
169 [&]( const wxTreeItemId& id )
170 {
171 wxCHECK_RET( id.IsOk(), wxT( "Invalid tree item" ) );
172
173 TREE_ITEM_DATA* itemData = static_cast<TREE_ITEM_DATA*>( m_tree->GetItemData( id ) );
174
175 // Skip items without data (e.g., project root node)
176 if( !itemData )
177 {
178 wxTreeItemIdValue cookie;
179 wxTreeItemId child = m_tree->GetFirstChild( id, cookie );
180
181 while( child.IsOk() )
182 {
183 recursiveDescent( child );
184 child = m_tree->GetNextChild( id, cookie );
185 }
186
187 return;
188 }
189
190 if( itemData->m_SheetPath == m_frame->GetCurrentSheet() )
191 {
192 wxTreeItemId parent = m_tree->GetItemParent( id );
193
194 if( parent.IsOk() )
195 {
196 // AT least on MSW, wxTreeCtrl::IsExpanded(item) and wxTreeCtrl::Expand(item)
197 // can be called only if item is visible.
198 // Otherwise wxWidgets alerts are thrown and Expand() say the item is invisible
199 if( m_tree->IsVisible( parent ) && !m_tree->IsExpanded( parent ) )
200 m_tree->Expand( parent );
201 }
202
203 if( !m_tree->IsVisible( id ) )
204 m_tree->EnsureVisible( id );
205
206 m_tree->SetItemBold( id, true );
207 m_tree->SetFocusedItem( id );
208 }
209 else
210 {
211 m_tree->SetItemBold( id, false );
212 }
213
214 wxTreeItemIdValue cookie;
215 wxTreeItemId child = m_tree->GetFirstChild( id, cookie );
216
217 while( child.IsOk() )
218 {
219 recursiveDescent( child );
220 child = m_tree->GetNextChild( id, cookie );
221 }
222 };
223
224 recursiveDescent( m_tree->GetRootItem() );
225
226 if( eventsWereBound )
227 {
228 // Enable selection events
229 Bind( wxEVT_TREE_ITEM_ACTIVATED, &HIERARCHY_PANE::onSelectSheetPath, this );
230 Bind( wxEVT_TREE_SEL_CHANGED, &HIERARCHY_PANE::onSelectSheetPath, this );
231 Bind( wxEVT_TREE_ITEM_RIGHT_CLICK, &HIERARCHY_PANE::onTreeItemRightClick, this );
232 m_tree->Bind( wxEVT_CONTEXT_MENU, &HIERARCHY_PANE::onContextMenu, this );
233
234 m_events_bound = true;
235 }
236}
237
238
240{
241 wxWindowUpdateLocker updateLock( this );
242
243 // If hierarchy hasn't been built yet (e.g., during frame construction before schematic
244 // is loaded), just return. The tree will be updated later when the schematic is loaded.
245 if( !m_frame->Schematic().HasHierarchy() )
246 return;
247
248 bool eventsWereBound = m_events_bound;
249
250 if( eventsWereBound )
251 {
252 // Disable selection events
253 Unbind( wxEVT_TREE_ITEM_ACTIVATED, &HIERARCHY_PANE::onSelectSheetPath, this );
254 Unbind( wxEVT_TREE_SEL_CHANGED, &HIERARCHY_PANE::onSelectSheetPath, this );
255 Unbind( wxEVT_TREE_ITEM_RIGHT_CLICK, &HIERARCHY_PANE::onTreeItemRightClick, this );
256 m_tree->Unbind( wxEVT_CONTEXT_MENU, &HIERARCHY_PANE::onContextMenu, this );
257
258 m_events_bound = false;
259 }
260
261 SCH_SHEET_LIST hierarchy = m_frame->Schematic().Hierarchy();
262 std::set<wxString> collapsedNodes = m_collapsedPaths;
263
264 std::function<void( const wxTreeItemId& )> getCollapsedNodes =
265 [&]( const wxTreeItemId& id )
266 {
267 wxCHECK_RET( id.IsOk(), wxT( "Invalid tree item" ) );
268
269 TREE_ITEM_DATA* itemData = static_cast<TREE_ITEM_DATA*>( m_tree->GetItemData( id ) );
270
271 // Skip items without data (e.g., project root node)
272 if( itemData && m_tree->ItemHasChildren( id ) && !m_tree->IsExpanded( id )
273 && hierarchy.HasPath( itemData->m_SheetPath.Path() ) )
274 {
275 collapsedNodes.emplace( itemData->m_SheetPath.PathAsString() );
276 return;
277 }
278
279 wxTreeItemIdValue cookie;
280 wxTreeItemId child = m_tree->GetFirstChild( id, cookie );
281
282 while( child.IsOk() )
283 {
284 getCollapsedNodes( child );
285 child = m_tree->GetNextChild( id, cookie );
286 }
287 };
288
289 // If we are clearing the tree, don't try to get collapsed nodes as they
290 // might be deleted
291 if( !aClear && !m_tree->IsEmpty() )
292 {
293 collapsedNodes.clear();
294 getCollapsedNodes( m_tree->GetRootItem() );
295 }
296
297 m_tree->DeleteAllItems();
298
299 // Create project root node (not associated with virtual root sheet)
300 wxTreeItemId projectRoot = m_tree->AddRoot( getRootString(), 0, 1 );
301 // Don't set item data for the project root - it doesn't correspond to a real sheet
302
303 // Get all top-level sheets
304 std::vector<SCH_SHEET*> topLevelSheets = m_frame->Schematic().GetTopLevelSheets();
305
306 // For each top-level sheet, build its hierarchy under the project root
307 for( SCH_SHEET* sheet : topLevelSheets )
308 {
309 if( sheet )
310 {
311 m_list.clear();
312 m_list.push_back( sheet );
313
314 wxString sheetNameBase = sheet->GetShownName( false );
315
316 // If the sheet name is empty, use the filename (without extension) as fallback
317 if( sheetNameBase.IsEmpty() && sheet->GetScreen() )
318 {
319 wxFileName fn( sheet->GetScreen()->GetFileName() );
320 sheetNameBase = fn.GetName();
321 }
322
323 // Create tree item for this top-level sheet
324 wxString sheetName = formatPageString( sheetNameBase, m_list.GetPageNumber() );
325 wxTreeItemId topLevelItem = m_tree->AppendItem( projectRoot, sheetName, 0, 1 );
326 m_tree->SetItemData( topLevelItem, new TREE_ITEM_DATA( m_list ) );
327
328 // Build hierarchy for this top-level sheet
329 buildHierarchyTree( &m_list, topLevelItem );
330 }
331 }
332
334
335 m_tree->ExpandAll();
336
337 std::function<void( const wxTreeItemId& )> collapseNodes =
338 [&]( const wxTreeItemId& id )
339 {
340 wxCHECK_RET( id.IsOk(), wxT( "Invalid tree item" ) );
341
342 TREE_ITEM_DATA* itemData =
343 static_cast<TREE_ITEM_DATA*>( m_tree->GetItemData( id ) );
344
345 if( id != projectRoot && itemData &&
346 collapsedNodes.find( itemData->m_SheetPath.PathAsString() ) != collapsedNodes.end() )
347 {
348 m_tree->Collapse( id );
349 return;
350 }
351
352 wxTreeItemIdValue cookie;
353 wxTreeItemId child = m_tree->GetFirstChild( id, cookie );
354
355 while( child.IsOk() )
356 {
357 collapseNodes( child );
358 child = m_tree->GetNextChild( id, cookie );
359 }
360 };
361
362 collapseNodes( projectRoot );
363 m_collapsedPaths = std::move( collapsedNodes );
364
365 if( !m_highlightedNet.IsEmpty() )
367
368 if( eventsWereBound )
369 {
370 // Enable selection events
371 Bind( wxEVT_TREE_ITEM_ACTIVATED, &HIERARCHY_PANE::onSelectSheetPath, this );
372 Bind( wxEVT_TREE_SEL_CHANGED, &HIERARCHY_PANE::onSelectSheetPath, this );
373 Bind( wxEVT_TREE_ITEM_RIGHT_CLICK, &HIERARCHY_PANE::onTreeItemRightClick, this );
374 m_tree->Bind( wxEVT_CONTEXT_MENU, &HIERARCHY_PANE::onContextMenu, this );
375
376 m_events_bound = true;
377 }
378}
379
380
381void HIERARCHY_PANE::onSelectSheetPath( wxTreeEvent& aEvent )
382{
383 wxTreeItemId itemSel = m_tree->GetSelection();
384
385 if( !itemSel.IsOk() )
386 return;
387
388 TREE_ITEM_DATA* itemData = static_cast<TREE_ITEM_DATA*>( m_tree->GetItemData( itemSel ) );
389
390 if( !itemData )
391 return;
392
393 SetCursor( wxCURSOR_ARROWWAIT );
394 m_frame->GetToolManager()->RunAction<SCH_SHEET_PATH*>( SCH_ACTIONS::changeSheet, &itemData->m_SheetPath );
395 SetCursor( wxCURSOR_ARROW );
396}
397
398
400{
401 // Update the labels of the hierarchical tree of the schematic.
402 // Must be called only for an up to date tree, to update displayed labels after
403 // a sheet name or a sheet number change.
404
405 std::function<void( const wxTreeItemId& )> updateLabel =
406 [&]( const wxTreeItemId& id )
407 {
408 TREE_ITEM_DATA* itemData = static_cast<TREE_ITEM_DATA*>( m_tree->GetItemData( id ) );
409
410 if( !itemData ) // happens if not shown in wxTreeCtrl m_tree (virtual sheet)
411 return;
412
413 SCH_SHEET* sheet = itemData->m_SheetPath.Last();
414 wxString sheetNameBase = sheet->GetField( FIELD_T::SHEET_NAME )->GetShownText( false );
415 wxString sheetName = formatPageString( sheetNameBase,
416 itemData->m_SheetPath.GetPageNumber() );
417
418 if( m_tree->GetItemText( id ) != sheetName )
419 m_tree->SetItemText( id, sheetName );
420 };
421
422 wxTreeItemId rootId = m_tree->GetRootItem();
423 updateLabel( rootId );
424
425 std::function<void( const wxTreeItemId& )> recursiveDescent =
426 [&]( const wxTreeItemId& id )
427 {
428 wxCHECK_RET( id.IsOk(), wxT( "Invalid tree item" ) );
429 wxTreeItemIdValue cookie;
430 wxTreeItemId child = m_tree->GetFirstChild( id, cookie );
431
432 while( child.IsOk() )
433 {
434 updateLabel( child );
435 recursiveDescent( child );
436 child = m_tree->GetNextChild( id, cookie );
437 }
438 };
439
440 recursiveDescent( rootId );
441}
442
443
444std::vector<wxString> HIERARCHY_PANE::GetCollapsedPaths() const
445{
446 std::vector<wxString> collapsed;
447
448 if( m_tree->IsEmpty() )
449 return collapsed;
450
451 std::function<void( const wxTreeItemId& )> collect =
452 [&]( const wxTreeItemId& id )
453 {
454 wxCHECK_RET( id.IsOk(), wxT( "Invalid tree item" ) );
455
456 TREE_ITEM_DATA* itemData = static_cast<TREE_ITEM_DATA*>( m_tree->GetItemData( id ) );
457
458 if( id != m_tree->GetRootItem() && m_tree->ItemHasChildren( id )
459 && !m_tree->IsExpanded( id ) )
460 {
461 collapsed.push_back( itemData->m_SheetPath.PathAsString() );
462 return;
463 }
464
465 wxTreeItemIdValue cookie;
466 wxTreeItemId child = m_tree->GetFirstChild( id, cookie );
467
468 while( child.IsOk() )
469 {
470 collect( child );
471 child = m_tree->GetNextChild( id, cookie );
472 }
473 };
474
475 collect( m_tree->GetRootItem() );
476 return collapsed;
477}
478
479
480void HIERARCHY_PANE::onTreeItemRightClick( wxTreeEvent& aEvent )
481{
482 // wxEVT_CONTEXT_MENU fires after wxEVT_TREE_ITEM_RIGHT_CLICK for the same right-click,
483 // so set a guard to prevent showing the context menu twice.
484 m_contextMenuOpen = true;
485 onRightClick( aEvent.GetItem() );
486 m_contextMenuOpen = false;
487}
488
489
490void HIERARCHY_PANE::onContextMenu( wxContextMenuEvent& aEvent )
491{
492 // wxEVT_CONTEXT_MENU fires after wxEVT_TREE_ITEM_RIGHT_CLICK for the same right-click.
493 // Skip if the tree item handler already showed the menu.
495 return;
496
497 // Handle right-click in empty space
498 onRightClick( wxTreeItemId() );
499}
500
501
502void HIERARCHY_PANE::onRightClick( wxTreeItemId aItem )
503{
504 wxMenu ctxMenu;
505 TREE_ITEM_DATA* itemData = nullptr;
506 bool isProjectRoot = false;
507
508 if( !aItem.IsOk() )
509 aItem = m_tree->GetSelection();
510
511 if( aItem.IsOk() )
512 {
513 itemData = static_cast<TREE_ITEM_DATA*>( m_tree->GetItemData( aItem ) );
514 isProjectRoot = ( m_tree->GetRootItem() == aItem.GetID() );
515 }
516
517 if( itemData )
518 {
519 ctxMenu.Append( EDIT_PAGE_NUMBER, _( "Edit Page Number" ) );
520 // The root item cannot be renamed
521 if( !isProjectRoot )
522 {
523 ctxMenu.Append( RENAME, _( "Rename" ), _( "Change name of this sheet" ) );
524
525 // Allow deleting top-level sheets (but not sub-sheets)
526 if( itemData->m_SheetPath.size() == 1 )
527 {
528 ctxMenu.Append( DELETE_TOP_LEVEL_SHEET, _( "Delete Top-Level Sheet" ),
529 _( "Remove this top-level sheet from the project" ) );
530 }
531 }
532
533 ctxMenu.AppendSeparator();
534 }
535
536 // Always allow creating new top-level sheets (root is hidden with wxTR_HIDE_ROOT)
537 ctxMenu.Append( NEW_TOP_LEVEL_SHEET, _( "New Top-Level Sheet" ), _( "Create a new top-level sheet" ) );
538 ctxMenu.AppendSeparator();
539
540 ctxMenu.Append( EXPAND_ALL, ACTIONS::expandAll.GetMenuItem() );
541 ctxMenu.Append( COLLAPSE_ALL, ACTIONS::collapseAll.GetMenuItem() );
542
543
544 int selected = GetPopupMenuSelectionFromUser( ctxMenu );
545
546 switch( selected )
547 {
549 {
550 // Create a new top-level sheet
551 wxTextEntryDialog dlg( m_frame, _( "Enter name for new top-level sheet:" ),
552 _( "New Top-Level Sheet" ),
553 _( "Untitled" ) );
554
555 if( dlg.ShowModal() == wxID_OK )
556 {
557 wxString newName = dlg.GetValue();
558
559 if( !newName.IsEmpty() )
560 {
561 SCH_COMMIT commit( m_frame );
562
563 // Create new sheet and screen
564 SCH_SHEET* newSheet = new SCH_SHEET( &m_frame->Schematic() );
565 SCH_SCREEN* newScreen = new SCH_SCREEN( &m_frame->Schematic() );
566
567 newSheet->SetScreen( newScreen );
568 newSheet->GetField( FIELD_T::SHEET_NAME )->SetText( newName );
569
570 // Generate a unique filename
571 wxString filename = newName;
572 filename.Replace( " ", "_" );
573 filename = filename.Lower();
574
575 if( !filename.EndsWith( ".kicad_sch" ) )
576 filename += ".kicad_sch";
577
578 newScreen->SetFileName( filename );
579
580 // Find the lowest unused page number
581 SCH_SHEET_LIST hierarchy = m_frame->Schematic().Hierarchy();
582 int nextPage = 1;
583 wxString pageStr;
584
585 do
586 {
587 pageStr = wxString::Format( "%d", nextPage++ );
588 } while( hierarchy.PageNumberExists( pageStr ) );
589
590 m_frame->Schematic().AddTopLevelSheet( newSheet );
591
592 SCH_SHEET_PATH newSheetPath;
593 newSheetPath.push_back( newSheet );
594 newSheetPath.SetPageNumber( pageStr );
595
596 commit.Push( _( "Add new top-level sheet" ) );
597
598 // Refresh the hierarchy tree
600 }
601 }
602 break;
603 }
604
606 {
607 if( itemData && itemData->m_SheetPath.size() == 1 )
608 {
609 SCH_SHEET* sheet = itemData->m_SheetPath.Last();
610
611 // Confirm deletion
612 wxString msg = wxString::Format( _( "Delete top-level sheet '%s'?\n\nThis cannot be undone." ),
613 sheet->GetName() );
614
615 if( wxMessageBox( msg, _( "Delete Top-Level Sheet" ), wxYES_NO | wxICON_QUESTION, m_frame ) == wxYES )
616 {
617 // Don't allow deleting the last top-level sheet
618 if( m_frame->Schematic().GetTopLevelSheets().size() <= 1 )
619 {
620 wxMessageBox( _( "Cannot delete the last top-level sheet." ), _( "Delete Top-Level Sheet" ),
621 wxOK | wxICON_ERROR, m_frame );
622 break;
623 }
624
625 SCH_COMMIT commit( m_frame );
626
627 // Remove from schematic
628 if( m_frame->Schematic().RemoveTopLevelSheet( sheet ) )
629 {
630 commit.Push( _( "Delete top-level sheet" ) );
631
632 // Refresh the hierarchy tree
634 }
635 }
636 }
637 break;
638 }
639
640 case EDIT_PAGE_NUMBER:
641 {
642 wxString msg;
643 wxString sheetPath = itemData->m_SheetPath.PathHumanReadable( false, true );
644 wxString pageNumber = itemData->m_SheetPath.GetPageNumber();
645
646 msg.Printf( _( "Enter page number for sheet path %s" ),
647 ( sheetPath.Length() > 20 ) ? wxS( " \n" ) + sheetPath + wxT( ": " )
648 : sheetPath + wxT( ": " ) );
649
650 wxTextEntryDialog dlg( m_frame, msg, _( "Edit Sheet Page Number" ), pageNumber );
651
652 dlg.SetTextValidator( wxFILTER_ALPHANUMERIC ); // No white space.
653
654 if( dlg.ShowModal() == wxID_OK && dlg.GetValue() != itemData->m_SheetPath.GetPageNumber() )
655 {
656 SCH_COMMIT commit( m_frame );
657 SCH_SCREEN* modifyScreen = nullptr;
658
659 if( itemData->m_SheetPath.size() == 1 )
660 {
661 modifyScreen = m_frame->Schematic().Root().GetScreen();
662 }
663 else
664 {
665 SCH_SHEET_PATH parentPath = itemData->m_SheetPath;
666 parentPath.pop_back();
667 modifyScreen = parentPath.LastScreen();
668 }
669
670 commit.Modify( itemData->m_SheetPath.Last(), modifyScreen );
671
672 itemData->m_SheetPath.SetPageNumber( dlg.GetValue() );
673
674 if( itemData->m_SheetPath == m_frame->GetCurrentSheet() )
675 {
676 if( m_frame->GetScreen() )
677 {
678 m_frame->GetScreen()->SetPageNumber( dlg.GetValue() );
679 m_frame->OnPageSettingsChange();
680 }
681 }
682
683 commit.Push( wxS( "Change sheet page number." ) );
684
686 }
687
688 break;
689 }
690 case EXPAND_ALL:
691 m_tree->ExpandAll();
692 break;
693 case COLLAPSE_ALL:
694 m_tree->CollapseAll();
695 break;
696 case RENAME:
697 m_tree->SetItemText( aItem, itemData->m_SheetPath.Last()->GetName() );
698 m_tree->EditLabel( aItem );
700 break;
701 }
702}
703
704
705void HIERARCHY_PANE::onTreeEditFinished( wxTreeEvent& event )
706{
707 // The frame is shutting down — schematic and current sheet state are no longer safe to access.
708 if( m_frame->IsClosing() )
709 {
710 event.Veto();
711 return;
712 }
713
714 TREE_ITEM_DATA* data = static_cast<TREE_ITEM_DATA*>( m_tree->GetItemData( event.GetItem() ) );
715 wxString newName = event.GetLabel();
716
717 if( data && data->m_SheetPath.Last() )
718 {
719 if( !newName.IsEmpty() )
720 {
721 // The editor holds only the page name as a text, while normally
722 // the tree items displaying it suffixed with the page number
723 if( data->m_SheetPath.Last()->GetName() != newName )
724 {
725 SCH_COMMIT commit( m_frame );
726 SCH_SCREEN* modifyScreen = nullptr;
727
728 // For top-level sheets (size == 1), modify on the virtual root's screen
729 // For sub-sheets, modify on the parent sheet's screen
730 if( data->m_SheetPath.size() == 1 )
731 {
732 modifyScreen = m_frame->Schematic().Root().GetScreen();
733 }
734 else
735 {
736 const SCH_SHEET* parentSheet = data->m_SheetPath.GetSheet( data->m_SheetPath.size() - 2 );
737 if( parentSheet )
738 modifyScreen = parentSheet->GetScreen();
739 }
740
741 if( modifyScreen )
742 {
744 modifyScreen );
745
746 data->m_SheetPath.Last()->SetName( newName );
747
748 renameIdenticalSheets( data->m_SheetPath, newName, &commit );
749
750 if( !commit.Empty() )
751 commit.Push( _( "Renaming sheet" ) );
752
753 if( data->m_SheetPath == m_frame->GetCurrentSheet() )
754 {
755 m_frame->OnPageSettingsChange();
756 }
757 }
758 }
759 }
760
761 m_tree->SetItemText( event.GetItem(), formatPageString( data->m_SheetPath.Last()->GetName(),
762 data->m_SheetPath.GetPageNumber() ) );
764 // The event needs to be rejected otherwise the SetItemText call above
765 // will be ineffective (the treeview item will hold the editor's content)
766 event.Veto();
767 }
768}
769
770
771void HIERARCHY_PANE::onCharHook( wxKeyEvent& aKeyStroke )
772{
773 int hotkey = aKeyStroke.GetKeyCode();
774
775 int mods = aKeyStroke.GetModifiers();
776
777 // the flag wxMOD_ALTGR is defined in wxWidgets as wxMOD_CONTROL|wxMOD_ALT
778 // So AltGr key cannot used as modifier key because it is the same as Alt key + Ctrl key.
779#if CAN_USE_ALTGR_KEY
780 if( wxmods & wxMOD_ALTGR )
781 mods |= MD_ALTGR;
782 else
783#endif
784 {
785 if( mods & wxMOD_CONTROL )
786 hotkey += MD_CTRL;
787
788 if( mods & wxMOD_ALT )
789 hotkey += MD_ALT;
790 }
791
792 if( mods & wxMOD_SHIFT )
793 hotkey += MD_SHIFT;
794
795#ifdef wxMOD_META
796 if( mods & wxMOD_META )
797 hotkey += MD_META;
798#endif
799
800#ifdef wxMOD_WIN
801 if( mods & wxMOD_WIN )
802 hotkey += MD_SUPER;
803#endif
804
805 if( hotkey == ACTIONS::expandAll.GetHotKey()
806 || hotkey == ACTIONS::expandAll.GetHotKeyAlt() )
807 {
808 m_tree->ExpandAll();
809 return;
810 }
811 else if( hotkey == ACTIONS::collapseAll.GetHotKey()
812 || hotkey == ACTIONS::collapseAll.GetHotKeyAlt() )
813 {
814 m_tree->CollapseAll();
815 return;
816 }
817 else
818 {
819 aKeyStroke.Skip();
820 }
821}
822
823
825{
826 // Pane may be repainting while schematic is in flux
827 if ( !m_frame->Schematic().IsValid() )
828 return _( "Schematic" );
829
830 // Return the project name for the root node
831 wxString projectName = m_frame->Schematic().Project().GetProjectName();
832
833 if( projectName.IsEmpty() )
834 projectName = _( "Schematic" );
835
836 return projectName;
837}
838
839
840wxString HIERARCHY_PANE::formatPageString( const wxString& aName, const wxString& aPage )
841{
842 return aName + wxT( " " ) + wxString::Format( _( "(page %s)" ), aPage );
843}
844
845
846void HIERARCHY_PANE::UpdateNetHighlight( const wxString& aNetName )
847{
848 m_highlightedNet = aNetName;
849
850 KIGFX::COLOR4D netColor = m_frame->GetRenderSettings()->GetLayerColor( LAYER_BRIGHTENED );
851 const wxColour markText = netColor.ToColour();
852
853 std::set<wxString> sheetsWithNet;
854
855 if( !aNetName.IsEmpty() && m_frame->Schematic().IsValid() )
856 {
857 CONNECTION_GRAPH* graph = m_frame->Schematic().ConnectionGraph();
858
859 if( graph )
860 {
861 for( const CONNECTION_SUBGRAPH* sg : graph->GetAllSubgraphs( aNetName ) )
862 {
863 if( sg && sg->GetSheet().Last() )
864 sheetsWithNet.insert( sg->GetSheet().PathAsString() );
865 }
866 }
867 }
868
869 std::function<void( const wxTreeItemId& )> recurse = [&]( const wxTreeItemId& id )
870 {
871 wxCHECK_RET( id.IsOk(), wxT( "Invalid tree item" ) );
872
873 TREE_ITEM_DATA* data = static_cast<TREE_ITEM_DATA*>( m_tree->GetItemData( id ) );
874
875 if( data )
876 {
877 bool mark = sheetsWithNet.count( data->m_SheetPath.PathAsString() ) > 0;
878 m_tree->SetItemTextColour( id, mark ? markText : wxNullColour );
879 }
880
881 wxTreeItemIdValue cookie;
882 wxTreeItemId child = m_tree->GetFirstChild( id, cookie );
883
884 while( child.IsOk() )
885 {
886 recurse( child );
887 child = m_tree->GetNextChild( id, cookie );
888 }
889 };
890
891 if( m_tree->GetRootItem().IsOk() )
892 recurse( m_tree->GetRootItem() );
893}
894
896{
897 std::function<void( const wxTreeItemId& )> recursiveDescent = [&]( const wxTreeItemId& id )
898 {
899 wxCHECK_RET( id.IsOk(), wxT( "Invalid tree item" ) );
900
901 TREE_ITEM_DATA* itemData = static_cast<TREE_ITEM_DATA*>( m_tree->GetItemData( id ) );
902
903 // Skip items without data (e.g., project root node)
904 if( itemData && itemData->m_SheetPath.Cmp( path ) != 0 && itemData->m_SheetPath.Last() == path.Last() )
905 {
906 wxFont font = m_tree->GetItemFont( id );
907 font.SetUnderlined( highLighted );
908 m_tree->SetItemFont( id, font );
909 }
910
911 wxTreeItemIdValue cookie;
912 wxTreeItemId child = m_tree->GetFirstChild( id, cookie );
913
914 while( child.IsOk() )
915 {
916 recursiveDescent( child );
917 child = m_tree->GetNextChild( id, cookie );
918 }
919 };
920
921 recursiveDescent( m_tree->GetRootItem() );
922}
923
925 const wxString newName, SCH_COMMIT* commit )
926{
927 std::function<void( const wxTreeItemId& )> recursiveDescent = [&]( const wxTreeItemId& id )
928 {
929 wxCHECK_RET( id.IsOk(), wxT( "Invalid tree item" ) );
930
931 TREE_ITEM_DATA* data = static_cast<TREE_ITEM_DATA*>( m_tree->GetItemData( id ) );
932
933 // Skip items without data (e.g., project root node)
934 if( !data )
935 {
936 wxTreeItemIdValue cookie;
937 wxTreeItemId child = m_tree->GetFirstChild( id, cookie );
938
939 while( child.IsOk() )
940 {
941 recursiveDescent( child );
942 child = m_tree->GetNextChild( id, cookie );
943 }
944
945 return;
946 }
947
948 // Check if this is an identical sheet that needs renaming (but not the renamed sheet itself)
949 if( data->m_SheetPath.Cmp( renamedSheet ) != 0
950 && data->m_SheetPath.Last() == renamedSheet.Last() )
951 {
952 SCH_SCREEN* modifyScreen = nullptr;
953
954 // For top-level sheets (size == 1), modify on the virtual root's screen
955 // For sub-sheets, modify on the parent sheet's screen
956 if( data->m_SheetPath.size() == 1 )
957 {
958 modifyScreen = m_frame->Schematic().Root().GetScreen();
959 }
960 else
961 {
962 const SCH_SHEET* parentSheet = data->m_SheetPath.GetSheet( data->m_SheetPath.size() - 2 );
963 if( parentSheet )
964 modifyScreen = parentSheet->GetScreen();
965 }
966
967 if( modifyScreen )
968 {
970 modifyScreen );
971
972 data->m_SheetPath.Last()->SetName( newName );
973
974 if( data->m_SheetPath == m_frame->GetCurrentSheet() )
975 {
976 m_frame->OnPageSettingsChange();
977 }
978
979 m_tree->SetItemText( id, formatPageString( data->m_SheetPath.Last()->GetName(),
980 data->m_SheetPath.GetPageNumber() ) );
981 }
982 }
983
984 wxTreeItemIdValue cookie;
985 wxTreeItemId child = m_tree->GetFirstChild( id, cookie );
986
987 while( child.IsOk() )
988 {
989 recursiveDescent( child );
990 child = m_tree->GetNextChild( id, cookie );
991 }
992 };
993
994 recursiveDescent( m_tree->GetRootItem() );
995}
wxBitmapBundle KiBitmapBundle(BITMAPS aBitmap, int aMinHeight)
Definition bitmap.cpp:106
static TOOL_ACTION expandAll
Definition actions.h:86
static TOOL_ACTION collapseAll
Definition actions.h:87
bool Empty() const
Definition commit.h:134
COMMIT & Modify(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr, RECURSE_MODE aRecurse=RECURSE_MODE::NO_RECURSE)
Modify a given item in the model.
Definition commit.h:102
Calculate the connectivity of a schematic and generates netlists.
const std::vector< CONNECTION_SUBGRAPH * > & GetAllSubgraphs(const wxString &aNetName) const
A subgraph is a set of items that are electrically connected on a single sheet.
SCH_EDIT_FRAME * m_frame
void onRightClick(wxTreeItemId aItem)
HIERARCHY_TREE * m_tree
void UpdateHierarchyTree(bool aClear=false)
Update the hierarchical tree of the schematic.
void onSelectSheetPath(wxTreeEvent &aEvent)
Open the selected sheet and display the corresponding screen when a tree item is selected.
void UpdateLabelsHierarchyTree()
Update the labels of the hierarchical tree of the schematic.
wxString formatPageString(const wxString &aName, const wxString &aPage)
wxString getRootString()
SCH_SHEET_PATH m_list
void UpdateHierarchySelection()
Updates the tree's selection to match current page.
wxString m_highlightedNet
HIERARCHY_PANE(SCH_EDIT_FRAME *aParent)
std::set< wxString > m_collapsedPaths
void onCharHook(wxKeyEvent &aKeyStroke)
void renameIdenticalSheets(const SCH_SHEET_PATH &renamedSheet, const wxString newName, SCH_COMMIT *commit)
Rename all sheets in a hierarchial desing which has the same source renamed sheet.
void buildHierarchyTree(SCH_SHEET_PATH *aList, const wxTreeItemId &aParent)
Create the hierarchical tree of the schematic.
void onContextMenu(wxContextMenuEvent &aEvent)
void onTreeItemRightClick(wxTreeEvent &aEvent)
std::vector< wxString > GetCollapsedPaths() const
Returns a list of sheet paths for nodes that are currently collapsed.
void onTreeEditFinished(wxTreeEvent &event)
void UpdateNetHighlight(const wxString &aNetName)
void setIdenticalSheetsHighlighted(const SCH_SHEET_PATH &path, bool highLighted=true)
When renaming the sheets in tree it is helpful to highlight the identical sheets which got renamed by...
Navigation hierarchy tree control.
int OnCompareItems(const wxTreeItemId &item1, const wxTreeItemId &item2) override
A color representation with 4 components: red, green, blue, alpha.
Definition color4d.h:101
wxColour ToColour() const
Definition color4d.cpp:221
The project local settings are things that are attached to a particular project, but also might be pa...
std::vector< wxString > m_SchHierarchyCollapsed
Collapsed nodes in the schematic hierarchy navigator.
static TOOL_ACTION changeSheet
virtual void Push(const wxString &aMessage=wxT("A commit"), int aCommitFlags=0) override
Execute the changes.
Schematic editor (Eeschema) main window.
wxString GetShownText(const SCH_SHEET_PATH *aPath, bool aAllowExtraText, int aDepth=0, const wxString &aVariantName=wxEmptyString) const
void SetText(const wxString &aText) override
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition sch_item.h:162
void SetFileName(const wxString &aFileName)
Set the file name for this screen to aFileName.
void GetSheets(std::vector< SCH_ITEM * > *aItems) const
Similar to Items().OfType( SCH_SHEET_T ), but return the sheets in a deterministic order (L-R,...
A container for handling SCH_SHEET_PATH objects in a flattened hierarchy.
bool PageNumberExists(const wxString &aPageNumber) const
bool HasPath(const KIID_PATH &aPath) const
Handle access to a stack of flattened SCH_SHEET objects by way of a path for creating a flattened sch...
const SCH_SHEET * GetSheet(unsigned aIndex) const
int ComparePageNum(const SCH_SHEET_PATH &aSheetPathToTest) const
Compare sheets by their page number.
KIID_PATH Path() const
Get the sheet path as an KIID_PATH.
SCH_SCREEN * LastScreen()
int Cmp(const SCH_SHEET_PATH &aSheetPathToTest) const
Compare if this is the same sheet path as aSheetPathToTest.
wxString GetPageNumber() const
wxString PathHumanReadable(bool aUseShortRootName=true, bool aStripTrailingSeparator=false, bool aEscapeSheetNames=false) const
Return the sheet path in a human readable form made from the sheet names.
wxString PathAsString() const
Return the path of time stamps which do not changes even when editing sheet parameters.
void SetPageNumber(const wxString &aPageNumber)
Set the sheet instance user definable page number.
SCH_SHEET * Last() const
Return a pointer to the last SCH_SHEET of the list.
void push_back(SCH_SHEET *aSheet)
Forwarded method from std::vector.
size_t size() const
Forwarded method from std::vector.
void pop_back()
Forwarded method from std::vector.
Sheet symbol placed in a schematic, and is the entry point for a sub schematic.
Definition sch_sheet.h:44
wxString GetFileName() const
Return the filename corresponding to this sheet.
Definition sch_sheet.h:370
SCH_FIELD * GetField(FIELD_T aFieldType)
Return a mandatory field in this sheet.
wxString GetName() const
Definition sch_sheet.h:136
void SetName(const wxString &aName)
Definition sch_sheet.h:137
SCH_SCREEN * GetScreen() const
Definition sch_sheet.h:139
void SetScreen(SCH_SCREEN *aScreen)
Set the SCH_SCREEN associated with this sheet to aScreen.
Store an SCH_SHEET_PATH of each sheet in hierarchy.
SCH_SHEET_PATH m_SheetPath
TREE_ITEM_DATA(SCH_SHEET_PATH &sheet)
WX_PANEL(wxWindow *parent, wxWindowID id=wxID_ANY, const wxPoint &pos=wxDefaultPosition, const wxSize &size=wxSize(-1,-1), long style=wxTAB_TRAVERSAL, const wxString &name=wxEmptyString)
Definition wx_panel.cpp:24
static void recursiveDescent(wxSizer *aSizer, std::map< int, wxString > &aLabels)
#define _(s)
wxIMPLEMENT_ABSTRACT_CLASS(HIERARCHY_TREE, wxTreeCtrl)
@ LAYER_BRIGHTENED
Definition layer_ids.h:489
std::string path
@ MD_META
Definition tool_event.h:143
@ MD_ALT
Definition tool_event.h:141
@ MD_CTRL
Definition tool_event.h:140
@ MD_SUPER
Definition tool_event.h:142
@ MD_ALTGR
Definition tool_event.h:144
@ MD_SHIFT
Definition tool_event.h:139