KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
pcb_dimension.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) 2012 Jean-Pierre Charras, jean-pierre.charras@ujf-grenoble.fr
5 * Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
6 * Copyright (C) 2012 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 <bitmaps.h>
29#include <pcb_edit_frame.h>
30#include <base_units.h>
32#include <font/font.h>
33#include <board.h>
34#include <pcb_dimension.h>
35#include <pcb_text.h>
41#include <trigo.h>
42#include <api/api_enums.h>
43#include <api/api_utils.h>
44#include <api/board/board_types.pb.h>
45
46
48
49static const EDA_ANGLE s_arrowAngle( 27.5, DEGREES_T );
50
51
60static OPT_VECTOR2I segPolyIntersection( const SHAPE_POLY_SET& aPoly, const SEG& aSeg,
61 bool aStart = true )
62{
63 VECTOR2I start( aStart ? aSeg.A : aSeg.B );
64 VECTOR2I endpoint( aStart ? aSeg.B : aSeg.A );
65
66 if( aPoly.Contains( start ) )
67 return std::nullopt;
68
69 for( SHAPE_POLY_SET::CONST_SEGMENT_ITERATOR seg = aPoly.CIterateSegments(); seg; ++seg )
70 {
71 if( OPT_VECTOR2I intersection = ( *seg ).Intersect( aSeg ) )
72 {
73 if( ( *intersection - start ).SquaredEuclideanNorm()
74 < ( endpoint - start ).SquaredEuclideanNorm() )
75 endpoint = *intersection;
76 }
77 }
78
79 if( start == endpoint )
80 return std::nullopt;
81
82 return OPT_VECTOR2I( endpoint );
83}
84
85
86static OPT_VECTOR2I segCircleIntersection( CIRCLE& aCircle, SEG& aSeg, bool aStart = true )
87{
88 VECTOR2I start( aStart ? aSeg.A : aSeg.B );
89 VECTOR2I endpoint( aStart ? aSeg.B : aSeg.A );
90
91 if( aCircle.Contains( start ) )
92 return std::nullopt;
93
94 std::vector<VECTOR2I> intersections = aCircle.Intersect( aSeg );
95
96 for( VECTOR2I& intersection : aCircle.Intersect( aSeg ) )
97 {
98 if( ( intersection - start ).SquaredEuclideanNorm()
99 < ( endpoint - start ).SquaredEuclideanNorm() )
100 endpoint = intersection;
101 }
102
103 if( start == endpoint )
104 return std::nullopt;
105
106 return OPT_VECTOR2I( endpoint );
107}
108
109
114static void CollectKnockedOutSegments( const SHAPE_POLY_SET& aPoly, const SEG& aSeg,
115 std::vector<std::shared_ptr<SHAPE>>& aSegmentsAfterKnockout )
116{
117 // Now we can draw 0, 1, or 2 crossbar lines depending on how the polygon collides
118 const bool containsA = aPoly.Contains( aSeg.A );
119 const bool containsB = aPoly.Contains( aSeg.B );
120
121 const OPT_VECTOR2I endpointA = segPolyIntersection( aPoly, aSeg );
122 const OPT_VECTOR2I endpointB = segPolyIntersection( aPoly, aSeg, false );
123
124 if( endpointA )
125 aSegmentsAfterKnockout.emplace_back( new SHAPE_SEGMENT( aSeg.A, *endpointA ) );
126
127 if( endpointB )
128 {
129 bool can_add = true;
130
131 if( endpointA )
132 {
133 if( ( *endpointB == aSeg.A && *endpointA == aSeg.B )
134 || ( *endpointA == *endpointB && aSeg.A == aSeg.B ) )
135 can_add = false;
136 }
137
138 if( can_add )
139 aSegmentsAfterKnockout.emplace_back( new SHAPE_SEGMENT( *endpointB, aSeg.B ) );
140 }
141
142 if( !containsA && !containsB && !endpointA && !endpointB )
143 aSegmentsAfterKnockout.emplace_back( new SHAPE_SEGMENT( aSeg ) );
144}
145
146
148 PCB_TEXT( aParent, aType ),
149 m_overrideTextEnabled( false ),
150 m_units( EDA_UNITS::INCH ),
151 m_autoUnits( false ),
152 m_unitsFormat( DIM_UNITS_FORMAT::BARE_SUFFIX ),
153 m_arrowDirection( DIM_ARROW_DIRECTION::OUTWARD ),
154 m_precision( DIM_PRECISION::X_XXXX ),
155 m_suppressZeroes( false ),
156 m_lineThickness( pcbIUScale.mmToIU( 0.2 ) ),
157 m_arrowLength( pcbIUScale.MilsToIU( 50 ) ),
158 m_extensionOffset( 0 ),
159 m_textPosition( DIM_TEXT_POSITION::OUTSIDE ),
160 m_keepTextAligned( true ),
161 m_measuredValue( 0 ),
162 m_inClearRenderCache( false )
163{
165 m_busy = false;
166}
167
168
170{
171 if( Type() != aOther.Type() )
172 return false;
173
174 const PCB_DIMENSION_BASE& other = static_cast<const PCB_DIMENSION_BASE&>( aOther );
175
176 return *this == other;
177}
178
179
181{
182 if( m_textPosition != aOther.m_textPosition )
183 return false;
184
186 return false;
187
188 if( m_units != aOther.m_units )
189 return false;
190
191 if( m_autoUnits != aOther.m_autoUnits )
192 return false;
193
194 if( m_unitsFormat != aOther.m_unitsFormat )
195 return false;
196
197 if( m_precision != aOther.m_precision )
198 return false;
199
200 if( m_suppressZeroes != aOther.m_suppressZeroes )
201 return false;
202
203 if( m_lineThickness != aOther.m_lineThickness )
204 return false;
205
206 if( m_arrowLength != aOther.m_arrowLength )
207 return false;
208
210 return false;
211
212 if( m_measuredValue != aOther.m_measuredValue )
213 return false;
214
215 return EDA_TEXT::operator==( aOther );
216}
217
218
219double PCB_DIMENSION_BASE::Similarity( const BOARD_ITEM& aOther ) const
220{
221 if( m_Uuid == aOther.m_Uuid )
222 return 1.0;
223
224 if( Type() != aOther.Type() )
225 return 0.0;
226
227 const PCB_DIMENSION_BASE& other = static_cast<const PCB_DIMENSION_BASE&>( aOther );
228
229 double similarity = 1.0;
230
231 if( m_textPosition != other.m_textPosition )
232 similarity *= 0.9;
233
235 similarity *= 0.9;
236
237 if( m_units != other.m_units )
238 similarity *= 0.9;
239
240 if( m_autoUnits != other.m_autoUnits )
241 similarity *= 0.9;
242
243 if( m_unitsFormat != other.m_unitsFormat )
244 similarity *= 0.9;
245
246 if( m_precision != other.m_precision )
247 similarity *= 0.9;
248
250 similarity *= 0.9;
251
252 if( m_lineThickness != other.m_lineThickness )
253 similarity *= 0.9;
254
255 if( m_arrowLength != other.m_arrowLength )
256 similarity *= 0.9;
257
259 similarity *= 0.9;
260
261 if( m_measuredValue != other.m_measuredValue )
262 similarity *= 0.9;
263
264 similarity *= EDA_TEXT::Similarity( other );
265
266 return similarity;
267}
268
269
270void PCB_DIMENSION_BASE::Serialize( google::protobuf::Any &aContainer ) const
271{
272 using namespace kiapi::common;
273 using namespace kiapi::board::types;
274 Dimension dimension;
275
276 dimension.mutable_id()->set_value( m_Uuid.AsStdString() );
277 dimension.set_layer( ToProtoEnum<PCB_LAYER_ID, BoardLayer>( GetLayer() ) );
278 dimension.set_locked( IsLocked() ? types::LockedState::LS_LOCKED
279 : types::LockedState::LS_UNLOCKED );
280
281 google::protobuf::Any any;
282 EDA_TEXT::Serialize( any );
283 any.UnpackTo( dimension.mutable_text() );
284
285 types::Text* text = dimension.mutable_text();
286 text->set_text( GetValueText() );
287
288 dimension.set_override_text_enabled( m_overrideTextEnabled );
289 dimension.set_override_text( m_valueString.ToUTF8() );
290 dimension.set_prefix( m_prefix.ToUTF8() );
291 dimension.set_suffix( m_suffix.ToUTF8() );
292
293 dimension.set_unit( ToProtoEnum<DIM_UNITS_MODE, DimensionUnit>( GetUnitsMode() ) );
294 dimension.set_unit_format(
295 ToProtoEnum<DIM_UNITS_FORMAT, DimensionUnitFormat>( m_unitsFormat ) );
296 dimension.set_arrow_direction(
297 ToProtoEnum<DIM_ARROW_DIRECTION, DimensionArrowDirection>( m_arrowDirection ) );
298 dimension.set_precision( ToProtoEnum<DIM_PRECISION, DimensionPrecision>( m_precision ) );
299 dimension.set_suppress_trailing_zeroes( m_suppressZeroes );
300
301 dimension.mutable_line_thickness()->set_value_nm( m_lineThickness );
302 dimension.mutable_arrow_length()->set_value_nm( m_arrowLength );
303 dimension.mutable_extension_offset()->set_value_nm( m_extensionOffset );
304 dimension.set_text_position(
305 ToProtoEnum<DIM_TEXT_POSITION, DimensionTextPosition>( m_textPosition ) );
306 dimension.set_keep_text_aligned( m_keepTextAligned );
307
308 aContainer.PackFrom( dimension );
309}
310
311
312bool PCB_DIMENSION_BASE::Deserialize( const google::protobuf::Any &aContainer )
313{
314 using namespace kiapi::common;
315 kiapi::board::types::Dimension dimension;
316
317 if( !aContainer.UnpackTo( &dimension ) )
318 return false;
319
320 SetLayer( FromProtoEnum<PCB_LAYER_ID, kiapi::board::types::BoardLayer>( dimension.layer() ) );
321 const_cast<KIID&>( m_Uuid ) = KIID( dimension.id().value() );
322 SetLocked( dimension.locked() == types::LockedState::LS_LOCKED );
323
324 google::protobuf::Any any;
325 any.PackFrom( dimension.text() );
327
328 SetOverrideTextEnabled( dimension.override_text_enabled() );
329 SetOverrideText( wxString::FromUTF8( dimension.override_text() ) );
330 SetPrefix( wxString::FromUTF8( dimension.prefix() ) );
331 SetSuffix( wxString::FromUTF8( dimension.suffix() ) );
332
333 SetUnitsMode( FromProtoEnum<DIM_UNITS_MODE>( dimension.unit() ) );
334 SetUnitsFormat( FromProtoEnum<DIM_UNITS_FORMAT>( dimension.unit_format() ) );
335 SetArrowDirection( FromProtoEnum<DIM_ARROW_DIRECTION>( dimension.arrow_direction() ) );
336 SetPrecision( FromProtoEnum<DIM_PRECISION>( dimension.precision() ) );
337 SetSuppressZeroes( dimension.suppress_trailing_zeroes() );
338
339 SetLineThickness( dimension.line_thickness().value_nm() );
340 SetArrowLength( dimension.arrow_length().value_nm() );
341 SetExtensionOffset( dimension.extension_offset().value_nm() );
342 SetTextPositionMode( FromProtoEnum<DIM_TEXT_POSITION>( dimension.text_position() ) );
343 SetKeepTextAligned( dimension.keep_text_aligned() );
344
345 Update();
346
347 return true;
348}
349
350
351void PCB_DIMENSION_BASE::drawAnArrow( VECTOR2I startPoint, EDA_ANGLE anAngle, int aLength )
352{
353 if( aLength )
354 {
355 VECTOR2I tailEnd( aLength, 0 );
356 RotatePoint( tailEnd, -anAngle );
357 m_shapes.emplace_back( new SHAPE_SEGMENT( startPoint, startPoint + tailEnd ) );
358 }
359
360 VECTOR2I arrowEndPos( m_arrowLength, 0 );
361 VECTOR2I arrowEndNeg( m_arrowLength, 0 );
362
363 RotatePoint( arrowEndPos, -anAngle + s_arrowAngle );
364 RotatePoint( arrowEndNeg, -anAngle - s_arrowAngle );
365
366 m_shapes.emplace_back( new SHAPE_SEGMENT( startPoint, startPoint + arrowEndPos ) );
367 m_shapes.emplace_back( new SHAPE_SEGMENT( startPoint, startPoint + arrowEndNeg ) );
368}
369
370
372{
374
375 switch( m_unitsFormat )
376 {
377 case DIM_UNITS_FORMAT::NO_SUFFIX: // no units
378 break;
379
380 case DIM_UNITS_FORMAT::BARE_SUFFIX: // normal
382 break;
383
384 case DIM_UNITS_FORMAT::PAREN_SUFFIX: // parenthetical
385 text += wxT( " (" ) + EDA_UNIT_UTILS::GetText( m_units ).Trim( false ) + wxT( ")" );
386 break;
387 }
388
389 text.Prepend( m_prefix );
390 text.Append( m_suffix );
391
392 SetText( text );
393}
394
395
397{
399
400 // We use EDA_TEXT::ClearRenderCache() as a signal that the properties of the EDA_TEXT
401 // have changed and we may need to update the dimension text
402
404 {
406 Update();
407 m_inClearRenderCache = false;
408 }
409}
410
411
412template<typename ShapeType>
413void PCB_DIMENSION_BASE::addShape( const ShapeType& aShape )
414{
415 m_shapes.push_back( std::make_shared<ShapeType>( aShape ) );
416}
417
418
420{
421 struct lconv* lc = localeconv();
422 wxChar sep = lc->decimal_point[0];
423
424 int val = GetMeasuredValue();
425 int precision = static_cast<int>( m_precision );
426 wxString text;
427
428 if( precision >= 6 )
429 {
430 switch( m_units )
431 {
432 case EDA_UNITS::INCH: precision = precision - 4; break;
433 case EDA_UNITS::MILS: precision = std::max( 0, precision - 7 ); break;
434 case EDA_UNITS::MM: precision = precision - 5; break;
435 default: precision = precision - 4; break;
436 }
437 }
438
439 wxString format = wxT( "%." ) + wxString::Format( wxT( "%i" ), precision ) + wxT( "f" );
440
441 text.Printf( format, EDA_UNIT_UTILS::UI::ToUserUnit( pcbIUScale, m_units, val ) );
442
443 if( m_suppressZeroes )
444 {
445 while( text.Last() == '0' )
446 {
447 text.RemoveLast();
448
449 if( text.Last() == '.' || text.Last() == sep )
450 {
451 text.RemoveLast();
452 break;
453 }
454 }
455 }
456
457 return text;
458}
459
460
461void PCB_DIMENSION_BASE::SetPrefix( const wxString& aPrefix )
462{
463 m_prefix = aPrefix;
464}
465
466
467void PCB_DIMENSION_BASE::SetSuffix( const wxString& aSuffix )
468{
469 m_suffix = aSuffix;
470}
471
472
474{
475 m_units = aUnits;
476}
477
478
480{
481 if( m_autoUnits )
482 {
483 return DIM_UNITS_MODE::AUTOMATIC;
484 }
485 else
486 {
487 switch( m_units )
488 {
489 default:
490 case EDA_UNITS::INCH: return DIM_UNITS_MODE::INCH;
491 case EDA_UNITS::MM: return DIM_UNITS_MODE::MM;
492 case EDA_UNITS::MILS: return DIM_UNITS_MODE::MILS;
493 }
494 }
495}
496
497
499{
500 switch( aMode )
501 {
502 case DIM_UNITS_MODE::INCH:
503 m_autoUnits = false;
504 m_units = EDA_UNITS::INCH;
505 break;
506
507 case DIM_UNITS_MODE::MILS:
508 m_autoUnits = false;
509 m_units = EDA_UNITS::MILS;
510 break;
511
512 case DIM_UNITS_MODE::MM:
513 m_autoUnits = false;
514 m_units = EDA_UNITS::MM;
515 break;
516
517 case DIM_UNITS_MODE::AUTOMATIC:
518 m_autoUnits = true;
519 m_units = GetBoard() ? GetBoard()->GetUserUnits() : EDA_UNITS::MM;
520 break;
521 }
522}
523
524
526{
527 SetTextAngleDegrees( aDegrees );
528 // Create or repair any knockouts
529 Update();
530}
531
532
534{
535 SetKeepTextAligned( aKeepAligned );
536 // Re-align the text and repair any knockouts
537 Update();
538}
539
540
542{
543 PCB_TEXT::Offset( offset );
544
545 m_start += offset;
546 m_end += offset;
547
548 Update();
549}
550
551
552void PCB_DIMENSION_BASE::Rotate( const VECTOR2I& aRotCentre, const EDA_ANGLE& aAngle )
553{
554 EDA_ANGLE newAngle = GetTextAngle() + aAngle;
555
556 newAngle.Normalize();
557
558 SetTextAngle( newAngle );
559
560 VECTOR2I pt = GetTextPos();
561 RotatePoint( pt, aRotCentre, aAngle );
562 SetTextPos( pt );
563
564 RotatePoint( m_start, aRotCentre, aAngle );
565 RotatePoint( m_end, aRotCentre, aAngle );
566
567 Update();
568}
569
570
571void PCB_DIMENSION_BASE::Flip( const VECTOR2I& aCentre, FLIP_DIRECTION aFlipDirection )
572{
573 Mirror( aCentre, aFlipDirection );
574
576}
577
578
579void PCB_DIMENSION_BASE::Mirror( const VECTOR2I& axis_pos, FLIP_DIRECTION aFlipDirection )
580{
581 VECTOR2I newPos = GetTextPos();
582
583 MIRROR( newPos, axis_pos, aFlipDirection );
584
585 SetTextPos( newPos );
586
587 // invert angle
589
590 MIRROR( m_start, axis_pos, aFlipDirection );
591 MIRROR( m_end, axis_pos, aFlipDirection );
592
593 if( IsSideSpecific() )
595
596 Update();
597}
598
599
601 std::vector<MSG_PANEL_ITEM>& aList )
602{
603 // for now, display only the text within the DIMENSION using class PCB_TEXT.
604 wxString msg;
605
606 wxCHECK_RET( m_parent != nullptr, wxT( "PCB_TEXT::GetMsgPanelInfo() m_Parent is NULL." ) );
607
608 // Don't use GetShownText(); we want to see the variable references here
609 aList.emplace_back( _( "Dimension" ), KIUI::EllipsizeStatusText( aFrame, GetText() ) );
610
611 aList.emplace_back( _( "Prefix" ), GetPrefix() );
612
614 {
615 aList.emplace_back( _( "Override Text" ), GetOverrideText() );
616 }
617 else
618 {
619 aList.emplace_back( _( "Value" ), GetValueText() );
620
621 switch( GetPrecision() )
622 {
623 case DIM_PRECISION::V_VV: msg = wxT( "0.00 in / 0 mils / 0.0 mm" ); break;
624 case DIM_PRECISION::V_VVV: msg = wxT( "0.000 in / 0 mils / 0.00 mm" ); break;
625 case DIM_PRECISION::V_VVVV: msg = wxT( "0.0000 in / 0.0 mils / 0.000 mm" ); break;
626 case DIM_PRECISION::V_VVVVV: msg = wxT( "0.00000 in / 0.00 mils / 0.0000 mm" ); break;
627 default: msg = wxT( "%" ) + wxString::Format( wxT( "1.%df" ), GetPrecision() );
628 }
629
630 aList.emplace_back( _( "Precision" ), wxString::Format( msg, 0.0 ) );
631 }
632
633 aList.emplace_back( _( "Suffix" ), GetSuffix() );
634
635 // Use our own UNITS_PROVIDER to report dimension info in dimension's units rather than
636 // in frame's units.
637 UNITS_PROVIDER unitsProvider( pcbIUScale, EDA_UNITS::MM );
638 unitsProvider.SetUserUnits( GetUnits() );
639
640 aList.emplace_back( _( "Units" ), EDA_UNIT_UTILS::GetLabel( GetUnits() ) );
641
642 aList.emplace_back( _( "Font" ), GetFont() ? GetFont()->GetName() : _( "Default" ) );
643 aList.emplace_back( _( "Text Thickness" ), unitsProvider.MessageTextFromValue( GetTextThickness() ) );
644 aList.emplace_back( _( "Text Width" ), unitsProvider.MessageTextFromValue( GetTextWidth() ) );
645 aList.emplace_back( _( "Text Height" ), unitsProvider.MessageTextFromValue( GetTextHeight() ) );
646
647 ORIGIN_TRANSFORMS& originTransforms = aFrame->GetOriginTransforms();
648
649 if( Type() == PCB_DIM_CENTER_T )
650 {
651 VECTOR2I startCoord = originTransforms.ToDisplayAbs( GetStart() );
652 wxString start = wxString::Format( wxT( "@(%s, %s)" ),
653 aFrame->MessageTextFromValue( startCoord.x ),
654 aFrame->MessageTextFromValue( startCoord.y ) );
655
656 aList.emplace_back( start, wxEmptyString );
657 }
658 else
659 {
660 VECTOR2I startCoord = originTransforms.ToDisplayAbs( GetStart() );
661 wxString start = wxString::Format( wxT( "@(%s, %s)" ),
662 aFrame->MessageTextFromValue( startCoord.x ),
663 aFrame->MessageTextFromValue( startCoord.y ) );
664 VECTOR2I endCoord = originTransforms.ToDisplayAbs( GetEnd() );
665 wxString end = wxString::Format( wxT( "@(%s, %s)" ),
666 aFrame->MessageTextFromValue( endCoord.x ),
667 aFrame->MessageTextFromValue( endCoord.y ) );
668
669 aList.emplace_back( start, end );
670 }
671
672 if( aFrame->GetName() == PCB_EDIT_FRAME_NAME && IsLocked() )
673 aList.emplace_back( _( "Status" ), _( "Locked" ) );
674
675 aList.emplace_back( _( "Layer" ), GetLayerName() );
676}
677
678
679std::shared_ptr<SHAPE> PCB_DIMENSION_BASE::GetEffectiveShape( PCB_LAYER_ID aLayer, FLASHING aFlash ) const
680{
681 std::shared_ptr<SHAPE_COMPOUND> effectiveShape = std::make_shared<SHAPE_COMPOUND>();
682
683 effectiveShape->AddShape( GetEffectiveTextShape()->Clone() );
684
685 for( const std::shared_ptr<SHAPE>& shape : GetShapes() )
686 effectiveShape->AddShape( shape->Clone() );
687
688 return effectiveShape;
689}
690
691
692bool PCB_DIMENSION_BASE::HitTest( const VECTOR2I& aPosition, int aAccuracy ) const
693{
694 if( TextHitTest( aPosition ) )
695 return true;
696
697 int dist_max = aAccuracy + ( m_lineThickness / 2 );
698
699 // Locate SEGMENTS
700
701 for( const std::shared_ptr<SHAPE>& shape : GetShapes() )
702 {
703 if( shape->Collide( aPosition, dist_max ) )
704 return true;
705 }
706
707 return false;
708}
709
710
711bool PCB_DIMENSION_BASE::HitTest( const BOX2I& aRect, bool aContained, int aAccuracy ) const
712{
713 BOX2I arect = aRect;
714 arect.Inflate( aAccuracy );
715
716 BOX2I rect = GetBoundingBox();
717
718 if( aAccuracy )
719 rect.Inflate( aAccuracy );
720
721 if( aContained )
722 return arect.Contains( rect );
723
724 return arect.Intersects( rect );
725}
726
727
729{
730 BOX2I bBox;
731 int xmin, xmax, ymin, ymax;
732
733 bBox = GetTextBox();
734 xmin = bBox.GetX();
735 xmax = bBox.GetRight();
736 ymin = bBox.GetY();
737 ymax = bBox.GetBottom();
738
739 for( const std::shared_ptr<SHAPE>& shape : GetShapes() )
740 {
741 BOX2I shapeBox = shape->BBox();
742 shapeBox.Inflate( m_lineThickness / 2 );
743
744 xmin = std::min( xmin, shapeBox.GetOrigin().x );
745 xmax = std::max( xmax, shapeBox.GetEnd().x );
746 ymin = std::min( ymin, shapeBox.GetOrigin().y );
747 ymax = std::max( ymax, shapeBox.GetEnd().y );
748 }
749
750 bBox.SetX( xmin );
751 bBox.SetY( ymin );
752 bBox.SetWidth( xmax - xmin + 1 );
753 bBox.SetHeight( ymax - ymin + 1 );
754
755 bBox.Normalize();
756
757 return bBox;
758}
759
760
761wxString PCB_DIMENSION_BASE::GetItemDescription( UNITS_PROVIDER* aUnitsProvider, bool aFull ) const
762{
763 return wxString::Format( _( "Dimension '%s' on %s" ),
764 aFull ? GetShownText( false ) : KIUI::EllipsizeMenuText( GetText() ),
765 GetLayerName() );
766}
767
768
769
771{
773 VECTOR2I( GetBoundingBox().GetSize() ) );
774 dimBBox.Merge( PCB_TEXT::ViewBBox() );
775
776 return dimBBox;
777}
778
779
781 int aClearance, int aError, ERROR_LOC aErrorLoc,
782 bool aIgnoreLineWidth ) const
783{
784 wxASSERT_MSG( !aIgnoreLineWidth, wxT( "IgnoreLineWidth has no meaning for dimensions." ) );
785
786 for( const std::shared_ptr<SHAPE>& shape : m_shapes )
787 {
788 const SHAPE_CIRCLE* circle = dynamic_cast<const SHAPE_CIRCLE*>( shape.get() );
789 const SHAPE_SEGMENT* seg = dynamic_cast<const SHAPE_SEGMENT*>( shape.get() );
790
791 if( circle )
792 {
794 circle->GetRadius() + m_lineThickness / 2 + aClearance,
795 aError, aErrorLoc );
796 }
797 else if( seg )
798 {
799 TransformOvalToPolygon( aBuffer, seg->GetSeg().A, seg->GetSeg().B,
800 m_lineThickness + 2 * aClearance, aError, aErrorLoc );
801 }
802 else
803 {
804 wxFAIL_MSG( wxT( "PCB_DIMENSION_BASE::TransformShapeToPolygon unknown shape type." ) );
805 }
806 }
807}
808
809
811 PCB_DIMENSION_BASE( aParent, aType ),
812 m_height( 0 )
813{
814 // To preserve look of old dimensions, initialize extension height based on default arrow length
815 m_extensionHeight = static_cast<int>( m_arrowLength * s_arrowAngle.Sin() );
816}
817
818
820{
821 return new PCB_DIM_ALIGNED( *this );
822}
823
824
825void PCB_DIM_ALIGNED::Serialize( google::protobuf::Any &aContainer ) const
826{
827 using namespace kiapi::common;
828 kiapi::board::types::Dimension dimension;
829
830 PCB_DIMENSION_BASE::Serialize( aContainer );
831 aContainer.UnpackTo( &dimension );
832
833 PackVector2( *dimension.mutable_aligned()->mutable_start(), m_start );
834 PackVector2( *dimension.mutable_aligned()->mutable_end(), m_end );
835 dimension.mutable_aligned()->mutable_height()->set_value_nm( m_height );
836 dimension.mutable_aligned()->mutable_extension_height()->set_value_nm( m_extensionHeight );
837
838 aContainer.PackFrom( dimension );
839}
840
841
842bool PCB_DIM_ALIGNED::Deserialize( const google::protobuf::Any &aContainer )
843{
844 using namespace kiapi::common;
845
846 if( !PCB_DIMENSION_BASE::Deserialize( aContainer ) )
847 return false;
848
849 kiapi::board::types::Dimension dimension;
850 aContainer.UnpackTo( &dimension );
851
852 if( !dimension.has_aligned() )
853 return false;
854
855 SetStart( UnpackVector2( dimension.aligned().start() ) );
856 SetEnd( UnpackVector2( dimension.aligned().end() ) );
857 SetHeight( dimension.aligned().height().value_nm());
858 SetExtensionHeight( dimension.aligned().extension_height().value_nm() );
859
860 Update();
861
862 return true;
863}
864
865
867{
868 wxASSERT( aImage->Type() == Type() );
869
870 m_shapes.clear();
871 static_cast<PCB_DIM_ALIGNED*>( aImage )->m_shapes.clear();
872
873 std::swap( *static_cast<PCB_DIM_ALIGNED*>( this ), *static_cast<PCB_DIM_ALIGNED*>( aImage ) );
874
875 Update();
876}
877
878
879void PCB_DIM_ALIGNED::Mirror( const VECTOR2I& axis_pos, FLIP_DIRECTION aFlipDirection )
880{
882 // Call this last for the Update()
883 PCB_DIMENSION_BASE::Mirror( axis_pos, aFlipDirection );
884}
885
886
888{
889 return BITMAPS::add_aligned_dimension;
890}
891
892
893void PCB_DIM_ALIGNED::UpdateHeight( const VECTOR2I& aCrossbarStart, const VECTOR2I& aCrossbarEnd )
894{
895 VECTOR2D height( aCrossbarStart - GetStart() );
896 VECTOR2D crossBar( aCrossbarEnd - aCrossbarStart );
897
898 if( height.Cross( crossBar ) > 0 )
899 m_height = -height.EuclideanNorm();
900 else
901 m_height = height.EuclideanNorm();
902
903 Update();
904}
905
906
908{
909 if( m_busy ) // Skeep reentrance that happens sometimes after calling updateText()
910 return;
911
912 m_busy = true;
913
914 m_shapes.clear();
915
916 VECTOR2I dimension( m_end - m_start );
917
918 m_measuredValue = KiROUND( dimension.EuclideanNorm() );
919
920 VECTOR2I extension;
921
922 if( m_height > 0 )
923 extension = VECTOR2I( -dimension.y, dimension.x );
924 else
925 extension = VECTOR2I( dimension.y, -dimension.x );
926
927 // Add extension lines
928 int extensionHeight = std::abs( m_height ) - m_extensionOffset + m_extensionHeight;
929
930 VECTOR2I extStart( m_start );
931 extStart += extension.Resize( m_extensionOffset );
932
933 addShape( SHAPE_SEGMENT( extStart, extStart + extension.Resize( extensionHeight ) ) );
934
935 extStart = VECTOR2I( m_end );
936 extStart += extension.Resize( m_extensionOffset );
937
938 addShape( SHAPE_SEGMENT( extStart, extStart + extension.Resize( extensionHeight ) ) );
939
940 // Add crossbar
941 VECTOR2I crossBarDistance = sign( m_height ) * extension.Resize( m_height );
942 m_crossBarStart = m_start + crossBarDistance;
943 m_crossBarEnd = m_end + crossBarDistance;
944
945 // Update text after calculating crossbar position but before adding crossbar lines
946 updateText();
947
948 // Now that we have the text updated, we can determine how to draw the crossbar.
949 // First we need to create an appropriate bounding polygon to collide with
951
952 SHAPE_POLY_SET polyBox;
953 polyBox.NewOutline();
954 polyBox.Append( textBox.GetOrigin() );
955 polyBox.Append( textBox.GetOrigin().x, textBox.GetEnd().y );
956 polyBox.Append( textBox.GetEnd() );
957 polyBox.Append( textBox.GetEnd().x, textBox.GetOrigin().y );
958 polyBox.Rotate( GetTextAngle(), textBox.GetCenter() );
959
960 // The ideal crossbar, if the text doesn't collide
961 SEG crossbar( m_crossBarStart, m_crossBarEnd );
962
963 CollectKnockedOutSegments( polyBox, crossbar, m_shapes );
964
965 if( m_arrowDirection == DIM_ARROW_DIRECTION::INWARD )
966 {
967 drawAnArrow( m_crossBarStart, EDA_ANGLE( dimension ) + EDA_ANGLE( 180 ),
969 drawAnArrow( m_crossBarEnd, EDA_ANGLE( dimension ),
971 }
972 else
973 {
974 drawAnArrow( m_crossBarStart, EDA_ANGLE( dimension ), 0 );
975 drawAnArrow( m_crossBarEnd, EDA_ANGLE( dimension ) + EDA_ANGLE( 180 ), 0 );
976 }
977
978 m_busy = false;
979}
980
981
983{
984 VECTOR2I crossbarCenter( ( m_crossBarEnd - m_crossBarStart ) / 2 );
985
986 if( m_textPosition == DIM_TEXT_POSITION::OUTSIDE )
987 {
988 int textOffsetDistance = GetEffectiveTextPenWidth() + GetTextHeight();
989 EDA_ANGLE rotation;
990
991 if( crossbarCenter.x == 0 )
992 rotation = ANGLE_90 * sign( -crossbarCenter.y );
993 else if( crossbarCenter.x < 0 )
994 rotation = -ANGLE_90;
995 else
996 rotation = ANGLE_90;
997
998 VECTOR2I textOffset = crossbarCenter;
999 RotatePoint( textOffset, rotation );
1000 textOffset = crossbarCenter + textOffset.Resize( textOffsetDistance );
1001
1002 SetTextPos( m_crossBarStart + textOffset );
1003 }
1004 else if( m_textPosition == DIM_TEXT_POSITION::INLINE )
1005 {
1006 SetTextPos( m_crossBarStart + crossbarCenter );
1007 }
1008
1009 if( m_keepTextAligned )
1010 {
1011 EDA_ANGLE textAngle = FULL_CIRCLE - EDA_ANGLE( crossbarCenter );
1012 textAngle.Normalize();
1013
1014 if( textAngle > ANGLE_90 && textAngle <= ANGLE_270 )
1015 textAngle -= ANGLE_180;
1016
1017 SetTextAngle( textAngle );
1018 }
1019
1021}
1022
1023
1024void PCB_DIM_ALIGNED::GetMsgPanelInfo( EDA_DRAW_FRAME* aFrame, std::vector<MSG_PANEL_ITEM>& aList )
1025{
1026 PCB_DIMENSION_BASE::GetMsgPanelInfo( aFrame, aList );
1027
1028 // Use our own UNITS_PROVIDER to report dimension info in dimension's units rather than
1029 // in frame's units.
1030 UNITS_PROVIDER unitsProvider( pcbIUScale, EDA_UNITS::MM );
1031 unitsProvider.SetUserUnits( GetUnits() );
1032
1033 aList.emplace_back( _( "Height" ), unitsProvider.MessageTextFromValue( m_height ) );
1034}
1035
1036
1039{
1040 // To preserve look of old dimensions, initialize extension height based on default arrow length
1041 m_extensionHeight = static_cast<int>( m_arrowLength * s_arrowAngle.Sin() );
1043}
1044
1045
1047{
1048 return new PCB_DIM_ORTHOGONAL( *this );
1049}
1050
1051
1052void PCB_DIM_ORTHOGONAL::Serialize( google::protobuf::Any &aContainer ) const
1053{
1054 using namespace kiapi::common;
1055 kiapi::board::types::Dimension dimension;
1056
1057 PCB_DIMENSION_BASE::Serialize( aContainer );
1058 aContainer.UnpackTo( &dimension );
1059
1060 PackVector2( *dimension.mutable_orthogonal()->mutable_start(), m_start );
1061 PackVector2( *dimension.mutable_orthogonal()->mutable_end(), m_end );
1062 dimension.mutable_orthogonal()->mutable_height()->set_value_nm( m_height );
1063 dimension.mutable_orthogonal()->mutable_extension_height()->set_value_nm( m_extensionHeight );
1064
1065 dimension.mutable_orthogonal()->set_alignment( m_orientation == DIR::VERTICAL
1066 ? types::AxisAlignment::AA_Y_AXIS
1067 : types::AxisAlignment::AA_X_AXIS );
1068 aContainer.PackFrom( dimension );
1069}
1070
1071
1072bool PCB_DIM_ORTHOGONAL::Deserialize( const google::protobuf::Any &aContainer )
1073{
1074 using namespace kiapi::common;
1075
1076 if( !PCB_DIMENSION_BASE::Deserialize( aContainer ) )
1077 return false;
1078
1079 kiapi::board::types::Dimension dimension;
1080 aContainer.UnpackTo( &dimension );
1081
1082 if( !dimension.has_orthogonal() )
1083 return false;
1084
1085 SetStart( UnpackVector2( dimension.orthogonal().start() ) );
1086 SetEnd( UnpackVector2( dimension.orthogonal().end() ) );
1087 SetHeight( dimension.orthogonal().height().value_nm());
1088 SetExtensionHeight( dimension.orthogonal().extension_height().value_nm() );
1089 SetOrientation( dimension.orthogonal().alignment() == types::AxisAlignment::AA_Y_AXIS
1091 : DIR::HORIZONTAL );
1092
1093 Update();
1094
1095 return true;
1096}
1097
1098
1100{
1101 wxASSERT( aImage->Type() == Type() );
1102
1103 m_shapes.clear();
1104 static_cast<PCB_DIM_ORTHOGONAL*>( aImage )->m_shapes.clear();
1105
1106 std::swap( *static_cast<PCB_DIM_ORTHOGONAL*>( this ),
1107 *static_cast<PCB_DIM_ORTHOGONAL*>( aImage ) );
1108
1109 Update();
1110}
1111
1112
1113void PCB_DIM_ORTHOGONAL::Mirror( const VECTOR2I& axis_pos, FLIP_DIRECTION aFlipDirection )
1114{
1115 // Only reverse the height if the height is aligned with the flip
1116 if( m_orientation == DIR::HORIZONTAL && aFlipDirection == FLIP_DIRECTION::TOP_BOTTOM )
1117 m_height = -m_height;
1118 else if( m_orientation == DIR::VERTICAL && aFlipDirection == FLIP_DIRECTION::LEFT_RIGHT )
1119 m_height = -m_height;
1120
1121 // Call this last, as we need the Update()
1122 PCB_DIMENSION_BASE::Mirror( axis_pos, aFlipDirection );
1123}
1124
1125
1127{
1128 return BITMAPS::add_orthogonal_dimension;
1129}
1130
1131
1133{
1134 if( m_busy ) // Skeep reentrance that happens sometimes after calling updateText()
1135 return;
1136
1137 m_busy = true;
1138 m_shapes.clear();
1139
1141 m_end.y - m_start.y );
1143
1144 VECTOR2I extension;
1145
1147 extension = VECTOR2I( 0, m_height );
1148 else
1149 extension = VECTOR2I( m_height, 0 );
1150
1151 // Add first extension line
1152 int extensionHeight = std::abs( m_height ) - m_extensionOffset + m_extensionHeight;
1153
1154 VECTOR2I extStart( m_start );
1155 extStart += extension.Resize( m_extensionOffset );
1156
1157 addShape( SHAPE_SEGMENT( extStart, extStart + extension.Resize( extensionHeight ) ) );
1158
1159 // Add crossbar
1160 VECTOR2I crossBarDistance = sign( m_height ) * extension.Resize( m_height );
1161 m_crossBarStart = m_start + crossBarDistance;
1162
1165 else
1167
1168 // Add second extension line (m_end to crossbar end)
1170 extension = VECTOR2I( 0, m_end.y - m_crossBarEnd.y );
1171 else
1172 extension = VECTOR2I( m_end.x - m_crossBarEnd.x, 0 );
1173
1174 extensionHeight = extension.EuclideanNorm() - m_extensionOffset + m_extensionHeight;
1175
1176 extStart = VECTOR2I( m_crossBarEnd );
1177 extStart -= extension.Resize( m_extensionHeight );
1178
1179 addShape( SHAPE_SEGMENT( extStart, extStart + extension.Resize( extensionHeight ) ) );
1180
1181 // Update text after calculating crossbar position but before adding crossbar lines
1182 updateText();
1183
1184 // Now that we have the text updated, we can determine how to draw the crossbar.
1185 // First we need to create an appropriate bounding polygon to collide with
1187
1188 SHAPE_POLY_SET polyBox;
1189 polyBox.NewOutline();
1190 polyBox.Append( textBox.GetOrigin() );
1191 polyBox.Append( textBox.GetOrigin().x, textBox.GetEnd().y );
1192 polyBox.Append( textBox.GetEnd() );
1193 polyBox.Append( textBox.GetEnd().x, textBox.GetOrigin().y );
1194 polyBox.Rotate( GetTextAngle(), textBox.GetCenter() );
1195
1196 // The ideal crossbar, if the text doesn't collide
1197 SEG crossbar( m_crossBarStart, m_crossBarEnd );
1198
1199 CollectKnockedOutSegments( polyBox, crossbar, m_shapes );
1200
1201 EDA_ANGLE crossBarAngle( m_crossBarEnd - m_crossBarStart );
1202
1203 if( m_arrowDirection == DIM_ARROW_DIRECTION::INWARD )
1204 {
1205 // Arrows with fixed length.
1206 drawAnArrow( m_crossBarStart, crossBarAngle + EDA_ANGLE( 180 ),
1209 }
1210 else
1211 {
1212 drawAnArrow( m_crossBarStart, crossBarAngle, 0 );
1213 drawAnArrow( m_crossBarEnd, crossBarAngle + EDA_ANGLE( 180 ), 0 );
1214 }
1215
1216 m_busy = false;
1217}
1218
1219
1221{
1222 VECTOR2I crossbarCenter( ( m_crossBarEnd - m_crossBarStart ) / 2 );
1223
1224 if( m_textPosition == DIM_TEXT_POSITION::OUTSIDE )
1225 {
1226 int textOffsetDistance = GetEffectiveTextPenWidth() + GetTextHeight();
1227
1228 VECTOR2I textOffset;
1229
1231 textOffset.y = -textOffsetDistance;
1232 else
1233 textOffset.x = -textOffsetDistance;
1234
1235 textOffset += crossbarCenter;
1236
1237 SetTextPos( m_crossBarStart + textOffset );
1238 }
1239 else if( m_textPosition == DIM_TEXT_POSITION::INLINE )
1240 {
1241 SetTextPos( m_crossBarStart + crossbarCenter );
1242 }
1243
1244 if( m_keepTextAligned )
1245 {
1246 if( abs( crossbarCenter.x ) > abs( crossbarCenter.y ) )
1248 else
1250 }
1251
1253}
1254
1255
1256void PCB_DIM_ORTHOGONAL::Rotate( const VECTOR2I& aRotCentre, const EDA_ANGLE& aAngle )
1257{
1258 EDA_ANGLE angle( aAngle );
1259
1260 // restrict angle to -179.9 to 180.0 degrees
1261 angle.Normalize180();
1262
1263 // adjust orientation and height to new angle
1264 // we can only handle the cases of -90, 0, 90, 180 degrees exactly;
1265 // in the other cases we will use the nearest 90 degree angle to
1266 // choose at least an approximate axis for the target orientation
1267 // In case of exactly 45 or 135 degrees, we will round towards zero for consistency
1268 if( angle > ANGLE_45 && angle <= ANGLE_135 )
1269 {
1270 // about 90 degree
1272 {
1274 }
1275 else
1276 {
1278 m_height = -m_height;
1279 }
1280 }
1281 else if( angle < -ANGLE_45 && angle >= -ANGLE_135 )
1282 {
1283 // about -90 degree
1285 {
1287 m_height = -m_height;
1288 }
1289 else
1290 {
1292 }
1293 }
1294 else if( angle > ANGLE_135 || angle < -ANGLE_135 )
1295 {
1296 // about 180 degree
1297 m_height = -m_height;
1298 }
1299
1300 // this will update m_crossBarStart and m_crossbarEnd
1301 PCB_DIMENSION_BASE::Rotate( aRotCentre, angle );
1302}
1303
1304
1307 m_textBorder( DIM_TEXT_BORDER::NONE )
1308{
1309 m_unitsFormat = DIM_UNITS_FORMAT::NO_SUFFIX;
1310 m_overrideTextEnabled = true;
1311 m_keepTextAligned = false;
1312
1313 SetOverrideText( _( "Leader" ) );
1314}
1315
1316
1317void PCB_DIM_LEADER::Serialize( google::protobuf::Any &aContainer ) const
1318{
1319 using namespace kiapi::common;
1320 kiapi::board::types::Dimension dimension;
1321
1322 PCB_DIMENSION_BASE::Serialize( aContainer );
1323 aContainer.UnpackTo( &dimension );
1324
1325 PackVector2( *dimension.mutable_leader()->mutable_start(), m_start );
1326 PackVector2( *dimension.mutable_leader()->mutable_end(), m_end );
1327 dimension.mutable_leader()->set_border_style(
1328 ToProtoEnum<DIM_TEXT_BORDER, kiapi::board::types::DimensionTextBorderStyle>(
1329 m_textBorder ) );
1330
1331 aContainer.PackFrom( dimension );
1332}
1333
1334
1335bool PCB_DIM_LEADER::Deserialize( const google::protobuf::Any &aContainer )
1336{
1337 using namespace kiapi::common;
1338
1339 if( !PCB_DIMENSION_BASE::Deserialize( aContainer ) )
1340 return false;
1341
1342 kiapi::board::types::Dimension dimension;
1343 aContainer.UnpackTo( &dimension );
1344
1345 if( !dimension.has_leader() )
1346 return false;
1347
1348 SetStart( UnpackVector2( dimension.leader().start() ) );
1349 SetEnd( UnpackVector2( dimension.leader().end() ) );
1350 SetTextBorder( FromProtoEnum<DIM_TEXT_BORDER>( dimension.leader().border_style() ) );
1351
1352 Update();
1353
1354 return true;
1355}
1356
1357
1359{
1360 return new PCB_DIM_LEADER( *this );
1361}
1362
1363
1365{
1366 wxASSERT( aImage->Type() == Type() );
1367
1368 m_shapes.clear();
1369 static_cast<PCB_DIM_LEADER*>( aImage )->m_shapes.clear();
1370
1371 std::swap( *static_cast<PCB_DIM_LEADER*>( this ), *static_cast<PCB_DIM_LEADER*>( aImage ) );
1372
1373 Update();
1374}
1375
1376
1378{
1379 return BITMAPS::add_leader;
1380}
1381
1382
1384{
1385 // Our geometry is dependent on the size of the text, so just update the whole shebang
1387}
1388
1389
1391{
1392 if( m_busy ) // Skeep reentrance that happens sometimes after calling updateText()
1393 return;
1394
1395 m_busy = true;
1396
1397 m_shapes.clear();
1398
1400
1401 // Now that we have the text updated, we can determine how to draw the second line
1402 // First we need to create an appropriate bounding polygon to collide with
1403 BOX2I textBox = GetTextBox().Inflate( GetTextWidth() / 2, GetEffectiveTextPenWidth() * 2 );
1404
1405 SHAPE_POLY_SET polyBox;
1406 polyBox.NewOutline();
1407 polyBox.Append( textBox.GetOrigin() );
1408 polyBox.Append( textBox.GetOrigin().x, textBox.GetEnd().y );
1409 polyBox.Append( textBox.GetEnd() );
1410 polyBox.Append( textBox.GetEnd().x, textBox.GetOrigin().y );
1411 polyBox.Rotate( GetTextAngle(), textBox.GetCenter() );
1412
1413 VECTOR2I firstLine( m_end - m_start );
1414 VECTOR2I start( m_start );
1415 start += firstLine.Resize( m_extensionOffset );
1416
1417 SEG arrowSeg( m_start, m_end );
1418 SEG textSeg( m_end, GetTextPos() );
1419 OPT_VECTOR2I arrowSegEnd;
1420 OPT_VECTOR2I textSegEnd;
1421
1422 if( m_textBorder == DIM_TEXT_BORDER::CIRCLE )
1423 {
1424 double penWidth = GetEffectiveTextPenWidth() / 2.0;
1425 double radius = ( textBox.GetWidth() / 2.0 ) - penWidth;
1426 CIRCLE circle( textBox.GetCenter(), radius );
1427
1428 arrowSegEnd = segCircleIntersection( circle, arrowSeg );
1429 textSegEnd = segCircleIntersection( circle, textSeg );
1430 }
1431 else
1432 {
1433 arrowSegEnd = segPolyIntersection( polyBox, arrowSeg );
1434 textSegEnd = segPolyIntersection( polyBox, textSeg );
1435 }
1436
1437 if( !arrowSegEnd )
1438 arrowSegEnd = m_end;
1439
1440 m_shapes.emplace_back( new SHAPE_SEGMENT( start, *arrowSegEnd ) );
1441
1442 drawAnArrow( start, EDA_ANGLE( firstLine ), 0 );
1443
1444 if( !GetText().IsEmpty() )
1445 {
1446 switch( m_textBorder )
1447 {
1448 case DIM_TEXT_BORDER::RECTANGLE:
1449 {
1450 for( SHAPE_POLY_SET::SEGMENT_ITERATOR seg = polyBox.IterateSegments(); seg; seg++ )
1451 m_shapes.emplace_back( new SHAPE_SEGMENT( *seg ) );
1452
1453 break;
1454 }
1455
1456 case DIM_TEXT_BORDER::CIRCLE:
1457 {
1458 double penWidth = GetEffectiveTextPenWidth() / 2.0;
1459 double radius = ( textBox.GetWidth() / 2.0 ) - penWidth;
1460 m_shapes.emplace_back( new SHAPE_CIRCLE( textBox.GetCenter(), radius ) );
1461
1462 break;
1463 }
1464
1465 default:
1466 break;
1467 }
1468 }
1469
1470 if( textSegEnd && *arrowSegEnd == m_end )
1471 m_shapes.emplace_back( new SHAPE_SEGMENT( m_end, *textSegEnd ) );
1472
1473 m_busy = false;
1474}
1475
1476
1477void PCB_DIM_LEADER::GetMsgPanelInfo( EDA_DRAW_FRAME* aFrame, std::vector<MSG_PANEL_ITEM>& aList )
1478{
1479 // Don't use GetShownText(); we want to see the variable references here
1480 aList.emplace_back( _( "Leader" ), KIUI::EllipsizeStatusText( aFrame, GetText() ) );
1481
1482 ORIGIN_TRANSFORMS& originTransforms = aFrame->GetOriginTransforms();
1483
1484 VECTOR2I startCoord = originTransforms.ToDisplayAbs( GetStart() );
1485 wxString start = wxString::Format( wxT( "@(%s, %s)" ),
1486 aFrame->MessageTextFromValue( startCoord.x ),
1487 aFrame->MessageTextFromValue( startCoord.y ) );
1488
1489 aList.emplace_back( start, wxEmptyString );
1490
1491 aList.emplace_back( _( "Layer" ), GetLayerName() );
1492}
1493
1494
1497{
1498 m_unitsFormat = DIM_UNITS_FORMAT::NO_SUFFIX;
1499 m_overrideTextEnabled = false;
1500 m_keepTextAligned = true;
1501 m_prefix = "R ";
1503}
1504
1505
1506void PCB_DIM_RADIAL::Serialize( google::protobuf::Any &aContainer ) const
1507{
1508 using namespace kiapi::common;
1509 kiapi::board::types::Dimension dimension;
1510
1511 PCB_DIMENSION_BASE::Serialize( aContainer );
1512 aContainer.UnpackTo( &dimension );
1513
1514 PackVector2( *dimension.mutable_radial()->mutable_center(), m_start );
1515 PackVector2( *dimension.mutable_radial()->mutable_radius_point(), m_end );
1516 dimension.mutable_radial()->mutable_leader_length()->set_value_nm( m_leaderLength );
1517
1518 aContainer.PackFrom( dimension );
1519}
1520
1521
1522bool PCB_DIM_RADIAL::Deserialize( const google::protobuf::Any &aContainer )
1523{
1524 using namespace kiapi::common;
1525
1526 if( !PCB_DIMENSION_BASE::Deserialize( aContainer ) )
1527 return false;
1528
1529 kiapi::board::types::Dimension dimension;
1530 aContainer.UnpackTo( &dimension );
1531
1532 if( !dimension.has_radial() )
1533 return false;
1534
1535 SetStart( UnpackVector2( dimension.radial().center() ) );
1536 SetEnd( UnpackVector2( dimension.radial().radius_point() ) );
1537 SetLeaderLength( dimension.radial().leader_length().value_nm() );
1538
1539 Update();
1540
1541 return true;
1542}
1543
1544
1546{
1547 return new PCB_DIM_RADIAL( *this );
1548}
1549
1550
1552{
1553 wxASSERT( aImage->Type() == Type() );
1554
1555 m_shapes.clear();
1556 static_cast<PCB_DIM_RADIAL*>( aImage )->m_shapes.clear();
1557
1558 std::swap( *static_cast<PCB_DIM_RADIAL*>( this ), *static_cast<PCB_DIM_RADIAL*>( aImage ) );
1559
1560 Update();
1561}
1562
1563
1565{
1566 return BITMAPS::add_radial_dimension;
1567}
1568
1569
1571{
1572 VECTOR2I radial( m_end - m_start );
1573
1574 return m_end + radial.Resize( m_leaderLength );
1575}
1576
1577
1579{
1580 if( m_keepTextAligned )
1581 {
1582 VECTOR2I textLine( GetTextPos() - GetKnee() );
1583 EDA_ANGLE textAngle = FULL_CIRCLE - EDA_ANGLE( textLine );
1584
1585 textAngle.Normalize();
1586
1587 if( textAngle > ANGLE_90 && textAngle <= ANGLE_270 )
1588 textAngle -= ANGLE_180;
1589
1590 // Round to nearest degree
1591 textAngle = EDA_ANGLE( KiROUND( textAngle.AsDegrees() ), DEGREES_T );
1592
1593 SetTextAngle( textAngle );
1594 }
1595
1597}
1598
1599
1601{
1602 if( m_busy ) // Skeep reentrance that happens sometimes after calling updateText()
1603 return;
1604
1605 m_busy = true;
1606
1607 m_shapes.clear();
1608
1610 VECTOR2I centerArm( 0, m_arrowLength );
1611
1612 m_shapes.emplace_back( new SHAPE_SEGMENT( center - centerArm, center + centerArm ) );
1613
1614 RotatePoint( centerArm, -ANGLE_90 );
1615
1616 m_shapes.emplace_back( new SHAPE_SEGMENT( center - centerArm, center + centerArm ) );
1617
1619
1620 m_measuredValue = KiROUND( radius.EuclideanNorm() );
1621
1622 updateText();
1623
1624 // Now that we have the text updated, we can determine how to draw the second line
1625 // First we need to create an appropriate bounding polygon to collide with
1627
1628 SHAPE_POLY_SET polyBox;
1629 polyBox.NewOutline();
1630 polyBox.Append( textBox.GetOrigin() );
1631 polyBox.Append( textBox.GetOrigin().x, textBox.GetEnd().y );
1632 polyBox.Append( textBox.GetEnd() );
1633 polyBox.Append( textBox.GetEnd().x, textBox.GetOrigin().y );
1634 polyBox.Rotate( GetTextAngle(), textBox.GetCenter() );
1635
1636 VECTOR2I radial( m_end - m_start );
1637 radial = radial.Resize( m_leaderLength );
1638
1639 SEG arrowSeg( m_end, m_end + radial );
1640 SEG textSeg( arrowSeg.B, GetTextPos() );
1641
1642 CollectKnockedOutSegments( polyBox, arrowSeg, m_shapes );
1643 CollectKnockedOutSegments( polyBox, textSeg, m_shapes );
1644
1645 drawAnArrow( m_end, EDA_ANGLE( radial ), 0 );
1646
1647 m_busy = false;
1648}
1649
1650
1653{
1654 m_unitsFormat = DIM_UNITS_FORMAT::NO_SUFFIX;
1655 m_overrideTextEnabled = true;
1656}
1657
1658
1659void PCB_DIM_CENTER::Serialize( google::protobuf::Any &aContainer ) const
1660{
1661 using namespace kiapi::common;
1662 kiapi::board::types::Dimension dimension;
1663
1664 PCB_DIMENSION_BASE::Serialize( aContainer );
1665 aContainer.UnpackTo( &dimension );
1666
1667 PackVector2( *dimension.mutable_center()->mutable_center(), m_start );
1668 PackVector2( *dimension.mutable_center()->mutable_end(), m_end );
1669
1670 aContainer.PackFrom( dimension );
1671}
1672
1673
1674bool PCB_DIM_CENTER::Deserialize( const google::protobuf::Any &aContainer )
1675{
1676 using namespace kiapi::common;
1677
1678 if( !PCB_DIMENSION_BASE::Deserialize( aContainer ) )
1679 return false;
1680
1681 kiapi::board::types::Dimension dimension;
1682 aContainer.UnpackTo( &dimension );
1683
1684 if( !dimension.has_center() )
1685 return false;
1686
1687 SetStart( UnpackVector2( dimension.center().center() ) );
1688 SetEnd( UnpackVector2( dimension.center().end() ) );
1689
1690 Update();
1691
1692 return true;
1693}
1694
1695
1697{
1698 return new PCB_DIM_CENTER( *this );
1699}
1700
1701
1703{
1704 wxASSERT( aImage->Type() == Type() );
1705
1706 std::swap( *static_cast<PCB_DIM_CENTER*>( this ), *static_cast<PCB_DIM_CENTER*>( aImage ) );
1707}
1708
1709
1711{
1712 return BITMAPS::add_center_dimension;
1713}
1714
1715
1717{
1718 int halfWidth = VECTOR2I( m_end - m_start ).x + ( m_lineThickness / 2.0 );
1719
1720 BOX2I bBox;
1721
1722 bBox.SetX( m_start.x - halfWidth );
1723 bBox.SetY( m_start.y - halfWidth );
1724 bBox.SetWidth( halfWidth * 2 );
1725 bBox.SetHeight( halfWidth * 2 );
1726
1727 bBox.Normalize();
1728
1729 return bBox;
1730}
1731
1732
1734{
1736 VECTOR2I( GetBoundingBox().GetSize() ) );
1737}
1738
1739
1741{
1742 if( m_busy ) // Skeep reentrance that happens sometimes after calling updateText()
1743 return;
1744
1745 m_busy = true;
1746
1747 m_shapes.clear();
1748
1750 VECTOR2I arm( m_end - m_start );
1751
1752 m_shapes.emplace_back( new SHAPE_SEGMENT( center - arm, center + arm ) );
1753
1754 RotatePoint( arm, -ANGLE_90 );
1755
1756 m_shapes.emplace_back( new SHAPE_SEGMENT( center - arm, center + arm ) );
1757
1758 updateText();
1759
1760 m_busy = false;
1761}
1762
1763
1764static struct DIMENSION_DESC
1765{
1767 {
1769 .Map( DIM_PRECISION::X, _HKI( "0" ) )
1770 .Map( DIM_PRECISION::X_X, _HKI( "0.0" ) )
1771 .Map( DIM_PRECISION::X_XX, _HKI( "0.00" ) )
1772 .Map( DIM_PRECISION::X_XXX, _HKI( "0.000" ) )
1773 .Map( DIM_PRECISION::X_XXXX, _HKI( "0.0000" ) )
1774 .Map( DIM_PRECISION::X_XXXXX, _HKI( "0.00000" ) )
1775 .Map( DIM_PRECISION::V_VV, _HKI( "0.00 in / 0 mils / 0.0 mm" ) )
1776 .Map( DIM_PRECISION::V_VVV, _HKI( "0.000 / 0 / 0.00" ) )
1777 .Map( DIM_PRECISION::V_VVVV, _HKI( "0.0000 / 0.0 / 0.000" ) )
1778 .Map( DIM_PRECISION::V_VVVVV, _HKI( "0.00000 / 0.00 / 0.0000" ) );
1779
1781 .Map( DIM_UNITS_FORMAT::NO_SUFFIX, _HKI( "1234.0" ) )
1782 .Map( DIM_UNITS_FORMAT::BARE_SUFFIX, _HKI( "1234.0 mm" ) )
1783 .Map( DIM_UNITS_FORMAT::PAREN_SUFFIX, _HKI( "1234.0 (mm)" ) );
1784
1786 .Map( DIM_UNITS_MODE::INCH, _HKI( "Inches" ) )
1787 .Map( DIM_UNITS_MODE::MILS, _HKI( "Mils" ) )
1788 .Map( DIM_UNITS_MODE::MM, _HKI( "Millimeters" ) )
1789 .Map( DIM_UNITS_MODE::AUTOMATIC, _HKI( "Automatic" ) );
1790
1792 .Map( DIM_ARROW_DIRECTION::INWARD, _HKI( "Inward" ) )
1793 .Map( DIM_ARROW_DIRECTION::OUTWARD, _HKI( "Outward" ) );
1794
1803
1804 propMgr.Mask( TYPE_HASH( PCB_DIMENSION_BASE ), TYPE_HASH( EDA_TEXT ), _HKI( "Orientation" ) );
1805
1806 const wxString groupDimension = _HKI( "Dimension Properties" );
1807
1808 auto isLeader =
1809 []( INSPECTABLE* aItem ) -> bool
1810 {
1811 return dynamic_cast<PCB_DIM_LEADER*>( aItem ) != nullptr;
1812 };
1813
1814 auto isNotLeader =
1815 []( INSPECTABLE* aItem ) -> bool
1816 {
1817 return dynamic_cast<PCB_DIM_LEADER*>( aItem ) == nullptr;
1818 };
1819
1820 auto isMultiArrowDirection =
1821 []( INSPECTABLE* aItem ) -> bool
1822 {
1823 return dynamic_cast<PCB_DIM_ALIGNED*>( aItem ) != nullptr;
1824 };
1825
1828 groupDimension )
1829 .SetAvailableFunc( isNotLeader );
1832 groupDimension )
1833 .SetAvailableFunc( isNotLeader );
1834 propMgr.AddProperty( new PROPERTY<PCB_DIMENSION_BASE, wxString>( _HKI( "Override Text" ),
1836 groupDimension )
1837 .SetAvailableFunc( isNotLeader );
1838
1841 groupDimension )
1842 .SetAvailableFunc( isLeader );
1843
1846 groupDimension )
1847 .SetAvailableFunc( isNotLeader );
1850 groupDimension )
1851 .SetAvailableFunc( isNotLeader );
1854 groupDimension )
1855 .SetAvailableFunc( isNotLeader );
1856 propMgr.AddProperty( new PROPERTY<PCB_DIMENSION_BASE, bool>( _HKI( "Suppress Trailing Zeroes" ),
1858 groupDimension )
1859 .SetAvailableFunc( isNotLeader );
1860
1863 groupDimension )
1864 .SetAvailableFunc( isMultiArrowDirection );
1865
1866 const wxString groupText = _HKI( "Text Properties" );
1867
1868 const auto isTextOrientationWriteable =
1869 []( INSPECTABLE* aItem ) -> bool
1870 {
1871 return !static_cast<PCB_DIMENSION_BASE*>( aItem )->GetKeepTextAligned();
1872 };
1873
1874 propMgr.AddProperty( new PROPERTY<PCB_DIMENSION_BASE, bool>( _HKI( "Keep Aligned with Dimension" ),
1877 groupText );
1878
1879 propMgr.AddProperty( new PROPERTY<PCB_DIMENSION_BASE, double>( _HKI( "Orientation" ),
1882 PROPERTY_DISPLAY::PT_DEGREE ),
1883 groupText )
1884 .SetWriteableFunc( isTextOrientationWriteable );
1885 }
1887
1892
1893
1895{
1897 {
1908
1909 const wxString groupDimension = _HKI( "Dimension Properties" );
1910
1911 propMgr.AddProperty( new PROPERTY<PCB_DIM_ALIGNED, int>( _HKI( "Crossbar Height" ),
1914 groupDimension );
1915 propMgr.AddProperty( new PROPERTY<PCB_DIM_ALIGNED, int>( _HKI( "Extension Line Overshoot" ),
1918 groupDimension );
1919
1921 _HKI( "Text" ),
1922 []( INSPECTABLE* aItem ) { return false; } );
1924 _HKI( "Vertical Justification" ),
1925 []( INSPECTABLE* aItem ) { return false; } );
1927 _HKI( "Hyperlink" ),
1928 []( INSPECTABLE* aItem ) { return false; } );
1930 _HKI( "Knockout" ),
1931 []( INSPECTABLE* aItem ) { return false; } );
1932 }
1934
1935
1937{
1939 {
1952
1954 _HKI( "Text" ),
1955 []( INSPECTABLE* aItem ) { return false; } );
1957 _HKI( "Vertical Justification" ),
1958 []( INSPECTABLE* aItem ) { return false; } );
1960 _HKI( "Hyperlink" ),
1961 []( INSPECTABLE* aItem ) { return false; } );
1963 _HKI( "Knockout" ),
1964 []( INSPECTABLE* aItem ) { return false; } );
1965 }
1967
1968
1970{
1972 {
1983
1984 const wxString groupDimension = _HKI( "Dimension Properties" );
1985
1986 propMgr.AddProperty( new PROPERTY<PCB_DIM_RADIAL, int>( _HKI( "Leader Length" ),
1988 PROPERTY_DISPLAY::PT_SIZE ),
1989 groupDimension );
1990
1992 _HKI( "Text" ),
1993 []( INSPECTABLE* aItem ) { return false; } );
1995 _HKI( "Vertical Justification" ),
1996 []( INSPECTABLE* aItem ) { return false; } );
1998 _HKI( "Hyperlink" ),
1999 []( INSPECTABLE* aItem ) { return false; } );
2001 _HKI( "Knockout" ),
2002 []( INSPECTABLE* aItem ) { return false; } );
2003 }
2005
2006
2008{
2010 {
2012 .Map( DIM_TEXT_BORDER::NONE, _HKI( "None" ) )
2013 .Map( DIM_TEXT_BORDER::RECTANGLE, _HKI( "Rectangle" ) )
2014 .Map( DIM_TEXT_BORDER::CIRCLE, _HKI( "Circle" ) );
2015
2026
2027 const wxString groupDimension = _HKI( "Dimension Properties" );
2028
2031 groupDimension );
2032
2034 _HKI( "Text" ),
2035 []( INSPECTABLE* aItem ) { return false; } );
2037 _HKI( "Vertical Justification" ),
2038 []( INSPECTABLE* aItem ) { return false; } );
2040 _HKI( "Hyperlink" ),
2041 []( INSPECTABLE* aItem ) { return false; } );
2043 _HKI( "Knockout" ),
2044 []( INSPECTABLE* aItem ) { return false; } );
2045 }
2047
2049
2050
2052{
2054 {
2065
2066
2068 _HKI( "Text" ),
2069 []( INSPECTABLE* aItem ) { return false; } );
2071 _HKI( "Vertical Justification" ),
2072 []( INSPECTABLE* aItem ) { return false; } );
2074 _HKI( "Hyperlink" ),
2075 []( INSPECTABLE* aItem ) { return false; } );
2077 _HKI( "Knockout" ),
2078 []( INSPECTABLE* aItem ) { return false; } );
2079 }
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
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
BITMAPS
A list of all bitmap identifiers.
Definition: bitmaps_list.h:33
BOX2< VECTOR2I > BOX2I
Definition: box2.h:922
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition: board_item.h:78
virtual PCB_LAYER_ID GetLayer() const
Return the primary layer this item is on.
Definition: board_item.h:229
virtual void SetLocked(bool aLocked)
Definition: board_item.h:320
PCB_LAYER_ID m_layer
Definition: board_item.h:445
virtual void SetLayer(PCB_LAYER_ID aLayer)
Set the layer this item is on.
Definition: board_item.h:280
virtual const BOARD * GetBoard() const
Return the BOARD in which this BOARD_ITEM resides, or NULL if none.
Definition: board_item.cpp:48
virtual bool IsLocked() const
Definition: board_item.cpp:76
bool IsSideSpecific() const
Definition: board_item.cpp:150
wxString GetLayerName() const
Return the name of the PCB layer on which the item resides.
Definition: board_item.cpp:140
EDA_UNITS GetUserUnits()
Definition: board.h:727
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 SetHeight(size_type val)
Definition: box2.h:292
constexpr const Vec GetEnd() const
Definition: box2.h:212
constexpr BOX2< Vec > & Normalize()
Ensure that the height and width are positive.
Definition: box2.h:146
constexpr coord_type GetY() const
Definition: box2.h:208
constexpr size_type GetWidth() const
Definition: box2.h:214
constexpr coord_type GetX() const
Definition: box2.h:207
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 const Vec GetCenter() const
Definition: box2.h:230
constexpr bool Contains(const Vec &aPoint) const
Definition: box2.h:168
constexpr void SetWidth(size_type val)
Definition: box2.h:287
constexpr void SetX(coord_type val)
Definition: box2.h:277
constexpr const Vec & GetOrigin() const
Definition: box2.h:210
constexpr void SetY(coord_type val)
Definition: box2.h:282
constexpr coord_type GetRight() const
Definition: box2.h:217
constexpr bool Intersects(const BOX2< Vec > &aRect) const
Definition: box2.h:311
constexpr coord_type GetBottom() const
Definition: box2.h:222
Represent basic circle geometry with utility geometry functions.
Definition: circle.h:33
std::vector< VECTOR2I > Intersect(const CIRCLE &aCircle) const
Compute the intersection points between this circle and aCircle.
Definition: circle.cpp:221
bool Contains(const VECTOR2I &aP) const
Return true if aP is on the circumference of this circle.
Definition: circle.cpp:188
EDA_ANGLE Normalize()
Definition: eda_angle.h:221
double Sin() const
Definition: eda_angle.h:170
double AsDegrees() const
Definition: eda_angle.h:113
EDA_ANGLE Normalize180()
Definition: eda_angle.h:260
ORIGIN_TRANSFORMS & GetOriginTransforms() override
Return a reference to the default ORIGIN_TRANSFORMS object.
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
const KIID m_Uuid
Definition: eda_item.h:498
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:108
EDA_ITEM * m_parent
Linked list: Link (parent struct).
Definition: eda_item.h:510
A mix-in class (via multiple inheritance) that handles texts such as labels, parts,...
Definition: eda_text.h:80
int GetTextHeight() const
Definition: eda_text.h:254
void Serialize(google::protobuf::Any &aContainer) const override
Serializes this object to the given Any message.
Definition: eda_text.cpp:187
const VECTOR2I & GetTextPos() const
Definition: eda_text.h:260
const EDA_ANGLE & GetTextAngle() const
Definition: eda_text.h:134
virtual const wxString & GetText() const
Return the string associated with the text object.
Definition: eda_text.h:98
void SetTextPos(const VECTOR2I &aPoint)
Definition: eda_text.cpp:571
bool Deserialize(const google::protobuf::Any &aContainer) override
Deserializes the given protobuf message into this object.
Definition: eda_text.cpp:223
KIFONT::FONT * GetFont() const
Definition: eda_text.h:234
BOX2I GetTextBox(int aLine=-1) const
Useful in multiline texts to calculate the full text or a line area (for zones filling,...
Definition: eda_text.cpp:723
void SetMirrored(bool isMirrored)
Definition: eda_text.cpp:386
int GetTextWidth() const
Definition: eda_text.h:251
void Offset(const VECTOR2I &aOffset)
Definition: eda_text.cpp:589
double Similarity(const EDA_TEXT &aOther) const
Definition: eda_text.cpp:1279
virtual void ClearRenderCache()
Definition: eda_text.cpp:651
bool IsMirrored() const
Definition: eda_text.h:177
int GetEffectiveTextPenWidth(int aDefaultPenWidth=0) const
The EffectiveTextPenWidth uses the text thickness if > 1 or aDefaultPenWidth.
Definition: eda_text.cpp:458
std::shared_ptr< SHAPE_COMPOUND > GetEffectiveTextShape(bool aTriangulate=true, const BOX2I &aBBox=BOX2I(), const EDA_ANGLE &aAngle=ANGLE_0) const
build a list of segments (SHAPE_SEGMENT) to describe a text shape.
Definition: eda_text.cpp:1120
void SetTextAngleDegrees(double aOrientation)
Definition: eda_text.h:137
virtual void SetText(const wxString &aText)
Definition: eda_text.cpp:270
virtual void SetTextAngle(const EDA_ANGLE &aAngle)
Definition: eda_text.cpp:292
int GetTextThickness() const
Definition: eda_text.h:126
bool operator==(const EDA_TEXT &aRhs) const
Definition: eda_text.h:382
static ENUM_MAP< T > & Instance()
Definition: property.h:680
Class that other classes need to inherit from, in order to be inspectable.
Definition: inspectable.h:37
Definition: kiid.h:49
std::string AsStdString() const
Definition: kiid.cpp:252
A class to perform either relative or absolute display origin transforms for a single axis of a point...
T ToDisplayAbs(const T &aValue) const
Abstract dimension API.
EDA_UNITS GetUnits() const
bool m_autoUnits
If true, follow the currently selected UI units.
void Flip(const VECTOR2I &aCentre, FLIP_DIRECTION aFlipDirection) override
Flip this object, i.e.
void Update()
Update the dimension's cached text and geometry.
void ChangeSuffix(const wxString &aSuffix)
wxString GetOverrideText() const
wxString GetSuffix() const
std::vector< std::shared_ptr< SHAPE > > m_shapes
double Similarity(const BOARD_ITEM &aOther) const override
Return a measure of how likely the other object is to represent the same object.
void TransformShapeToPolygon(SHAPE_POLY_SET &aBuffer, PCB_LAYER_ID aLayer, int aClearance, int aError, ERROR_LOC aErrorLoc, bool aIgnoreLineWidth=false) const override
Convert the item shape to a closed polygon.
void Serialize(google::protobuf::Any &aContainer) const override
Serializes this object to the given Any message.
void ClearRenderCache() override
int m_lineThickness
Thickness used for all graphics in the dimension.
void Move(const VECTOR2I &offset) override
Move this object.
void SetUnitsFormat(const DIM_UNITS_FORMAT aFormat)
bool m_suppressZeroes
Suppress trailing zeroes.
wxString GetItemDescription(UNITS_PROVIDER *aUnitsProvider, bool aFull) const override
Return a user-visible description string of this item.
void SetUnits(EDA_UNITS aUnits)
DIM_PRECISION m_precision
Number of digits to display after decimal.
std::shared_ptr< SHAPE > GetEffectiveShape(PCB_LAYER_ID aLayer, FLASHING aFlash=FLASHING::DEFAULT) const override
Some pad shapes can be complex (rounded/chamfered rectangle), even without considering custom shapes.
void addShape(const ShapeType &aShape)
void SetPrefix(const wxString &aPrefix)
wxString m_suffix
String appended to the value.
const BOX2I GetBoundingBox() const override
Return the orthogonal bounding box of this object for display purposes.
void ChangeOverrideText(const wxString &aValue)
void SetExtensionOffset(int aOffset)
void SetSuppressZeroes(bool aSuppress)
bool m_inClearRenderCache
re-entrancy guard
PCB_DIMENSION_BASE(BOARD_ITEM *aParent, KICAD_T aType=PCB_DIMENSION_T)
int m_extensionOffset
Distance from feature points to extension line start.
void ChangeArrowDirection(const DIM_ARROW_DIRECTION &aDirection)
bool GetKeepTextAligned() const
void ChangeTextAngleDegrees(double aDegrees)
bool m_keepTextAligned
Calculate text orientation to match dimension.
DIM_PRECISION GetPrecision() const
wxString GetPrefix() const
void SetOverrideTextEnabled(bool aOverride)
void SetSuffix(const wxString &aSuffix)
bool operator==(const PCB_DIMENSION_BASE &aOther) const
const std::vector< std::shared_ptr< SHAPE > > & GetShapes() const
DIM_UNITS_MODE GetUnitsMode() const
void drawAnArrow(VECTOR2I aStartPoint, EDA_ANGLE anAngle, int aLength)
Draws an arrow and updates the shape container.
void SetTextPositionMode(DIM_TEXT_POSITION aMode)
virtual void updateText()
Update the text field value from the current geometry (called by updateGeometry normally).
bool HitTest(const VECTOR2I &aPosition, int aAccuracy) const override
Test if aPosition is inside or on the boundary of this item.
EDA_UNITS m_units
0 = inches, 1 = mm
int m_measuredValue
value of PCB dimensions
DIM_UNITS_FORMAT GetUnitsFormat() const
void SetLineThickness(int aWidth)
void SetArrowLength(int aLength)
virtual const VECTOR2I & GetStart() const
The dimension's origin is the first feature point for the dimension.
DIM_ARROW_DIRECTION GetArrowDirection() const
wxString m_valueString
Displayed value when m_overrideValue = true.
void ChangePrecision(DIM_PRECISION aPrecision)
virtual void SetEnd(const VECTOR2I &aPoint)
void SetPrecision(DIM_PRECISION aPrecision)
bool m_overrideTextEnabled
Manually specify the displayed measurement value.
virtual void SetStart(const VECTOR2I &aPoint)
DIM_UNITS_FORMAT m_unitsFormat
How to render the units suffix.
int GetMeasuredValue() const
bool GetSuppressZeroes() const
void SetArrowDirection(const DIM_ARROW_DIRECTION &aDirection)
const BOX2I ViewBBox() const override
Return the bounding box of the item covering all its layers.
DIM_TEXT_POSITION m_textPosition
How to position the text.
void SetOverrideText(const wxString &aValue)
wxString GetValueText() const
void ChangePrefix(const wxString &aPrefix)
wxString m_prefix
String prepended to the value.
bool Deserialize(const google::protobuf::Any &aContainer) override
Deserializes the given protobuf message into this object.
void Rotate(const VECTOR2I &aRotCentre, const EDA_ANGLE &aAngle) override
Rotate this object.
DIM_ARROW_DIRECTION m_arrowDirection
direction of dimension arrow.
void ChangeKeepTextAligned(bool aKeepAligned)
void ChangeUnitsFormat(const DIM_UNITS_FORMAT aFormat)
void ChangeSuppressZeroes(bool aSuppress)
void GetMsgPanelInfo(EDA_DRAW_FRAME *aFrame, std::vector< MSG_PANEL_ITEM > &aList) override
Populate aList of MSG_PANEL_ITEM objects with it's internal state for display purposes.
double GetTextAngleDegreesProp() const
bool GetOverrideTextEnabled() const
void SetUnitsMode(DIM_UNITS_MODE aMode)
int m_arrowLength
Length of arrow shapes.
virtual const VECTOR2I & GetEnd() const
virtual void Mirror(const VECTOR2I &axis_pos, FLIP_DIRECTION aFlipDirection) override
Mirror the dimension relative to a given horizontal axis.
VECTOR2I m_end
Internal cache of drawn shapes.
void ChangeUnitsMode(DIM_UNITS_MODE aMode)
void SetKeepTextAligned(bool aKeepAligned)
VECTOR2I GetPosition() const override
For better understanding of the points that make a dimension:
int GetHeight() const
EDA_ITEM * Clone() const override
Create a duplicate of this item with linked list members set to NULL.
BITMAPS GetMenuImage() const override
Return a pointer to an image to be used in menus.
bool Deserialize(const google::protobuf::Any &aContainer) override
Deserializes the given protobuf message into this object.
int m_height
Perpendicular distance from features to crossbar.
void ChangeExtensionHeight(int aHeight)
void updateText() override
Update the text field value from the current geometry (called by updateGeometry normally).
void SetExtensionHeight(int aHeight)
VECTOR2I m_crossBarStart
Crossbar start control point.
void Serialize(google::protobuf::Any &aContainer) const override
Serializes this object to the given Any message.
void Mirror(const VECTOR2I &axis_pos, FLIP_DIRECTION aFlipDirection) override
Mirror the dimension relative to a given horizontal axis.
void UpdateHeight(const VECTOR2I &aCrossbarStart, const VECTOR2I &aCrossbarEnd)
Update the stored height basing on points coordinates.
virtual void swapData(BOARD_ITEM *aImage) override
int m_extensionHeight
Length of extension lines past the crossbar.
void SetHeight(int aHeight)
Set the distance from the feature points to the crossbar line.
VECTOR2I m_crossBarEnd
Crossbar end control point.
PCB_DIM_ALIGNED(BOARD_ITEM *aParent, KICAD_T aType=PCB_DIM_ALIGNED_T)
void ChangeHeight(int aHeight)
void updateGeometry() override
Update the cached geometry of the dimension after changing any of its properties.
int GetExtensionHeight() const
void GetMsgPanelInfo(EDA_DRAW_FRAME *aFrame, std::vector< MSG_PANEL_ITEM > &aList) override
Populate aList of MSG_PANEL_ITEM objects with it's internal state for display purposes.
Mark the center of a circle or arc with a cross shape.
EDA_ITEM * Clone() const override
Create a duplicate of this item with linked list members set to NULL.
const BOX2I GetBoundingBox() const override
Return the orthogonal bounding box of this object for display purposes.
void updateGeometry() override
Update the cached geometry of the dimension after changing any of its properties.
const BOX2I ViewBBox() const override
Return the bounding box of the item covering all its layers.
void Serialize(google::protobuf::Any &aContainer) const override
Serializes this object to the given Any message.
virtual void swapData(BOARD_ITEM *aImage) override
BITMAPS GetMenuImage() const override
Return a pointer to an image to be used in menus.
bool Deserialize(const google::protobuf::Any &aContainer) override
Deserializes the given protobuf message into this object.
PCB_DIM_CENTER(BOARD_ITEM *aParent)
A leader is a dimension-like object pointing to a specific point.
void GetMsgPanelInfo(EDA_DRAW_FRAME *aFrame, std::vector< MSG_PANEL_ITEM > &aList) override
Populate aList of MSG_PANEL_ITEM objects with it's internal state for display purposes.
DIM_TEXT_BORDER m_textBorder
BITMAPS GetMenuImage() const override
Return a pointer to an image to be used in menus.
DIM_TEXT_BORDER GetTextBorder() const
virtual void swapData(BOARD_ITEM *aImage) override
void updateGeometry() override
Update the cached geometry of the dimension after changing any of its properties.
void updateText() override
Update the text field value from the current geometry (called by updateGeometry normally).
void SetTextBorder(DIM_TEXT_BORDER aBorder)
EDA_ITEM * Clone() const override
Create a duplicate of this item with linked list members set to NULL.
PCB_DIM_LEADER(BOARD_ITEM *aParent)
void Serialize(google::protobuf::Any &aContainer) const override
Serializes this object to the given Any message.
void ChangeTextBorder(DIM_TEXT_BORDER aBorder)
bool Deserialize(const google::protobuf::Any &aContainer) override
Deserializes the given protobuf message into this object.
An orthogonal dimension is like an aligned dimension, but the extension lines are locked to the X or ...
void swapData(BOARD_ITEM *aImage) override
bool Deserialize(const google::protobuf::Any &aContainer) override
Deserializes the given protobuf message into this object.
void updateText() override
Update the text field value from the current geometry (called by updateGeometry normally).
void SetOrientation(DIR aOrientation)
Set the orientation of the dimension line (so, perpendicular to the feature lines).
void updateGeometry() override
Update the cached geometry of the dimension after changing any of its properties.
BITMAPS GetMenuImage() const override
Return a pointer to an image to be used in menus.
void Serialize(google::protobuf::Any &aContainer) const override
Serializes this object to the given Any message.
EDA_ITEM * Clone() const override
Create a duplicate of this item with linked list members set to NULL.
DIR m_orientation
What axis to lock the dimension line to.
void Mirror(const VECTOR2I &axis_pos, FLIP_DIRECTION aFlipDirection) override
Mirror the dimension relative to a given horizontal axis.
void Rotate(const VECTOR2I &aRotCentre, const EDA_ANGLE &aAngle) override
Rotate this object.
PCB_DIM_ORTHOGONAL(BOARD_ITEM *aParent)
A radial dimension indicates either the radius or diameter of an arc or circle.
int GetLeaderLength() const
PCB_DIM_RADIAL(BOARD_ITEM *aParent)
EDA_ITEM * Clone() const override
Create a duplicate of this item with linked list members set to NULL.
void updateText() override
Update the text field value from the current geometry (called by updateGeometry normally).
virtual void swapData(BOARD_ITEM *aImage) override
void SetLeaderLength(int aLength)
void Serialize(google::protobuf::Any &aContainer) const override
Serializes this object to the given Any message.
void updateGeometry() override
Update the cached geometry of the dimension after changing any of its properties.
void ChangeLeaderLength(int aLength)
bool Deserialize(const google::protobuf::Any &aContainer) override
Deserializes the given protobuf message into this object.
VECTOR2I GetKnee() const
BITMAPS GetMenuImage() const override
Return a pointer to an image to be used in menus.
const BOX2I ViewBBox() const override
Return the bounding box of the item covering all its layers.
Definition: pcb_text.cpp:204
wxString GetShownText(bool aAllowExtraText, int aDepth=0) const override
Return the string actually shown after processing of the base text.
Definition: pcb_text.cpp:140
bool TextHitTest(const VECTOR2I &aPoint, int aAccuracy=0) const override
Test if aPoint is within the bounds of this object.
Definition: pcb_text.cpp:353
EDA_ITEM * Clone() const override
Create a duplicate of this item with linked list members set to NULL.
Definition: pcb_text.cpp:456
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 & SetWriteableFunc(std::function< bool(INSPECTABLE *)> aFunc)
Definition: property.h:268
Provide class metadata.Helper macro to map type hashes to names.
Definition: property_mgr.h:85
void InheritsAfter(TYPE_ID aDerived, TYPE_ID aBase)
Declare an inheritance relationship between types.
void Mask(TYPE_ID aDerived, TYPE_ID aBase, const wxString &aName)
Sets a base class property as masked in a derived class.
static PROPERTY_MANAGER & Instance()
Definition: property_mgr.h:87
PROPERTY_BASE & AddProperty(PROPERTY_BASE *aProperty, const wxString &aGroup=wxEmptyString)
Register a property.
void OverrideAvailability(TYPE_ID aDerived, TYPE_ID aBase, const wxString &aName, std::function< bool(INSPECTABLE *)> aFunc)
Sets an override availability functor for a base class property of a given derived class.
void AddTypeCast(TYPE_CAST_BASE *aCast)
Register a type converter.
Definition: seg.h:42
VECTOR2I A
Definition: seg.h:49
VECTOR2I B
Definition: seg.h:50
int GetRadius() const
Definition: shape_circle.h:118
const VECTOR2I GetCenter() const
Definition: shape_circle.h:123
Base class for iterating over all segments in a given SHAPE_POLY_SET.
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.
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)
int NewOutline()
Creates a new empty polygon in the set and returns its index.
SEGMENT_ITERATOR IterateSegments(int aFirst, int aLast, bool aIterateHoles=false)
Return an iterator object, for iterating between aFirst and aLast outline, with or without holes (def...
CONST_SEGMENT_ITERATOR CIterateSegments(int aFirst, int aLast, bool aIterateHoles=false) const
Return an iterator object, for iterating between aFirst and aLast outline, with or without holes (def...
bool Contains(const VECTOR2I &aP, int aSubpolyIndex=-1, int aAccuracy=0, bool aUseBBoxCaches=false) const
Return true if a given subpolygon contains the point aP.
const SEG & GetSeg() const
wxString MessageTextFromValue(double aValue, bool aAddUnitLabel=true, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
A lower-precision version of StringFromValue().
void SetUserUnits(EDA_UNITS aUnits)
constexpr extended_type Cross(const VECTOR2< T > &aVector) const
Compute cross product of self with aVector.
Definition: vector2d.h:546
T EuclideanNorm() const
Compute the Euclidean norm of the vector, which is defined as sqrt(x ** 2 + y ** 2).
Definition: vector2d.h:283
VECTOR2< T > Resize(T aNewLength) const
Return a vector of the same direction, but length specified in aNewLength.
Definition: vector2d.h:385
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 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_90
Definition: eda_angle.h:403
@ DEGREES_T
Definition: eda_angle.h:31
static constexpr EDA_ANGLE ANGLE_VERTICAL
Definition: eda_angle.h:398
static constexpr EDA_ANGLE ANGLE_HORIZONTAL
Definition: eda_angle.h:397
static constexpr EDA_ANGLE ANGLE_45
Definition: eda_angle.h:402
static constexpr EDA_ANGLE ANGLE_270
Definition: eda_angle.h:406
static constexpr EDA_ANGLE FULL_CIRCLE
Definition: eda_angle.h:399
static constexpr EDA_ANGLE ANGLE_180
Definition: eda_angle.h:405
static constexpr EDA_ANGLE ANGLE_135
Definition: eda_angle.h:404
#define PCB_EDIT_FRAME_NAME
@ NONE
Definition: eda_shape.h:69
EDA_UNITS
Definition: eda_units.h:46
PCB_LAYER_ID FlipLayer(PCB_LAYER_ID aLayerId, int aCopperLayersCount)
Definition: layer_id.cpp:169
FLASHING
Enum used during connectivity building to ensure we do not query connectivity while building the data...
Definition: layer_ids.h:184
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ Dwgs_User
Definition: layer_ids.h:107
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 double ToUserUnit(const EDA_IU_SCALE &aIuScale, EDA_UNITS aUnit, double aValue)
Convert aValue in internal units to the appropriate user units defined by aUnit.
Definition: eda_units.cpp:261
KICOMMON_API wxString GetText(EDA_UNITS aUnits, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
Get the units string for a given units type.
Definition: eda_units.cpp:127
KICOMMON_API wxString GetLabel(EDA_UNITS aUnits, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
Get the units string for a given units type.
Definition: eda_units.cpp:156
KICOMMON_API wxString EllipsizeMenuText(const wxString &aString)
Ellipsize text (at the end) to be no more than 36 characters.
Definition: ui_common.cpp:215
KICOMMON_API wxString EllipsizeStatusText(wxWindow *aWindow, const wxString &aString)
Ellipsize text (at the end) to be no more than 1/3 of the window width.
Definition: ui_common.cpp:197
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
Definition: eda_angle.h:390
static const int INWARD_ARROW_LENGTH_TO_HEAD_RATIO
static const EDA_ANGLE s_arrowAngle(27.5, DEGREES_T)
static struct DIMENSION_DESC _DIMENSION_DESC
static OPT_VECTOR2I segPolyIntersection(const SHAPE_POLY_SET &aPoly, const SEG &aSeg, bool aStart=true)
Find the intersection between a given segment and polygon outline.
static OPT_VECTOR2I segCircleIntersection(CIRCLE &aCircle, SEG &aSeg, bool aStart=true)
static void CollectKnockedOutSegments(const SHAPE_POLY_SET &aPoly, const SEG &aSeg, std::vector< std::shared_ptr< SHAPE > > &aSegmentsAfterKnockout)
Knockout a polygon from a segment.
DIM_TEXT_POSITION
Where to place the text on a dimension.
Definition: pcb_dimension.h:62
@ OUTSIDE
Text appears outside the dimension line (default)
DIM_UNITS_FORMAT
How to display the units in a dimension's text.
Definition: pcb_dimension.h:40
DIM_UNITS_MODE
Used for storing the units selection in the file because EDA_UNITS alone doesn't cut it.
Definition: pcb_dimension.h:72
DIM_ARROW_DIRECTION
Used for dimension's arrow.
Definition: pcb_dimension.h:83
DIM_TEXT_BORDER
Frame to show around dimension text.
Definition: pcb_dimension.h:92
DIM_PRECISION
Definition: pcb_dimension.h:47
#define TYPE_HASH(x)
Definition: property.h:71
#define ENUM_TO_WXANY(type)
Macro to define read-only fields (no setter method available)
Definition: property.h:782
@ PT_SIZE
Size expressed in distance units (mm/inch)
Definition: property.h:62
#define REGISTER_TYPE(x)
Definition: property_mgr.h:371
@ INCH
Definition: rs274x.cpp:62
std::optional< VECTOR2I > OPT_VECTOR2I
Definition: seg.h:39
VECTOR2I center
int radius
VECTOR2I end
SHAPE_CIRCLE circle(c.m_circle_center, c.m_circle_radius)
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
KICAD_T
The set of class identification values stored in EDA_ITEM::m_structType.
Definition: typeinfo.h:78
@ PCB_DIM_ORTHOGONAL_T
class PCB_DIM_ORTHOGONAL, a linear dimension constrained to x/y
Definition: typeinfo.h:105
@ PCB_DIM_LEADER_T
class PCB_DIM_LEADER, a leader dimension (graphic item)
Definition: typeinfo.h:102
@ PCB_DIM_CENTER_T
class PCB_DIM_CENTER, a center point marking (graphic item)
Definition: typeinfo.h:103
@ PCB_DIM_RADIAL_T
class PCB_DIM_RADIAL, a radius or diameter dimension
Definition: typeinfo.h:104
constexpr int sign(T val)
Definition: util.h:159
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695