30#include <nlohmann/json.hpp>
35#include <wx/wfstream.h>
36#include <wx/stdstream.h>
74 if( !aValue.ToCDouble( &value ) )
75 THROW_IO_ERROR( wxString::Format(
_(
"Failed to parse value: '%s'" ), aValue ) );
104 case 9:
return F_Fab;
105 case 10:
return B_Fab;
204 const wxString& modelUuid,
205 const wxString& modelTitle,
206 const wxString& modelTransform )
const
209 const wxString easyedaModelDir = wxS(
"EASYEDA_MODELS" );
210 const wxString kicadModelPrefix = wxS(
"${KIPRJMOD}/" ) + easyedaModelDir + wxS(
"/" );
221 footprint->
Add( field );
226 wxArrayString arr = wxSplit( modelTransform,
',',
'\0' );
231 if( fitXmm > 0.0 && fitYmm > 0.0 )
236 field->
SetText( wxString::FromCDouble( fitXmm ) + wxS(
" " )
237 + wxString::FromCDouble( fitYmm ) );
238 footprint->
Add( field );
241 kmodelRotation.
z = -
Convert( arr[3] );
242 kmodelRotation.
x = -
Convert( arr[4] );
243 kmodelRotation.
y = -
Convert( arr[5] );
250 if( !modelTitle.IsEmpty() && footprint->
Models().empty() )
254 +
EscapeString( modelTitle, ESCAPE_CONTEXT::CTX_FILENAME )
258 footprint->
Models().push_back( model );
263std::vector<std::unique_ptr<PCB_SHAPE>>
265 bool aClosed,
bool aInFill )
const
267 std::vector<std::unique_ptr<PCB_SHAPE>> results;
270 for(
int i = 0; i < polyData.size(); i++ )
272 nlohmann::json val = polyData.at( i );
274 if( val.is_string() )
277 if( str == wxS(
"CIRCLE" ) )
280 center.
x = ( polyData.at( ++i ) );
281 center.
y = ( polyData.at( ++i ) );
282 double r = ( polyData.at( ++i ) );
284 std::unique_ptr<PCB_SHAPE> shape =
285 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::CIRCLE );
289 shape->SetFilled( aClosed );
291 results.emplace_back( std::move( shape ) );
293 else if( str == wxS(
"R" ) )
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;
305 std::unique_ptr<PCB_SHAPE> shape =
306 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::RECTANGLE );
308 shape->SetStart(
ScalePos( start ) );
309 shape->SetEnd(
ScalePos( start + size ) );
310 shape->SetFilled( aClosed );
313 results.emplace_back( std::move( shape ) );
321 std::unique_ptr<PCB_SHAPE> shape =
322 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::SEGMENT );
324 shape->SetStart(
ScalePos( aStart ) );
326 shape->SetFilled( aClosed );
329 results.emplace_back( std::move( shape ) );
334 std::unique_ptr<PCB_SHAPE> shape =
335 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::ARC );
337 shape->SetStart(
ScalePos( aStart ) );
340 shape->SetFilled( aClosed );
343 results.emplace_back( std::move( shape ) );
351 addArc( {
end.
x - cr, start.
y }, {
end.
x, start.
y - cr },
352 {
end.
x - cr, start.
y - cr } );
357 addArc( { start.
x + cr,
end.
y }, { start.
x,
end.
y + cr },
358 { start.
x + cr,
end.
y + cr } );
360 addArc( { start.
x, start.
y - cr }, { start.
x + cr, start.
y },
361 { start.
x + cr, start.
y - cr } );
364 else if( str == wxS(
"ARC" ) || str == wxS(
"CARC" ) )
367 double angle = polyData.at( ++i ).get<
double>() / ( aInFill ? 10 : 1 );
368 end.
x = ( polyData.at( ++i ) );
369 end.
y = ( polyData.at( ++i ) );
371 std::unique_ptr<PCB_SHAPE> shape =
372 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::ARC );
376 shape->SetStart(
ScalePos( prevPt ) );
382 shape->SetEnd(
ScalePos( prevPt ) );
388 double ha = angle / 2;
389 double hd =
delta.EuclideanNorm() / 2;
390 double cdist = hd / tan(
DEG2RAD( ha ) );
394 shape->SetFilled( aClosed );
396 results.emplace_back( std::move( shape ) );
400 else if( str == wxS(
"L" ) )
405 while( i < polyData.size() - 2 && polyData.at( i + 1 ).is_number() )
408 pt.
x = ( polyData.at( ++i ) );
409 pt.
y = ( polyData.at( ++i ) );
418 std::unique_ptr<PCB_SHAPE> shape =
419 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::POLY );
426 shape->SetFilled(
true );
427 shape->SetPolyShape(
chain );
429 results.emplace_back( std::move( shape ) );
438 std::unique_ptr<PCB_SHAPE> shape =
439 std::make_unique<PCB_SHAPE>( aContainer, SHAPE_T::SEGMENT );
441 shape->SetStart( seg.
A );
442 shape->SetEnd( seg.
B );
444 results.emplace_back( std::move( shape ) );
449 else if( val.is_number() )
451 prevPt.
x = ( polyData.at( i ) );
452 prevPt.
y = ( polyData.at( ++i ) );
462 double aArcAccuracy )
const
467 for(
int i = 0; i < polyData.size(); i++ )
469 nlohmann::json val = polyData.at( i );
471 if( val.is_string() )
474 if( str == wxS(
"CIRCLE" ) )
477 center.
x = ( polyData.at( ++i ) );
478 center.
y = ( polyData.at( ++i ) );
479 double r = ( polyData.at( ++i ) );
484 else if( str == wxS(
"R" ) )
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;
498 VECTOR2D kcenter = kstart + ksize / 2;
505 result.
Append( poly.Outline( 0 ) );
507 else if( str == wxS(
"ARC" ) || str == wxS(
"CARC" ) )
510 double angle = polyData.at( ++i ).get<
double>();
515 end.
x = ( polyData.at( ++i ) );
516 end.
y = ( polyData.at( ++i ) );
525 double ha = angle / 2;
526 double hd =
delta.EuclideanNorm() / 2;
527 double cdist = hd / tan(
DEG2RAD( ha ) );
534 result.
Append( sarc, aArcAccuracy );
538 else if( str == wxS(
"C" ) )
541 pt1.
x = ( polyData.at( ++i ) );
542 pt1.
y = ( polyData.at( ++i ) );
545 pt2.
x = ( polyData.at( ++i ) );
546 pt2.
y = ( polyData.at( ++i ) );
549 pt3.
x = ( polyData.at( ++i ) );
550 pt3.
y = ( polyData.at( ++i ) );
556 std::vector<VECTOR2I> bezierPoints;
557 converter.
GetPoly( bezierPoints, aArcAccuracy );
559 result.
Append( bezierPoints );
563 else if( str == wxS(
"L" ) )
567 while( i < polyData.size() - 2 && polyData.at( i + 1 ).is_number() )
570 pt.
x = ( polyData.at( ++i ) );
571 pt.
y = ( polyData.at( ++i ) );
579 else if( val.is_number() )
581 prevPt.
x = ( polyData.at( i ) );
582 prevPt.
y = ( polyData.at( ++i ) );
591 const nlohmann::json& line )
593 wxString uuid = line.at( 1 );
600 wxString netname = line.at( 3 );
601 int layer = line.at( 4 ).get<
int>();
604 wxString padNumber = line.at( 5 );
610 double orientation = line.at( 8 );
612 nlohmann::json padHole = line.at( 9 );
613 nlohmann::json padShape = line.at( 10 );
615 std::unique_ptr<PAD>
pad = std::make_unique<PAD>( aFootprint );
617 pad->SetNumber( padNumber );
619 pad->SetOrientationDegrees( orientation );
621 if( !padHole.is_null() )
623 double drill_dir = 0;
625 if( line.at( 14 ).is_number() )
626 drill_dir = line.at( 14 );
628 wxString holeShape = padHole.at( 0 );
630 if( holeShape == wxS(
"ROUND" ) || holeShape == wxS(
"SLOT" ) )
633 drill.
x = padHole.at( 1 );
634 drill.
y = padHole.at( 2 );
639 std::swap( drill.
x, drill.
y );
641 if( holeShape == wxS(
"SLOT" ) )
643 pad->SetDrillShape( PAD_DRILL_SHAPE::OBLONG );
648 pad->SetAttribute( PAD_ATTRIB::PTH );
657 else if( klayer ==
B_Cu )
662 pad->SetAttribute( PAD_ATTRIB::SMD );
665 wxString padSh = padShape.at( 0 );
666 if( padSh == wxS(
"RECT" ) )
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;
685 else if( padSh == wxS(
"ELLIPSE" ) )
688 size.
x = padShape.at( 1 );
689 size.
y = padShape.at( 2 );
694 else if( padSh == wxS(
"OVAL" ) )
697 size.
x = padShape.at( 1 );
698 size.
y = padShape.at( 2 );
703 else if( padSh == wxS(
"POLY" ) )
709 nlohmann::json polyData = padShape.at( 1 );
711 std::vector<std::unique_ptr<PCB_SHAPE>> results =
712 ParsePoly( aFootprint, polyData,
true,
false );
714 for(
auto& shape : results )
716 shape->SetLayer( klayer );
717 shape->SetWidth( 0 );
719 shape->Move( -
pad->GetPosition() );
727 return std::move(
pad );
732 const wxString& aFpUuid,
733 const std::vector<nlohmann::json>& aLines )
735 std::unique_ptr<FOOTPRINT> footprintPtr = std::make_unique<FOOTPRINT>(
m_board );
736 FOOTPRINT* footprint = footprintPtr.get();
743 field->SetTextSize( defaultTextSize );
744 field->SetTextThickness( defaultTextThickness );
747 for(
const nlohmann::json& line : aLines )
749 if( line.size() == 0 )
752 wxString type = line.at( 0 );
754 if( type == wxS(
"POLY" ) || type == wxS(
"PAD" ) || type == wxS(
"FILL" )
755 || type == wxS(
"ATTR" ) )
757 wxString uuid = line.at( 1 );
764 wxString netname = line.at( 3 );
765 int layer = line.at( 4 ).get<
int>();
768 if( type == wxS(
"POLY" ) )
770 double thickness = ( line.at( 5 ) );
771 nlohmann::json polyData = line.at( 6 );
773 std::vector<std::unique_ptr<PCB_SHAPE>> results =
774 ParsePoly( footprint, polyData,
false,
false );
776 for(
auto& shape : results )
778 shape->SetLayer( klayer );
779 shape->SetWidth(
ScaleSize( thickness ) );
781 footprint->
Add( shape.release(), ADD_MODE::APPEND );
784 else if( type == wxS(
"PAD" ) )
786 std::unique_ptr<PAD>
pad =
createPAD( footprint, line );
788 footprint->
Add(
pad.release(), ADD_MODE::APPEND );
790 else if( type == wxS(
"FILL" ) )
792 int layer = line.at( 4 ).get<
int>();
795 double width = line.at( 5 );
797 nlohmann::json polyDataList = line.at( 7 );
799 if( !polyDataList.is_null() && !polyDataList.empty() )
801 if( !polyDataList.at( 0 ).is_array() )
802 polyDataList = nlohmann::json::array( { polyDataList } );
804 std::vector<SHAPE_LINE_CHAIN> contours;
805 for( nlohmann::json& polyData : polyDataList )
810 contours.push_back( contour );
820 std::unique_ptr<PCB_GROUP>
group;
823 group = std::make_unique<PCB_GROUP>( footprint );
829 std::unique_ptr<PCB_SHAPE> shape =
830 std::make_unique<PCB_SHAPE>( footprint, SHAPE_T::POLY );
832 shape->SetFilled(
true );
833 shape->SetPolyShape( poly );
834 shape->SetLayer( klayer );
835 shape->SetWidth( 0 );
838 group->AddItem( shape.get() );
840 footprint->
Add( shape.release(), ADD_MODE::APPEND );
844 footprint->
Add(
group.release(), ADD_MODE::APPEND );
847 else if( type == wxS(
"ATTR" ) )
851 if( attr.
key == wxS(
"Designator" ) )
855 else if( type == wxS(
"REGION" ) )
857 wxString uuid = line.at( 1 );
864 int layer = line.at( 3 ).get<
int>();
867 double width = line.at( 4 );
868 std::set<int> flags = line.at( 5 );
869 nlohmann::json polyDataList = line.at( 6 );
871 for( nlohmann::json& polyData : polyDataList )
875 std::vector<std::unique_ptr<PCB_SHAPE>> results =
876 ParsePoly(
nullptr, polyData,
true,
false );
878 for(
auto& shape : results )
880 shape->SetFilled(
true );
887 std::unique_ptr<ZONE> zone = std::make_unique<ZONE>( footprint );
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 ) );
897 zone->SetLayer( klayer );
898 zone->Outline()->Append( polySet );
900 footprint->
Add( zone.release(), ADD_MODE::APPEND );
905 if( aProject.is_object() && aProject.contains(
"devices" ) )
907 std::map<wxString, EASYEDAPRO::PRJ_DEVICE> devicesMap = aProject.at(
"devices" );
908 std::map<wxString, wxString> compAttrs;
910 for(
auto& [devUuid, devData] : devicesMap )
912 if(
auto fp =
get_opt( devData.attributes,
"Footprint" ) )
916 compAttrs = devData.attributes;
922 wxString modelUuid, modelTitle, modelTransform;
924 modelUuid =
get_def( compAttrs,
"3D Model",
"" );
925 modelTitle =
get_def( compAttrs,
"3D Model Title", modelUuid );
926 modelTransform =
get_def( compAttrs,
"3D Model Transform",
"" );
932 std::vector<PCB_SHAPE*> shapes;
933 std::vector<std::unique_ptr<PCB_SHAPE>> newShapes;
941 shapes.push_back(
static_cast<PCB_SHAPE*
>( item ) );
946 for( std::unique_ptr<PCB_SHAPE>& ptr : newShapes )
947 footprint->
Add( ptr.release(), ADD_MODE::APPEND );
949 return footprintPtr.release();
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 )
960 std::map<wxString, std::vector<nlohmann::json>> componentLines;
961 std::map<wxString, std::vector<nlohmann::json>> ruleLines;
963 std::multimap<wxString, EASYEDAPRO::POURED> boardPouredMap = aPouredMap;
964 std::map<wxString, ZONE*> poursToFill;
968 for(
const nlohmann::json& line : aLines )
970 if( line.size() == 0 )
973 wxString type = line.at( 0 );
975 if( type == wxS(
"LAYER" ) )
977 int layer = line.at( 1 );
980 wxString layerType = line.at( 2 );
981 wxString layerName = line.at( 3 );
982 int layerFlag = line.at( 4 );
987 blayers.
set( klayer );
992 else if( type == wxS(
"NET" ) )
994 wxString netname = line.at( 1 );
999 else if( type == wxS(
"RULE" ) )
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 );
1006 if( ruleType == wxS(
"3" ) && isDefault )
1008 wxString units = ruleData.at( 0 );
1009 double minVal = ruleData.at( 1 );
1010 double optVal = ruleData.at( 2 );
1011 double maxVal = ruleData.at( 3 );
1015 else if( ruleType == wxS(
"1" ) && isDefault )
1017 wxString units = ruleData.at( 0 );
1018 nlohmann::json
table = ruleData.at( 1 );
1020 int minVal = INT_MAX;
1021 for(
const std::vector<int>& arr :
table )
1023 for(
int val : arr )
1033 ruleLines[ruleType].push_back( line );
1035 else if( type == wxS(
"POURED" ) )
1037 if( !line.at( 2 ).is_string() )
1041 boardPouredMap.emplace( poured.
parentId, poured );
1043 else if( type == wxS(
"VIA" ) || type == wxS(
"LINE" ) || type == wxS(
"ARC" )
1044 || type == wxS(
"POLY" ) || type == wxS(
"FILL" ) || type == wxS(
"POUR" ) )
1046 wxString uuid = line.at( 1 );
1053 wxString netname = line.at( 3 );
1055 if( type == wxS(
"VIA" ) )
1061 double drill = line.at( 7 );
1062 double dia = line.at( 8 );
1064 std::unique_ptr<PCB_VIA>
via = std::make_unique<PCB_VIA>( aBoard );
1072 aBoard->
Add(
via.release(), ADD_MODE::APPEND );
1074 else if( type == wxS(
"LINE" ) )
1076 int layer = line.at( 4 ).get<
int>();
1080 start.
x = line.at( 5 );
1081 start.
y = line.at( 6 );
1084 end.
x = line.at( 7 );
1085 end.
y = line.at( 8 );
1087 double width = line.at( 9 );
1089 std::unique_ptr<PCB_TRACK> track = std::make_unique<PCB_TRACK>( aBoard );
1091 track->SetLayer( klayer );
1092 track->SetStart(
ScalePos( start ) );
1096 track->SetNet( aBoard->
FindNet( netname ) );
1098 aBoard->
Add( track.release(), ADD_MODE::APPEND );
1100 else if( type == wxS(
"ARC" ) )
1102 int layer = line.at( 4 ).get<
int>();
1106 start.
x = line.at( 5 );
1107 start.
y = line.at( 6 );
1110 end.
x = line.at( 7 );
1111 end.
y = line.at( 8 );
1113 double angle = line.at( 9 );
1114 double width = line.at( 10 );
1119 double ha = angle / 2;
1120 double hd =
delta.EuclideanNorm() / 2;
1121 double cdist = hd / tan(
DEG2RAD( ha ) );
1128 std::unique_ptr<PCB_ARC> arc = std::make_unique<PCB_ARC>( aBoard, &sarc );
1131 arc->SetLayer( klayer );
1132 arc->SetNet( aBoard->
FindNet( netname ) );
1134 aBoard->
Add( arc.release(), ADD_MODE::APPEND );
1136 else if( type == wxS(
"FILL" ) )
1138 int layer = line.at( 4 ).get<
int>();
1141 double width = line.at( 5 );
1143 nlohmann::json polyDataList = line.at( 7 );
1145 if( !polyDataList.at( 0 ).is_array() )
1146 polyDataList = nlohmann::json::array( { polyDataList } );
1148 std::vector<SHAPE_LINE_CHAIN> contours;
1149 for( nlohmann::json& polyData : polyDataList )
1154 contours.push_back( contour );
1165 std::unique_ptr<ZONE> zone = std::make_unique<ZONE>( aBoard );
1167 zone->SetNet( aBoard->
FindNet( netname ) );
1168 zone->SetLayer( klayer );
1170 zone->SetFilledPolysList( klayer, zoneFillPoly );
1171 zone->SetAssignedPriority( 500 );
1172 zone->SetIsFilled(
true );
1173 zone->SetNeedRefill(
false );
1178 aBoard->
Add( zone.release(), ADD_MODE::APPEND );
1180 else if( type == wxS(
"POLY" ) )
1182 int layer = line.at( 4 );
1185 double thickness = line.at( 5 );
1186 nlohmann::json polyData = line.at( 6 );
1188 std::vector<std::unique_ptr<PCB_SHAPE>> results =
1189 ParsePoly( aBoard, polyData,
false,
false );
1191 for(
auto& shape : results )
1193 shape->SetLayer( klayer );
1194 shape->SetWidth(
ScaleSize( thickness ) );
1196 aBoard->
Add( shape.release(), ADD_MODE::APPEND );
1199 else if( type == wxS(
"POUR" ) )
1201 int layer = line.at( 4 ).get<
int>();
1204 double lineWidth = line.at( 5 );
1205 wxString pourname = line.at( 6 );
1206 int fillOrder = line.at( 7 ).get<
int>();
1207 nlohmann::json polyDataList = line.at( 8 );
1209 std::unique_ptr<ZONE> zone = std::make_unique<ZONE>( aBoard );
1211 zone->SetNet( aBoard->
FindNet( netname ) );
1212 zone->SetLayer( klayer );
1213 zone->SetAssignedPriority( 500 - fillOrder );
1217 for( nlohmann::json& polyData : polyDataList )
1222 zone->Outline()->Append( contour );
1225 wxASSERT( zone->Outline()->OutlineCount() == 1 );
1227 poursToFill.emplace( uuid, zone.get() );
1229 aBoard->
Add( zone.release(), ADD_MODE::APPEND );
1232 else if( type == wxS(
"TEARDROP" ) )
1234 wxString uuid = line.at( 1 );
1235 wxString netname = line.at( 2 );
1236 int layer = line.at( 3 ).get<
int>();
1239 nlohmann::json polyData = line.at( 4 );
1244 std::unique_ptr<ZONE> zone = std::make_unique<ZONE>( aBoard );
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 );
1253 zone->SetAssignedPriority( 600 );
1254 zone->SetLocalClearance( 0 );
1255 zone->SetMinThickness( 0 );
1256 zone->SetTeardropAreaType( TEARDROP_TYPE::TD_UNSPECIFIED );
1258 aBoard->
Add( zone.release(), ADD_MODE::APPEND );
1260 else if( type == wxS(
"REGION" ) )
1262 wxString uuid = line.at( 1 );
1269 int layer = line.at( 3 ).get<
int>();
1272 double width = line.at( 4 );
1273 std::set<int> flags = line.at( 5 );
1274 nlohmann::json polyDataList = line.at( 6 );
1276 for( nlohmann::json& polyData : polyDataList )
1281 std::unique_ptr<ZONE> zone = std::make_unique<ZONE>( aBoard );
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 ) );
1291 zone->SetLayer( klayer );
1292 zone->Outline()->Append( contour );
1294 aBoard->
Add( zone.release(), ADD_MODE::APPEND );
1297 else if( type == wxS(
"PAD" ) )
1299 wxString netname = line.at( 3 );
1301 std::unique_ptr<FOOTPRINT> footprint = std::make_unique<FOOTPRINT>( aBoard );
1302 std::unique_ptr<PAD>
pad =
createPAD( footprint.get(), line );
1312 footprint->Add(
pad.release(), ADD_MODE::APPEND );
1313 footprint->SetPosition( pos );
1314 footprint->SetOrientation( orient );
1316 wxString fpName = wxS(
"Pad_" ) + line.at( 1 ).get<wxString>();
1319 footprint->SetFPID( fpID );
1320 footprint->Reference().SetVisible(
true );
1321 footprint->Value().SetVisible(
true );
1322 footprint->AutoPositionFields();
1324 aBoard->
Add( footprint.release(), ADD_MODE::APPEND );
1326 else if( type == wxS(
"IMAGE" ) )
1328 wxString uuid = line.at( 1 );
1335 int layer = line.at( 3 ).get<
int>();
1338 VECTOR2D start( line.at( 4 ), line.at( 5 ) );
1339 VECTOR2D size( line.at( 6 ), line.at( 7 ) );
1341 double angle = line.at( 8 );
1342 int mirror = line.at( 9 );
1343 nlohmann::json polyDataList = line.at( 10 );
1346 std::vector<SHAPE_LINE_CHAIN> contours;
1347 for( nlohmann::json& polyData : polyDataList )
1352 contours.push_back( contour );
1364 for(
int i = 0; i < contour.PointCount(); i++ )
1375 std::unique_ptr<PCB_GROUP>
group;
1378 group = std::make_unique<PCB_GROUP>( aBoard );
1384 std::unique_ptr<PCB_SHAPE> shape =
1385 std::make_unique<PCB_SHAPE>( aBoard, SHAPE_T::POLY );
1387 shape->SetFilled(
true );
1388 shape->SetPolyShape( poly );
1389 shape->SetLayer( klayer );
1390 shape->SetWidth( 0 );
1398 ? FLIP_DIRECTION::TOP_BOTTOM
1399 : FLIP_DIRECTION::LEFT_RIGHT;
1400 shape->Mirror(
ScalePos( start ), flipDirection );
1404 group->AddItem( shape.get() );
1406 aBoard->
Add( shape.release(), ADD_MODE::APPEND );
1410 aBoard->
Add(
group.release(), ADD_MODE::APPEND );
1412 else if( type == wxS(
"OBJ" ) )
1415 wxString mimeType, base64Data;
1419 if( !line.at( 3 ).is_number() )
1422 int layer = line.at( 3 ).get<
int>();
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 );
1430 wxString imageUrl = line.at( 11 );
1432 if( imageUrl.BeforeFirst(
':' ) == wxS(
"blob" ) )
1434 wxString objectId = imageUrl.AfterLast(
':' );
1436 if(
auto blob =
get_opt( aBlobMap, objectId ) )
1438 wxString blobUrl = blob->url;
1440 if( blobUrl.BeforeFirst(
':' ) == wxS(
"data" ) )
1442 wxArrayString paramsArr =
1443 wxSplit( blobUrl.AfterFirst(
':' ).BeforeFirst(
',' ),
';',
'\0' );
1445 base64Data = blobUrl.AfterFirst(
',' );
1447 if( paramsArr.size() > 0 )
1448 mimeType = paramsArr[0];
1456 if( mimeType.empty() || base64Data.empty() )
1459 wxMemoryBuffer buf = wxBase64Decode( base64Data );
1461 if( mimeType == wxS(
"image/svg+xml" ) )
1467 VECTOR2D kcenter = kstart + ksize / 2;
1469 std::unique_ptr<PCB_REFERENCE_IMAGE> bitmap =
1470 std::make_unique<PCB_REFERENCE_IMAGE>( aBoard, kcenter, klayer );
1473 wxImage::SetDefaultLoadFlags( wxImage::GetDefaultLoadFlags()
1474 & ~wxImage::Load_Verbose );
1486 int x = bitmap->GetPosition().x;
1493 aBoard->
Add( bitmap.release(), ADD_MODE::APPEND );
1497 else if( type == wxS(
"STRING" ) )
1499 wxString uuid = line.at( 1 );
1506 int layer = line.at( 3 ).get<
int>();
1510 wxString
string = line.at( 6 );
1511 wxString font = line.at( 7 );
1513 double height = line.at( 8 );
1514 double strokew = line.at( 9 );
1516 int align = line.at( 12 );
1517 double angle = line.at( 13 );
1518 int inverted = line.at( 14 );
1519 int mirror = line.at( 16 );
1523 text->SetText(
string );
1524 text->SetLayer( klayer );
1526 text->SetIsKnockout( inverted );
1530 if( font != wxS(
"default" ) )
1543 text->SetMirrored(
true );
1544 text->SetTextAngleDegrees( -angle );
1548 text->SetTextAngleDegrees( angle );
1551 aBoard->
Add(
text, ADD_MODE::APPEND );
1553 else if( type == wxS(
"COMPONENT" ) )
1555 wxString compId = line.at( 1 );
1556 componentLines[compId].push_back( line );
1558 else if( type == wxS(
"ATTR" ) )
1560 wxString compId = line.at( 3 );
1561 componentLines[compId].push_back( line );
1563 else if( type == wxS(
"PAD_NET" ) )
1565 wxString compId = line.at( 1 );
1566 componentLines[compId].push_back( line );
1570 for(
auto const& [compId, lines] : componentLines )
1573 wxString fpIdOverride;
1574 wxString fpDesignator;
1575 std::map<wxString, wxString> localCompAttribs;
1577 for(
auto& line : lines )
1579 if( line.size() == 0 )
1582 wxString type = line.at( 0 );
1584 if( type == wxS(
"COMPONENT" ) )
1586 localCompAttribs = line.at( 7 );
1588 else if( type == wxS(
"ATTR" ) )
1592 if( attr.
key == wxS(
"Device" ) )
1593 deviceId = attr.
value;
1595 else if( attr.
key == wxS(
"Footprint" ) )
1596 fpIdOverride = attr.
value;
1598 else if( attr.
key == wxS(
"Designator" ) )
1599 fpDesignator = attr.
value;
1603 if( deviceId.empty() )
1606 nlohmann::json compAttrs = aProject.at(
"devices" ).at( deviceId ).at(
"attributes" );
1610 if( !fpIdOverride.IsEmpty() )
1611 fpId = fpIdOverride;
1613 fpId = compAttrs.at(
"Footprint" ).get<wxString>();
1615 auto it = aFootprintMap.find( fpId );
1616 if( it == aFootprintMap.end() )
1618 wxLogError(
"Footprint of '%s' with uuid '%s' not found.", fpDesignator, fpId );
1622 std::unique_ptr<FOOTPRINT>& footprintOrig = it->second;
1623 std::unique_ptr<FOOTPRINT> footprint(
static_cast<FOOTPRINT*
>( footprintOrig->Clone() ) );
1625 wxString modelUuid, modelTitle, modelTransform;
1627 if(
auto val =
get_opt( localCompAttribs,
"3D Model" ) )
1630 modelUuid = compAttrs.value<wxString>(
"3D Model",
"" );
1632 if(
auto val =
get_opt( localCompAttribs,
"3D Model Title" ) )
1633 modelTitle = val->Trim();
1635 modelTitle = compAttrs.value<wxString>(
"3D Model Title", modelUuid ).Trim();
1637 if(
auto val =
get_opt( localCompAttribs,
"3D Model Transform" ) )
1638 modelTransform = *val;
1640 modelTransform = compAttrs.value<wxString>(
"3D Model Transform",
"" );
1644 footprint->SetParent( aBoard );
1646 for(
auto& line : lines )
1648 if( line.size() == 0 )
1651 wxString type = line.at( 0 );
1653 if( type == wxS(
"COMPONENT" ) )
1655 int layer = line.at( 3 );
1660 double orient = line.at( 6 );
1663 if( klayer ==
B_Cu )
1664 footprint->Flip( footprint->GetPosition(), FLIP_DIRECTION::TOP_BOTTOM );
1666 footprint->SetOrientationDegrees( orient );
1669 else if( type == wxS(
"ATTR" ) )
1677 if( attr.
key == wxS(
"Designator" ) )
1679 if( attr.
key == wxS(
"Designator" ) )
1681 field = footprint->GetField( FIELD_T::REFERENCE );
1685 field =
new PCB_FIELD( footprint.get(), FIELD_T::USER, attr.
key );
1686 footprint->Add( field, ADD_MODE::APPEND );
1689 if( attr.
fontName != wxS(
"default" ) )
1718 else if( type == wxS(
"PAD_NET" ) )
1720 wxString padNumber = line.at( 2 );
1721 wxString padNet = line.at( 3 );
1723 PAD*
pad = footprint->FindPadByNumber( padNumber );
1735 aBoard->
Add( footprint.release(), ADD_MODE::APPEND );
1741 for(
auto& [uuid, zone] : poursToFill )
1746 auto range = boardPouredMap.equal_range( uuid );
1747 for(
auto& it = range.first; it != range.second; ++it )
1750 int unki = poured.
unki;
1754 for(
int dataId = 0; dataId < poured.
polyData.size(); dataId++ )
1756 const nlohmann::json& fillData = poured.
polyData[dataId];
1757 const double ptScale = 10;
1763 for(
int i = 0; i < contour.
PointCount(); i++ )
1776 thisPoly.
Append( simple );
1787 for(
int segId = 0; segId < contour.
SegmentCount(); segId++ )
1797 fillPolySet.
Append( thisPoly );
1806 fillPolySet.
Inflate( strokeWidth / 2, CORNER_STRATEGY::ROUND_ALL_CORNERS,
1813 zone->SetFilledPolysList( zone->GetFirstLayer(), fillPolySet );
1814 zone->SetNeedRefill(
false );
1815 zone->SetIsFilled(
true );
1821 std::vector<PCB_SHAPE*> shapes;
1822 std::vector<std::unique_ptr<PCB_SHAPE>> newShapes;
1830 shapes.push_back(
static_cast<PCB_SHAPE*
>( item ) );
1835 for( std::unique_ptr<PCB_SHAPE>& ptr : newShapes )
1836 aBoard->
Add( ptr.release(), ADD_MODE::APPEND );
1848 offset.
x =
KiROUND( offset.
x / alignGrid ) * alignGrid;
1849 offset.
y =
KiROUND( offset.
y / alignGrid ) * alignGrid;
1851 aBoard->
Move( offset );
constexpr int ARC_HIGH_DEF
constexpr EDA_IU_SCALE pcbIUScale
constexpr int ARC_LOW_DEF
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
BASE_SET & set(size_t pos)
Bezier curves to polygon converter.
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...
virtual void SetIsKnockout(bool aKnockout)
virtual void SetLayer(PCB_LAYER_ID aLayer)
Set the layer this item is on.
Information pertinent to a Pcbnew printed circuit board.
LSET GetEnabledLayers() const
A proxy function that calls the corresponding function in m_BoardSettings.
void Add(BOARD_ITEM *aItem, ADD_MODE aMode=ADD_MODE::INSERT, bool aSkipConnectivity=false) override
Removes an item from the container.
const PAGE_INFO & GetPageSettings() const
NETINFO_ITEM * FindNet(int aNetcode) const
Search for a net with the given netcode.
bool SetLayerName(PCB_LAYER_ID aLayer, const wxString &aLayerName)
Changes the name of the layer given by aLayer.
void Move(const VECTOR2I &aMoveVector) override
Move this object.
BOX2I ComputeBoundingBox(bool aBoardEdgesOnly=false) const
Calculate the bounding box containing all board items (or board edge segments).
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
unsigned GetNetCount() const
void SetEnabledLayers(const LSET &aLayerMask)
A proxy function that calls the correspondent function in m_BoardSettings.
const DRAWINGS & Drawings() const
constexpr BOX2< Vec > & Merge(const BOX2< Vec > &aRect)
Modify the position and size of the rectangle in order to contain aRect.
constexpr const Vec GetCenter() const
constexpr const Vec & GetOrigin() const
constexpr const SizeVec & GetSize() const
A mix-in class (via multiple inheritance) that handles texts such as labels, parts,...
void SetTextSize(VECTOR2I aNewSize, bool aEnforceMinTextSize=true)
virtual void SetVisible(bool aVisible)
void SetTextThickness(int aWidth)
The TextThickness is that set by the user.
void SetTextAngleDegrees(double aOrientation)
virtual void SetText(const wxString &aText)
void SetFont(KIFONT::FONT *aFont)
VECTOR3D m_Offset
3D model offset (mm)
VECTOR3D m_Rotation
3D model rotation (degrees)
wxString m_Filename
The 3D shape filename in 3D library.
static FONT * GetFont(const wxString &aFontName=wxEmptyString, bool aBold=false, bool aItalic=false, const std::vector< wxString > *aEmbeddedFiles=nullptr, bool aForDrawingSheet=false)
A logical library item identifier and consists of various portions much like a URI.
LSET is a set of PCB_LAYER_IDs.
Handle the data for a net.
static constexpr PCB_LAYER_ID ALL_LAYERS
! Temporary layer identifier to identify code that is not padstack-aware
static LSET PTHMask()
layer set for a through hole pad
static LSET SMDMask()
layer set for a SMD pad on Front layer
Describe the page size and margins of a paper page on which to eventually print or plot.
double GetHeightMils() const
double GetWidthMils() const
static T ScaleSize(T aValue)
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
~PCB_IO_EASYEDAPRO_PARSER()
virtual void SetPosition(const VECTOR2I &aPos) override
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.
void SetImageScale(double aScale)
Set the image "zoom" value.
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.
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
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.
static constexpr EDA_ANGLE ANGLE_0
static constexpr EDA_ANGLE ANGLE_90
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)
bool IsBackLayer(PCB_LAYER_ID aLayerId)
Layer classification: check if it's a back layer.
PCB_LAYER_ID
A quick note on layer IDs:
wxString get_def(const std::map< wxString, wxString > &aMap, const char *aKey, const char *aDefval="")
std::optional< V > get_opt(const std::map< wxString, V > &aMap, const wxString &aKey)
constexpr void MIRROR(T &aPoint, const T &aMirrorRef)
Updates aPoint with the mirror of aPoint relative to the aMirrorRef.
LIB_ID ToKiCadLibID(const wxString &aLibName, const wxString &aLibReference)
static const bool IMPORT_POURED
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
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)
wxString EscapeString(const wxString &aSource, ESCAPE_CONTEXT aContext)
The Escape/Unescape routines use HTML-entity-reference-style encoding to handle characters which are:...
constexpr double IUTomm(int iu) const
constexpr int MilsToIU(int mils) const
constexpr int mmToIU(double mm) const
const SHAPE_LINE_CHAIN chain
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.
double DEG2RAD(double deg)
@ PCB_SHAPE_T
class PCB_SHAPE, a segment not on copper layers
VECTOR2< int32_t > VECTOR2I
VECTOR2< double > VECTOR2D