59#include <wx/filename.h>
72 const unsigned PROGRESS_DELTA = 250;
93 wxCHECK_MSG( !aStr.empty(), 0.0,
"Empty string passed to readDouble" );
95 std::istringstream istr( aStr );
96 istr.imbue( std::locale::classic() );
107 wxCHECK_MSG( !aStr.empty(), 0,
"Empty string passed to readInt" );
109 std::istringstream istr( aStr );
110 istr.imbue( std::locale::classic() );
120 std::ifstream ifs( aFile, std::ios::in | std::ios::binary );
128 ifs.ignore( std::numeric_limits<std::streamsize>::max() );
129 std::streamsize length = ifs.gcount();
131 ifs.seekg( 0, std::ios_base::beg );
133 std::string buffer( std::istreambuf_iterator<char>{ ifs }, {} );
135 std::vector < std::string > row;
139 row.reserve( length / 100 );
145 for(
auto& ch : buffer )
151 if( cell.empty() || cell[0] ==
'"' )
160 row.push_back( cell );
172 row.push_back( cell );
175 rows.push_back( row );
184 cell += std::toupper( ch );
189 if( !cell.empty() || !row.empty() )
191 row.push_back( cell );
193 rows.push_back( row );
207 row =
rows.at( aOffset );
209 catch( std::out_of_range& )
217 if( row[0].back() !=
'A' )
220 std::string row1 = row[1];
221 std::string row2 = row[2];
226 std::erase_if( row1, [](
char c ){
return c ==
'_'; } );
227 std::erase_if( row2, [](
char c ){
return c ==
'_'; } );
232 std::erase_if( row3, [](
char c ){
return c ==
'_'; } );
235 if( row1 ==
"REFDES" && row2 ==
"COMPCLASS" )
238 if( row1 ==
"NETNAME" && row2 ==
"REFDES" )
241 if( row1 ==
"CLASS" && row2 ==
"SUBCLASS" && row3.empty() )
244 if( row1 ==
"GRAPHICDATANAME" && row2 ==
"GRAPHICDATANUMBER" )
247 if( row1 ==
"CLASS" && row2 ==
"SUBCLASS" && row3 ==
"GRAPHICDATANAME" )
250 if( row1 ==
"SYMNAME" && row2 ==
"PINNAME" )
253 if( row1 ==
"SYMNAME" && row2 ==
"SYMMIRROR" && row3 ==
"PINNAME" )
256 if( row1 ==
"VIAX" && row2 ==
"VIAY" )
259 if( row1 ==
"SUBCLASS" && row2 ==
"PADSHAPENAME" )
262 if( row1 ==
"PADNAME" )
265 if( row1 ==
"LAYERSORT" )
268 wxLogError(
_(
"Unknown FABMASTER section %s:%s at row %zu." ),
281 if( aRow >=
rows.size() )
284 if(
rows[aRow].size() < 11 )
286 wxLogError(
_(
"Invalid row size in J row %zu. Expecting 11 elements but found %zu." ),
292 for(
int i = 7; i < 10 && retval < 1.0; ++i )
294 std::string units =
rows[aRow][i];
295 std::transform(units.begin(), units.end(),units.begin(), ::toupper);
297 if( units ==
"MILS" )
299 else if( units ==
"MILLIMETERS" )
301 else if( units ==
"MICRONS" )
303 else if( units ==
"INCHES" )
309 wxLogError(
_(
"Could not find units value, defaulting to mils." ) );
319 if( aRow >=
rows.size() )
322 std::vector<std::string> header =
rows[aRow];
324 for(
size_t i = 0; i < header.size(); i++ )
328 std::erase_if( header[i], [](
const char c ) {
return c ==
'_'; } );
330 if( header[i] == aStr )
334 THROW_IO_ERROR( wxString::Format(
_(
"Could not find column label %s." ), aStr.c_str() ) );
341 const auto& kicad_layer =
layers.find( aLayerName);
343 if( kicad_layer ==
layers.end() )
346 return static_cast<PCB_LAYER_ID>( kicad_layer->second.layerid );
352 size_t rownum = aRow + 2;
354 if( rownum >=
rows.size() )
362 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
366 if( row.size() != header.size() )
368 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
375 auto& pad_num = row[pad_num_col];
376 auto& pad_layer = row[pad_lay_col];
379 if( pad_layer ==
"INTERNAL_PAD_DEF" || pad_layer ==
"internal_pad_def" )
383 if( pad_layer[0] ==
'~' )
392 layer.
name = pad_layer;
410 size_t rownum = aRow + 2;
412 if( rownum >=
rows.size() )
418 if( scale_factor <= 0.0 )
432 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
437 if( row.size() != header.size() )
439 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
446 auto& pad_name = row[pad_name_col];
447 auto& pad_num = row[pad_num_col];
448 auto& pad_layer = row[pad_lay_col];
449 auto& pad_is_via = row[pad_via_col];
450 auto& pad_shape = row[pad_shape_col];
451 auto& pad_width = row[pad_width_col];
452 auto& pad_height = row[pad_height_col];
453 auto& pad_xoff = row[pad_xoff_col];
454 auto& pad_yoff = row[pad_yoff_col];
455 auto& pad_shapename = row[pad_shape_name_col];
458 if( pad_layer ==
"INTERNAL_PAD_DEF" || pad_layer ==
"internal_pad_def" )
463 auto new_pad =
pads.find( pad_name );
465 if( new_pad !=
pads.end() )
466 pad = &new_pad->second;
471 pad->name = pad_name;
475 if( pad_layer ==
"~DRILL" )
489 wxLogError(
_(
"Expecting drill size value but found %s!%s!%s in row %zu." ),
507 if( drill_x == drill_y )
509 pad->drill_size_x = drill_hit;
510 pad->drill_size_y = drill_hit;
514 pad->drill_size_x = drill_x;
515 pad->drill_size_y = drill_y;
518 if( !pad_shapename.empty() && pad_shapename[0] ==
'P' )
524 if( pad_shape.empty() )
537 wxLogError(
_(
"Expecting pad size values but found %s : %s in row %zu." ),
544 auto layer =
layers.find( pad_layer );
546 if( w > 0.0 && layer !=
layers.end() && layer->second.conductive )
547 pad->copper_layers.insert( pad_layer );
552 if( layer !=
layers.end() )
554 if( layer->second.layerid ==
F_Cu )
556 else if( layer->second.layerid ==
B_Cu )
560 if( w > std::numeric_limits<int>::max() || h > std::numeric_limits<int>::max() )
562 wxLogError(
_(
"Invalid pad size in row %zu." ), rownum );
566 if( pad_layer ==
"~TSM" || pad_layer ==
"~BSM" )
568 if( w > 0.0 && h > 0.0 )
576 if( pad_layer ==
"~TSP" || pad_layer ==
"~BSP" )
578 if( w > 0.0 && h > 0.0 )
587 if( pad_layer[0] ==
'~' )
597 wxLogError(
_(
"Expecting pad offset values but found %s:%s in row %zu." ),
604 if( w > 0.0 && h > 0.0 && recnum == 1 )
608 pad->via = ( std::toupper( pad_is_via[0] ) !=
'V' );
610 if( pad_shape ==
"CIRCLE" )
615 else if( pad_shape ==
"RECTANGLE" )
619 else if( pad_shape ==
"ROUNDED_RECT" )
623 else if( pad_shape ==
"SQUARE" )
628 else if( pad_shape ==
"OBLONG" || pad_shape ==
"OBLONG_X" || pad_shape ==
"OBLONG_Y" )
630 else if( pad_shape ==
"OCTAGON" )
633 pad->is_octogon =
true;
635 else if( pad_shape ==
"SHAPE" )
638 pad->custom_name = pad_shapename;
642 wxLogError(
_(
"Unknown pad shape name '%s' on layer '%s' in row %zu." ),
651 return rownum - aRow;
657 size_t rownum = aRow + 2;
659 if( rownum >=
rows.size() )
662 auto& header =
rows[aRow];
665 if( scale_factor <= 0.0 )
671 if( layer_class_col < 0 || layer_subclass_col < 0 )
674 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
678 if( row.size() != header.size() )
680 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
690 layer.
name = row[layer_subclass_col];
694 if( row[layer_class_col] ==
"ANTI ETCH" )
699 else if( row[layer_class_col] ==
"ETCH" )
705 return rownum - aRow;
711 std::vector<std::pair<std::string, int>> extra_layers
713 {
"ASSEMBLY_TOP",
F_Fab },
714 {
"ASSEMBLY_BOTTOM",
B_Fab },
715 {
"PLACE_BOUND_TOP",
F_CrtYd },
716 {
"PLACE_BOUND_BOTTOM",
B_CrtYd },
719 std::vector<FABMASTER_LAYER*> layer_order;
721 int next_user_layer =
User_1;
730 layer_order.push_back( &layer );
732 else if( ( layer.
name.find(
"SILK" ) != std::string::npos
733 && layer.
name.find(
"AUTOSILK" )
734 == std::string::npos )
735 || layer.
name.find(
"DISPLAY" ) != std::string::npos )
737 if( layer.
name.find(
"B" ) != std::string::npos )
742 else if( layer.
name.find(
"MASK" ) != std::string::npos ||
743 layer.
name.find(
"MSK" ) != std::string::npos )
745 if( layer.
name.find(
"B" ) != std::string::npos )
750 else if( layer.
name.find(
"PAST" ) != std::string::npos )
752 if( layer.
name.find(
"B" ) != std::string::npos )
757 else if( layer.
name.find(
"NCLEGEND" ) != std::string::npos )
766 if( layer.
name.find(
"AUTOSILK" ) == std::string::npos )
768 if( next_user_layer <=
User_9 )
771 layer.
layerid = next_user_layer;
772 next_user_layer += 2;
780 wxLogWarning(
_(
"No user layer to put layer %s" ), layer.
name );
788 for(
size_t layeri = 0; layeri < layer_order.size(); ++layeri )
793 else if( layeri == layer_order.size() - 1 )
796 layer->
layerid = layeri * 2 + 2;
799 for(
auto& new_pair : extra_layers )
803 new_layer.
name = new_pair.first;
804 new_layer.
layerid = new_pair.second;
807 auto result =
layers.emplace( new_pair.first, new_layer );
811 result.first->second.layerid = new_pair.second;
812 result.first->second.disable =
false;
816 for(
const auto& [layer_name, fabmaster_layer] :
layers )
818 wxLogTrace(
traceFabmaster, wxT(
"Layer %s -> KiCad layer %d" ), layer_name,
819 fabmaster_layer.layerid );
833 size_t rownum = aRow + 2;
835 if( rownum >=
rows.size() )
838 auto& header =
rows[aRow];
841 if( scale_factor <= 0.0 )
849 int layer_er_col =
getColFromName( aRow,
"LAYERDIELECTRICCONSTANT" );
850 int layer_rho_col =
getColFromName( aRow,
"LAYERELECTRICALCONDUCTIVITY" );
853 if( layer_sort_col < 0 || layer_subclass_col < 0 || layer_art_col < 0 || layer_use_col < 0
854 || layer_cond_col < 0 || layer_er_col < 0 || layer_rho_col < 0 || layer_mat_col < 0 )
857 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
861 if( row.size() != header.size() )
863 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
870 auto& layer_sort = row[layer_sort_col];
871 auto& layer_subclass = row[layer_subclass_col];
872 auto& layer_art = row[layer_art_col];
873 auto& layer_cond = row[layer_cond_col];
874 auto& layer_mat = row[layer_mat_col];
876 if( layer_mat ==
"AIR" )
881 if( layer_subclass.empty() )
883 if( layer_cond !=
"NO" )
884 layer.
name =
"In.Cu" + layer_sort;
886 layer.
name =
"Dielectric" + layer_sort;
889 layer.
positive = ( layer_art !=
"NEGATIVE" );
894 return rownum - aRow;
905 size_t rownum = aRow + 2;
907 if( rownum >=
rows.size() )
910 auto& header =
rows[aRow];
913 if( scale_factor <= 0.0 )
918 int pad_grdata_name_col =
getColFromName( aRow,
"GRAPHICDATANAME" );
919 int pad_grdata_num_col =
getColFromName( aRow,
"GRAPHICDATANUMBER" );
934 if( pad_subclass_col < 0 || pad_shape_name_col < 0 || pad_grdata1_col < 0 || pad_grdata2_col < 0
935 || pad_grdata3_col < 0 || pad_grdata4_col < 0 || pad_grdata5_col < 0
936 || pad_grdata6_col < 0 || pad_grdata7_col < 0 || pad_grdata8_col < 0
937 || pad_grdata9_col < 0 || pad_stack_name_col < 0 || pad_refdes_col < 0
938 || pad_pin_num_col < 0 )
941 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
945 if( row.size() != header.size() )
947 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
955 auto& pad_layer = row[pad_subclass_col];
956 auto pad_shape_name = row[pad_shape_name_col];
957 auto& pad_record_tag = row[pad_record_tag_col];
972 auto& pad_stack_name = row[pad_stack_name_col];
973 auto& pad_refdes = row[pad_refdes_col];
974 auto& pad_pin_num = row[pad_pin_num_col];
978 std::string prefix(
"FIG_SHAPE " );
980 if( pad_shape_name.length() <= prefix.length()
981 || !std::equal( prefix.begin(), prefix.end(), pad_shape_name.begin() ) )
991 if( std::sscanf( pad_record_tag.c_str(),
"%d %d", &
id, &seq ) != 2 )
993 wxLogError(
_(
"Invalid format for id string '%s' in custom pad row %zu." ),
994 pad_record_tag.c_str(),
999 auto name = pad_shape_name.substr( prefix.length() );
1000 name +=
"_" + pad_refdes +
"_" + pad_pin_num;
1003 auto& custom_pad = ret.first->second;
1009 custom_pad.name =
name;
1010 custom_pad.padstack = pad_stack_name;
1011 custom_pad.pinnum = pad_pin_num;
1012 custom_pad.refdes = pad_refdes;
1019 auto gr_item = std::unique_ptr<GRAPHIC_ITEM>(
processGraphic( gr_data, scale_factor ) );
1023 gr_item->layer = pad_layer;
1024 gr_item->refdes = pad_refdes;
1026 gr_item->subseq = 0;
1031 auto retval = pad_it.first->second.insert( std::move(gr_item ) );
1033 if( !retval.second )
1035 wxLogError(
_(
"Could not insert graphical item %d into padstack '%s'." ),
1037 pad_stack_name.c_str() );
1042 wxLogError(
_(
"Unrecognized pad shape primitive '%s' in row %zu." ),
1048 return rownum - aRow;
1098 angle = endangle - startangle;
1133 std::unique_ptr<GRAPHIC_ARC> new_circle = std::make_unique<GRAPHIC_ARC>();
1144 if( size.
x != size.
y )
1146 wxLogError(
_(
"Circle with unequal x and y radii (x=%d, y=%d)" ), size.
x, size.
y );
1152 new_circle->radius = size.
x / 2;
1158 new_circle->start_x = start.
x;
1159 new_circle->start_y = start.
y;
1161 new_circle->end_x = start.
x;
1162 new_circle->end_y = start.
y;
1164 new_circle->center_x =
center.x;
1165 new_circle->center_y =
center.y;
1167 new_circle->clockwise =
true;
1169 new_circle->result =
SHAPE_ARC{ start, mid, start, 0 };
1171 return new_circle.release();
1191 new_rect->
width = 0;
1205 auto new_rect = std::make_unique<GRAPHIC_RECTANGLE>();
1214 new_rect->start_x = center_x - size_x / 2;
1215 new_rect->start_y = center_y + size_y / 2;
1216 new_rect->end_x = center_x + size_x / 2;
1217 new_rect->end_y = center_y - size_y / 2;
1219 new_rect->width = 0;
1221 return new_rect.release();
1246 auto new_oblong = std::make_unique<GRAPHIC_OBLONG>();
1258 return new_oblong.release();
1288 auto new_poly = std::make_unique<GRAPHIC_POLYGON>();
1293 bool across_corners =
true;
1327 across_corners =
false;
1333 wxCHECK_MSG(
false,
nullptr,
1334 wxString::Format(
"Unhandled polygon type: %s", aData.
graphic_dataname ) );
1339 return new_poly.release();
1350 auto new_cross = std::make_unique<GRAPHIC_CROSS>();
1359 return new_cross.release();
1383 if( toks.size() < 8 )
1386 wxLogError(
_(
"Invalid token count. Expected 8 but found %zu." ), toks.size() );
1388 new_text->
width = 0;
1389 new_text->
ital =
false;
1463 size_t rownum = aRow + 2;
1465 if( rownum >=
rows.size() )
1471 if( scale_factor <= 0.0 )
1490 if( geo_name_col < 0 || geo_num_col < 0 || geo_grdata1_col < 0 || geo_grdata2_col < 0
1491 || geo_grdata3_col < 0 || geo_grdata4_col < 0 || geo_grdata5_col < 0
1492 || geo_grdata6_col < 0 || geo_grdata7_col < 0 || geo_grdata8_col < 0
1493 || geo_grdata9_col < 0 || geo_subclass_col < 0 || geo_sym_name_col < 0
1494 || geo_refdes_col < 0 )
1497 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1501 if( row.size() != header.size() )
1503 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1510 auto& geo_tag = row[geo_tag_col];
1525 auto& geo_refdes = row[geo_refdes_col];
1533 if( std::sscanf( geo_tag.c_str(),
"%d %d %d", &
id, &seq, &subseq ) < 2 )
1535 wxLogError(
_(
"Invalid format for record_tag string '%s' in row %zu." ),
1541 auto gr_item = std::unique_ptr<GRAPHIC_ITEM>(
processGraphic( gr_data, scale_factor ) );
1546 gr_item->layer = row[geo_subclass_col];
1548 gr_item->subseq = subseq;
1550 if( geo_refdes.empty() )
1555 new_gr.
subclass = row[geo_subclass_col];
1556 new_gr.
refdes = row[geo_refdes_col];
1557 new_gr.
name = row[geo_sym_name_col];
1559 new_gr.
elements = std::make_unique<graphic_element>();
1564 graphic.
elements->emplace( std::move( gr_item ) );
1569 std::map<int, GEOM_GRAPHIC>{} );
1570 auto map_it = sym_gr_it.first->second.emplace(
id,
GEOM_GRAPHIC{} );
1571 auto& gr = map_it.first;
1575 gr->second.subclass = row[geo_subclass_col];
1576 gr->second.refdes = row[geo_refdes_col];
1577 gr->second.name = row[geo_sym_name_col];
1579 gr->second.elements = std::make_unique<graphic_element>();
1582 gr->second.elements->emplace( std::move( gr_item ) );
1586 return rownum - aRow;
1595 size_t rownum = aRow + 2;
1597 if( rownum >=
rows.size() )
1603 if( scale_factor <= 0.0 )
1612 if( viax_col < 0 || viay_col < 0 || padstack_name_col < 0 || net_name_col < 0
1613 || test_point_col < 0 )
1616 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1620 if( row.size() != header.size() )
1622 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1629 vias.emplace_back( std::make_unique<FM_VIA>() );
1634 via->padstack = row[padstack_name_col];
1635 via->net = row[net_name_col];
1636 via->test_point = ( row[test_point_col] ==
"YES" );
1639 return rownum - aRow;
1650 size_t rownum = aRow + 2;
1652 if( rownum >=
rows.size() )
1658 if( scale_factor <= 0.0 )
1664 int grdata_num_col =
getColFromName( aRow,
"GRAPHICDATANUMBER" );
1677 if( class_col < 0 || layer_col < 0 || grdata_name_col < 0 || grdata_num_col < 0
1678 || tag_col < 0 || grdata1_col < 0 || grdata2_col < 0 || grdata3_col < 0
1679 || grdata4_col < 0 || grdata5_col < 0 || grdata6_col < 0 || grdata7_col < 0
1680 || grdata8_col < 0 || grdata9_col < 0 || netname_col < 0 )
1683 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1687 if( row.size() != header.size() )
1689 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1709 const std::string& geo_tag = row[tag_col];
1716 if( std::sscanf( geo_tag.c_str(),
"%d %d %d", &
id, &seq, &subseq ) < 2 )
1718 wxLogError(
_(
"Invalid format for record_tag string '%s' in row %zu." ),
1724 auto gr_item = std::unique_ptr<GRAPHIC_ITEM>(
processGraphic( gr_data, scale_factor ) );
1728 wxLogTrace(
traceFabmaster,
_(
"Unhandled graphic item '%s' in row %zu." ),
1734 auto new_trace = std::make_unique<TRACE>();
1736 new_trace->layer = row[layer_col];
1737 new_trace->netname = row[netname_col];
1738 new_trace->lclass = row[class_col];
1740 gr_item->layer = row[layer_col];
1742 gr_item->subseq = subseq;
1745 if( new_trace->lclass ==
"REF DES" )
1747 auto result =
refdes.emplace( std::move( new_trace ) );
1748 auto& ref = *
result.first;
1749 ref->segment.emplace( std::move( gr_item ) );
1751 else if( new_trace->lclass ==
"DEVICE TYPE" || new_trace->lclass ==
"COMPONENT VALUE"
1752 || new_trace->lclass ==
"TOLERANCE" )
1765 else if( gr_item->width == 0 )
1767 auto result =
zones.emplace( std::move( new_trace ) );
1768 auto& zone = *
result.first;
1769 auto gr_result = zone->segment.emplace( std::move( gr_item ) );
1771 if( !gr_result.second )
1773 wxLogError(
_(
"Duplicate item for ID %d and sequence %d in row %zu." ),
1781 auto result =
traces.emplace( std::move( new_trace ) );
1782 auto& trace = *
result.first;
1783 auto gr_result = trace->segment.emplace( std::move( gr_item ) );
1785 if( !gr_result.second )
1787 wxLogError(
_(
"Duplicate item for ID %d and sequence %d in row %zu." ),
1795 return rownum - aRow;
1801 if( aSymType ==
"PACKAGE" )
1803 else if( aSymType ==
"DRAFTING")
1805 else if( aSymType ==
"MECHANICAL" )
1807 else if( aSymType ==
"FORMAT" )
1816 if( aCmpClass ==
"IO" )
1818 else if( aCmpClass ==
"IC" )
1820 else if( aCmpClass ==
"DISCRETE" )
1833 size_t rownum = aRow + 2;
1835 if( rownum >=
rows.size() )
1841 if( scale_factor <= 0.0 )
1849 int compinscode_col =
getColFromName( aRow,
"COMPINSERTIONCODE" );
1860 if( refdes_col < 0 || compclass_col < 0 || comppartnum_col < 0 || compheight_col < 0
1861 || compdevlabelcol < 0 || compinscode_col < 0 || symtype_col < 0 || symname_col < 0
1862 || symmirror_col < 0 || symrotate_col < 0 || symx_col < 0 || symy_col < 0
1863 || compvalue_col < 0 || comptol_col < 0 || compvolt_col < 0 )
1866 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1870 if( row.size() != header.size() )
1872 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1879 const wxString& comp_refdes = row[refdes_col];
1881 if( row[symx_col].
empty() || row[symy_col].
empty() || row[symrotate_col].
empty() )
1883 wxLogError(
_(
"Missing X, Y, or rotation data in row %zu for refdes %s. "
1884 "This may be an unplaced component." ),
1885 rownum, comp_refdes );
1889 auto cmp = std::make_unique<COMPONENT>();
1891 cmp->refdes = comp_refdes;
1893 cmp->pn = row[comppartnum_col];
1894 cmp->height = row[compheight_col];
1895 cmp->dev_label = row[compdevlabelcol];
1896 cmp->insert_code = row[compinscode_col];
1898 cmp->name = row[symname_col];
1899 cmp->mirror = ( row[symmirror_col] ==
"YES" );
1900 cmp->rotate =
readDouble( row[symrotate_col] );
1903 cmp->value = row[compvalue_col];
1904 cmp->tol = row[comptol_col];
1905 cmp->voltage = row[compvolt_col];
1911 auto retval =
components.insert( std::make_pair( cmp->refdes, std::vector<std::unique_ptr<COMPONENT>>{} ) );
1916 vec->second.push_back( std::move( cmp ) );
1919 return rownum - aRow;
1929 size_t rownum = aRow + 2;
1931 if( rownum >=
rows.size() )
1937 if( scale_factor <= 0.0 )
1951 if( symname_col < 0 ||symmirror_col < 0 || pinname_col < 0 || pinnum_col < 0 || pinx_col < 0
1952 || piny_col < 0 || padstack_col < 0 || refdes_col < 0 || pinrot_col < 0
1953 || testpoint_col < 0 )
1956 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1960 if( row.size() != header.size() )
1962 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1969 auto pin = std::make_unique<PIN>();
1971 pin->name = row[symname_col];
1972 pin->mirror = ( row[symmirror_col] ==
"YES" );
1973 pin->pin_name = row[pinname_col];
1974 pin->pin_number = row[pinnum_col];
1977 pin->padstack = row[padstack_col];
1978 pin->refdes = row[refdes_col];
1984 std::string pin_key =
pin->refdes.empty() ?
pin->name :
pin->refdes;
1986 auto map_it =
pins.find( pin_key );
1988 if( map_it ==
pins.end() )
1990 auto retval =
pins.insert( std::make_pair( pin_key, std::set<std::unique_ptr<PIN>,
1992 map_it = retval.first;
1995 map_it->second.insert( std::move(
pin ) );
1998 return rownum - aRow;
2007 size_t rownum = aRow + 2;
2009 if( rownum >=
rows.size() )
2015 if( scale_factor <= 0.0 )
2025 if( netname_col < 0 || refdes_col < 0 || pinnum_col < 0 || pinname_col < 0 || pingnd_col < 0
2029 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
2033 if( row.size() != header.size() )
2035 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
2043 new_net.
name = row[netname_col];
2044 new_net.
refdes = row[refdes_col];
2045 new_net.
pin_num = row[pinnum_col];
2046 new_net.
pin_name = row[pinname_col];
2047 new_net.
pin_gnd = ( row[pingnd_col] ==
"YES" );
2048 new_net.
pin_pwr = ( row[pinpwr_col] ==
"YES" );
2051 netnames.insert( row[netname_col] );
2054 return rownum - aRow;
2061 for(
size_t i = 0; i <
rows.size(); )
2075 i += std::max( retval, 1 );
2083 i += std::max( retval, 1 );
2091 i += std::max( retval, 1 );
2099 i += std::max( retval, 1 );
2107 i += std::max( retval, 1 );
2115 i += std::max( retval, 1 );
2123 i += std::max( retval, 1 );
2131 i += std::max( retval, 1 );
2139 i += std::max( retval, 1 );
2147 i += std::max( retval, 1 );
2164 for(
auto& zone :
zones )
2174 if( zone->layer ==
"OUTLINE" || zone->layer ==
"DESIGN_OUTLINE" )
2195 std::set<ZONE*> zones_to_delete;
2196 std::set<ZONE*> matched_fills;
2198 for(
auto zone : aBoard->
Zones() )
2200 if( zone->GetNetCode() > 0 )
2201 zones_to_delete.insert( zone );
2204 for(
auto zone1 : aBoard->
Zones() )
2206 if( zone1->GetNetCode() > 0 )
2211 std::map<int, std::vector<ZONE*>> net_to_fills;
2213 for(
auto zone2 : aBoard->
Zones() )
2215 if( zone2->GetNetCode() <= 0 )
2220 if( zone1->GetLayer() != zone2->GetLayer() )
2226 size_t match_count = 0;
2228 for(
auto& pt1 : outline1.
CPoints() )
2234 for(
auto& pt2 : outline2.
CPoints() )
2240 if( match_count > 0 )
2242 overlaps[zone2->GetNetCode()] += match_count;
2243 net_to_fills[zone2->GetNetCode()].push_back( zone2 );
2248 size_t max_net_id = 0;
2250 for(
size_t el = 1; el < overlaps.size(); ++el )
2252 if( overlaps[el] > max_net )
2254 max_net = overlaps[el];
2261 zone1->SetNetCode( max_net_id );
2263 for(
ZONE* fill : net_to_fills[max_net_id] )
2264 matched_fills.insert( fill );
2268 for(
auto zone : zones_to_delete )
2270 if( matched_fills.find( zone ) != matched_fills.end() )
2291 if( aMirrorPoint.has_value() )
2324 for(
const auto& [pinKey, pinSet] :
pins )
2326 if( pinSet.empty() )
2332 const auto& firstPin = *pinSet.begin();
2334 int minX = firstPin->pin_x;
2335 int maxX = firstPin->pin_x;
2336 int minY = firstPin->pin_y;
2337 int maxY = firstPin->pin_y;
2339 for(
const auto&
pin : pinSet )
2341 minX = std::min( minX,
pin->pin_x );
2342 maxX = std::max( maxX,
pin->pin_x );
2343 minY = std::min( minY,
pin->pin_y );
2344 maxY = std::max( maxY,
pin->pin_y );
2347 auto cmp = std::make_unique<COMPONENT>();
2348 cmp->refdes = pinKey;
2349 cmp->name = firstPin->name;
2350 cmp->mirror = firstPin->mirror;
2352 cmp->x = ( minX + maxX ) / 2;
2353 cmp->y = ( minY + maxY ) / 2;
2357 std::vector<std::unique_ptr<COMPONENT>> compVec;
2358 compVec.push_back( std::move( cmp ) );
2359 components.insert( std::make_pair( pinKey, std::move( compVec ) ) );
2373 bool has_multiple = mod.second.size() > 1;
2375 for(
int i = 0; i < (int) mod.second.size(); ++i )
2377 auto& src = mod.second[i];
2381 wxString mod_ref = src->name;
2385 mod_ref.Append( wxString::Format( wxT(
"_%d" ), i ) );
2390 wxString key = !lib_ref.empty() ? lib_ref + wxT(
":" ) + mod_ref : mod_ref;
2393 fpID.
Parse( key,
true );
2402 wxString reference = src->refdes;
2404 if( !std::isalpha( src->refdes[0] ) )
2405 reference.Prepend(
"UNK" );
2418 for(
auto& ref :
refdes )
2421 static_cast<const GRAPHIC_TEXT&
>( **ref->segment.begin() );
2423 if( lsrc.
text == src->refdes )
2430 wxLogTrace(
traceFabmaster, wxS(
"The layer %s is not mapped?" ),
2431 ref->layer.c_str() );
2442 flip_point =
VECTOR2I( src->x, src->y );
2447 setupText( lsrc, layer, *txt, *aBoard, flip_point );
2462 for(
auto& gr_ref : gr_it->second )
2464 auto& graphic = gr_ref.second;
2466 for(
auto& seg : *graphic.elements )
2473 STROKE_PARAMS defaultStroke( ds.GetLineThickness( layer ) );
2475 switch( seg->shape )
2498 if( lsrc->
width == 0 )
2511 circle->SetLayer( layer );
2523 if( lsrc.
width == 0 )
2529 if( lsrc.
layer ==
"DISPLAY_TOP" || lsrc.
layer ==
"DISPLAY_BOTTOM" )
2530 circle->SetFilled(
true );
2532 circle->SetWidth( ds.GetLineThickness(
circle->GetLayer() ) );
2546 std::unique_ptr<PCB_SHAPE> arc =
2560 arc->SetLayer( layer );
2564 if( lsrc->
width == 0 )
2565 arc->SetStroke( defaultStroke );
2604 std::unique_ptr<PCB_TEXT> txt = std::make_unique<PCB_TEXT>( fp );
2609 flip_point =
VECTOR2I( src->x, src->y );
2611 setupText( lsrc, layer, *txt, *aBoard, flip_point );
2636 auto pin_it =
pins.find( src->refdes );
2639 if( pin_it ==
pins.end() )
2640 pin_it =
pins.find( src->name );
2642 if( pin_it !=
pins.end() )
2644 for(
auto&
pin : pin_it->second )
2646 auto pin_net_it =
pin_nets.find( std::make_pair(
pin->refdes,
2647 pin->pin_number ) );
2648 auto padstack =
pads.find(
pin->padstack );
2649 std::string netname =
"";
2652 netname = pin_net_it->second.name;
2654 auto net_it = netinfo.find( netname );
2656 std::unique_ptr<PAD> newpad = std::make_unique<PAD>( fp );
2658 if( net_it != netinfo.end() )
2659 newpad->SetNet( net_it->second );
2661 newpad->SetNetCode( 0 );
2663 newpad->SetX(
pin->pin_x );
2666 newpad->SetY( 2 * src->y -
pin->pin_y );
2668 newpad->SetY(
pin->pin_y );
2670 newpad->SetNumber(
pin->pin_number );
2672 if( padstack ==
pads.end() )
2674 wxLogError(
_(
"Unable to locate padstack %s in file %s\n" ),
2680 auto&
pad = padstack->second;
2688 int pad_size = std::min(
pad.width,
pad.height );
2691 VECTOR2I( pad_size / 2, pad_size / 2 ) );
2693 std::string custom_name =
pad.custom_name +
"_" +
pin->refdes +
"_" +
2695 auto custom_it =
pad_shapes.find( custom_name );
2701 int last_subseq = 0;
2708 for(
const auto& el : (*custom_it).second.elements )
2714 if(
getLayer( ( *( el.second.begin() ) )->layer ) != primary_layer )
2717 for(
const auto& seg : el.second )
2719 if( seg->subseq > 0 || seg->subseq != last_subseq )
2721 poly_outline.
Polygon(0).back().SetClosed(
true );
2729 if( poly_outline.
VertexCount( 0, hole_idx ) == 0 )
2749 wxLogError(
_(
"Invalid custom pad '%s'. Replacing with "
2751 custom_name.c_str() );
2758 poly_outline.
Move( -newpad->GetPosition() );
2781 wxLogError(
_(
"Invalid custom pad '%s'. Replacing with "
2783 custom_name.c_str() );
2789 wxLogError(
_(
"Could not find custom pad '%s'." ),
2790 custom_name.c_str() );
2812 if(
pad.drill_size_x ==
pad.drill_size_y )
2817 newpad->SetDrillSize(
VECTOR2I(
pad.drill_size_x,
pad.drill_size_y ) );
2825 else if(
pad.bottom )
2826 newpad->SetLayerSet(
PAD::SMDMask().FlipStandardLayers() );
2831 newpad->SetOrientation(
EDA_ANGLE( -src->rotate +
pin->rotation,
2834 newpad->SetOrientation(
EDA_ANGLE( src->rotate -
pin->rotation,
2837 if( newpad->GetSizeX() > 0 || newpad->GetSizeY() > 0 )
2843 wxLogError(
_(
"Invalid zero-sized pad ignored in\nfile: %s" ),
2870 for(
auto& layer :
layers )
2875 layer_set.
set( layer.second.layerid );
2880 for(
auto& layer :
layers )
2882 if( layer.second.conductive )
2885 layer.second.name );
2899 std::vector<const FABMASTER_LAYER*> conductiveLayers;
2901 for(
const auto& layer :
layers )
2903 if( layer.second.conductive )
2904 conductiveLayers.push_back( &layer.second );
2913 auto net_it = netinfo.find(
via->net );
2914 auto padstack =
pads.find(
via->padstack );
2920 if( net_it != netinfo.end() )
2921 new_via->
SetNet( net_it->second );
2923 if( padstack ==
pads.end() )
2927 if( !ds.m_ViasDimensionsList.empty() )
2930 new_via->
SetDrill( ds.m_ViasDimensionsList[0].m_Drill );
2940 new_via->
SetDrill( padstack->second.drill_size_x );
2943 const std::set<std::string>& viaLayers = padstack->second.copper_layers;
2945 if( viaLayers.size() >= 2 )
2953 if( viaLayers.count( layer->name ) )
2962 if( topLayer && botLayer && topLayer != botLayer )
2968 bool isThrough = ( topLayerId ==
F_Cu && botLayerId ==
B_Cu );
2974 if( topLayerId ==
F_Cu || botLayerId ==
B_Cu )
3009 auto net_it = netinfo.find( aLine->netname );
3011 for(
const auto& seg : aLine->segment )
3017 switch( seg->shape )
3030 if( net_it != netinfo.end() )
3031 trk->
SetNet( net_it->second );
3045 if( net_it != netinfo.end() )
3046 trk->
SetNet( net_it->second );
3054 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3062 wxLogError(
_(
"Expecting etch data to be on copper layer. Row found on layer '%s'" ),
3063 seg->layer.c_str() );
3074 int last_subseq = 0;
3079 for(
const auto& seg : aElement )
3081 if( seg->subseq > 0 || seg->subseq != last_subseq )
3088 if( poly_outline.
VertexCount( 0, hole_idx ) == 0 )
3102 return poly_outline;
3129 if( aLine.
segment.size() == 0 )
3135 int first_subseq = -1;
3136 bool have_multiple_subseqs =
false;
3138 for(
const std::unique_ptr<GRAPHIC_ITEM>& gr_item : aLine.
segment )
3140 if( first ==
nullptr )
3142 first = gr_item.get();
3143 first_subseq = gr_item->subseq;
3145 else if( gr_item->subseq == first_subseq )
3147 last = gr_item.get();
3151 have_multiple_subseqs =
true;
3157 wxCHECK( first,
true );
3175 switch( last->
shape )
3197 if(
end.has_value() && start ==
end )
3205std::vector<std::unique_ptr<BOARD_ITEM>>
3208 std::vector<std::unique_ptr<BOARD_ITEM>> new_items;
3213 const auto setShapeParameters = [&](
PCB_SHAPE& aShape )
3217 if( aShape.GetWidth() == 0 )
3218 aShape.SetStroke( defaultStroke );
3221 switch( aGraphic.
shape )
3227 auto new_text = std::make_unique<PCB_TEXT>( &aBoard );
3231 new_text->SetMirrored(
true );
3234 setupText( src, aLayer, *new_text, aBoard, std::nullopt );
3236 new_items.emplace_back( std::move( new_text ) );
3249 for(
const SEG& seg : segs )
3251 auto line = std::make_unique<PCB_SHAPE>( &aBoard );
3253 line->SetStart( seg.A );
3254 line->SetEnd( seg.B );
3256 setShapeParameters( *line );
3257 new_items.emplace_back( std::move( line ) );
3265 auto new_shape = std::make_unique<PCB_SHAPE>( &aBoard );
3267 setShapeParameters( *new_shape );
3269 switch( aGraphic.
shape )
3298 new_shape->SetRadius( src.
radius );
3310 new_shape->SetFilled( src.
fill );
3318 new_shape->SetPolyPoints( src.
m_pts );
3348 new_shape->SetPolyShape( poly );
3353 wxLogError(
_(
"Unhandled shape type %d in polygon on layer %s, seq %d %d" ),
3357 new_items.emplace_back( std::move( new_shape ) );
3361 for( std::unique_ptr<BOARD_ITEM>& new_item : new_items )
3363 new_item->SetLayer( aLayer );
3367 if( new_items.size() > 1 )
3369 auto new_group = std::make_unique<PCB_GROUP>( &aBoard );
3370 for( std::unique_ptr<BOARD_ITEM>& new_item : new_items )
3372 new_group->AddItem( new_item.get() );
3374 new_items.emplace_back( std::move( new_group ) );
3383 if( aLine->segment.empty() )
3397 for(
const auto& seg : aLine->segment )
3399 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3446 if( aLine->segment.size() < 3 )
3450 ZONE* zone =
nullptr;
3453 auto net_it = netinfo.find( aLine->netname );
3455 auto new_layer =
getLayer( aLine->layer );
3460 zone =
new ZONE( aBoard );
3463 if( net_it != netinfo.end() )
3464 zone->
SetNet( net_it->second );
3466 if( aLine->layer ==
"ALL" )
3478 if( aLine->lclass ==
"ROUTE KEEPOUT")
3483 else if( aLine->lclass ==
"VIA KEEPOUT")
3498 std::unique_ptr<SHAPE_LINE_CHAIN> pending_hole =
nullptr;
3501 const auto add_hole_if_valid = [&]()
3505 pending_hole->SetClosed(
true );
3511 wxLogMessage(
_(
"Invalid hole with %d points in zone on layer %s with net %s" ),
3516 pending_hole.reset();
3520 int last_subseq = 0;
3521 for(
const auto& seg : aLine->segment )
3523 if( seg->subseq > 0 && seg->subseq != last_subseq )
3527 if( aLine->lclass ==
"BOUNDARY" )
3530 add_hole_if_valid();
3531 pending_hole = std::make_unique<SHAPE_LINE_CHAIN>();
3532 active_chain = pending_hole.get();
3533 last_subseq = seg->subseq;
3544 active_chain->
Append( start );
3553 wxLogError(
_(
"Outline seems discontinuous: last point was %s, "
3554 "start point of next segment is %s" ),
3569 wxLogError(
_(
"Invalid shape type %d in zone outline" ), seg->shape );
3574 add_hole_if_valid();
3583 delete( zone_outline );
3595 if( aLine->lclass ==
"BOARD GEOMETRY" && aLine->layer !=
"DIMENSION" )
3597 else if( aLine->lclass ==
"DRAWING FORMAT" )
3602 for(
auto& seg : aLine->segment )
3604 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3624 if( geom.subclass ==
"PIN_NUMBER" )
3632 if( !geom.elements->empty() )
3635 if( ( *( geom.elements->begin() ) )->width == 0 )
3656 for(
auto& seg : *geom.elements )
3658 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3672 std::vector<ZONE*> sortedZones;
3673 std::copy( aBoard->
Zones().begin(), aBoard->
Zones().end(), std::back_inserter( sortedZones ) );
3674 std::sort( sortedZones.begin(), sortedZones.end(),
3675 [&](
const ZONE* a,
const ZONE* b )
3677 if( a->GetLayer() == b->GetLayer() )
3678 return a->GetBoundingBox().GetArea() > b->GetBoundingBox().GetArea();
3680 return a->GetLayer() < b->GetLayer();
3684 unsigned int priority = 0;
3686 for(
ZONE* zone : sortedZones )
3689 if( zone->GetIsRuleArea() )
3692 if( zone->GetLayer() != layer )
3694 layer = zone->GetLayer();
3698 zone->SetAssignedPriority( priority );
3728 for(
auto& track :
traces )
3732 if( track->lclass ==
"ETCH" )
3734 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)
@ 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.
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
@ FULL
pads are covered by copper