KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
sch_group.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, you may find one here:
18 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
19 * or you may search the http://www.gnu.org website for the version 2 license,
20 * or you may write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
22 */
23#include <bitmaps.h>
24#include <eda_draw_frame.h>
25#include <eda_group.h>
27#include <sch_item.h>
28#include <sch_group.h>
29#include <sch_symbol.h>
30#include <symbol.h>
31#include <confirm.h>
32#include <widgets/msgpanel.h>
33#include <view/view.h>
34
35#include <wx/debug.h>
36
38{
39}
40
42{
43}
44
45
47{
48 switch( aType )
49 {
50 case SCH_SYMBOL_T:
51 case SCH_PIN_T:
52 case SCH_SHAPE_T:
53 case SCH_BITMAP_T:
54 case SCH_FIELD_T:
55 case SCH_TEXT_T:
56 case SCH_TEXTBOX_T:
57 case SCH_TABLE_T:
58 case SCH_GROUP_T:
59 case SCH_LINE_T:
60 case SCH_JUNCTION_T:
64 case SCH_LABEL_T:
67 case SCH_RULE_AREA_T:
69 case SCH_SHEET_PIN_T:
70 case SCH_SHEET_T:
71 return true;
72 default:
73 return false;
74 }
75}
76
77
79{
80 wxCHECK_MSG( aItem, false, wxT( "Nullptr added to group." ) );
81
82 wxCHECK_MSG( IsGroupableType( aItem->Type() ), false,
83 wxT( "Invalid item type added to group: " ) + aItem->GetTypeDesc() );
84
85 // Items can only be in one group at a time
86 if( aItem->GetParentGroup() )
87 aItem->GetParentGroup()->RemoveItem( aItem );
88
89 m_items.insert( aItem );
90 aItem->SetParentGroup( this );
91 return true;
92}
93
94
96{
97 wxCHECK_MSG( aItem, false, wxT( "Nullptr removed from group." ) );
98
99 // Only clear the item's group field if it was inside this group
100 if( m_items.erase( aItem ) == 1 )
101 {
102 aItem->SetParentGroup( nullptr );
103 return true;
104 }
105
106 return false;
107}
108
109
111{
112 for( EDA_ITEM* item : m_items )
113 item->SetParentGroup( nullptr );
114
115 m_items.clear();
116}
117
118
119std::unordered_set<SCH_ITEM*> SCH_GROUP::GetSchItems() const
120{
121 std::unordered_set<SCH_ITEM*> items;
122
123 for( EDA_ITEM* item : m_items )
124 {
125 SCH_ITEM* sch_item = dynamic_cast<SCH_ITEM*>( item );
126
127 if( sch_item )
128 {
129 items.insert( sch_item );
130 }
131 }
132
133 return items;
134}
135
136
137/*
138 * @return if not in the symbol editor and aItem is in a symbol, returns the
139 * symbol's parent group. Otherwise, returns the aItem's parent group.
140 */
141EDA_GROUP* getClosestGroup( SCH_ITEM* aItem, bool isSymbolEditor )
142{
143 if( !isSymbolEditor && aItem->GetParent() && aItem->GetParent()->Type() == SCH_SYMBOL_T )
144 return static_cast<SCH_SYMBOL*>( aItem->GetParent() )->GetParentGroup();
145 else
146 return aItem->GetParentGroup();
147}
148
149
151EDA_GROUP* getNestedGroup( SCH_ITEM* aItem, EDA_GROUP* aScope, bool isSymbolEditor )
152{
153 EDA_GROUP* group = getClosestGroup( aItem, isSymbolEditor );
154
155 if( group == aScope )
156 return nullptr;
157
158 while( group && group->AsEdaItem()->GetParentGroup() && group->AsEdaItem()->GetParentGroup() != aScope )
159 {
160 if( group->AsEdaItem()->GetParent()->Type() == LIB_SYMBOL_T && isSymbolEditor )
161 break;
162
163 group = group->AsEdaItem()->GetParentGroup();
164 }
165
166 return group;
167}
168
169
170EDA_GROUP* SCH_GROUP::TopLevelGroup( SCH_ITEM* aItem, EDA_GROUP* aScope, bool isSymbolEditor )
171{
172 return getNestedGroup( aItem, aScope, isSymbolEditor );
173}
174
175
176bool SCH_GROUP::WithinScope( SCH_ITEM* aItem, SCH_GROUP* aScope, bool isSymbolEditor )
177{
178 EDA_GROUP* group = getClosestGroup( aItem, isSymbolEditor );
179
180 if( group && group == aScope )
181 return true;
182
183 EDA_GROUP* nested = getNestedGroup( aItem, aScope, isSymbolEditor );
184
185 return nested && nested->AsEdaItem()->GetParentGroup() && nested->AsEdaItem()->GetParentGroup() == aScope;
186}
187
188
190{
191 return GetBoundingBox().Centre();
192}
193
194
195void SCH_GROUP::SetPosition( const VECTOR2I& aNewpos )
196{
197 VECTOR2I delta = aNewpos - GetPosition();
198
199 Move( delta );
200}
201
202
204{
205 // Use copy constructor to get the same uuid and other fields
206 SCH_GROUP* newGroup = new SCH_GROUP( *this );
207 return newGroup;
208}
209
210
212{
213 // Use copy constructor to get the same uuid and other fields
214 SCH_GROUP* newGroup = new SCH_GROUP( *this );
215 newGroup->m_items.clear();
216
217 for( EDA_ITEM* member : m_items )
218 {
219 if( member->Type() == SCH_GROUP_T )
220 newGroup->AddItem( static_cast<SCH_GROUP*>( member )->DeepClone() );
221 else
222 newGroup->AddItem( static_cast<SCH_ITEM*>( member->Clone() ) );
223 }
224
225 return newGroup;
226}
227
228
230{
231 SCH_GROUP* newGroup = static_cast<SCH_GROUP*>( Duplicate() );
232 newGroup->m_items.clear();
233
234 for( EDA_ITEM* member : m_items )
235 {
236 if( member->Type() == SCH_GROUP_T )
237 newGroup->AddItem( static_cast<SCH_GROUP*>( member )->DeepDuplicate() );
238 else
239 newGroup->AddItem( static_cast<SCH_ITEM*>( member )->Duplicate() );
240 }
241
242 return newGroup;
243}
244
245
247{
248 assert( aImage->Type() == SCH_GROUP_T );
249
250 std::swap( *( (SCH_GROUP*) this ), *( (SCH_GROUP*) aImage ) );
251}
252
253
254bool SCH_GROUP::HitTest( const VECTOR2I& aPosition, int aAccuracy ) const
255{
256 // Groups are selected by promoting a selection of one of their children
257 return false;
258}
259
260
261bool SCH_GROUP::HitTest( const BOX2I& aRect, bool aContained, int aAccuracy ) const
262{
263 // Groups are selected by promoting a selection of one of their children
264 return false;
265}
266
267
269{
270 BOX2I bbox;
271
272 for( EDA_ITEM* item : m_items )
273 {
274 if( item->Type() == SCH_SYMBOL_T || item->Type() == LIB_SYMBOL_T )
275 bbox.Merge( static_cast<SYMBOL*>( item )->GetBoundingBox() );
276 else
277 bbox.Merge( item->GetBoundingBox() );
278 }
279
280 bbox.Inflate( schIUScale.MilsToIU( 5 ) ); // Give a min size to the bbox
281
282 return bbox;
283}
284
285
286INSPECT_RESULT SCH_GROUP::Visit( INSPECTOR aInspector, void* aTestData,
287 const std::vector<KICAD_T>& aScanTypes )
288{
289 for( KICAD_T scanType : aScanTypes )
290 {
291 if( scanType == Type() )
292 {
293 if( INSPECT_RESULT::QUIT == aInspector( this, aTestData ) )
294 return INSPECT_RESULT::QUIT;
295 }
296 }
297
298 return INSPECT_RESULT::CONTINUE;
299}
300
301
302std::vector<int> SCH_GROUP::ViewGetLayers() const
303{
304 return { LAYER_SCHEMATIC_ANCHOR };
305}
306
307
308double SCH_GROUP::ViewGetLOD( int aLayer, const KIGFX::VIEW* aView ) const
309{
311 return LOD_SHOW;
312
313 return LOD_HIDE;
314}
315
316
317void SCH_GROUP::Move( const VECTOR2I& aMoveVector )
318{
319 for( EDA_ITEM* member : m_items )
320 static_cast<SCH_ITEM*>( member )->Move( aMoveVector );
321}
322
323
324void SCH_GROUP::Rotate( const VECTOR2I& aCenter, bool aRotateCCW )
325{
326 for( EDA_ITEM* member : m_items )
327 static_cast<SCH_ITEM*>( member )->Rotate( aCenter, aRotateCCW );
328}
329
330
332{
333 for( EDA_ITEM* item : m_items )
334 static_cast<SCH_ITEM*>( item )->MirrorHorizontally( aCenter );
335}
336
337
339{
340 for( EDA_ITEM* item : m_items )
341 static_cast<SCH_ITEM*>( item )->MirrorVertically( aCenter );
342}
343
344
345wxString SCH_GROUP::GetItemDescription( UNITS_PROVIDER* aUnitsProvider, bool aFull ) const
346{
347 if( m_name.empty() )
348 return wxString::Format( _( "Unnamed Group, %zu members" ), m_items.size() );
349 else
350 return wxString::Format( _( "Group '%s', %zu members" ), m_name, m_items.size() );
351}
352
353
355{
356 return BITMAPS::module;
357}
358
359
360void SCH_GROUP::GetMsgPanelInfo( EDA_DRAW_FRAME* aFrame, std::vector<MSG_PANEL_ITEM>& aList )
361{
362 aList.emplace_back( _( "Group" ), m_name.empty() ? _( "<unnamed>" ) : m_name );
363 aList.emplace_back( _( "Members" ), wxString::Format( wxT( "%zu" ), m_items.size() ) );
364}
365
366
367void SCH_GROUP::RunOnChildren( const std::function<void( SCH_ITEM* )>& aFunction, RECURSE_MODE aMode )
368{
369 try
370 {
371 for( EDA_ITEM* item : m_items )
372 {
373 aFunction( static_cast<SCH_ITEM*>( item ) );
374
375 if( item->Type() == SCH_GROUP_T )
376 static_cast<SCH_ITEM*>( item )->RunOnChildren( aFunction, RECURSE_MODE::RECURSE );
377 }
378 }
379 catch( std::bad_function_call& )
380 {
381 wxFAIL_MSG( wxT( "Error calling function in SCH_GROUP::RunOnChildren" ) );
382 }
383}
384
385
386bool SCH_GROUP::operator==( const SCH_ITEM& aSchItem ) const
387{
388 if( aSchItem.Type() != Type() )
389 return false;
390
391 const SCH_GROUP& other = static_cast<const SCH_GROUP&>( aSchItem );
392
393 return *this == other;
394}
395
396
397bool SCH_GROUP::operator==( const SCH_GROUP& aOther ) const
398{
399 if( m_items.size() != aOther.m_items.size() )
400 return false;
401
402 // The items in groups are in unordered sets hashed by the pointer value, so we need to
403 // order them by UUID (EDA_ITEM_SET) to compare
404 EDA_ITEM_SET itemSet( m_items.begin(), m_items.end() );
405 EDA_ITEM_SET otherItemSet( aOther.m_items.begin(), aOther.m_items.end() );
406
407 for( auto it1 = itemSet.begin(), it2 = otherItemSet.begin(); it1 != itemSet.end(); ++it1, ++it2 )
408 {
409 // Compare UUID instead of the items themselves because we only care if the contents
410 // of the group has changed, not which elements in the group have changed
411 if( ( *it1 )->m_Uuid != ( *it2 )->m_Uuid )
412 return false;
413 }
414
415 return true;
416}
417
418
419double SCH_GROUP::Similarity( const SCH_ITEM& aOther ) const
420{
421 if( aOther.Type() != Type() )
422 return 0.0;
423
424 const SCH_GROUP& other = static_cast<const SCH_GROUP&>( aOther );
425
426 double similarity = 0.0;
427
428 for( EDA_ITEM* item : m_items )
429 {
430 for( EDA_ITEM* otherItem : other.m_items )
431 {
432 similarity += static_cast<SCH_ITEM*>( item )->Similarity( *static_cast<SCH_ITEM*>( otherItem ) );
433 }
434 }
435
436 return similarity / m_items.size();
437}
438
439
440static struct SCH_GROUP_DESC
441{
443 {
450
451 propMgr.Mask( TYPE_HASH( SCH_GROUP ), TYPE_HASH( SCH_ITEM ), _HKI( "Position X" ) );
452 propMgr.Mask( TYPE_HASH( SCH_GROUP ), TYPE_HASH( SCH_ITEM ), _HKI( "Position Y" ) );
453
454 const wxString groupTab = _HKI( "Group Properties" );
455
456 propMgr.AddProperty(
458 groupTab );
459 }
constexpr EDA_IU_SCALE schIUScale
Definition: base_units.h:110
BITMAPS
A list of all bitmap identifiers.
Definition: bitmaps_list.h:33
constexpr BOX2< Vec > & Inflate(coord_type dx, coord_type dy)
Inflates the rectangle horizontally by dx and vertically by dy.
Definition: box2.h:558
constexpr Vec Centre() const
Definition: box2.h:97
constexpr BOX2< Vec > & Merge(const BOX2< Vec > &aRect)
Modify the position and size of the rectangle in order to contain aRect.
Definition: box2.h:658
The base class for create windows for drawing purpose.
A set of EDA_ITEMs (i.e., without duplicates).
Definition: eda_group.h:45
wxString m_name
Definition: eda_group.h:92
std::unordered_set< EDA_ITEM * > m_items
Check if the proposed type can be added to a group.
Definition: eda_group.h:91
wxString GetName() const
Definition: eda_group.h:50
virtual bool RemoveItem(EDA_ITEM *aItem)=0
Remove item from group.
virtual EDA_ITEM * AsEdaItem()=0
void SetName(const wxString &aName)
Definition: eda_group.h:51
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:96
wxString GetTypeDesc() const
Return a translated description of the type for this EDA_ITEM for display in user facing messages.
Definition: eda_item.cpp:365
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:108
virtual void SetParentGroup(EDA_GROUP *aGroup)
Definition: eda_item.h:113
EDA_GROUP * GetParentGroup() const
Definition: eda_item.h:114
EDA_ITEM * GetParent() const
Definition: eda_item.h:110
static constexpr double LOD_HIDE
Return this constant from ViewGetLOD() to hide the item unconditionally.
Definition: view_item.h:174
static constexpr double LOD_SHOW
Return this constant from ViewGetLOD() to show the item unconditionally.
Definition: view_item.h:179
Hold a (potentially large) number of VIEW_ITEMs and renders them on a graphics device provided by the...
Definition: view.h:67
bool IsLayerVisible(int aLayer) const
Return information about visibility of a particular layer.
Definition: view.h:418
Provide class metadata.Helper macro to map type hashes to names.
Definition: property_mgr.h:85
void InheritsAfter(TYPE_ID aDerived, TYPE_ID aBase)
Declare an inheritance relationship between types.
void Mask(TYPE_ID aDerived, TYPE_ID aBase, const wxString &aName)
Sets a base class property as masked in a derived class.
static PROPERTY_MANAGER & Instance()
Definition: property_mgr.h:87
PROPERTY_BASE & AddProperty(PROPERTY_BASE *aProperty, const wxString &aGroup=wxEmptyString)
Register a property.
void AddTypeCast(TYPE_CAST_BASE *aCast)
Register a type converter.
A set of SCH_ITEMs (i.e., without duplicates).
Definition: sch_group.h:52
SCH_GROUP * DeepDuplicate() const override
Definition: sch_group.cpp:229
void Move(const VECTOR2I &aMoveVector) override
Move the item by aMoveVector to a new position.
Definition: sch_group.cpp:317
std::vector< int > ViewGetLayers() const override
Definition: sch_group.cpp:302
BITMAPS GetMenuImage() const override
Return a pointer to an image to be used in menus.
Definition: sch_group.cpp:354
void Rotate(const VECTOR2I &aCenter, bool aRotateCCW) override
Rotate the item around aCenter 90 degrees in the clockwise direction.
Definition: sch_group.cpp:324
double Similarity(const SCH_ITEM &aOther) const override
Return a measure of how likely the other object is to represent the same object.
Definition: sch_group.cpp:419
EDA_ITEM * Clone() const override
Create a duplicate of this item with linked list members set to NULL.
Definition: sch_group.cpp:203
bool AddItem(EDA_ITEM *aItem) override
Add item to group.
Definition: sch_group.cpp:78
static bool WithinScope(SCH_ITEM *aItem, SCH_GROUP *aScope, bool isSymbolEditor)
Definition: sch_group.cpp:176
bool operator==(const SCH_GROUP &aOther) const
Definition: sch_group.cpp:397
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_group.cpp:360
wxString GetItemDescription(UNITS_PROVIDER *aUnitsProvider, bool aFull) const override
Return a user-visible description string of this item.
Definition: sch_group.cpp:345
double ViewGetLOD(int aLayer, const KIGFX::VIEW *aView) const override
Definition: sch_group.cpp:308
INSPECT_RESULT Visit(INSPECTOR aInspector, void *aTestData, const std::vector< KICAD_T > &aScanTypes) override
May be re-implemented for each derived class in order to handle all the types given by its member dat...
Definition: sch_group.cpp:286
void MirrorVertically(int aCenter) override
Mirror item vertically about aCenter.
Definition: sch_group.cpp:338
void RunOnChildren(const std::function< void(SCH_ITEM *)> &aFunction, RECURSE_MODE aMode) override
Definition: sch_group.cpp:367
void SetPosition(const VECTOR2I &aNewpos) override
Definition: sch_group.cpp:195
std::unordered_set< SCH_ITEM * > GetSchItems() const
Definition: sch_group.cpp:119
static EDA_GROUP * TopLevelGroup(SCH_ITEM *aItem, EDA_GROUP *aScope, bool isSymbolEditor)
Definition: sch_group.cpp:170
void swapData(SCH_ITEM *aImage) override
Swap the internal data structures aItem with the schematic item.
Definition: sch_group.cpp:246
void MirrorHorizontally(int aCenter) override
Mirror item horizontally about aCenter.
Definition: sch_group.cpp:331
bool HitTest(const VECTOR2I &aPosition, int aAccuracy=0) const override
Test if aPosition is inside or on the boundary of this item.
Definition: sch_group.cpp:254
bool RemoveItem(EDA_ITEM *aItem) override
Remove item from group.
Definition: sch_group.cpp:95
const BOX2I GetBoundingBox() const override
Return the orthogonal bounding box of this object for display purposes.
Definition: sch_group.cpp:268
SCH_GROUP * DeepClone() const override
Definition: sch_group.cpp:211
VECTOR2I GetPosition() const override
Definition: sch_group.cpp:189
void RemoveAll() override
Definition: sch_group.cpp:110
static bool IsGroupableType(KICAD_T aType)
Check if the proposed type can be added to a group.
Definition: sch_group.cpp:46
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition: sch_item.h:167
SCH_ITEM * Duplicate(bool doClone=false) const
Routine to create a new copy of given item.
Definition: sch_item.cpp:131
Schematic symbol object.
Definition: sch_symbol.h:75
A base class for LIB_SYMBOL and SCH_SYMBOL.
Definition: symbol.h:63
This file is part of the common library.
#define _HKI(x)
#define _(s)
RECURSE_MODE
Definition: eda_item.h:49
INSPECT_RESULT
Definition: eda_item.h:43
std::set< EDA_ITEM *, CompareByUuid > EDA_ITEM_SET
Definition: eda_item.h:550
const INSPECTOR_FUNC & INSPECTOR
std::function passed to nested users by ref, avoids copying std::function.
Definition: eda_item.h:89
@ LAYER_SCHEMATIC_ANCHOR
Definition: layer_ids.h:488
Message panel definition file.
#define TYPE_HASH(x)
Definition: property.h:71
#define REGISTER_TYPE(x)
Definition: property_mgr.h:371
static struct SCH_GROUP_DESC _SCH_GROUP_DESC
EDA_GROUP * getNestedGroup(SCH_ITEM *aItem, EDA_GROUP *aScope, bool isSymbolEditor)
Returns the top level group inside the aScope group, or nullptr.
Definition: sch_group.cpp:151
EDA_GROUP * getClosestGroup(SCH_ITEM *aItem, bool isSymbolEditor)
Definition: sch_group.cpp:141
Class to handle a set of SCH_ITEMs.
constexpr int MilsToIU(int mils) const
Definition: base_units.h:93
constexpr int delta
KICAD_T
The set of class identification values stored in EDA_ITEM::m_structType.
Definition: typeinfo.h:78
@ SCH_GROUP_T
Definition: typeinfo.h:173
@ SCH_TABLE_T
Definition: typeinfo.h:165
@ SCH_LINE_T
Definition: typeinfo.h:163
@ LIB_SYMBOL_T
Definition: typeinfo.h:148
@ SCH_NO_CONNECT_T
Definition: typeinfo.h:160
@ SCH_SYMBOL_T
Definition: typeinfo.h:172
@ SCH_FIELD_T
Definition: typeinfo.h:150
@ SCH_DIRECTIVE_LABEL_T
Definition: typeinfo.h:171
@ SCH_LABEL_T
Definition: typeinfo.h:167
@ SCH_SHEET_T
Definition: typeinfo.h:175
@ SCH_SHAPE_T
Definition: typeinfo.h:149
@ SCH_RULE_AREA_T
Definition: typeinfo.h:170
@ SCH_HIER_LABEL_T
Definition: typeinfo.h:169
@ SCH_BUS_BUS_ENTRY_T
Definition: typeinfo.h:162
@ SCH_SHEET_PIN_T
Definition: typeinfo.h:174
@ SCH_TEXT_T
Definition: typeinfo.h:151
@ SCH_BUS_WIRE_ENTRY_T
Definition: typeinfo.h:161
@ SCH_BITMAP_T
Definition: typeinfo.h:164
@ SCH_TEXTBOX_T
Definition: typeinfo.h:152
@ SCH_GLOBAL_LABEL_T
Definition: typeinfo.h:168
@ SCH_JUNCTION_T
Definition: typeinfo.h:159
@ SCH_PIN_T
Definition: typeinfo.h:153