KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
eda_shape.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) 2018 Jean-Pierre Charras, jp.charras at wanadoo.fr
5 * Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
6 * Copyright (C) 2011 Wayne Stambaugh <stambaughw@gmail.com>
7 * Copyright (C) 2023 CERN
8 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
9 *
10 * This program is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU General Public License
12 * as published by the Free Software Foundation; either version 2
13 * of the License, or (at your option) any later version.
14 *
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, you may find one here:
22 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
23 * or you may search the http://www.gnu.org website for the version 2 license,
24 * or you may write to the Free Software Foundation, Inc.,
25 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
26 */
27
28#include <eda_shape.h>
29
30#include <bezier_curves.h>
32#include <eda_draw_frame.h>
33#include <geometry/shape_arc.h>
37#include <geometry/shape_rect.h>
38#include <macros.h>
39#include <math/util.h> // for KiROUND
40#include <eda_item.h>
41#include <plotters/plotter.h>
42#include <api/api_enums.h>
43#include <api/api_utils.h>
44#include <api/common/types/base_types.pb.h>
45
46
47EDA_SHAPE::EDA_SHAPE( SHAPE_T aType, int aLineWidth, FILL_T aFill ) :
48 m_endsSwapped( false ),
49 m_shape( aType ),
50 m_stroke( aLineWidth, LINE_STYLE::DEFAULT, COLOR4D::UNSPECIFIED ),
51 m_fill( aFill ),
52 m_fillColor( COLOR4D::UNSPECIFIED ),
53 m_rectangleHeight( 0 ),
54 m_rectangleWidth( 0 ),
55 m_segmentLength( 0 ),
56 m_editState( 0 ),
57 m_proxyItem( false )
58{
59}
60
61
63{
64}
65
66
67EDA_SHAPE::EDA_SHAPE( const SHAPE& aShape ) :
68 m_endsSwapped( false ),
69 m_stroke( 0, LINE_STYLE::DEFAULT, COLOR4D::UNSPECIFIED ),
70 m_fill(),
71 m_rectangleHeight( 0 ),
72 m_rectangleWidth( 0 ),
73 m_segmentLength( 0 ),
74 m_editState( 0 ),
75 m_proxyItem( false )
76{
77 switch( aShape.Type() )
78 {
79 case SH_RECT:
80 {
81 auto rect = static_cast<const SHAPE_RECT&>( aShape );
82 m_shape = SHAPE_T::RECTANGLE;
83 SetStart( rect.GetPosition() );
84 SetEnd( rect.GetPosition() + rect.GetSize() );
85 break;
86 }
87
88 case SH_SEGMENT:
89 {
90 auto seg = static_cast<const SHAPE_SEGMENT&>( aShape );
91 m_shape = SHAPE_T::SEGMENT;
92 SetStart( seg.GetSeg().A );
93 SetEnd( seg.GetSeg().B );
94 SetWidth( seg.GetWidth() );
95 break;
96 }
97
98 case SH_LINE_CHAIN:
99 {
100 auto line = static_cast<const SHAPE_LINE_CHAIN&>( aShape );
101 m_shape = SHAPE_T::POLY;
103 m_poly.AddOutline( line );
104 SetWidth( line.Width() );
105 break;
106 }
107
108 case SH_CIRCLE:
109 {
110 auto circle = static_cast<const SHAPE_CIRCLE&>( aShape );
111 m_shape = SHAPE_T::CIRCLE;
114 break;
115 }
116
117 case SH_ARC:
118 {
119 auto arc = static_cast<const SHAPE_ARC&>( aShape );
120 m_shape = SHAPE_T::ARC;
121 SetArcGeometry( arc.GetP0(), arc.GetArcMid(), arc.GetP1() );
122 SetWidth( arc.GetWidth() );
123 break;
124 }
125
126 case SH_SIMPLE:
127 {
128 auto poly = static_cast<const SHAPE_SIMPLE&>( aShape );
129 m_shape = SHAPE_T::POLY;
130 poly.TransformToPolygon( m_poly, 0, ERROR_INSIDE );
131 break;
132 }
133
134 // currently unhandled
135 case SH_POLY_SET:
136 case SH_COMPOUND:
137 case SH_NULL:
139 default:
140 m_shape = SHAPE_T::UNDEFINED;
141 break;
142 }
143}
144
145
146void EDA_SHAPE::Serialize( google::protobuf::Any &aContainer ) const
147{
148 using namespace kiapi::common;
149 types::GraphicShape shape;
150
151 types::StrokeAttributes* stroke = shape.mutable_attributes()->mutable_stroke();
152 types::GraphicFillAttributes* fill = shape.mutable_attributes()->mutable_fill();
153
154 stroke->mutable_width()->set_value_nm( GetWidth() );
155
156 switch( GetLineStyle() )
157 {
158 case LINE_STYLE::DEFAULT: stroke->set_style( types::SLS_DEFAULT ); break;
159 case LINE_STYLE::SOLID: stroke->set_style( types::SLS_SOLID ); break;
160 case LINE_STYLE::DASH: stroke->set_style( types::SLS_DASH ); break;
161 case LINE_STYLE::DOT: stroke->set_style( types::SLS_DOT ); break;
162 case LINE_STYLE::DASHDOT: stroke->set_style( types::SLS_DASHDOT ); break;
163 case LINE_STYLE::DASHDOTDOT: stroke->set_style( types::SLS_DASHDOTDOT ); break;
164 default: break;
165 }
166
167 switch( GetFillMode() )
168 {
169 case FILL_T::FILLED_SHAPE: fill->set_fill_type( types::GFT_FILLED ); break;
170 default: fill->set_fill_type( types::GFT_UNFILLED ); break;
171 }
172
173 switch( GetShape() )
174 {
175 case SHAPE_T::SEGMENT:
176 {
177 types::GraphicSegmentAttributes* segment = shape.mutable_segment();
178 PackVector2( *segment->mutable_start(), GetStart() );
179 PackVector2( *segment->mutable_end(), GetEnd() );
180 break;
181 }
182
183 case SHAPE_T::RECTANGLE:
184 {
185 types::GraphicRectangleAttributes* rectangle = shape.mutable_rectangle();
186 PackVector2( *rectangle->mutable_top_left(), GetStart() );
187 PackVector2( *rectangle->mutable_bottom_right(), GetEnd() );
188 break;
189 }
190
191 case SHAPE_T::ARC:
192 {
193 types::GraphicArcAttributes* arc = shape.mutable_arc();
194 PackVector2( *arc->mutable_start(), GetStart() );
195 PackVector2( *arc->mutable_mid(), GetArcMid() );
196 PackVector2( *arc->mutable_end(), GetEnd() );
197 break;
198 }
199
200 case SHAPE_T::CIRCLE:
201 {
202 types::GraphicCircleAttributes* circle = shape.mutable_circle();
203 PackVector2( *circle->mutable_center(), GetStart() );
204 PackVector2( *circle->mutable_radius_point(), GetEnd() );
205 break;
206 }
207
208 case SHAPE_T::POLY:
209 {
210 PackPolySet( *shape.mutable_polygon(), GetPolyShape() );
211 break;
212 }
213
214 case SHAPE_T::BEZIER:
215 {
216 types::GraphicBezierAttributes* bezier = shape.mutable_bezier();
217 PackVector2( *bezier->mutable_start(), GetStart() );
218 PackVector2( *bezier->mutable_control1(), GetBezierC1() );
219 PackVector2( *bezier->mutable_control2(), GetBezierC2() );
220 PackVector2( *bezier->mutable_end(), GetEnd() );
221 break;
222 }
223
224 default:
225 wxASSERT_MSG( false, "Unhandled shape in PCB_SHAPE::Serialize" );
226 }
227
228 // TODO m_hasSolderMask and m_solderMaskMargin
229
230 aContainer.PackFrom( shape );
231}
232
233
234bool EDA_SHAPE::Deserialize( const google::protobuf::Any &aContainer )
235{
236 using namespace kiapi::common;
237
238 types::GraphicShape shape;
239
240 if( !aContainer.UnpackTo( &shape ) )
241 return false;
242
243 // Initialize everything to a known state that doesn't get touched by every
244 // codepath below, to make sure the equality operator is consistent
245 m_start = {};
246 m_end = {};
247 m_arcCenter = {};
248 m_arcMidData = {};
249 m_bezierC1 = {};
250 m_bezierC2 = {};
251 m_editState = 0;
252 m_proxyItem = false;
253 m_endsSwapped = false;
254
255 SetFilled( shape.attributes().fill().fill_type() == types::GFT_FILLED );
256 SetWidth( shape.attributes().stroke().width().value_nm() );
257
258 switch( shape.attributes().stroke().style() )
259 {
260 case types::SLS_DEFAULT: SetLineStyle( LINE_STYLE::DEFAULT ); break;
261 case types::SLS_SOLID: SetLineStyle( LINE_STYLE::SOLID ); break;
262 case types::SLS_DASH: SetLineStyle( LINE_STYLE::DASH ); break;
263 case types::SLS_DOT: SetLineStyle( LINE_STYLE::DOT ); break;
264 case types::SLS_DASHDOT: SetLineStyle( LINE_STYLE::DASHDOT ); break;
265 case types::SLS_DASHDOTDOT: SetLineStyle( LINE_STYLE::DASHDOTDOT ); break;
266 default: break;
267 }
268
269 if( shape.has_segment() )
270 {
271 SetShape( SHAPE_T::SEGMENT );
272 SetStart( UnpackVector2( shape.segment().start() ) );
273 SetEnd( UnpackVector2( shape.segment().end() ) );
274 }
275 else if( shape.has_rectangle() )
276 {
277 SetShape( SHAPE_T::RECTANGLE );
278 SetStart( UnpackVector2( shape.rectangle().top_left() ) );
279 SetEnd( UnpackVector2( shape.rectangle().bottom_right() ) );
280 }
281 else if( shape.has_arc() )
282 {
283 SetShape( SHAPE_T::ARC );
284 SetArcGeometry( UnpackVector2( shape.arc().start() ),
285 UnpackVector2( shape.arc().mid() ),
286 UnpackVector2( shape.arc().end() ) );
287 }
288 else if( shape.has_circle() )
289 {
290 SetShape( SHAPE_T::CIRCLE );
291 SetStart( UnpackVector2( shape.circle().center() ) );
292 SetEnd( UnpackVector2( shape.circle().radius_point() ) );
293 }
294 else if( shape.has_polygon() )
295 {
296 SetShape( SHAPE_T::POLY );
297 SetPolyShape( UnpackPolySet( shape.polygon() ) );
298 }
299 else if( shape.has_bezier() )
300 {
301 SetShape( SHAPE_T::BEZIER );
302 SetStart( UnpackVector2( shape.bezier().start() ) );
303 SetBezierC1( UnpackVector2( shape.bezier().control1() ) );
304 SetBezierC2( UnpackVector2( shape.bezier().control2() ) );
305 SetEnd( UnpackVector2( shape.bezier().end() ) );
307 }
308
309 return true;
310}
311
312
313wxString EDA_SHAPE::ShowShape() const
314{
315 if( IsProxyItem() )
316 {
317 switch( m_shape )
318 {
319 case SHAPE_T::SEGMENT: return _( "Thermal Spoke" );
320 case SHAPE_T::RECTANGLE: return _( "Number Box" );
321 default: return wxT( "??" );
322 }
323 }
324 else
325 {
326 switch( m_shape )
327 {
328 case SHAPE_T::SEGMENT: return _( "Line" );
329 case SHAPE_T::RECTANGLE: return _( "Rect" );
330 case SHAPE_T::ARC: return _( "Arc" );
331 case SHAPE_T::CIRCLE: return _( "Circle" );
332 case SHAPE_T::BEZIER: return _( "Bezier Curve" );
333 case SHAPE_T::POLY: return _( "Polygon" );
334 default: return wxT( "??" );
335 }
336 }
337}
338
339
341{
342 switch( m_shape )
343 {
344 case SHAPE_T::SEGMENT: return wxS( "S_SEGMENT" );
345 case SHAPE_T::RECTANGLE: return wxS( "S_RECT" );
346 case SHAPE_T::ARC: return wxS( "S_ARC" );
347 case SHAPE_T::CIRCLE: return wxS( "S_CIRCLE" );
348 case SHAPE_T::POLY: return wxS( "S_POLYGON" );
349 case SHAPE_T::BEZIER: return wxS( "S_CURVE" );
350 case SHAPE_T::UNDEFINED: return wxS( "UNDEFINED" );
351 }
352
353 return wxEmptyString; // Just to quiet GCC.
354}
355
356
358{
359 move( aPos - getPosition() );
360}
361
362
364{
365 if( m_shape == SHAPE_T::ARC )
366 return getCenter();
367 else if( m_shape == SHAPE_T::POLY )
368 return m_poly.CVertex( 0 );
369 else
370 return m_start;
371}
372
373
375{
376 double length = 0.0;
377
378 switch( m_shape )
379 {
380 case SHAPE_T::BEZIER:
381 for( size_t ii = 1; ii < m_bezierPoints.size(); ++ii )
382 length += m_bezierPoints[ ii - 1].Distance( m_bezierPoints[ii] );
383
384 return length;
385
386 case SHAPE_T::SEGMENT:
387 return GetStart().Distance( GetEnd() );
388
389 case SHAPE_T::POLY:
390 for( int ii = 0; ii < m_poly.COutline( 0 ).SegmentCount(); ii++ )
391 length += m_poly.COutline( 0 ).CSegment( ii ).Length();
392
393 return length;
394
395 case SHAPE_T::ARC:
396 return GetRadius() * GetArcAngle().AsRadians();
397
398 default:
400 return 0.0;
401 }
402}
403
404
406{
407 switch( m_shape )
408 {
409 case SHAPE_T::RECTANGLE:
410 return GetEndY() - GetStartY();
411
412 default:
414 return 0;
415 }
416}
417
418
420{
421 switch( m_shape )
422 {
423 case SHAPE_T::RECTANGLE:
424 return GetEndX() - GetStartX();
425
426 default:
428 return 0;
429 }
430}
431
432
433void EDA_SHAPE::SetLength( const double& aLength )
434{
435 switch( m_shape )
436 {
437 case SHAPE_T::SEGMENT:
438 m_segmentLength = aLength; break;
439
440 default:
442 }
443}
444
445
446void EDA_SHAPE::SetRectangleHeight( const int& aHeight )
447{
448 switch ( m_shape )
449 {
450 case SHAPE_T::RECTANGLE:
451 m_rectangleHeight = aHeight;
453 break;
454
455 default:
457 }
458}
459
460
461void EDA_SHAPE::SetRectangleWidth( const int& aWidth )
462{
463 switch ( m_shape )
464 {
465 case SHAPE_T::RECTANGLE:
466 m_rectangleWidth = aWidth;
468 break;
469
470 default:
472 }
473}
474
475
476void EDA_SHAPE::SetRectangle( const long long int& aHeight, const long long int& aWidth )
477{
478 switch ( m_shape )
479 {
480 case SHAPE_T::RECTANGLE:
481 m_rectangleHeight = aHeight;
482 m_rectangleWidth = aWidth;
483 break;
484
485 default:
487 }
488}
489
490
492{
493 switch( m_shape )
494 {
495 case SHAPE_T::SEGMENT:
496 m_segmentAngle = aAngle;
497 break;
498
499 default:
501 }
502}
503
504
506{
507 switch( m_shape )
508 {
509 case SHAPE_T::CIRCLE:
510 case SHAPE_T::RECTANGLE:
511 return true;
512
513 case SHAPE_T::ARC:
514 case SHAPE_T::SEGMENT:
515 return false;
516
517 case SHAPE_T::POLY:
518 if( m_poly.IsEmpty() )
519 return false;
520 else
521 return m_poly.Outline( 0 ).IsClosed();
522
523 case SHAPE_T::BEZIER:
524 if( m_bezierPoints.size() < 3 )
525 return false;
526 else
527 return m_bezierPoints[0] == m_bezierPoints[ m_bezierPoints.size() - 1 ];
528
529 default:
531 return false;
532 }
533}
534
535
537{
538 m_fill = aFill;
539 m_hatchingDirty = true;
540}
541
542
544{
545 switch( aFill )
546 {
547 case UI_FILL_MODE::NONE: SetFillMode( FILL_T::NO_FILL ); break;
548 case UI_FILL_MODE::HATCH: SetFillMode( FILL_T::HATCH ); break;
549 case UI_FILL_MODE::REVERSE_HATCH: SetFillMode( FILL_T::REVERSE_HATCH ); break;
550 case UI_FILL_MODE::CROSS_HATCH: SetFillMode( FILL_T::CROSS_HATCH ); break;
551 default: SetFilled( true ); break;
552 }
553}
554
555
557{
558 switch( m_fill )
559 {
560 case FILL_T::NO_FILL: return UI_FILL_MODE::NONE;
561 case FILL_T::HATCH: return UI_FILL_MODE::HATCH;
562 case FILL_T::REVERSE_HATCH: return UI_FILL_MODE::REVERSE_HATCH;
563 case FILL_T::CROSS_HATCH: return UI_FILL_MODE::CROSS_HATCH;
564 default: return UI_FILL_MODE::SOLID;
565 }
566}
567
568
570{
571 if( m_hatchingDirty )
572 {
574 m_hatchingDirty = false;
575 }
576
577 return m_hatching;
578}
579
580
582{
584
585 std::vector<double> slopes;
586 int lineWidth = GetHatchLineWidth();
587 int spacing = GetHatchLineSpacing();
588 SHAPE_POLY_SET shapeBuffer;
589
590 if( GetFillMode() == FILL_T::CROSS_HATCH )
591 slopes = { 1.0, -1.0 };
592 else if( GetFillMode() == FILL_T::HATCH )
593 slopes = { -1.0 };
594 else if( GetFillMode() == FILL_T::REVERSE_HATCH )
595 slopes = { 1.0 };
596 else
597 return;
598
599 if( spacing == 0 )
600 return;
601
602 auto addHatchLines =
603 [&]( const std::vector<SEG>& hatchLines )
604 {
605 for( const SEG& seg : hatchLines )
606 {
607 TransformOvalToPolygon( m_hatching, seg.A, seg.B, lineWidth, ARC_LOW_DEF,
608 ERROR_INSIDE );
609 }
610 };
611
612 switch( m_shape )
613 {
614 case SHAPE_T::ARC:
615 case SHAPE_T::SEGMENT:
616 case SHAPE_T::BEZIER:
617 break;
618
619 case SHAPE_T::RECTANGLE:
620 shapeBuffer.NewOutline();
621
622 for( const VECTOR2I& pt : GetRectCorners() )
623 shapeBuffer.Append( pt );
624
625 addHatchLines( shapeBuffer.GenerateHatchLines( slopes, spacing, -1 ) );
626 break;
627
628 case SHAPE_T::CIRCLE:
630 addHatchLines( shapeBuffer.GenerateHatchLines( slopes, spacing, -1 ) );
631 break;
632
633 case SHAPE_T::POLY:
634 if( IsClosed() )
635 addHatchLines( m_poly.GenerateHatchLines( slopes, spacing, -1 ) );
636
637 break;
638
639 default:
641 break;
642 }
643
645}
646
647
648void EDA_SHAPE::move( const VECTOR2I& aMoveVector )
649{
650 switch ( m_shape )
651 {
652 case SHAPE_T::ARC:
653 m_arcCenter += aMoveVector;
654 m_arcMidData.center += aMoveVector;
655 m_arcMidData.start += aMoveVector;
656 m_arcMidData.end += aMoveVector;
657 m_arcMidData.mid += aMoveVector;
659
660 case SHAPE_T::SEGMENT:
661 case SHAPE_T::RECTANGLE:
662 case SHAPE_T::CIRCLE:
663 m_start += aMoveVector;
664 m_end += aMoveVector;
665 break;
666
667 case SHAPE_T::POLY:
668 m_poly.Move( aMoveVector );
669 break;
670
671 case SHAPE_T::BEZIER:
672 m_start += aMoveVector;
673 m_end += aMoveVector;
674 m_bezierC1 += aMoveVector;
675 m_bezierC2 += aMoveVector;
676
677 for( VECTOR2I& pt : m_bezierPoints )
678 pt += aMoveVector;
679
680 break;
681
682 default:
684 break;
685 }
686
687 m_hatchingDirty = true;
688}
689
690
691void EDA_SHAPE::scale( double aScale )
692{
693 auto scalePt = [&]( VECTOR2I& pt )
694 {
695 pt.x = KiROUND( pt.x * aScale );
696 pt.y = KiROUND( pt.y * aScale );
697 };
698
699 switch( m_shape )
700 {
701 case SHAPE_T::ARC:
702 scalePt( m_arcCenter );
704
705 case SHAPE_T::SEGMENT:
706 case SHAPE_T::RECTANGLE:
707 scalePt( m_start );
708 scalePt( m_end );
709 break;
710
711 case SHAPE_T::CIRCLE: // ring or circle
712 scalePt( m_start );
713 m_end.x = m_start.x + KiROUND( GetRadius() * aScale );
714 m_end.y = m_start.y;
715 break;
716
717 case SHAPE_T::POLY: // polygon
718 {
719 std::vector<VECTOR2I> pts;
720
721 for( int ii = 0; ii < m_poly.OutlineCount(); ++ ii )
722 {
723 for( const VECTOR2I& pt : m_poly.Outline( ii ).CPoints() )
724 {
725 pts.emplace_back( pt );
726 scalePt( pts.back() );
727 }
728 }
729
730 SetPolyPoints( pts );
731 }
732 break;
733
734 case SHAPE_T::BEZIER:
735 scalePt( m_start );
736 scalePt( m_end );
737 scalePt( m_bezierC1 );
738 scalePt( m_bezierC2 );
740 break;
741
742 default:
744 break;
745 }
746
747 m_hatchingDirty = true;
748}
749
750
751void EDA_SHAPE::rotate( const VECTOR2I& aRotCentre, const EDA_ANGLE& aAngle )
752{
753 switch( m_shape )
754 {
755 case SHAPE_T::SEGMENT:
756 case SHAPE_T::CIRCLE:
757 RotatePoint( m_start, aRotCentre, aAngle );
758 RotatePoint( m_end, aRotCentre, aAngle );
759 break;
760
761 case SHAPE_T::ARC:
762 RotatePoint( m_start, aRotCentre, aAngle );
763 RotatePoint( m_end, aRotCentre, aAngle );
764 RotatePoint( m_arcCenter, aRotCentre, aAngle );
765 RotatePoint( m_arcMidData.start, aRotCentre, aAngle );
766 RotatePoint( m_arcMidData.end, aRotCentre, aAngle );
767 RotatePoint( m_arcMidData.mid, aRotCentre, aAngle );
768 RotatePoint( m_arcMidData.center, aRotCentre, aAngle );
769 break;
770
771 case SHAPE_T::RECTANGLE:
772 if( aAngle.IsCardinal() )
773 {
774 RotatePoint( m_start, aRotCentre, aAngle );
775 RotatePoint( m_end, aRotCentre, aAngle );
776 break;
777 }
778
779 // Convert non-cardinally-rotated rect to a diamond
780 m_shape = SHAPE_T::POLY;
787
789
790 case SHAPE_T::POLY:
791 m_poly.Rotate( aAngle, aRotCentre );
792 break;
793
794 case SHAPE_T::BEZIER:
795 RotatePoint( m_start, aRotCentre, aAngle );
796 RotatePoint( m_end, aRotCentre, aAngle );
797 RotatePoint( m_bezierC1, aRotCentre, aAngle );
798 RotatePoint( m_bezierC2, aRotCentre, aAngle );
799
800 for( VECTOR2I& pt : m_bezierPoints )
801 RotatePoint( pt, aRotCentre, aAngle);
802
803 break;
804
805 default:
807 break;
808 }
809
810 m_hatchingDirty = true;
811}
812
813
814void EDA_SHAPE::flip( const VECTOR2I& aCentre, FLIP_DIRECTION aFlipDirection )
815{
816 switch ( m_shape )
817 {
818 case SHAPE_T::SEGMENT:
819 case SHAPE_T::RECTANGLE:
820 MIRROR( m_start, aCentre, aFlipDirection );
821 MIRROR( m_end, aCentre, aFlipDirection );
822 break;
823
824 case SHAPE_T::CIRCLE:
825 MIRROR( m_start, aCentre, aFlipDirection );
826 MIRROR( m_end, aCentre, aFlipDirection );
827 break;
828
829 case SHAPE_T::ARC:
830 MIRROR( m_start, aCentre, aFlipDirection );
831 MIRROR( m_end, aCentre, aFlipDirection );
832 MIRROR( m_arcCenter, aCentre, aFlipDirection );
833
834 std::swap( m_start, m_end );
835 break;
836
837 case SHAPE_T::POLY:
838 m_poly.Mirror( aCentre, aFlipDirection );
839 break;
840
841 case SHAPE_T::BEZIER:
842 MIRROR( m_start, aCentre, aFlipDirection );
843 MIRROR( m_end, aCentre, aFlipDirection );
844 MIRROR( m_bezierC1, aCentre, aFlipDirection );
845 MIRROR( m_bezierC2, aCentre, aFlipDirection );
846
848 break;
849
850 default:
852 break;
853 }
854
855 m_hatchingDirty = true;
856}
857
858
860{
861 // Has meaning only for SHAPE_T::BEZIER
862 if( m_shape != SHAPE_T::BEZIER )
863 {
864 m_bezierPoints.clear();
865 return;
866 }
867
868 // Rebuild the m_BezierPoints vertex list that approximate the Bezier curve
870}
871
872
873const std::vector<VECTOR2I> EDA_SHAPE::buildBezierToSegmentsPointsList( int aMaxError ) const
874{
875 std::vector<VECTOR2I> bezierPoints;
876
877 // Rebuild the m_BezierPoints vertex list that approximate the Bezier curve
878 std::vector<VECTOR2I> ctrlPoints = { m_start, m_bezierC1, m_bezierC2, m_end };
879 BEZIER_POLY converter( ctrlPoints );
880 converter.GetPoly( bezierPoints, aMaxError );
881
882 return bezierPoints;
883}
884
885
887{
888 switch( m_shape )
889 {
890 case SHAPE_T::ARC:
891 return m_arcCenter;
892
893 case SHAPE_T::CIRCLE:
894 return m_start;
895
896 case SHAPE_T::SEGMENT:
897 // Midpoint of the line
898 return ( m_start + m_end ) / 2;
899
900 case SHAPE_T::POLY:
901 case SHAPE_T::RECTANGLE:
902 case SHAPE_T::BEZIER:
903 return getBoundingBox().Centre();
904
905 default:
907 return VECTOR2I();
908 }
909}
910
911
912void EDA_SHAPE::SetCenter( const VECTOR2I& aCenter )
913{
914 switch( m_shape )
915 {
916 case SHAPE_T::ARC:
917 m_arcCenter = aCenter;
918 break;
919
920 case SHAPE_T::CIRCLE:
921 m_start = aCenter;
922 m_hatchingDirty = true;
923 break;
924
925 default:
927 }
928}
929
930
932{
933 // If none of the input data have changed since we loaded the arc,
934 // keep the original mid point data to minimize churn
937 return m_arcMidData.mid;
938
939 VECTOR2I mid = m_start;
940 RotatePoint( mid, m_arcCenter, -GetArcAngle() / 2.0 );
941 return mid;
942}
943
944
945void EDA_SHAPE::CalcArcAngles( EDA_ANGLE& aStartAngle, EDA_ANGLE& aEndAngle ) const
946{
947 VECTOR2D startRadial( GetStart() - getCenter() );
948 VECTOR2D endRadial( GetEnd() - getCenter() );
949
950 aStartAngle = EDA_ANGLE( startRadial );
951 aEndAngle = EDA_ANGLE( endRadial );
952
953 if( aEndAngle == aStartAngle )
954 aEndAngle = aStartAngle + ANGLE_360; // ring, not null
955
956 while( aEndAngle < aStartAngle )
957 aEndAngle += ANGLE_360;
958}
959
960
962{
963 double radius = 0.0;
964
965 switch( m_shape )
966 {
967 case SHAPE_T::ARC:
969 break;
970
971 case SHAPE_T::CIRCLE:
973 break;
974
975 default:
977 }
978
979 // don't allow degenerate circles/arcs
980 return std::max( 1, KiROUND( radius ) );
981}
982
983
984void EDA_SHAPE::SetCachedArcData( const VECTOR2I& aStart, const VECTOR2I& aMid,
985 const VECTOR2I& aEnd, const VECTOR2I& aCenter )
986{
987 m_arcMidData.start = aStart;
988 m_arcMidData.end = aEnd;
989 m_arcMidData.center = aCenter;
990 m_arcMidData.mid = aMid;
991}
992
993
994void EDA_SHAPE::SetArcGeometry( const VECTOR2I& aStart, const VECTOR2I& aMid, const VECTOR2I& aEnd )
995{
996 m_arcMidData = {};
997 m_start = aStart;
998 m_end = aEnd;
999 m_arcCenter = CalcArcCenter( aStart, aMid, aEnd );
1000 VECTOR2I new_mid = GetArcMid();
1001
1002 m_endsSwapped = false;
1003
1004 // Watch the ordering here. GetArcMid above needs to be called prior to initializing the
1005 // m_arcMidData structure in order to ensure we get the calculated variant, not the cached
1006 SetCachedArcData( aStart, aMid, aEnd, m_arcCenter );
1007
1008 /*
1009 * If the input winding doesn't match our internal winding, the calculated midpoint will end
1010 * up on the other side of the arc. In this case, we need to flip the start/end points and
1011 * flag this change for the system.
1012 */
1013 VECTOR2D dist( new_mid - aMid );
1014 VECTOR2D dist2( new_mid - m_arcCenter );
1015
1016 if( dist.SquaredEuclideanNorm() > dist2.SquaredEuclideanNorm() )
1017 {
1018 std::swap( m_start, m_end );
1019 m_endsSwapped = true;
1020 }
1021}
1022
1023
1025{
1026 EDA_ANGLE angle( atan2( static_cast<double>( GetStart().y - GetEnd().y ),
1027 static_cast<double>( GetEnd().x - GetStart().x ) ), RADIANS_T );
1028
1029 return angle;
1030}
1031
1032
1034{
1035 EDA_ANGLE startAngle;
1036 EDA_ANGLE endAngle;
1037
1038 CalcArcAngles( startAngle, endAngle );
1039
1040 return endAngle - startAngle;
1041}
1042
1043
1045{
1046 if( m_shape == SHAPE_T::ARC )
1047 {
1048 VECTOR2D mid = GetArcMid();
1049
1050 double orient = ( mid.x - m_start.x ) * ( m_end.y - m_start.y )
1051 - ( mid.y - m_start.y ) * ( m_end.x - m_start.x );
1052
1053 return orient < 0;
1054 }
1055
1057 return false;
1058}
1059
1060
1061void EDA_SHAPE::SetArcAngleAndEnd( const EDA_ANGLE& aAngle, bool aCheckNegativeAngle )
1062{
1063 EDA_ANGLE angle( aAngle );
1064
1065 m_end = m_start;
1067
1068 if( aCheckNegativeAngle && aAngle < ANGLE_0 )
1069 {
1070 std::swap( m_start, m_end );
1071 m_endsSwapped = true;
1072 }
1073}
1074
1075
1077{
1078 if( IsProxyItem() )
1079 {
1080 switch( m_shape )
1081 {
1082 case SHAPE_T::RECTANGLE: return _( "Pad Number Box" );
1083 case SHAPE_T::SEGMENT: return _( "Thermal Spoke Template" );
1084 default: return _( "Unrecognized" );
1085 }
1086 }
1087 else
1088 {
1089 switch( m_shape )
1090 {
1091 case SHAPE_T::CIRCLE: return _( "Circle" );
1092 case SHAPE_T::ARC: return _( "Arc" );
1093 case SHAPE_T::BEZIER: return _( "Curve" );
1094 case SHAPE_T::POLY: return _( "Polygon" );
1095 case SHAPE_T::RECTANGLE: return _( "Rectangle" );
1096 case SHAPE_T::SEGMENT: return _( "Segment" );
1097 default: return _( "Unrecognized" );
1098 }
1099 }
1100}
1101
1102
1103void EDA_SHAPE::ShapeGetMsgPanelInfo( EDA_DRAW_FRAME* aFrame, std::vector<MSG_PANEL_ITEM>& aList )
1104{
1105 wxString msg;
1106
1107 wxString shape = _( "Shape" );
1108 aList.emplace_back( shape, getFriendlyName() );
1109
1110 switch( m_shape )
1111 {
1112 case SHAPE_T::CIRCLE:
1113 aList.emplace_back( _( "Radius" ), aFrame->MessageTextFromValue( GetRadius() ) );
1114 break;
1115
1116 case SHAPE_T::ARC:
1117 aList.emplace_back( _( "Length" ), aFrame->MessageTextFromValue( GetLength() ) );
1118
1120 aList.emplace_back( _( "Angle" ), msg );
1121
1122 aList.emplace_back( _( "Radius" ), aFrame->MessageTextFromValue( GetRadius() ) );
1123 break;
1124
1125 case SHAPE_T::BEZIER:
1126 aList.emplace_back( _( "Length" ), aFrame->MessageTextFromValue( GetLength() ) );
1127 break;
1128
1129 case SHAPE_T::POLY:
1130 msg.Printf( wxS( "%d" ), GetPolyShape().Outline(0).PointCount() );
1131 aList.emplace_back( _( "Points" ), msg );
1132 break;
1133
1134 case SHAPE_T::RECTANGLE:
1135 aList.emplace_back( _( "Width" ),
1136 aFrame->MessageTextFromValue( std::abs( GetEnd().x - GetStart().x ) ) );
1137
1138 aList.emplace_back( _( "Height" ),
1139 aFrame->MessageTextFromValue( std::abs( GetEnd().y - GetStart().y ) ) );
1140 break;
1141
1142 case SHAPE_T::SEGMENT:
1143 {
1144 aList.emplace_back( _( "Length" ),
1145 aFrame->MessageTextFromValue( GetStart().Distance( GetEnd() ) ));
1146
1147 // angle counter-clockwise from 3'o-clock
1148 EDA_ANGLE angle( atan2( (double)( GetStart().y - GetEnd().y ),
1149 (double)( GetEnd().x - GetStart().x ) ), RADIANS_T );
1150 aList.emplace_back( _( "Angle" ), EDA_UNIT_UTILS::UI::MessageTextFromValue( angle ) );
1151 break;
1152 }
1153
1154 default:
1155 break;
1156 }
1157
1158 m_stroke.GetMsgPanelInfo( aFrame, aList );
1159}
1160
1161
1163{
1164 BOX2I bbox;
1165
1166 switch( m_shape )
1167 {
1168 case SHAPE_T::RECTANGLE:
1169 for( VECTOR2I& pt : GetRectCorners() )
1170 bbox.Merge( pt );
1171
1172 break;
1173
1174 case SHAPE_T::SEGMENT:
1175 bbox.SetOrigin( GetStart() );
1176 bbox.SetEnd( GetEnd() );
1177 break;
1178
1179 case SHAPE_T::CIRCLE:
1180 bbox.SetOrigin( GetStart() );
1181 bbox.Inflate( GetRadius() );
1182 break;
1183
1184 case SHAPE_T::ARC:
1185 computeArcBBox( bbox );
1186 break;
1187
1188 case SHAPE_T::POLY:
1189 if( m_poly.IsEmpty() )
1190 break;
1191
1192 for( auto iter = m_poly.CIterate(); iter; iter++ )
1193 bbox.Merge( *iter );
1194
1195 break;
1196
1197 case SHAPE_T::BEZIER:
1198 // Bezier BBoxes are not trivial to compute, so we approximate it by
1199 // using the bounding box of the curve (not control!) points.
1200 for( const VECTOR2I& pt : m_bezierPoints )
1201 bbox.Merge( pt );
1202 break;
1203
1204 default:
1206 break;
1207 }
1208
1209 bbox.Inflate( std::max( 0, GetWidth() ) / 2 );
1210 bbox.Normalize();
1211
1212 return bbox;
1213}
1214
1215
1216bool EDA_SHAPE::hitTest( const VECTOR2I& aPosition, int aAccuracy ) const
1217{
1218 double maxdist = aAccuracy;
1219
1220 if( GetWidth() > 0 )
1221 maxdist += GetWidth() / 2.0;
1222
1223 switch( m_shape )
1224 {
1225 case SHAPE_T::CIRCLE:
1226 {
1227 double radius = GetRadius();
1228 double dist = aPosition.Distance( getCenter() );
1229
1230 if( IsFilledForHitTesting() )
1231 return dist <= radius + maxdist; // Filled circle hit-test
1232 else if( abs( radius - dist ) <= maxdist ) // Ring hit-test
1233 return true;
1234
1235 if( IsHatchedFill() && GetHatching().Collide( aPosition, maxdist ) )
1236 return true;
1237
1238 return false;
1239 }
1240
1241 case SHAPE_T::ARC:
1242 {
1243 if( aPosition.Distance( m_start ) <= maxdist )
1244 return true;
1245
1246 if( aPosition.Distance( m_end ) <= maxdist )
1247 return true;
1248
1249 double radius = GetRadius();
1250 VECTOR2D relPos( VECTOR2D( aPosition ) - getCenter() );
1251 double dist = relPos.EuclideanNorm();
1252
1253 if( IsFilledForHitTesting() )
1254 {
1255 // Check distance from arc center
1256 if( dist > radius + maxdist )
1257 return false;
1258 }
1259 else
1260 {
1261 // Check distance from arc circumference
1262 if( abs( radius - dist ) > maxdist )
1263 return false;
1264 }
1265
1266 // Finally, check to see if it's within arc's swept angle.
1267 EDA_ANGLE startAngle;
1268 EDA_ANGLE endAngle;
1269 CalcArcAngles( startAngle, endAngle );
1270
1271 EDA_ANGLE relPosAngle( relPos );
1272
1273 startAngle.Normalize();
1274 endAngle.Normalize();
1275 relPosAngle.Normalize();
1276
1277 if( endAngle > startAngle )
1278 return relPosAngle >= startAngle && relPosAngle <= endAngle;
1279 else
1280 return relPosAngle >= startAngle || relPosAngle <= endAngle;
1281 }
1282
1283 case SHAPE_T::BEZIER:
1284 {
1285 const std::vector<VECTOR2I>* pts = &m_bezierPoints;
1286 std::vector<VECTOR2I> updatedBezierPoints;
1287
1288 if( m_bezierPoints.empty() )
1289 {
1291 converter.GetPoly( updatedBezierPoints, aAccuracy / 2 );
1292 pts = &updatedBezierPoints;
1293 }
1294
1295 for( unsigned int i = 1; i < pts->size(); i++ )
1296 {
1297 if( TestSegmentHit( aPosition, ( *pts )[i - 1], ( *pts )[i], maxdist ) )
1298 return true;
1299 }
1300
1301 return false;
1302 }
1303 case SHAPE_T::SEGMENT:
1304 return TestSegmentHit( aPosition, GetStart(), GetEnd(), maxdist );
1305
1306 case SHAPE_T::RECTANGLE:
1307 if( IsProxyItem() || IsFilledForHitTesting() ) // Filled rect hit-test
1308 {
1309 SHAPE_POLY_SET poly;
1310 poly.NewOutline();
1311
1312 for( const VECTOR2I& pt : GetRectCorners() )
1313 poly.Append( pt );
1314
1315 return poly.Collide( aPosition, maxdist );
1316 }
1317 else
1318 {
1319 std::vector<VECTOR2I> pts = GetRectCorners();
1320
1321 if( TestSegmentHit( aPosition, pts[0], pts[1], maxdist )
1322 || TestSegmentHit( aPosition, pts[1], pts[2], maxdist )
1323 || TestSegmentHit( aPosition, pts[2], pts[3], maxdist )
1324 || TestSegmentHit( aPosition, pts[3], pts[0], maxdist ) )
1325 {
1326 return true;
1327 }
1328
1329 if( IsHatchedFill() && GetHatching().Collide( aPosition, maxdist ) )
1330 return true;
1331
1332 return false;
1333 }
1334
1335 case SHAPE_T::POLY:
1336 if( IsFilledForHitTesting() )
1337 {
1338 if( !m_poly.COutline( 0 ).IsClosed() )
1339 {
1340 // Only one outline is expected
1342 copy.SetClosed( true );
1343 return copy.Collide( aPosition, maxdist );
1344 }
1345 else
1346 {
1347 return m_poly.Collide( aPosition, maxdist );
1348 }
1349 }
1350 else
1351 {
1352 if( m_poly.CollideEdge( aPosition, nullptr, maxdist ) )
1353 return true;
1354
1355 if( IsHatchedFill() && GetHatching().Collide( aPosition, maxdist ) )
1356 return true;
1357
1358 return false;
1359 }
1360
1361 default:
1363 return false;
1364 }
1365}
1366
1367
1368bool EDA_SHAPE::hitTest( const BOX2I& aRect, bool aContained, int aAccuracy ) const
1369{
1370 BOX2I arect = aRect;
1371 arect.Normalize();
1372 arect.Inflate( aAccuracy );
1373
1374 BOX2I bbox = getBoundingBox();
1375
1376 switch( m_shape )
1377 {
1378 case SHAPE_T::CIRCLE:
1379 // Test if area intersects or contains the circle:
1380 if( aContained )
1381 {
1382 return arect.Contains( bbox );
1383 }
1384 else
1385 {
1386 // If the rectangle does not intersect the bounding box, this is a much quicker test
1387 if( !arect.Intersects( bbox ) )
1388 return false;
1389 else
1390 return arect.IntersectsCircleEdge( getCenter(), GetRadius(), GetWidth() );
1391 }
1392
1393 case SHAPE_T::ARC:
1394 // Test for full containment of this arc in the rect
1395 if( aContained )
1396 {
1397 return arect.Contains( bbox );
1398 }
1399 // Test if the rect crosses the arc
1400 else
1401 {
1402 if( !arect.Intersects( bbox ) )
1403 return false;
1404
1405 if( IsAnyFill() )
1406 {
1407 return ( arect.Intersects( getCenter(), GetStart() )
1408 || arect.Intersects( getCenter(), GetEnd() )
1409 || arect.IntersectsCircleEdge( getCenter(), GetRadius(), GetWidth() ) );
1410 }
1411 else
1412 {
1413 return arect.IntersectsCircleEdge( getCenter(), GetRadius(), GetWidth() );
1414 }
1415 }
1416
1417 case SHAPE_T::RECTANGLE:
1418 if( aContained )
1419 {
1420 return arect.Contains( bbox );
1421 }
1422 else
1423 {
1424 std::vector<VECTOR2I> pts = GetRectCorners();
1425
1426 // Account for the width of the lines
1427 arect.Inflate( GetWidth() / 2 );
1428 return ( arect.Intersects( pts[0], pts[1] )
1429 || arect.Intersects( pts[1], pts[2] )
1430 || arect.Intersects( pts[2], pts[3] )
1431 || arect.Intersects( pts[3], pts[0] ) );
1432 }
1433
1434 case SHAPE_T::SEGMENT:
1435 if( aContained )
1436 {
1437 return arect.Contains( GetStart() ) && aRect.Contains( GetEnd() );
1438 }
1439 else
1440 {
1441 // Account for the width of the line
1442 arect.Inflate( GetWidth() / 2 );
1443 return arect.Intersects( GetStart(), GetEnd() );
1444 }
1445
1446 case SHAPE_T::POLY:
1447 if( aContained )
1448 {
1449 return arect.Contains( bbox );
1450 }
1451 else
1452 {
1453 // Fast test: if aRect is outside the polygon bounding box,
1454 // rectangles cannot intersect
1455 if( !arect.Intersects( bbox ) )
1456 return false;
1457
1458 // Account for the width of the line
1459 arect.Inflate( GetWidth() / 2 );
1460
1461 for( int ii = 0; ii < m_poly.OutlineCount(); ++ii )
1462 {
1463 const SHAPE_LINE_CHAIN& poly = m_poly.Outline( ii );
1464 int count = poly.GetPointCount();
1465
1466 for( int jj = 0; jj < count; jj++ )
1467 {
1468 VECTOR2I vertex = poly.GetPoint( jj );
1469
1470 // Test if the point is within aRect
1471 if( arect.Contains( vertex ) )
1472 return true;
1473
1474 if( jj + 1 < count )
1475 {
1476 VECTOR2I vertexNext = poly.GetPoint( jj + 1 );
1477
1478 // Test if this edge intersects aRect
1479 if( arect.Intersects( vertex, vertexNext ) )
1480 return true;
1481 }
1482 else if( poly.IsClosed() )
1483 {
1484 VECTOR2I vertexNext = poly.GetPoint( 0 );
1485
1486 // Test if this edge intersects aRect
1487 if( arect.Intersects( vertex, vertexNext ) )
1488 return true;
1489 }
1490 }
1491 }
1492
1493 return false;
1494 }
1495
1496 case SHAPE_T::BEZIER:
1497 if( aContained )
1498 {
1499 return arect.Contains( bbox );
1500 }
1501 else
1502 {
1503 // Fast test: if aRect is outside the polygon bounding box,
1504 // rectangles cannot intersect
1505 if( !arect.Intersects( bbox ) )
1506 return false;
1507
1508 // Account for the width of the line
1509 arect.Inflate( GetWidth() / 2 );
1510 const std::vector<VECTOR2I>* pts = &m_bezierPoints;
1511 std::vector<VECTOR2I> updatedBezierPoints;
1512
1513 if( m_bezierPoints.empty() )
1514 {
1516 converter.GetPoly( updatedBezierPoints, aAccuracy / 2 );
1517 pts = &updatedBezierPoints;
1518 }
1519
1520 for( unsigned ii = 1; ii < pts->size(); ii++ )
1521 {
1522 VECTOR2I vertex = ( *pts )[ii - 1];
1523 VECTOR2I vertexNext = ( *pts )[ii];
1524
1525 // Test if the point is within aRect
1526 if( arect.Contains( vertex ) )
1527 return true;
1528
1529 // Test if this edge intersects aRect
1530 if( arect.Intersects( vertex, vertexNext ) )
1531 return true;
1532 }
1533
1534 return false;
1535 }
1536
1537 default:
1539 return false;
1540 }
1541}
1542
1543
1544std::vector<VECTOR2I> EDA_SHAPE::GetRectCorners() const
1545{
1546 std::vector<VECTOR2I> pts;
1547 VECTOR2I topLeft = GetStart();
1548 VECTOR2I botRight = GetEnd();
1549
1550 pts.emplace_back( topLeft );
1551 pts.emplace_back( botRight.x, topLeft.y );
1552 pts.emplace_back( botRight );
1553 pts.emplace_back( topLeft.x, botRight.y );
1554
1555 return pts;
1556}
1557
1558
1559std::vector<VECTOR2I> EDA_SHAPE::GetCornersInSequence() const
1560{
1561 std::vector<VECTOR2I> pts;
1562 EDA_ANGLE textAngle( getDrawRotation() );
1563
1564 textAngle.Normalize();
1565
1566 BOX2I bbox = getBoundingBox();
1567 bbox.Normalize();
1568
1569 if( textAngle.IsCardinal() )
1570 {
1571 if( textAngle == ANGLE_0 )
1572 {
1573 pts.emplace_back( VECTOR2I( bbox.GetLeft(), bbox.GetTop() ) );
1574 pts.emplace_back( VECTOR2I( bbox.GetRight(), bbox.GetTop() ) );
1575 pts.emplace_back( VECTOR2I( bbox.GetRight(), bbox.GetBottom() ) );
1576 pts.emplace_back( VECTOR2I( bbox.GetLeft(), bbox.GetBottom() ) );
1577 }
1578 else if( textAngle == ANGLE_90 )
1579 {
1580 pts.emplace_back( VECTOR2I( bbox.GetLeft(), bbox.GetBottom() ) );
1581 pts.emplace_back( VECTOR2I( bbox.GetLeft(), bbox.GetTop() ) );
1582 pts.emplace_back( VECTOR2I( bbox.GetRight(), bbox.GetTop() ) );
1583 pts.emplace_back( VECTOR2I( bbox.GetRight(), bbox.GetBottom() ) );
1584 }
1585 else if( textAngle == ANGLE_180 )
1586 {
1587 pts.emplace_back( VECTOR2I( bbox.GetRight(), bbox.GetBottom() ) );
1588 pts.emplace_back( VECTOR2I( bbox.GetLeft(), bbox.GetBottom() ) );
1589 pts.emplace_back( VECTOR2I( bbox.GetLeft(), bbox.GetTop() ) );
1590 pts.emplace_back( VECTOR2I( bbox.GetRight(), bbox.GetTop() ) );
1591 }
1592 else if( textAngle == ANGLE_270 )
1593 {
1594 pts.emplace_back( VECTOR2I( bbox.GetRight(), bbox.GetTop() ) );
1595 pts.emplace_back( VECTOR2I( bbox.GetRight(), bbox.GetBottom() ) );
1596 pts.emplace_back( VECTOR2I( bbox.GetLeft(), bbox.GetBottom() ) );
1597 pts.emplace_back( VECTOR2I( bbox.GetLeft(), bbox.GetTop() ) );
1598 }
1599 }
1600 else
1601 {
1602 std::vector<VECTOR2I> corners = GetRectCorners();
1603
1604 VECTOR2I minX = corners[0];
1605 VECTOR2I maxX = corners[0];
1606 VECTOR2I minY = corners[0];
1607 VECTOR2I maxY = corners[0];
1608
1609 for( const VECTOR2I& corner : corners )
1610 {
1611 if( corner.x < minX.x )
1612 minX = corner;
1613
1614 if( corner.x > maxX.x )
1615 maxX = corner;
1616
1617 if( corner.y < minY.y )
1618 minY = corner;
1619
1620 if( corner.y > maxY.y )
1621 maxY = corner;
1622 }
1623
1624 if( textAngle < ANGLE_90 )
1625 {
1626 pts.emplace_back( minX );
1627 pts.emplace_back( minY );
1628 pts.emplace_back( maxX );
1629 pts.emplace_back( maxY );
1630 }
1631 else if( textAngle < ANGLE_180 )
1632 {
1633 pts.emplace_back( maxY );
1634 pts.emplace_back( minX );
1635 pts.emplace_back( minY );
1636 pts.emplace_back( maxX );
1637 }
1638 else if( textAngle < ANGLE_270 )
1639 {
1640 pts.emplace_back( maxX );
1641 pts.emplace_back( maxY );
1642 pts.emplace_back( minX );
1643 pts.emplace_back( minY );
1644 }
1645 else
1646 {
1647 pts.emplace_back( minY );
1648 pts.emplace_back( maxX );
1649 pts.emplace_back( maxY );
1650 pts.emplace_back( minX );
1651 }
1652 }
1653
1654 return pts;
1655}
1656
1657
1659{
1660 // Start, end, and each inflection point the arc crosses will enclose the entire arc.
1661 // Only include the center when filled; it's not necessarily inside the BB of an unfilled
1662 // arc with a small included angle.
1663 aBBox.SetOrigin( m_start );
1664 aBBox.Merge( m_end );
1665
1666 if( IsAnyFill() )
1667 aBBox.Merge( m_arcCenter );
1668
1669 int radius = GetRadius();
1670 EDA_ANGLE t1, t2;
1671
1672 CalcArcAngles( t1, t2 );
1673
1674 t1.Normalize();
1675 t2.Normalize();
1676
1677 if( t2 > t1 )
1678 {
1679 if( t1 < ANGLE_0 && t2 > ANGLE_0 )
1680 aBBox.Merge( VECTOR2I( m_arcCenter.x + radius, m_arcCenter.y ) ); // right
1681
1682 if( t1 < ANGLE_90 && t2 > ANGLE_90 )
1683 aBBox.Merge( VECTOR2I( m_arcCenter.x, m_arcCenter.y + radius ) ); // down
1684
1685 if( t1 < ANGLE_180 && t2 > ANGLE_180 )
1686 aBBox.Merge( VECTOR2I( m_arcCenter.x - radius, m_arcCenter.y ) ); // left
1687
1688 if( t1 < ANGLE_270 && t2 > ANGLE_270 )
1689 aBBox.Merge( VECTOR2I( m_arcCenter.x, m_arcCenter.y - radius ) ); // up
1690 }
1691 else
1692 {
1693 if( t1 < ANGLE_0 || t2 > ANGLE_0 )
1694 aBBox.Merge( VECTOR2I( m_arcCenter.x + radius, m_arcCenter.y ) ); // right
1695
1696 if( t1 < ANGLE_90 || t2 > ANGLE_90 )
1697 aBBox.Merge( VECTOR2I( m_arcCenter.x, m_arcCenter.y + radius ) ); // down
1698
1699 if( t1 < ANGLE_180 || t2 > ANGLE_180 )
1700 aBBox.Merge( VECTOR2I( m_arcCenter.x - radius, m_arcCenter.y ) ); // left
1701
1702 if( t1 < ANGLE_270 || t2 > ANGLE_270 )
1703 aBBox.Merge( VECTOR2I( m_arcCenter.x, m_arcCenter.y - radius ) ); // up
1704 }
1705}
1706
1707
1708void EDA_SHAPE::SetPolyPoints( const std::vector<VECTOR2I>& aPoints )
1709{
1712
1713 for( const VECTOR2I& p : aPoints )
1714 m_poly.Append( p.x, p.y );
1715}
1716
1717
1718std::vector<SHAPE*> EDA_SHAPE::makeEffectiveShapes( bool aEdgeOnly, bool aLineChainOnly ) const
1719{
1720 std::vector<SHAPE*> effectiveShapes;
1721 int width = GetEffectiveWidth();
1722 bool solidFill = ( IsSolidFill() || IsHatchedFill() || IsProxyItem() ) && !aEdgeOnly;
1723
1724 switch( m_shape )
1725 {
1726 case SHAPE_T::ARC:
1727 effectiveShapes.emplace_back( new SHAPE_ARC( m_arcCenter, m_start, GetArcAngle(), width ) );
1728 break;
1729
1730 case SHAPE_T::SEGMENT:
1731 effectiveShapes.emplace_back( new SHAPE_SEGMENT( m_start, m_end, width ) );
1732 break;
1733
1734 case SHAPE_T::RECTANGLE:
1735 {
1736 std::vector<VECTOR2I> pts = GetRectCorners();
1737
1738 if( solidFill )
1739 effectiveShapes.emplace_back( new SHAPE_SIMPLE( pts ) );
1740
1741 if( width > 0 || !solidFill )
1742 {
1743 effectiveShapes.emplace_back( new SHAPE_SEGMENT( pts[0], pts[1], width ) );
1744 effectiveShapes.emplace_back( new SHAPE_SEGMENT( pts[1], pts[2], width ) );
1745 effectiveShapes.emplace_back( new SHAPE_SEGMENT( pts[2], pts[3], width ) );
1746 effectiveShapes.emplace_back( new SHAPE_SEGMENT( pts[3], pts[0], width ) );
1747 }
1748 break;
1749 }
1750
1751 case SHAPE_T::CIRCLE:
1752 {
1753 if( solidFill )
1754 effectiveShapes.emplace_back( new SHAPE_CIRCLE( getCenter(), GetRadius() ) );
1755
1756 if( width > 0 || !solidFill )
1757 effectiveShapes.emplace_back( new SHAPE_ARC( getCenter(), GetEnd(), ANGLE_360, width ) );
1758
1759 break;
1760 }
1761
1762 case SHAPE_T::BEZIER:
1763 {
1764 std::vector<VECTOR2I> bezierPoints = buildBezierToSegmentsPointsList( width / 2 );
1765 VECTOR2I start_pt = bezierPoints[0];
1766
1767 for( unsigned int jj = 1; jj < bezierPoints.size(); jj++ )
1768 {
1769 VECTOR2I end_pt = bezierPoints[jj];
1770 effectiveShapes.emplace_back( new SHAPE_SEGMENT( start_pt, end_pt, width ) );
1771 start_pt = end_pt;
1772 }
1773
1774 break;
1775 }
1776
1777 case SHAPE_T::POLY:
1778 {
1779 if( GetPolyShape().OutlineCount() == 0 ) // malformed/empty polygon
1780 break;
1781
1782 for( int ii = 0; ii < GetPolyShape().OutlineCount(); ++ii )
1783 {
1784 const SHAPE_LINE_CHAIN& l = GetPolyShape().COutline( ii );
1785
1786 if( solidFill )
1787 effectiveShapes.emplace_back( new SHAPE_SIMPLE( l ) );
1788
1789 if( width > 0 || !IsSolidFill() || aEdgeOnly )
1790 {
1791 int segCount = l.SegmentCount();
1792
1793 if( aLineChainOnly && l.IsClosed() )
1794 segCount--; // Treat closed chain as open
1795
1796 for( int jj = 0; jj < segCount; jj++ )
1797 effectiveShapes.emplace_back( new SHAPE_SEGMENT( l.CSegment( jj ), width ) );
1798 }
1799 }
1800 }
1801 break;
1802
1803 default:
1805 break;
1806 }
1807
1808 return effectiveShapes;
1809}
1810
1811
1812void EDA_SHAPE::DupPolyPointsList( std::vector<VECTOR2I>& aBuffer ) const
1813{
1814 for( int ii = 0; ii < m_poly.OutlineCount(); ++ii )
1815 {
1816 int pointCount = m_poly.COutline( ii ).PointCount();
1817
1818 if( pointCount )
1819 {
1820 aBuffer.reserve( pointCount );
1821
1822 for ( auto iter = m_poly.CIterate(); iter; iter++ )
1823 aBuffer.emplace_back( iter->x, iter->y );
1824 }
1825 }
1826}
1827
1828
1830{
1831 // return true if the polygonal shape is valid (has more than 2 points)
1832 return GetPolyShape().OutlineCount() > 0 && GetPolyShape().Outline( 0 ).PointCount() > 2;
1833}
1834
1835
1837{
1838 // return the number of corners of the polygonal shape
1839 // this shape is expected to be only one polygon without hole
1840 return GetPolyShape().OutlineCount() ? GetPolyShape().VertexCount( 0 ) : 0;
1841}
1842
1843
1844void EDA_SHAPE::beginEdit( const VECTOR2I& aPosition )
1845{
1846 switch( GetShape() )
1847 {
1848 case SHAPE_T::SEGMENT:
1849 case SHAPE_T::CIRCLE:
1850 case SHAPE_T::RECTANGLE:
1851 SetStart( aPosition );
1852 SetEnd( aPosition );
1853 break;
1854
1855 case SHAPE_T::ARC:
1856 SetArcGeometry( aPosition, aPosition, aPosition );
1857 m_editState = 1;
1858 break;
1859
1860 case SHAPE_T::BEZIER:
1861 SetStart( aPosition );
1862 SetEnd( aPosition );
1863 SetBezierC1( aPosition );
1864 SetBezierC2( aPosition );
1865 m_editState = 1;
1866
1868 break;
1869
1870 case SHAPE_T::POLY:
1872 m_poly.Outline( 0 ).SetClosed( false );
1873
1874 // Start and end of the first segment (co-located for now)
1875 m_poly.Outline( 0 ).Append( aPosition );
1876 m_poly.Outline( 0 ).Append( aPosition, true );
1877 break;
1878
1879 default:
1881 }
1882}
1883
1884
1885bool EDA_SHAPE::continueEdit( const VECTOR2I& aPosition )
1886{
1887 switch( GetShape() )
1888 {
1889 case SHAPE_T::ARC:
1890 case SHAPE_T::SEGMENT:
1891 case SHAPE_T::CIRCLE:
1892 case SHAPE_T::RECTANGLE:
1893 return false;
1894
1895 case SHAPE_T::BEZIER:
1896 if( m_editState == 3 )
1897 return false;
1898
1899 m_editState++;
1900 return true;
1901
1902 case SHAPE_T::POLY:
1903 {
1904 SHAPE_LINE_CHAIN& poly = m_poly.Outline( 0 );
1905
1906 // do not add zero-length segments
1907 if( poly.CPoint( poly.GetPointCount() - 2 ) != poly.CLastPoint() )
1908 poly.Append( aPosition, true );
1909 }
1910 return true;
1911
1912 default:
1914 return false;
1915 }
1916}
1917
1918
1919void EDA_SHAPE::calcEdit( const VECTOR2I& aPosition )
1920{
1921#define sq( x ) pow( x, 2 )
1922
1923 switch( GetShape() )
1924 {
1925 case SHAPE_T::SEGMENT:
1926 case SHAPE_T::CIRCLE:
1927 case SHAPE_T::RECTANGLE:
1928 SetEnd( aPosition );
1929 break;
1930
1931 case SHAPE_T::BEZIER:
1932 {
1933 switch( m_editState )
1934 {
1935 case 0:
1936 SetStart( aPosition );
1937 SetEnd( aPosition );
1938 SetBezierC1( aPosition );
1939 SetBezierC2( aPosition );
1940 break;
1941 case 1:
1942 SetBezierC2( aPosition );
1943 SetEnd( aPosition );
1944 break;
1945 case 2: SetBezierC1( aPosition ); break;
1946 case 3: SetBezierC2( aPosition ); break;
1947 }
1948
1950 }
1951 break;
1952
1953 case SHAPE_T::ARC:
1954 {
1955 double radius = GetRadius();
1956 EDA_ANGLE lastAngle = GetArcAngle();
1957
1958 // Edit state 0: drawing: place start
1959 // Edit state 1: drawing: place end (center calculated for 90-degree subtended angle)
1960 // Edit state 2: point edit: move start (center calculated for invariant subtended angle)
1961 // Edit state 3: point edit: move end (center calculated for invariant subtended angle)
1962 // Edit state 4: point edit: move center
1963 // Edit state 5: point edit: move arc-mid-point
1964
1965 switch( m_editState )
1966 {
1967 case 0:
1968 SetArcGeometry( aPosition, aPosition, aPosition );
1969 return;
1970
1971 case 1:
1972 m_end = aPosition;
1973 radius = m_start.Distance( m_end ) * M_SQRT1_2;
1974 break;
1975
1976 case 2:
1977 case 3:
1978 {
1979 VECTOR2I v = m_start - m_end;
1980 double chordBefore = v.SquaredEuclideanNorm();
1981
1982 if( m_editState == 2 )
1983 m_start = aPosition;
1984 else
1985 m_end = aPosition;
1986
1987 v = m_start - m_end;
1988
1989 double chordAfter = v.SquaredEuclideanNorm();
1990 double ratio = 0.0;
1991
1992 if( chordBefore > 0 )
1993 ratio = chordAfter / chordBefore;
1994
1995 if( ratio != 0 )
1996 radius = std::max( sqrt( sq( radius ) * ratio ), sqrt( chordAfter ) / 2 );
1997 }
1998 break;
1999
2000 case 4:
2001 {
2002 double radialA = m_start.Distance( aPosition );
2003 double radialB = m_end.Distance( aPosition );
2004 radius = ( radialA + radialB ) / 2.0;
2005 }
2006 break;
2007
2008 case 5:
2009 SetArcGeometry( GetStart(), aPosition, GetEnd() );
2010 return;
2011 }
2012
2013 // Calculate center based on start, end, and radius
2014 //
2015 // Let 'l' be the length of the chord and 'm' the middle point of the chord
2016 double l = m_start.Distance( m_end );
2017 VECTOR2D m = ( m_start + m_end ) / 2;
2018 double sqRadDiff = ( radius * radius ) - ( l * l ) / 4.0;
2019
2020 // Calculate 'd', the vector from the chord midpoint to the center
2021 VECTOR2D d;
2022
2023 if( l > 0 && sqRadDiff >= 0 )
2024 {
2025 d.x = sqrt( sqRadDiff ) * ( m_start.y - m_end.y ) / l;
2026 d.y = sqrt( sqRadDiff ) * ( m_end.x - m_start.x ) / l;
2027 }
2028
2029 VECTOR2I c1 = KiROUND( m + d );
2030 VECTOR2I c2 = KiROUND( m - d );
2031
2032 // Solution gives us 2 centers; we need to pick one:
2033 switch( m_editState )
2034 {
2035 case 1:
2036 // Keep arc clockwise while drawing i.e. arc angle = 90 deg.
2037 // it can be 90 or 270 deg depending on the arc center choice (c1 or c2)
2038 m_arcCenter = c1; // first trial
2039
2040 if( GetArcAngle() > ANGLE_180 )
2041 m_arcCenter = c2;
2042
2043 break;
2044
2045 case 2:
2046 case 3:
2047 // Pick the one of c1, c2 to keep arc on the same side
2048 m_arcCenter = c1; // first trial
2049
2050 if( ( lastAngle < ANGLE_180 ) != ( GetArcAngle() < ANGLE_180 ) )
2051 m_arcCenter = c2;
2052
2053 break;
2054
2055 case 4:
2056 // Pick the one closer to the mouse position
2057 m_arcCenter = c1.Distance( aPosition ) < c2.Distance( aPosition ) ? c1 : c2;
2058 break;
2059 }
2060 }
2061 break;
2062
2063 case SHAPE_T::POLY:
2064 m_poly.Outline( 0 ).SetPoint( m_poly.Outline( 0 ).GetPointCount() - 1, aPosition );
2065 break;
2066
2067 default:
2069 }
2070}
2071
2072
2073void EDA_SHAPE::endEdit( bool aClosed )
2074{
2075 switch( GetShape() )
2076 {
2077 case SHAPE_T::ARC:
2078 case SHAPE_T::SEGMENT:
2079 case SHAPE_T::CIRCLE:
2080 case SHAPE_T::RECTANGLE:
2081 case SHAPE_T::BEZIER:
2082 break;
2083
2084 case SHAPE_T::POLY:
2085 {
2086 SHAPE_LINE_CHAIN& poly = m_poly.Outline( 0 );
2087
2088 // do not include last point twice
2089 if( poly.GetPointCount() > 2 )
2090 {
2091 if( poly.CPoint( poly.GetPointCount() - 2 ) == poly.CLastPoint() )
2092 {
2093 poly.SetClosed( aClosed );
2094 }
2095 else
2096 {
2097 poly.SetClosed( false );
2098 poly.Remove( poly.GetPointCount() - 1 );
2099 }
2100 }
2101 }
2102 break;
2103
2104 default:
2106 }
2107}
2108
2109
2111{
2112 EDA_SHAPE* image = dynamic_cast<EDA_SHAPE*>( aImage );
2113 assert( image );
2114
2115 #define SWAPITEM( x ) std::swap( x, image->x )
2116 SWAPITEM( m_stroke );
2117 SWAPITEM( m_start );
2118 SWAPITEM( m_end );
2120 SWAPITEM( m_shape );
2124 SWAPITEM( m_poly );
2125 SWAPITEM( m_fill );
2129 #undef SWAPITEM
2130
2131 m_hatchingDirty = true;
2132}
2133
2134
2135int EDA_SHAPE::Compare( const EDA_SHAPE* aOther ) const
2136{
2137#define EPSILON 2 // Should be enough for rounding errors on calculated items
2138
2139#define TEST( a, b ) { if( a != b ) return a - b; }
2140#define TEST_E( a, b ) { if( abs( a - b ) > EPSILON ) return a - b; }
2141#define TEST_PT( a, b ) { TEST_E( a.x, b.x ); TEST_E( a.y, b.y ); }
2142
2143 TEST_PT( m_start, aOther->m_start );
2144 TEST_PT( m_end, aOther->m_end );
2145
2146 TEST( (int) m_shape, (int) aOther->m_shape );
2147
2148 if( m_shape == SHAPE_T::ARC )
2149 {
2150 TEST_PT( m_arcCenter, aOther->m_arcCenter );
2151 }
2152 else if( m_shape == SHAPE_T::BEZIER )
2153 {
2154 TEST_PT( m_bezierC1, aOther->m_bezierC1 );
2155 TEST_PT( m_bezierC2, aOther->m_bezierC2 );
2156 }
2157 else if( m_shape == SHAPE_T::POLY )
2158 {
2160 }
2161
2162 for( size_t ii = 0; ii < m_bezierPoints.size(); ++ii )
2163 TEST_PT( m_bezierPoints[ii], aOther->m_bezierPoints[ii] );
2164
2165 for( int ii = 0; ii < m_poly.TotalVertices(); ++ii )
2166 TEST_PT( m_poly.CVertex( ii ), aOther->m_poly.CVertex( ii ) );
2167
2168 TEST_E( m_stroke.GetWidth(), aOther->m_stroke.GetWidth() );
2169 TEST( (int) m_stroke.GetLineStyle(), (int) aOther->m_stroke.GetLineStyle() );
2170 TEST( (int) m_fill, (int) aOther->m_fill );
2171
2172 return 0;
2173}
2174
2175
2176void EDA_SHAPE::TransformShapeToPolygon( SHAPE_POLY_SET& aBuffer, int aClearance, int aError,
2177 ERROR_LOC aErrorLoc, bool ignoreLineWidth,
2178 bool includeFill ) const
2179{
2180 bool solidFill = IsSolidFill() || ( IsHatchedFill() && !includeFill ) || IsProxyItem();
2181 int width = ignoreLineWidth ? 0 : GetWidth();
2182
2183 width += 2 * aClearance;
2184
2185 switch( m_shape )
2186 {
2187 case SHAPE_T::CIRCLE:
2188 {
2189 int r = GetRadius();
2190
2191 if( solidFill )
2192 TransformCircleToPolygon( aBuffer, getCenter(), r + width / 2, aError, aErrorLoc );
2193 else
2194 TransformRingToPolygon( aBuffer, getCenter(), r, width, aError, aErrorLoc );
2195
2196 break;
2197 }
2198
2199 case SHAPE_T::RECTANGLE:
2200 {
2201 std::vector<VECTOR2I> pts = GetRectCorners();
2202
2203 if( solidFill )
2204 {
2205 aBuffer.NewOutline();
2206
2207 for( const VECTOR2I& pt : pts )
2208 aBuffer.Append( pt );
2209 }
2210
2211 if( width > 0 || !solidFill )
2212 {
2213 // Add in segments
2214 TransformOvalToPolygon( aBuffer, pts[0], pts[1], width, aError, aErrorLoc );
2215 TransformOvalToPolygon( aBuffer, pts[1], pts[2], width, aError, aErrorLoc );
2216 TransformOvalToPolygon( aBuffer, pts[2], pts[3], width, aError, aErrorLoc );
2217 TransformOvalToPolygon( aBuffer, pts[3], pts[0], width, aError, aErrorLoc );
2218 }
2219
2220 break;
2221 }
2222
2223 case SHAPE_T::ARC:
2224 TransformArcToPolygon( aBuffer, GetStart(), GetArcMid(), GetEnd(), width, aError, aErrorLoc );
2225 break;
2226
2227 case SHAPE_T::SEGMENT:
2228 TransformOvalToPolygon( aBuffer, GetStart(), GetEnd(), width, aError, aErrorLoc );
2229 break;
2230
2231 case SHAPE_T::POLY:
2232 {
2233 if( !IsPolyShapeValid() )
2234 break;
2235
2236 if( solidFill )
2237 {
2238 for( int ii = 0; ii < m_poly.OutlineCount(); ++ii )
2239 {
2240 const SHAPE_LINE_CHAIN& poly = m_poly.Outline( ii );
2241 SHAPE_POLY_SET tmp;
2242 tmp.NewOutline();
2243
2244 for( int jj = 0; jj < (int) poly.GetPointCount(); ++jj )
2245 tmp.Append( poly.GetPoint( jj ) );
2246
2247 if( width > 0 )
2248 {
2249 int inflate = width / 2;
2250
2251 if( aErrorLoc == ERROR_OUTSIDE )
2252 inflate += aError;
2253
2254 tmp.Inflate( inflate, CORNER_STRATEGY::ROUND_ALL_CORNERS, aError );
2255 }
2256
2257 aBuffer.Append( tmp );
2258 }
2259 }
2260 else
2261 {
2262 for( int ii = 0; ii < m_poly.OutlineCount(); ++ii )
2263 {
2264 const SHAPE_LINE_CHAIN& poly = m_poly.Outline( ii );
2265
2266 for( int jj = 0; jj < (int) poly.SegmentCount(); ++jj )
2267 {
2268 const SEG& seg = poly.GetSegment( jj );
2269 TransformOvalToPolygon( aBuffer, seg.A, seg.B, width, aError, aErrorLoc );
2270 }
2271 }
2272 }
2273
2274 break;
2275 }
2276
2277 case SHAPE_T::BEZIER:
2278 {
2279 std::vector<VECTOR2I> ctrlPts = { GetStart(), GetBezierC1(), GetBezierC2(), GetEnd() };
2280 BEZIER_POLY converter( ctrlPts );
2281 std::vector<VECTOR2I> poly;
2282 converter.GetPoly( poly, aError );
2283
2284 for( unsigned ii = 1; ii < poly.size(); ii++ )
2285 TransformOvalToPolygon( aBuffer, poly[ii - 1], poly[ii], width, aError, aErrorLoc );
2286
2287 break;
2288 }
2289
2290 default:
2292 break;
2293 }
2294
2295 if( IsHatchedFill() && includeFill )
2296 {
2297 for( int ii = 0; ii < GetHatching().OutlineCount(); ++ii )
2298 aBuffer.AddOutline( GetHatching().COutline( ii ) );
2299 }
2300}
2301
2302
2303void EDA_SHAPE::SetWidth( int aWidth )
2304{
2305 m_stroke.SetWidth( aWidth );
2306 m_hatchingDirty = true;
2307}
2308
2309
2311{
2312 m_stroke.SetLineStyle( aStyle );
2313}
2314
2315
2317{
2318 if( m_stroke.GetLineStyle() != LINE_STYLE::DEFAULT )
2319 return m_stroke.GetLineStyle();
2320
2321 return LINE_STYLE::SOLID;
2322}
2323
2324
2325bool EDA_SHAPE::operator==( const EDA_SHAPE& aOther ) const
2326{
2327 if( GetShape() != aOther.GetShape() )
2328 return false;
2329
2330 if( m_fill != aOther.m_fill )
2331 return false;
2332
2333 if( m_stroke.GetWidth() != aOther.m_stroke.GetWidth() )
2334 return false;
2335
2336 if( m_stroke.GetLineStyle() != aOther.m_stroke.GetLineStyle() )
2337 return false;
2338
2339 if( m_fillColor != aOther.m_fillColor )
2340 return false;
2341
2342 if( m_start != aOther.m_start )
2343 return false;
2344
2345 if( m_end != aOther.m_end )
2346 return false;
2347
2348 if( m_arcCenter != aOther.m_arcCenter )
2349 return false;
2350
2351 if( m_bezierC1 != aOther.m_bezierC1 )
2352 return false;
2353
2354 if( m_bezierC2 != aOther.m_bezierC2 )
2355 return false;
2356
2357 if( m_bezierPoints != aOther.m_bezierPoints )
2358 return false;
2359
2360 for( int ii = 0; ii < m_poly.TotalVertices(); ++ii )
2361 {
2362 if( m_poly.CVertex( ii ) != aOther.m_poly.CVertex( ii ) )
2363 return false;
2364 }
2365
2366 return true;
2367}
2368
2369
2370double EDA_SHAPE::Similarity( const EDA_SHAPE& aOther ) const
2371{
2372 if( GetShape() != aOther.GetShape() )
2373 return 0.0;
2374
2375 double similarity = 1.0;
2376
2377 if( m_fill != aOther.m_fill )
2378 similarity *= 0.9;
2379
2380 if( m_stroke.GetWidth() != aOther.m_stroke.GetWidth() )
2381 similarity *= 0.9;
2382
2383 if( m_stroke.GetLineStyle() != aOther.m_stroke.GetLineStyle() )
2384 similarity *= 0.9;
2385
2386 if( m_fillColor != aOther.m_fillColor )
2387 similarity *= 0.9;
2388
2389 if( m_start != aOther.m_start )
2390 similarity *= 0.9;
2391
2392 if( m_end != aOther.m_end )
2393 similarity *= 0.9;
2394
2395 if( m_arcCenter != aOther.m_arcCenter )
2396 similarity *= 0.9;
2397
2398 if( m_bezierC1 != aOther.m_bezierC1 )
2399 similarity *= 0.9;
2400
2401 if( m_bezierC2 != aOther.m_bezierC2 )
2402 similarity *= 0.9;
2403
2404 {
2405 int m = m_bezierPoints.size();
2406 int n = aOther.m_bezierPoints.size();
2407
2408 size_t longest = alg::longest_common_subset( m_bezierPoints, aOther.m_bezierPoints );
2409
2410 similarity *= std::pow( 0.9, m + n - 2 * longest );
2411 }
2412
2413 {
2414 int m = m_poly.TotalVertices();
2415 int n = aOther.m_poly.TotalVertices();
2416 std::vector<VECTOR2I> poly;
2417 std::vector<VECTOR2I> otherPoly;
2418 VECTOR2I lastPt( 0, 0 );
2419
2420 // We look for the longest common subset of the two polygons, but we need to
2421 // offset each point because we're actually looking for overall similarity, not just
2422 // exact matches. So if the zone is moved by 1IU, we only want one point to be
2423 // considered "moved" rather than the entire polygon. In this case, the first point
2424 // will not be a match but the rest of the sequence will.
2425 for( int ii = 0; ii < m; ++ii )
2426 {
2427 poly.emplace_back( lastPt - m_poly.CVertex( ii ) );
2428 lastPt = m_poly.CVertex( ii );
2429 }
2430
2431 lastPt = VECTOR2I( 0, 0 );
2432
2433 for( int ii = 0; ii < n; ++ii )
2434 {
2435 otherPoly.emplace_back( lastPt - aOther.m_poly.CVertex( ii ) );
2436 lastPt = aOther.m_poly.CVertex( ii );
2437 }
2438
2439 size_t longest = alg::longest_common_subset( poly, otherPoly );
2440
2441 similarity *= std::pow( 0.9, m + n - 2 * longest );
2442 }
2443
2444 return similarity;
2445}
2446
2447
2451
2452
2453static struct EDA_SHAPE_DESC
2454{
2456 {
2458 .Map( SHAPE_T::SEGMENT, _HKI( "Segment" ) )
2459 .Map( SHAPE_T::RECTANGLE, _HKI( "Rectangle" ) )
2460 .Map( SHAPE_T::ARC, _HKI( "Arc" ) )
2461 .Map( SHAPE_T::CIRCLE, _HKI( "Circle" ) )
2462 .Map( SHAPE_T::POLY, _HKI( "Polygon" ) )
2463 .Map( SHAPE_T::BEZIER, _HKI( "Bezier" ) );
2464
2466
2467 if( lineStyleEnum.Choices().GetCount() == 0 )
2468 {
2469 lineStyleEnum.Map( LINE_STYLE::SOLID, _HKI( "Solid" ) )
2470 .Map( LINE_STYLE::DASH, _HKI( "Dashed" ) )
2471 .Map( LINE_STYLE::DOT, _HKI( "Dotted" ) )
2472 .Map( LINE_STYLE::DASHDOT, _HKI( "Dash-Dot" ) )
2473 .Map( LINE_STYLE::DASHDOTDOT, _HKI( "Dash-Dot-Dot" ) );
2474 }
2475
2477
2478 if( hatchModeEnum.Choices().GetCount() == 0 )
2479 {
2480 hatchModeEnum.Map( UI_FILL_MODE::NONE, _HKI( "None" ) );
2481 hatchModeEnum.Map( UI_FILL_MODE::SOLID, _HKI( "Solid" ) );
2482 hatchModeEnum.Map( UI_FILL_MODE::HATCH, _HKI( "Hatch" ) );
2483 hatchModeEnum.Map( UI_FILL_MODE::REVERSE_HATCH, _HKI( "Reverse Hatch" ) );
2484 hatchModeEnum.Map( UI_FILL_MODE::CROSS_HATCH, _HKI( "Cross-hatch" ) );
2485 }
2486
2489
2490 auto isNotPolygonOrCircle = []( INSPECTABLE* aItem ) -> bool
2491 {
2492 // Polygons, unlike other shapes, have no meaningful start or end coordinates
2493 if( EDA_SHAPE* shape = dynamic_cast<EDA_SHAPE*>( aItem ) )
2494 return shape->GetShape() != SHAPE_T::POLY && shape->GetShape() != SHAPE_T::CIRCLE;
2495
2496 return false;
2497 };
2498
2499 auto isCircle = []( INSPECTABLE* aItem ) -> bool
2500 {
2501 // Polygons, unlike other shapes, have no meaningful start or end coordinates
2502 if( EDA_SHAPE* shape = dynamic_cast<EDA_SHAPE*>( aItem ) )
2503 return shape->GetShape() == SHAPE_T::CIRCLE;
2504
2505 return false;
2506 };
2507
2508 auto isRectangle = []( INSPECTABLE* aItem ) -> bool
2509 {
2510 // Polygons, unlike other shapes, have no meaningful start or end coordinates
2511 if( EDA_SHAPE* shape = dynamic_cast<EDA_SHAPE*>( aItem ) )
2512 return shape->GetShape() == SHAPE_T::RECTANGLE;
2513
2514 return false;
2515 };
2516
2517 const wxString shapeProps = _HKI( "Shape Properties" );
2518
2519 auto shape = new PROPERTY_ENUM<EDA_SHAPE, SHAPE_T>( _HKI( "Shape" ),
2521 propMgr.AddProperty( shape, shapeProps );
2522
2523 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, int>( _HKI( "Start X" ),
2526 shapeProps )
2527 .SetAvailableFunc( isNotPolygonOrCircle );
2528 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, int>( _HKI( "Start Y" ),
2531 shapeProps )
2532 .SetAvailableFunc( isNotPolygonOrCircle );
2533
2534 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, int>( _HKI( "Center X" ),
2537 shapeProps )
2538 .SetAvailableFunc( isCircle );
2539
2540 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, int>( _HKI( "Center Y" ),
2543 shapeProps )
2544 .SetAvailableFunc( isCircle );
2545
2546 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, int>( _HKI( "Radius" ),
2549 shapeProps )
2550 .SetAvailableFunc( isCircle );
2551
2552 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, int>( _HKI( "End X" ),
2555 shapeProps )
2556 .SetAvailableFunc( isNotPolygonOrCircle );
2557
2558 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, int>( _HKI( "End Y" ),
2561 shapeProps )
2562 .SetAvailableFunc( isNotPolygonOrCircle );
2563
2564 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, int>( _HKI( "Width" ),
2567 shapeProps )
2568 .SetAvailableFunc( isRectangle );
2569
2570 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, int>( _HKI( "Height" ),
2573 shapeProps )
2574 .SetAvailableFunc( isRectangle );
2575
2576 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, int>( _HKI( "Line Width" ),
2578 shapeProps );
2579
2580 propMgr.AddProperty( new PROPERTY_ENUM<EDA_SHAPE, LINE_STYLE>( _HKI( "Line Style" ),
2582 shapeProps );
2583
2584 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, COLOR4D>( _HKI( "Line Color" ),
2586 shapeProps )
2588
2589 auto angle = new PROPERTY<EDA_SHAPE, EDA_ANGLE>( _HKI( "Angle" ),
2592 angle->SetAvailableFunc(
2593 [=]( INSPECTABLE* aItem ) -> bool
2594 {
2595 if( EDA_SHAPE* curr_shape = dynamic_cast<EDA_SHAPE*>( aItem ) )
2596 return curr_shape->GetShape() == SHAPE_T::ARC;
2597
2598 return false;
2599 } );
2600 propMgr.AddProperty( angle, shapeProps );
2601
2602 auto fillAvailable =
2603 [=]( INSPECTABLE* aItem ) -> bool
2604 {
2605 if( EDA_ITEM* edaItem = dynamic_cast<EDA_ITEM*>( aItem ) )
2606 {
2607 // For some reason masking "Filled" and "Fill Color" at the
2608 // PCB_TABLECELL level doesn't work.
2609 if( edaItem->Type() == PCB_TABLECELL_T )
2610 return false;
2611 }
2612
2613 if( EDA_SHAPE* edaShape = dynamic_cast<EDA_SHAPE*>( aItem ) )
2614 {
2615 switch( edaShape->GetShape() )
2616 {
2617 case SHAPE_T::POLY:
2618 case SHAPE_T::RECTANGLE:
2619 case SHAPE_T::CIRCLE:
2620 case SHAPE_T::BEZIER:
2621 return true;
2622
2623 default:
2624 return false;
2625 }
2626 }
2627
2628 return false;
2629 };
2630
2633 shapeProps )
2634 .SetAvailableFunc( fillAvailable );
2635
2636 propMgr.AddProperty( new PROPERTY<EDA_SHAPE, COLOR4D>( _HKI( "Fill Color" ),
2638 shapeProps )
2639 .SetAvailableFunc( fillAvailable )
2641 }
ERROR_LOC
When approximating an arc or circle, should the error be placed on the outside or inside of the curve...
Definition: approximation.h:32
@ ERROR_OUTSIDE
Definition: approximation.h:33
@ ERROR_INSIDE
Definition: approximation.h:34
constexpr int ARC_HIGH_DEF
Definition: base_units.h:120
constexpr int ARC_LOW_DEF
Definition: base_units.h:119
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
Bezier curves to polygon converter.
Definition: bezier_curves.h:38
void GetPoly(std::vector< VECTOR2I > &aOutput, int aMaxError=10)
Convert a Bezier curve to a polygon.
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 void SetOrigin(const Vec &pos)
Definition: box2.h:237
constexpr BOX2< Vec > & Normalize()
Ensure that the height and width are positive.
Definition: box2.h:146
constexpr Vec Centre() const
Definition: box2.h:97
bool IntersectsCircleEdge(const Vec &aCenter, const int aRadius, const int aWidth) const
Definition: box2.h:523
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
constexpr coord_type GetLeft() const
Definition: box2.h:228
constexpr bool Contains(const Vec &aPoint) const
Definition: box2.h:168
constexpr coord_type GetRight() const
Definition: box2.h:217
constexpr void SetEnd(coord_type x, coord_type y)
Definition: box2.h:297
constexpr coord_type GetTop() const
Definition: box2.h:229
constexpr bool Intersects(const BOX2< Vec > &aRect) const
Definition: box2.h:311
constexpr coord_type GetBottom() const
Definition: box2.h:222
EDA_ANGLE Normalize()
Definition: eda_angle.h:221
bool IsCardinal() const
Definition: eda_angle.cpp:40
EDA_ANGLE Normalize720()
Definition: eda_angle.h:271
double AsRadians() const
Definition: eda_angle.h:117
The base class for create windows for drawing purpose.
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:96
UI_FILL_MODE GetFillModeProp() const
Definition: eda_shape.cpp:556
virtual int GetHatchLineSpacing() const
Definition: eda_shape.h:159
EDA_ANGLE GetArcAngle() const
Definition: eda_shape.cpp:1033
SHAPE_T m_shape
Definition: eda_shape.h:484
void TransformShapeToPolygon(SHAPE_POLY_SET &aBuffer, int aClearance, int aError, ERROR_LOC aErrorLoc, bool ignoreLineWidth=false, bool includeFill=false) const
Convert the shape to a closed polygon.
Definition: eda_shape.cpp:2176
void SetStartX(int x)
Definition: eda_shape.h:191
bool m_proxyItem
Definition: eda_shape.h:511
bool m_hatchingDirty
Definition: eda_shape.h:490
bool m_endsSwapped
Definition: eda_shape.h:483
const VECTOR2I & GetBezierC2() const
Definition: eda_shape.h:258
void SetBezierC2(const VECTOR2I &aPt)
Definition: eda_shape.h:257
void move(const VECTOR2I &aMoveVector)
Definition: eda_shape.cpp:648
void SetCenter(const VECTOR2I &aCenter)
Definition: eda_shape.cpp:912
VECTOR2I getCenter() const
Definition: eda_shape.cpp:886
int GetStartY() const
Definition: eda_shape.h:174
void SetFillModeProp(UI_FILL_MODE)
Definition: eda_shape.cpp:543
void SetLength(const double &aLength)
Definition: eda_shape.cpp:433
int m_editState
Definition: eda_shape.h:510
void rotate(const VECTOR2I &aRotCentre, const EDA_ANGLE &aAngle)
Definition: eda_shape.cpp:751
const std::vector< VECTOR2I > buildBezierToSegmentsPointsList(int aMaxError) const
Definition: eda_shape.cpp:873
const SHAPE_POLY_SET & GetHatching() const
Definition: eda_shape.cpp:569
FILL_T GetFillMode() const
Definition: eda_shape.h:142
virtual ~EDA_SHAPE()
Definition: eda_shape.cpp:62
long long int m_rectangleHeight
Definition: eda_shape.h:492
void SetEndY(int aY)
Definition: eda_shape.h:226
virtual int GetEffectiveWidth() const
Definition: eda_shape.h:157
COLOR4D GetLineColor() const
Definition: eda_shape.h:165
int GetEndX() const
Definition: eda_shape.h:217
SHAPE_POLY_SET m_hatching
Definition: eda_shape.h:489
int GetRectangleWidth() const
Definition: eda_shape.cpp:419
void SetLineStyle(const LINE_STYLE aStyle)
Definition: eda_shape.cpp:2310
void calcEdit(const VECTOR2I &aPosition)
Definition: eda_shape.cpp:1919
void SetStartY(int y)
Definition: eda_shape.h:184
SHAPE_POLY_SET & GetPolyShape()
Definition: eda_shape.h:337
void SetCenterY(int y)
Definition: eda_shape.h:198
void CalcArcAngles(EDA_ANGLE &aStartAngle, EDA_ANGLE &aEndAngle) const
Calc arc start and end angles such that aStartAngle < aEndAngle.
Definition: eda_shape.cpp:945
void ShapeGetMsgPanelInfo(EDA_DRAW_FRAME *aFrame, std::vector< MSG_PANEL_ITEM > &aList)
Definition: eda_shape.cpp:1103
bool operator==(const EDA_SHAPE &aOther) const
Definition: eda_shape.cpp:2325
int GetRadius() const
Definition: eda_shape.cpp:961
SHAPE_T GetShape() const
Definition: eda_shape.h:168
void SetPolyShape(const SHAPE_POLY_SET &aShape)
Definition: eda_shape.h:345
bool Deserialize(const google::protobuf::Any &aContainer) override
Deserializes the given protobuf message into this object.
Definition: eda_shape.cpp:234
void SetRectangleHeight(const int &aHeight)
Definition: eda_shape.cpp:446
bool IsHatchedFill() const
Definition: eda_shape.h:124
virtual void updateHatching() const
Definition: eda_shape.cpp:581
VECTOR2I m_arcCenter
Definition: eda_shape.h:501
double m_segmentLength
Definition: eda_shape.h:495
void SetCenterX(int x)
Definition: eda_shape.h:205
virtual void SetFilled(bool aFlag)
Definition: eda_shape.h:136
virtual bool IsFilledForHitTesting() const
Definition: eda_shape.h:131
bool continueEdit(const VECTOR2I &aPosition)
Definition: eda_shape.cpp:1885
wxString ShowShape() const
Definition: eda_shape.cpp:313
ARC_MID m_arcMidData
Definition: eda_shape.h:502
void SetFillColor(const COLOR4D &aColor)
Definition: eda_shape.h:153
int GetEndY() const
Definition: eda_shape.h:216
std::vector< SHAPE * > makeEffectiveShapes(bool aEdgeOnly, bool aLineChainOnly=false) const
Make a set of SHAPE objects representing the EDA_SHAPE.
Definition: eda_shape.cpp:1718
bool hitTest(const VECTOR2I &aPosition, int aAccuracy=0) const
Definition: eda_shape.cpp:1216
void SetCachedArcData(const VECTOR2I &aStart, const VECTOR2I &aMid, const VECTOR2I &aEnd, const VECTOR2I &aCenter)
Set the data used for mid point caching.
Definition: eda_shape.cpp:984
void SetEndX(int aX)
Definition: eda_shape.h:233
void RebuildBezierToSegmentsPointsList(int aMaxError)
Rebuild the m_bezierPoints vertex list that approximate the Bezier curve by a list of segments.
Definition: eda_shape.cpp:859
virtual int GetHatchLineWidth() const
Definition: eda_shape.h:158
bool IsSolidFill() const
Definition: eda_shape.h:117
void flip(const VECTOR2I &aCentre, FLIP_DIRECTION aFlipDirection)
Definition: eda_shape.cpp:814
EDA_SHAPE(SHAPE_T aType, int aLineWidth, FILL_T aFill)
Definition: eda_shape.cpp:47
void beginEdit(const VECTOR2I &aStartPoint)
Definition: eda_shape.cpp:1844
VECTOR2I m_start
Definition: eda_shape.h:498
int GetPointCount() const
Definition: eda_shape.cpp:1836
const VECTOR2I & GetEnd() const
Return the ending point of the graphic.
Definition: eda_shape.h:215
bool IsClosed() const
Definition: eda_shape.cpp:505
void SetRadius(int aX)
Definition: eda_shape.h:240
void SetStart(const VECTOR2I &aStart)
Definition: eda_shape.h:177
void DupPolyPointsList(std::vector< VECTOR2I > &aBuffer) const
Duplicate the list of corners in a std::vector<VECTOR2I>.
Definition: eda_shape.cpp:1812
LINE_STYLE GetLineStyle() const
Definition: eda_shape.cpp:2316
void endEdit(bool aClosed=true)
Finish editing the shape.
Definition: eda_shape.cpp:2073
const VECTOR2I & GetStart() const
Return the starting point of the graphic.
Definition: eda_shape.h:173
void SetLineColor(const COLOR4D &aColor)
Definition: eda_shape.h:164
COLOR4D GetFillColor() const
Definition: eda_shape.h:152
void SetRectangle(const long long int &aHeight, const long long int &aWidth)
Definition: eda_shape.cpp:476
void SetShape(SHAPE_T aShape)
Definition: eda_shape.h:167
void SwapShape(EDA_SHAPE *aImage)
Definition: eda_shape.cpp:2110
std::vector< VECTOR2I > GetRectCorners() const
Definition: eda_shape.cpp:1544
void SetSegmentAngle(const EDA_ANGLE &aAngle)
Definition: eda_shape.cpp:491
std::vector< VECTOR2I > GetCornersInSequence() const
Definition: eda_shape.cpp:1559
std::vector< VECTOR2I > m_bezierPoints
Definition: eda_shape.h:507
bool IsAnyFill() const
Definition: eda_shape.h:112
void setPosition(const VECTOR2I &aPos)
Definition: eda_shape.cpp:357
virtual bool IsProxyItem() const
Definition: eda_shape.h:109
void computeArcBBox(BOX2I &aBBox) const
Definition: eda_shape.cpp:1658
void SetEnd(const VECTOR2I &aEnd)
Definition: eda_shape.h:219
void SetRectangleWidth(const int &aWidth)
Definition: eda_shape.cpp:461
void SetBezierC1(const VECTOR2I &aPt)
Definition: eda_shape.h:254
void SetArcGeometry(const VECTOR2I &aStart, const VECTOR2I &aMid, const VECTOR2I &aEnd)
Set the three controlling points for an arc.
Definition: eda_shape.cpp:994
double GetLength() const
Definition: eda_shape.cpp:374
wxString SHAPE_T_asString() const
Definition: eda_shape.cpp:340
void scale(double aScale)
Definition: eda_shape.cpp:691
int GetStartX() const
Definition: eda_shape.h:175
double Similarity(const EDA_SHAPE &aOther) const
Definition: eda_shape.cpp:2370
const VECTOR2I & GetBezierC1() const
Definition: eda_shape.h:255
VECTOR2I m_end
Definition: eda_shape.h:499
const BOX2I getBoundingBox() const
Definition: eda_shape.cpp:1162
virtual EDA_ANGLE getDrawRotation() const
Definition: eda_shape.h:444
void SetArcAngleAndEnd(const EDA_ANGLE &aAngle, bool aCheckNegativeAngle=false)
Set the end point from the angle center and start.
Definition: eda_shape.cpp:1061
SHAPE_POLY_SET m_poly
Definition: eda_shape.h:508
int GetRectangleHeight() const
Definition: eda_shape.cpp:405
virtual int GetWidth() const
Definition: eda_shape.h:156
VECTOR2I getPosition() const
Definition: eda_shape.cpp:363
bool IsClockwiseArc() const
Definition: eda_shape.cpp:1044
STROKE_PARAMS m_stroke
Definition: eda_shape.h:485
EDA_ANGLE m_segmentAngle
Definition: eda_shape.h:496
void SetPolyPoints(const std::vector< VECTOR2I > &aPoints)
Definition: eda_shape.cpp:1708
wxString getFriendlyName() const
Definition: eda_shape.cpp:1076
VECTOR2I m_bezierC1
Definition: eda_shape.h:504
FILL_T m_fill
Definition: eda_shape.h:486
COLOR4D m_fillColor
Definition: eda_shape.h:487
void SetWidth(int aWidth)
Definition: eda_shape.cpp:2303
EDA_ANGLE GetSegmentAngle() const
Definition: eda_shape.cpp:1024
void SetFillMode(FILL_T aFill)
Definition: eda_shape.cpp:536
long long int m_rectangleWidth
Definition: eda_shape.h:493
VECTOR2I m_bezierC2
Definition: eda_shape.h:505
void Serialize(google::protobuf::Any &aContainer) const override
Serializes this object to the given Any message.
Definition: eda_shape.cpp:146
bool IsPolyShapeValid() const
Definition: eda_shape.cpp:1829
int Compare(const EDA_SHAPE *aOther) const
Definition: eda_shape.cpp:2135
VECTOR2I GetArcMid() const
Definition: eda_shape.cpp:931
ENUM_MAP & Map(T aValue, const wxString &aName)
Definition: property.h:686
static ENUM_MAP< T > & Instance()
Definition: property.h:680
wxPGChoices & Choices()
Definition: property.h:729
Class that other classes need to inherit from, in order to be inspectable.
Definition: inspectable.h:37
A color representation with 4 components: red, green, blue, alpha.
Definition: color4d.h:104
PROPERTY_BASE & SetAvailableFunc(std::function< bool(INSPECTABLE *)> aFunc)
Set a callback function to determine whether an object provides this property.
Definition: property.h:257
PROPERTY_BASE & SetIsHiddenFromRulesEditor(bool aHide=true)
Definition: property.h:307
Provide class metadata.Helper macro to map type hashes to names.
Definition: property_mgr.h:85
static PROPERTY_MANAGER & Instance()
Definition: property_mgr.h:87
PROPERTY_BASE & AddProperty(PROPERTY_BASE *aProperty, const wxString &aGroup=wxEmptyString)
Register a property.
Definition: seg.h:42
VECTOR2I A
Definition: seg.h:49
VECTOR2I B
Definition: seg.h:50
int Length() const
Return the length (this).
Definition: seg.h:333
SHAPE_TYPE Type() const
Return the type of the shape.
Definition: shape.h:98
int GetRadius() const
Definition: shape_circle.h:118
const VECTOR2I GetCenter() const
Definition: shape_circle.h:123
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
bool IsClosed() const override
virtual const VECTOR2I GetPoint(int aIndex) const override
void SetPoint(int aIndex, const VECTOR2I &aPos)
Move a point to a specific location.
void SetClosed(bool aClosed)
Mark the line chain as closed (i.e.
int PointCount() const
Return the number of points (vertices) in this line chain.
virtual size_t GetPointCount() const override
void Append(int aX, int aY, bool aAllowDuplication=false)
Append a new point at the end of the line chain.
virtual const SEG GetSegment(int aIndex) const override
const VECTOR2I & CPoint(int aIndex) const
Return a reference to a given point in the line chain.
int SegmentCount() const
Return the number of segments in this line chain.
const VECTOR2I & CLastPoint() const
Return the last point in the line chain.
void Remove(int aStartIndex, int aEndIndex)
Remove the range of points [start_index, end_index] from the line chain.
const SEG CSegment(int aIndex) const
Return a constant copy of the aIndex segment in the line chain.
const std::vector< VECTOR2I > & CPoints() const
Represent a set of closed polygons.
void Rotate(const EDA_ANGLE &aAngle, const VECTOR2I &aCenter={ 0, 0 }) override
Rotate all vertices by a given angle.
void RemoveAllContours()
Remove all outlines & holes (clears) the polygon set.
bool CollideEdge(const VECTOR2I &aPoint, VERTEX_INDEX *aClosestVertex=nullptr, int aClearance=0) const
Check whether aPoint collides with any edge of any of the contours of the polygon.
int AddOutline(const SHAPE_LINE_CHAIN &aOutline)
Adds a new outline to the set and returns its index.
int VertexCount(int aOutline=-1, int aHole=-1) const
Return the number of vertices in a given outline/hole.
bool IsEmpty() const
Return true if the set is empty (no polygons at all)
CONST_ITERATOR CIterate(int aFirst, int aLast, bool aIterateHoles=false) const
void Fracture()
Convert a set of polygons with holes to a single outline with "slits"/"fractures" connecting the oute...
bool Collide(const SHAPE *aShape, int aClearance=0, int *aActual=nullptr, VECTOR2I *aLocation=nullptr) const override
Check if the boundary of shape (this) lies closer to the shape aShape than aClearance,...
int TotalVertices() const
Return total number of vertices stored in the set.
void Inflate(int aAmount, CORNER_STRATEGY aCornerStrategy, int aMaxError, bool aSimplify=false)
Perform outline inflation/deflation.
int Append(int x, int y, int aOutline=-1, int aHole=-1, bool aAllowDuplication=false)
Appends a vertex at the end of the given outline/hole (default: the last outline)
const std::vector< SEG > GenerateHatchLines(const std::vector< double > &aSlopes, int aSpacing, int aLineLength) const
SHAPE_LINE_CHAIN & Outline(int aIndex)
Return the reference to aIndex-th outline in the set.
int NewOutline()
Creates a new empty polygon in the set and returns its index.
void Mirror(const VECTOR2I &aRef, FLIP_DIRECTION aFlipDirection)
Mirror the line points about y or x (or both)
const VECTOR2I & CVertex(int aIndex, int aOutline, int aHole) const
Return the index-th vertex in a given hole outline within a given outline.
int OutlineCount() const
Return the number of outlines in the set.
void Move(const VECTOR2I &aVector) override
const SHAPE_LINE_CHAIN & COutline(int aIndex) const
Represent a simple polygon consisting of a zero-thickness closed chain of connected line segments.
Definition: shape_simple.h:42
An abstract shape on 2D plane.
Definition: shape.h:126
int GetWidth() const
void SetLineStyle(LINE_STYLE aLineStyle)
void SetWidth(int aWidth)
void GetMsgPanelInfo(UNITS_PROVIDER *aUnitsProvider, std::vector< MSG_PANEL_ITEM > &aList, bool aIncludeStyle=true, bool aIncludeWidth=true)
LINE_STYLE GetLineStyle() const
wxString MessageTextFromValue(double aValue, bool aAddUnitLabel=true, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
A lower-precision version of StringFromValue().
double Distance(const VECTOR2< extended_type > &aVector) const
Compute the distance between two vectors.
Definition: vector2d.h:561
constexpr extended_type SquaredEuclideanNorm() const
Compute the squared euclidean norm of the vector, which is defined as (x ** 2 + y ** 2).
Definition: vector2d.h:307
T EuclideanNorm() const
Compute the Euclidean norm of the vector, which is defined as sqrt(x ** 2 + y ** 2).
Definition: vector2d.h:283
void TransformRingToPolygon(SHAPE_POLY_SET &aBuffer, const VECTOR2I &aCentre, int aRadius, int aWidth, int aError, ERROR_LOC aErrorLoc)
Convert arcs to multiple straight segments.
void TransformCircleToPolygon(SHAPE_LINE_CHAIN &aBuffer, const VECTOR2I &aCenter, int aRadius, int aError, ERROR_LOC aErrorLoc, int aMinSegCount=0)
Convert a circle to a polygon, using multiple straight lines.
void TransformArcToPolygon(SHAPE_POLY_SET &aBuffer, const VECTOR2I &aStart, const VECTOR2I &aMid, const VECTOR2I &aEnd, int aWidth, int aError, ERROR_LOC aErrorLoc)
Convert arc to multiple straight segments.
void TransformOvalToPolygon(SHAPE_POLY_SET &aBuffer, const VECTOR2I &aStart, const VECTOR2I &aEnd, int aWidth, int aError, ERROR_LOC aErrorLoc, int aMinSegCount=0)
Convert a oblong shape to a polygon, using multiple segments.
#define _HKI(x)
#define _(s)
static constexpr EDA_ANGLE ANGLE_0
Definition: eda_angle.h:401
static constexpr EDA_ANGLE ANGLE_90
Definition: eda_angle.h:403
@ RADIANS_T
Definition: eda_angle.h:32
static constexpr EDA_ANGLE ANGLE_270
Definition: eda_angle.h:406
static constexpr EDA_ANGLE ANGLE_360
Definition: eda_angle.h:407
static constexpr EDA_ANGLE ANGLE_180
Definition: eda_angle.h:405
#define TEST_PT(a, b)
#define TEST(a, b)
#define TEST_E(a, b)
static struct EDA_SHAPE_DESC _EDA_SHAPE_DESC
#define SWAPITEM(x)
#define sq(x)
SHAPE_T
Definition: eda_shape.h:43
@ RECTANGLE
Use RECTANGLE instead of RECT to avoid collision in a Windows header.
UI_FILL_MODE
Definition: eda_shape.h:68
@ REVERSE_HATCH
Definition: eda_shape.h:72
@ SOLID
Definition: eda_shape.h:70
@ HATCH
Definition: eda_shape.h:71
@ NONE
Definition: eda_shape.h:69
@ CROSS_HATCH
Definition: eda_shape.h:73
FILL_T
Definition: eda_shape.h:56
This file contains miscellaneous commonly used macros and functions.
#define KI_FALLTHROUGH
The KI_FALLTHROUGH macro is to be used when switch statement cases should purposely fallthrough from ...
Definition: macros.h:83
#define UNIMPLEMENTED_FOR(type)
Definition: macros.h:96
constexpr void MIRROR(T &aPoint, const T &aMirrorRef)
Updates aPoint with the mirror of aPoint relative to the aMirrorRef.
Definition: mirror.h:45
FLIP_DIRECTION
Definition: mirror.h:27
KICOMMON_API wxString MessageTextFromValue(const EDA_IU_SCALE &aIuScale, EDA_UNITS aUnits, double aValue, bool aAddUnitsText=true, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
A helper to convert the double length aValue to a string in inches, millimeters, or unscaled units.
Definition: eda_units.cpp:361
size_t longest_common_subset(const _Container &__c1, const _Container &__c2)
Returns the length of the longest common subset of values between two containers.
Definition: kicad_algo.h:208
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
Definition: eda_angle.h:390
#define IMPLEMENT_ENUM_TO_WXANY(type)
Definition: property.h:780
#define NO_SETTER(owner, type)
Definition: property.h:791
@ PT_COORD
Coordinate expressed in distance units (mm/inch)
Definition: property.h:64
@ PT_DECIDEGREE
Angle expressed in decidegrees.
Definition: property.h:66
@ PT_SIZE
Size expressed in distance units (mm/inch)
Definition: property.h:62
#define REGISTER_TYPE(x)
Definition: property_mgr.h:371
@ SH_POLY_SET
set of polygons (with holes, etc.)
Definition: shape.h:52
@ SH_RECT
axis-aligned rectangle
Definition: shape.h:47
@ SH_CIRCLE
circle
Definition: shape.h:50
@ SH_SIMPLE
simple polygon
Definition: shape.h:51
@ SH_NULL
empty shape (no shape...),
Definition: shape.h:55
@ SH_SEGMENT
line segment
Definition: shape.h:48
@ SH_ARC
circular arc
Definition: shape.h:54
@ SH_POLY_SET_TRIANGLE
a single triangle belonging to a POLY_SET triangulation
Definition: shape.h:56
@ SH_LINE_CHAIN
line chain (polyline)
Definition: shape.h:49
@ SH_COMPOUND
compound shape, consisting of multiple simple shapes
Definition: shape.h:53
static bool Collide(const SHAPE_CIRCLE &aA, const SHAPE_CIRCLE &aB, int aClearance, int *aActual, VECTOR2I *aLocation, VECTOR2I *aMTV)
LINE_STYLE
Dashed line types.
Definition: stroke_params.h:46
VECTOR2I end
Definition: eda_shape.h:82
VECTOR2I center
Definition: eda_shape.h:83
VECTOR2I start
Definition: eda_shape.h:81
VECTOR2I mid
Definition: eda_shape.h:80
int radius
SHAPE_CIRCLE circle(c.m_circle_center, c.m_circle_radius)
bool TestSegmentHit(const VECTOR2I &aRefPoint, const VECTOR2I &aStart, const VECTOR2I &aEnd, int aDist)
Test if aRefPoint is with aDistance on the line defined by aStart and aEnd.
Definition: trigo.cpp:175
void RotatePoint(int *pX, int *pY, const EDA_ANGLE &aAngle)
Calculate the new point of coord coord pX, pY, for a rotation center 0, 0.
Definition: trigo.cpp:229
const VECTOR2I CalcArcCenter(const VECTOR2I &aStart, const VECTOR2I &aMid, const VECTOR2I &aEnd)
Determine the center of an arc or circle given three points on its circumference.
Definition: trigo.cpp:521
@ PCB_TABLECELL_T
class PCB_TABLECELL, PCB_TEXTBOX for use in tables
Definition: typeinfo.h:95
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695
VECTOR2< double > VECTOR2D
Definition: vector2d.h:694