KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
pcb_io_easyedapro_parser.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) 2023 Alex Shvartzkop <dudesuchamazing@gmail.com>
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, you may find one here:
19 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
20 * or you may search the http://www.gnu.org website for the version 2 license,
21 * or you may write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 */
24
27
28#include <memory>
29
30#include <nlohmann/json.hpp>
32#include <core/map_helpers.h>
33#include <string_utils.h>
34
35#include <wx/wfstream.h>
36#include <wx/stdstream.h>
37#include <wx/log.h>
38#include <glm/glm.hpp>
39
40#include <progress_reporter.h>
41#include <footprint.h>
42#include <board.h>
44#include <bezier_curves.h>
45#include <pcb_group.h>
46#include <pcb_track.h>
47#include <pcb_shape.h>
48#include <pcb_text.h>
49#include <font/font.h>
50#include <geometry/shape_arc.h>
53#include <geometry/shape_rect.h>
55#include <zone.h>
56#include <pad.h>
58#include <project.h>
59#include <fix_board_shape.h>
60#include <pcb_reference_image.h>
61#include <core/mirror.h>
62
63
64static const wxString QUERY_MODEL_UUID_KEY = wxS( "JLC_3DModel_Q" );
65static const wxString MODEL_SIZE_KEY = wxS( "JLC_3D_Size" );
66
67static const int SHAPE_JOIN_DISTANCE = pcbIUScale.mmToIU( 1.5 );
68
69
70double PCB_IO_EASYEDAPRO_PARSER::Convert( wxString aValue )
71{
72 double value = 0;
73
74 if( !aValue.ToCDouble( &value ) )
75 THROW_IO_ERROR( wxString::Format( _( "Failed to parse value: '%s'" ), aValue ) );
76
77 return value;
78}
79
80
82{
83 m_board = aBoard;
84}
85
86
88{
89}
90
91
93{
94 switch( aLayer )
95 {
96 case 1: return F_Cu;
97 case 2: return B_Cu;
98 case 3: return F_SilkS;
99 case 4: return B_SilkS;
100 case 5: return F_Mask;
101 case 6: return B_Mask;
102 case 7: return F_Paste;
103 case 8: return B_Paste;
104 case 9: return F_Fab;
105 case 10: return B_Fab;
106 case 11: return Edge_Cuts;
107 case 12: return Edge_Cuts; // Multi
108 case 13: return Dwgs_User;
109 case 14: return Eco2_User;
110
111 case 15: return In1_Cu;
112 case 16: return In2_Cu;
113 case 17: return In3_Cu;
114 case 18: return In4_Cu;
115 case 19: return In5_Cu;
116 case 20: return In6_Cu;
117 case 21: return In7_Cu;
118 case 22: return In8_Cu;
119 case 23: return In9_Cu;
120 case 24: return In10_Cu;
121 case 25: return In11_Cu;
122 case 26: return In12_Cu;
123 case 27: return In13_Cu;
124 case 28: return In14_Cu;
125 case 29: return In15_Cu;
126 case 30: return In16_Cu;
127 case 31: return In17_Cu;
128 case 32: return In18_Cu;
129 case 33: return In19_Cu;
130 case 34: return In20_Cu;
131 case 35: return In21_Cu;
132 case 36: return In22_Cu;
133 case 37: return In23_Cu;
134 case 38: return In24_Cu;
135 case 39: return In25_Cu;
136 case 40: return In26_Cu;
137 case 41: return In27_Cu;
138 case 42: return In28_Cu;
139 case 43: return In29_Cu;
140 case 44: return In30_Cu;
141
142 case 48: return User_2; // Component shape layer
143 case 49: return User_3; // Component marking
144
145 case 53: return User_4; // 3D shell outline
146 case 54: return User_5; // 3D shell top
147 case 55: return User_6; // 3D shell bot
148 case 56: return User_7; // Drill drawing
149
150 default: break;
151 }
152
153 return User_1;
154}
155
156
157static void AlignText( EDA_TEXT* text, int align )
158{
159 switch( align )
160 {
161 case 1:
162 text->SetVertJustify( GR_TEXT_V_ALIGN_TOP );
163 text->SetHorizJustify( GR_TEXT_H_ALIGN_LEFT );
164 break;
165 case 2:
166 text->SetVertJustify( GR_TEXT_V_ALIGN_CENTER );
167 text->SetHorizJustify( GR_TEXT_H_ALIGN_LEFT );
168 break;
169 case 3:
170 text->SetVertJustify( GR_TEXT_V_ALIGN_BOTTOM );
171 text->SetHorizJustify( GR_TEXT_H_ALIGN_LEFT );
172 break;
173
174 case 4:
175 text->SetVertJustify( GR_TEXT_V_ALIGN_TOP );
176 text->SetHorizJustify( GR_TEXT_H_ALIGN_CENTER );
177 break;
178 case 5:
179 text->SetVertJustify( GR_TEXT_V_ALIGN_CENTER );
180 text->SetHorizJustify( GR_TEXT_H_ALIGN_CENTER );
181 break;
182 case 6:
183 text->SetVertJustify( GR_TEXT_V_ALIGN_BOTTOM );
184 text->SetHorizJustify( GR_TEXT_H_ALIGN_CENTER );
185 break;
186
187 case 7:
188 text->SetVertJustify( GR_TEXT_V_ALIGN_TOP );
189 text->SetHorizJustify( GR_TEXT_H_ALIGN_RIGHT );
190 break;
191 case 8:
192 text->SetVertJustify( GR_TEXT_V_ALIGN_CENTER );
193 text->SetHorizJustify( GR_TEXT_H_ALIGN_RIGHT );
194 break;
195 case 9:
196 text->SetVertJustify( GR_TEXT_V_ALIGN_BOTTOM );
197 text->SetHorizJustify( GR_TEXT_H_ALIGN_RIGHT );
198 break;
199 }
200}
201
202
204 const wxString& modelUuid,
205 const wxString& modelTitle,
206 const wxString& modelTransform ) const
207{
208 // TODO: make this path configurable?
209 const wxString easyedaModelDir = wxS( "EASYEDA_MODELS" );
210 const wxString kicadModelPrefix = wxS( "${KIPRJMOD}/" ) + easyedaModelDir + wxS( "/" );
211
212 VECTOR3D kmodelOffset;
213 VECTOR3D kmodelRotation;
214
215 if( !modelUuid.IsEmpty() && !footprint->GetField( QUERY_MODEL_UUID_KEY ) )
216 {
217 PCB_FIELD* field = new PCB_FIELD( footprint, FIELD_T::USER, QUERY_MODEL_UUID_KEY );
218 field->SetLayer( Cmts_User );
219 field->SetVisible( false );
220 field->SetText( modelUuid );
221 footprint->Add( field );
222 }
223
224 if( !modelTransform.IsEmpty() && !footprint->GetField( MODEL_SIZE_KEY ) )
225 {
226 wxArrayString arr = wxSplit( modelTransform, ',', '\0' );
227
228 double fitXmm = pcbIUScale.IUTomm( ScaleSize( Convert( arr[0] ) ) );
229 double fitYmm = pcbIUScale.IUTomm( ScaleSize( Convert( arr[1] ) ) );
230
231 if( fitXmm > 0.0 && fitYmm > 0.0 )
232 {
233 PCB_FIELD* field = new PCB_FIELD( footprint, FIELD_T::USER, MODEL_SIZE_KEY );
234 field->SetLayer( Cmts_User );
235 field->SetVisible( false );
236 field->SetText( wxString::FromCDouble( fitXmm ) + wxS( " " )
237 + wxString::FromCDouble( fitYmm ) );
238 footprint->Add( field );
239 }
240
241 kmodelRotation.z = -Convert( arr[3] );
242 kmodelRotation.x = -Convert( arr[4] );
243 kmodelRotation.y = -Convert( arr[5] );
244
245 kmodelOffset.x = pcbIUScale.IUTomm( ScaleSize( Convert( arr[6] ) ) );
246 kmodelOffset.y = pcbIUScale.IUTomm( ScaleSize( Convert( arr[7] ) ) );
247 kmodelOffset.z = pcbIUScale.IUTomm( ScaleSize( Convert( arr[8] ) ) );
248 }
249
250 if( !modelTitle.IsEmpty() && footprint->Models().empty() )
251 {
252 FP_3DMODEL model;
253 model.m_Filename = kicadModelPrefix
254 + EscapeString( modelTitle, ESCAPE_CONTEXT::CTX_FILENAME )
255 + wxS( ".step" );
256 model.m_Offset = kmodelOffset;
257 model.m_Rotation = kmodelRotation;
258 footprint->Models().push_back( model );
259 }
260}
261
262
263std::vector<std::unique_ptr<PCB_SHAPE>>
264PCB_IO_EASYEDAPRO_PARSER::ParsePoly( BOARD_ITEM_CONTAINER* aContainer, nlohmann::json polyData,
265 bool aClosed, bool aInFill ) const
266{
267 std::vector<std::unique_ptr<PCB_SHAPE>> results;
268
269 VECTOR2D prevPt;
270 for( int i = 0; i < polyData.size(); i++ )
271 {
272 nlohmann::json val = polyData.at( i );
273
274 if( val.is_string() )
275 {
276 wxString str = val;
277 if( str == wxS( "CIRCLE" ) )
278 {
280 center.x = ( polyData.at( ++i ) );
281 center.y = ( polyData.at( ++i ) );
282 double r = ( polyData.at( ++i ) );
283
284 std::unique_ptr<PCB_SHAPE> shape =
285 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::CIRCLE );
286
287 shape->SetCenter( ScalePos( center ) );
288 shape->SetEnd( ScalePos( center + VECTOR2D( r, 0 ) ) );
289 shape->SetFilled( aClosed );
290
291 results.emplace_back( std::move( shape ) );
292 }
293 else if( str == wxS( "R" ) )
294 {
295 VECTOR2D start, size;
296 start.x = ( polyData.at( ++i ) );
297 start.y = ( polyData.at( ++i ) );
298 size.x = ( polyData.at( ++i ) );
299 size.y = -( polyData.at( ++i ).get<double>() );
300 double angle = polyData.at( ++i );
301 double cr = ( i + 1 ) < polyData.size() ? polyData.at( ++i ).get<double>() : 0;
302
303 if( cr == 0 )
304 {
305 std::unique_ptr<PCB_SHAPE> shape =
306 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::RECTANGLE );
307
308 shape->SetStart( ScalePos( start ) );
309 shape->SetEnd( ScalePos( start + size ) );
310 shape->SetFilled( aClosed );
311 shape->Rotate( ScalePos( start ), EDA_ANGLE( angle, DEGREES_T ) );
312
313 results.emplace_back( std::move( shape ) );
314 }
315 else
316 {
317 VECTOR2D end = start + size;
318
319 auto addSegment = [&]( VECTOR2D aStart, VECTOR2D aEnd )
320 {
321 std::unique_ptr<PCB_SHAPE> shape =
322 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::SEGMENT );
323
324 shape->SetStart( ScalePos( aStart ) );
325 shape->SetEnd( ScalePos( aEnd ) );
326 shape->SetFilled( aClosed );
327 shape->Rotate( ScalePos( start ), EDA_ANGLE( angle, DEGREES_T ) );
328
329 results.emplace_back( std::move( shape ) );
330 };
331
332 auto addArc = [&]( VECTOR2D aStart, VECTOR2D aEnd, VECTOR2D center )
333 {
334 std::unique_ptr<PCB_SHAPE> shape =
335 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::ARC );
336
337 shape->SetStart( ScalePos( aStart ) );
338 shape->SetEnd( ScalePos( aEnd ) );
339 shape->SetCenter( ScalePos( center ) );
340 shape->SetFilled( aClosed );
341 shape->Rotate( ScalePos( start ), EDA_ANGLE( angle, DEGREES_T ) );
342
343 results.emplace_back( std::move( shape ) );
344 };
345
346 addSegment( { start.x + cr, start.y }, { end.x - cr, start.y } );
347 addSegment( { end.x, start.y - cr }, { end.x, end.y + cr } );
348 addSegment( { start.x + cr, end.y }, { end.x - cr, end.y } );
349 addSegment( { start.x, start.y - cr }, { start.x, end.y + cr } );
350
351 addArc( { end.x - cr, start.y }, { end.x, start.y - cr },
352 { end.x - cr, start.y - cr } );
353
354 addArc( { end.x, end.y + cr }, { end.x - cr, end.y },
355 { end.x - cr, end.y + cr } );
356
357 addArc( { start.x + cr, end.y }, { start.x, end.y + cr },
358 { start.x + cr, end.y + cr } );
359
360 addArc( { start.x, start.y - cr }, { start.x + cr, start.y },
361 { start.x + cr, start.y - cr } );
362 }
363 }
364 else if( str == wxS( "ARC" ) || str == wxS( "CARC" ) )
365 {
367 double angle = polyData.at( ++i ).get<double>() / ( aInFill ? 10 : 1 );
368 end.x = ( polyData.at( ++i ) );
369 end.y = ( polyData.at( ++i ) );
370
371 std::unique_ptr<PCB_SHAPE> shape =
372 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::ARC );
373
374 if( angle < 0 )
375 {
376 shape->SetStart( ScalePos( prevPt ) );
377 shape->SetEnd( ScalePos( end ) );
378 }
379 else
380 {
381 shape->SetStart( ScalePos( end ) );
382 shape->SetEnd( ScalePos( prevPt ) );
383 }
384
385 VECTOR2D delta = end - prevPt;
386 VECTOR2D mid = ( prevPt + delta / 2 );
387
388 double ha = angle / 2;
389 double hd = delta.EuclideanNorm() / 2;
390 double cdist = hd / tan( DEG2RAD( ha ) );
391 VECTOR2D center = mid + delta.Perpendicular().Resize( cdist );
392 shape->SetCenter( ScalePos( center ) );
393
394 shape->SetFilled( aClosed );
395
396 results.emplace_back( std::move( shape ) );
397
398 prevPt = end;
399 }
400 else if( str == wxS( "L" ) )
401 {
403 chain.Append( ScalePos( prevPt ) );
404
405 while( i < polyData.size() - 2 && polyData.at( i + 1 ).is_number() )
406 {
407 VECTOR2D pt;
408 pt.x = ( polyData.at( ++i ) );
409 pt.y = ( polyData.at( ++i ) );
410
411 chain.Append( ScalePos( pt ) );
412
413 prevPt = pt;
414 }
415
416 if( aClosed )
417 {
418 std::unique_ptr<PCB_SHAPE> shape =
419 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::POLY );
420
421 wxASSERT( chain.PointCount() > 2 );
422
423 if( chain.PointCount() > 2 )
424 {
425 chain.SetClosed( true );
426 shape->SetFilled( true );
427 shape->SetPolyShape( chain );
428
429 results.emplace_back( std::move( shape ) );
430 }
431 }
432 else
433 {
434 for( int s = 0; s < chain.SegmentCount(); s++ )
435 {
436 SEG seg = chain.Segment( s );
437
438 std::unique_ptr<PCB_SHAPE> shape =
439 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::SEGMENT );
440
441 shape->SetStart( seg.A );
442 shape->SetEnd( seg.B );
443
444 results.emplace_back( std::move( shape ) );
445 }
446 }
447 }
448 }
449 else if( val.is_number() )
450 {
451 prevPt.x = ( polyData.at( i ) );
452 prevPt.y = ( polyData.at( ++i ) );
453 }
454 }
455
456 return results;
457}
458
459
461PCB_IO_EASYEDAPRO_PARSER::ParseContour( nlohmann::json polyData, bool aInFill,
462 double aArcAccuracy ) const
463{
464 SHAPE_LINE_CHAIN result;
465 VECTOR2D prevPt;
466
467 for( int i = 0; i < polyData.size(); i++ )
468 {
469 nlohmann::json val = polyData.at( i );
470
471 if( val.is_string() )
472 {
473 wxString str = val;
474 if( str == wxS( "CIRCLE" ) )
475 {
477 center.x = ( polyData.at( ++i ) );
478 center.y = ( polyData.at( ++i ) );
479 double r = ( polyData.at( ++i ) );
480
482 ERROR_INSIDE );
483 }
484 else if( str == wxS( "R" ) )
485 {
486 VECTOR2D start, size;
487 start.x = ( polyData.at( ++i ) );
488 start.y = ( polyData.at( ++i ) );
489 size.x = ( polyData.at( ++i ) );
490 size.y = ( polyData.at( ++i ).get<double>() );
491 double angle = polyData.at( ++i );
492 double cr = ( i + 1 ) < polyData.size() ? polyData.at( ++i ).get<double>() : 0;
493
494 SHAPE_POLY_SET poly;
495
496 VECTOR2D kstart = ScalePos( start );
497 VECTOR2D ksize = ScaleSize( size );
498 VECTOR2D kcenter = kstart + ksize / 2;
499 RotatePoint( kcenter, kstart, EDA_ANGLE( angle, DEGREES_T ) );
500
502 poly, kcenter, ksize, EDA_ANGLE( angle, DEGREES_T ), ScaleSize( cr ), 0, 0,
504
505 result.Append( poly.Outline( 0 ) );
506 }
507 else if( str == wxS( "ARC" ) || str == wxS( "CARC" ) )
508 {
510 double angle = polyData.at( ++i ).get<double>();
511
512 if( aInFill ) // In .epcb fills, the angle is 10x for some reason
513 angle /= 10;
514
515 end.x = ( polyData.at( ++i ) );
516 end.y = ( polyData.at( ++i ) );
517
518 VECTOR2D arcStart, arcEnd;
519 arcStart = prevPt;
520 arcEnd = end;
521
522 VECTOR2D delta = end - prevPt;
523 VECTOR2D mid = ( prevPt + delta / 2 );
524
525 double ha = angle / 2;
526 double hd = delta.EuclideanNorm() / 2;
527 double cdist = hd / tan( DEG2RAD( ha ) );
528 VECTOR2D center = mid + delta.Perpendicular().Resize( cdist );
529
530 SHAPE_ARC sarc;
531 sarc.ConstructFromStartEndCenter( ScalePos( arcStart ), ScalePos( arcEnd ),
532 ScalePos( center ), angle >= 0, 0 );
533
534 result.Append( sarc, aArcAccuracy );
535
536 prevPt = end;
537 }
538 else if( str == wxS( "C" ) )
539 {
540 VECTOR2D pt1;
541 pt1.x = ( polyData.at( ++i ) );
542 pt1.y = ( polyData.at( ++i ) );
543
544 VECTOR2D pt2;
545 pt2.x = ( polyData.at( ++i ) );
546 pt2.y = ( polyData.at( ++i ) );
547
548 VECTOR2D pt3;
549 pt3.x = ( polyData.at( ++i ) );
550 pt3.y = ( polyData.at( ++i ) );
551
552 std::vector<VECTOR2I> ctrlPoints = { ScalePos( prevPt ), ScalePos( pt1 ),
553 ScalePos( pt2 ), ScalePos( pt3 ) };
554 BEZIER_POLY converter( ctrlPoints );
555
556 std::vector<VECTOR2I> bezierPoints;
557 converter.GetPoly( bezierPoints, aArcAccuracy );
558
559 result.Append( bezierPoints );
560
561 prevPt = pt3;
562 }
563 else if( str == wxS( "L" ) )
564 {
565 result.Append( ScalePos( prevPt ) );
566
567 while( i < polyData.size() - 2 && polyData.at( i + 1 ).is_number() )
568 {
569 VECTOR2D pt;
570 pt.x = ( polyData.at( ++i ) );
571 pt.y = ( polyData.at( ++i ) );
572
573 result.Append( ScalePos( pt ) );
574
575 prevPt = pt;
576 }
577 }
578 }
579 else if( val.is_number() )
580 {
581 prevPt.x = ( polyData.at( i ) );
582 prevPt.y = ( polyData.at( ++i ) );
583 }
584 }
585
586 return result;
587}
588
589
590std::unique_ptr<PAD> PCB_IO_EASYEDAPRO_PARSER::createPAD( FOOTPRINT* aFootprint,
591 const nlohmann::json& line )
592{
593 wxString uuid = line.at( 1 );
594
595 // if( line.at( 2 ).is_number() )
596 // int unk = line.at( 2 ).get<int>();
597 // else if( line.at( 2 ).is_string() )
598 // int unk = wxAtoi( line.at( 2 ).get<wxString>() );
599
600 wxString netname = line.at( 3 );
601 int layer = line.at( 4 ).get<int>();
602 PCB_LAYER_ID klayer = LayerToKi( layer );
603
604 wxString padNumber = line.at( 5 );
605
607 center.x = line.at( 6 );
608 center.y = line.at( 7 );
609
610 double orientation = line.at( 8 );
611
612 nlohmann::json padHole = line.at( 9 );
613 nlohmann::json padShape = line.at( 10 );
614
615 std::unique_ptr<PAD> pad = std::make_unique<PAD>( aFootprint );
616
617 pad->SetNumber( padNumber );
618 pad->SetPosition( ScalePos( center ) );
619 pad->SetOrientationDegrees( orientation );
620
621 if( !padHole.is_null() )
622 {
623 double drill_dir = 0;
624
625 if( line.at( 14 ).is_number() )
626 drill_dir = line.at( 14 );
627
628 wxString holeShape = padHole.at( 0 );
629
630 if( holeShape == wxS( "ROUND" ) || holeShape == wxS( "SLOT" ) )
631 {
632 VECTOR2D drill;
633 drill.x = padHole.at( 1 );
634 drill.y = padHole.at( 2 );
635
636 double deg = EDA_ANGLE( drill_dir, DEGREES_T ).Normalize90().AsDegrees();
637
638 if( std::abs( deg ) >= 45 )
639 std::swap( drill.x, drill.y ); // KiCad doesn't support arbitrary hole direction
640
641 if( holeShape == wxS( "SLOT" ) )
642 {
643 pad->SetDrillShape( PAD_DRILL_SHAPE::OBLONG );
644 }
645
646 pad->SetDrillSize( ScaleSize( drill ) );
647 pad->SetLayerSet( PAD::PTHMask() );
648 pad->SetAttribute( PAD_ATTRIB::PTH );
649 }
650 }
651 else
652 {
653 if( klayer == F_Cu )
654 {
655 pad->SetLayerSet( PAD::SMDMask() );
656 }
657 else if( klayer == B_Cu )
658 {
659 pad->SetLayerSet( PAD::SMDMask().FlipStandardLayers() );
660 }
661
662 pad->SetAttribute( PAD_ATTRIB::SMD );
663 }
664
665 wxString padSh = padShape.at( 0 );
666 if( padSh == wxS( "RECT" ) )
667 {
668 VECTOR2D size;
669 size.x = padShape.at( 1 );
670 size.y = padShape.at( 2 );
671 double cr_p = padShape.size() > 3 ? padShape.at( 3 ).get<double>() : 0;
672
673 pad->SetSize( PADSTACK::ALL_LAYERS, ScaleSize( size ) );
674
675 if( cr_p == 0 )
676 {
677 pad->SetShape( PADSTACK::ALL_LAYERS, PAD_SHAPE::RECTANGLE );
678 }
679 else
680 {
681 pad->SetShape( PADSTACK::ALL_LAYERS, PAD_SHAPE::ROUNDRECT );
682 pad->SetRoundRectRadiusRatio( PADSTACK::ALL_LAYERS, cr_p / 100 );
683 }
684 }
685 else if( padSh == wxS( "ELLIPSE" ) )
686 {
687 VECTOR2D size;
688 size.x = padShape.at( 1 );
689 size.y = padShape.at( 2 );
690
691 pad->SetSize( PADSTACK::ALL_LAYERS, ScaleSize( size ) );
692 pad->SetShape( PADSTACK::ALL_LAYERS, PAD_SHAPE::CIRCLE );
693 }
694 else if( padSh == wxS( "OVAL" ) )
695 {
696 VECTOR2D size;
697 size.x = padShape.at( 1 );
698 size.y = padShape.at( 2 );
699
700 pad->SetSize( PADSTACK::ALL_LAYERS, ScaleSize( size ) );
701 pad->SetShape( PADSTACK::ALL_LAYERS, PAD_SHAPE::OVAL );
702 }
703 else if( padSh == wxS( "POLY" ) )
704 {
705 pad->SetShape( PADSTACK::ALL_LAYERS, PAD_SHAPE::CUSTOM );
706 pad->SetAnchorPadShape( PADSTACK::ALL_LAYERS, PAD_SHAPE::CIRCLE );
707 pad->SetSize( PADSTACK::ALL_LAYERS, { 1, 1 } );
708
709 nlohmann::json polyData = padShape.at( 1 );
710
711 std::vector<std::unique_ptr<PCB_SHAPE>> results =
712 ParsePoly( aFootprint, polyData, true, false );
713
714 for( auto& shape : results )
715 {
716 shape->SetLayer( klayer );
717 shape->SetWidth( 0 );
718
719 shape->Move( -pad->GetPosition() );
720
721 pad->AddPrimitive( PADSTACK::ALL_LAYERS, shape.release() );
722 }
723 }
724
725 pad->SetThermalSpokeAngle( ANGLE_90 );
726
727 return std::move( pad );
728}
729
730
732 const wxString& aFpUuid,
733 const std::vector<nlohmann::json>& aLines )
734{
735 std::unique_ptr<FOOTPRINT> footprintPtr = std::make_unique<FOOTPRINT>( m_board );
736 FOOTPRINT* footprint = footprintPtr.get();
737
738 const VECTOR2I defaultTextSize( pcbIUScale.mmToIU( 1.0 ), pcbIUScale.mmToIU( 1.0 ) );
739 const int defaultTextThickness( pcbIUScale.mmToIU( 0.15 ) );
740
741 for( PCB_FIELD* field : footprint->GetFields() )
742 {
743 field->SetTextSize( defaultTextSize );
744 field->SetTextThickness( defaultTextThickness );
745 }
746
747 for( const nlohmann::json& line : aLines )
748 {
749 if( line.size() == 0 )
750 continue;
751
752 wxString type = line.at( 0 );
753
754 if( type == wxS( "POLY" ) || type == wxS( "PAD" ) || type == wxS( "FILL" )
755 || type == wxS( "ATTR" ) )
756 {
757 wxString uuid = line.at( 1 );
758
759 // if( line.at( 2 ).is_number() )
760 // int unk = line.at( 2 ).get<int>();
761 // else if( line.at( 2 ).is_string() )
762 // int unk = wxAtoi( line.at( 2 ).get<wxString>() );
763
764 wxString netname = line.at( 3 );
765 int layer = line.at( 4 ).get<int>();
766 PCB_LAYER_ID klayer = LayerToKi( layer );
767
768 if( type == wxS( "POLY" ) )
769 {
770 double thickness = ( line.at( 5 ) );
771 nlohmann::json polyData = line.at( 6 );
772
773 std::vector<std::unique_ptr<PCB_SHAPE>> results =
774 ParsePoly( footprint, polyData, false, false );
775
776 for( auto& shape : results )
777 {
778 shape->SetLayer( klayer );
779 shape->SetWidth( ScaleSize( thickness ) );
780
781 footprint->Add( shape.release(), ADD_MODE::APPEND );
782 }
783 }
784 else if( type == wxS( "PAD" ) )
785 {
786 std::unique_ptr<PAD> pad = createPAD( footprint, line );
787
788 footprint->Add( pad.release(), ADD_MODE::APPEND );
789 }
790 else if( type == wxS( "FILL" ) )
791 {
792 int layer = line.at( 4 ).get<int>();
793 PCB_LAYER_ID klayer = LayerToKi( layer );
794
795 double width = line.at( 5 );
796
797 nlohmann::json polyDataList = line.at( 7 );
798
799 if( !polyDataList.is_null() && !polyDataList.empty() )
800 {
801 if( !polyDataList.at( 0 ).is_array() )
802 polyDataList = nlohmann::json::array( { polyDataList } );
803
804 std::vector<SHAPE_LINE_CHAIN> contours;
805 for( nlohmann::json& polyData : polyDataList )
806 {
807 SHAPE_LINE_CHAIN contour = ParseContour( polyData, false );
808 contour.SetClosed( true );
809
810 contours.push_back( contour );
811 }
812
813 SHAPE_POLY_SET polySet;
814
815 for( SHAPE_LINE_CHAIN& contour : contours )
816 polySet.AddOutline( contour );
817
818 polySet.RebuildHolesFromContours();
819
820 std::unique_ptr<PCB_GROUP> group;
821
822 if( polySet.OutlineCount() > 1 )
823 group = std::make_unique<PCB_GROUP>( footprint );
824
825 BOX2I polyBBox = polySet.BBox();
826
827 for( const SHAPE_POLY_SET::POLYGON& poly : polySet.CPolygons() )
828 {
829 std::unique_ptr<PCB_SHAPE> shape =
830 std::make_unique<PCB_SHAPE>( footprint, SHAPE_T::POLY );
831
832 shape->SetFilled( true );
833 shape->SetPolyShape( poly );
834 shape->SetLayer( klayer );
835 shape->SetWidth( 0 );
836
837 if( group )
838 group->AddItem( shape.get() );
839
840 footprint->Add( shape.release(), ADD_MODE::APPEND );
841 }
842
843 if( group )
844 footprint->Add( group.release(), ADD_MODE::APPEND );
845 }
846 }
847 else if( type == wxS( "ATTR" ) )
848 {
849 EASYEDAPRO::PCB_ATTR attr = line;
850
851 if( attr.key == wxS( "Designator" ) )
852 footprint->GetField( FIELD_T::REFERENCE )->SetText( attr.value );
853 }
854 }
855 else if( type == wxS( "REGION" ) )
856 {
857 wxString uuid = line.at( 1 );
858
859 // if( line.at( 2 ).is_number() )
860 // int unk = line.at( 2 ).get<int>();
861 // else if( line.at( 2 ).is_string() )
862 // int unk = wxAtoi( line.at( 2 ).get<wxString>() );
863
864 int layer = line.at( 3 ).get<int>();
865 PCB_LAYER_ID klayer = LayerToKi( layer );
866
867 double width = line.at( 4 );
868 std::set<int> flags = line.at( 5 );
869 nlohmann::json polyDataList = line.at( 6 );
870
871 for( nlohmann::json& polyData : polyDataList )
872 {
873 SHAPE_POLY_SET polySet;
874
875 std::vector<std::unique_ptr<PCB_SHAPE>> results =
876 ParsePoly( nullptr, polyData, true, false );
877
878 for( auto& shape : results )
879 {
880 shape->SetFilled( true );
881 shape->TransformShapeToPolygon( polySet, klayer, 0, ARC_HIGH_DEF, ERROR_INSIDE,
882 true );
883 }
884
885 polySet.Simplify();
886
887 std::unique_ptr<ZONE> zone = std::make_unique<ZONE>( footprint );
888
889 zone->SetIsRuleArea( true );
890 zone->SetDoNotAllowFootprints( !!flags.count( 2 ) );
891 zone->SetDoNotAllowZoneFills( !!flags.count( 7 ) || !!flags.count( 6 )
892 || !!flags.count( 8 ) );
893 zone->SetDoNotAllowPads( !!flags.count( 7 ) );
894 zone->SetDoNotAllowTracks( !!flags.count( 7 ) || !!flags.count( 5 ) );
895 zone->SetDoNotAllowVias( !!flags.count( 7 ) );
896
897 zone->SetLayer( klayer );
898 zone->Outline()->Append( polySet );
899
900 footprint->Add( zone.release(), ADD_MODE::APPEND );
901 }
902 }
903 }
904
905 if( aProject.is_object() && aProject.contains( "devices" ) )
906 {
907 std::map<wxString, EASYEDAPRO::PRJ_DEVICE> devicesMap = aProject.at( "devices" );
908 std::map<wxString, wxString> compAttrs;
909
910 for( auto& [devUuid, devData] : devicesMap )
911 {
912 if( auto fp = get_opt( devData.attributes, "Footprint" ) )
913 {
914 if( *fp == aFpUuid )
915 {
916 compAttrs = devData.attributes;
917 break;
918 }
919 }
920 }
921
922 wxString modelUuid, modelTitle, modelTransform;
923
924 modelUuid = get_def( compAttrs, "3D Model", "" );
925 modelTitle = get_def( compAttrs, "3D Model Title", modelUuid );
926 modelTransform = get_def( compAttrs, "3D Model Transform", "" );
927
928 fillFootprintModelInfo( footprint, modelUuid, modelTitle, modelTransform );
929 }
930
931 // Heal board outlines
932 std::vector<PCB_SHAPE*> shapes;
933 std::vector<std::unique_ptr<PCB_SHAPE>> newShapes;
934
935 for( BOARD_ITEM* item : footprint->GraphicalItems() )
936 {
937 if( !item->IsOnLayer( Edge_Cuts ) )
938 continue;
939
940 if( item->Type() == PCB_SHAPE_T )
941 shapes.push_back( static_cast<PCB_SHAPE*>( item ) );
942 }
943
944 ConnectBoardShapes( shapes, newShapes, SHAPE_JOIN_DISTANCE );
945
946 for( std::unique_ptr<PCB_SHAPE>& ptr : newShapes )
947 footprint->Add( ptr.release(), ADD_MODE::APPEND );
948
949 return footprintPtr.release();
950}
951
952
954 BOARD* aBoard, const nlohmann::json& aProject,
955 std::map<wxString, std::unique_ptr<FOOTPRINT>>& aFootprintMap,
956 const std::map<wxString, EASYEDAPRO::BLOB>& aBlobMap,
957 const std::multimap<wxString, EASYEDAPRO::POURED>& aPouredMap,
958 const std::vector<nlohmann::json>& aLines, const wxString& aFpLibName )
959{
960 std::map<wxString, std::vector<nlohmann::json>> componentLines;
961 std::map<wxString, std::vector<nlohmann::json>> ruleLines;
962
963 std::multimap<wxString, EASYEDAPRO::POURED> boardPouredMap = aPouredMap;
964 std::map<wxString, ZONE*> poursToFill;
965
967
968 for( const nlohmann::json& line : aLines )
969 {
970 if( line.size() == 0 )
971 continue;
972
973 wxString type = line.at( 0 );
974
975 if( type == wxS( "LAYER" ) )
976 {
977 int layer = line.at( 1 );
978 PCB_LAYER_ID klayer = LayerToKi( layer );
979
980 wxString layerType = line.at( 2 );
981 wxString layerName = line.at( 3 );
982 int layerFlag = line.at( 4 );
983
984 if( layerFlag != 0 )
985 {
986 LSET blayers = aBoard->GetEnabledLayers();
987 blayers.set( klayer );
988 aBoard->SetEnabledLayers( blayers );
989 aBoard->SetLayerName( klayer, layerName );
990 }
991 }
992 else if( type == wxS( "NET" ) )
993 {
994 wxString netname = line.at( 1 );
995
996 aBoard->Add( new NETINFO_ITEM( aBoard, netname, aBoard->GetNetCount() + 1 ),
997 ADD_MODE::APPEND );
998 }
999 else if( type == wxS( "RULE" ) )
1000 {
1001 wxString ruleType = line.at( 1 );
1002 wxString ruleName = line.at( 2 );
1003 int isDefault = line.at( 3 );
1004 nlohmann::json ruleData = line.at( 4 );
1005
1006 if( ruleType == wxS( "3" ) && isDefault ) // Track width
1007 {
1008 wxString units = ruleData.at( 0 );
1009 double minVal = ruleData.at( 1 );
1010 double optVal = ruleData.at( 2 );
1011 double maxVal = ruleData.at( 3 );
1012
1013 bds.m_TrackMinWidth = ScaleSize( minVal );
1014 }
1015 else if( ruleType == wxS( "1" ) && isDefault )
1016 {
1017 wxString units = ruleData.at( 0 );
1018 nlohmann::json table = ruleData.at( 1 );
1019
1020 int minVal = INT_MAX;
1021 for( const std::vector<int>& arr : table )
1022 {
1023 for( int val : arr )
1024 {
1025 if( val < minVal )
1026 minVal = val;
1027 }
1028 }
1029
1030 bds.m_MinClearance = ScaleSize( minVal );
1031 }
1032
1033 ruleLines[ruleType].push_back( line );
1034 }
1035 else if( type == wxS( "POURED" ) )
1036 {
1037 if( !line.at( 2 ).is_string() )
1038 continue; // Unknown type of POURED
1039
1040 EASYEDAPRO::POURED poured = line;
1041 boardPouredMap.emplace( poured.parentId, poured );
1042 }
1043 else if( type == wxS( "VIA" ) || type == wxS( "LINE" ) || type == wxS( "ARC" )
1044 || type == wxS( "POLY" ) || type == wxS( "FILL" ) || type == wxS( "POUR" ) )
1045 {
1046 wxString uuid = line.at( 1 );
1047
1048 // if( line.at( 2 ).is_number() )
1049 // int unk = line.at( 2 ).get<int>();
1050 // else if( line.at( 2 ).is_string() )
1051 // int unk = wxAtoi( line.at( 2 ).get<wxString>() );
1052
1053 wxString netname = line.at( 3 );
1054
1055 if( type == wxS( "VIA" ) )
1056 {
1058 center.x = line.at( 5 );
1059 center.y = line.at( 6 );
1060
1061 double drill = line.at( 7 );
1062 double dia = line.at( 8 );
1063
1064 std::unique_ptr<PCB_VIA> via = std::make_unique<PCB_VIA>( aBoard );
1065
1066 via->SetPosition( ScalePos( center ) );
1067 via->SetDrill( ScaleSize( drill ) );
1068 via->SetWidth( PADSTACK::ALL_LAYERS, ScaleSize( dia ) );
1069
1070 via->SetNet( aBoard->FindNet( netname ) );
1071
1072 aBoard->Add( via.release(), ADD_MODE::APPEND );
1073 }
1074 else if( type == wxS( "LINE" ) )
1075 {
1076 int layer = line.at( 4 ).get<int>();
1077 PCB_LAYER_ID klayer = LayerToKi( layer );
1078
1079 VECTOR2D start;
1080 start.x = line.at( 5 );
1081 start.y = line.at( 6 );
1082
1083 VECTOR2D end;
1084 end.x = line.at( 7 );
1085 end.y = line.at( 8 );
1086
1087 double width = line.at( 9 );
1088
1089 std::unique_ptr<PCB_TRACK> track = std::make_unique<PCB_TRACK>( aBoard );
1090
1091 track->SetLayer( klayer );
1092 track->SetStart( ScalePos( start ) );
1093 track->SetEnd( ScalePos( end ) );
1094 track->SetWidth( ScaleSize( width ) );
1095
1096 track->SetNet( aBoard->FindNet( netname ) );
1097
1098 aBoard->Add( track.release(), ADD_MODE::APPEND );
1099 }
1100 else if( type == wxS( "ARC" ) )
1101 {
1102 int layer = line.at( 4 ).get<int>();
1103 PCB_LAYER_ID klayer = LayerToKi( layer );
1104
1105 VECTOR2D start;
1106 start.x = line.at( 5 );
1107 start.y = line.at( 6 );
1108
1109 VECTOR2D end;
1110 end.x = line.at( 7 );
1111 end.y = line.at( 8 );
1112
1113 double angle = line.at( 9 );
1114 double width = line.at( 10 );
1115
1116 VECTOR2D delta = end - start;
1117 VECTOR2D mid = ( start + delta / 2 );
1118
1119 double ha = angle / 2;
1120 double hd = delta.EuclideanNorm() / 2;
1121 double cdist = hd / tan( DEG2RAD( ha ) );
1122 VECTOR2D center = mid + delta.Perpendicular().Resize( cdist );
1123
1124 SHAPE_ARC sarc;
1126 ScalePos( center ), angle >= 0, width );
1127
1128 std::unique_ptr<PCB_ARC> arc = std::make_unique<PCB_ARC>( aBoard, &sarc );
1129 arc->SetWidth( ScaleSize( width ) );
1130
1131 arc->SetLayer( klayer );
1132 arc->SetNet( aBoard->FindNet( netname ) );
1133
1134 aBoard->Add( arc.release(), ADD_MODE::APPEND );
1135 }
1136 else if( type == wxS( "FILL" ) )
1137 {
1138 int layer = line.at( 4 ).get<int>();
1139 PCB_LAYER_ID klayer = LayerToKi( layer );
1140
1141 double width = line.at( 5 );
1142
1143 nlohmann::json polyDataList = line.at( 7 );
1144
1145 if( !polyDataList.at( 0 ).is_array() )
1146 polyDataList = nlohmann::json::array( { polyDataList } );
1147
1148 std::vector<SHAPE_LINE_CHAIN> contours;
1149 for( nlohmann::json& polyData : polyDataList )
1150 {
1151 SHAPE_LINE_CHAIN contour = ParseContour( polyData, true );
1152 contour.SetClosed( true );
1153
1154 contours.push_back( contour );
1155 }
1156
1157 SHAPE_POLY_SET zoneFillPoly;
1158
1159 for( SHAPE_LINE_CHAIN& contour : contours )
1160 zoneFillPoly.AddOutline( contour );
1161
1162 zoneFillPoly.RebuildHolesFromContours();
1163 zoneFillPoly.Fracture();
1164
1165 std::unique_ptr<ZONE> zone = std::make_unique<ZONE>( aBoard );
1166
1167 zone->SetNet( aBoard->FindNet( netname ) );
1168 zone->SetLayer( klayer );
1169 zone->Outline()->Append( SHAPE_RECT( zoneFillPoly.BBox() ).Outline() );
1170 zone->SetFilledPolysList( klayer, zoneFillPoly );
1171 zone->SetAssignedPriority( 500 );
1172 zone->SetIsFilled( true );
1173 zone->SetNeedRefill( false );
1174
1175 zone->SetLocalClearance( bds.m_MinClearance );
1176 zone->SetMinThickness( bds.m_TrackMinWidth );
1177
1178 aBoard->Add( zone.release(), ADD_MODE::APPEND );
1179 }
1180 else if( type == wxS( "POLY" ) )
1181 {
1182 int layer = line.at( 4 );
1183 PCB_LAYER_ID klayer = LayerToKi( layer );
1184
1185 double thickness = line.at( 5 );
1186 nlohmann::json polyData = line.at( 6 );
1187
1188 std::vector<std::unique_ptr<PCB_SHAPE>> results =
1189 ParsePoly( aBoard, polyData, false, false );
1190
1191 for( auto& shape : results )
1192 {
1193 shape->SetLayer( klayer );
1194 shape->SetWidth( ScaleSize( thickness ) );
1195
1196 aBoard->Add( shape.release(), ADD_MODE::APPEND );
1197 }
1198 }
1199 else if( type == wxS( "POUR" ) )
1200 {
1201 int layer = line.at( 4 ).get<int>();
1202 PCB_LAYER_ID klayer = LayerToKi( layer );
1203
1204 double lineWidth = line.at( 5 ); // Doesn't matter
1205 wxString pourname = line.at( 6 );
1206 int fillOrder = line.at( 7 ).get<int>();
1207 nlohmann::json polyDataList = line.at( 8 );
1208
1209 std::unique_ptr<ZONE> zone = std::make_unique<ZONE>( aBoard );
1210
1211 zone->SetNet( aBoard->FindNet( netname ) );
1212 zone->SetLayer( klayer );
1213 zone->SetAssignedPriority( 500 - fillOrder );
1214 zone->SetLocalClearance( bds.m_MinClearance );
1215 zone->SetMinThickness( bds.m_TrackMinWidth );
1216
1217 for( nlohmann::json& polyData : polyDataList )
1218 {
1219 SHAPE_LINE_CHAIN contour = ParseContour( polyData, false );
1220 contour.SetClosed( true );
1221
1222 zone->Outline()->Append( contour );
1223 }
1224
1225 wxASSERT( zone->Outline()->OutlineCount() == 1 );
1226
1227 poursToFill.emplace( uuid, zone.get() );
1228
1229 aBoard->Add( zone.release(), ADD_MODE::APPEND );
1230 }
1231 }
1232 else if( type == wxS( "TEARDROP" ) )
1233 {
1234 wxString uuid = line.at( 1 );
1235 wxString netname = line.at( 2 );
1236 int layer = line.at( 3 ).get<int>();
1237 PCB_LAYER_ID klayer = LayerToKi( layer );
1238
1239 nlohmann::json polyData = line.at( 4 );
1240
1241 SHAPE_LINE_CHAIN contour = ParseContour( polyData, false );
1242 contour.SetClosed( true );
1243
1244 std::unique_ptr<ZONE> zone = std::make_unique<ZONE>( aBoard );
1245
1246 zone->SetNet( aBoard->FindNet( netname ) );
1247 zone->SetLayer( klayer );
1248 zone->Outline()->Append( contour );
1249 zone->SetFilledPolysList( klayer, contour );
1250 zone->SetNeedRefill( false );
1251 zone->SetIsFilled( true );
1252
1253 zone->SetAssignedPriority( 600 );
1254 zone->SetLocalClearance( 0 );
1255 zone->SetMinThickness( 0 );
1256 zone->SetTeardropAreaType( TEARDROP_TYPE::TD_UNSPECIFIED );
1257
1258 aBoard->Add( zone.release(), ADD_MODE::APPEND );
1259 }
1260 else if( type == wxS( "REGION" ) )
1261 {
1262 wxString uuid = line.at( 1 );
1263
1264 // if( line.at( 2 ).is_number() )
1265 // int unk = line.at( 2 ).get<int>();
1266 // else if( line.at( 2 ).is_string() )
1267 // int unk = wxAtoi( line.at( 2 ).get<wxString>() );
1268
1269 int layer = line.at( 3 ).get<int>();
1270 PCB_LAYER_ID klayer = LayerToKi( layer );
1271
1272 double width = line.at( 4 );
1273 std::set<int> flags = line.at( 5 );
1274 nlohmann::json polyDataList = line.at( 6 );
1275
1276 for( nlohmann::json& polyData : polyDataList )
1277 {
1278 SHAPE_LINE_CHAIN contour = ParseContour( polyData, false );
1279 contour.SetClosed( true );
1280
1281 std::unique_ptr<ZONE> zone = std::make_unique<ZONE>( aBoard );
1282
1283 zone->SetIsRuleArea( true );
1284 zone->SetDoNotAllowFootprints( !!flags.count( 2 ) );
1285 zone->SetDoNotAllowZoneFills( !!flags.count( 7 ) || !!flags.count( 6 )
1286 || !!flags.count( 8 ) );
1287 zone->SetDoNotAllowPads( !!flags.count( 7 ) );
1288 zone->SetDoNotAllowTracks( !!flags.count( 7 ) || !!flags.count( 5 ) );
1289 zone->SetDoNotAllowVias( !!flags.count( 7 ) );
1290
1291 zone->SetLayer( klayer );
1292 zone->Outline()->Append( contour );
1293
1294 aBoard->Add( zone.release(), ADD_MODE::APPEND );
1295 }
1296 }
1297 else if( type == wxS( "PAD" ) )
1298 {
1299 wxString netname = line.at( 3 );
1300
1301 std::unique_ptr<FOOTPRINT> footprint = std::make_unique<FOOTPRINT>( aBoard );
1302 std::unique_ptr<PAD> pad = createPAD( footprint.get(), line );
1303
1304 pad->SetNet( aBoard->FindNet( netname ) );
1305
1306 VECTOR2I pos = pad->GetPosition();
1307 EDA_ANGLE orient = pad->GetOrientation();
1308
1309 pad->SetPosition( VECTOR2I() );
1310 pad->SetOrientation( ANGLE_0 );
1311
1312 footprint->Add( pad.release(), ADD_MODE::APPEND );
1313 footprint->SetPosition( pos );
1314 footprint->SetOrientation( orient );
1315
1316 wxString fpName = wxS( "Pad_" ) + line.at( 1 ).get<wxString>();
1317 LIB_ID fpID = EASYEDAPRO::ToKiCadLibID( wxEmptyString, fpName );
1318
1319 footprint->SetFPID( fpID );
1320 footprint->Reference().SetVisible( true );
1321 footprint->Value().SetVisible( true );
1322 footprint->AutoPositionFields();
1323
1324 aBoard->Add( footprint.release(), ADD_MODE::APPEND );
1325 }
1326 else if( type == wxS( "IMAGE" ) )
1327 {
1328 wxString uuid = line.at( 1 );
1329
1330 // if( line.at( 2 ).is_number() )
1331 // int unk = line.at( 2 ).get<int>();
1332 // else if( line.at( 2 ).is_string() )
1333 // int unk = wxAtoi( line.at( 2 ).get<wxString>() );
1334
1335 int layer = line.at( 3 ).get<int>();
1336 PCB_LAYER_ID klayer = LayerToKi( layer );
1337
1338 VECTOR2D start( line.at( 4 ), line.at( 5 ) );
1339 VECTOR2D size( line.at( 6 ), line.at( 7 ) );
1340
1341 double angle = line.at( 8 ); // from top left corner
1342 int mirror = line.at( 9 );
1343 nlohmann::json polyDataList = line.at( 10 );
1344
1345 BOX2I bbox;
1346 std::vector<SHAPE_LINE_CHAIN> contours;
1347 for( nlohmann::json& polyData : polyDataList )
1348 {
1349 SHAPE_LINE_CHAIN contour = ParseContour( polyData, false );
1350 contour.SetClosed( true );
1351
1352 contours.push_back( contour );
1353
1354 bbox.Merge( contour.BBox() );
1355 }
1356
1357 VECTOR2D scale( ScaleSize( size.x ) / bbox.GetSize().x,
1358 ScaleSize( size.y ) / bbox.GetSize().y );
1359
1360 SHAPE_POLY_SET polySet;
1361
1362 for( SHAPE_LINE_CHAIN& contour : contours )
1363 {
1364 for( int i = 0; i < contour.PointCount(); i++ )
1365 {
1366 VECTOR2I pt = contour.CPoint( i );
1367 contour.SetPoint( i, VECTOR2I( pt.x * scale.x, pt.y * scale.y ) );
1368 }
1369
1370 polySet.AddOutline( contour );
1371 }
1372
1373 polySet.RebuildHolesFromContours();
1374
1375 std::unique_ptr<PCB_GROUP> group;
1376
1377 if( polySet.OutlineCount() > 1 )
1378 group = std::make_unique<PCB_GROUP>( aBoard );
1379
1380 BOX2I polyBBox = polySet.BBox();
1381
1382 for( const SHAPE_POLY_SET::POLYGON& poly : polySet.CPolygons() )
1383 {
1384 std::unique_ptr<PCB_SHAPE> shape =
1385 std::make_unique<PCB_SHAPE>( aBoard, SHAPE_T::POLY );
1386
1387 shape->SetFilled( true );
1388 shape->SetPolyShape( poly );
1389 shape->SetLayer( klayer );
1390 shape->SetWidth( 0 );
1391
1392 shape->Move( ScalePos( start ) - polyBBox.GetOrigin() );
1393 shape->Rotate( ScalePos( start ), EDA_ANGLE( angle, DEGREES_T ) );
1394
1395 if( IsBackLayer( klayer ) ^ !!mirror )
1396 {
1397 FLIP_DIRECTION flipDirection = IsBackLayer( klayer )
1398 ? FLIP_DIRECTION::TOP_BOTTOM
1399 : FLIP_DIRECTION::LEFT_RIGHT;
1400 shape->Mirror( ScalePos( start ), flipDirection );
1401 }
1402
1403 if( group )
1404 group->AddItem( shape.get() );
1405
1406 aBoard->Add( shape.release(), ADD_MODE::APPEND );
1407 }
1408
1409 if( group )
1410 aBoard->Add( group.release(), ADD_MODE::APPEND );
1411 }
1412 else if( type == wxS( "OBJ" ) )
1413 {
1414 VECTOR2D start, size;
1415 wxString mimeType, base64Data;
1416 double angle = 0;
1417 int flipped = 0;
1418
1419 if( !line.at( 3 ).is_number() )
1420 continue;
1421
1422 int layer = line.at( 3 ).get<int>();
1423 PCB_LAYER_ID klayer = LayerToKi( layer );
1424
1425 start = VECTOR2D( line.at( 5 ), line.at( 6 ) );
1426 size = VECTOR2D( line.at( 7 ), line.at( 8 ) );
1427 angle = line.at( 9 );
1428 flipped = line.at( 10 );
1429
1430 wxString imageUrl = line.at( 11 );
1431
1432 if( imageUrl.BeforeFirst( ':' ) == wxS( "blob" ) )
1433 {
1434 wxString objectId = imageUrl.AfterLast( ':' );
1435
1436 if( auto blob = get_opt( aBlobMap, objectId ) )
1437 {
1438 wxString blobUrl = blob->url;
1439
1440 if( blobUrl.BeforeFirst( ':' ) == wxS( "data" ) )
1441 {
1442 wxArrayString paramsArr =
1443 wxSplit( blobUrl.AfterFirst( ':' ).BeforeFirst( ',' ), ';', '\0' );
1444
1445 base64Data = blobUrl.AfterFirst( ',' );
1446
1447 if( paramsArr.size() > 0 )
1448 mimeType = paramsArr[0];
1449 }
1450 }
1451 }
1452
1453 VECTOR2D kstart = ScalePos( start );
1454 VECTOR2D ksize = ScaleSize( size );
1455
1456 if( mimeType.empty() || base64Data.empty() )
1457 continue;
1458
1459 wxMemoryBuffer buf = wxBase64Decode( base64Data );
1460
1461 if( mimeType == wxS( "image/svg+xml" ) )
1462 {
1463 // Not yet supported by EasyEDA
1464 }
1465 else
1466 {
1467 VECTOR2D kcenter = kstart + ksize / 2;
1468
1469 std::unique_ptr<PCB_REFERENCE_IMAGE> bitmap =
1470 std::make_unique<PCB_REFERENCE_IMAGE>( aBoard, kcenter, klayer );
1471 REFERENCE_IMAGE& refImage = bitmap->GetReferenceImage();
1472
1473 wxImage::SetDefaultLoadFlags( wxImage::GetDefaultLoadFlags()
1474 & ~wxImage::Load_Verbose );
1475
1476 if( refImage.ReadImageFile( buf ) )
1477 {
1478 double scaleFactor = ScaleSize( size.x ) / refImage.GetSize().x;
1479 refImage.SetImageScale( scaleFactor );
1480
1481 // TODO: support non-90-deg angles
1482 bitmap->Rotate( kstart, EDA_ANGLE( angle, DEGREES_T ) );
1483
1484 if( flipped )
1485 {
1486 int x = bitmap->GetPosition().x;
1487 MIRROR( x, KiROUND( kstart.x ) );
1488 bitmap->SetX( x );
1489
1490 refImage.MutableImage().Mirror( FLIP_DIRECTION::LEFT_RIGHT );
1491 }
1492
1493 aBoard->Add( bitmap.release(), ADD_MODE::APPEND );
1494 }
1495 }
1496 }
1497 else if( type == wxS( "STRING" ) )
1498 {
1499 wxString uuid = line.at( 1 );
1500
1501 // if( line.at( 2 ).is_number() )
1502 // int unk = line.at( 2 ).get<int>();
1503 // else if( line.at( 2 ).is_string() )
1504 // int unk = wxAtoi( line.at( 2 ).get<wxString>() );
1505
1506 int layer = line.at( 3 ).get<int>();
1507 PCB_LAYER_ID klayer = LayerToKi( layer );
1508
1509 VECTOR2D location( line.at( 4 ), line.at( 5 ) );
1510 wxString string = line.at( 6 );
1511 wxString font = line.at( 7 );
1512
1513 double height = line.at( 8 );
1514 double strokew = line.at( 9 );
1515
1516 int align = line.at( 12 );
1517 double angle = line.at( 13 );
1518 int inverted = line.at( 14 );
1519 int mirror = line.at( 16 );
1520
1521 PCB_TEXT* text = new PCB_TEXT( aBoard );
1522
1523 text->SetText( string );
1524 text->SetLayer( klayer );
1525 text->SetPosition( ScalePos( location ) );
1526 text->SetIsKnockout( inverted );
1527 text->SetTextThickness( ScaleSize( strokew ) );
1528 text->SetTextSize( VECTOR2D( ScaleSize( height * 0.6 ), ScaleSize( height * 0.7 ) ) );
1529
1530 if( font != wxS( "default" ) )
1531 {
1532 text->SetFont( KIFONT::FONT::GetFont( font ) );
1533 //text->SetupRenderCache( text->GetShownText(), text->GetFont(), EDA_ANGLE( angle, DEGREES_T ) );
1534
1535 //text->AddRenderCacheGlyph();
1536 // TODO: import geometry cache
1537 }
1538
1539 AlignText( text, align );
1540
1541 if( IsBackLayer( klayer ) ^ !!mirror )
1542 {
1543 text->SetMirrored( true );
1544 text->SetTextAngleDegrees( -angle );
1545 }
1546 else
1547 {
1548 text->SetTextAngleDegrees( angle );
1549 }
1550
1551 aBoard->Add( text, ADD_MODE::APPEND );
1552 }
1553 else if( type == wxS( "COMPONENT" ) )
1554 {
1555 wxString compId = line.at( 1 );
1556 componentLines[compId].push_back( line );
1557 }
1558 else if( type == wxS( "ATTR" ) )
1559 {
1560 wxString compId = line.at( 3 );
1561 componentLines[compId].push_back( line );
1562 }
1563 else if( type == wxS( "PAD_NET" ) )
1564 {
1565 wxString compId = line.at( 1 );
1566 componentLines[compId].push_back( line );
1567 }
1568 }
1569
1570 for( auto const& [compId, lines] : componentLines )
1571 {
1572 wxString deviceId;
1573 wxString fpIdOverride;
1574 wxString fpDesignator;
1575 std::map<wxString, wxString> localCompAttribs;
1576
1577 for( auto& line : lines )
1578 {
1579 if( line.size() == 0 )
1580 continue;
1581
1582 wxString type = line.at( 0 );
1583
1584 if( type == wxS( "COMPONENT" ) )
1585 {
1586 localCompAttribs = line.at( 7 );
1587 }
1588 else if( type == wxS( "ATTR" ) )
1589 {
1590 EASYEDAPRO::PCB_ATTR attr = line;
1591
1592 if( attr.key == wxS( "Device" ) )
1593 deviceId = attr.value;
1594
1595 else if( attr.key == wxS( "Footprint" ) )
1596 fpIdOverride = attr.value;
1597
1598 else if( attr.key == wxS( "Designator" ) )
1599 fpDesignator = attr.value;
1600 }
1601 }
1602
1603 if( deviceId.empty() )
1604 continue;
1605
1606 nlohmann::json compAttrs = aProject.at( "devices" ).at( deviceId ).at( "attributes" );
1607
1608 wxString fpId;
1609
1610 if( !fpIdOverride.IsEmpty() )
1611 fpId = fpIdOverride;
1612 else
1613 fpId = compAttrs.at( "Footprint" ).get<wxString>();
1614
1615 auto it = aFootprintMap.find( fpId );
1616 if( it == aFootprintMap.end() )
1617 {
1618 wxLogError( "Footprint of '%s' with uuid '%s' not found.", fpDesignator, fpId );
1619 continue;
1620 }
1621
1622 std::unique_ptr<FOOTPRINT>& footprintOrig = it->second;
1623 std::unique_ptr<FOOTPRINT> footprint( static_cast<FOOTPRINT*>( footprintOrig->Clone() ) );
1624
1625 wxString modelUuid, modelTitle, modelTransform;
1626
1627 if( auto val = get_opt( localCompAttribs, "3D Model" ) )
1628 modelUuid = *val;
1629 else
1630 modelUuid = compAttrs.value<wxString>( "3D Model", "" );
1631
1632 if( auto val = get_opt( localCompAttribs, "3D Model Title" ) )
1633 modelTitle = val->Trim();
1634 else
1635 modelTitle = compAttrs.value<wxString>( "3D Model Title", modelUuid ).Trim();
1636
1637 if( auto val = get_opt( localCompAttribs, "3D Model Transform" ) )
1638 modelTransform = *val;
1639 else
1640 modelTransform = compAttrs.value<wxString>( "3D Model Transform", "" );
1641
1642 fillFootprintModelInfo( footprint.get(), modelUuid, modelTitle, modelTransform );
1643
1644 footprint->SetParent( aBoard );
1645
1646 for( auto& line : lines )
1647 {
1648 if( line.size() == 0 )
1649 continue;
1650
1651 wxString type = line.at( 0 );
1652
1653 if( type == wxS( "COMPONENT" ) )
1654 {
1655 int layer = line.at( 3 );
1656 PCB_LAYER_ID klayer = LayerToKi( layer );
1657
1658 VECTOR2D center( line.at( 4 ), line.at( 5 ) );
1659
1660 double orient = line.at( 6 );
1661 //std::map<wxString, wxString> props = line.at( 7 );
1662
1663 if( klayer == B_Cu )
1664 footprint->Flip( footprint->GetPosition(), FLIP_DIRECTION::TOP_BOTTOM );
1665
1666 footprint->SetOrientationDegrees( orient );
1667 footprint->SetPosition( ScalePos( center ) );
1668 }
1669 else if( type == wxS( "ATTR" ) )
1670 {
1671 EASYEDAPRO::PCB_ATTR attr = line;
1672
1673 PCB_LAYER_ID klayer = LayerToKi( attr.layer );
1674
1675 PCB_FIELD* field = nullptr;
1676
1677 if( attr.key == wxS( "Designator" ) )
1678 {
1679 if( attr.key == wxS( "Designator" ) )
1680 {
1681 field = footprint->GetField( FIELD_T::REFERENCE );
1682 }
1683 else
1684 {
1685 field = new PCB_FIELD( footprint.get(), FIELD_T::USER, attr.key );
1686 footprint->Add( field, ADD_MODE::APPEND );
1687 }
1688
1689 if( attr.fontName != wxS( "default" ) )
1690 field->SetFont( KIFONT::FONT::GetFont( attr.fontName ) );
1691
1692 if( attr.valVisible && attr.keyVisible )
1693 {
1694 field->SetText( attr.key + ':' + attr.value );
1695 }
1696 else if( attr.keyVisible )
1697 {
1698 field->SetText( attr.key );
1699 }
1700 else
1701 {
1702 field->SetVisible( false );
1703 field->SetText( attr.value );
1704 }
1705
1706 field->SetLayer( klayer );
1707 field->SetPosition( ScalePos( attr.position ) );
1708 field->SetTextAngleDegrees( footprint->IsFlipped() ? -attr.rotation
1709 : attr.rotation );
1710 field->SetIsKnockout( attr.inverted );
1711 field->SetTextThickness( ScaleSize( attr.strokeWidth ) );
1712 field->SetTextSize( VECTOR2D( ScaleSize( attr.height * 0.55 ),
1713 ScaleSize( attr.height * 0.6 ) ) );
1714
1715 AlignText( field, attr.textOrigin );
1716 }
1717 }
1718 else if( type == wxS( "PAD_NET" ) )
1719 {
1720 wxString padNumber = line.at( 2 );
1721 wxString padNet = line.at( 3 );
1722
1723 PAD* pad = footprint->FindPadByNumber( padNumber );
1724 if( pad )
1725 {
1726 pad->SetNet( aBoard->FindNet( padNet ) );
1727 }
1728 else
1729 {
1730 // Not a pad
1731 }
1732 }
1733 }
1734
1735 aBoard->Add( footprint.release(), ADD_MODE::APPEND );
1736 }
1737
1738 // Set zone fills
1740 {
1741 for( auto& [uuid, zone] : poursToFill )
1742 {
1743 SHAPE_POLY_SET fillPolySet;
1744 SHAPE_POLY_SET thermalSpokes;
1745
1746 auto range = boardPouredMap.equal_range( uuid );
1747 for( auto& it = range.first; it != range.second; ++it )
1748 {
1749 const EASYEDAPRO::POURED& poured = it->second;
1750 int unki = poured.unki;
1751
1752 SHAPE_POLY_SET thisPoly;
1753
1754 for( int dataId = 0; dataId < poured.polyData.size(); dataId++ )
1755 {
1756 const nlohmann::json& fillData = poured.polyData[dataId];
1757 const double ptScale = 10;
1758
1759 SHAPE_LINE_CHAIN contour =
1760 ParseContour( fillData, false, ARC_HIGH_DEF / ptScale );
1761
1762 // Scale the fill
1763 for( int i = 0; i < contour.PointCount(); i++ )
1764 contour.SetPoint( i, contour.GetPoint( i ) * ptScale );
1765
1766 if( poured.isPoly )
1767 {
1768 contour.SetClosed( true );
1769
1770 // The contour can be self-intersecting
1771 SHAPE_POLY_SET simple( contour );
1772 simple.Simplify();
1773
1774 if( dataId == 0 )
1775 {
1776 thisPoly.Append( simple );
1777 }
1778 else
1779 {
1780 thisPoly.BooleanSubtract( simple );
1781 }
1782 }
1783 else
1784 {
1785 const int thermalWidth = pcbIUScale.mmToIU( 0.2 ); // Generic
1786
1787 for( int segId = 0; segId < contour.SegmentCount(); segId++ )
1788 {
1789 const SEG& seg = contour.CSegment( segId );
1790
1791 TransformOvalToPolygon( thermalSpokes, seg.A, seg.B, thermalWidth,
1793 }
1794 }
1795 }
1796
1797 fillPolySet.Append( thisPoly );
1798 }
1799
1800 if( !fillPolySet.IsEmpty() )
1801 {
1802 fillPolySet.Simplify();
1803
1804 const int strokeWidth = pcbIUScale.MilsToIU( 8 ); // Seems to be 8 mils
1805
1806 fillPolySet.Inflate( strokeWidth / 2, CORNER_STRATEGY::ROUND_ALL_CORNERS,
1807 ARC_HIGH_DEF, false );
1808
1809 fillPolySet.BooleanAdd( thermalSpokes );
1810
1811 fillPolySet.Fracture();
1812
1813 zone->SetFilledPolysList( zone->GetFirstLayer(), fillPolySet );
1814 zone->SetNeedRefill( false );
1815 zone->SetIsFilled( true );
1816 }
1817 }
1818 }
1819
1820 // Heal board outlines
1821 std::vector<PCB_SHAPE*> shapes;
1822 std::vector<std::unique_ptr<PCB_SHAPE>> newShapes;
1823
1824 for( BOARD_ITEM* item : aBoard->Drawings() )
1825 {
1826 if( !item->IsOnLayer( Edge_Cuts ) )
1827 continue;
1828
1829 if( item->Type() == PCB_SHAPE_T )
1830 shapes.push_back( static_cast<PCB_SHAPE*>( item ) );
1831 }
1832
1833 ConnectBoardShapes( shapes, newShapes, SHAPE_JOIN_DISTANCE );
1834
1835 for( std::unique_ptr<PCB_SHAPE>& ptr : newShapes )
1836 aBoard->Add( ptr.release(), ADD_MODE::APPEND );
1837
1838 // Center the board
1839 BOX2I outlineBbox = aBoard->ComputeBoundingBox( true );
1840 PAGE_INFO pageInfo = aBoard->GetPageSettings();
1841
1842 VECTOR2D pageCenter( pcbIUScale.MilsToIU( pageInfo.GetWidthMils() / 2 ),
1843 pcbIUScale.MilsToIU( pageInfo.GetHeightMils() / 2 ) );
1844
1845 VECTOR2D offset = pageCenter - outlineBbox.GetCenter();
1846
1847 int alignGrid = pcbIUScale.mmToIU( 10 );
1848 offset.x = KiROUND( offset.x / alignGrid ) * alignGrid;
1849 offset.y = KiROUND( offset.y / alignGrid ) * alignGrid;
1850
1851 aBoard->Move( offset );
1852 bds.SetAuxOrigin( offset );
1853}
@ ERROR_INSIDE
Definition: approximation.h:34
constexpr int ARC_HIGH_DEF
Definition: base_units.h:120
constexpr EDA_IU_SCALE pcbIUScale
Definition: base_units.h:108
constexpr int ARC_LOW_DEF
Definition: base_units.h:119
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
BASE_SET & set(size_t pos)
Definition: base_set.h:116
Bezier curves to polygon converter.
Definition: bezier_curves.h:38
void GetPoly(std::vector< VECTOR2I > &aOutput, int aMaxError=10)
Convert a Bezier curve to a polygon.
void Mirror(FLIP_DIRECTION aFlipDirection)
Mirror image vertically (i.e.
Container for design settings for a BOARD object.
void SetAuxOrigin(const VECTOR2I &aOrigin)
Abstract interface for BOARD_ITEMs capable of storing other items inside.
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition: board_item.h:78
virtual void SetIsKnockout(bool aKnockout)
Definition: board_item.h:317
virtual void SetLayer(PCB_LAYER_ID aLayer)
Set the layer this item is on.
Definition: board_item.h:280
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:297
LSET GetEnabledLayers() const
A proxy function that calls the corresponding function in m_BoardSettings.
Definition: board.cpp:829
void Add(BOARD_ITEM *aItem, ADD_MODE aMode=ADD_MODE::INSERT, bool aSkipConnectivity=false) override
Removes an item from the container.
Definition: board.cpp:1060
const PAGE_INFO & GetPageSettings() const
Definition: board.h:715
NETINFO_ITEM * FindNet(int aNetcode) const
Search for a net with the given netcode.
Definition: board.cpp:2011
bool SetLayerName(PCB_LAYER_ID aLayer, const wxString &aLayerName)
Changes the name of the layer given by aLayer.
Definition: board.cpp:632
void Move(const VECTOR2I &aMoveVector) override
Move this object.
Definition: board.cpp:503
BOX2I ComputeBoundingBox(bool aBoardEdgesOnly=false) const
Calculate the bounding box containing all board items (or board edge segments).
Definition: board.cpp:1763
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Definition: board.cpp:946
unsigned GetNetCount() const
Definition: board.h:928
void SetEnabledLayers(const LSET &aLayerMask)
A proxy function that calls the correspondent function in m_BoardSettings.
Definition: board.cpp:849
const DRAWINGS & Drawings() const
Definition: board.h:340
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 const Vec & GetOrigin() const
Definition: box2.h:210
constexpr const SizeVec & GetSize() const
Definition: box2.h:206
EDA_ANGLE Normalize90()
Definition: eda_angle.h:249
double AsDegrees() const
Definition: eda_angle.h:113
A mix-in class (via multiple inheritance) that handles texts such as labels, parts,...
Definition: eda_text.h:80
void SetTextSize(VECTOR2I aNewSize, bool aEnforceMinTextSize=true)
Definition: eda_text.cpp:526
virtual void SetVisible(bool aVisible)
Definition: eda_text.cpp:379
void SetTextThickness(int aWidth)
The TextThickness is that set by the user.
Definition: eda_text.cpp:284
void SetTextAngleDegrees(double aOrientation)
Definition: eda_text.h:137
virtual void SetText(const wxString &aText)
Definition: eda_text.cpp:270
void SetFont(KIFONT::FONT *aFont)
Definition: eda_text.cpp:492
PCB_FIELD * GetField(FIELD_T aFieldType)
Return a mandatory field in this footprint.
Definition: footprint.cpp:582
void Add(BOARD_ITEM *aItem, ADD_MODE aMode=ADD_MODE::INSERT, bool aSkipConnectivity=false) override
Removes an item from the container.
Definition: footprint.cpp:1069
void GetFields(std::vector< PCB_FIELD * > &aVector, bool aVisibleOnly) const
Populate a std::vector with PCB_TEXTs.
Definition: footprint.cpp:627
std::vector< FP_3DMODEL > & Models()
Definition: footprint.h:225
DRAWINGS & GraphicalItems()
Definition: footprint.h:214
VECTOR3D m_Offset
3D model offset (mm)
Definition: footprint.h:106
VECTOR3D m_Rotation
3D model rotation (degrees)
Definition: footprint.h:105
wxString m_Filename
The 3D shape filename in 3D library.
Definition: footprint.h:108
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
A logical library item identifier and consists of various portions much like a URI.
Definition: lib_id.h:49
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:37
Handle the data for a net.
Definition: netinfo.h:56
static constexpr PCB_LAYER_ID ALL_LAYERS
! Temporary layer identifier to identify code that is not padstack-aware
Definition: padstack.h:144
Definition: pad.h:54
static LSET PTHMask()
layer set for a through hole pad
Definition: pad.cpp:284
static LSET SMDMask()
layer set for a SMD pad on Front layer
Definition: pad.cpp:291
Describe the page size and margins of a paper page on which to eventually print or plot.
Definition: page_info.h:59
double GetHeightMils() const
Definition: page_info.h:141
double GetWidthMils() const
Definition: page_info.h:136
SHAPE_LINE_CHAIN ParseContour(nlohmann::json polyData, bool aInFill, double aArcAccuracy=SHAPE_ARC::DefaultAccuracyForPCB()) const
static VECTOR2< T > ScalePos(VECTOR2< T > aValue)
PCB_IO_EASYEDAPRO_PARSER(BOARD *aBoard, PROGRESS_REPORTER *aProgressReporter)
std::vector< std::unique_ptr< PCB_SHAPE > > ParsePoly(BOARD_ITEM_CONTAINER *aContainer, nlohmann::json polyData, bool aClosed, bool aInFill) const
void ParseBoard(BOARD *aBoard, const nlohmann::json &aProject, std::map< wxString, std::unique_ptr< FOOTPRINT > > &aFootprintMap, const std::map< wxString, EASYEDAPRO::BLOB > &aBlobMap, const std::multimap< wxString, EASYEDAPRO::POURED > &aPouredMap, const std::vector< nlohmann::json > &aLines, const wxString &aFpLibName)
std::unique_ptr< PAD > createPAD(FOOTPRINT *aFootprint, const nlohmann::json &line)
PCB_LAYER_ID LayerToKi(int aLayer)
FOOTPRINT * ParseFootprint(const nlohmann::json &aProject, const wxString &aFpUuid, const std::vector< nlohmann::json > &aLines)
static double Convert(wxString aValue)
void fillFootprintModelInfo(FOOTPRINT *footprint, const wxString &modelUuid, const wxString &modelTitle, const wxString &modelTransform) const
virtual void SetPosition(const VECTOR2I &aPos) override
Definition: pcb_text.h:89
A progress reporter interface for use in multi-threaded environments.
A REFERENCE_IMAGE is a wrapper around a BITMAP_IMAGE that is displayed in an editor as a reference fo...
BITMAP_BASE & MutableImage() const
Only use this if you really need to modify the underlying image.
bool ReadImageFile(const wxString &aFullFilename)
Read and store an image file.
VECTOR2I GetSize() const
void SetImageScale(double aScale)
Set the image "zoom" value.
Definition: seg.h:42
VECTOR2I A
Definition: seg.h:49
VECTOR2I B
Definition: seg.h:50
SHAPE_ARC & ConstructFromStartEndCenter(const VECTOR2I &aStart, const VECTOR2I &aEnd, const VECTOR2I &aCenter, bool aClockwise=false, double aWidth=0)
Constructs this arc from the given start, end and center.
Definition: shape_arc.cpp:213
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
virtual const VECTOR2I GetPoint(int aIndex) const override
void SetPoint(int aIndex, const VECTOR2I &aPos)
Move a point to a specific location.
void SetClosed(bool aClosed)
Mark the line chain as closed (i.e.
int PointCount() const
Return the number of points (vertices) in this line chain.
SEG Segment(int aIndex) const
Return a copy of the aIndex-th segment in the line chain.
void Append(int aX, int aY, bool aAllowDuplication=false)
Append a new point at the end of the line chain.
int SegmentCount() const
Return the number of segments in this line chain.
const SEG CSegment(int aIndex) const
Return a constant copy of the aIndex segment in the line chain.
const BOX2I BBox(int aClearance=0) const override
Compute a bounding box of the shape, with a margin of aClearance a collision.
Represent a set of closed polygons.
void BooleanAdd(const SHAPE_POLY_SET &b)
Perform boolean polyset union.
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 Fracture()
Convert a set of polygons with holes to a single outline with "slits"/"fractures" connecting the oute...
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)
std::vector< SHAPE_LINE_CHAIN > POLYGON
represents a single polygon outline with holes.
void RebuildHolesFromContours()
Extract all contours from this polygon set, then recreate polygons with holes.
int OutlineCount() const
Return the number of outlines in the set.
void BooleanSubtract(const SHAPE_POLY_SET &b)
Perform boolean polyset difference.
const std::vector< POLYGON > & CPolygons() const
const BOX2I BBox(int aClearance=0) const override
Compute a bounding box of the shape, with a margin of aClearance a collision.
const SHAPE_LINE_CHAIN Outline() const
Definition: shape_rect.h:212
T y
Definition: vector3.h:64
T z
Definition: vector3.h:65
T x
Definition: vector3.h:63
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 TransformRoundChamferedRectToPolygon(SHAPE_POLY_SET &aBuffer, const VECTOR2I &aPosition, const VECTOR2I &aSize, const EDA_ANGLE &aRotation, int aCornerRadius, double aChamferRatio, int aChamferCorners, int aInflate, int aError, ERROR_LOC aErrorLoc)
Convert a rectangle with rounded corners and/or chamfered corners to a polygon.
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 _(s)
static constexpr EDA_ANGLE ANGLE_0
Definition: eda_angle.h:401
static constexpr EDA_ANGLE ANGLE_90
Definition: eda_angle.h:403
@ DEGREES_T
Definition: eda_angle.h:31
void ConnectBoardShapes(std::vector< PCB_SHAPE * > &aShapeList, std::vector< std::unique_ptr< PCB_SHAPE > > &aNewShapes, int aChainingEpsilon)
Connects shapes to each other, making continious contours (adjacent shapes will have a common vertex)...
#define THROW_IO_ERROR(msg)
Definition: ki_exception.h:39
bool IsBackLayer(PCB_LAYER_ID aLayerId)
Layer classification: check if it's a back layer.
Definition: layer_ids.h:788
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ In22_Cu
Definition: layer_ids.h:87
@ In11_Cu
Definition: layer_ids.h:76
@ In29_Cu
Definition: layer_ids.h:94
@ In30_Cu
Definition: layer_ids.h:95
@ In17_Cu
Definition: layer_ids.h:82
@ Edge_Cuts
Definition: layer_ids.h:112
@ Dwgs_User
Definition: layer_ids.h:107
@ F_Paste
Definition: layer_ids.h:104
@ In9_Cu
Definition: layer_ids.h:74
@ Cmts_User
Definition: layer_ids.h:108
@ User_6
Definition: layer_ids.h:129
@ User_7
Definition: layer_ids.h:130
@ In19_Cu
Definition: layer_ids.h:84
@ In7_Cu
Definition: layer_ids.h:72
@ In28_Cu
Definition: layer_ids.h:93
@ In26_Cu
Definition: layer_ids.h:91
@ B_Mask
Definition: layer_ids.h:98
@ B_Cu
Definition: layer_ids.h:65
@ User_5
Definition: layer_ids.h:128
@ F_Mask
Definition: layer_ids.h:97
@ In21_Cu
Definition: layer_ids.h:86
@ In23_Cu
Definition: layer_ids.h:88
@ B_Paste
Definition: layer_ids.h:105
@ In15_Cu
Definition: layer_ids.h:80
@ In2_Cu
Definition: layer_ids.h:67
@ F_Fab
Definition: layer_ids.h:119
@ In10_Cu
Definition: layer_ids.h:75
@ F_SilkS
Definition: layer_ids.h:100
@ In4_Cu
Definition: layer_ids.h:69
@ Eco2_User
Definition: layer_ids.h:110
@ In16_Cu
Definition: layer_ids.h:81
@ In24_Cu
Definition: layer_ids.h:89
@ In1_Cu
Definition: layer_ids.h:66
@ User_3
Definition: layer_ids.h:126
@ User_1
Definition: layer_ids.h:124
@ B_SilkS
Definition: layer_ids.h:101
@ In13_Cu
Definition: layer_ids.h:78
@ User_4
Definition: layer_ids.h:127
@ In8_Cu
Definition: layer_ids.h:73
@ In14_Cu
Definition: layer_ids.h:79
@ User_2
Definition: layer_ids.h:125
@ In12_Cu
Definition: layer_ids.h:77
@ In27_Cu
Definition: layer_ids.h:92
@ In6_Cu
Definition: layer_ids.h:71
@ In5_Cu
Definition: layer_ids.h:70
@ In3_Cu
Definition: layer_ids.h:68
@ In20_Cu
Definition: layer_ids.h:85
@ F_Cu
Definition: layer_ids.h:64
@ In18_Cu
Definition: layer_ids.h:83
@ In25_Cu
Definition: layer_ids.h:90
@ B_Fab
Definition: layer_ids.h:118
wxString get_def(const std::map< wxString, wxString > &aMap, const char *aKey, const char *aDefval="")
Definition: map_helpers.h:64
std::optional< V > get_opt(const std::map< wxString, V > &aMap, const wxString &aKey)
Definition: map_helpers.h:34
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
LIB_ID ToKiCadLibID(const wxString &aLibName, const wxString &aLibReference)
static const bool IMPORT_POURED
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
Definition: eda_angle.h:390
Class to handle a set of BOARD_ITEMs.
static const wxString MODEL_SIZE_KEY
static const int SHAPE_JOIN_DISTANCE
static void AlignText(EDA_TEXT *text, int align)
static const wxString MODEL_SIZE_KEY
static const int SHAPE_JOIN_DISTANCE
static const wxString QUERY_MODEL_UUID_KEY
static bool addSegment(VRML_LAYER &model, IDF_SEGMENT *seg, int icont, int iseg)
const int scale
wxString EscapeString(const wxString &aSource, ESCAPE_CONTEXT aContext)
The Escape/Unescape routines use HTML-entity-reference-style encoding to handle characters which are:...
nlohmann::json polyData
constexpr double IUTomm(int iu) const
Definition: base_units.h:86
constexpr int MilsToIU(int mils) const
Definition: base_units.h:93
constexpr int mmToIU(double mm) const
Definition: base_units.h:88
VECTOR2I center
const SHAPE_LINE_CHAIN chain
VECTOR2I end
VECTOR2I location
constexpr int delta
@ GR_TEXT_H_ALIGN_CENTER
@ GR_TEXT_H_ALIGN_RIGHT
@ GR_TEXT_H_ALIGN_LEFT
@ GR_TEXT_V_ALIGN_BOTTOM
@ GR_TEXT_V_ALIGN_CENTER
@ GR_TEXT_V_ALIGN_TOP
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
double DEG2RAD(double deg)
Definition: trigo.h:166
@ PCB_SHAPE_T
class PCB_SHAPE, a segment not on copper layers
Definition: typeinfo.h:88
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695
VECTOR2< double > VECTOR2D
Definition: vector2d.h:694