KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
create_3Dgraphic_brd_items.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) 2015-2016 Mario Luzeiro <mrluzeiro@ua.pt>
5 * Copyright (C) 2023 CERN
6 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version 2
11 * of the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, you may find one here:
20 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
21 * or you may search the http://www.gnu.org website for the version 2 license,
22 * or you may write to the Free Software Foundation, Inc.,
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
24 */
25
32#include "../3d_rendering/raytracing/shapes2D/ring_2d.h"
33#include "../3d_rendering/raytracing/shapes2D/filled_circle_2d.h"
34#include "../3d_rendering/raytracing/shapes2D/round_segment_2d.h"
35#include "../3d_rendering/raytracing/shapes2D/triangle_2d.h"
36#include <board_adapter.h>
37#include <board.h>
38#include <footprint.h>
39#include <pad.h>
40#include <pcb_text.h>
41#include <pcb_textbox.h>
42#include <pcb_table.h>
44#include <pcb_painter.h> // for PCB_RENDER_SETTINGS
45#include <zone.h>
47#include <trigo.h>
51#include <geometry/shape_rect.h>
53#include <utility>
54#include <vector>
55#include <wx/log.h>
56#include <macros.h>
57#include <callback_gal.h>
58
59
60#define TO_3DU( x ) ( ( x ) * m_biuTo3Dunits )
61
62#define TO_SFVEC2F( vec ) SFVEC2F( TO_3DU( vec.x ), TO_3DU( -vec.y ) )
63
64
65void addFILLED_CIRCLE_2D( CONTAINER_2D_BASE* aContainer, const SFVEC2F& aCenter, float aRadius,
66 const BOARD_ITEM& aBoardItem )
67{
68 if( aRadius > 0.0f )
69 aContainer->Add( new FILLED_CIRCLE_2D( aCenter, aRadius, aBoardItem ) );
70}
71
72
73void addRING_2D( CONTAINER_2D_BASE* aContainer, const SFVEC2F& aCenter, float aInnerRadius,
74 float aOuterRadius, const BOARD_ITEM& aBoardItem )
75{
76 if( aOuterRadius > aInnerRadius && aInnerRadius > 0.0f )
77 aContainer->Add( new RING_2D( aCenter, aInnerRadius, aOuterRadius, aBoardItem ) );
78}
79
80
81void addROUND_SEGMENT_2D( CONTAINER_2D_BASE* aContainer, const SFVEC2F& aStart, const SFVEC2F& aEnd,
82 float aWidth, const BOARD_ITEM& aBoardItem )
83{
84 if( Is_segment_a_circle( aStart, aEnd ) )
85 {
86 // Cannot add segments that have the same start and end point
87 addFILLED_CIRCLE_2D( aContainer, aStart, aWidth / 2, aBoardItem );
88 return;
89 }
90
91 if( aWidth > 0.0f )
92 aContainer->Add( new ROUND_SEGMENT_2D( aStart, aEnd, aWidth, aBoardItem ) );
93}
94
95
96void BOARD_ADAPTER::addText( const EDA_TEXT* aText, CONTAINER_2D_BASE* aContainer,
97 const BOARD_ITEM* aOwner )
98{
99 int maxError = m_board->GetDesignSettings().m_MaxError;
101 TEXT_ATTRIBUTES attrs = aText->GetAttributes();
102 float penWidth_3DU = TO_3DU( aText->GetEffectiveTextPenWidth() );
103 KIFONT::FONT* font = aText->GetFont();
104 wxString shownText = aText->GetShownText( true );
105
106 if( !font )
107 font = KIFONT::FONT::GetFont( wxEmptyString, aText->IsBold(), aText->IsItalic() );
108
109 if( aOwner && aOwner->IsKnockout() )
110 {
111 SHAPE_POLY_SET finalPoly;
112 const PCB_TEXT* pcbText = static_cast<const PCB_TEXT*>( aOwner );
113
114 pcbText->TransformTextToPolySet( finalPoly, 0, maxError, ERROR_INSIDE );
115
116 // Do not call finalPoly.Fracture() here: ConvertPolygonToTriangles() call it
117 // if needed, and Fracture() called twice can create bad results and is useless
118 ConvertPolygonToTriangles( finalPoly, *aContainer, m_biuTo3Dunits, *aOwner );
119 }
120 else
121 {
122 CALLBACK_GAL callback_gal( empty_opts,
123 // Stroke callback
124 [&]( const VECTOR2I& aPt1, const VECTOR2I& aPt2 )
125 {
126 addROUND_SEGMENT_2D( aContainer, TO_SFVEC2F( aPt1 ), TO_SFVEC2F( aPt2 ),
127 penWidth_3DU, *aOwner );
128 },
129 // Triangulation callback
130 [&]( const VECTOR2I& aPt1, const VECTOR2I& aPt2, const VECTOR2I& aPt3 )
131 {
132 aContainer->Add( new TRIANGLE_2D( TO_SFVEC2F( aPt1 ), TO_SFVEC2F( aPt2 ),
133 TO_SFVEC2F( aPt3 ), *aOwner ) );
134 } );
135
136 attrs.m_Angle = aText->GetDrawRotation();
137
138 if( auto* cache = aText->GetRenderCache( font, shownText ) )
139 {
140 callback_gal.DrawGlyphs( *cache );
141 }
142 else
143 {
144 font->Draw( &callback_gal, shownText, aText->GetDrawPos(), attrs,
145 aOwner->GetFontMetrics() );
146 }
147 }
148}
149
150
152 const BOARD_ITEM* aOwner )
153{
154 addText( aDimension, aContainer, aDimension );
155
156 const int linewidth = aDimension->GetLineThickness();
157
158 for( const std::shared_ptr<SHAPE>& shape : aDimension->GetShapes() )
159 {
160 switch( shape->Type() )
161 {
162 case SH_SEGMENT:
163 {
164 const SEG& seg = static_cast<const SHAPE_SEGMENT*>( shape.get() )->GetSeg();
165
166 addROUND_SEGMENT_2D( aContainer, TO_SFVEC2F( seg.A ), TO_SFVEC2F( seg.B ),
167 TO_3DU( linewidth ), *aOwner );
168 break;
169 }
170
171 case SH_CIRCLE:
172 {
173 int radius = static_cast<const SHAPE_CIRCLE*>( shape.get() )->GetRadius();
174 float innerR3DU = TO_3DU( radius ) - TO_3DU( aDimension->GetLineThickness() ) / 2.0f;
175 float outerR3DU = TO_3DU( radius ) + TO_3DU( aDimension->GetLineThickness() ) / 2.0f;
176
177 addRING_2D( aContainer, TO_SFVEC2F( shape->Centre() ), innerR3DU, outerR3DU, *aOwner );
178
179 break;
180 }
181
182 default:
183 break;
184 }
185 }
186}
187
188
190 PCB_LAYER_ID aLayerId,
191 const std::bitset<LAYER_3D_END>& aFlags )
192{
194
195 for( PCB_FIELD* field : aFootprint->GetFields() )
196 {
197 if( !aFlags.test( LAYER_FP_TEXT ) )
198 continue;
199
200 if( field->IsReference() && !aFlags.test( LAYER_FP_REFERENCES ) )
201 continue;
202
203 if( field->IsValue() && !aFlags.test( LAYER_FP_VALUES ) )
204 continue;
205
206 if( field->GetLayer() == aLayerId && field->IsVisible() )
207 addText( field, aContainer, field );
208 }
209
210 for( BOARD_ITEM* item : aFootprint->GraphicalItems() )
211 {
212 switch( item->Type() )
213 {
214 case PCB_TEXT_T:
215 {
216 PCB_TEXT* text = static_cast<PCB_TEXT*>( item );
217
218 if( !aFlags.test( LAYER_FP_TEXT ) )
219 continue;
220
221 if( text->GetText() == wxT( "${REFERENCE}" ) && !aFlags.test( LAYER_FP_REFERENCES ) )
222 continue;
223
224 if( text->GetText() == wxT( "${VALUE}" ) && !aFlags.test( LAYER_FP_VALUES ) )
225 continue;
226
227 if( text->GetLayer() == aLayerId )
228 addText( text, aContainer, text );
229
230 break;
231 }
232
233 case PCB_TEXTBOX_T:
234 {
235 PCB_TEXTBOX* textbox = static_cast<PCB_TEXTBOX*>( item );
236
237 if( textbox->GetLayer() == aLayerId )
238 addShape( textbox, aContainer, aFootprint );
239
240 break;
241 }
242
243 case PCB_TABLE_T:
244 {
245 PCB_TABLE* table = static_cast<PCB_TABLE*>( item );
246
247 if( table->GetLayer() == aLayerId )
248 addTable( table, aContainer, aFootprint );
249
250 break;
251 }
252
254 case PCB_DIM_CENTER_T:
256 case PCB_DIM_RADIAL_T:
257 case PCB_DIM_LEADER_T:
258 {
259 PCB_DIMENSION_BASE* dimension = static_cast<PCB_DIMENSION_BASE*>( item );
260
261 if( dimension->GetLayer() == aLayerId )
262 addShape( dimension, aContainer, aFootprint );
263
264 break;
265 }
266
267 case PCB_SHAPE_T:
268 {
269 PCB_SHAPE* shape = static_cast<PCB_SHAPE*>( item );
270
271 if( shape->IsOnLayer( aLayerId ) )
272 addShape( shape, aContainer, aFootprint, aLayerId );
273
274 break;
275 }
276
277 default:
278 break;
279 }
280 }
281}
282
283
285 PCB_LAYER_ID aLayer, int aMargin )
286{
287 SFVEC2F start3DU = TO_SFVEC2F( aTrack->GetStart() );
288 SFVEC2F end3DU = TO_SFVEC2F( aTrack->GetEnd() );
289
290 switch( aTrack->Type() )
291 {
292 case PCB_VIA_T:
293 {
294 const PCB_VIA* via = static_cast<const PCB_VIA*>( aTrack );
295 float width3DU = TO_3DU( via->GetWidth( aLayer ) + aMargin * 2 );
296
297 addFILLED_CIRCLE_2D( aDstContainer, start3DU, width3DU / 2.0, *aTrack );
298 break;
299 }
300
301 case PCB_ARC_T:
302 {
303 const PCB_ARC* arc = static_cast<const PCB_ARC*>( aTrack );
304
305 if( arc->IsDegenerated() )
306 {
307 // Draw this very small arc like a track segment (a PCB_TRACE_T)
308 PCB_TRACK track( arc->GetParent() );
309 track.SetStart( arc->GetStart() );
310 track.SetEnd( arc->GetEnd() );
311 track.SetWidth( arc->GetWidth() );
312 track.SetLayer( arc->GetLayer() );
313
314 createTrackWithMargin( &track, aDstContainer, aLayer, aMargin );
315 return;
316 }
317
318 VECTOR2I center( arc->GetCenter() );
319 EDA_ANGLE arc_angle = arc->GetAngle();
320 double radius = arc->GetRadius();
321 int arcsegcount = GetArcToSegmentCount( KiROUND( radius ), ARC_HIGH_DEF, arc_angle );
322 int circlesegcount;
323
324 // Avoid arcs that cannot be drawn
325 if( radius < std::numeric_limits<double>::min() || arc_angle.IsZero() )
326 break;
327
328 // We need a circle to segment count. However, the arc angle can be small, and the
329 // radius very big. so we calculate a reasonable value for circlesegcount.
330 if( arcsegcount <= 1 ) // The arc will be approximated by a segment
331 {
332 circlesegcount = 1;
333 }
334 else
335 {
336 circlesegcount = KiROUND( arcsegcount * 360.0 / std::abs( arc_angle.AsDegrees() ) );
337 circlesegcount = std::clamp( circlesegcount, 1, 128 );
338 }
339
340 createArcSegments( center, arc->GetStart(), arc_angle, circlesegcount,
341 arc->GetWidth() + aMargin * 2, aDstContainer, *arc );
342 break;
343 }
344
345 case PCB_TRACE_T: // Track is a usual straight segment
346 {
347 float width3DU = TO_3DU( aTrack->GetWidth() + aMargin * 2 );
348 addROUND_SEGMENT_2D( aDstContainer, start3DU, end3DU, width3DU, *aTrack );
349 break;
350 }
351
352 default:
353 break;
354 }
355}
356
357
359 PCB_LAYER_ID aLayer, const VECTOR2I& aMargin ) const
360{
361 SHAPE_POLY_SET poly;
362 int maxError = GetBoard()->GetDesignSettings().m_MaxError;
363 VECTOR2I clearance = aMargin;
364
365 // Our shape-based builder can't handle negative or differing x:y clearance values (the
366 // former are common for solder paste while the later get generated when a relative paste
367 // margin is used with an oblong pad). So we apply this huge hack and fake a larger pad to
368 // run the general-purpose polygon builder on.
369 // Of course being a hack it falls down when dealing with custom shape pads (where the size
370 // is only the size of the anchor), so for those we punt and just use aMargin.x.
371
372 if( ( clearance.x < 0 || clearance.x != clearance.y ) && aPad->GetShape( aLayer ) != PAD_SHAPE::CUSTOM )
373 {
374 VECTOR2I dummySize = VECTOR2I( aPad->GetSize( aLayer ) ) + clearance + clearance;
375
376 if( dummySize.x <= 0 || dummySize.y <= 0 )
377 return;
378
379 PAD dummy( *aPad );
380 dummy.SetSize( aLayer, VECTOR2I( dummySize.x, dummySize.y ) );
381 dummy.TransformShapeToPolygon( poly, aLayer, 0, maxError, ERROR_INSIDE );
382 clearance = { 0, 0 };
383
384 // Remove group membership from dummy item before deleting
385 dummy.SetParentGroup( nullptr );
386 }
387 else if( aPad->GetShape( aLayer ) == PAD_SHAPE::CUSTOM )
388 {
389 // A custom pad can have many complex subshape items. To avoid issues, use its
390 // final polygon shape, not its basic shape set. One cannot apply the clearance
391 // to each subshape: it does no work
392 aPad->TransformShapeToPolygon( poly, aLayer, 0, maxError );
393 }
394 else
395 {
396 auto padShapes = std::static_pointer_cast<SHAPE_COMPOUND>( aPad->GetEffectiveShape( aLayer ) );
397
398 for( const SHAPE* shape : padShapes->Shapes() )
399 {
400 switch( shape->Type() )
401 {
402 case SH_SEGMENT:
403 {
404 const SHAPE_SEGMENT* seg = static_cast<const SHAPE_SEGMENT*>( shape );
405
406 addROUND_SEGMENT_2D( aContainer,
407 TO_SFVEC2F( seg->GetSeg().A ),
408 TO_SFVEC2F( seg->GetSeg().B ),
409 TO_3DU( seg->GetWidth() + clearance.x * 2 ),
410 *aPad );
411 break;
412 }
413
414 case SH_CIRCLE:
415 {
416 const SHAPE_CIRCLE* circle = static_cast<const SHAPE_CIRCLE*>( shape );
417
418 addFILLED_CIRCLE_2D( aContainer,
421 *aPad );
422 break;
423 }
424
425 case SH_RECT:
426 {
427 const SHAPE_RECT* rect = static_cast<const SHAPE_RECT*>( shape );
428
429 poly.NewOutline();
430 poly.Append( rect->GetPosition() );
431 poly.Append( rect->GetPosition().x + rect->GetSize().x, rect->GetPosition().y );
432 poly.Append( rect->GetPosition() + rect->GetSize() );
433 poly.Append( rect->GetPosition().x, rect->GetPosition().y + rect->GetSize().y );
434 break;
435 }
436
437 case SH_SIMPLE:
438 poly.AddOutline( static_cast<const SHAPE_SIMPLE*>( shape )->Vertices() );
439 break;
440
441 case SH_POLY_SET:
442 poly.Append( *static_cast<const SHAPE_POLY_SET*>( shape ) );
443 break;
444
445 case SH_ARC:
446 {
447 const SHAPE_ARC* arc = static_cast<const SHAPE_ARC*>( shape );
448 SHAPE_LINE_CHAIN l = arc->ConvertToPolyline( maxError );
449
450 for( int i = 0; i < l.SegmentCount(); i++ )
451 {
452 SHAPE_SEGMENT seg( l.Segment( i ).A, l.Segment( i ).B, arc->GetWidth() );
453
454 addROUND_SEGMENT_2D( aContainer,
455 TO_SFVEC2F( seg.GetSeg().A ),
456 TO_SFVEC2F( seg.GetSeg().B ),
457 TO_3DU( arc->GetWidth() + clearance.x * 2 ),
458 *aPad );
459 }
460
461 break;
462 }
463
464 default:
465 UNIMPLEMENTED_FOR( SHAPE_TYPE_asString( shape->Type() ) );
466 break;
467 }
468 }
469 }
470
471 if( !poly.IsEmpty() )
472 {
473 if( clearance.x )
474 poly.Inflate( clearance.x, CORNER_STRATEGY::ROUND_ALL_CORNERS, maxError );
475
476 // Add the PAD polygon
477 ConvertPolygonToTriangles( poly, *aContainer, m_biuTo3Dunits, *aPad );
478 }
479}
480
481
483 int aInflateValue )
484{
485 if( !aPad->HasHole() )
486 {
487 wxLogTrace( m_logTrace, wxT( "BOARD_ADAPTER::createPadWithHole - found an invalid pad" ) );
488 return;
489 }
490
491 std::shared_ptr<SHAPE_SEGMENT> slot = aPad->GetEffectiveHoleShape();
492
493 addROUND_SEGMENT_2D( aDstContainer,
494 TO_SFVEC2F( slot->GetSeg().A ),
495 TO_SFVEC2F( slot->GetSeg().B ),
496 TO_3DU( slot->GetWidth() + aInflateValue * 2 ),
497 *aPad );
498}
499
500
501void BOARD_ADAPTER::addPads( const FOOTPRINT* aFootprint, CONTAINER_2D_BASE* aContainer,
502 PCB_LAYER_ID aLayerId )
503{
504 for( PAD* pad : aFootprint->Pads() )
505 {
506 if( !pad->IsOnLayer( aLayerId ) )
507 continue;
508
509 if( IsCopperLayer( aLayerId ) )
510 {
511 // Skip pad annulus when there isn't one (note: this is more discerning than
512 // pad->IsOnLayer(), which doesn't check for NPTH pads with holes that consume
513 // the entire pad).
514 if( !pad->IsOnCopperLayer() )
515 continue;
516
517 // Skip pad annulus when not connected on this layer (if removing is enabled)
518 if( !pad->FlashLayer( aLayerId ) )
519 continue;
520 }
521
522 VECTOR2I margin( 0, 0 );
523
524 switch( aLayerId )
525 {
526 case F_Mask:
527 case B_Mask:
528 margin.x += pad->GetSolderMaskExpansion( aLayerId );
529 margin.y += pad->GetSolderMaskExpansion( aLayerId );
530 break;
531
532 case F_Paste:
533 case B_Paste:
534 margin += pad->GetSolderPasteMargin( aLayerId );
535 break;
536
537 default:
538 break;
539 }
540
541 createPadWithMargin( pad, aContainer, aLayerId, margin );
542 }
543}
544
545
546// based on TransformArcToPolygon function from
547// common/convert_basic_shapes_to_polygon.cpp
548void BOARD_ADAPTER::createArcSegments( const VECTOR2I& aCentre, const VECTOR2I& aStart,
549 const EDA_ANGLE& aArcAngle, int aCircleToSegmentsCount,
550 int aWidth, CONTAINER_2D_BASE* aContainer,
551 const BOARD_ITEM& aOwner )
552{
553 // Don't attempt to render degenerate shapes
554 if( aWidth == 0 )
555 return;
556
557 VECTOR2I arc_start, arc_end;
558 EDA_ANGLE arcAngle( aArcAngle );
559 EDA_ANGLE delta = ANGLE_360 / aCircleToSegmentsCount; // rotate angle
560
561 arc_end = arc_start = aStart;
562
563 if( arcAngle != ANGLE_360 )
564 RotatePoint( arc_end, aCentre, -arcAngle );
565
566 if( arcAngle < ANGLE_0 )
567 {
568 std::swap( arc_start, arc_end );
569 arcAngle = -arcAngle;
570 }
571
572 // Compute the ends of segments and creates poly
573 VECTOR2I curr_end = arc_start;
574 VECTOR2I curr_start = arc_start;
575
576 for( EDA_ANGLE ii = delta; ii < arcAngle; ii += delta )
577 {
578 curr_end = arc_start;
579 RotatePoint( curr_end, aCentre, -ii );
580
581 addROUND_SEGMENT_2D( aContainer, TO_SFVEC2F( curr_start ), TO_SFVEC2F( curr_end ),
582 TO_3DU( aWidth ), aOwner );
583
584 curr_start = curr_end;
585 }
586
587 if( curr_end != arc_end )
588 {
589 addROUND_SEGMENT_2D( aContainer, TO_SFVEC2F( curr_end ), TO_SFVEC2F( arc_end ),
590 TO_3DU( aWidth ), aOwner );
591 }
592}
593
594
595void BOARD_ADAPTER::addShape( const PCB_SHAPE* aShape, CONTAINER_2D_BASE* aContainer,
596 const BOARD_ITEM* aOwner, PCB_LAYER_ID aLayer )
597{
598 // The full width of the lines to create
599 int linewidth = aShape->GetWidth();
600 int margin = 0;
601
602 if( IsSolderMaskLayer( aLayer )
603 && aShape->HasSolderMask()
604 && IsExternalCopperLayer( aShape->GetLayer() ) )
605 {
606 margin = aShape->GetSolderMaskExpansion();
607 linewidth += margin * 2;
608 }
609
610 float linewidth3DU = TO_3DU( linewidth );
611 LINE_STYLE lineStyle = aShape->GetStroke().GetLineStyle();
612
613 if( lineStyle <= LINE_STYLE::FIRST_TYPE )
614 {
615 switch( aShape->GetShape() )
616 {
617 case SHAPE_T::CIRCLE:
618 {
619 SFVEC2F center3DU = TO_SFVEC2F( aShape->GetCenter() );
620 float innerR3DU = TO_3DU( aShape->GetRadius() ) - linewidth3DU / 2.0;
621 float outerR3DU = TO_3DU( aShape->GetRadius() ) + linewidth3DU / 2.0;
622
623 if( aShape->IsSolidFill() || innerR3DU <= 0.0 )
624 addFILLED_CIRCLE_2D( aContainer, center3DU, outerR3DU, *aOwner );
625 else
626 addRING_2D( aContainer, center3DU, innerR3DU, outerR3DU, *aOwner );
627
628 break;
629 }
630
631 case SHAPE_T::RECTANGLE:
632 if( aShape->IsSolidFill() )
633 {
634 SHAPE_POLY_SET polyList;
635
637
638 polyList.Simplify();
639
640 if( margin != 0 )
641 {
642 polyList.Inflate( margin, CORNER_STRATEGY::ROUND_ALL_CORNERS,
643 GetBoard()->GetDesignSettings().m_MaxError );
644 }
645
646 ConvertPolygonToTriangles( polyList, *aContainer, m_biuTo3Dunits, *aOwner );
647 }
648 else
649 {
650 std::vector<VECTOR2I> pts = aShape->GetRectCorners();
651
652 addROUND_SEGMENT_2D( aContainer, TO_SFVEC2F( pts[0] ), TO_SFVEC2F( pts[1] ),
653 linewidth3DU, *aOwner );
654 addROUND_SEGMENT_2D( aContainer, TO_SFVEC2F( pts[1] ), TO_SFVEC2F( pts[2] ),
655 linewidth3DU, *aOwner );
656 addROUND_SEGMENT_2D( aContainer, TO_SFVEC2F( pts[2] ), TO_SFVEC2F( pts[3] ),
657 linewidth3DU, *aOwner );
658 addROUND_SEGMENT_2D( aContainer, TO_SFVEC2F( pts[3] ), TO_SFVEC2F( pts[0] ),
659 linewidth3DU, *aOwner );
660 }
661 break;
662
663 case SHAPE_T::ARC:
664 {
665 unsigned int segCount = GetCircleSegmentCount( aShape->GetBoundingBox().GetSizeMax() );
666
667 createArcSegments( aShape->GetCenter(), aShape->GetStart(), aShape->GetArcAngle(),
668 segCount, linewidth, aContainer, *aOwner );
669 break;
670 }
671
672 case SHAPE_T::SEGMENT:
673 addROUND_SEGMENT_2D( aContainer, TO_SFVEC2F( aShape->GetStart() ),
674 TO_SFVEC2F( aShape->GetEnd() ), linewidth3DU, *aOwner );
675 break;
676
677 case SHAPE_T::BEZIER:
678 case SHAPE_T::POLY:
679 {
680 SHAPE_POLY_SET polyList;
681
683
684 // Some polygons can be a bit complex (especially when coming from a
685 // picture of a text converted to a polygon
686 // So call Simplify before calling ConvertPolygonToTriangles, just in case.
687 polyList.Simplify();
688
689 if( polyList.IsEmpty() ) // Just for caution
690 break;
691
692 if( margin != 0 )
693 {
694 CORNER_STRATEGY cornerStr = margin >= 0 ? CORNER_STRATEGY::ROUND_ALL_CORNERS
695 : CORNER_STRATEGY::ALLOW_ACUTE_CORNERS;
696
697 polyList.Inflate( margin, cornerStr, GetBoard()->GetDesignSettings().m_MaxError );
698 }
699
700 ConvertPolygonToTriangles( polyList, *aContainer, m_biuTo3Dunits, *aOwner );
701 break;
702 }
703
704 default:
705 wxFAIL_MSG( wxT( "BOARD_ADAPTER::addShape no implementation for " )
706 + aShape->SHAPE_T_asString() );
707 break;
708 }
709 }
710 else
711 {
712 std::vector<SHAPE*> shapes = aShape->MakeEffectiveShapes( true );
713 SFVEC2F a3DU;
714 SFVEC2F b3DU;
715
716 const PCB_PLOT_PARAMS& plotParams = aShape->GetBoard()->GetPlotOptions();
717 KIGFX::PCB_RENDER_SETTINGS renderSettings;
718
719 renderSettings.SetDashLengthRatio( plotParams.GetDashedLineDashRatio() );
720 renderSettings.SetGapLengthRatio( plotParams.GetDashedLineGapRatio() );
721
722 for( SHAPE* shape : shapes )
723 {
724 STROKE_PARAMS::Stroke( shape, lineStyle, aShape->GetWidth(), &renderSettings,
725 [&]( const VECTOR2I& a, const VECTOR2I& b )
726 {
727 addROUND_SEGMENT_2D( aContainer, TO_SFVEC2F( a ), TO_SFVEC2F( b ),
728 linewidth3DU, *aOwner );
729 } );
730 }
731
732 for( SHAPE* shape : shapes )
733 delete shape;
734 }
735
736 if( aShape->IsHatchedFill() )
737 ConvertPolygonToTriangles( aShape->GetHatching(), *aContainer, m_biuTo3Dunits, *aOwner );
738}
739
740
741void BOARD_ADAPTER::addShape( const PCB_TEXTBOX* aTextBox, CONTAINER_2D_BASE* aContainer,
742 const BOARD_ITEM* aOwner )
743{
744 addText( aTextBox, aContainer, aOwner );
745
746 if( !aTextBox->IsBorderEnabled() )
747 return;
748
749 // We cannot use PCB_TEXTBOX::TransformShapeToPolygon because it convert the textbox
750 // as filled polygon even if there's no background colour.
751 // So for polygon, we use PCB_SHAPE::TransformShapeToPolygon
752
753 if( aTextBox->GetShape() == SHAPE_T::RECTANGLE )
754 {
755 addShape( static_cast<const PCB_SHAPE*>( aTextBox ), aContainer, aOwner, UNDEFINED_LAYER );
756 }
757 else
758 {
759 SHAPE_POLY_SET polyList;
760
761 aTextBox->PCB_SHAPE::TransformShapeToPolygon( polyList, UNDEFINED_LAYER, 0, ARC_HIGH_DEF,
762 ERROR_INSIDE );
763
764 ConvertPolygonToTriangles( polyList, *aContainer, m_biuTo3Dunits, *aOwner );
765 }
766}
767
768
769void BOARD_ADAPTER::addTable( const PCB_TABLE* aTable, CONTAINER_2D_BASE* aContainer,
770 const BOARD_ITEM* aOwner )
771{
772 aTable->DrawBorders(
773 [&]( const VECTOR2I& ptA, const VECTOR2I& ptB, const STROKE_PARAMS& stroke )
774 {
775 addROUND_SEGMENT_2D( aContainer, TO_SFVEC2F( ptA ), TO_SFVEC2F( ptB ),
776 TO_3DU( stroke.GetWidth() ), *aOwner );
777 } );
778
779 for( PCB_TABLECELL* cell : aTable->GetCells() )
780 {
781 if( cell->GetColSpan() > 0 && cell->GetRowSpan() > 0 )
782 addText( cell, aContainer, aOwner );
783 }
784}
785
786
788 PCB_LAYER_ID aLayerId )
789{
790 // This convert the poly in outline and holes
791 ConvertPolygonToTriangles( *aZone->GetFilledPolysList( aLayerId ), *aContainer, m_biuTo3Dunits, *aZone );
792}
793
794
796 CONTAINER_2D_BASE* aContainer, int aWidth )
797{
798 if( aPad->GetShape( aLayer ) == PAD_SHAPE::CIRCLE ) // Draw a ring
799 {
800 const SFVEC2F center3DU = TO_SFVEC2F( aPad->ShapePos( aLayer ) );
801 const int radius = aPad->GetSize( aLayer ).x / 2;
802 const float inner_radius3DU = TO_3DU( radius - aWidth / 2.0 );
803 const float outer_radius3DU = TO_3DU( radius + aWidth / 2.0 );
804
805 addRING_2D( aContainer, center3DU, inner_radius3DU, outer_radius3DU, *aPad );
806 }
807 else
808 {
809 // For other shapes, add outlines as thick segments in polygon buffer
810 const std::shared_ptr<SHAPE_POLY_SET>& corners = aPad->GetEffectivePolygon( aLayer, ERROR_INSIDE );
811 const SHAPE_LINE_CHAIN& path = corners->COutline( 0 );
812
813 for( int j = 0; j < path.PointCount(); j++ )
814 {
815 SFVEC2F start3DU = TO_SFVEC2F( path.CPoint( j ) );
816 SFVEC2F end3DU = TO_SFVEC2F( path.CPoint( j + 1 ) );
817
818 addROUND_SEGMENT_2D( aContainer, start3DU, end3DU, TO_3DU( aWidth ), *aPad );
819 }
820 }
821}
@ ERROR_INSIDE
Definition: approximation.h:34
constexpr int ARC_HIGH_DEF
Definition: base_units.h:120
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
void addPads(const FOOTPRINT *aFootprint, CONTAINER_2D_BASE *aDstContainer, PCB_LAYER_ID aLayerId)
void createTrackWithMargin(const PCB_TRACK *aTrack, CONTAINER_2D_BASE *aDstContainer, PCB_LAYER_ID aLayer, int aMargin=0)
void addSolidAreasShapes(const ZONE *aZone, CONTAINER_2D_BASE *aDstContainer, PCB_LAYER_ID aLayerId)
void addFootprintShapes(const FOOTPRINT *aFootprint, CONTAINER_2D_BASE *aDstContainer, PCB_LAYER_ID aLayerId, const std::bitset< LAYER_3D_END > &aVisibilityFlags)
void createArcSegments(const VECTOR2I &aCentre, const VECTOR2I &aStart, const EDA_ANGLE &aArcAngle, int aCircleToSegmentsCount, int aWidth, CONTAINER_2D_BASE *aContainer, const BOARD_ITEM &aOwner)
void createPadWithMargin(const PAD *aPad, CONTAINER_2D_BASE *aDstContainer, PCB_LAYER_ID aLayer, const VECTOR2I &aMargin) const
const BOARD * GetBoard() const noexcept
void createPadWithHole(const PAD *aPad, CONTAINER_2D_BASE *aDstContainer, int aInflateValue)
void buildPadOutlineAsSegments(const PAD *aPad, PCB_LAYER_ID aLayer, CONTAINER_2D_BASE *aDstContainer, int aWidth)
void addTable(const PCB_TABLE *aTable, CONTAINER_2D_BASE *aContainer, const BOARD_ITEM *aOwner)
unsigned int GetCircleSegmentCount(float aDiameter3DU) const
void addShape(const PCB_SHAPE *aShape, CONTAINER_2D_BASE *aContainer, const BOARD_ITEM *aOwner, PCB_LAYER_ID aLayer)
double m_biuTo3Dunits
Scale factor to convert board internal units to 3D units normalized between -1.0 and 1....
void addText(const EDA_TEXT *aText, CONTAINER_2D_BASE *aDstContainer, const BOARD_ITEM *aOwner)
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 bool IsKnockout() const
Definition: board_item.h:316
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
const KIFONT::METRICS & GetFontMetrics() const
Definition: board_item.cpp:101
BOARD_ITEM_CONTAINER * GetParent() const
Definition: board_item.h:207
const PCB_PLOT_PARAMS & GetPlotOptions() const
Definition: board.h:718
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Definition: board.cpp:946
constexpr int GetSizeMax() const
Definition: box2.h:235
void Add(OBJECT_2D *aObject)
Definition: container_2d.h:49
double AsDegrees() const
Definition: eda_angle.h:113
bool IsZero() const
Definition: eda_angle.h:133
KICAD_T Type() const
Returns the type of object.
Definition: eda_item.h:108
EDA_ANGLE GetArcAngle() const
Definition: eda_shape.cpp:1033
const SHAPE_POLY_SET & GetHatching() const
Definition: eda_shape.cpp:569
virtual std::vector< SHAPE * > MakeEffectiveShapes(bool aEdgeOnly=false) const
Make a set of SHAPE objects representing the EDA_SHAPE.
Definition: eda_shape.h:379
int GetRadius() const
Definition: eda_shape.cpp:961
SHAPE_T GetShape() const
Definition: eda_shape.h:168
bool IsHatchedFill() const
Definition: eda_shape.h:124
bool IsSolidFill() const
Definition: eda_shape.h:117
const VECTOR2I & GetEnd() const
Return the ending point of the graphic.
Definition: eda_shape.h:215
const VECTOR2I & GetStart() const
Return the starting point of the graphic.
Definition: eda_shape.h:173
std::vector< VECTOR2I > GetRectCorners() const
Definition: eda_shape.cpp:1544
wxString SHAPE_T_asString() const
Definition: eda_shape.cpp:340
A mix-in class (via multiple inheritance) that handles texts such as labels, parts,...
Definition: eda_text.h:80
bool IsItalic() const
Definition: eda_text.h:156
KIFONT::FONT * GetFont() const
Definition: eda_text.h:234
std::vector< std::unique_ptr< KIFONT::GLYPH > > * GetRenderCache(const KIFONT::FONT *aFont, const wxString &forResolvedText, const VECTOR2I &aOffset={ 0, 0 }) const
Definition: eda_text.cpp:664
virtual EDA_ANGLE GetDrawRotation() const
Definition: eda_text.h:365
virtual VECTOR2I GetDrawPos() const
Definition: eda_text.h:366
const TEXT_ATTRIBUTES & GetAttributes() const
Definition: eda_text.h:218
int GetEffectiveTextPenWidth(int aDefaultPenWidth=0) const
The EffectiveTextPenWidth uses the text thickness if > 1 or aDefaultPenWidth.
Definition: eda_text.cpp:458
bool IsBold() const
Definition: eda_text.h:171
virtual wxString GetShownText(bool aAllowExtraText, int aDepth=0) const
Return the string actually shown after processing of the base text.
Definition: eda_text.h:109
std::deque< PAD * > & Pads()
Definition: footprint.h:211
void GetFields(std::vector< PCB_FIELD * > &aVector, bool aVisibleOnly) const
Populate a std::vector with PCB_TEXTs.
Definition: footprint.cpp:627
DRAWINGS & GraphicalItems()
Definition: footprint.h:214
FONT is an abstract base class for both outline and stroke fonts.
Definition: font.h:131
static FONT * GetFont(const wxString &aFontName=wxEmptyString, bool aBold=false, bool aItalic=false, const std::vector< wxString > *aEmbeddedFiles=nullptr, bool aForDrawingSheet=false)
Definition: font.cpp:147
void Draw(KIGFX::GAL *aGal, const wxString &aText, const VECTOR2I &aPosition, const VECTOR2I &aCursor, const TEXT_ATTRIBUTES &aAttributes, const METRICS &aFontMetrics) const
Draw a string.
Definition: font.cpp:250
PCB specific render settings.
Definition: pcb_painter.h:79
void SetGapLengthRatio(double aRatio)
void SetDashLengthRatio(double aRatio)
Definition: pad.h:54
virtual std::shared_ptr< SHAPE > GetEffectiveShape(PCB_LAYER_ID aLayer, FLASHING flashPTHPads=FLASHING::DEFAULT) const override
Some pad shapes can be complex (rounded/chamfered rectangle), even without considering custom shapes.
Definition: pad.cpp:525
PAD_SHAPE GetShape(PCB_LAYER_ID aLayer) const
Definition: pad.h:195
void TransformShapeToPolygon(SHAPE_POLY_SET &aBuffer, PCB_LAYER_ID aLayer, int aClearance, int aMaxError, ERROR_LOC aErrorLoc=ERROR_INSIDE, bool ignoreLineWidth=false) const override
Convert the pad shape to a closed polygon.
Definition: pad.cpp:1957
const std::shared_ptr< SHAPE_POLY_SET > & GetEffectivePolygon(PCB_LAYER_ID aLayer, ERROR_LOC aErrorLoc=ERROR_INSIDE) const
Definition: pad.cpp:513
bool HasHole() const override
Definition: pad.h:106
VECTOR2I ShapePos(PCB_LAYER_ID aLayer) const
Definition: pad.cpp:1043
std::shared_ptr< SHAPE_SEGMENT > GetEffectiveHoleShape() const override
Return a SHAPE_SEGMENT object representing the pad's hole.
Definition: pad.cpp:582
const VECTOR2I & GetSize(PCB_LAYER_ID aLayer) const
Definition: pad.h:264
bool IsDegenerated(int aThreshold=5) const
Definition: pcb_track.cpp:2068
double GetRadius() const
Definition: pcb_track.cpp:2033
EDA_ANGLE GetAngle() const
Definition: pcb_track.cpp:2040
virtual VECTOR2I GetCenter() const override
This defaults to the center of the bounding box if not overridden.
Definition: pcb_track.h:344
Abstract dimension API.
int GetLineThickness() const
const std::vector< std::shared_ptr< SHAPE > > & GetShapes() const
Parameters and options when plotting/printing a board.
double GetDashedLineGapRatio() const
double GetDashedLineDashRatio() const
const BOX2I GetBoundingBox() const override
Return the orthogonal bounding box of this object for display purposes.
Definition: pcb_shape.h:121
VECTOR2I GetCenter() const override
This defaults to the center of the bounding box if not overridden.
Definition: pcb_shape.h:81
int GetWidth() const override
Definition: pcb_shape.cpp:375
bool HasSolderMask() const
Definition: pcb_shape.h:188
int GetSolderMaskExpansion() const
Definition: pcb_shape.cpp:188
void TransformShapeToPolySet(SHAPE_POLY_SET &aBuffer, PCB_LAYER_ID aLayer, int aClearance, int aError, ERROR_LOC aErrorLoc) const override
Convert the item shape to a polyset.
Definition: pcb_shape.cpp:854
void TransformShapeToPolygon(SHAPE_POLY_SET &aBuffer, PCB_LAYER_ID aLayer, int aClearance, int aError, ERROR_LOC aErrorLoc, bool ignoreLineWidth=false) const override
Convert the shape to a closed polygon.
Definition: pcb_shape.cpp:845
STROKE_PARAMS GetStroke() const override
Definition: pcb_shape.h:91
bool IsOnLayer(PCB_LAYER_ID aLayer) const override
Test to see if this object is on the given layer.
Definition: pcb_shape.cpp:209
PCB_LAYER_ID GetLayer() const override
Return the primary layer this item is on.
Definition: pcb_shape.h:71
std::vector< PCB_TABLECELL * > GetCells() const
Definition: pcb_table.h:150
void DrawBorders(const std::function< void(const VECTOR2I &aPt1, const VECTOR2I &aPt2, const STROKE_PARAMS &aStroke)> &aCallback) const
Definition: pcb_table.cpp:239
bool IsBorderEnabled() const
Disables the border, this is done by changing the stroke internally.
void TransformTextToPolySet(SHAPE_POLY_SET &aBuffer, int aClearance, int aMaxError, ERROR_LOC aErrorLoc) const
Function TransformTextToPolySet Convert the text to a polygonSet describing the actual character stro...
Definition: pcb_text.cpp:514
void SetEnd(const VECTOR2I &aEnd)
Definition: pcb_track.h:148
void SetStart(const VECTOR2I &aStart)
Definition: pcb_track.h:151
const VECTOR2I & GetStart() const
Definition: pcb_track.h:152
const VECTOR2I & GetEnd() const
Definition: pcb_track.h:149
virtual void SetWidth(int aWidth)
Definition: pcb_track.h:145
virtual int GetWidth() const
Definition: pcb_track.h:146
Definition: seg.h:42
VECTOR2I A
Definition: seg.h:49
VECTOR2I B
Definition: seg.h:50
int GetWidth() const
Definition: shape_arc.h:210
const SHAPE_LINE_CHAIN ConvertToPolyline(double aAccuracy=DefaultAccuracyForPCB(), double *aEffectiveAccuracy=nullptr) const
Construct a SHAPE_LINE_CHAIN of segments from a given arc.
Definition: shape_arc.cpp:886
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...
SEG Segment(int aIndex) const
Return a copy of the aIndex-th segment in the line chain.
int SegmentCount() const
Return the number of segments in this line chain.
Represent a set of closed polygons.
int AddOutline(const SHAPE_LINE_CHAIN &aOutline)
Adds a new outline to the set and returns its index.
bool IsEmpty() const
Return true if the set is empty (no polygons at all)
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)
void Simplify()
Simplify the polyset (merges overlapping polys, eliminates degeneracy/self-intersections)
int NewOutline()
Creates a new empty polygon in the set and returns its index.
const VECTOR2I & GetPosition() const
Definition: shape_rect.h:160
const VECTOR2I GetSize() const
Definition: shape_rect.h:168
const SEG & GetSeg() const
int GetWidth() const
Represent a simple polygon consisting of a zero-thickness closed chain of connected line segments.
Definition: shape_simple.h:42
const SHAPE_LINE_CHAIN & Vertices() const
Return the list of vertices defining this simple polygon.
Definition: shape_simple.h:124
An abstract shape on 2D plane.
Definition: shape.h:126
Simple container to manage line stroke parameters.
Definition: stroke_params.h:94
int GetWidth() const
LINE_STYLE GetLineStyle() const
static void Stroke(const SHAPE *aShape, LINE_STYLE aLineStyle, int aWidth, const KIGFX::RENDER_SETTINGS *aRenderSettings, const std::function< void(const VECTOR2I &a, const VECTOR2I &b)> &aStroker)
Handle a list of polygons defining a copper zone.
Definition: zone.h:74
const std::shared_ptr< SHAPE_POLY_SET > & GetFilledPolysList(PCB_LAYER_ID aLayer) const
Definition: zone.h:648
CORNER_STRATEGY
define how inflate transform build inflated polygon
#define TO_3DU(x)
void addFILLED_CIRCLE_2D(CONTAINER_2D_BASE *aContainer, const SFVEC2F &aCenter, float aRadius, const BOARD_ITEM &aBoardItem)
#define TO_SFVEC2F(vec)
void addROUND_SEGMENT_2D(CONTAINER_2D_BASE *aContainer, const SFVEC2F &aStart, const SFVEC2F &aEnd, float aWidth, const BOARD_ITEM &aBoardItem)
void addRING_2D(CONTAINER_2D_BASE *aContainer, const SFVEC2F &aCenter, float aInnerRadius, float aOuterRadius, const BOARD_ITEM &aBoardItem)
static constexpr EDA_ANGLE ANGLE_0
Definition: eda_angle.h:401
static constexpr EDA_ANGLE ANGLE_360
Definition: eda_angle.h:407
a few functions useful in geometry calculations.
int GetArcToSegmentCount(int aRadius, int aErrorMax, const EDA_ANGLE &aArcAngle)
static const wxChar * m_logTrace
Trace mask used to enable or disable debug output for this class.
bool IsSolderMaskLayer(int aLayer)
Definition: layer_ids.h:733
bool IsCopperLayer(int aLayerId)
Test whether a layer is a copper layer.
Definition: layer_ids.h:663
@ LAYER_FP_REFERENCES
Show footprints references (when texts are visible).
Definition: layer_ids.h:265
@ LAYER_FP_TEXT
Definition: layer_ids.h:239
@ LAYER_FP_VALUES
Show footprints values (when texts are visible).
Definition: layer_ids.h:262
bool IsExternalCopperLayer(int aLayerId)
Test whether a layer is an external (F_Cu or B_Cu) copper layer.
Definition: layer_ids.h:674
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ F_Paste
Definition: layer_ids.h:104
@ B_Mask
Definition: layer_ids.h:98
@ F_Mask
Definition: layer_ids.h:97
@ B_Paste
Definition: layer_ids.h:105
@ UNDEFINED_LAYER
Definition: layer_ids.h:61
This file contains miscellaneous commonly used macros and functions.
#define UNIMPLEMENTED_FOR(type)
Definition: macros.h:96
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
Definition: eda_angle.h:390
bool Is_segment_a_circle(const SFVEC2F &aStart, const SFVEC2F &aEnd)
Check if segment start and end is very close to each other.
@ 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_SEGMENT
line segment
Definition: shape.h:48
@ SH_ARC
circular arc
Definition: shape.h:54
static wxString SHAPE_TYPE_asString(SHAPE_TYPE a)
Definition: shape.h:59
std::vector< FAB_LAYER_COLOR > dummy
LINE_STYLE
Dashed line types.
Definition: stroke_params.h:46
VECTOR2I center
int radius
SHAPE_CIRCLE circle(c.m_circle_center, c.m_circle_radius)
int clearance
constexpr int delta
void ConvertPolygonToTriangles(const SHAPE_POLY_SET &aPolyList, CONTAINER_2D_BASE &aDstContainer, float aBiuTo3dUnitsScale, const BOARD_ITEM &aBoardItem)
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
@ PCB_SHAPE_T
class PCB_SHAPE, a segment not on copper layers
Definition: typeinfo.h:88
@ 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_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
Definition: typeinfo.h:97
@ PCB_DIM_CENTER_T
class PCB_DIM_CENTER, a center point marking (graphic item)
Definition: typeinfo.h:103
@ PCB_TEXTBOX_T
class PCB_TEXTBOX, wrapped text on a layer
Definition: typeinfo.h:93
@ PCB_TEXT_T
class PCB_TEXT, text on a layer
Definition: typeinfo.h:92
@ PCB_DIM_ALIGNED_T
class PCB_DIM_ALIGNED, a linear dimension (graphic item)
Definition: typeinfo.h:101
@ PCB_ARC_T
class PCB_ARC, an arc track segment on a copper layer
Definition: typeinfo.h:98
@ PCB_TABLE_T
class PCB_TABLE, table of PCB_TABLECELLs
Definition: typeinfo.h:94
@ PCB_TRACE_T
class PCB_TRACK, a track segment (segment on a copper layer)
Definition: typeinfo.h:96
@ PCB_DIM_RADIAL_T
class PCB_DIM_RADIAL, a radius or diameter dimension
Definition: typeinfo.h:104
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695
glm::vec2 SFVEC2F
Definition: xv3d_types.h:42