KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
bus-wire-junction.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2004 Jean-Pierre Charras, jean-pierre.charras@gipsa-lab.inpg.fr
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
25#include <core/kicad_algo.h>
26#include <general.h>
27#include <sch_bus_entry.h>
28#include <sch_edit_frame.h>
29#include <sch_junction.h>
30#include <sch_line.h>
31#include <sch_no_connect.h>
32#include <sch_commit.h>
33#include <tool/tool_manager.h>
35#include <trigo.h>
36
37
39{
40 std::function<void( SCH_ITEM* )> changeHandler =
41 [&]( SCH_ITEM* aChangedItem ) -> void
42 {
43 GetCanvas()->GetView()->Update( aChangedItem, KIGFX::REPAINT );
44 };
45
46 GetScreen()->TestDanglingEnds( nullptr, &changeHandler );
47}
48
49
50bool SCH_EDIT_FRAME::TrimWire( SCH_COMMIT* aCommit, const VECTOR2I& aStart, const VECTOR2I& aEnd )
51{
52 if( aStart == aEnd )
53 return false;
54
55 SCH_SCREEN* screen = GetScreen();
56 std::vector<SCH_LINE*> wires;
57 BOX2I bb( aStart );
58
59 bb.Merge( aEnd );
60
61 // We cannot modify the RTree while iterating, so push the possible
62 // wires into a separate structure.
63 for( EDA_ITEM* item : screen->Items().Overlapping( bb ) )
64 {
65 SCH_LINE* line = static_cast<SCH_LINE*>( item );
66
67 if( item->Type() == SCH_LINE_T && line->GetLayer() == LAYER_WIRE )
68 wires.push_back( line );
69 }
70
71 for( SCH_LINE* line : wires )
72 {
73 // Don't remove wires that are already deleted or are currently being dragged
74 if( line->GetEditFlags() & ( STRUCT_DELETED | IS_MOVING | SKIP_STRUCT ) )
75 continue;
76
77 if( !IsPointOnSegment( line->GetStartPoint(), line->GetEndPoint(), aStart ) ||
78 !IsPointOnSegment( line->GetStartPoint(), line->GetEndPoint(), aEnd ) )
79 {
80 continue;
81 }
82
83 // Don't remove entire wires
84 if( ( line->GetStartPoint() == aStart && line->GetEndPoint() == aEnd )
85 || ( line->GetStartPoint() == aEnd && line->GetEndPoint() == aStart ) )
86 {
87 continue;
88 }
89
90 // Step 1: break the segment on one end.
91 // Ensure that *line points to the segment containing aEnd
92 SCH_LINE* new_line;
93 BreakSegment( aCommit, line, aStart, &new_line, screen );
94
95 if( IsPointOnSegment( new_line->GetStartPoint(), new_line->GetEndPoint(), aEnd ) )
96 line = new_line;
97
98 // Step 2: break the remaining segment.
99 // Ensure that *line _also_ contains aStart. This is our overlapping segment
100 BreakSegment( aCommit, line, aEnd, &new_line, screen );
101
102 if( IsPointOnSegment( new_line->GetStartPoint(), new_line->GetEndPoint(), aStart ) )
103 line = new_line;
104
105 RemoveFromScreen( line, screen );
106 aCommit->Removed( line, screen );
107
108 return true;
109 }
110
111 return false;
112}
113
114
116{
118 std::vector<SCH_LINE*> lines;
119 std::vector<SCH_JUNCTION*> junctions;
120 std::vector<SCH_NO_CONNECT*> ncs;
121 std::vector<SCH_ITEM*> items_to_remove;
122 bool changed = true;
123
124 if( aScreen == nullptr )
125 aScreen = GetScreen();
126
127 auto remove_item = [&]( SCH_ITEM* aItem ) -> void
128 {
129 changed = true;
130
131 if( !( aItem->GetFlags() & STRUCT_DELETED ) )
132 {
133 aItem->SetFlags( STRUCT_DELETED );
134
135 if( aItem->IsSelected() )
136 selectionTool->RemoveItemFromSel( aItem, true /*quiet mode*/ );
137
138 RemoveFromScreen( aItem, aScreen );
139 aCommit->Removed( aItem, aScreen );
140 }
141 };
142
143 BreakSegmentsOnJunctions( aCommit, aScreen );
144
145 for( SCH_ITEM* item : aScreen->Items().OfType( SCH_JUNCTION_T ) )
146 {
147 if( !aScreen->IsExplicitJunction( item->GetPosition() ) )
148 items_to_remove.push_back( item );
149 else
150 junctions.push_back( static_cast<SCH_JUNCTION*>( item ) );
151 }
152
153 for( SCH_ITEM* item : items_to_remove )
154 remove_item( item );
155
156 for( SCH_ITEM* item : aScreen->Items().OfType( SCH_NO_CONNECT_T ) )
157 ncs.push_back( static_cast<SCH_NO_CONNECT*>( item ) );
158
159 alg::for_all_pairs( junctions.begin(), junctions.end(),
160 [&]( SCH_JUNCTION* aFirst, SCH_JUNCTION* aSecond )
161 {
162 if( ( aFirst->GetEditFlags() & STRUCT_DELETED )
163 || ( aSecond->GetEditFlags() & STRUCT_DELETED ) )
164 {
165 return;
166 }
167
168 if( aFirst->GetPosition() == aSecond->GetPosition() )
169 remove_item( aSecond );
170 } );
171
172 alg::for_all_pairs( ncs.begin(), ncs.end(),
173 [&]( SCH_NO_CONNECT* aFirst, SCH_NO_CONNECT* aSecond )
174 {
175 if( ( aFirst->GetEditFlags() & STRUCT_DELETED )
176 || ( aSecond->GetEditFlags() & STRUCT_DELETED ) )
177 {
178 return;
179 }
180
181 if( aFirst->GetPosition() == aSecond->GetPosition() )
182 remove_item( aSecond );
183 } );
184
185
186 auto minX = []( const SCH_LINE* l )
187 {
188 return std::min( l->GetStartPoint().x, l->GetEndPoint().x );
189 };
190
191 auto maxX = []( const SCH_LINE* l )
192 {
193 return std::max( l->GetStartPoint().x, l->GetEndPoint().x );
194 };
195
196 auto minY = []( const SCH_LINE* l )
197 {
198 return std::min( l->GetStartPoint().y, l->GetEndPoint().y );
199 };
200
201 auto maxY = []( const SCH_LINE* l )
202 {
203 return std::max( l->GetStartPoint().y, l->GetEndPoint().y );
204 };
205
206 // Would be nice to put lines in a canonical form here by swapping
207 // start <-> end as needed but I don't know what swapping breaks.
208 while( changed )
209 {
210 changed = false;
211 lines.clear();
212
213 for( SCH_ITEM* item : aScreen->Items().OfType( SCH_LINE_T ) )
214 {
215 if( item->GetLayer() == LAYER_WIRE || item->GetLayer() == LAYER_BUS )
216 lines.push_back( static_cast<SCH_LINE*>( item ) );
217 }
218
219 // Sort by minimum X position
220 std::sort( lines.begin(), lines.end(),
221 [&]( const SCH_LINE* a, const SCH_LINE* b )
222 {
223 return minX( a ) < minX( b );
224 } );
225
226 for( auto it1 = lines.begin(); it1 != lines.end(); ++it1 )
227 {
228 SCH_LINE* firstLine = *it1;
229
230 if( firstLine->GetEditFlags() & STRUCT_DELETED )
231 continue;
232
233 if( firstLine->IsNull() )
234 {
235 remove_item( firstLine );
236 continue;
237 }
238
239 int firstRightXEdge = maxX( firstLine );
240 auto it2 = it1;
241
242 for( ++it2; it2 != lines.end(); ++it2 )
243 {
244 SCH_LINE* secondLine = *it2;
245 int secondLeftXEdge = minX( secondLine );
246
247 // impossible to overlap remaining lines
248 if( secondLeftXEdge > firstRightXEdge )
249 break;
250
251 // No Y axis overlap
252 if( !( std::max( minY( firstLine ), minY( secondLine ) )
253 <= std::min( maxY( firstLine ), maxY( secondLine ) ) ) )
254 {
255 continue;
256 }
257
258 if( secondLine->GetFlags() & STRUCT_DELETED )
259 continue;
260
261 if( !secondLine->IsParallel( firstLine )
262 || !secondLine->IsStrokeEquivalent( firstLine )
263 || secondLine->GetLayer() != firstLine->GetLayer() )
264 {
265 continue;
266 }
267
268 // Remove identical lines
269 if( firstLine->IsEndPoint( secondLine->GetStartPoint() )
270 && firstLine->IsEndPoint( secondLine->GetEndPoint() ) )
271 {
272 remove_item( secondLine );
273 continue;
274 }
275
276 // See if we can merge an overlap (or two colinear touching segments with
277 // no junction where they meet).
278 SCH_LINE* mergedLine = secondLine->MergeOverlap( aScreen, firstLine, true );
279
280 if( mergedLine != nullptr )
281 {
282 remove_item( firstLine );
283 remove_item( secondLine );
284
285 AddToScreen( mergedLine, aScreen );
286 aCommit->Added( mergedLine, aScreen );
287
288 if( firstLine->IsSelected() || secondLine->IsSelected() )
289 selectionTool->AddItemToSel( mergedLine, true /*quiet mode*/ );
290
291 break;
292 }
293 }
294 }
295 }
296}
297
298
299void SCH_EDIT_FRAME::BreakSegment( SCH_COMMIT* aCommit, SCH_LINE* aSegment, const VECTOR2I& aPoint,
300 SCH_LINE** aNewSegment, SCH_SCREEN* aScreen )
301{
302 // Save the copy of aSegment before breaking it
303 aCommit->Modify( aSegment, aScreen );
304
305 SCH_LINE* newSegment = aSegment->BreakAt( aPoint );
306
307 aSegment->SetFlags( IS_CHANGED | IS_BROKEN );
308 newSegment->SetFlags( IS_NEW | IS_BROKEN );
309
310 AddToScreen( newSegment, aScreen );
311 aCommit->Added( newSegment, aScreen );
312
313 *aNewSegment = newSegment;
314}
315
316
317bool SCH_EDIT_FRAME::BreakSegments( SCH_COMMIT* aCommit, const VECTOR2I& aPos, SCH_SCREEN* aScreen )
318{
319 bool brokenSegments = false;
320 SCH_LINE* new_line;
321
322 for( SCH_LINE* wire : aScreen->GetBusesAndWires( aPos, true ) )
323 {
324 BreakSegment( aCommit, wire, aPos, &new_line, aScreen );
325 brokenSegments = true;
326 }
327
328 return brokenSegments;
329}
330
331
333{
334 bool brokenSegments = false;
335
336 std::set<VECTOR2I> point_set;
337
338 for( SCH_ITEM* item : aScreen->Items().OfType( SCH_JUNCTION_T ) )
339 point_set.insert( item->GetPosition() );
340
341 for( SCH_ITEM* item : aScreen->Items().OfType( SCH_BUS_WIRE_ENTRY_T ) )
342 {
343 SCH_BUS_WIRE_ENTRY* entry = static_cast<SCH_BUS_WIRE_ENTRY*>( item );
344 point_set.insert( entry->GetPosition() );
345 point_set.insert( entry->GetEnd() );
346 }
347
348 for( const VECTOR2I& pt : point_set )
349 {
350 BreakSegments( aCommit, pt, aScreen );
351 brokenSegments = true;
352 }
353
354 return brokenSegments;
355}
356
357
359{
360 SCH_SCREEN* screen = GetScreen();
361 PICKED_ITEMS_LIST undoList;
363
364 aJunction->SetFlags( STRUCT_DELETED );
365 RemoveFromScreen( aJunction, screen );
366 aCommit->Removed( aJunction, screen );
367
370 std::list<SCH_LINE*> lines;
371
372 for( SCH_ITEM* item : screen->Items().Overlapping( SCH_LINE_T, aJunction->GetPosition() ) )
373 {
374 SCH_LINE* line = static_cast<SCH_LINE*>( item );
375
376 if( ( line->IsWire() || line->IsBus() )
377 && line->IsEndPoint( aJunction->GetPosition() )
378 && !( line->GetEditFlags() & STRUCT_DELETED ) )
379 {
380 lines.push_back( line );
381 }
382 }
383
384 alg::for_all_pairs( lines.begin(), lines.end(),
385 [&]( SCH_LINE* firstLine, SCH_LINE* secondLine )
386 {
387 if( ( firstLine->GetEditFlags() & STRUCT_DELETED )
388 || ( secondLine->GetEditFlags() & STRUCT_DELETED )
389 || !secondLine->IsParallel( firstLine ) )
390 {
391 return;
392 }
393
394 // Remove identical lines
395 if( firstLine->IsEndPoint( secondLine->GetStartPoint() )
396 && firstLine->IsEndPoint( secondLine->GetEndPoint() ) )
397 {
398 firstLine->SetFlags( STRUCT_DELETED );
399 return;
400 }
401
402 // Try to merge the remaining lines
403 if( SCH_LINE* new_line = secondLine->MergeOverlap( screen, firstLine, false ) )
404 {
405 firstLine->SetFlags( STRUCT_DELETED );
406 secondLine->SetFlags( STRUCT_DELETED );
407 AddToScreen( new_line, screen );
408 aCommit->Added( new_line, screen );
409
410 if( new_line->IsSelected() )
411 selectionTool->AddItemToSel( new_line, true /*quiet mode*/ );
412
413 lines.push_back( new_line );
414 }
415 } );
416
417 for( SCH_LINE* line : lines )
418 {
419 if( line->GetEditFlags() & STRUCT_DELETED )
420 {
421 if( line->IsSelected() )
422 selectionTool->RemoveItemFromSel( line, true /*quiet mode*/ );
423
424 RemoveFromScreen( line, screen );
425 aCommit->Removed( line, screen );
426 }
427 }
428}
429
430
432 const VECTOR2I& aPos )
433{
434 SCH_JUNCTION* junction = new SCH_JUNCTION( aPos );
435
436 AddToScreen( junction, aScreen );
437 aCommit->Added( junction, aScreen );
438
439 BreakSegments( aCommit, aPos, aScreen );
440
441 return junction;
442}
443
444
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
COMMIT & Modify(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Modify a given item in the model.
Definition: commit.h:108
COMMIT & Added(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Notify observers that aItem has been added.
Definition: commit.h:86
COMMIT & Removed(EDA_ITEM *aItem, BASE_SCREEN *aScreen=nullptr)
Definition: commit.h:98
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
EDA_ITEM_FLAGS GetEditFlags() const
Definition: eda_item.h:141
void SetFlags(EDA_ITEM_FLAGS aMask)
Definition: eda_item.h:135
bool IsSelected() const
Definition: eda_item.h:120
EDA_ITEM_FLAGS GetFlags() const
Definition: eda_item.h:138
EE_TYPE Overlapping(const BOX2I &aRect) const
Definition: sch_rtree.h:246
EE_TYPE OfType(KICAD_T aType) const
Definition: sch_rtree.h:241
virtual void Update(const VIEW_ITEM *aItem, int aUpdateFlags) const
For dynamic VIEWs, inform the associated VIEW that the graphical representation of this item has chan...
Definition: view.cpp:1673
A holder to handle information on schematic or board items.
void AddToScreen(EDA_ITEM *aItem, SCH_SCREEN *aScreen=nullptr)
Add an item to the screen (and view) aScreen is the screen the item is located on,...
SCH_DRAW_PANEL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
void RemoveFromScreen(EDA_ITEM *aItem, SCH_SCREEN *aScreen)
Remove an item from the screen (and view) aScreen is the screen the item is located on,...
VECTOR2I GetPosition() const override
VECTOR2I GetEnd() const
Class for a wire to bus entry.
KIGFX::SCH_VIEW * GetView() const override
Return a pointer to the #VIEW instance used in the panel.
SCH_SCREEN * GetScreen() const override
Return a pointer to a BASE_SCREEN or one of its derivatives.
bool BreakSegmentsOnJunctions(SCH_COMMIT *aCommit, SCH_SCREEN *aScreen)
Test all junctions and bus entries in the schematic for intersections with wires and buses and breaks...
void SchematicCleanUp(SCH_COMMIT *aCommit, SCH_SCREEN *aScreen=nullptr)
Perform routine schematic cleaning including breaking wire and buses and deleting identical objects s...
bool BreakSegments(SCH_COMMIT *aCommit, const VECTOR2I &aPoint, SCH_SCREEN *aScreen)
Check every wire and bus for a intersection at aPoint and break into two segments at aPoint if an int...
void BreakSegment(SCH_COMMIT *aCommit, SCH_LINE *aSegment, const VECTOR2I &aPoint, SCH_LINE **aNewSegment, SCH_SCREEN *aScreen)
Break a single segment into two at the specified point.
bool TrimWire(SCH_COMMIT *aCommit, const VECTOR2I &aStart, const VECTOR2I &aEnd)
If any single wire passes through both points, remove the portion between the two points,...
void TestDanglingEnds()
Test all of the connectable objects in the schematic for unused connection points.
void DeleteJunction(SCH_COMMIT *aCommit, SCH_ITEM *aItem)
Remove a given junction and heals any wire segments under the junction.
SCH_JUNCTION * AddJunction(SCH_COMMIT *aCommit, SCH_SCREEN *aScreen, const VECTOR2I &aPos)
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition: sch_item.h:167
SCH_LAYER_ID GetLayer() const
Return the layer this item is on.
Definition: sch_item.h:291
VECTOR2I GetPosition() const override
Definition: sch_junction.h:107
Segment description base class to describe items which have 2 end points (track, wire,...
Definition: sch_line.h:41
bool IsWire() const
Return true if the line is a wire.
Definition: sch_line.cpp:933
bool IsParallel(const SCH_LINE *aLine) const
Definition: sch_line.cpp:406
VECTOR2I GetEndPoint() const
Definition: sch_line.h:143
VECTOR2I GetStartPoint() const
Definition: sch_line.h:138
SCH_LINE * MergeOverlap(SCH_SCREEN *aScreen, SCH_LINE *aLine, bool aCheckJunctions)
Check line against aLine to see if it overlaps and merge if it does.
Definition: sch_line.cpp:419
bool IsBus() const
Return true if the line is a bus.
Definition: sch_line.cpp:939
bool IsNull() const
Definition: sch_line.h:136
bool IsEndPoint(const VECTOR2I &aPoint) const
Definition: sch_line.h:90
bool IsStrokeEquivalent(const SCH_LINE *aLine)
Definition: sch_line.h:195
SCH_LINE * BreakAt(const VECTOR2I &aPoint)
Break this segment into two at the specified point.
Definition: sch_line.cpp:540
VECTOR2I GetPosition() const override
void TestDanglingEnds(const SCH_SHEET_PATH *aPath=nullptr, std::function< void(SCH_ITEM *)> *aChangedHandler=nullptr) const
Test all of the connectable objects in the schematic for unused connection points.
std::vector< SCH_LINE * > GetBusesAndWires(const VECTOR2I &aPosition, bool aIgnoreEndpoints=false) const
Return buses and wires passing through aPosition.
bool IsExplicitJunction(const VECTOR2I &aPosition) const
Indicate that a junction dot is necessary at the given location.
Definition: sch_screen.cpp:489
EE_RTREE & Items()
Get the full RTree, usually for iterating.
Definition: sch_screen.h:112
int RemoveItemFromSel(const TOOL_EVENT &aEvent)
TOOL_MANAGER * m_toolManager
Definition: tools_holder.h:171
#define IS_CHANGED
Item was edited, and modified.
#define IS_NEW
New item, just created.
#define IS_BROKEN
Is a segment just broken by BreakSegment.
#define STRUCT_DELETED
flag indication structures to be erased
#define SKIP_STRUCT
flag indicating that the structure should be ignored
#define IS_MOVING
Item being moved.
@ LAYER_WIRE
Definition: layer_ids.h:441
@ LAYER_BUS
Definition: layer_ids.h:442
@ REPAINT
Item needs to be redrawn.
Definition: view_item.h:58
void for_all_pairs(_InputIterator __first, _InputIterator __last, _Function __f)
Apply a function to every possible pair of elements of a sequence.
Definition: kicad_algo.h:84
bool IsPointOnSegment(const VECTOR2I &aSegStart, const VECTOR2I &aSegEnd, const VECTOR2I &aTestPoint)
Test if aTestPoint is on line defined by aSegStart and aSegEnd.
Definition: trigo.cpp:89
@ SCH_LINE_T
Definition: typeinfo.h:163
@ SCH_NO_CONNECT_T
Definition: typeinfo.h:160
@ SCH_BUS_WIRE_ENTRY_T
Definition: typeinfo.h:161
@ SCH_JUNCTION_T
Definition: typeinfo.h:159