KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
dialog_label_properties.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) 2013 Jean-Pierre Charras, jp.charras at wanadoo.fr
5 * Copyright (C) 2013 Wayne Stambaugh <stambaughw@gmail.com>
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, you may find one here:
20 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
21 * or you may search the http://www.gnu.org website for the version 2 license,
22 * or you may write to the Free Software Foundation, Inc.,
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
24 */
25
27#include <widgets/font_choice.h>
31#include <sch_edit_frame.h>
32#include <tool/tool_manager.h>
33#include <gr_text.h>
34#include <confirm.h>
35#include <schematic.h>
38#include <string_utils.h>
39#include <kiface_base.h>
40#include <sch_label.h>
41#include <sch_commit.h>
42
43
45 SCH_LABEL_BASE* aLabel,
46 bool aNew ) :
48 m_Parent( aParent ),
49 m_currentLabel( aLabel ),
50 m_activeTextEntry( nullptr ),
51 m_netNameValidator( true ),
52 m_fields( nullptr ),
53 m_textSize( aParent, m_textSizeLabel, m_textSizeCtrl, m_textSizeUnits, false ),
54 m_helpWindow( nullptr )
55{
56 COLOR_SETTINGS* colorSettings = m_Parent->GetColorSettings();
57 COLOR4D schematicBackground = colorSettings->GetColor( LAYER_SCHEMATIC_BACKGROUND );
58 bool multiLine = false;
59
60 if( EESCHEMA_SETTINGS* cfg = dynamic_cast<EESCHEMA_SETTINGS*>( Kiface().KifaceSettings() ) )
61 multiLine = cfg->m_Appearance.edit_label_multiple;
62
63 m_cbMultiLine->SetValue( multiLine );
64
65 m_fields = new FIELDS_GRID_TABLE( this, aParent, m_grid, m_currentLabel );
66 m_width = 100; // Will be later set to a better value
69
71 {
74
75 m_labelSingleLine->Show( false );
76 m_valueSingleLine->Show( false );
77
78 if( multiLine && aNew )
79 {
82 m_labelCombo->Show( false );
83 m_valueCombo->Show( false );
84 }
85 else
86 {
87 m_labelMultiLine->Show( false );
88 m_valueMultiLine->Show( false );
89 m_valueCombo->SetValidator( m_netNameValidator );
90 }
91 }
92 else if( m_currentLabel->Type() == SCH_HIER_LABEL_T )
93 {
96
97 m_labelCombo->Show( false );
98 m_valueCombo->Show( false );
99
100 if( multiLine && aNew )
101 {
104 m_labelSingleLine->Show( false );
105 m_valueSingleLine->Show( false );
106 }
107 else
108 {
109 m_labelMultiLine->Show( false );
110 m_valueMultiLine->Show( false );
111 m_valueSingleLine->SetValidator( m_netNameValidator );
112 }
113 }
115 {
118
119 m_labelSingleLine->Show( false );
120 m_valueSingleLine->Show( false );
121 m_labelCombo->Show( false );
122 m_valueCombo->Show( false );
123 m_syntaxHelp->Show( false );
124 m_textEntrySizer->Show( false );
125 m_labelCombo->Show( false );
126 m_valueCombo->Show( false );
127 m_cbMultiLine->Show( false );
128
129 m_textSizeLabel->SetLabel( _( "Pin length:" ) );
130 }
131
132 if( !aNew )
133 m_cbMultiLine->Show( false );
134
135 switch( m_currentLabel->Type() )
136 {
137 case SCH_GLOBAL_LABEL_T: SetTitle( _( "Global Label Properties" ) ); break;
138 case SCH_HIER_LABEL_T: SetTitle( _( "Hierarchical Label Properties" ) ); break;
139 case SCH_LABEL_T: SetTitle( _( "Label Properties" ) ); break;
140 case SCH_DIRECTIVE_LABEL_T: SetTitle( _( "Directive Label Properties" ) ); break;
141 case SCH_SHEET_PIN_T: SetTitle( _( "Hierarchical Sheet Pin Properties" ) ); break;
142 default: UNIMPLEMENTED_FOR( m_currentLabel->GetClass() ); break;
143 }
144
145 // Give a bit more room for combobox editors
146 m_grid->SetDefaultRowSize( m_grid->GetDefaultRowSize() + 4 );
147
149 m_grid->PushEventHandler( new FIELDS_GRID_TRICKS( m_grid, this, nullptr,
150 [&]( wxCommandEvent& aEvent )
151 {
152 OnAddField( aEvent );
153 } ) );
154 m_grid->SetSelectionMode( wxGrid::wxGridSelectRows );
155
156 // Show/hide columns according to user's preference
157 if( EESCHEMA_SETTINGS* cfg = dynamic_cast<EESCHEMA_SETTINGS*>( Kiface().KifaceSettings() ) )
158 {
159 m_grid->ShowHideColumns( cfg->m_Appearance.edit_label_visible_columns );
161 }
162
163 // Configure button logos
164 m_bpAdd->SetBitmap( KiBitmapBundle( BITMAPS::small_plus ) );
165 m_bpDelete->SetBitmap( KiBitmapBundle( BITMAPS::small_trash ) );
166 m_bpMoveUp->SetBitmap( KiBitmapBundle( BITMAPS::small_up ) );
167 m_bpMoveDown->SetBitmap( KiBitmapBundle( BITMAPS::small_down ) );
168
170
172 m_bold->SetBitmap( KiBitmapBundle( BITMAPS::text_bold ) );
174 m_italic->SetBitmap( KiBitmapBundle( BITMAPS::text_italic ) );
175
177
182
184
185 m_textColorSwatch->SetDefaultColor( COLOR4D::UNSPECIFIED );
186 m_textColorSwatch->SetSwatchBackground( schematicBackground );
187
188 // Show/hide relevant controls
190 {
191 m_dot->Hide();
192 m_circle->Hide();
193 m_diamond->Hide();
194 m_rectangle->Hide();
195
196 m_spin0->SetBitmap( KiBitmapBundle( BITMAPS::label_align_left ) );
197 m_spin1->SetBitmap( KiBitmapBundle( BITMAPS::label_align_right ) );
198 m_spin2->SetBitmap( KiBitmapBundle( BITMAPS::label_align_bottom ) );
199 m_spin3->SetBitmap( KiBitmapBundle( BITMAPS::label_align_top ) );
200 }
202 {
203 m_input->Hide();
204 m_output->Hide();
205 m_bidirectional->Hide();
206 m_triState->Hide();
207 m_passive->Hide();
208
209 m_fontLabel->SetLabel( _( "Orientation:" ) );
210 m_fontCtrl->Hide();
211 m_separator1->Hide();
212 m_bold->Hide();
213 m_italic->Hide();
214 m_separator2->Hide();
215 m_spin0->SetBitmap( KiBitmapBundle( BITMAPS::pinorient_down ) );
216 m_spin1->SetBitmap( KiBitmapBundle( BITMAPS::pinorient_up ) );
217 m_spin2->SetBitmap( KiBitmapBundle( BITMAPS::pinorient_right ) );
218 m_spin3->SetBitmap( KiBitmapBundle( BITMAPS::pinorient_left ) );
219 m_separator3->Hide();
220
221 m_formattingGB->Detach( m_fontCtrl );
222 m_formattingGB->Detach( m_iconBar );
223 m_formattingGB->Add( m_iconBar, wxGBPosition( 0, 1 ), wxGBSpan( 1, 1 ),
224 wxEXPAND | wxRIGHT, 5 );
225 }
226 else
227 {
228 m_shapeSizer->Show( false );
229
230 m_spin0->SetBitmap( KiBitmapBundle( BITMAPS::text_align_left ) );
231 m_spin1->SetBitmap( KiBitmapBundle( BITMAPS::text_align_right ) );
232 m_spin2->SetBitmap( KiBitmapBundle( BITMAPS::text_align_bottom ) );
233 m_spin3->SetBitmap( KiBitmapBundle( BITMAPS::text_align_top ) );
234 }
235
237 {
238 m_autoRotate->Hide();
239 wxSizer* parentSizer = m_autoRotate->GetContainingSizer();
240 parentSizer->Detach( m_autoRotate );
241 parentSizer->Layout();
242 }
243
245
246 // DIALOG_SHIM needs a unique hash_key because classname is not sufficient because the
247 // various versions have different controls so we want to store sizes for each version.
248 m_hash_key = TO_UTF8( GetTitle() );
249
250 m_spin0->Bind( wxEVT_BUTTON, &DIALOG_LABEL_PROPERTIES::onSpinButton, this );
251 m_spin1->Bind( wxEVT_BUTTON, &DIALOG_LABEL_PROPERTIES::onSpinButton, this );
252 m_spin2->Bind( wxEVT_BUTTON, &DIALOG_LABEL_PROPERTIES::onSpinButton, this );
253 m_spin3->Bind( wxEVT_BUTTON, &DIALOG_LABEL_PROPERTIES::onSpinButton, this );
254
255 // Now all widgets have the size fixed, call FinishDialogSettings
257
258 if( EESCHEMA_SETTINGS* cfg = dynamic_cast<EESCHEMA_SETTINGS*>( Kiface().KifaceSettings() ) )
259 {
260 if( cfg->m_Appearance.edit_label_width > 0 && cfg->m_Appearance.edit_label_height > 0 )
261 SetSize( cfg->m_Appearance.edit_label_width, cfg->m_Appearance.edit_label_height );
262 }
263
264 // Bind the event to handle filtering
265 m_valueCombo->Bind( wxEVT_TEXT, &DIALOG_LABEL_PROPERTIES::OnLabelFilter, this );
266 m_valueCombo->Bind( wxEVT_COMBOBOX, &DIALOG_LABEL_PROPERTIES::OnLabelItemSelected, this );
267}
268
269
271{
272 if( EESCHEMA_SETTINGS* cfg = dynamic_cast<EESCHEMA_SETTINGS*>( Kiface().KifaceSettings() ) )
273 {
274 cfg->m_Appearance.edit_label_visible_columns = m_grid->GetShownColumnsAsString();
275 cfg->m_Appearance.edit_label_width = GetSize().x;
276 cfg->m_Appearance.edit_label_height = GetSize().y;
277 cfg->m_Appearance.edit_label_multiple = m_cbMultiLine->IsChecked();
278 }
279
280 // Prevents crash bug in wxGrid's d'tor
282
283 // Delete the GRID_TRICKS.
284 m_grid->PopEventHandler( true );
285
286 if( m_helpWindow )
287 m_helpWindow->Destroy();
288
289 m_valueCombo->Unbind( wxEVT_TEXT, &DIALOG_LABEL_PROPERTIES::OnLabelFilter, this );
290 m_valueCombo->Unbind( wxEVT_COMBOBOX, &DIALOG_LABEL_PROPERTIES::OnLabelItemSelected, this );
291}
292
293
295{
296 if( !wxDialog::TransferDataToWindow() )
297 return false;
298
300 {
301 // show control characters in a human-readable format
302 wxString text = UnescapeString( m_currentLabel->GetText() );
303
304 // show text variable cross-references in a human-readable format
306
307 m_activeTextEntry->SetValue( text );
308 }
309
311 {
312 // Load the combobox with the existing labels of the same type
313 std::set<wxString> existingLabels;
314 std::vector<std::shared_ptr<BUS_ALIAS>> busAliases;
315 SCH_SCREENS allScreens( m_Parent->Schematic().Root() );
316
317 for( SCH_SCREEN* screen = allScreens.GetFirst(); screen; screen = allScreens.GetNext() )
318 {
319 for( SCH_ITEM* item : screen->Items().OfType( m_currentLabel->Type() ) )
320 {
321 const SCH_LABEL_BASE* label = static_cast<const SCH_LABEL_BASE*>( item );
322 existingLabels.insert( UnescapeString( label->GetText() ) );
323 }
324
325 // Add global power labels from power symbols
327 {
328 for( SCH_ITEM* item : screen->Items().OfType( SCH_SYMBOL_LOCATE_POWER_T ) )
329 {
330 const SCH_SYMBOL* power = static_cast<const SCH_SYMBOL*>( item );
331
332 // Ensure the symbol has the Power (i.e. equivalent to a global label
333 // before adding its value in list
334 if( power->IsSymbolLikePowerGlobalLabel() )
335 {
336 const SCH_FIELD* valueField = power->GetField( FIELD_T::VALUE );
337 existingLabels.insert( UnescapeString( valueField->GetText() ) );
338 }
339 }
340 }
341
342 // Add local power labels from power symbols
344 {
345 for( SCH_ITEM* item : screen->Items().OfType( SCH_SYMBOL_LOCATE_POWER_T ) )
346 {
347 const SCH_SYMBOL* power = static_cast<const SCH_SYMBOL*>( item );
348
349 // Ensure the symbol has the Power (i.e. equivalent to a local label
350 // before adding its value in list
351 if( power->IsSymbolLikePowerLocalLabel() )
352 {
353 const SCH_FIELD* valueField = power->GetField( FIELD_T::VALUE );
354 existingLabels.insert( UnescapeString( valueField->GetText() ) );
355 }
356 }
357 }
358 // Add bus aliases from the current screen
359
360 auto& sheetAliases = screen->GetBusAliases();
361 busAliases.insert( busAliases.end(), sheetAliases.begin(), sheetAliases.end() );
362 }
363
364 // Add bus aliases to label list
365 for( const std::shared_ptr<BUS_ALIAS>& busAlias : busAliases )
366 existingLabels.insert( wxT( "{" ) + busAlias->GetName() + wxT( "}" ) );
367
368 for( const wxString& label : existingLabels )
369 m_existingLabelArray.push_back( label );
370
372 }
373
374 // Push a copy of each field into m_updateFields
375 for( SCH_FIELD& field : m_currentLabel->GetFields() )
376 {
377 SCH_FIELD field_copy( field );
378
379 // change offset to be symbol-relative
380 field_copy.Offset( -m_currentLabel->GetPosition() );
381
382 m_fields->push_back( field_copy );
383 }
384
385 // notify the grid
386 wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, (int) m_fields->size() );
387 m_grid->ProcessTableMessage( msg );
388 AdjustGridColumns( m_grid->GetRect().GetWidth() );
389
390 if( m_shapeSizer->AreAnyItemsShown() )
391 {
392 switch( m_currentLabel->GetShape() )
393 {
394 case LABEL_FLAG_SHAPE::L_INPUT: m_input->SetValue( true ); break;
395 case LABEL_FLAG_SHAPE::L_OUTPUT: m_output->SetValue( true ); break;
396 case LABEL_FLAG_SHAPE::L_BIDI: m_bidirectional->SetValue( true ); break;
397 case LABEL_FLAG_SHAPE::L_TRISTATE: m_triState->SetValue( true ); break;
398 case LABEL_FLAG_SHAPE::L_UNSPECIFIED: m_passive->SetValue( true ); break;
399 case LABEL_FLAG_SHAPE::F_DOT: m_dot->SetValue( true ); break;
400 case LABEL_FLAG_SHAPE::F_ROUND: m_circle->SetValue( true ); break;
401 case LABEL_FLAG_SHAPE::F_DIAMOND: m_diamond->SetValue( true ); break;
402 case LABEL_FLAG_SHAPE::F_RECTANGLE: m_rectangle->SetValue( true ); break;
403 }
404 }
405
407
409 m_textSize.SetValue( static_cast<SCH_DIRECTIVE_LABEL*>( m_currentLabel )->GetPinLength() );
410 else
412
416
417 switch( m_currentLabel->GetSpinStyle() )
418 {
419 case SPIN_STYLE::RIGHT: m_spin0->Check( true ); break;
420 case SPIN_STYLE::LEFT: m_spin1->Check( true ); break;
421 case SPIN_STYLE::UP: m_spin2->Check( true ); break;
422 case SPIN_STYLE::BOTTOM: m_spin3->Check( true ); break;
423 }
424
427
428 return true;
429}
430
431
432void DIALOG_LABEL_PROPERTIES::OnEnterKey( wxCommandEvent& aEvent )
433{
434 wxPostEvent( this, wxCommandEvent( wxEVT_COMMAND_BUTTON_CLICKED, wxID_OK ) );
435}
436
437
439{
440 if( aEvent.GetKeyCode() == WXK_TAB )
441 {
442 if( aEvent.ShiftDown() )
443 {
444 m_textSizeCtrl->SetFocusFromKbd();
445 }
446 else if( !m_fields->empty() )
447 {
448 m_grid->SetFocusFromKbd();
449 m_grid->MakeCellVisible( 0, 0 );
450 m_grid->SetGridCursor( 0, 0 );
451 }
452 else
453 {
454 m_textSizeCtrl->SetFocusFromKbd();
455 }
456 }
457 else
458 {
459 aEvent.Skip();
460 }
461}
462
463
464static bool positioningChanged( const SCH_FIELD& a, const SCH_FIELD& b )
465{
466 return a.GetPosition() != b.GetPosition()
467 || a.GetHorizJustify() != b.GetHorizJustify()
468 || a.GetVertJustify() != b.GetVertJustify()
469 || a.GetTextAngle() != b.GetTextAngle();
470}
471
472
473static bool positioningChanged( FIELDS_GRID_TABLE* a, std::vector<SCH_FIELD>& b )
474{
475 for( size_t i = 0; i < a->size() && i < b.size(); ++i )
476 {
477 if( positioningChanged( a->at( i ), b.at( i ) ) )
478 return true;
479 }
480
481 return false;
482}
483
484
486{
488 return false;
489
490 if( !wxDialog::TransferDataFromWindow() )
491 return false;
492
493 // Don't allow text to disappear; it can be difficult to correct if you can't select it
494 if( !m_textSize.Validate( 0.01, 1000.0, EDA_UNITS::MM ) )
495 return false;
496
497 SCH_COMMIT commit( m_Parent );
498 wxString text;
499
500 /* save old text in undo list if not already in edit */
501 if( m_currentLabel->GetEditFlags() == 0 )
503
505
507 {
508 // labels need escaping
510
511 // convert any text variable cross-references to their UUIDs
513
514#ifdef __WXMAC__
515 // On macOS CTRL+Enter produces '\r' instead of '\n' regardless of EOL setting
516 text.Replace( wxS( "\r" ), wxS( "\n" ) );
517#endif
518
519 if( text.IsEmpty() && !m_currentLabel->IsNew() )
520 {
521 DisplayError( this, _( "Label can not be empty." ) );
522 return false;
523 }
524
526 }
527
528 // change all field positions from relative to absolute
529 for( SCH_FIELD& field : *m_fields )
530 {
531 field.Offset( m_currentLabel->GetPosition() );
532
533 if( field.GetCanonicalName() == wxT( "Netclass" ) )
534 {
535 field.SetLayer( LAYER_NETCLASS_REFS );
536 }
537 else if( field.GetId() == FIELD_T::INTERSHEET_REFS )
538 {
539 if( field.IsVisible() != m_Parent->Schematic().Settings().m_IntersheetRefsShow )
540 {
541 DisplayInfoMessage( this, _( "Intersheet reference visibility is "
542 "controlled globally from "
543 "Schematic Setup > General > Formatting" ) );
544 }
545
546 field.SetLayer( LAYER_INTERSHEET_REFS );
547 }
548 else
549 {
550 field.SetLayer( LAYER_FIELDS );
551 }
552 }
553
556
557 for( int ii = m_fields->GetNumberRows() - 1; ii >= 0; ii-- )
558 {
559 SCH_FIELD& field = m_fields->at( ii );
560 const wxString& fieldName = field.GetCanonicalName();
561 const wxString& fieldText = field.GetText();
562
563 if( fieldName.IsEmpty() && fieldText.IsEmpty() )
564 {
565 // delete empty, unnamed fields
566 m_fields->erase( m_fields->begin() + ii );
567 }
568 else if( fieldName == wxT( "Netclass" ) && fieldText.IsEmpty() )
569 {
570 // delete empty Netclass fields if there are other Netclass fields present
571 int netclassFieldCount = 0;
572
573 for( int jj = 0; jj < m_fields->GetNumberRows(); ++jj )
574 {
575 if( m_fields->at( jj ).GetCanonicalName() == wxT( "Netclass" ) )
576 netclassFieldCount++;
577 }
578
579 if( netclassFieldCount > 1 )
580 m_fields->erase( m_fields->begin() + ii );
581 }
582 else if( fieldName.IsEmpty() )
583 {
584 // give non-empty, unnamed fields a name
585 field.SetName( _( "untitled" ) );
586 }
587 }
588
589 int ordinal = 42; // Arbitrarily larger than any mandatory FIELD_T ids.
590
591 for( SCH_FIELD& field : *m_fields )
592 {
593 if( !field.IsMandatory() )
594 field.SetOrdinal( ordinal++ );
595 }
596
598
599 if( m_shapeSizer->AreAnyItemsShown() )
600 {
601 if( m_bidirectional->GetValue() )
602 m_currentLabel->SetShape( LABEL_FLAG_SHAPE::L_BIDI );
603 else if( m_input->GetValue() )
604 m_currentLabel->SetShape( LABEL_FLAG_SHAPE::L_INPUT );
605 else if( m_output->GetValue() )
606 m_currentLabel->SetShape( LABEL_FLAG_SHAPE::L_OUTPUT );
607 else if( m_triState->GetValue() )
608 m_currentLabel->SetShape( LABEL_FLAG_SHAPE::L_TRISTATE );
609 else if( m_passive->GetValue() )
610 m_currentLabel->SetShape( LABEL_FLAG_SHAPE::L_UNSPECIFIED );
611 else if( m_dot->GetValue() )
612 m_currentLabel->SetShape( LABEL_FLAG_SHAPE::F_DOT );
613 else if( m_circle->GetValue() )
614 m_currentLabel->SetShape( LABEL_FLAG_SHAPE::F_ROUND );
615 else if( m_diamond->GetValue() )
616 m_currentLabel->SetShape( LABEL_FLAG_SHAPE::F_DIAMOND );
617 else if( m_rectangle->GetValue() )
618 m_currentLabel->SetShape( LABEL_FLAG_SHAPE::F_RECTANGLE );
619 }
620
622 {
624 m_italic->IsChecked() ) );
625 }
626
628 static_cast<SCH_DIRECTIVE_LABEL*>( m_currentLabel )->SetPinLength( m_textSize.GetIntValue() );
632
633 // Must come after SetTextSize()
636
638
639 SPIN_STYLE selectedSpinStyle= SPIN_STYLE::LEFT;
640
641 if( m_spin0->IsChecked() ) selectedSpinStyle = SPIN_STYLE::RIGHT;
642 else if( m_spin1->IsChecked() ) selectedSpinStyle = SPIN_STYLE::LEFT;
643 else if( m_spin2->IsChecked() ) selectedSpinStyle = SPIN_STYLE::UP;
644 else if( m_spin3->IsChecked() ) selectedSpinStyle = SPIN_STYLE::BOTTOM;
645
647 {
648 SCH_EDIT_FRAME* frame = static_cast<SCH_EDIT_FRAME*>( m_parentFrame );
650 frame->AutoRotateItem( frame->GetScreen(), m_currentLabel );
651 }
652 else
653 {
655 }
656
658 && m_currentLabel->GetSpinStyle() != selectedSpinStyle )
659 {
660 m_currentLabel->SetSpinStyle( selectedSpinStyle );
661 }
662
664
665 if( fieldsAutoplaced == AUTOPLACE_AUTO || fieldsAutoplaced == AUTOPLACE_MANUAL )
666 m_currentLabel->AutoplaceFields( m_Parent->GetScreen(), fieldsAutoplaced );
667
668 if( !commit.Empty() )
669 commit.Push( _( "Edit Label Properties" ) );
670
671 else if( m_activeTextEntry && m_labelList )
672 {
673 text = m_activeTextEntry->GetValue();
674 // On macOS CTRL+Enter produces '\r' instead of '\n' regardless of EOL setting
675 text.Replace( wxS( "\r" ), wxS( "\n" ) );
676 wxArrayString lines = wxSplit( text, '\n' );
677
678 for( const wxString& line : lines )
679 {
680 text = EscapeString( line, CTX_NETNAME );
681 text.Trim( false ).Trim( true );
682
683 if( text.empty() )
684 continue;
685
686 // convert any text variable cross-references to their UUIDs
688
689 switch ( m_currentLabel->Type() )
690 {
692 {
693 SCH_GLOBALLABEL* label = new SCH_GLOBALLABEL( *static_cast<SCH_GLOBALLABEL*>( m_currentLabel ) );
694 label->SetText( text );
695 m_labelList->push_back( std::unique_ptr<SCH_LABEL_BASE>( label ) );
696 break;
697 }
698 case SCH_HIER_LABEL_T:
699 {
700 SCH_HIERLABEL* label = new SCH_HIERLABEL( *static_cast<SCH_HIERLABEL*>( m_currentLabel ) );
701 label->SetText( text );
702 m_labelList->push_back( std::unique_ptr<SCH_LABEL_BASE>( label ) );
703 break;
704 }
705 case SCH_LABEL_T:
706 {
707 SCH_LABEL* label = new SCH_LABEL( *static_cast<SCH_LABEL*>( m_currentLabel ) );
708 label->SetText( text );
709 m_labelList->push_back( std::unique_ptr<SCH_LABEL_BASE>( label ) );
710 break;
711 }
712 default:
713 break;
714 }
715 }
716 }
718 {
719 SCH_DIRECTIVE_LABEL* label =
721 m_labelList->push_back( std::unique_ptr<SCH_LABEL_BASE>( label ) );
722 }
723
724 return true;
725}
726
727
728void DIALOG_LABEL_PROPERTIES::onSpinButton( wxCommandEvent& aEvent )
729{
730 for( BITMAP_BUTTON* btn : { m_spin0, m_spin1, m_spin2, m_spin3 } )
731 {
732 if( btn->IsChecked() && btn != aEvent.GetEventObject() )
733 btn->Check( false );
734 }
735}
736
737
738void DIALOG_LABEL_PROPERTIES::OnFormattingHelp( wxHyperlinkEvent& aEvent )
739{
741}
742
743
744void DIALOG_LABEL_PROPERTIES::OnAddField( wxCommandEvent& event )
745{
747 return;
748
749 wxString fieldName = wxT( "Netclass" );
750
751 for( SCH_FIELD& field : *m_fields )
752 {
753 if( field.GetId() != FIELD_T::INTERSHEET_REFS && field.GetName() != wxT( "Netclass" ) )
754 {
755 fieldName = wxEmptyString;
756 break;
757 }
758 }
759
760 fieldName = SCH_LABEL_BASE::GetDefaultFieldName( fieldName, true );
761
762 SCH_FIELD newField( VECTOR2I( 0, 0 ), FIELD_T::USER, m_currentLabel, fieldName );
763
764 if( m_fields->size() > 0 )
765 {
766 newField.SetVisible( m_fields->at( m_fields->size() - 1 ).IsVisible() );
767 newField.SetTextAngle( m_fields->at( m_fields->size() - 1 ).GetTextAngle() );
768 newField.SetItalic( m_fields->at( m_fields->size() - 1 ).IsItalic() );
769 newField.SetBold( m_fields->at( m_fields->size() - 1 ).IsBold() );
770 }
771 else
772 {
773 newField.SetVisible( true );
774 newField.SetItalic( true );
775 }
776
777 m_fields->push_back( newField );
778
779 // notify the grid
780 wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_APPENDED, 1 );
781 m_grid->ProcessTableMessage( msg );
782
783 m_grid->MakeCellVisible( (int) m_fields->size() - 1, 0 );
784 m_grid->SetGridCursor( (int) m_fields->size() - 1, 0 );
785
786 m_grid->EnableCellEditControl();
787 m_grid->ShowCellEditControl();
788}
789
790
791void DIALOG_LABEL_PROPERTIES::OnDeleteField( wxCommandEvent& event )
792{
793 wxArrayInt selectedRows = m_grid->GetSelectedRows();
794
795 if( selectedRows.empty() && m_grid->GetGridCursorRow() >= 0 )
796 selectedRows.push_back( m_grid->GetGridCursorRow() );
797
798 if( selectedRows.empty() )
799 return;
800
801 for( int row : selectedRows )
802 {
803 if( row < m_currentLabel->GetMandatoryFieldCount() )
804 {
805 DisplayError( this, _( "The first field is mandatory." ) );
806 return;
807 }
808 }
809
810 m_grid->CommitPendingChanges( true /* quiet mode */ );
811
812 // Reverse sort so deleting a row doesn't change the indexes of the other rows.
813 selectedRows.Sort( []( int* first, int* second )
814 {
815 return *second - *first;
816 } );
817
818 for( int row : selectedRows )
819 {
820 //avoids an assert if we deselect early here
821 m_grid->DeselectRow( row );
822 m_fields->erase( m_fields->begin() + row );
823
824 // notify the grid
825 wxGridTableMessage msg( m_fields, wxGRIDTABLE_NOTIFY_ROWS_DELETED, row, 1 );
826 m_grid->ProcessTableMessage( msg );
827
828 if( m_grid->GetNumberRows() > 0 )
829 {
830 m_grid->MakeCellVisible( std::max( 0, row-1 ), m_grid->GetGridCursorCol() );
831 m_grid->SetGridCursor( std::max( 0, row-1 ), m_grid->GetGridCursorCol() );
832 }
833 }
834}
835
836
837void DIALOG_LABEL_PROPERTIES::OnMoveUp( wxCommandEvent& event )
838{
840 return;
841
842 int i = m_grid->GetGridCursorRow();
843
845 {
846 SCH_FIELD tmp = m_fields->at( (unsigned) i );
847 m_fields->erase( m_fields->begin() + i, m_fields->begin() + i + 1 );
848 m_fields->insert( m_fields->begin() + i - 1, tmp );
849 m_grid->ForceRefresh();
850
851 m_grid->SetGridCursor( i - 1, m_grid->GetGridCursorCol() );
852 m_grid->MakeCellVisible( m_grid->GetGridCursorRow(), m_grid->GetGridCursorCol() );
853 }
854 else
855 {
856 wxBell();
857 }
858}
859
860
861void DIALOG_LABEL_PROPERTIES::OnMoveDown( wxCommandEvent& event )
862{
864 return;
865
866 int i = m_grid->GetGridCursorRow();
867
868 if( i >= m_currentLabel->GetMandatoryFieldCount() && i < m_grid->GetNumberRows() - 1 )
869 {
870 SCH_FIELD tmp = m_fields->at( (unsigned) i );
871 m_fields->erase( m_fields->begin() + i, m_fields->begin() + i + 1 );
872 m_fields->insert( m_fields->begin() + i + 1, tmp );
873 m_grid->ForceRefresh();
874
875 m_grid->SetGridCursor( i + 1, m_grid->GetGridCursorCol() );
876 m_grid->MakeCellVisible( m_grid->GetGridCursorRow(), m_grid->GetGridCursorCol() );
877 }
878 else
879 {
880 wxBell();
881 }
882}
883
884
886{
887 m_width = aWidth;
888 // Account for scroll bars
889 aWidth -= ( m_grid->GetSize().x - m_grid->GetClientSize().x );
890
891 m_grid->AutoSizeColumn( 0 );
892 m_grid->SetColSize( 0, std::max( 72, m_grid->GetColSize( 0 ) ) );
893
894 int fixedColsWidth = m_grid->GetColSize( 0 );
895
896 for( int i = 2; i < m_grid->GetNumberCols(); i++ )
897 fixedColsWidth += m_grid->GetColSize( i );
898
899 m_grid->SetColSize( 1, std::max( 120, aWidth - fixedColsWidth ) );
900}
901
902
903void DIALOG_LABEL_PROPERTIES::OnUpdateUI( wxUpdateUIEvent& event )
904{
905 std::bitset<64> shownColumns = m_grid->GetShownColumns();
906
907 if( shownColumns != m_shownColumns )
908 {
909 m_shownColumns = shownColumns;
910
911 if( !m_grid->IsCellEditControlShown() )
912 AdjustGridColumns( m_grid->GetRect().GetWidth() );
913 }
914
915 // Handle a delayed focus
916 if( m_delayedFocusRow >= 0 && m_delayedFocusRow < m_grid->GetNumberRows() )
917 {
918 m_grid->SetFocus();
919 m_grid->MakeCellVisible( m_delayedFocusRow, m_delayedFocusColumn );
921 }
922
925}
926
927
928void DIALOG_LABEL_PROPERTIES::OnSizeGrid( wxSizeEvent& event )
929{
930 int new_size = event.GetSize().GetX();
931
932 if( m_width != new_size )
933 AdjustGridColumns( new_size );
934
935 // Always propagate for a grid repaint (needed if the height changes, as well as width)
936 event.Skip();
937}
938
939
958void DIALOG_LABEL_PROPERTIES::OnLabelFilter( wxCommandEvent& event )
959{
960 static bool isFiltering = false; // Prevent re-entry
961
962 if( isFiltering )
963 return;
964
965 isFiltering = true; // Set the flag
966
967 wxString currentLabelText = m_valueCombo->GetValue();
968
969 // Check if the text has changed compared to the previous value
970 if( currentLabelText != m_previousLabelText )
971 {
972 m_previousLabelText = currentLabelText; // Update the previous text
973
974 // Save the current cursor position
975 long insertionPoint = m_valueCombo->GetInsertionPoint();
976
977 wxArrayString filteredLabels;
978
979 if( currentLabelText.IsEmpty() )
980 {
981 // If the input is empty, append all existing labels
982 filteredLabels = m_existingLabelArray;
983 }
984 else
985 {
986 // Filter the items based on the user input
987 wxString filterText = currentLabelText.Lower();
988
989 std::copy_if( m_existingLabelArray.begin(), m_existingLabelArray.end(),
990 std::back_inserter( filteredLabels ),
991 [&filterText]( const wxString& label )
992 {
993 return label.Lower().Contains( filterText );
994 } );
995 }
996
997 // Update the dropdown items
998 m_valueCombo->Freeze(); // Prevent visual flickering
999 m_valueCombo->Clear();
1000 m_valueCombo->Append( filteredLabels );
1001 m_valueCombo->Thaw(); // Resume rendering
1002
1003 // Restore the user's input and cursor position
1004 m_valueCombo->ChangeValue( currentLabelText ); // Set without triggering wxEVT_TEXT
1005 m_valueCombo->SetInsertionPoint( insertionPoint );
1006 }
1007
1008 isFiltering = false; // Reset the flag
1009}
1010
1011
1013{
1014 // Get the selected item's value
1015 wxString selectedValue = m_valueCombo->GetValue();
1016 m_previousLabelText = selectedValue; // Update the previous text to match the selected value
1017}
1018
1019
1021{
1023 {
1024 m_labelCombo->Show( !m_cbMultiLine->IsChecked() );
1025 m_valueCombo->Show( !m_cbMultiLine->IsChecked() );
1026 m_labelMultiLine->Show( m_cbMultiLine->IsChecked() );
1027 m_valueMultiLine->Show( m_cbMultiLine->IsChecked() );
1028
1029 if( m_cbMultiLine->IsChecked() )
1030 {
1031 m_valueMultiLine->SetValue( m_valueCombo->GetValue() );
1034 }
1035 else
1036 {
1037 wxString multiText = m_valueMultiLine->GetValue();
1038 m_valueCombo->SetValue( multiText.BeforeFirst( '\n' ) );
1041 }
1042 }
1043 else if( m_currentLabel->Type() == SCH_HIER_LABEL_T )
1044 {
1045 m_labelSingleLine->Show( !m_cbMultiLine->IsChecked() );
1046 m_valueSingleLine->Show( !m_cbMultiLine->IsChecked() );
1047 m_labelMultiLine->Show( m_cbMultiLine->IsChecked() );
1048 m_valueMultiLine->Show( m_cbMultiLine->IsChecked() );
1049
1050 if( m_cbMultiLine->IsChecked() )
1051 {
1052 m_valueMultiLine->SetValue( m_valueSingleLine->GetValue() );
1055 }
1056 else
1057 {
1058 wxString multiText = m_valueMultiLine->GetValue();
1059 m_valueSingleLine->SetValue( multiText.BeforeFirst( '\n' ) );
1062 }
1063 }
1064
1065 Layout();
1066}
KIFACE_BASE & Kiface()
Global KIFACE_BASE "get" accessor.
wxBitmapBundle KiBitmapBundle(BITMAPS aBitmap, int aMinHeight)
Definition: bitmap.cpp:110
A bitmap button widget that behaves like an AUI toolbar item's button when it is drawn.
Definition: bitmap_button.h:42
void SetIsRadioButton()
bool IsChecked() const
void Check(bool aCheck=true)
Check the control.
void SetIsSeparator()
Render button as a toolbar separator.
void SetIsCheckButton()
Setup the control as a two-state button (checked or unchecked).
void SetBitmap(const wxBitmapBundle &aBmp)
Set the bitmap shown when the button is enabled.
Color settings are a bit different than most of the settings objects in that there can be more than o...
COLOR4D GetColor(int aLayer) const
void SetSwatchColor(const KIGFX::COLOR4D &aColor, bool aSendEvent)
Set the current swatch color directly.
KIGFX::COLOR4D GetSwatchColor() const
void SetDefaultColor(const KIGFX::COLOR4D &aColor)
Sets the color that will be chosen with the "Reset to Default" button in the chooser.
void SetSwatchBackground(const KIGFX::COLOR4D &aBackground)
Set the swatch background color.
COMMIT & Modify(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Modify a given item in the model.
Definition: commit.h:108
bool Empty() const
Definition: commit.h:150
Class DIALOG_LABEL_PROPERTIES_BASE.
void OnLabelItemSelected(wxCommandEvent &event)
Handles the selection of an item from the wxComboBox dropdown.
std::list< std::unique_ptr< SCH_LABEL_BASE > > * m_labelList
void OnSizeGrid(wxSizeEvent &event) override
void OnMoveUp(wxCommandEvent &event) override
void OnMoveDown(wxCommandEvent &event) override
DIALOG_LABEL_PROPERTIES(SCH_EDIT_FRAME *parent, SCH_LABEL_BASE *aLabel, bool aNew)
void OnEnterKey(wxCommandEvent &aEvent) override
wxEVT_COMMAND_ENTER event handler for single-line control.
void OnAddField(wxCommandEvent &event) override
SCH_NETNAME_VALIDATOR m_netNameValidator
void OnFormattingHelp(wxHyperlinkEvent &aEvent) override
void OnUpdateUI(wxUpdateUIEvent &event) override
void onSpinButton(wxCommandEvent &aEvent)
void onMultiLabelCheck(wxCommandEvent &aEvent) override
void OnDeleteField(wxCommandEvent &event) override
void OnLabelFilter(wxCommandEvent &event)
Handles the filtering of items in the wxComboBox based on user input.
void OnValueCharHook(wxKeyEvent &aEvent) override
void SetInitialFocus(wxWindow *aWindow)
Sets the window (usually a wxTextCtrl) that should be focused when the dialog is shown.
Definition: dialog_shim.h:66
void SetupStandardButtons(std::map< int, wxString > aLabels={})
std::string m_hash_key
Definition: dialog_shim.h:194
void finishDialogSettings()
In all dialogs, we must call the same functions to fix minimal dlg size, the default position and per...
EDA_BASE_FRAME * m_parentFrame
Definition: dialog_shim.h:212
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
EDA_ITEM_FLAGS GetEditFlags() const
Definition: eda_item.h:141
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:108
bool IsNew() const
Definition: eda_item.h:117
void SetTextColor(const COLOR4D &aColor)
Definition: eda_text.h:256
COLOR4D GetTextColor() const
Definition: eda_text.h:257
bool IsItalic() const
Definition: eda_text.h:156
const EDA_ANGLE & GetTextAngle() const
Definition: eda_text.h:134
void SetTextSize(VECTOR2I aNewSize, bool aEnforceMinTextSize=true)
Definition: eda_text.cpp:526
virtual const wxString & GetText() const
Return the string associated with the text object.
Definition: eda_text.h:98
KIFONT::FONT * GetFont() const
Definition: eda_text.h:234
int GetTextWidth() const
Definition: eda_text.h:251
void Offset(const VECTOR2I &aOffset)
Definition: eda_text.cpp:589
GR_TEXT_H_ALIGN_T GetHorizJustify() const
Definition: eda_text.h:187
virtual void SetVisible(bool aVisible)
Definition: eda_text.cpp:379
void SetBold(bool aBold)
Set the text to be bold - this will also update the font if needed.
Definition: eda_text.cpp:328
bool IsBold() const
Definition: eda_text.h:171
GR_TEXT_V_ALIGN_T GetVertJustify() const
Definition: eda_text.h:190
virtual void SetText(const wxString &aText)
Definition: eda_text.cpp:270
virtual void SetTextAngle(const EDA_ANGLE &aAngle)
Definition: eda_text.cpp:292
void SetItalic(bool aItalic)
Set the text to be italic - this will also update the font if needed.
Definition: eda_text.cpp:300
void SetFont(KIFONT::FONT *aFont)
Definition: eda_text.cpp:492
int GetNumberRows() override
void SetFontSelection(KIFONT::FONT *aFont, bool aSilentMode=false)
Set the selection in wxChoice widget.
Definition: font_choice.cpp:73
KIFONT::FONT * GetFontSelection(bool aBold, bool aItalic, bool aForDrawingSheet=false) const
bool HaveFontSelection() const
Definition: font_choice.cpp:95
APP_SETTINGS_BASE * KifaceSettings() const
Definition: kiface_base.h:95
A color representation with 4 components: red, green, blue, alpha.
Definition: color4d.h:104
SCHEMATIC_SETTINGS & Settings() const
Definition: schematic.cpp:306
wxString ConvertKIIDsToRefs(const wxString &aSource) const
Definition: schematic.cpp:575
wxString ConvertRefsToKIIDs(const wxString &aSource) const
Definition: schematic.cpp:505
SCH_SHEET & Root() const
Definition: schematic.h:117
SCH_DRAW_PANEL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
COLOR_SETTINGS * GetColorSettings(bool aForceRefresh=false) const override
Returns a pointer to the active color theme settings.
virtual void Push(const wxString &aMessage=wxT("A commit"), int aCommitFlags=0) override
Execute the changes.
Definition: sch_commit.cpp:435
Schematic editor (Eeschema) main window.
SCH_SCREEN * GetScreen() const override
Return a pointer to a BASE_SCREEN or one of its derivatives.
SCHEMATIC & Schematic() const
void AutoRotateItem(SCH_SCREEN *aScreen, SCH_ITEM *aItem)
Automatically set the rotation of an item (if the item supports it).
VECTOR2I GetPosition() const override
Definition: sch_field.cpp:1314
wxString GetCanonicalName() const
Get a non-language-specific name for a field which can be used for storage, variable look-up,...
Definition: sch_field.cpp:1098
void SetName(const wxString &aName)
Definition: sch_field.cpp:1059
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition: sch_item.h:167
SCHEMATIC * Schematic() const
Search the item hierarchy to find a SCHEMATIC.
Definition: sch_item.cpp:151
void SetFieldsAutoplaced(AUTOPLACE_ALGO aAlgo)
Definition: sch_item.h:562
AUTOPLACE_ALGO GetFieldsAutoplaced() const
Return whether the fields have been automatically placed.
Definition: sch_item.h:561
void SetFields(const std::vector< SCH_FIELD > &aFields)
Set multiple schematic fields.
Definition: sch_label.h:213
bool AutoRotateOnPlacement() const
autoRotateOnPlacement
Definition: sch_label.cpp:1414
SPIN_STYLE GetSpinStyle() const
Definition: sch_label.cpp:337
void SetShape(LABEL_FLAG_SHAPE aShape)
Definition: sch_label.h:177
LABEL_FLAG_SHAPE GetShape() const
Definition: sch_label.h:176
virtual bool AutoRotateOnPlacementSupported() const =0
void SetAutoRotateOnPlacement(bool autoRotate=true)
Definition: sch_label.cpp:1420
static const wxString GetDefaultFieldName(const wxString &aName, bool aUseDefaultName)
Definition: sch_label.cpp:202
void AutoplaceFields(SCH_SCREEN *aScreen, AUTOPLACE_ALGO aAlgo) override
Definition: sch_label.cpp:566
virtual int GetMandatoryFieldCount()
Definition: sch_label.h:203
std::vector< SCH_FIELD > & GetFields()
Definition: sch_label.h:205
virtual wxString GetClass() const override=0
Return the class name.
virtual void SetSpinStyle(SPIN_STYLE aSpinStyle)
Definition: sch_label.cpp:302
Container class that holds multiple SCH_SCREEN objects in a hierarchy.
Definition: sch_screen.h:734
SCH_SCREEN * GetNext()
SCH_SCREEN * GetFirst()
Schematic symbol object.
Definition: sch_symbol.h:75
bool IsSymbolLikePowerGlobalLabel() const
bool IsSymbolLikePowerLocalLabel() const
SCH_FIELD * GetField(FIELD_T aFieldType)
Return a mandatory field in this symbol.
Definition: sch_symbol.cpp:813
static HTML_MESSAGE_BOX * ShowSyntaxHelp(wxWindow *aParentWindow)
Definition: sch_label.cpp:2192
VECTOR2I GetPosition() const override
Definition: sch_text.h:139
void SetBitmap(const wxBitmapBundle &aBmp)
int GetIntValue()
Definition: unit_binder.h:129
virtual bool Validate(double aMin, double aMax, EDA_UNITS aUnits=EDA_UNITS::UNSCALED)
Validate the control against the given range, informing the user of any errors found.
virtual void SetValue(long long int aValue)
Set new value (in Internal Units) for the text field, taking care of units conversion.
void ShowHideColumns(const wxString &shownColumns)
Show/hide the grid columns based on a tokenized string of shown column indexes.
Definition: wx_grid.cpp:494
void SetTable(wxGridTableBase *table, bool aTakeOwnership=false)
Hide wxGrid's SetTable() method with one which doesn't mess up the grid column widths when setting th...
Definition: wx_grid.cpp:275
void DestroyTable(wxGridTableBase *aTable)
Work-around for a bug in wxGrid which crashes when deleting the table if the cell edit control was no...
Definition: wx_grid.cpp:449
wxString GetShownColumnsAsString()
Get a tokenized string containing the shown column indexes.
Definition: wx_grid.cpp:464
std::bitset< 64 > GetShownColumns()
Definition: wx_grid.cpp:483
bool CommitPendingChanges(bool aQuietMode=false)
Close any open cell edit controls.
Definition: wx_grid.cpp:644
void DisplayInfoMessage(wxWindow *aParent, const wxString &aMessage, const wxString &aExtraInfo)
Display an informational message box with aMessage.
Definition: confirm.cpp:221
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.
static bool positioningChanged(const SCH_FIELD &a, const SCH_FIELD &b)
#define _(s)
@ FDC_VALUE
@ LAYER_FIELDS
Definition: layer_ids.h:451
@ LAYER_NETCLASS_REFS
Definition: layer_ids.h:453
@ LAYER_SCHEMATIC_BACKGROUND
Definition: layer_ids.h:477
@ LAYER_INTERSHEET_REFS
Definition: layer_ids.h:452
#define UNIMPLEMENTED_FOR(type)
Definition: macros.h:96
AUTOPLACE_ALGO
Definition: sch_item.h:68
@ AUTOPLACE_MANUAL
Definition: sch_item.h:71
@ AUTOPLACE_NONE
Definition: sch_item.h:69
@ AUTOPLACE_AUTO
Definition: sch_item.h:70
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:...
#define TO_UTF8(wxstring)
Convert a wxString to a UTF8 encoded C string for all wxWidgets build modes.
Definition: string_utils.h:403
@ CTX_NETNAME
Definition: string_utils.h:53
@ SCH_DIRECTIVE_LABEL_T
Definition: typeinfo.h:171
@ SCH_LABEL_T
Definition: typeinfo.h:167
@ SCH_HIER_LABEL_T
Definition: typeinfo.h:169
@ SCH_SHEET_PIN_T
Definition: typeinfo.h:174
@ SCH_SYMBOL_LOCATE_POWER_T
Definition: typeinfo.h:196
@ SCH_GLOBAL_LABEL_T
Definition: typeinfo.h:168
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695