60#include <wx/filename.h>
73 const unsigned PROGRESS_DELTA = 250;
94 wxCHECK_MSG( !aStr.empty(), 0.0,
"Empty string passed to readDouble" );
96 std::istringstream istr( aStr );
97 istr.imbue( std::locale::classic() );
108 wxCHECK_MSG( !aStr.empty(), 0,
"Empty string passed to readInt" );
110 std::istringstream istr( aStr );
111 istr.imbue( std::locale::classic() );
121 std::ifstream ifs( aFile, std::ios::in | std::ios::binary );
129 ifs.ignore( std::numeric_limits<std::streamsize>::max() );
130 std::streamsize length = ifs.gcount();
132 ifs.seekg( 0, std::ios_base::beg );
134 std::string buffer( std::istreambuf_iterator<char>{ ifs }, {} );
136 std::vector < std::string > row;
140 row.reserve( length / 100 );
146 for(
auto& ch : buffer )
152 if( cell.empty() || cell[0] ==
'"' )
161 row.push_back( cell );
173 row.push_back( cell );
176 rows.push_back( row );
185 cell += std::toupper( ch );
190 if( !cell.empty() || !row.empty() )
192 row.push_back( cell );
194 rows.push_back( row );
208 row =
rows.at( aOffset );
210 catch( std::out_of_range& )
218 if( row[0].back() !=
'A' )
221 std::string row1 = row[1];
222 std::string row2 = row[2];
227 std::erase_if( row1, [](
char c ){
return c ==
'_'; } );
228 std::erase_if( row2, [](
char c ){
return c ==
'_'; } );
233 std::erase_if( row3, [](
char c ){
return c ==
'_'; } );
236 if( row1 ==
"REFDES" && row2 ==
"COMPCLASS" )
239 if( row1 ==
"NETNAME" && row2 ==
"REFDES" )
242 if( row1 ==
"CLASS" && row2 ==
"SUBCLASS" && row3.empty() )
245 if( row1 ==
"GRAPHICDATANAME" && row2 ==
"GRAPHICDATANUMBER" )
248 if( row1 ==
"CLASS" && row2 ==
"SUBCLASS" && row3 ==
"GRAPHICDATANAME" )
251 if( row1 ==
"SYMNAME" && row2 ==
"PINNAME" )
254 if( row1 ==
"SYMNAME" && row2 ==
"SYMMIRROR" && row3 ==
"PINNAME" )
257 if( row1 ==
"VIAX" && row2 ==
"VIAY" )
260 if( row1 ==
"SUBCLASS" && row2 ==
"PADSHAPENAME" )
263 if( row1 ==
"PADNAME" )
266 if( row1 ==
"LAYERSORT" )
269 wxLogError(
_(
"Unknown FABMASTER section %s:%s at row %zu." ),
282 if( aRow >=
rows.size() )
285 if(
rows[aRow].size() < 11 )
287 wxLogError(
_(
"Invalid row size in J row %zu. Expecting 11 elements but found %zu." ),
293 for(
int i = 7; i < 10 && retval < 1.0; ++i )
295 std::string units =
rows[aRow][i];
296 std::transform(units.begin(), units.end(),units.begin(), ::toupper);
298 if( units ==
"MILS" )
300 else if( units ==
"MILLIMETERS" )
302 else if( units ==
"MICRONS" )
304 else if( units ==
"INCHES" )
310 wxLogError(
_(
"Could not find units value, defaulting to mils." ) );
320 if( aRow >=
rows.size() )
325 for(
size_t i = 0; i <
header.size(); i++ )
329 std::erase_if(
header[i], [](
const char c ) {
return c ==
'_'; } );
335 THROW_IO_ERROR( wxString::Format(
_(
"Could not find column label %s." ), aStr.c_str() ) );
342 const auto& kicad_layer =
layers.find( aLayerName);
344 if( kicad_layer ==
layers.end() )
347 return static_cast<PCB_LAYER_ID>( kicad_layer->second.layerid );
353 size_t rownum = aRow + 2;
355 if( rownum >=
rows.size() )
363 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
367 if( row.size() !=
header.size() )
369 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
376 auto& pad_num = row[pad_num_col];
377 auto& pad_layer = row[pad_lay_col];
380 if( pad_layer ==
"INTERNAL_PAD_DEF" || pad_layer ==
"internal_pad_def" )
384 if( pad_layer[0] ==
'~' )
393 layer.
name = pad_layer;
411 size_t rownum = aRow + 2;
413 if( rownum >=
rows.size() )
419 if( scale_factor <= 0.0 )
433 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
438 if( row.size() !=
header.size() )
440 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
447 auto& pad_name = row[pad_name_col];
448 auto& pad_num = row[pad_num_col];
449 auto& pad_layer = row[pad_lay_col];
450 auto& pad_is_via = row[pad_via_col];
451 auto& pad_shape = row[pad_shape_col];
452 auto& pad_width = row[pad_width_col];
453 auto& pad_height = row[pad_height_col];
454 auto& pad_xoff = row[pad_xoff_col];
455 auto& pad_yoff = row[pad_yoff_col];
456 auto& pad_shapename = row[pad_shape_name_col];
459 if( pad_layer ==
"INTERNAL_PAD_DEF" || pad_layer ==
"internal_pad_def" )
464 auto new_pad =
pads.find( pad_name );
466 if( new_pad !=
pads.end() )
467 pad = &new_pad->second;
472 pad->name = pad_name;
476 if( pad_layer ==
"~DRILL" )
490 wxLogError(
_(
"Expecting drill size value but found %s!%s!%s in row %zu." ),
508 if( drill_x == drill_y )
510 pad->drill_size_x = drill_hit;
511 pad->drill_size_y = drill_hit;
515 pad->drill_size_x = drill_x;
516 pad->drill_size_y = drill_y;
519 if( !pad_shapename.empty() && pad_shapename[0] ==
'P' )
525 if( pad_shape.empty() )
538 wxLogError(
_(
"Expecting pad size values but found %s : %s in row %zu." ),
545 auto layer =
layers.find( pad_layer );
547 if( w > 0.0 && layer !=
layers.end() && layer->second.conductive )
548 pad->copper_layers.insert( pad_layer );
553 if( layer !=
layers.end() )
555 if( layer->second.layerid ==
F_Cu )
557 else if( layer->second.layerid ==
B_Cu )
561 if( w > std::numeric_limits<int>::max() || h > std::numeric_limits<int>::max() )
563 wxLogError(
_(
"Invalid pad size in row %zu." ), rownum );
567 if( pad_layer ==
"~TSM" || pad_layer ==
"~BSM" )
569 if( w > 0.0 && h > 0.0 )
577 if( pad_layer ==
"~TSP" || pad_layer ==
"~BSP" )
579 if( w > 0.0 && h > 0.0 )
588 if( pad_layer[0] ==
'~' )
598 wxLogError(
_(
"Expecting pad offset values but found %s:%s in row %zu." ),
605 if( w > 0.0 && h > 0.0 && recnum == 1 )
609 pad->via = ( std::toupper( pad_is_via[0] ) !=
'V' );
611 if( pad_shape ==
"CIRCLE" )
616 else if( pad_shape ==
"RECTANGLE" )
620 else if( pad_shape ==
"ROUNDED_RECT" )
624 else if( pad_shape ==
"SQUARE" )
629 else if( pad_shape ==
"OBLONG" || pad_shape ==
"OBLONG_X" || pad_shape ==
"OBLONG_Y" )
631 else if( pad_shape ==
"OCTAGON" )
634 pad->is_octogon =
true;
636 else if( pad_shape ==
"SHAPE" )
639 pad->custom_name = pad_shapename;
643 wxLogError(
_(
"Unknown pad shape name '%s' on layer '%s' in row %zu." ),
652 return rownum - aRow;
658 size_t rownum = aRow + 2;
660 if( rownum >=
rows.size() )
666 if( scale_factor <= 0.0 )
672 if( layer_class_col < 0 || layer_subclass_col < 0 )
675 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
679 if( row.size() !=
header.size() )
681 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
691 layer.
name = row[layer_subclass_col];
695 if( row[layer_class_col] ==
"ANTI ETCH" )
700 else if( row[layer_class_col] ==
"ETCH" )
706 return rownum - aRow;
712 std::vector<std::pair<std::string, int>> extra_layers
714 {
"ASSEMBLY_TOP",
F_Fab },
715 {
"ASSEMBLY_BOTTOM",
B_Fab },
716 {
"PLACE_BOUND_TOP",
F_CrtYd },
717 {
"PLACE_BOUND_BOTTOM",
B_CrtYd },
720 std::vector<FABMASTER_LAYER*> layer_order;
722 int next_user_layer =
User_1;
731 layer_order.push_back( &layer );
733 else if( ( layer.
name.find(
"SILK" ) != std::string::npos
734 && layer.
name.find(
"AUTOSILK" )
735 == std::string::npos )
736 || layer.
name.find(
"DISPLAY" ) != std::string::npos )
738 if( layer.
name.find(
"B" ) != std::string::npos )
743 else if( layer.
name.find(
"MASK" ) != std::string::npos ||
744 layer.
name.find(
"MSK" ) != std::string::npos )
746 if( layer.
name.find(
"B" ) != std::string::npos )
751 else if( layer.
name.find(
"PAST" ) != std::string::npos )
753 if( layer.
name.find(
"B" ) != std::string::npos )
758 else if( layer.
name.find(
"NCLEGEND" ) != std::string::npos )
767 if( layer.
name.find(
"AUTOSILK" ) == std::string::npos )
769 if( next_user_layer <=
User_9 )
772 layer.
layerid = next_user_layer;
773 next_user_layer += 2;
781 wxLogWarning(
_(
"No user layer to put layer %s" ), layer.
name );
789 for(
size_t layeri = 0; layeri < layer_order.size(); ++layeri )
794 else if( layeri == layer_order.size() - 1 )
797 layer->
layerid = layeri * 2 + 2;
800 for(
auto& new_pair : extra_layers )
804 new_layer.
name = new_pair.first;
805 new_layer.
layerid = new_pair.second;
808 auto result =
layers.emplace( new_pair.first, new_layer );
812 result.first->second.layerid = new_pair.second;
813 result.first->second.disable =
false;
817 for(
const auto& [layer_name, fabmaster_layer] :
layers )
819 wxLogTrace(
traceFabmaster, wxT(
"Layer %s -> KiCad layer %d" ), layer_name,
820 fabmaster_layer.layerid );
834 size_t rownum = aRow + 2;
836 if( rownum >=
rows.size() )
842 if( scale_factor <= 0.0 )
850 int layer_er_col =
getColFromName( aRow,
"LAYERDIELECTRICCONSTANT" );
851 int layer_rho_col =
getColFromName( aRow,
"LAYERELECTRICALCONDUCTIVITY" );
854 if( layer_sort_col < 0 || layer_subclass_col < 0 || layer_art_col < 0 || layer_use_col < 0
855 || layer_cond_col < 0 || layer_er_col < 0 || layer_rho_col < 0 || layer_mat_col < 0 )
858 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
862 if( row.size() !=
header.size() )
864 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
871 auto& layer_sort = row[layer_sort_col];
872 auto& layer_subclass = row[layer_subclass_col];
873 auto& layer_art = row[layer_art_col];
874 auto& layer_cond = row[layer_cond_col];
875 auto& layer_mat = row[layer_mat_col];
877 if( layer_mat ==
"AIR" )
882 if( layer_subclass.empty() )
884 if( layer_cond !=
"NO" )
885 layer.
name =
"In.Cu" + layer_sort;
887 layer.
name =
"Dielectric" + layer_sort;
890 layer.
positive = ( layer_art !=
"NEGATIVE" );
895 return rownum - aRow;
906 size_t rownum = aRow + 2;
908 if( rownum >=
rows.size() )
914 if( scale_factor <= 0.0 )
919 int pad_grdata_name_col =
getColFromName( aRow,
"GRAPHICDATANAME" );
920 int pad_grdata_num_col =
getColFromName( aRow,
"GRAPHICDATANUMBER" );
935 if( pad_subclass_col < 0 || pad_shape_name_col < 0 || pad_grdata1_col < 0 || pad_grdata2_col < 0
936 || pad_grdata3_col < 0 || pad_grdata4_col < 0 || pad_grdata5_col < 0
937 || pad_grdata6_col < 0 || pad_grdata7_col < 0 || pad_grdata8_col < 0
938 || pad_grdata9_col < 0 || pad_stack_name_col < 0 || pad_refdes_col < 0
939 || pad_pin_num_col < 0 )
942 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
946 if( row.size() !=
header.size() )
948 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
956 auto& pad_layer = row[pad_subclass_col];
957 auto pad_shape_name = row[pad_shape_name_col];
958 auto& pad_record_tag = row[pad_record_tag_col];
973 auto& pad_stack_name = row[pad_stack_name_col];
974 auto& pad_refdes = row[pad_refdes_col];
975 auto& pad_pin_num = row[pad_pin_num_col];
979 std::string prefix(
"FIG_SHAPE " );
981 if( pad_shape_name.length() <= prefix.length()
982 || !std::equal( prefix.begin(), prefix.end(), pad_shape_name.begin() ) )
992 if( std::sscanf( pad_record_tag.c_str(),
"%d %d", &
id, &seq ) != 2 )
994 wxLogError(
_(
"Invalid format for id string '%s' in custom pad row %zu." ),
995 pad_record_tag.c_str(),
1000 auto name = pad_shape_name.substr( prefix.length() );
1001 name +=
"_" + pad_refdes +
"_" + pad_pin_num;
1004 auto& custom_pad = ret.first->second;
1010 custom_pad.name =
name;
1011 custom_pad.padstack = pad_stack_name;
1012 custom_pad.pinnum = pad_pin_num;
1013 custom_pad.refdes = pad_refdes;
1020 auto gr_item = std::unique_ptr<GRAPHIC_ITEM>(
processGraphic( gr_data, scale_factor ) );
1024 gr_item->layer = pad_layer;
1025 gr_item->refdes = pad_refdes;
1027 gr_item->subseq = 0;
1032 auto retval = pad_it.first->second.insert( std::move(gr_item ) );
1034 if( !retval.second )
1036 wxLogError(
_(
"Could not insert graphical item %d into padstack '%s'." ),
1038 pad_stack_name.c_str() );
1043 wxLogError(
_(
"Unrecognized pad shape primitive '%s' in row %zu." ),
1049 return rownum - aRow;
1099 angle = endangle - startangle;
1134 std::unique_ptr<GRAPHIC_ARC> new_circle = std::make_unique<GRAPHIC_ARC>();
1145 if( size.
x != size.
y )
1147 wxLogError(
_(
"Circle with unequal x and y radii (x=%d, y=%d)" ), size.
x, size.
y );
1153 new_circle->radius = size.
x / 2;
1159 new_circle->start_x = start.
x;
1160 new_circle->start_y = start.
y;
1162 new_circle->end_x = start.
x;
1163 new_circle->end_y = start.
y;
1165 new_circle->center_x =
center.x;
1166 new_circle->center_y =
center.y;
1168 new_circle->clockwise =
true;
1170 new_circle->result =
SHAPE_ARC{ start, mid, start, 0 };
1172 return new_circle.release();
1192 new_rect->
width = 0;
1206 auto new_rect = std::make_unique<GRAPHIC_RECTANGLE>();
1215 new_rect->start_x = center_x - size_x / 2;
1216 new_rect->start_y = center_y + size_y / 2;
1217 new_rect->end_x = center_x + size_x / 2;
1218 new_rect->end_y = center_y - size_y / 2;
1220 new_rect->width = 0;
1222 return new_rect.release();
1247 auto new_oblong = std::make_unique<GRAPHIC_OBLONG>();
1259 return new_oblong.release();
1289 auto new_poly = std::make_unique<GRAPHIC_POLYGON>();
1294 bool across_corners =
true;
1328 across_corners =
false;
1334 wxCHECK_MSG(
false,
nullptr,
1335 wxString::Format(
"Unhandled polygon type: %s", aData.
graphic_dataname ) );
1340 return new_poly.release();
1351 auto new_cross = std::make_unique<GRAPHIC_CROSS>();
1360 return new_cross.release();
1384 if( toks.size() < 8 )
1387 wxLogError(
_(
"Invalid token count. Expected 8 but found %zu." ), toks.size() );
1389 new_text->
width = 0;
1390 new_text->
ital =
false;
1464 size_t rownum = aRow + 2;
1466 if( rownum >=
rows.size() )
1472 if( scale_factor <= 0.0 )
1491 if( geo_name_col < 0 || geo_num_col < 0 || geo_grdata1_col < 0 || geo_grdata2_col < 0
1492 || geo_grdata3_col < 0 || geo_grdata4_col < 0 || geo_grdata5_col < 0
1493 || geo_grdata6_col < 0 || geo_grdata7_col < 0 || geo_grdata8_col < 0
1494 || geo_grdata9_col < 0 || geo_subclass_col < 0 || geo_sym_name_col < 0
1495 || geo_refdes_col < 0 )
1498 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1502 if( row.size() !=
header.size() )
1504 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1511 auto& geo_tag = row[geo_tag_col];
1526 auto& geo_refdes = row[geo_refdes_col];
1534 if( std::sscanf( geo_tag.c_str(),
"%d %d %d", &
id, &seq, &subseq ) < 2 )
1536 wxLogError(
_(
"Invalid format for record_tag string '%s' in row %zu." ),
1542 auto gr_item = std::unique_ptr<GRAPHIC_ITEM>(
processGraphic( gr_data, scale_factor ) );
1547 gr_item->layer = row[geo_subclass_col];
1549 gr_item->subseq = subseq;
1551 if( geo_refdes.empty() )
1556 new_gr.
subclass = row[geo_subclass_col];
1557 new_gr.
refdes = row[geo_refdes_col];
1558 new_gr.
name = row[geo_sym_name_col];
1560 new_gr.
elements = std::make_unique<graphic_element>();
1565 graphic.
elements->emplace( std::move( gr_item ) );
1570 std::map<int, GEOM_GRAPHIC>{} );
1571 auto map_it = sym_gr_it.first->second.emplace(
id,
GEOM_GRAPHIC{} );
1572 auto& gr = map_it.first;
1576 gr->second.subclass = row[geo_subclass_col];
1577 gr->second.refdes = row[geo_refdes_col];
1578 gr->second.name = row[geo_sym_name_col];
1580 gr->second.elements = std::make_unique<graphic_element>();
1583 gr->second.elements->emplace( std::move( gr_item ) );
1587 return rownum - aRow;
1596 size_t rownum = aRow + 2;
1598 if( rownum >=
rows.size() )
1604 if( scale_factor <= 0.0 )
1613 if( viax_col < 0 || viay_col < 0 || padstack_name_col < 0 || net_name_col < 0
1614 || test_point_col < 0 )
1617 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1621 if( row.size() !=
header.size() )
1623 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1630 vias.emplace_back( std::make_unique<FM_VIA>() );
1635 via->padstack = row[padstack_name_col];
1636 via->net = row[net_name_col];
1637 via->test_point = ( row[test_point_col] ==
"YES" );
1640 return rownum - aRow;
1651 size_t rownum = aRow + 2;
1653 if( rownum >=
rows.size() )
1659 if( scale_factor <= 0.0 )
1665 int grdata_num_col =
getColFromName( aRow,
"GRAPHICDATANUMBER" );
1678 if( class_col < 0 || layer_col < 0 || grdata_name_col < 0 || grdata_num_col < 0
1679 || tag_col < 0 || grdata1_col < 0 || grdata2_col < 0 || grdata3_col < 0
1680 || grdata4_col < 0 || grdata5_col < 0 || grdata6_col < 0 || grdata7_col < 0
1681 || grdata8_col < 0 || grdata9_col < 0 || netname_col < 0 )
1684 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1688 if( row.size() !=
header.size() )
1690 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1710 const std::string& geo_tag = row[tag_col];
1717 if( std::sscanf( geo_tag.c_str(),
"%d %d %d", &
id, &seq, &subseq ) < 2 )
1719 wxLogError(
_(
"Invalid format for record_tag string '%s' in row %zu." ),
1725 auto gr_item = std::unique_ptr<GRAPHIC_ITEM>(
processGraphic( gr_data, scale_factor ) );
1729 wxLogTrace(
traceFabmaster,
_(
"Unhandled graphic item '%s' in row %zu." ),
1735 auto new_trace = std::make_unique<TRACE>();
1737 new_trace->layer = row[layer_col];
1738 new_trace->netname = row[netname_col];
1739 new_trace->lclass = row[class_col];
1741 gr_item->layer = row[layer_col];
1743 gr_item->subseq = subseq;
1746 if( new_trace->lclass ==
"REF DES" )
1748 auto result =
refdes.emplace( std::move( new_trace ) );
1749 auto& ref = *
result.first;
1750 ref->segment.emplace( std::move( gr_item ) );
1752 else if( new_trace->lclass ==
"DEVICE TYPE" || new_trace->lclass ==
"COMPONENT VALUE"
1753 || new_trace->lclass ==
"TOLERANCE" )
1766 else if( gr_item->width == 0 )
1768 auto result =
zones.emplace( std::move( new_trace ) );
1769 auto& zone = *
result.first;
1770 auto gr_result = zone->segment.emplace( std::move( gr_item ) );
1772 if( !gr_result.second )
1774 wxLogError(
_(
"Duplicate item for ID %d and sequence %d in row %zu." ),
1782 auto result =
traces.emplace( std::move( new_trace ) );
1783 auto& trace = *
result.first;
1784 auto gr_result = trace->segment.emplace( std::move( gr_item ) );
1786 if( !gr_result.second )
1788 wxLogError(
_(
"Duplicate item for ID %d and sequence %d in row %zu." ),
1796 return rownum - aRow;
1802 if( aSymType ==
"PACKAGE" )
1804 else if( aSymType ==
"DRAFTING")
1806 else if( aSymType ==
"MECHANICAL" )
1808 else if( aSymType ==
"FORMAT" )
1817 if( aCmpClass ==
"IO" )
1819 else if( aCmpClass ==
"IC" )
1821 else if( aCmpClass ==
"DISCRETE" )
1834 size_t rownum = aRow + 2;
1836 if( rownum >=
rows.size() )
1842 if( scale_factor <= 0.0 )
1850 int compinscode_col =
getColFromName( aRow,
"COMPINSERTIONCODE" );
1861 if( refdes_col < 0 || compclass_col < 0 || comppartnum_col < 0 || compheight_col < 0
1862 || compdevlabelcol < 0 || compinscode_col < 0 || symtype_col < 0 || symname_col < 0
1863 || symmirror_col < 0 || symrotate_col < 0 || symx_col < 0 || symy_col < 0
1864 || compvalue_col < 0 || comptol_col < 0 || compvolt_col < 0 )
1867 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1871 if( row.size() !=
header.size() )
1873 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1880 const wxString& comp_refdes = row[refdes_col];
1882 if( row[symx_col].
empty() || row[symy_col].
empty() || row[symrotate_col].
empty() )
1884 wxLogError(
_(
"Missing X, Y, or rotation data in row %zu for refdes %s. "
1885 "This may be an unplaced component." ),
1886 rownum, comp_refdes );
1890 auto cmp = std::make_unique<COMPONENT>();
1892 cmp->refdes = comp_refdes;
1894 cmp->pn = row[comppartnum_col];
1895 cmp->height = row[compheight_col];
1896 cmp->dev_label = row[compdevlabelcol];
1897 cmp->insert_code = row[compinscode_col];
1899 cmp->name = row[symname_col];
1900 cmp->mirror = ( row[symmirror_col] ==
"YES" );
1901 cmp->rotate =
readDouble( row[symrotate_col] );
1904 cmp->value = row[compvalue_col];
1905 cmp->tol = row[comptol_col];
1906 cmp->voltage = row[compvolt_col];
1912 auto retval =
components.insert( std::make_pair( cmp->refdes, std::vector<std::unique_ptr<COMPONENT>>{} ) );
1917 vec->second.push_back( std::move( cmp ) );
1920 return rownum - aRow;
1930 size_t rownum = aRow + 2;
1932 if( rownum >=
rows.size() )
1938 if( scale_factor <= 0.0 )
1952 if( symname_col < 0 ||symmirror_col < 0 || pinname_col < 0 || pinnum_col < 0 || pinx_col < 0
1953 || piny_col < 0 || padstack_col < 0 || refdes_col < 0 || pinrot_col < 0
1954 || testpoint_col < 0 )
1957 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1961 if( row.size() !=
header.size() )
1963 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1970 auto pin = std::make_unique<PIN>();
1972 pin->name = row[symname_col];
1973 pin->mirror = ( row[symmirror_col] ==
"YES" );
1974 pin->pin_name = row[pinname_col];
1975 pin->pin_number = row[pinnum_col];
1978 pin->padstack = row[padstack_col];
1979 pin->refdes = row[refdes_col];
1985 std::string pin_key =
pin->refdes.empty() ?
pin->name :
pin->refdes;
1987 auto map_it =
pins.find( pin_key );
1989 if( map_it ==
pins.end() )
1991 auto retval =
pins.insert( std::make_pair( pin_key, std::set<std::unique_ptr<PIN>,
1993 map_it = retval.first;
1996 map_it->second.insert( std::move(
pin ) );
1999 return rownum - aRow;
2008 size_t rownum = aRow + 2;
2010 if( rownum >=
rows.size() )
2016 if( scale_factor <= 0.0 )
2026 if( netname_col < 0 || refdes_col < 0 || pinnum_col < 0 || pinname_col < 0 || pingnd_col < 0
2030 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
2034 if( row.size() !=
header.size() )
2036 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
2044 new_net.
name = row[netname_col];
2045 new_net.
refdes = row[refdes_col];
2046 new_net.
pin_num = row[pinnum_col];
2047 new_net.
pin_name = row[pinname_col];
2048 new_net.
pin_gnd = ( row[pingnd_col] ==
"YES" );
2049 new_net.
pin_pwr = ( row[pinpwr_col] ==
"YES" );
2052 netnames.insert( row[netname_col] );
2055 return rownum - aRow;
2062 for(
size_t i = 0; i <
rows.size(); )
2076 i += std::max( retval, 1 );
2084 i += std::max( retval, 1 );
2092 i += std::max( retval, 1 );
2100 i += std::max( retval, 1 );
2108 i += std::max( retval, 1 );
2116 i += std::max( retval, 1 );
2124 i += std::max( retval, 1 );
2132 i += std::max( retval, 1 );
2140 i += std::max( retval, 1 );
2148 i += std::max( retval, 1 );
2165 for(
auto& zone :
zones )
2175 if( zone->layer ==
"OUTLINE" || zone->layer ==
"DESIGN_OUTLINE" )
2196 std::set<ZONE*> zones_to_delete;
2197 std::set<ZONE*> matched_fills;
2199 for(
auto zone : aBoard->
Zones() )
2201 if( zone->GetNetCode() > 0 )
2202 zones_to_delete.insert( zone );
2205 for(
auto zone1 : aBoard->
Zones() )
2207 if( zone1->GetNetCode() > 0 )
2212 std::map<int, std::vector<ZONE*>> net_to_fills;
2214 for(
auto zone2 : aBoard->
Zones() )
2216 if( zone2->GetNetCode() <= 0 )
2221 if( zone1->GetLayer() != zone2->GetLayer() )
2227 size_t match_count = 0;
2229 for(
auto& pt1 : outline1.
CPoints() )
2235 for(
auto& pt2 : outline2.
CPoints() )
2241 if( match_count > 0 )
2243 overlaps[zone2->GetNetCode()] += match_count;
2244 net_to_fills[zone2->GetNetCode()].push_back( zone2 );
2249 size_t max_net_id = 0;
2251 for(
size_t el = 1; el < overlaps.size(); ++el )
2253 if( overlaps[el] > max_net )
2255 max_net = overlaps[el];
2262 zone1->SetNetCode( max_net_id );
2264 for(
ZONE* fill : net_to_fills[max_net_id] )
2265 matched_fills.insert( fill );
2269 for(
auto zone : zones_to_delete )
2271 if( matched_fills.find( zone ) != matched_fills.end() )
2292 if( aMirrorPoint.has_value() )
2325 for(
const auto& [pinKey, pinSet] :
pins )
2327 if( pinSet.empty() )
2333 const auto& firstPin = *pinSet.begin();
2335 int minX = firstPin->pin_x;
2336 int maxX = firstPin->pin_x;
2337 int minY = firstPin->pin_y;
2338 int maxY = firstPin->pin_y;
2340 for(
const auto&
pin : pinSet )
2342 minX = std::min( minX,
pin->pin_x );
2343 maxX = std::max( maxX,
pin->pin_x );
2344 minY = std::min( minY,
pin->pin_y );
2345 maxY = std::max( maxY,
pin->pin_y );
2348 auto cmp = std::make_unique<COMPONENT>();
2349 cmp->refdes = pinKey;
2350 cmp->name = firstPin->name;
2351 cmp->mirror = firstPin->mirror;
2353 cmp->x = ( minX + maxX ) / 2;
2354 cmp->y = ( minY + maxY ) / 2;
2358 std::vector<std::unique_ptr<COMPONENT>> compVec;
2359 compVec.push_back( std::move( cmp ) );
2360 components.insert( std::make_pair( pinKey, std::move( compVec ) ) );
2374 bool has_multiple = mod.second.size() > 1;
2376 for(
int i = 0; i < (int) mod.second.size(); ++i )
2378 auto& src = mod.second[i];
2382 wxString mod_ref = src->name;
2386 mod_ref.Append( wxString::Format( wxT(
"_%d" ), i ) );
2391 wxString key = !lib_ref.empty() ? lib_ref + wxT(
":" ) + mod_ref : mod_ref;
2394 fpID.
Parse( key,
true );
2403 wxString reference = src->refdes;
2405 if( !std::isalpha( src->refdes[0] ) )
2406 reference.Prepend(
"UNK" );
2419 for(
auto& ref :
refdes )
2422 static_cast<const GRAPHIC_TEXT&
>( **ref->segment.begin() );
2424 if( lsrc.
text == src->refdes )
2431 wxLogTrace(
traceFabmaster, wxS(
"The layer %s is not mapped?" ),
2432 ref->layer.c_str() );
2443 flip_point =
VECTOR2I( src->x, src->y );
2448 setupText( lsrc, layer, *txt, *aBoard, flip_point );
2463 for(
auto& gr_ref : gr_it->second )
2465 auto& graphic = gr_ref.second;
2467 for(
auto& seg : *graphic.elements )
2474 STROKE_PARAMS defaultStroke( ds.GetLineThickness( layer ) );
2476 switch( seg->shape )
2499 if( lsrc->
width == 0 )
2512 circle->SetLayer( layer );
2524 if( lsrc.
width == 0 )
2530 if( lsrc.
layer ==
"DISPLAY_TOP" || lsrc.
layer ==
"DISPLAY_BOTTOM" )
2531 circle->SetFilled(
true );
2533 circle->SetWidth( ds.GetLineThickness(
circle->GetLayer() ) );
2547 std::unique_ptr<PCB_SHAPE> arc =
2561 arc->SetLayer( layer );
2565 if( lsrc->
width == 0 )
2566 arc->SetStroke( defaultStroke );
2605 std::unique_ptr<PCB_TEXT> txt = std::make_unique<PCB_TEXT>( fp );
2610 flip_point =
VECTOR2I( src->x, src->y );
2612 setupText( lsrc, layer, *txt, *aBoard, flip_point );
2637 auto pin_it =
pins.find( src->refdes );
2640 if( pin_it ==
pins.end() )
2641 pin_it =
pins.find( src->name );
2643 if( pin_it !=
pins.end() )
2645 for(
auto&
pin : pin_it->second )
2647 auto pin_net_it =
pin_nets.find( std::make_pair(
pin->refdes,
2648 pin->pin_number ) );
2649 auto padstack =
pads.find(
pin->padstack );
2650 std::string netname =
"";
2653 netname = pin_net_it->second.name;
2655 auto net_it = netinfo.find( netname );
2657 std::unique_ptr<PAD> newpad = std::make_unique<PAD>( fp );
2659 if( net_it != netinfo.end() )
2660 newpad->SetNet( net_it->second );
2662 newpad->SetNetCode( 0 );
2664 newpad->SetX(
pin->pin_x );
2667 newpad->SetY( 2 * src->y -
pin->pin_y );
2669 newpad->SetY(
pin->pin_y );
2671 newpad->SetNumber(
pin->pin_number );
2673 if( padstack ==
pads.end() )
2675 wxLogError(
_(
"Unable to locate padstack %s in file %s\n" ),
2681 auto&
pad = padstack->second;
2689 int pad_size = std::min(
pad.width,
pad.height );
2692 VECTOR2I( pad_size / 2, pad_size / 2 ) );
2694 std::string custom_name =
pad.custom_name +
"_" +
pin->refdes +
"_" +
2696 auto custom_it =
pad_shapes.find( custom_name );
2702 int last_subseq = 0;
2709 for(
const auto& el : (*custom_it).second.elements )
2715 if(
getLayer( ( *( el.second.begin() ) )->layer ) != primary_layer )
2718 for(
const auto& seg : el.second )
2720 if( seg->subseq > 0 || seg->subseq != last_subseq )
2722 poly_outline.
Polygon(0).back().SetClosed(
true );
2730 if( poly_outline.
VertexCount( 0, hole_idx ) == 0 )
2750 wxLogError(
_(
"Invalid custom pad '%s'. Replacing with "
2752 custom_name.c_str() );
2759 poly_outline.
Move( -newpad->GetPosition() );
2782 wxLogError(
_(
"Invalid custom pad '%s'. Replacing with "
2784 custom_name.c_str() );
2790 wxLogError(
_(
"Could not find custom pad '%s'." ),
2791 custom_name.c_str() );
2813 if(
pad.drill_size_x ==
pad.drill_size_y )
2818 newpad->SetDrillSize(
VECTOR2I(
pad.drill_size_x,
pad.drill_size_y ) );
2826 else if(
pad.bottom )
2827 newpad->SetLayerSet(
PAD::SMDMask().FlipStandardLayers() );
2832 newpad->SetOrientation(
EDA_ANGLE( -src->rotate +
pin->rotation,
2835 newpad->SetOrientation(
EDA_ANGLE( src->rotate -
pin->rotation,
2838 if( newpad->GetSizeX() > 0 || newpad->GetSizeY() > 0 )
2844 wxLogError(
_(
"Invalid zero-sized pad ignored in\nfile: %s" ),
2871 for(
auto& layer :
layers )
2876 layer_set.
set( layer.second.layerid );
2881 for(
auto& layer :
layers )
2883 if( layer.second.conductive )
2886 layer.second.name );
2900 std::vector<const FABMASTER_LAYER*> conductiveLayers;
2902 for(
const auto& layer :
layers )
2904 if( layer.second.conductive )
2905 conductiveLayers.push_back( &layer.second );
2914 auto net_it = netinfo.find(
via->net );
2915 auto padstack =
pads.find(
via->padstack );
2921 if( net_it != netinfo.end() )
2922 new_via->
SetNet( net_it->second );
2924 if( padstack ==
pads.end() )
2928 if( !ds.m_ViasDimensionsList.empty() )
2931 new_via->
SetDrill( ds.m_ViasDimensionsList[0].m_Drill );
2941 new_via->
SetDrill( padstack->second.drill_size_x );
2944 const std::set<std::string>& viaLayers = padstack->second.copper_layers;
2946 if( viaLayers.size() >= 2 )
2954 if( viaLayers.count( layer->name ) )
2963 if( topLayer && botLayer && topLayer != botLayer )
2969 bool isThrough = ( topLayerId ==
F_Cu && botLayerId ==
B_Cu );
2975 if( topLayerId ==
F_Cu || botLayerId ==
B_Cu )
3010 auto net_it = netinfo.find( aLine->netname );
3012 for(
const auto& seg : aLine->segment )
3018 switch( seg->shape )
3031 if( net_it != netinfo.end() )
3032 trk->
SetNet( net_it->second );
3046 if( net_it != netinfo.end() )
3047 trk->
SetNet( net_it->second );
3055 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3063 wxLogError(
_(
"Expecting etch data to be on copper layer. Row found on layer '%s'" ),
3064 seg->layer.c_str() );
3075 int last_subseq = 0;
3080 for(
const auto& seg : aElement )
3082 if( seg->subseq > 0 || seg->subseq != last_subseq )
3089 if( poly_outline.
VertexCount( 0, hole_idx ) == 0 )
3103 return poly_outline;
3130 if( aLine.
segment.size() == 0 )
3136 int first_subseq = -1;
3137 bool have_multiple_subseqs =
false;
3139 for(
const std::unique_ptr<GRAPHIC_ITEM>& gr_item : aLine.
segment )
3141 if( first ==
nullptr )
3143 first = gr_item.get();
3144 first_subseq = gr_item->subseq;
3146 else if( gr_item->subseq == first_subseq )
3148 last = gr_item.get();
3152 have_multiple_subseqs =
true;
3158 wxCHECK( first,
true );
3176 switch( last->
shape )
3198 if(
end.has_value() && start ==
end )
3206std::vector<std::unique_ptr<BOARD_ITEM>>
3209 std::vector<std::unique_ptr<BOARD_ITEM>> new_items;
3214 const auto setShapeParameters = [&](
PCB_SHAPE& aShape )
3218 if( aShape.GetWidth() == 0 )
3219 aShape.SetStroke( defaultStroke );
3222 switch( aGraphic.
shape )
3228 auto new_text = std::make_unique<PCB_TEXT>( &aBoard );
3232 new_text->SetMirrored(
true );
3235 setupText( src, aLayer, *new_text, aBoard, std::nullopt );
3237 new_items.emplace_back( std::move( new_text ) );
3250 for(
const SEG& seg : segs )
3252 auto line = std::make_unique<PCB_SHAPE>( &aBoard );
3254 line->SetStart( seg.A );
3255 line->SetEnd( seg.B );
3257 setShapeParameters( *line );
3258 new_items.emplace_back( std::move( line ) );
3266 auto new_shape = std::make_unique<PCB_SHAPE>( &aBoard );
3268 setShapeParameters( *new_shape );
3270 switch( aGraphic.
shape )
3299 new_shape->SetRadius( src.
radius );
3311 new_shape->SetFilled( src.
fill );
3319 new_shape->SetPolyPoints( src.
m_pts );
3349 new_shape->SetPolyShape( poly );
3354 wxLogError(
_(
"Unhandled shape type %d in polygon on layer %s, seq %d %d" ),
3358 new_items.emplace_back( std::move( new_shape ) );
3362 for( std::unique_ptr<BOARD_ITEM>& new_item : new_items )
3364 new_item->SetLayer( aLayer );
3368 if( new_items.size() > 1 )
3370 auto new_group = std::make_unique<PCB_GROUP>( &aBoard );
3371 for( std::unique_ptr<BOARD_ITEM>& new_item : new_items )
3373 new_group->AddItem( new_item.get() );
3375 new_items.emplace_back( std::move( new_group ) );
3384 if( aLine->segment.empty() )
3398 for(
const auto& seg : aLine->segment )
3400 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3447 if( aLine->segment.size() < 3 )
3451 ZONE* zone =
nullptr;
3454 auto net_it = netinfo.find( aLine->netname );
3456 auto new_layer =
getLayer( aLine->layer );
3461 zone =
new ZONE( aBoard );
3464 if( net_it != netinfo.end() )
3465 zone->
SetNet( net_it->second );
3467 if( aLine->layer ==
"ALL" )
3479 if( aLine->lclass ==
"ROUTE KEEPOUT")
3484 else if( aLine->lclass ==
"VIA KEEPOUT")
3499 std::unique_ptr<SHAPE_LINE_CHAIN> pending_hole =
nullptr;
3502 const auto add_hole_if_valid = [&]()
3506 pending_hole->SetClosed(
true );
3512 wxLogMessage(
_(
"Invalid hole with %d points in zone on layer %s with net %s" ),
3517 pending_hole.reset();
3521 int last_subseq = 0;
3522 for(
const auto& seg : aLine->segment )
3524 if( seg->subseq > 0 && seg->subseq != last_subseq )
3528 if( aLine->lclass ==
"BOUNDARY" )
3531 add_hole_if_valid();
3532 pending_hole = std::make_unique<SHAPE_LINE_CHAIN>();
3533 active_chain = pending_hole.get();
3534 last_subseq = seg->subseq;
3545 active_chain->
Append( start );
3554 wxLogError(
_(
"Outline seems discontinuous: last point was %s, "
3555 "start point of next segment is %s" ),
3570 wxLogError(
_(
"Invalid shape type %d in zone outline" ), seg->shape );
3575 add_hole_if_valid();
3584 delete( zone_outline );
3596 if( aLine->lclass ==
"BOARD GEOMETRY" && aLine->layer !=
"DIMENSION" )
3598 else if( aLine->lclass ==
"DRAWING FORMAT" )
3603 for(
auto& seg : aLine->segment )
3605 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3625 if( geom.subclass ==
"PIN_NUMBER" )
3633 if( !geom.elements->empty() )
3636 if( ( *( geom.elements->begin() ) )->width == 0 )
3657 for(
auto& seg : *geom.elements )
3659 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3700 for(
auto& track :
traces )
3704 if( track->lclass ==
"ETCH" )
3706 else if( track->layer ==
"OUTLINE" || track->layer ==
"DIMENSION" )
constexpr EDA_IU_SCALE pcbIUScale
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
BASE_SET & set(size_t pos)
wxString GetNetname() const
void SetLayer(PCB_LAYER_ID aLayer) override
Set the layer this item is on.
void SetNet(NETINFO_ITEM *aNetInfo)
Set a NET_INFO object for the item.
Container for design settings for a BOARD object.
int GetLineThickness(PCB_LAYER_ID aLayer) const
Return the default graphic segment thickness from the layer class for the given layer.
virtual void SetLayer(PCB_LAYER_ID aLayer)
Set the layer this item is on.
wxString GetLayerName() const
Return the name of the PCB layer on which the item resides.
Information pertinent to a Pcbnew printed circuit board.
const NETINFO_LIST & GetNetInfo() const
void Add(BOARD_ITEM *aItem, ADD_MODE aMode=ADD_MODE::INSERT, bool aSkipConnectivity=false) override
Removes an item from the container.
void SetFileName(const wxString &aFileName)
LSET GetLayerSet() const override
Return a std::bitset of all layers on which the item physically resides.
const ZONES & Zones() const
bool SetLayerName(PCB_LAYER_ID aLayer, const wxString &aLayerName)
Changes the name of the layer given by aLayer.
PCB_LAYER_ID FlipLayer(PCB_LAYER_ID aLayer) const
const wxString & GetFileName() const
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
void Remove(BOARD_ITEM *aBoardItem, REMOVE_MODE aMode=REMOVE_MODE::NORMAL) override
Removes an item from the container.
void SetEnabledLayers(const LSET &aLayerMask)
A proxy function that calls the correspondent function in m_BoardSettings.
constexpr bool Intersects(const BOX2< Vec > &aRect) const
EDA_ANGLE Normalized() const
void SetPolyShape(const SHAPE_POLY_SET &aShape)
virtual void SetFilled(bool aFlag)
void SetStart(const VECTOR2I &aStart)
void SetShape(SHAPE_T aShape)
void SetEnd(const VECTOR2I &aEnd)
void SetTextPos(const VECTOR2I &aPoint)
void SetMirrored(bool isMirrored)
void SetVertJustify(GR_TEXT_V_ALIGN_T aType)
void SetTextWidth(int aWidth)
virtual void SetVisible(bool aVisible)
void SetTextThickness(int aWidth)
The TextThickness is that set by the user.
void SetTextHeight(int aHeight)
void SetKeepUpright(bool aKeepUpright)
virtual void SetText(const wxString &aText)
virtual void SetTextAngle(const EDA_ANGLE &aAngle)
void SetItalic(bool aItalic)
Set the text to be italic - this will also update the font if needed.
void SetHorizJustify(GR_TEXT_H_ALIGN_T aType)
size_t processFootprints(size_t aRow)
A!REFDES!COMP_CLASS!COMP_PART_NUMBER!COMP_HEIGHT!COMP_DEVICE_LABEL!COMP_INSERTION_CODE!...
size_t processPins(size_t aRow)
A!SYM_NAME!SYM_MIRROR!PIN_NAME!PIN_NUMBER!PIN_X!PIN_Y!PAD_STACK_NAME!REFDES!PIN_ROTATION!
int readInt(const std::string &aStr) const
GRAPHIC_OBLONG * processOblong(const GRAPHIC_DATA &aData, double aScale)
size_t processGeometry(size_t aRow)
A!GRAPHIC_DATA_NAME!GRAPHIC_DATA_NUMBER!RECORD_TAG!GRAPHIC_DATA_1!GRAPHIC_DATA_2!GRAPHIC_DATA_3!
static std::vector< std::unique_ptr< BOARD_ITEM > > createBoardItems(BOARD &aBoard, PCB_LAYER_ID aLayer, FABMASTER::GRAPHIC_ITEM &aGraphic)
Convert one Fabmaster graphic item to one or more PCB items.
bool Read(const std::string &aFile)
bool loadNets(BOARD *aBoard)
std::map< std::string, std::map< int, GEOM_GRAPHIC > > comp_graphics
GRAPHIC_CROSS * processCross(const GRAPHIC_DATA &aData, double aScale)
unsigned m_lastProgressCount
SYMTYPE parseSymType(const std::string &aSymType)
GRAPHIC_TEXT * processText(const GRAPHIC_DATA &aData, double aScale)
bool loadLayers(BOARD *aBoard)
static void setupText(const FABMASTER::GRAPHIC_TEXT &aGraphicText, PCB_LAYER_ID aLayer, PCB_TEXT &aText, const BOARD &aBoard, const OPT_VECTOR2I &aMirrorPoint)
Set parameters for graphic text.
PCB_LAYER_ID getLayer(const std::string &aLayerName)
GRAPHIC_RECTANGLE * processSquare(const GRAPHIC_DATA &aData, double aScale)
static bool traceIsOpen(const FABMASTER::TRACE &aLine)
bool loadZones(BOARD *aBoard)
Loads sections of the database into the board.
GRAPHIC_RECTANGLE * processRectangle(const GRAPHIC_DATA &aData, double aScale)
std::vector< std::string > single_row
size_t processSimpleLayers(size_t aRow)
PROGRESS_REPORTER * m_progressReporter
optional; may be nullptr
bool loadFootprints(BOARD *aBoard)
GRAPHIC_ARC * processCircle(const GRAPHIC_DATA &aData, double aScale)
std::unordered_map< std::string, FM_PAD > pads
int getColFromName(size_t aRow, const std::string &aStr)
bool loadVias(BOARD *aBoard)
size_t processLayers(size_t aRow)
A!LAYER_SORT!LAYER_SUBCLASS!LAYER_ARTWORK!LAYER_USE!LAYER_CONDUCTOR!LAYER_DIELECTRIC_CONSTANT!
std::map< std::string, FABMASTER_LAYER > layers
COMPCLASS parseCompClass(const std::string &aCompClass)
bool loadEtch(BOARD *aBoard, const std::unique_ptr< TRACE > &aLine)
GRAPHIC_RECTANGLE * processFigRectangle(const GRAPHIC_DATA &aData, double aScale)
std::map< std::string, std::set< std::unique_ptr< PIN >, PIN::BY_NUM > > pins
std::set< std::unique_ptr< GRAPHIC_ITEM >, GRAPHIC_ITEM::SEQ_CMP > graphic_element
std::map< std::pair< std::string, std::string >, NETNAME > pin_nets
GRAPHIC_ITEM * processGraphic(const GRAPHIC_DATA &aData, double aScale)
Specialty functions for processing graphical data rows into the internal database.
std::deque< single_row > rows
GRAPHIC_POLYGON * processPolygon(const GRAPHIC_DATA &aData, double aScale)
void createComponentsFromOrphanPins()
Creates synthetic COMPONENT entries from pins that have no matching component.
bool loadOutline(BOARD *aBoard, const std::unique_ptr< TRACE > &aLine)
size_t processPadStacks(size_t aRow)
A!PADNAME!RECNUMBER!LAYER!FIXFLAG!VIAFLAG!PADSHAPE1!PADWIDTH!PADHGHT!
std::vector< GEOM_GRAPHIC > board_graphics
section_type detectType(size_t aOffset)
double readDouble(const std::string &aStr) const
Reads the double/integer value from a std string independent of the user locale.
bool loadZone(BOARD *aBoard, const std::unique_ptr< FABMASTER::TRACE > &aLine)
GRAPHIC_ARC * processArc(const GRAPHIC_DATA &aData, double aScale)
SHAPE_POLY_SET loadShapePolySet(const graphic_element &aLine)
bool loadGraphics(BOARD *aBoard)
std::unordered_map< std::string, FABMASTER_PAD_SHAPE > pad_shapes
bool LoadBoard(BOARD *aBoard, PROGRESS_REPORTER *aProgressReporter)
std::vector< std::unique_ptr< FM_VIA > > vias
std::set< std::unique_ptr< TRACE >, TRACE::BY_ID > traces
std::set< std::unique_ptr< TRACE >, TRACE::BY_ID > zones
std::map< std::string, std::vector< std::unique_ptr< COMPONENT > > > components
bool loadPolygon(BOARD *aBoard, const std::unique_ptr< FABMASTER::TRACE > &aLine)
std::set< std::unique_ptr< TRACE >, TRACE::BY_ID > refdes
@ GR_SHAPE_OBLONG
!< Actually 360° arcs (for both arcs where start==end and real circles)
@ GR_SHAPE_CROSS
!< X/Y oblongs
size_t processPadStackLayers(size_t aRow)
std::set< std::string > netnames
size_t processTraces(size_t aRow)
A!CLASS!SUBCLASS!GRAPHIC_DATA_NAME!GRAPHIC_DATA_NUMBER!RECORD_TAG!GRAPHIC_DATA_1!GRAPHIC_DATA_2!
size_t processNets(size_t aRow)
A!NET_NAME!REFDES!PIN_NUMBER!PIN_NAME!PIN_GROUND!PIN_POWER!
bool orderZones(BOARD *aBoard)
Sets zone priorities based on zone BB size.
double processScaleFactor(size_t aRow)
Processes data from text vectors into internal database for further ordering.
size_t processVias(size_t aRow)
A!VIA_X!VIA_Y!PAD_STACK_NAME!NET_NAME!TEST_POINT!
unsigned m_totalCount
for progress reporting
size_t processCustomPads(size_t aRow)
A!SUBCLASS!PAD_SHAPE_NAME!GRAPHIC_DATA_NAME!GRAPHIC_DATA_NUMBER!RECORD_TAG!GRAPHIC_DATA_1!
GRAPHIC_LINE * processLine(const GRAPHIC_DATA &aData, double aScale)
A logical library item identifier and consists of various portions much like a URI.
int Parse(const UTF8 &aId, bool aFix=false)
Parse LIB_ID with the information from aId.
LSET is a set of PCB_LAYER_IDs.
static const LSET & UserMask()
static const LSET & AllTechMask()
Return a mask holding all technical layers (no CU layer) on both side.
static LSET AllCuMask(int aCuLayerCount)
Return a mask holding the requested number of Cu PCB_LAYER_IDs.
Handle the data for a net.
unsigned GetNetCount() const
const NETNAMES_MAP & NetsByName() const
Return the name map, at least for python.
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 UnplatedHoleMask()
layer set for a mechanical unplated through hole pad
static LSET SMDMask()
layer set for a SMD pad on Front layer
int GetWidth() const override
void SetLayer(PCB_LAYER_ID aLayer) override
Set the layer this item is on.
void SetStroke(const STROKE_PARAMS &aStroke) override
void SetEnd(const VECTOR2I &aEnd)
void SetStart(const VECTOR2I &aStart)
virtual void SetWidth(int aWidth)
void SetDrillDefault()
Set the drill value for vias to the default value UNDEFINED_DRILL_DIAMETER.
void SetDrill(int aDrill)
void SetPosition(const VECTOR2I &aPoint) override
void SetLayerPair(PCB_LAYER_ID aTopLayer, PCB_LAYER_ID aBottomLayer)
For a via m_layer contains the top layer, the other layer is in m_bottomLayer/.
void SetViaType(VIATYPE aViaType)
void SetWidth(int aWidth) override
A progress reporter interface for use in multi-threaded environments.
const VECTOR2I & GetArcMid() const
void Mirror(const VECTOR2I &aRef, FLIP_DIRECTION aFlipDirection)
const VECTOR2I & GetP1() const
const VECTOR2I & GetP0() const
const VECTOR2I & GetCenter() const
bool PointOnEdge(const VECTOR2I &aP, int aAccuracy=0) const
Check if point aP lies on an edge or vertex of the line chain.
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
int PointCount() const
Return the number of points (vertices) in this line chain.
void Append(int aX, int aY, bool aAllowDuplication=false)
Append a new point at the end of the line chain.
const VECTOR2I & CLastPoint() const
Return the last point in the line chain.
const std::vector< VECTOR2I > & CPoints() const
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 Rotate(const EDA_ANGLE &aAngle, const VECTOR2I &aCenter={ 0, 0 }) override
Rotate all vertices by a given angle.
int VertexCount(int aOutline=-1, int aHole=-1) const
Return the number of vertices in a given outline/hole.
POLYGON & Polygon(int aIndex)
Return the aIndex-th subpolygon in the set.
int Append(int x, int y, int aOutline=-1, int aHole=-1, bool aAllowDuplication=false)
Appends a vertex at the end of the given outline/hole (default: the last outline)
int AddHole(const SHAPE_LINE_CHAIN &aHole, int aOutline=-1)
Adds a new hole to the given outline (default: last) and returns its index.
SHAPE_LINE_CHAIN & Outline(int aIndex)
Return the reference to aIndex-th outline in the set.
SHAPE_LINE_CHAIN & Hole(int aOutline, int aHole)
Return the reference to aHole-th hole in the aIndex-th outline.
int NewOutline()
Creates a new empty polygon in the set and returns its index.
void Mirror(const VECTOR2I &aRef, FLIP_DIRECTION aFlipDirection)
Mirror the line points about y or x (or both)
int OutlineCount() const
Return the number of outlines in the set.
void Move(const VECTOR2I &aVector) override
void Fracture(bool aSimplify=true)
Convert a set of polygons with holes to a single outline with "slits"/"fractures" connecting the oute...
const SHAPE_LINE_CHAIN & COutline(int aIndex) const
void TransformToPolygon(SHAPE_POLY_SET &aBuffer, int aError, ERROR_LOC aErrorLoc) const override
Fills a SHAPE_POLY_SET with a polygon representation of this shape.
Simple container to manage line stroke parameters.
const std::string Format() const
Return the vector formatted as a string.
Handle a list of polygons defining a copper zone.
void SetDoNotAllowPads(bool aEnable)
void SetLocalClearance(std::optional< int > aClearance)
virtual void SetLayer(PCB_LAYER_ID aLayer) override
Set the layer this item is on.
void SetIsRuleArea(bool aEnable)
void SetDoNotAllowTracks(bool aEnable)
void SetLayerSet(const LSET &aLayerSet) override
void SetDoNotAllowVias(bool aEnable)
void SetDoNotAllowFootprints(bool aEnable)
void SetDoNotAllowZoneFills(bool aEnable)
void SetAssignedPriority(unsigned aPriority)
void SetPadConnection(ZONE_CONNECTION aPadConnection)
void SetOutline(SHAPE_POLY_SET *aOutline)
static bool empty(const wxTextEntryBase *aCtrl)
static constexpr EDA_ANGLE ANGLE_0
static constexpr EDA_ANGLE ANGLE_90
static constexpr EDA_ANGLE FULL_CIRCLE
static constexpr EDA_ANGLE ANGLE_360
static constexpr EDA_ANGLE ANGLE_180
@ RECTANGLE
Use RECTANGLE instead of RECT to avoid collision in a Windows header.
static const wxChar traceFabmaster[]
Flag to enable FABMASTER plugin debugging output.
#define THROW_IO_ERROR(msg)
macro which captures the "call site" values of FILE_, __FUNCTION & LINE
bool IsPcbLayer(int aLayer)
Test whether a layer is a valid layer for Pcbnew.
constexpr PCB_LAYER_ID PCBNEW_LAYER_ID_START
bool IsBackLayer(PCB_LAYER_ID aLayerId)
Layer classification: check if it's a back layer.
bool IsCopperLayer(int aLayerId)
Test whether a layer is a copper layer.
PCB_LAYER_ID
A quick note on layer IDs:
@ LEFT_RIGHT
Flip left to right (around the Y axis)
@ TOP_BOTTOM
Flip top to bottom (around the X axis)
bool AddHoleIfValid(SHAPE_POLY_SET &aOutline, SHAPE_LINE_CHAIN &&aHole)
Adds a hole to a polygon if it is valid (i.e.
std::vector< VECTOR2I > MakeRegularPolygonPoints(const VECTOR2I &aCenter, size_t aN, const VECTOR2I &aPt0)
Get the corners of a regular polygon from the centre, one point and the number of sides.
std::vector< SEG > MakeCrossSegments(const VECTOR2I &aCenter, const VECTOR2I &aSize, EDA_ANGLE aAngle)
Create the two segments for a cross.
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
std::map< wxString, NETINFO_ITEM * > NETNAMES_MAP
@ NPTH
like PAD_PTH, but not plated mechanical use only, no connection allowed
@ SMD
Smd pad, appears on the solder paste layer (default)
@ PTH
Plated through hole pad.
Class to handle a set of BOARD_ITEMs.
std::optional< VECTOR2I > OPT_VECTOR2I
Utility functions for working with shapes.
bool ReplaceIllegalFileNameChars(std::string &aName, int aReplaceChar)
Checks aName for illegal file name characters.
static std::vector< std::string > split(const std::string &aStr, const std::string &aDelim)
Split the input string into a vector of output strings.
A!LAYER_SORT!LAYER_SUBCLASS!LAYER_ARTWORK!LAYER_USE!LAYER_CONDUCTOR!LAYER_DIELECTRIC_CONSTANT !...
bool disable
! if true, prevent the layer elements from being used
std::string name
! LAYER_SUBCLASS
int layerid
! pcbnew layer (assigned)
bool conductive
! LAYER_CONDUCTOR
bool positive
! LAYER_ARTWORK (either POSITIVE or NEGATIVE)
A!SUBCLASS!PAD_SHAPE_NAME!GRAPHIC_DATA_NAME!GRAPHIC_DATA_NUMBER!RECORD_TAG!GRAPHIC_DATA_1!
std::string name
! SYM_NAME
std::string refdes
! REFDES
std::string subclass
! SUBCLASS
std::unique_ptr< graphic_element > elements
int end_x
! GRAPHIC_DATA_3
SHAPE_ARC result
! KiCad-style arc representation
int center_x
! GRAPHIC_DATA_5
bool clockwise
! GRAPHIC_DATA_9
int center_y
! GRAPHIC_DATA_6
int end_y
! GRAPHIC_DATA_4
int size_y
! GRAPHIC_DATA_4
int size_x
! GRAPHIC_DATA_3
std::string graphic_data8
std::string graphic_data1
std::string graphic_dataname
std::string graphic_data6
std::string graphic_data7
std::string graphic_data2
std::string graphic_data3
std::string graphic_data10
std::string graphic_datanum
std::string graphic_data4
std::string graphic_data5
std::string graphic_data9
std::string layer
! SUBCLASS
int subseq
! RECORD_TAG[1]
int width
! Various sections depending on type
GRAPHIC_SHAPE shape
! Shape of the graphic_item
int start_y
! GRAPHIC_DATA_2
int start_x
! GRAPHIC_DATA_1
GRAPHIC_TYPE type
! Type of graphic item
int end_x
! GRAPHIC_DATA_3
bool oblong_x
! OBLONG_X (as opposed to OBLONG_Y)
int size_x
! GRAPHIC_DATA_3
int size_y
! GRAPHIC_DATA_4
std::vector< VECTOR2I > m_pts
bool fill
! GRAPHIC_DATA_5
int end_y
! GRAPHIC_DATA_4
int end_x
! GRAPHIC_DATA_3
double rotation
! GRAPHIC_DATA_3
std::string text
! GRAPHIC_DATA_7
int height
! GRAPHIC_DATA_6[2]
int thickness
! GRAPHIC_DATA_6[6]
GR_TEXT_H_ALIGN_T orient
! GRAPHIC_DATA_5
bool ital
! GRAPHIC_DATA_6[4] != 0.0
bool mirror
! GRAPHIC_DATA_4
std::string refdes
!< NET_NAME
std::string pin_num
!< REFDES
std::string pin_name
!< PIN_NUMBER
graphic_element segment
! GRAPHIC_DATA (can be either LINE or ARC)
@ USER
The field ID hasn't been set yet; field is invalid.
std::vector< std::string > header
const SHAPE_LINE_CHAIN chain
SHAPE_CIRCLE circle(c.m_circle_center, c.m_circle_radius)
wxString result
Test unit parsing edge cases and error handling.
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.
VECTOR2< int32_t > VECTOR2I
VECTOR2< double > VECTOR2D
bool AutoAssignZonePriorities(BOARD *aBoard, PROGRESS_REPORTER *aReporter)
Automatically assign zone priorities based on connectivity analysis of overlapping regions.
@ FULL
pads are covered by copper