KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
symbol_editor_edit_tool.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) 2019 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, you may find one here:
19 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
20 * or you may search the http://www.gnu.org website for the version 2 license,
21 * or you may write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 */
24
26
27#include <tool/picker_tool.h>
33#include <clipboard.h>
34#include <sch_actions.h>
35#include <increment.h>
36#include <pin_layout_cache.h>
37#include <string_utils.h>
38#include <symbol_edit_frame.h>
39#include <sch_commit.h>
41#include <dialogs/dialog_text_properties.h>
46#include <view/view_controls.h>
47#include <richio.h>
49#include <sch_textbox.h>
50#include <wx/textdlg.h> // for wxTextEntryDialog
51#include <math/util.h> // for KiROUND
53
55 SCH_TOOL_BASE( "eeschema.SymbolEditTool" ),
56 m_pickerItem( nullptr )
57{
58}
59
60
62{
64
67
68 wxASSERT_MSG( drawingTools, "eeschema.SymbolDrawing tool is not available" );
69
70 auto haveSymbolCondition =
71 [&]( const SELECTION& sel )
72 {
74 };
75
76 auto canEdit =
77 [&]( const SELECTION& sel )
78 {
80 return false;
81
82 if( m_frame->IsSymbolAlias() )
83 {
84 for( EDA_ITEM* item : sel )
85 {
86 if( item->Type() != SCH_FIELD_T )
87 return false;
88 }
89 }
90
91 return true;
92 };
93
94 const auto canCopyText = SCH_CONDITIONS::OnlyTypes( {
101 } );
102
103 // clang-format off
104 // Add edit actions to the move tool menu
105 if( moveTool )
106 {
107 CONDITIONAL_MENU& moveMenu = moveTool->GetToolMenu().GetMenu();
108
109 moveMenu.AddSeparator( 200 );
110 moveMenu.AddItem( SCH_ACTIONS::rotateCCW, canEdit && SCH_CONDITIONS::NotEmpty, 200 );
111 moveMenu.AddItem( SCH_ACTIONS::rotateCW, canEdit && SCH_CONDITIONS::NotEmpty, 200 );
112 moveMenu.AddItem( SCH_ACTIONS::mirrorV, canEdit && SCH_CONDITIONS::NotEmpty, 200 );
113 moveMenu.AddItem( SCH_ACTIONS::mirrorH, canEdit && SCH_CONDITIONS::NotEmpty, 200 );
114
115 moveMenu.AddItem( SCH_ACTIONS::swap, canEdit && SELECTION_CONDITIONS::MoreThan( 1 ), 200);
116 moveMenu.AddItem( SCH_ACTIONS::properties, canEdit && SCH_CONDITIONS::Count( 1 ), 200 );
117
118 moveMenu.AddSeparator( 300 );
121 moveMenu.AddItem( ACTIONS::copyAsText, canCopyText && SCH_CONDITIONS::IdleSelection, 300 );
122 moveMenu.AddItem( ACTIONS::duplicate, canEdit && SCH_CONDITIONS::NotEmpty, 300 );
123 moveMenu.AddItem( ACTIONS::doDelete, canEdit && SCH_CONDITIONS::NotEmpty, 200 );
124
125 moveMenu.AddSeparator( 400 );
126 moveMenu.AddItem( ACTIONS::selectAll, haveSymbolCondition, 400 );
127 moveMenu.AddItem( ACTIONS::unselectAll, haveSymbolCondition, 400 );
128 }
129
130 // Add editing actions to the drawing tool menu
131 CONDITIONAL_MENU& drawMenu = drawingTools->GetToolMenu().GetMenu();
132
133 drawMenu.AddSeparator( 200 );
138
139 drawMenu.AddItem( SCH_ACTIONS::properties, canEdit && SCH_CONDITIONS::Count( 1 ), 200 );
140
141 // Add editing actions to the selection tool menu
143
144 selToolMenu.AddItem( SCH_ACTIONS::rotateCCW, canEdit && SCH_CONDITIONS::NotEmpty, 200 );
145 selToolMenu.AddItem( SCH_ACTIONS::rotateCW, canEdit && SCH_CONDITIONS::NotEmpty, 200 );
146 selToolMenu.AddItem( SCH_ACTIONS::mirrorV, canEdit && SCH_CONDITIONS::NotEmpty, 200 );
147 selToolMenu.AddItem( SCH_ACTIONS::mirrorH, canEdit && SCH_CONDITIONS::NotEmpty, 200 );
148
149 selToolMenu.AddItem( SCH_ACTIONS::swap, canEdit && SELECTION_CONDITIONS::MoreThan( 1 ), 200 );
150 selToolMenu.AddItem( SCH_ACTIONS::properties, canEdit && SCH_CONDITIONS::Count( 1 ), 200 );
151
152 selToolMenu.AddSeparator( 300 );
155 selToolMenu.AddItem( ACTIONS::copyAsText, canCopyText && SCH_CONDITIONS::IdleSelection, 300 );
156 selToolMenu.AddItem( ACTIONS::paste, canEdit && SCH_CONDITIONS::Idle, 300 );
157 selToolMenu.AddItem( ACTIONS::duplicate, canEdit && SCH_CONDITIONS::NotEmpty, 300 );
158 selToolMenu.AddItem( ACTIONS::doDelete, canEdit && SCH_CONDITIONS::NotEmpty, 300 );
159
160 selToolMenu.AddSeparator( 400 );
161 selToolMenu.AddItem( ACTIONS::selectAll, haveSymbolCondition, 400 );
162 selToolMenu.AddItem( ACTIONS::unselectAll, haveSymbolCondition, 400 );
163 // clang-format on
164
165 return true;
166}
167
168
170{
172
173 if( selection.GetSize() == 0 )
174 return 0;
175
176 VECTOR2I rotPoint;
177 bool ccw = ( aEvent.Matches( SCH_ACTIONS::rotateCCW.MakeEvent() ) );
178 SCH_ITEM* item = static_cast<SCH_ITEM*>( selection.Front() );
179 SCH_COMMIT localCommit( m_toolMgr );
180 SCH_COMMIT* commit = dynamic_cast<SCH_COMMIT*>( aEvent.Commit() );
181
182 if( !commit )
183 commit = &localCommit;
184
185 if( !item->IsMoving() )
186 commit->Modify( m_frame->GetCurSymbol(), m_frame->GetScreen() );
187
188 if( selection.GetSize() == 1 )
189 rotPoint = item->GetPosition();
190 else
191 rotPoint = m_frame->GetNearestHalfGridPosition( selection.GetCenter() );
192
193 for( unsigned ii = 0; ii < selection.GetSize(); ii++ )
194 {
195 item = static_cast<SCH_ITEM*>( selection.GetItem( ii ) );
196 item->Rotate( rotPoint, ccw );
197 m_frame->UpdateItem( item, false, true );
198 }
199
200 if( item->IsMoving() )
201 {
203 }
204 else
205 {
206 if( selection.IsHover() )
208
209 if( !localCommit.Empty() )
210 localCommit.Push( _( "Rotate" ) );
211 }
212
213 return 0;
214}
215
216
218{
220
221 if( selection.GetSize() == 0 )
222 return 0;
223
224 VECTOR2I mirrorPoint;
225 bool xAxis = ( aEvent.Matches( SCH_ACTIONS::mirrorV.MakeEvent() ) );
226 SCH_ITEM* item = static_cast<SCH_ITEM*>( selection.Front() );
227
228 if( !item->IsMoving() )
229 saveCopyInUndoList( m_frame->GetCurSymbol(), UNDO_REDO::LIBEDIT );
230
231 if( selection.GetSize() == 1 )
232 {
233 mirrorPoint = item->GetPosition();
234
235 switch( item->Type() )
236 {
237 case SCH_FIELD_T:
238 {
239 SCH_FIELD* field = static_cast<SCH_FIELD*>( item );
240
241 if( xAxis )
243 else
245
246 break;
247 }
248
249 default:
250 if( xAxis )
251 item->MirrorVertically( mirrorPoint.y );
252 else
253 item->MirrorHorizontally( mirrorPoint.x );
254
255 break;
256 }
257
258
259 m_frame->UpdateItem( item, false, true );
260 }
261 else
262 {
263 mirrorPoint = m_frame->GetNearestHalfGridPosition( selection.GetCenter() );
264
265 for( unsigned ii = 0; ii < selection.GetSize(); ii++ )
266 {
267 item = static_cast<SCH_ITEM*>( selection.GetItem( ii ) );
268
269 if( xAxis )
270 item->MirrorVertically( mirrorPoint.y );
271 else
272 item->MirrorHorizontally( mirrorPoint.x );
273
274 m_frame->UpdateItem( item, false, true );
275 }
276 }
277
278 if( item->IsMoving() )
279 {
281 }
282 else
283 {
284 if( selection.IsHover() )
286
287 m_frame->OnModify();
288 }
289
290 return 0;
291}
292
293
294const std::vector<KICAD_T> swappableItems = {
295 LIB_SYMBOL_T, // Allows swapping the anchor
296 SCH_PIN_T,
301};
302
303
305{
307 std::vector<EDA_ITEM*> sorted = selection.GetItemsSortedBySelectionOrder();
308
309 if( selection.Size() < 2 )
310 return 0;
311
312 EDA_ITEM* front = selection.Front();
313 bool isMoving = front->IsMoving();
314
315 // Save copy for undo if not in edit (edit command already handle the save copy)
316 if( front->GetEditFlags() == 0 )
317 saveCopyInUndoList( front->GetParent(), UNDO_REDO::LIBEDIT );
318
319 for( size_t i = 0; i < sorted.size() - 1; i++ )
320 {
321 SCH_ITEM* a = static_cast<SCH_ITEM*>( sorted[i] );
322 SCH_ITEM* b = static_cast<SCH_ITEM*>( sorted[( i + 1 ) % sorted.size()] );
323
324 VECTOR2I aPos = a->GetPosition(), bPos = b->GetPosition();
325 std::swap( aPos, bPos );
326
327 a->SetPosition( aPos );
328 b->SetPosition( bPos );
329
330 // Special case some common swaps
331 if( a->Type() == b->Type() )
332 {
333 switch( a->Type() )
334 {
335 case SCH_PIN_T:
336 {
337 SCH_PIN* aPin = static_cast<SCH_PIN*>( a );
338 SCH_PIN* bBpin = static_cast<SCH_PIN*>( b );
339
340 PIN_ORIENTATION aOrient = aPin->GetOrientation();
341 PIN_ORIENTATION bOrient = bBpin->GetOrientation();
342
343 aPin->SetOrientation( bOrient );
344 bBpin->SetOrientation( aOrient );
345
346 break;
347 }
348 default: break;
349 }
350 }
351
352 m_frame->UpdateItem( a, false, true );
353 m_frame->UpdateItem( b, false, true );
354 }
355
356 // Update R-Tree for modified items
357 for( EDA_ITEM* selected : selection )
358 updateItem( selected, true );
359
360 if( isMoving )
361 {
363 }
364 else
365 {
366 if( selection.IsHover() )
368
369 m_frame->OnModify();
370 }
371
372 return 0;
373}
374
375
376static std::vector<KICAD_T> nonFields =
377{
383};
384
385
387{
388 LIB_SYMBOL* symbol = m_frame->GetCurSymbol();
389 std::deque<EDA_ITEM*> items = m_selectionTool->RequestSelection().GetItems();
390 SCH_COMMIT commit( m_frame );
391
392 if( items.empty() )
393 return 0;
394
395 // Don't leave a freed pointer in the selection
397
398 commit.Modify( symbol, m_frame->GetScreen() );
399
400 std::set<SCH_ITEM*> toDelete;
401 int fieldsHidden = 0;
402 int fieldsAlreadyHidden = 0;
403
404 for( EDA_ITEM* item : items )
405 {
406 if( item->Type() == SCH_PIN_T )
407 {
408 SCH_PIN* curr_pin = static_cast<SCH_PIN*>( item );
409 VECTOR2I pos = curr_pin->GetPosition();
410
411 toDelete.insert( curr_pin );
412
413 // when pin editing is synchronized, pins in the same position, with the same name
414 // in different units are also removed. But only one pin per unit (matching)
415 if( m_frame->SynchronizePins() )
416 {
417 std::vector<bool> got_unit( symbol->GetUnitCount() + 1 );
418
419 got_unit[curr_pin->GetUnit()] = true;
420
421 for( SCH_PIN* pin : symbol->GetPins() )
422 {
423 if( got_unit[pin->GetUnit()] )
424 continue;
425
426 if( pin->GetPosition() != pos )
427 continue;
428
429 if( pin->GetBodyStyle() != curr_pin->GetBodyStyle() )
430 continue;
431
432 if( pin->GetType() != curr_pin->GetType() )
433 continue;
434
435 if( pin->GetName() != curr_pin->GetName() )
436 continue;
437
438 toDelete.insert( pin );
439 got_unit[pin->GetUnit()] = true;
440 }
441 }
442 }
443 else if( item->Type() == SCH_FIELD_T )
444 {
445 SCH_FIELD* field = static_cast<SCH_FIELD*>( item );
446
447 // Hide "deleted" fields
448 if( field->IsVisible() )
449 {
450 field->SetVisible( false );
451 fieldsHidden++;
452 }
453 else
454 {
455 fieldsAlreadyHidden++;
456 }
457 }
458 else if( SCH_ITEM* schItem = dynamic_cast<SCH_ITEM*>( item ) )
459 {
460 toDelete.insert( schItem );
461 }
462 }
463
464 for( SCH_ITEM* item : toDelete )
465 symbol->RemoveDrawItem( item );
466
467 if( toDelete.size() == 0 )
468 {
469 if( fieldsHidden == 1 )
470 commit.Push( _( "Hide Field" ) );
471 else if( fieldsHidden > 1 )
472 commit.Push( _( "Hide Fields" ) );
473 else if( fieldsAlreadyHidden > 0 )
474 m_frame->ShowInfoBarError( _( "Use the Symbol Properties dialog to remove fields." ) );
475 }
476 else
477 {
478 commit.Push( _( "Delete" ) );
479 }
480
482 return 0;
483}
484
485
486#define HITTEST_THRESHOLD_PIXELS 5
487
488
490{
492
494 m_pickerItem = nullptr;
495
496 // Deactivate other tools; particularly important if another PICKER is currently running
497 Activate();
498
499 picker->SetCursor( KICURSOR::REMOVE );
500
501 picker->SetClickHandler(
502 [this]( const VECTOR2D& aPosition ) -> bool
503 {
504 if( m_pickerItem )
505 {
507 selectionTool->UnbrightenItem( m_pickerItem );
508 selectionTool->AddItemToSel( m_pickerItem, true /*quiet mode*/ );
510 m_pickerItem = nullptr;
511 }
512
513 return true;
514 } );
515
516 picker->SetMotionHandler(
517 [this]( const VECTOR2D& aPos )
518 {
520 SCH_COLLECTOR collector;
521
522 selectionTool->CollectHits( collector, aPos, nonFields );
523
524 // Remove unselectable items
525 for( int i = collector.GetCount() - 1; i >= 0; --i )
526 {
527 if( !selectionTool->Selectable( collector[ i ] ) )
528 collector.Remove( i );
529 }
530
531 if( collector.GetCount() > 1 )
532 selectionTool->GuessSelectionCandidates( collector, aPos );
533
534 EDA_ITEM* item = collector.GetCount() == 1 ? collector[ 0 ] : nullptr;
535
536 if( m_pickerItem != item )
537 {
538 if( m_pickerItem )
539 selectionTool->UnbrightenItem( m_pickerItem );
540
541 m_pickerItem = item;
542
543 if( m_pickerItem )
544 selectionTool->BrightenItem( m_pickerItem );
545 }
546 } );
547
548 picker->SetFinalizeHandler(
549 [this]( const int& aFinalState )
550 {
551 if( m_pickerItem )
552 m_toolMgr->GetTool<SCH_SELECTION_TOOL>()->UnbrightenItem( m_pickerItem );
553
554 // Wake the selection tool after exiting to ensure the cursor gets updated
556 } );
557
559
560 return 0;
561}
562
563
565{
567
568 if( selection.Empty() || aEvent.IsAction( &SCH_ACTIONS::symbolProperties ) )
569 {
570 if( m_frame->GetCurSymbol() )
572 }
573 else if( selection.Size() == 1 )
574 {
575 SCH_ITEM* item = static_cast<SCH_ITEM*>( selection.Front() );
576
577 // Save copy for undo if not in edit (edit command already handle the save copy)
578 if( item->GetEditFlags() == 0 )
579 saveCopyInUndoList( item->GetParent(), UNDO_REDO::LIBEDIT );
580
581 switch( item->Type() )
582 {
583 case SCH_PIN_T:
584 {
585 SCH_PIN& pin = static_cast<SCH_PIN&>( *item );
586
587 // Mouse, not cursor, as grid points may well not be under any text
588 const VECTOR2I& mousePos = m_toolMgr->GetMousePosition();
589 PIN_LAYOUT_CACHE& layout = pin.GetLayoutCache();
590
591 bool mouseOverNumber = false;
592 if( OPT_BOX2I numberBox = layout.GetPinNumberBBox() )
593 {
594 mouseOverNumber = numberBox->Contains( mousePos );
595 }
596
598 pinTool->EditPinProperties( &pin, mouseOverNumber );
599
600 break;
601 }
602 case SCH_SHAPE_T:
603 editShapeProperties( static_cast<SCH_SHAPE*>( item ) );
604 break;
605
606 case SCH_TEXT_T:
607 editTextProperties( item );
608 break;
609
610 case SCH_TEXTBOX_T:
611 editTextBoxProperties( item );
612 break;
613
614 case SCH_FIELD_T:
615 editFieldProperties( static_cast<SCH_FIELD*>( item ) );
616 break;
617
618 default:
619 wxFAIL_MSG( wxT( "Unhandled item <" ) + item->GetClass() + wxT( ">" ) );
620 break;
621 }
622 }
623
624 if( selection.IsHover() )
626
627 return 0;
628}
629
630
632{
633 DIALOG_SHAPE_PROPERTIES dlg( m_frame, aShape );
634
635 if( dlg.ShowModal() != wxID_OK )
636 return;
637
638 updateItem( aShape, true );
640 m_frame->OnModify();
641
644 drawingTools->SetDrawSpecificUnit( !dlg.GetApplyToAllUnits() );
645
646 std::vector<MSG_PANEL_ITEM> items;
647 aShape->GetMsgPanelInfo( m_frame, items );
648 m_frame->SetMsgPanel( items );
649}
650
651
653{
654 if ( aItem->Type() != SCH_TEXT_T )
655 return;
656
657 DIALOG_TEXT_PROPERTIES dlg( m_frame, static_cast<SCH_TEXT*>( aItem ) );
658
659 if( dlg.ShowModal() != wxID_OK )
660 return;
661
662 updateItem( aItem, true );
664 m_frame->OnModify( );
665}
666
667
669{
670 if ( aItem->Type() != SCH_TEXTBOX_T )
671 return;
672
673 DIALOG_TEXT_PROPERTIES dlg( m_frame, static_cast<SCH_TEXTBOX*>( aItem ) );
674
675 if( dlg.ShowModal() != wxID_OK )
676 return;
677
678 updateItem( aItem, true );
680 m_frame->OnModify( );
681}
682
683
685{
686 if( aField == nullptr )
687 return;
688
689 wxString caption;
690
691 if( aField->IsMandatory() )
692 caption.Printf( _( "Edit %s Field" ), TitleCaps( aField->GetName() ) );
693 else
694 caption.Printf( _( "Edit '%s' Field" ), aField->GetName() );
695
696 DIALOG_FIELD_PROPERTIES dlg( m_frame, caption, aField );
697
698 // The dialog may invoke a kiway player for footprint fields
699 // so we must use a quasimodal dialog.
700 if( dlg.ShowQuasiModal() != wxID_OK )
701 return;
702
703 wxString newFieldValue = EscapeString( dlg.GetText(), CTX_LIBID );
704 wxString oldFieldValue = aField->GetFullText( m_frame->GetUnit() );
705
706 SCH_COMMIT commit( m_toolMgr );
707 commit.Modify( aField, m_frame->GetScreen() );
708
709 dlg.UpdateField( aField );
710
711 commit.Push( caption );
712
715}
716
717
719{
720 LIB_SYMBOL* symbol = m_frame->GetCurSymbol();
721 bool partLocked = symbol->UnitsLocked();
722
725
727
728 // This dialog itself subsequently can invoke a KIWAY_PLAYER as a quasimodal
729 // frame. Therefore this dialog as a modal frame parent, MUST be run under
730 // quasimodal mode for the quasimodal frame support to work. So don't use
731 // the QUASIMODAL macros here.
732 if( dlg.ShowQuasiModal() != wxID_OK )
733 return;
734
735 m_frame->OnModify();
736
737 // if m_UnitSelectionLocked has changed, set some edit options or defaults
738 // to the best value
739 if( partLocked != symbol->UnitsLocked() )
740 {
742
743 // Enable synchronized pin edit mode for symbols with interchangeable units
744 m_frame->m_SyncPinEdit = !symbol->UnitsLocked();
745
746 // also set default edit options to the better value
747 // Usually if units are locked, graphic items are specific to each unit
748 // and if units are interchangeable, graphic items are common to units
749 tools->SetDrawSpecificUnit( symbol->UnitsLocked() );
750 }
751}
752
754 int& aSymbolLastPinNumber )
755{
756 if( !aNewPin->GetNumber().IsEmpty() )
757 {
758 // when duplicating a pin in symbol editor, assigning identical pin number
759 // to the old one does not makes any sense, so assign the next unassigned number to it
760 aSymbolLastPinNumber++;
761 aNewPin->SetNumber( wxString::Format( wxT( "%i" ), aSymbolLastPinNumber ) );
762 }
763}
764
766{
767 SCH_COMMIT commit( m_frame );
768 LIB_SYMBOL* symbol = m_frame->GetCurSymbol();
769
770 if( !symbol )
771 return 0;
772
773 commit.Modify( symbol );
774
776
777 DIALOG_LIB_EDIT_PIN_TABLE dlg( m_frame, symbol );
778
779 if( dlg.ShowModal() == wxID_CANCEL )
780 return -1;
781
782 commit.Push( _( "Edit Pins" ) );
784
785 return 0;
786}
787
788
790{
791 LIB_SYMBOL* symbol = m_frame->GetCurSymbol();
792
793 if( !symbol )
794 return 0;
795
796 if( !symbol->IsDerived() )
797 {
798 m_frame->ShowInfoBarError( _( "Symbol is not derived from another symbol." ) );
799 }
800 else
801 {
803
804 if( dlg.ShowModal() == wxID_CANCEL )
805 return -1;
806 }
807
808 return 0;
809}
810
811
813{
814 LIB_SYMBOL* symbol = m_frame->GetCurSymbol();
815
816 if( !symbol )
817 return 0;
818
819 int unitid = m_frame->GetUnit();
820
821 if( unitid == 0 )
822 {
823 return -1;
824 }
825
826 wxString promptText = wxString::Format( _( "Enter display name for unit %s" ),
827 symbol->GetUnitReference( unitid ) );
828 wxString currentvalue;
829
830 if( symbol->HasUnitDisplayName( unitid ) )
831 {
832 currentvalue = symbol->GetUnitDisplayName( unitid );
833 }
834
835 wxTextEntryDialog dlg( m_frame, promptText, _( "Set Unit Display Name" ), currentvalue );
836
837 if( dlg.ShowModal() == wxID_OK )
838 {
839 saveCopyInUndoList( symbol, UNDO_REDO::LIBEDIT );
840 symbol->SetUnitDisplayName( unitid, dlg.GetValue() );
842 m_frame->OnModify();
843 }
844 else
845 {
846 return -1;
847 }
848
849 return 0;
850}
851
852
854{
856
857 // Nuke the selection for later rebuilding. This does *not* clear the flags on any items;
858 // it just clears the SELECTION's reference to them.
859 selTool->GetSelection().Clear();
860 {
862 }
863 selTool->RebuildSelection();
864
865 return 0;
866}
867
868
870{
872
873 // Nuke the selection for later rebuilding. This does *not* clear the flags on any items;
874 // it just clears the SELECTION's reference to them.
875 selTool->GetSelection().Clear();
876 {
878 }
879 selTool->RebuildSelection();
880
881 return 0;
882}
883
884
886{
887 int retVal = Copy( aEvent );
888
889 if( retVal == 0 )
890 retVal = DoDelete( aEvent );
891
892 return retVal;
893}
894
895
897{
898 LIB_SYMBOL* symbol = m_frame->GetCurSymbol();
900
901 if( !symbol || !selection.GetSize() )
902 return 0;
903
904 for( SCH_ITEM& item : symbol->GetDrawItems() )
905 {
906 if( item.Type() == SCH_FIELD_T )
907 continue;
908
909 wxASSERT( !item.HasFlag( STRUCT_DELETED ) );
910
911 if( !item.IsSelected() )
912 item.SetFlags( STRUCT_DELETED );
913 }
914
915 LIB_SYMBOL* partCopy = new LIB_SYMBOL( *symbol );
916
917 STRING_FORMATTER formatter;
918 SCH_IO_KICAD_SEXPR::FormatLibSymbol( partCopy, formatter );
919
920 delete partCopy;
921
922 for( SCH_ITEM& item : symbol->GetDrawItems() )
924
925 std::string prettyData = formatter.GetString();
926 KICAD_FORMAT::Prettify( prettyData, true );
927
928 if( SaveClipboard( prettyData ) )
929 return 0;
930 else
931 return -1;
932}
933
934
936{
938 SCH_SELECTION& selection = selTool->RequestSelection();
939
940 if( selection.Empty() )
941 return 0;
942
943 wxString itemsAsText = GetSelectedItemsAsText( selection );
944
945 if( selection.IsHover() )
947
948 return SaveClipboard( itemsAsText.ToStdString() );
949}
950
951
953{
954 LIB_SYMBOL* symbol = m_frame->GetCurSymbol();
955 LIB_SYMBOL* newPart = nullptr;
956
957 if( !symbol || symbol->IsDerived() )
958 return 0;
959
960 std::string clipboardData = GetClipboardUTF8();
961
962 try
963 {
964 std::vector<LIB_SYMBOL*> newParts = SCH_IO_KICAD_SEXPR::ParseLibSymbols( clipboardData, "Clipboard" );
965
966 if( newParts.empty() || !newParts[0] )
967 return -1;
968
969 newPart = newParts[0];
970 }
971 catch( IO_ERROR& )
972 {
973 // If it's not a symbol then paste as text
974 newPart = new LIB_SYMBOL( "dummy_part" );
975
976 wxString pasteText( clipboardData );
977
978 // Limit of 5000 is totally arbitrary. Without a limit, pasting a bitmap image from
979 // eeschema makes KiCad appear to hang.
980 if( pasteText.Length() > 5000 )
981 pasteText = pasteText.Left( 5000 ) + wxT( "..." );
982
983 SCH_TEXT* newText = new SCH_TEXT( { 0, 0 }, pasteText, LAYER_DEVICE );
984 newPart->AddDrawItem( newText );
985 }
986
987 SCH_COMMIT commit( m_toolMgr );
988
989 commit.Modify( symbol );
991
992 for( SCH_ITEM& item : symbol->GetDrawItems() )
993 item.ClearFlags( IS_NEW | IS_PASTED | SELECTED );
994
995 for( SCH_ITEM& item : newPart->GetDrawItems() )
996 {
997 if( item.Type() == SCH_FIELD_T )
998 continue;
999
1000 SCH_ITEM* newItem = item.Duplicate();
1001 newItem->SetParent( symbol );
1002 newItem->SetFlags( IS_NEW | IS_PASTED | SELECTED );
1003
1004 newItem->SetUnit( newItem->GetUnit() ? m_frame->GetUnit() : 0 );
1005 newItem->SetBodyStyle( newItem->GetBodyStyle() ? m_frame->GetBodyStyle() : 0 );
1006
1007 symbol->AddDrawItem( newItem );
1008 getView()->Add( newItem );
1009 }
1010
1011 delete newPart;
1012
1014
1016
1017 if( !selection.Empty() )
1018 {
1019 selection.SetReferencePoint( getViewControls()->GetCursorPosition( true ) );
1020
1022 commit.Push( _( "Paste" ) );
1023 else
1024 commit.Revert();
1025 }
1026
1027 return 0;
1028}
1029
1030
1032{
1033 LIB_SYMBOL* symbol = m_frame->GetCurSymbol();
1035 SCH_COMMIT commit( m_toolMgr );
1036
1037 if( selection.GetSize() == 0 )
1038 return 0;
1039
1040 commit.Modify( symbol, m_frame->GetScreen() );
1041
1042 std::vector<EDA_ITEM*> oldItems;
1043 std::vector<EDA_ITEM*> newItems;
1044
1045 std::copy( selection.begin(), selection.end(), std::back_inserter( oldItems ) );
1046 std::sort( oldItems.begin(), oldItems.end(), []( EDA_ITEM* a, EDA_ITEM* b )
1047 {
1048 int cmp;
1049
1050 if( a->Type() != b->Type() )
1051 return a->Type() < b->Type();
1052
1053 // Create the new pins in the same order as the old pins
1054 if( a->Type() == SCH_PIN_T )
1055 {
1056 const wxString& aNum = static_cast<SCH_PIN*>( a )->GetNumber();
1057 const wxString& bNum = static_cast<SCH_PIN*>( b )->GetNumber();
1058
1059 cmp = StrNumCmp( aNum, bNum );
1060
1061 // If the pin numbers are not numeric, then just number them by their position
1062 // on the screen.
1063 if( aNum.IsNumber() && bNum.IsNumber() && cmp != 0 )
1064 return cmp < 0;
1065 }
1066
1068
1069 if( cmp != 0 )
1070 return cmp < 0;
1071
1072 return a->m_Uuid < b->m_Uuid;
1073 } );
1074
1075 for( EDA_ITEM* item : oldItems )
1076 {
1077 SCH_ITEM* oldItem = static_cast<SCH_ITEM*>( item );
1078 SCH_ITEM* newItem = oldItem->Duplicate();
1079
1080 if( newItem->Type() == SCH_PIN_T )
1081 {
1082 SCH_PIN* newPin = static_cast<SCH_PIN*>( newItem );
1083
1084 if( !newPin->GetNumber().IsEmpty() )
1085 newPin->SetNumber( wxString::Format( wxT( "%i" ), symbol->GetMaxPinNumber() + 1 ) );
1086 }
1087
1088 oldItem->ClearFlags( IS_NEW | IS_PASTED | SELECTED );
1089 newItem->SetFlags( IS_NEW | IS_PASTED | SELECTED );
1090 newItem->SetParent( symbol );
1091 newItems.push_back( newItem );
1092
1093 symbol->AddDrawItem( newItem );
1094 getView()->Add( newItem );
1095 }
1096
1097 m_toolMgr->RunAction( ACTIONS::selectionClear );
1098 m_toolMgr->RunAction<EDA_ITEMS*>( ACTIONS::selectItems, &newItems );
1099
1100 selection.SetReferencePoint( getViewControls()->GetCursorPosition( true ) );
1101
1102 if( m_toolMgr->RunSynchronousAction( SCH_ACTIONS::move, &commit ) )
1103 commit.Push( _( "Duplicate" ) );
1104 else
1105 commit.Revert();
1106
1107 return 0;
1108}
1109
1110
1112{
1113 const ACTIONS::INCREMENT incParam = aEvent.Parameter<ACTIONS::INCREMENT>();
1115
1116 if( selection.Empty() )
1117 return 0;
1118
1119 KICAD_T type = selection.Front()->Type();
1120 bool allSameType = true;
1121 for( EDA_ITEM* item : selection )
1122 {
1123 if( item->Type() != type )
1124 {
1125 allSameType = false;
1126 break;
1127 }
1128 }
1129
1130 // Incrementing multiple types at once seems confusing
1131 // though it would work.
1132 if( !allSameType )
1133 return 0;
1134
1135 const VECTOR2I mousePosition = getViewControls()->GetMousePosition();
1136
1137 STRING_INCREMENTER incrementer;
1138 incrementer.SetSkipIOSQXZ( true );
1139
1140 // If we're coming via another action like 'Move', use that commit
1141 SCH_COMMIT localCommit( m_toolMgr );
1142 SCH_COMMIT* commit = dynamic_cast<SCH_COMMIT*>( aEvent.Commit() );
1143
1144 if( !commit )
1145 commit = &localCommit;
1146
1147 const auto modifyItem = [&]( EDA_ITEM& aItem )
1148 {
1149 if( aItem.IsNew() )
1151 else
1152 commit->Modify( &aItem, m_frame->GetScreen() );
1153 };
1154
1155 for( EDA_ITEM* item : selection )
1156 {
1157 switch( item->Type() )
1158 {
1159 case SCH_PIN_T:
1160 {
1161 SCH_PIN& pin = static_cast<SCH_PIN&>( *item );
1162 PIN_LAYOUT_CACHE& layout = pin.GetLayoutCache();
1163
1164 bool found = false;
1165 OPT_BOX2I bbox = layout.GetPinNumberBBox();
1166
1167 if( bbox && bbox->Contains( mousePosition ) )
1168 {
1169 std::optional<wxString> nextNumber =
1170 incrementer.Increment( pin.GetNumber(), incParam.Delta, incParam.Index );
1171 if( nextNumber )
1172 {
1173 modifyItem( pin );
1174 pin.SetNumber( *nextNumber );
1175 }
1176 found = true;
1177 }
1178
1179 if( !found )
1180 {
1181 bbox = layout.GetPinNameBBox();
1182
1183 if( bbox && bbox->Contains( mousePosition ) )
1184 {
1185 std::optional<wxString> nextName =
1186 incrementer.Increment( pin.GetName(), incParam.Delta, incParam.Index );
1187 if( nextName )
1188 {
1189 modifyItem( pin );
1190 pin.SetName( *nextName );
1191 }
1192 found = true;
1193 }
1194 }
1195 break;
1196 }
1197 case SCH_TEXT_T:
1198 {
1199 SCH_TEXT& label = static_cast<SCH_TEXT&>( *item );
1200
1201 std::optional<wxString> newLabel =
1202 incrementer.Increment( label.GetText(), incParam.Delta, incParam.Index );
1203 if( newLabel )
1204 {
1205 modifyItem( label );
1206 label.SetText( *newLabel );
1207 }
1208 break;
1209 }
1210 default:
1211 // No increment for other items
1212 break;
1213 }
1214 }
1215
1216 commit->Push( _( "Increment" ) );
1217
1218 return 0;
1219}
1220
1221
1223{
1224 // clang-format off
1232
1240
1246
1252 // clang-format on
1253}
std::optional< BOX2I > OPT_BOX2I
Definition: box2.h:926
static TOOL_ACTION decrementPrimary
Definition: actions.h:96
static TOOL_ACTION paste
Definition: actions.h:80
static TOOL_ACTION cancelInteractive
Definition: actions.h:72
static TOOL_ACTION unselectAll
Definition: actions.h:83
static TOOL_ACTION decrementSecondary
Definition: actions.h:98
static TOOL_ACTION copy
Definition: actions.h:78
static TOOL_ACTION pickerTool
Definition: actions.h:248
static TOOL_ACTION undo
Definition: actions.h:75
static TOOL_ACTION selectionActivate
Activation of the selection tool.
Definition: actions.h:214
static TOOL_ACTION incrementSecondary
Definition: actions.h:97
static TOOL_ACTION duplicate
Definition: actions.h:84
static TOOL_ACTION incrementPrimary
Definition: actions.h:95
static TOOL_ACTION doDelete
Definition: actions.h:85
static TOOL_ACTION redo
Definition: actions.h:76
static TOOL_ACTION deleteTool
Definition: actions.h:86
static TOOL_ACTION increment
Definition: actions.h:94
static TOOL_ACTION selectionClear
Clear the current selection.
Definition: actions.h:220
static TOOL_ACTION cut
Definition: actions.h:77
static TOOL_ACTION copyAsText
Definition: actions.h:79
static TOOL_ACTION refreshPreview
Definition: actions.h:156
static TOOL_ACTION selectAll
Definition: actions.h:82
static TOOL_ACTION selectItems
Select a list of items (specified as the event parameter)
Definition: actions.h:228
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
void AddItem(const TOOL_ACTION &aAction, const SELECTION_CONDITION &aCondition, int aOrder=ANY_ORDER)
Add a menu entry to run a TOOL_ACTION on selected items.
void AddSeparator(int aOrder=ANY_ORDER)
Add a separator to the menu.
This class is setup in expectation of its children possibly using Kiway player so DIALOG_SHIM::ShowQu...
void UpdateField(SCH_FIELD *aField)
const wxString & GetText() const
int ShowQuasiModal()
int ShowModal() override
Dialog to update or change schematic library symbols.
void ShowInfoBarError(const wxString &aErrorMsg, bool aShowCloseButton=false, WX_INFOBAR::MESSAGE_TYPE aType=WX_INFOBAR::MESSAGE_TYPE::GENERIC)
Show the WX_INFOBAR displayed on the top of the canvas with a message and an error icon on the left o...
void SetMsgPanel(const std::vector< MSG_PANEL_ITEM > &aList)
Clear the message panel and populates it with the contents of aList.
VECTOR2I GetNearestHalfGridPosition(const VECTOR2I &aPosition) const
Return the nearest aGridSize / 2 location to aPosition.
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:96
virtual VECTOR2I GetPosition() const
Definition: eda_item.h:252
virtual void SetPosition(const VECTOR2I &aPos)
Definition: eda_item.h:253
EDA_ITEM_FLAGS GetEditFlags() const
Definition: eda_item.h:141
void SetFlags(EDA_ITEM_FLAGS aMask)
Definition: eda_item.h:135
const KIID m_Uuid
Definition: eda_item.h:498
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:108
void ClearFlags(EDA_ITEM_FLAGS aMask=EDA_ITEM_ALL_FLAGS)
Definition: eda_item.h:137
virtual void SetParent(EDA_ITEM *aParent)
Definition: eda_item.h:111
EDA_ITEM * GetParent() const
Definition: eda_item.h:110
bool IsMoving() const
Definition: eda_item.h:118
virtual const wxString & GetText() const
Return the string associated with the text object.
Definition: eda_text.h:98
virtual bool IsVisible() const
Definition: eda_text.h:174
void SetVertJustify(GR_TEXT_V_ALIGN_T aType)
Definition: eda_text.cpp:410
GR_TEXT_H_ALIGN_T GetHorizJustify() const
Definition: eda_text.h:187
virtual void SetVisible(bool aVisible)
Definition: eda_text.cpp:379
GR_TEXT_V_ALIGN_T GetVertJustify() const
Definition: eda_text.h:190
virtual void SetText(const wxString &aText)
Definition: eda_text.cpp:270
void SetHorizJustify(GR_TEXT_H_ALIGN_T aType)
Definition: eda_text.cpp:402
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
Definition: ki_exception.h:77
virtual VECTOR2D GetMousePosition(bool aWorldCoordinates=true) const =0
Return the current mouse pointer position.
virtual void Add(VIEW_ITEM *aItem, int aDrawPriority=-1)
Add a VIEW_ITEM to the view.
Definition: view.cpp:297
Define a library symbol object.
Definition: lib_symbol.h:85
std::vector< SCH_PIN * > GetPins(int aUnit, int aBodyStyle) const
Return a list of pin object pointers from the draw item list.
Definition: lib_symbol.cpp:797
bool UnitsLocked() const
Check whether symbol units are interchangeable.
Definition: lib_symbol.h:289
bool IsDerived() const
Definition: lib_symbol.h:207
wxString GetUnitDisplayName(int aUnit) override
Return the user-defined display name for aUnit for symbols with units.
Definition: lib_symbol.cpp:298
LIB_ITEMS_CONTAINER & GetDrawItems()
Return a reference to the draw item list.
Definition: lib_symbol.h:519
void RemoveDrawItem(SCH_ITEM *aItem)
Remove draw aItem from list.
Definition: lib_symbol.cpp:758
bool HasUnitDisplayName(int aUnit) override
Return true if the given unit aUnit has a display name defined.
Definition: lib_symbol.cpp:292
wxString GetUnitReference(int aUnit) override
Return an identifier for aUnit for symbols with units.
Definition: lib_symbol.cpp:286
int GetUnitCount() const override
void SetUnitDisplayName(int aUnit, const wxString &aName)
Set the user-defined display name for aUnit to aName for symbols with units.
Definition: lib_symbol.cpp:314
void AddDrawItem(SCH_ITEM *aItem, bool aSort=true)
Add a new draw aItem to the draw object list and sort according to aSort.
Definition: lib_symbol.cpp:783
void SetMotionHandler(MOTION_HANDLER aHandler)
Set a handler for mouse motion.
Definition: picker_tool.h:84
void SetClickHandler(CLICK_HANDLER aHandler)
Set a handler for mouse click event.
Definition: picker_tool.h:73
void SetCursor(KICURSOR aCursor)
Definition: picker_tool.h:64
void SetFinalizeHandler(FINALIZE_HANDLER aHandler)
Set a handler for the finalize event.
Definition: picker_tool.h:104
A pin layout helper is a class that manages the layout of the parts of a pin on a schematic symbol:
OPT_BOX2I GetPinNumberBBox()
Get the bounding box of the pin number, if there is one.
OPT_BOX2I GetPinNameBBox()
Get the bounding box of the pin name, if there is one.
static TOOL_ACTION rotateCCW
Definition: sch_actions.h:119
static TOOL_ACTION mirrorV
Definition: sch_actions.h:120
static TOOL_ACTION swap
Definition: sch_actions.h:122
static TOOL_ACTION pinTable
Definition: sch_actions.h:150
static TOOL_ACTION properties
Definition: sch_actions.h:123
static TOOL_ACTION rotateCW
Definition: sch_actions.h:118
static TOOL_ACTION setUnitDisplayName
Definition: sch_actions.h:213
static TOOL_ACTION mirrorH
Definition: sch_actions.h:121
static TOOL_ACTION symbolProperties
Definition: sch_actions.h:149
static TOOL_ACTION updateSymbolFields
Definition: sch_actions.h:212
static TOOL_ACTION move
Definition: sch_actions.h:115
SCH_SCREEN * GetScreen() const override
Return a pointer to a BASE_SCREEN or one of its derivatives.
SCH_DRAW_PANEL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
virtual void Push(const wxString &aMessage=wxT("A commit"), int aCommitFlags=0) override
Execute the changes.
Definition: sch_commit.cpp:435
virtual void Revert() override
Revert the commit by restoring the modified items state.
Definition: sch_commit.cpp:513
bool IsMandatory() const
Definition: sch_field.cpp:1336
wxString GetFullText(int unit=1) const
Return the text of a field.
Definition: sch_field.cpp:306
wxString GetName(bool aUseDefaultName=true) const
Return the field name (not translated).
Definition: sch_field.cpp:1084
static void FormatLibSymbol(LIB_SYMBOL *aPart, OUTPUTFORMATTER &aFormatter)
static std::vector< LIB_SYMBOL * > ParseLibSymbols(std::string &aSymbolText, std::string aSource, int aFileVersion=SEXPR_SCHEMATIC_FILE_VERSION)
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition: sch_item.h:167
virtual void SetBodyStyle(int aBodyStyle)
Definition: sch_item.h:238
int GetBodyStyle() const
Definition: sch_item.h:239
virtual void MirrorHorizontally(int aCenter)
Mirror item horizontally about aCenter.
Definition: sch_item.h:355
int GetUnit() const
Definition: sch_item.h:236
virtual void Rotate(const VECTOR2I &aCenter, bool aRotateCCW)
Rotate the item around aCenter 90 degrees in the clockwise direction.
Definition: sch_item.h:372
virtual void SetUnit(int aUnit)
Definition: sch_item.h:235
wxString GetClass() const override
Return the class name.
Definition: sch_item.h:177
SCH_ITEM * Duplicate(bool doClone=false) const
Routine to create a new copy of given item.
Definition: sch_item.cpp:131
virtual void MirrorVertically(int aCenter)
Mirror item vertically about aCenter.
Definition: sch_item.h:364
void SetNumber(const wxString &aNumber)
Definition: sch_pin.cpp:545
void SetOrientation(PIN_ORIENTATION aOrientation)
Definition: sch_pin.h:92
const wxString & GetName() const
Definition: sch_pin.cpp:397
PIN_ORIENTATION GetOrientation() const
Definition: sch_pin.cpp:256
VECTOR2I GetPosition() const override
Definition: sch_pin.cpp:248
const wxString & GetNumber() const
Definition: sch_pin.h:123
ELECTRICAL_PINTYPE GetType() const
Definition: sch_pin.cpp:305
SCH_SELECTION & RequestSelection(const std::vector< KICAD_T > &aScanTypes={ SCH_LOCATE_ANY_T }, bool aPromoteCellSelections=false)
Return either an existing selection (filtered), or the selection at the current cursor position if th...
int ClearSelection(const TOOL_EVENT &aEvent)
Select all visible items in sheet.
bool CollectHits(SCH_COLLECTOR &aCollector, const VECTOR2I &aWhere, const std::vector< KICAD_T > &aScanTypes={ SCH_LOCATE_ANY_T })
Collect one or more items at a given point.
void GuessSelectionCandidates(SCH_COLLECTOR &collector, const VECTOR2I &aPos)
Apply heuristics to try and determine a single object when multiple are found under the cursor.
void RebuildSelection()
Rebuild the selection from the EDA_ITEMs' selection flags.
SCH_SELECTION & GetSelection()
bool Selectable(const EDA_ITEM *aItem, const VECTOR2I *aPos=nullptr, bool checkVisibilityOnly=false) const
Check conditions for an item to be selected.
void GetMsgPanelInfo(EDA_DRAW_FRAME *aFrame, std::vector< MSG_PANEL_ITEM > &aList) override
Populate aList of MSG_PANEL_ITEM objects with it's internal state for display purposes.
Definition: sch_shape.cpp:309
A foundation class for a tool operating on a schematic or symbol.
Definition: sch_tool_base.h:48
void updateItem(EDA_ITEM *aItem, bool aUpdateRTree) const
Similar to getView()->Update(), but handles items that are redrawn by their parents and updating the ...
void saveCopyInUndoList(EDA_ITEM *aItem, UNDO_REDO aType, bool aAppend=false, bool aDirtyConnectivity=true)
bool Init() override
Init() is called once upon a registration of the tool.
Definition: sch_tool_base.h:64
SCH_SELECTION_TOOL * m_selectionTool
static bool NotEmpty(const SELECTION &aSelection)
Test if there are any items selected.
static SELECTION_CONDITION MoreThan(int aNumber)
Create a functor that tests if the number of selected items is greater than the value given as parame...
static bool Idle(const SELECTION &aSelection)
Test if there no items selected or being edited.
static bool IdleSelection(const SELECTION &aSelection)
Test if all selected items are not being edited.
static SELECTION_CONDITION Count(int aNumber)
Create a functor that tests if the number of selected items is equal to the value given as parameter.
static SELECTION_CONDITION OnlyTypes(std::vector< KICAD_T > aTypes)
Create a functor that tests if the selected items are only of given types.
void BrightenItem(EDA_ITEM *aItem)
int AddItemToSel(const TOOL_EVENT &aEvent)
void UnbrightenItem(EDA_ITEM *aItem)
virtual KIGFX::VIEW_ITEM * GetItem(unsigned int aIdx) const override
Definition: selection.cpp:75
ITER end()
Definition: selection.h:75
const std::deque< EDA_ITEM * > GetItems() const
Definition: selection.h:121
ITER begin()
Definition: selection.h:74
virtual VECTOR2I GetCenter() const
Returns the center point of the selection area bounding box.
Definition: selection.cpp:92
bool IsHover() const
Definition: selection.h:84
virtual unsigned int GetSize() const override
Return the number of stored items.
Definition: selection.h:100
EDA_ITEM * Front() const
Definition: selection.h:172
virtual void Clear() override
Remove all the stored items from the group.
Definition: selection.h:93
int Size() const
Returns the number of selected parts.
Definition: selection.h:116
std::vector< EDA_ITEM * > GetItemsSortedBySelectionOrder() const
Definition: selection.cpp:263
void SetReferencePoint(const VECTOR2I &aP)
Definition: selection.cpp:178
bool Empty() const
Checks if there is anything selected.
Definition: selection.h:110
Implement an OUTPUTFORMATTER to a memory buffer.
Definition: richio.h:449
const std::string & GetString()
Definition: richio.h:472
Heuristically increment a string's n'th part from the right.
Definition: increment.h:48
void SetSkipIOSQXZ(bool aSkip)
If a alphabetic part is found, skip the letters I, O, S, Q, X, Z.
Definition: increment.h:54
std::optional< wxString > Increment(const wxString &aStr, int aDelta, size_t aRightIndex) const
Increment the n-th part from the right of the given string.
Definition: increment.cpp:86
SYMBOL_EDITOR_DRAWING_TOOLS.
void SetDrawSpecificBodyStyle(bool aSpecific)
int Undo(const TOOL_EVENT &aEvent)
void setTransitions() override
This method is meant to be overridden in order to specify handlers for events.
void editTextBoxProperties(SCH_ITEM *aItem)
int PinTable(const TOOL_EVENT &aEvent)
int Increment(const TOOL_EVENT &aEvent)
int Copy(const TOOL_EVENT &aEvent)
int CopyAsText(const TOOL_EVENT &aEvent)
int Paste(const TOOL_EVENT &aEvent)
int Cut(const TOOL_EVENT &aEvent)
bool Init() override
Init() is called once upon a registration of the tool.
int Redo(const TOOL_EVENT &aEvent)
void editTextProperties(SCH_ITEM *aItem)
int Swap(const TOOL_EVENT &aEvent)
void editFieldProperties(SCH_FIELD *aField)
void handlePinDuplication(SCH_PIN *aOldPin, SCH_PIN *aNewPin, int &aSymbolLastPinNumber)
Set up handlers for various events.
void editShapeProperties(SCH_SHAPE *aShape)
int Duplicate(const TOOL_EVENT &aEvent)
int Mirror(const TOOL_EVENT &aEvent)
int InteractiveDelete(const TOOL_EVENT &aEvent)
int Rotate(const TOOL_EVENT &aEvent)
int Properties(const TOOL_EVENT &aEvent)
int SetUnitDisplayName(const TOOL_EVENT &aEvent)
int UpdateSymbolFields(const TOOL_EVENT &aEvent)
int DoDelete(const TOOL_EVENT &aEvent)
Delete the selected items, or the item under the cursor.
void UpdateItem(EDA_ITEM *aItem, bool isAddOrDelete=false, bool aUpdateRtree=false) override
Mark an item for refresh.
bool IsSymbolAlias() const
Return true if aLibId is an alias for the editor screen symbol.
int GetBodyStyle() const
bool m_SyncPinEdit
Set to true to synchronize pins at the same position when editing symbols with multiple units or mult...
bool IsSymbolEditable() const
Test if a symbol is loaded and can be edited.
LIB_SYMBOL * GetCurSymbol() const
Return the current symbol being edited or NULL if none selected.
void UpdateSymbolMsgPanelInfo()
Display the documentation of the selected symbol.
void OnModify() override
Must be called after a schematic change in order to set the "modify" flag of the current symbol.
KIGFX::VIEW_CONTROLS * getViewControls() const
Return the instance of VIEW_CONTROLS object used in the application.
Definition: tool_base.cpp:44
TOOL_MANAGER * m_toolMgr
Definition: tool_base.h:220
KIGFX::VIEW * getView() const
Returns the instance of #VIEW object used in the application.
Definition: tool_base.cpp:38
Generic, UI-independent tool event.
Definition: tool_event.h:168
bool Matches(const TOOL_EVENT &aEvent) const
Test whether two events match in terms of category & action or command.
Definition: tool_event.h:389
COMMIT * Commit() const
Definition: tool_event.h:280
bool IsAction(const TOOL_ACTION *aAction) const
Test if the event contains an action issued upon activation of the given TOOL_ACTION.
Definition: tool_event.cpp:82
T Parameter() const
Return a parameter assigned to the event.
Definition: tool_event.h:465
void Go(int(T::*aStateFunc)(const TOOL_EVENT &), const TOOL_EVENT_LIST &aConditions=TOOL_EVENT(TC_ANY, TA_ANY))
Define which state (aStateFunc) to go when a certain event arrives (aConditions).
TOOL_MENU & GetToolMenu()
void Activate()
Run the tool.
bool RunAction(const std::string &aActionName, T aParam)
Run the specified action immediately, pausing the current action to run the new one.
Definition: tool_manager.h:150
VECTOR2D GetMousePosition() const
bool PostAction(const std::string &aActionName, T aParam)
Run the specified action after the current action (coroutine) ends.
Definition: tool_manager.h:235
bool RunSynchronousAction(const TOOL_ACTION &aAction, COMMIT *aCommit, T aParam)
Run the specified action immediately, pausing the current action to run the new one.
Definition: tool_manager.h:197
CONDITIONAL_MENU & GetMenu()
Definition: tool_menu.cpp:44
bool SaveClipboard(const std::string &aTextUTF8)
Store information to the system clipboard.
Definition: clipboard.cpp:31
std::string GetClipboardUTF8()
Return the information currently stored in the system clipboard.
Definition: clipboard.cpp:51
#define _(s)
std::vector< EDA_ITEM * > EDA_ITEMS
Define list of drawing items for screens.
Definition: eda_item.h:548
#define IS_PASTED
Modifier on IS_NEW which indicates it came from clipboard.
#define IS_NEW
New item, just created.
#define SELECTED
Item was manually selected by the user.
#define STRUCT_DELETED
flag indication structures to be erased
@ LAYER_DEVICE
Definition: layer_ids.h:455
void Prettify(std::string &aSource, bool aCompactSave)
PIN_ORIENTATION
The symbol library pin object orientations.
Definition: pin_type.h:80
const std::vector< KICAD_T > swappableItems
wxString GetSelectedItemsAsText(const SELECTION &aSel)
wxString TitleCaps(const wxString &aString)
Capitalize the first letter in each word.
wxString EscapeString(const wxString &aSource, ESCAPE_CONTEXT aContext)
The Escape/Unescape routines use HTML-entity-reference-style encoding to handle characters which are:...
@ CTX_LIBID
Definition: string_utils.h:54
static std::vector< KICAD_T > nonFields
const std::vector< KICAD_T > swappableItems
constexpr GR_TEXT_H_ALIGN_T GetFlippedAlignment(GR_TEXT_H_ALIGN_T aAlign)
Get the reverse alignment: left-right are swapped, others are unchanged.
KICAD_T
The set of class identification values stored in EDA_ITEM::m_structType.
Definition: typeinfo.h:78
@ SCH_TABLE_T
Definition: typeinfo.h:165
@ LIB_SYMBOL_T
Definition: typeinfo.h:148
@ SCH_TABLECELL_T
Definition: typeinfo.h:166
@ SCH_FIELD_T
Definition: typeinfo.h:150
@ SCH_SHAPE_T
Definition: typeinfo.h:149
@ SCH_TEXT_T
Definition: typeinfo.h:151
@ SCH_TEXTBOX_T
Definition: typeinfo.h:152
@ SCH_PIN_T
Definition: typeinfo.h:153
constexpr int LexicographicalCompare(const VECTOR2< T > &aA, const VECTOR2< T > &aB)
Definition: vector2d.h:644