56#include <wx/filename.h>
69 const unsigned PROGRESS_DELTA = 250;
90 wxCHECK_MSG( !aStr.empty(), 0.0,
"Empty string passed to readDouble" );
92 std::istringstream istr( aStr );
93 istr.imbue( std::locale::classic() );
104 wxCHECK_MSG( !aStr.empty(), 0,
"Empty string passed to readInt" );
106 std::istringstream istr( aStr );
107 istr.imbue( std::locale::classic() );
117 std::ifstream ifs( aFile, std::ios::in | std::ios::binary );
125 ifs.ignore( std::numeric_limits<std::streamsize>::max() );
126 std::streamsize length = ifs.gcount();
128 ifs.seekg( 0, std::ios_base::beg );
130 std::string buffer( std::istreambuf_iterator<char>{ ifs }, {} );
132 std::vector < std::string > row;
136 row.reserve( length / 100 );
142 for(
auto& ch : buffer )
148 if( cell.empty() || cell[0] ==
'"' )
157 row.push_back( cell );
169 row.push_back( cell );
172 rows.push_back( row );
181 cell += std::toupper( ch );
186 if( !cell.empty() || !row.empty() )
188 row.push_back( cell );
190 rows.push_back( row );
204 row =
rows.at( aOffset );
206 catch( std::out_of_range& )
214 if( row[0].back() !=
'A' )
217 std::string row1 = row[1];
218 std::string row2 = row[2];
223 std::erase_if( row1, [](
char c ){
return c ==
'_'; } );
224 std::erase_if( row2, [](
char c ){
return c ==
'_'; } );
229 std::erase_if( row3, [](
char c ){
return c ==
'_'; } );
232 if( row1 ==
"REFDES" && row2 ==
"COMPCLASS" )
235 if( row1 ==
"NETNAME" && row2 ==
"REFDES" )
238 if( row1 ==
"CLASS" && row2 ==
"SUBCLASS" && row3.empty() )
241 if( row1 ==
"GRAPHICDATANAME" && row2 ==
"GRAPHICDATANUMBER" )
244 if( row1 ==
"CLASS" && row2 ==
"SUBCLASS" && row3 ==
"GRAPHICDATANAME" )
247 if( row1 ==
"SYMNAME" && row2 ==
"PINNAME" )
250 if( row1 ==
"SYMNAME" && row2 ==
"SYMMIRROR" && row3 ==
"PINNAME" )
253 if( row1 ==
"VIAX" && row2 ==
"VIAY" )
256 if( row1 ==
"SUBCLASS" && row2 ==
"PADSHAPENAME" )
259 if( row1 ==
"PADNAME" )
262 if( row1 ==
"LAYERSORT" )
265 wxLogError(
_(
"Unknown FABMASTER section %s:%s at row %zu." ),
278 if( aRow >=
rows.size() )
281 if(
rows[aRow].size() < 11 )
283 wxLogError(
_(
"Invalid row size in J row %zu. Expecting 11 elements but found %zu." ),
289 for(
int i = 7; i < 10 && retval < 1.0; ++i )
291 std::string units =
rows[aRow][i];
292 std::transform(units.begin(), units.end(),units.begin(), ::toupper);
294 if( units ==
"MILS" )
296 else if( units ==
"MILLIMETERS" )
298 else if( units ==
"MICRONS" )
300 else if( units ==
"INCHES" )
306 wxLogError(
_(
"Could not find units value, defaulting to mils." ) );
316 if( aRow >=
rows.size() )
321 for(
size_t i = 0; i <
header.size(); i++ )
325 std::erase_if(
header[i], [](
const char c ) {
return c ==
'_'; } );
331 THROW_IO_ERROR( wxString::Format(
_(
"Could not find column label %s." ), aStr.c_str() ) );
338 const auto& kicad_layer =
layers.find( aLayerName);
340 if( kicad_layer ==
layers.end() )
343 return static_cast<PCB_LAYER_ID>( kicad_layer->second.layerid );
349 size_t rownum = aRow + 2;
351 if( rownum >=
rows.size() )
359 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
363 if( row.size() !=
header.size() )
365 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
372 auto& pad_num = row[pad_num_col];
373 auto& pad_layer = row[pad_lay_col];
376 if( pad_layer ==
"INTERNAL_PAD_DEF" || pad_layer ==
"internal_pad_def" )
380 if( pad_layer[0] ==
'~' )
389 layer.
name = pad_layer;
407 size_t rownum = aRow + 2;
409 if( rownum >=
rows.size() )
415 if( scale_factor <= 0.0 )
429 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
434 if( row.size() !=
header.size() )
436 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
443 auto& pad_name = row[pad_name_col];
444 auto& pad_num = row[pad_num_col];
445 auto& pad_layer = row[pad_lay_col];
446 auto& pad_is_via = row[pad_via_col];
447 auto& pad_shape = row[pad_shape_col];
448 auto& pad_width = row[pad_width_col];
449 auto& pad_height = row[pad_height_col];
450 auto& pad_xoff = row[pad_xoff_col];
451 auto& pad_yoff = row[pad_yoff_col];
452 auto& pad_shapename = row[pad_shape_name_col];
455 if( pad_layer ==
"INTERNAL_PAD_DEF" || pad_layer ==
"internal_pad_def" )
460 auto new_pad =
pads.find( pad_name );
462 if( new_pad !=
pads.end() )
463 pad = &new_pad->second;
468 pad->name = pad_name;
472 if( pad_layer ==
"~DRILL" )
486 wxLogError(
_(
"Expecting drill size value but found %s!%s!%s in row %zu." ),
504 if( drill_x == drill_y )
506 pad->drill_size_x = drill_hit;
507 pad->drill_size_y = drill_hit;
511 pad->drill_size_x = drill_x;
512 pad->drill_size_y = drill_y;
515 if( !pad_shapename.empty() && pad_shapename[0] ==
'P' )
521 if( pad_shape.empty() )
534 wxLogError(
_(
"Expecting pad size values but found %s : %s in row %zu." ),
541 auto layer =
layers.find( pad_layer );
543 if( w > 0.0 && layer !=
layers.end() && layer->second.conductive )
544 pad->copper_layers.insert( pad_layer );
549 if( layer !=
layers.end() )
551 if( layer->second.layerid ==
F_Cu )
553 else if( layer->second.layerid ==
B_Cu )
557 if( w > std::numeric_limits<int>::max() || h > std::numeric_limits<int>::max() )
559 wxLogError(
_(
"Invalid pad size in row %zu." ), rownum );
563 if( pad_layer ==
"~TSM" || pad_layer ==
"~BSM" )
565 if( w > 0.0 && h > 0.0 )
573 if( pad_layer ==
"~TSP" || pad_layer ==
"~BSP" )
575 if( w > 0.0 && h > 0.0 )
584 if( pad_layer[0] ==
'~' )
587 int layer_x_offset = 0;
588 int layer_y_offset = 0;
597 wxLogError(
_(
"Expecting pad offset values but found %s:%s in row %zu." ),
606 pad->x_offset = layer_x_offset;
607 pad->y_offset = layer_y_offset;
610 if( w > 0.0 && h > 0.0 )
615 layer_data.
x_offset = layer_x_offset;
616 layer_data.
y_offset = layer_y_offset;
618 if( pad_shape ==
"CIRCLE" )
623 else if( pad_shape ==
"RECTANGLE" )
627 else if( pad_shape ==
"ROUNDED_RECT" )
631 else if( pad_shape ==
"SQUARE" )
636 else if( pad_shape ==
"OBLONG" || pad_shape ==
"OBLONG_X"
637 || pad_shape ==
"OBLONG_Y" )
641 else if( pad_shape ==
"OCTAGON" )
646 else if( pad_shape ==
"SHAPE" )
653 wxLogError(
_(
"Unknown pad shape name '%s' on layer '%s' in row %zu." ),
660 pad->layer_shapes[pad_layer] = layer_data;
668 pad->via = pad_is_via.empty()
669 || std::toupper(
static_cast<unsigned char>( pad_is_via[0] ) ) !=
'V';
672 pad->custom_name = pad_shapename;
677 return rownum - aRow;
683 size_t rownum = aRow + 2;
685 if( rownum >=
rows.size() )
691 if( scale_factor <= 0.0 )
697 if( layer_class_col < 0 || layer_subclass_col < 0 )
700 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
704 if( row.size() !=
header.size() )
706 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
716 layer.
name = row[layer_subclass_col];
720 if( row[layer_class_col] ==
"ANTI ETCH" )
725 else if( row[layer_class_col] ==
"ETCH" )
731 return rownum - aRow;
737 std::vector<std::pair<std::string, int>> extra_layers
739 {
"ASSEMBLY_TOP",
F_Fab },
740 {
"ASSEMBLY_BOTTOM",
B_Fab },
741 {
"PLACE_BOUND_TOP",
F_CrtYd },
742 {
"PLACE_BOUND_BOTTOM",
B_CrtYd },
745 std::vector<FABMASTER_LAYER*> layer_order;
747 int next_user_layer =
User_1;
756 layer_order.push_back( &layer );
758 else if( ( layer.
name.find(
"SILK" ) != std::string::npos
759 && layer.
name.find(
"AUTOSILK" )
760 == std::string::npos )
761 || layer.
name.find(
"DISPLAY" ) != std::string::npos )
763 if( layer.
name.find(
"B" ) != std::string::npos )
768 else if( layer.
name.find(
"MASK" ) != std::string::npos ||
769 layer.
name.find(
"MSK" ) != std::string::npos )
771 if( layer.
name.find(
"B" ) != std::string::npos )
776 else if( layer.
name.find(
"PAST" ) != std::string::npos )
778 if( layer.
name.find(
"B" ) != std::string::npos )
783 else if( layer.
name.find(
"NCLEGEND" ) != std::string::npos )
792 if( layer.
name.find(
"AUTOSILK" ) == std::string::npos )
794 if( next_user_layer <=
User_9 )
797 layer.
layerid = next_user_layer;
798 next_user_layer += 2;
806 wxLogWarning(
_(
"No user layer to put layer %s" ), layer.
name );
814 for(
size_t layeri = 0; layeri < layer_order.size(); ++layeri )
819 else if( layeri == layer_order.size() - 1 )
822 layer->
layerid = layeri * 2 + 2;
825 for(
auto& new_pair : extra_layers )
829 new_layer.
name = new_pair.first;
830 new_layer.
layerid = new_pair.second;
833 auto result =
layers.emplace( new_pair.first, new_layer );
837 result.first->second.layerid = new_pair.second;
838 result.first->second.disable =
false;
842 for(
const auto& [layer_name, fabmaster_layer] :
layers )
844 wxLogTrace(
traceFabmaster, wxT(
"Layer %s -> KiCad layer %d" ), layer_name,
845 fabmaster_layer.layerid );
859 size_t rownum = aRow + 2;
861 if( rownum >=
rows.size() )
867 if( scale_factor <= 0.0 )
875 int layer_er_col =
getColFromName( aRow,
"LAYERDIELECTRICCONSTANT" );
876 int layer_rho_col =
getColFromName( aRow,
"LAYERELECTRICALCONDUCTIVITY" );
879 if( layer_sort_col < 0 || layer_subclass_col < 0 || layer_art_col < 0 || layer_use_col < 0
880 || layer_cond_col < 0 || layer_er_col < 0 || layer_rho_col < 0 || layer_mat_col < 0 )
883 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
887 if( row.size() !=
header.size() )
889 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
896 auto& layer_sort = row[layer_sort_col];
897 auto& layer_subclass = row[layer_subclass_col];
898 auto& layer_art = row[layer_art_col];
899 auto& layer_cond = row[layer_cond_col];
900 auto& layer_mat = row[layer_mat_col];
902 if( layer_mat ==
"AIR" )
907 if( layer_subclass.empty() )
909 if( layer_cond !=
"NO" )
910 layer.
name =
"In.Cu" + layer_sort;
912 layer.
name =
"Dielectric" + layer_sort;
915 layer.
positive = ( layer_art !=
"NEGATIVE" );
920 return rownum - aRow;
931 size_t rownum = aRow + 2;
933 if( rownum >=
rows.size() )
939 if( scale_factor <= 0.0 )
944 int pad_grdata_name_col =
getColFromName( aRow,
"GRAPHICDATANAME" );
945 int pad_grdata_num_col =
getColFromName( aRow,
"GRAPHICDATANUMBER" );
960 if( pad_subclass_col < 0 || pad_shape_name_col < 0 || pad_grdata1_col < 0 || pad_grdata2_col < 0
961 || pad_grdata3_col < 0 || pad_grdata4_col < 0 || pad_grdata5_col < 0
962 || pad_grdata6_col < 0 || pad_grdata7_col < 0 || pad_grdata8_col < 0
963 || pad_grdata9_col < 0 || pad_stack_name_col < 0 || pad_refdes_col < 0
964 || pad_pin_num_col < 0 )
967 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
971 if( row.size() !=
header.size() )
973 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
981 auto& pad_layer = row[pad_subclass_col];
982 auto pad_shape_name = row[pad_shape_name_col];
983 auto& pad_record_tag = row[pad_record_tag_col];
998 auto& pad_stack_name = row[pad_stack_name_col];
999 auto& pad_refdes = row[pad_refdes_col];
1000 auto& pad_pin_num = row[pad_pin_num_col];
1004 std::string prefix(
"FIG_SHAPE " );
1006 if( pad_shape_name.length() <= prefix.length()
1007 || !std::equal( prefix.begin(), prefix.end(), pad_shape_name.begin() ) )
1017 if( std::sscanf( pad_record_tag.c_str(),
"%d %d", &
id, &seq ) != 2 )
1019 wxLogError(
_(
"Invalid format for id string '%s' in custom pad row %zu." ),
1020 pad_record_tag.c_str(),
1025 auto name = pad_shape_name.substr( prefix.length() );
1026 name +=
"_" + pad_refdes +
"_" + pad_pin_num;
1029 auto& custom_pad = ret.first->second;
1035 custom_pad.name =
name;
1036 custom_pad.padstack = pad_stack_name;
1037 custom_pad.pinnum = pad_pin_num;
1038 custom_pad.refdes = pad_refdes;
1045 auto gr_item = std::unique_ptr<GRAPHIC_ITEM>(
processGraphic( gr_data, scale_factor ) );
1049 gr_item->layer = pad_layer;
1050 gr_item->refdes = pad_refdes;
1052 gr_item->subseq = 0;
1057 auto retval = pad_it.first->second.insert( std::move(gr_item ) );
1059 if( !retval.second )
1061 wxLogError(
_(
"Could not insert graphical item %d into padstack '%s'." ),
1063 pad_stack_name.c_str() );
1068 wxLogError(
_(
"Unrecognized pad shape primitive '%s' in row %zu." ),
1074 return rownum - aRow;
1124 angle = endangle - startangle;
1159 std::unique_ptr<GRAPHIC_ARC> new_circle = std::make_unique<GRAPHIC_ARC>();
1170 if( size.
x != size.
y )
1172 wxLogError(
_(
"Circle with unequal x and y radii (x=%d, y=%d)" ), size.
x, size.
y );
1178 new_circle->radius = size.
x / 2;
1184 new_circle->start_x = start.
x;
1185 new_circle->start_y = start.
y;
1187 new_circle->end_x = start.
x;
1188 new_circle->end_y = start.
y;
1190 new_circle->center_x =
center.x;
1191 new_circle->center_y =
center.y;
1193 new_circle->clockwise =
true;
1195 new_circle->result =
SHAPE_ARC{ start, mid, start, 0 };
1197 return new_circle.release();
1217 new_rect->
width = 0;
1231 auto new_rect = std::make_unique<GRAPHIC_RECTANGLE>();
1240 new_rect->start_x = center_x - size_x / 2;
1241 new_rect->start_y = center_y + size_y / 2;
1242 new_rect->end_x = center_x + size_x / 2;
1243 new_rect->end_y = center_y - size_y / 2;
1245 new_rect->width = 0;
1247 return new_rect.release();
1272 auto new_oblong = std::make_unique<GRAPHIC_OBLONG>();
1284 return new_oblong.release();
1314 auto new_poly = std::make_unique<GRAPHIC_POLYGON>();
1319 bool across_corners =
true;
1353 across_corners =
false;
1359 wxCHECK_MSG(
false,
nullptr,
1360 wxString::Format(
"Unhandled polygon type: %s", aData.
graphic_dataname ) );
1365 return new_poly.release();
1376 auto new_cross = std::make_unique<GRAPHIC_CROSS>();
1385 return new_cross.release();
1409 if( toks.size() < 8 )
1412 wxLogError(
_(
"Invalid token count. Expected 8 but found %zu." ), toks.size() );
1414 new_text->
width = 0;
1415 new_text->
ital =
false;
1489 size_t rownum = aRow + 2;
1491 if( rownum >=
rows.size() )
1497 if( scale_factor <= 0.0 )
1516 if( geo_name_col < 0 || geo_num_col < 0 || geo_grdata1_col < 0 || geo_grdata2_col < 0
1517 || geo_grdata3_col < 0 || geo_grdata4_col < 0 || geo_grdata5_col < 0
1518 || geo_grdata6_col < 0 || geo_grdata7_col < 0 || geo_grdata8_col < 0
1519 || geo_grdata9_col < 0 || geo_subclass_col < 0 || geo_sym_name_col < 0
1520 || geo_refdes_col < 0 )
1523 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1527 if( row.size() !=
header.size() )
1529 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1536 auto& geo_tag = row[geo_tag_col];
1551 auto& geo_refdes = row[geo_refdes_col];
1559 if( std::sscanf( geo_tag.c_str(),
"%d %d %d", &
id, &seq, &subseq ) < 2 )
1561 wxLogError(
_(
"Invalid format for record_tag string '%s' in row %zu." ),
1567 auto gr_item = std::unique_ptr<GRAPHIC_ITEM>(
processGraphic( gr_data, scale_factor ) );
1572 gr_item->layer = row[geo_subclass_col];
1574 gr_item->subseq = subseq;
1576 if( geo_refdes.empty() )
1581 new_gr.
subclass = row[geo_subclass_col];
1582 new_gr.
refdes = row[geo_refdes_col];
1583 new_gr.
name = row[geo_sym_name_col];
1585 new_gr.
elements = std::make_unique<graphic_element>();
1590 graphic.
elements->emplace( std::move( gr_item ) );
1595 std::map<int, GEOM_GRAPHIC>{} );
1596 auto map_it = sym_gr_it.first->second.emplace(
id,
GEOM_GRAPHIC{} );
1597 auto& gr = map_it.first;
1601 gr->second.subclass = row[geo_subclass_col];
1602 gr->second.refdes = row[geo_refdes_col];
1603 gr->second.name = row[geo_sym_name_col];
1605 gr->second.elements = std::make_unique<graphic_element>();
1608 gr->second.elements->emplace( std::move( gr_item ) );
1612 return rownum - aRow;
1621 size_t rownum = aRow + 2;
1623 if( rownum >=
rows.size() )
1629 if( scale_factor <= 0.0 )
1638 if( viax_col < 0 || viay_col < 0 || padstack_name_col < 0 || net_name_col < 0
1639 || test_point_col < 0 )
1642 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1646 if( row.size() !=
header.size() )
1648 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1655 vias.emplace_back( std::make_unique<FM_VIA>() );
1660 via->padstack = row[padstack_name_col];
1661 via->net = row[net_name_col];
1662 via->test_point = ( row[test_point_col] ==
"YES" );
1665 return rownum - aRow;
1676 size_t rownum = aRow + 2;
1678 if( rownum >=
rows.size() )
1684 if( scale_factor <= 0.0 )
1690 int grdata_num_col =
getColFromName( aRow,
"GRAPHICDATANUMBER" );
1703 if( class_col < 0 || layer_col < 0 || grdata_name_col < 0 || grdata_num_col < 0
1704 || tag_col < 0 || grdata1_col < 0 || grdata2_col < 0 || grdata3_col < 0
1705 || grdata4_col < 0 || grdata5_col < 0 || grdata6_col < 0 || grdata7_col < 0
1706 || grdata8_col < 0 || grdata9_col < 0 || netname_col < 0 )
1709 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1713 if( row.size() !=
header.size() )
1715 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1735 const std::string& geo_tag = row[tag_col];
1742 if( std::sscanf( geo_tag.c_str(),
"%d %d %d", &
id, &seq, &subseq ) < 2 )
1744 wxLogError(
_(
"Invalid format for record_tag string '%s' in row %zu." ),
1750 auto gr_item = std::unique_ptr<GRAPHIC_ITEM>(
processGraphic( gr_data, scale_factor ) );
1754 wxLogTrace(
traceFabmaster,
_(
"Unhandled graphic item '%s' in row %zu." ),
1760 auto new_trace = std::make_unique<TRACE>();
1762 new_trace->layer = row[layer_col];
1763 new_trace->netname = row[netname_col];
1764 new_trace->lclass = row[class_col];
1766 gr_item->layer = row[layer_col];
1768 gr_item->subseq = subseq;
1771 if( new_trace->lclass ==
"REF DES" )
1773 auto result =
refdes.emplace( std::move( new_trace ) );
1774 auto& ref = *
result.first;
1775 ref->segment.emplace( std::move( gr_item ) );
1777 else if( new_trace->lclass ==
"DEVICE TYPE" || new_trace->lclass ==
"COMPONENT VALUE"
1778 || new_trace->lclass ==
"TOLERANCE" )
1791 else if( gr_item->width == 0 )
1793 auto result =
zones.emplace( std::move( new_trace ) );
1794 auto& zone = *
result.first;
1795 auto gr_result = zone->segment.emplace( std::move( gr_item ) );
1797 if( !gr_result.second )
1799 wxLogError(
_(
"Duplicate item for ID %d and sequence %d in row %zu." ),
1807 auto result =
traces.emplace( std::move( new_trace ) );
1808 auto& trace = *
result.first;
1809 auto gr_result = trace->segment.emplace( std::move( gr_item ) );
1811 if( !gr_result.second )
1813 wxLogError(
_(
"Duplicate item for ID %d and sequence %d in row %zu." ),
1821 return rownum - aRow;
1827 if( aSymType ==
"PACKAGE" )
1829 else if( aSymType ==
"DRAFTING")
1831 else if( aSymType ==
"MECHANICAL" )
1833 else if( aSymType ==
"FORMAT" )
1842 if( aCmpClass ==
"IO" )
1844 else if( aCmpClass ==
"IC" )
1846 else if( aCmpClass ==
"DISCRETE" )
1859 size_t rownum = aRow + 2;
1861 if( rownum >=
rows.size() )
1867 if( scale_factor <= 0.0 )
1875 int compinscode_col =
getColFromName( aRow,
"COMPINSERTIONCODE" );
1886 if( refdes_col < 0 || compclass_col < 0 || comppartnum_col < 0 || compheight_col < 0
1887 || compdevlabelcol < 0 || compinscode_col < 0 || symtype_col < 0 || symname_col < 0
1888 || symmirror_col < 0 || symrotate_col < 0 || symx_col < 0 || symy_col < 0
1889 || compvalue_col < 0 || comptol_col < 0 || compvolt_col < 0 )
1892 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1896 if( row.size() !=
header.size() )
1898 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1905 const wxString& comp_refdes = row[refdes_col];
1907 if( row[symx_col].
empty() || row[symy_col].
empty() || row[symrotate_col].
empty() )
1909 wxLogError(
_(
"Missing X, Y, or rotation data in row %zu for refdes %s. "
1910 "This may be an unplaced component." ),
1911 rownum, comp_refdes );
1915 auto cmp = std::make_unique<COMPONENT>();
1917 cmp->refdes = comp_refdes;
1919 cmp->pn = row[comppartnum_col];
1920 cmp->height = row[compheight_col];
1921 cmp->dev_label = row[compdevlabelcol];
1922 cmp->insert_code = row[compinscode_col];
1924 cmp->name = row[symname_col];
1925 cmp->mirror = ( row[symmirror_col] ==
"YES" );
1926 cmp->rotate =
readDouble( row[symrotate_col] );
1929 cmp->value = row[compvalue_col];
1930 cmp->tol = row[comptol_col];
1931 cmp->voltage = row[compvolt_col];
1937 auto retval =
components.insert( std::make_pair( cmp->refdes, std::vector<std::unique_ptr<COMPONENT>>{} ) );
1942 vec->second.push_back( std::move( cmp ) );
1945 return rownum - aRow;
1955 size_t rownum = aRow + 2;
1957 if( rownum >=
rows.size() )
1963 if( scale_factor <= 0.0 )
1977 if( symname_col < 0 ||symmirror_col < 0 || pinname_col < 0 || pinnum_col < 0 || pinx_col < 0
1978 || piny_col < 0 || padstack_col < 0 || refdes_col < 0 || pinrot_col < 0
1979 || testpoint_col < 0 )
1982 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
1986 if( row.size() !=
header.size() )
1988 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
1995 auto pin = std::make_unique<PIN>();
1997 pin->name = row[symname_col];
1998 pin->mirror = ( row[symmirror_col] ==
"YES" );
1999 pin->pin_name = row[pinname_col];
2000 pin->pin_number = row[pinnum_col];
2003 pin->padstack = row[padstack_col];
2004 pin->refdes = row[refdes_col];
2010 std::string pin_key =
pin->refdes.empty() ?
pin->name :
pin->refdes;
2012 auto map_it =
pins.find( pin_key );
2014 if( map_it ==
pins.end() )
2016 auto retval =
pins.insert( std::make_pair( pin_key, std::set<std::unique_ptr<PIN>,
2018 map_it = retval.first;
2021 map_it->second.insert( std::move(
pin ) );
2024 return rownum - aRow;
2033 size_t rownum = aRow + 2;
2035 if( rownum >=
rows.size() )
2041 if( scale_factor <= 0.0 )
2051 if( netname_col < 0 || refdes_col < 0 || pinnum_col < 0 || pinname_col < 0 || pingnd_col < 0
2055 for( ; rownum <
rows.size() &&
rows[rownum].size() > 0 &&
rows[rownum][0] ==
"S"; ++rownum )
2059 if( row.size() !=
header.size() )
2061 wxLogError(
_(
"Invalid row size in row %zu. Expecting %zu elements but found %zu." ),
2069 new_net.
name = row[netname_col];
2070 new_net.
refdes = row[refdes_col];
2071 new_net.
pin_num = row[pinnum_col];
2072 new_net.
pin_name = row[pinname_col];
2073 new_net.
pin_gnd = ( row[pingnd_col] ==
"YES" );
2074 new_net.
pin_pwr = ( row[pinpwr_col] ==
"YES" );
2077 netnames.insert( row[netname_col] );
2080 return rownum - aRow;
2087 for(
size_t i = 0; i <
rows.size(); )
2101 i += std::max( retval, 1 );
2109 i += std::max( retval, 1 );
2117 i += std::max( retval, 1 );
2125 i += std::max( retval, 1 );
2133 i += std::max( retval, 1 );
2141 i += std::max( retval, 1 );
2149 i += std::max( retval, 1 );
2157 i += std::max( retval, 1 );
2165 i += std::max( retval, 1 );
2173 i += std::max( retval, 1 );
2190 for(
auto& zone :
zones )
2200 if( zone->layer ==
"OUTLINE" || zone->layer ==
"DESIGN_OUTLINE" )
2221 std::set<ZONE*> zones_to_delete;
2222 std::set<ZONE*> matched_fills;
2224 for(
auto zone : aBoard->
Zones() )
2226 if( zone->GetNetCode() > 0 )
2227 zones_to_delete.insert( zone );
2230 for(
auto zone1 : aBoard->
Zones() )
2232 if( zone1->GetNetCode() > 0 )
2237 std::map<int, std::vector<ZONE*>> net_to_fills;
2239 for(
auto zone2 : aBoard->
Zones() )
2241 if( zone2->GetNetCode() <= 0 )
2246 if( zone1->GetLayer() != zone2->GetLayer() )
2252 size_t match_count = 0;
2254 for(
auto& pt1 : outline1.
CPoints() )
2260 for(
auto& pt2 : outline2.
CPoints() )
2266 if( match_count > 0 )
2268 overlaps[zone2->GetNetCode()] += match_count;
2269 net_to_fills[zone2->GetNetCode()].push_back( zone2 );
2274 size_t max_net_id = 0;
2276 for(
size_t el = 1; el < overlaps.size(); ++el )
2278 if( overlaps[el] > max_net )
2280 max_net = overlaps[el];
2287 zone1->SetNetCode( max_net_id );
2289 for(
ZONE* fill : net_to_fills[max_net_id] )
2290 matched_fills.insert( fill );
2294 for(
auto zone : zones_to_delete )
2296 if( matched_fills.find( zone ) != matched_fills.end() )
2317 if( aMirrorPoint.has_value() )
2350 for(
const auto& [pinKey, pinSet] :
pins )
2352 if( pinSet.empty() )
2358 const auto& firstPin = *pinSet.begin();
2360 int minX = firstPin->pin_x;
2361 int maxX = firstPin->pin_x;
2362 int minY = firstPin->pin_y;
2363 int maxY = firstPin->pin_y;
2365 for(
const auto&
pin : pinSet )
2367 minX = std::min( minX,
pin->pin_x );
2368 maxX = std::max( maxX,
pin->pin_x );
2369 minY = std::min( minY,
pin->pin_y );
2370 maxY = std::max( maxY,
pin->pin_y );
2373 auto cmp = std::make_unique<COMPONENT>();
2374 cmp->refdes = pinKey;
2375 cmp->name = firstPin->name;
2376 cmp->mirror = firstPin->mirror;
2378 cmp->x = ( minX + maxX ) / 2;
2379 cmp->y = ( minY + maxY ) / 2;
2383 std::vector<std::unique_ptr<COMPONENT>> compVec;
2384 compVec.push_back( std::move( cmp ) );
2385 components.insert( std::make_pair( pinKey, std::move( compVec ) ) );
2399 bool has_multiple = mod.second.size() > 1;
2401 for(
int i = 0; i < (int) mod.second.size(); ++i )
2403 auto& src = mod.second[i];
2407 wxString mod_ref = src->name;
2411 mod_ref.Append( wxString::Format( wxT(
"_%d" ), i ) );
2416 wxString key = !lib_ref.empty() ? lib_ref + wxT(
":" ) + mod_ref : mod_ref;
2419 fpID.
Parse( key,
true );
2428 wxString reference = src->refdes;
2430 if( !std::isalpha( src->refdes[0] ) )
2431 reference.Prepend(
"UNK" );
2444 for(
auto& ref :
refdes )
2447 static_cast<const GRAPHIC_TEXT&
>( **ref->segment.begin() );
2449 if( lsrc.
text == src->refdes )
2456 wxLogTrace(
traceFabmaster, wxS(
"The layer %s is not mapped?" ),
2457 ref->layer.c_str() );
2468 flip_point =
VECTOR2I( src->x, src->y );
2473 setupText( lsrc, layer, *txt, *aBoard, flip_point );
2488 for(
auto& gr_ref : gr_it->second )
2490 auto& graphic = gr_ref.second;
2492 for(
auto& seg : *graphic.elements )
2499 STROKE_PARAMS defaultStroke( ds.GetLineThickness( layer ) );
2501 switch( seg->shape )
2524 if( lsrc->
width == 0 )
2537 circle->SetLayer( layer );
2549 if( lsrc.
width == 0 )
2555 if( lsrc.
layer ==
"DISPLAY_TOP" || lsrc.
layer ==
"DISPLAY_BOTTOM" )
2556 circle->SetFilled(
true );
2558 circle->SetWidth( ds.GetLineThickness(
circle->GetLayer() ) );
2572 std::unique_ptr<PCB_SHAPE> arc =
2586 arc->SetLayer( layer );
2590 if( lsrc->
width == 0 )
2591 arc->SetStroke( defaultStroke );
2630 std::unique_ptr<PCB_TEXT> txt = std::make_unique<PCB_TEXT>( fp );
2635 flip_point =
VECTOR2I( src->x, src->y );
2637 setupText( lsrc, layer, *txt, *aBoard, flip_point );
2662 auto pin_it =
pins.find( src->refdes );
2665 if( pin_it ==
pins.end() )
2666 pin_it =
pins.find( src->name );
2668 if( pin_it !=
pins.end() )
2670 for(
auto&
pin : pin_it->second )
2672 auto pin_net_it =
pin_nets.find( std::make_pair(
pin->refdes,
2673 pin->pin_number ) );
2674 auto padstack =
pads.find(
pin->padstack );
2675 std::string netname =
"";
2678 netname = pin_net_it->second.name;
2680 auto net_it = netinfo.find( netname );
2682 std::unique_ptr<PAD> newpad = std::make_unique<PAD>( fp );
2684 if( net_it != netinfo.end() )
2685 newpad->SetNet( net_it->second );
2687 newpad->SetNetCode( 0 );
2689 newpad->SetX(
pin->pin_x );
2692 newpad->SetY( 2 * src->y -
pin->pin_y );
2694 newpad->SetY(
pin->pin_y );
2696 newpad->SetNumber(
pin->pin_number );
2698 if( padstack ==
pads.end() )
2700 wxLogError(
_(
"Unable to locate padstack %s in file %s\n" ),
2706 auto&
pad = padstack->second;
2714 for(
const auto& [layer_name, layer_data] :
pad.layer_shapes )
2716 auto layer_it =
layers.find( layer_name );
2718 if( layer_it ==
layers.end() || !layer_it->second.conductive )
2724 if( kicad_layer ==
F_Cu )
2725 front_layer = &layer_data;
2726 else if( kicad_layer ==
B_Cu )
2727 back_layer = &layer_data;
2729 inner_layer = &layer_data;
2734 return aA.
shape != aB.shape
2735 || aA.
width != aB.width
2736 || aA.
height != aB.height
2743 std::vector<const FM_PAD_LAYER*> copper_defs;
2745 for(
const FM_PAD_LAYER* def : { front_layer, inner_layer, back_layer } )
2748 copper_defs.push_back( def );
2751 bool needs_padstack =
false;
2753 for(
size_t ii = 1; ii < copper_defs.size(); ++ii )
2755 if( layersDiffer( *copper_defs[0], *copper_defs[ii] ) )
2757 needs_padstack =
true;
2762 if( needs_padstack )
2768 newpad->SetShape( aLayer, aLayerData.shape );
2772 newpad->SetSize( aLayer,
2773 VECTOR2I( aLayerData.width, aLayerData.width ) );
2777 newpad->SetSize( aLayer,
2778 VECTOR2I( aLayerData.width, aLayerData.height ) );
2781 newpad->SetOffset( aLayer,
2782 VECTOR2I( aLayerData.x_offset, aLayerData.y_offset ) );
2785 if( needs_padstack )
2788 applyLayerShape(
F_Cu, *front_layer );
2791 applyLayerShape(
B_Cu, *back_layer );
2795 else if( front_layer )
2800 wxLogWarning(
_(
"Pad '%s' has custom shape with per-layer "
2801 "geometry; custom primitives not supported "
2802 "in padstack mode." ),
2812 int pad_size = std::min(
pad.width,
pad.height );
2815 VECTOR2I( pad_size / 2, pad_size / 2 ) );
2817 std::string custom_name =
pad.custom_name +
"_" +
pin->refdes +
"_" +
2819 auto custom_it =
pad_shapes.find( custom_name );
2825 int last_subseq = 0;
2832 for(
const auto& el : (*custom_it).second.elements )
2838 if(
getLayer( ( *( el.second.begin() ) )->layer ) != primary_layer )
2841 for(
const auto& seg : el.second )
2843 if( seg->subseq > 0 || seg->subseq != last_subseq )
2845 poly_outline.
Polygon(0).back().SetClosed(
true );
2853 if( poly_outline.
VertexCount( 0, hole_idx ) == 0 )
2873 wxLogError(
_(
"Invalid custom pad '%s'. Replacing with "
2875 custom_name.c_str() );
2882 poly_outline.
Move( -newpad->GetPosition() );
2905 wxLogError(
_(
"Invalid custom pad '%s'. Replacing with "
2907 custom_name.c_str() );
2913 wxLogError(
_(
"Could not find custom pad '%s'." ),
2914 custom_name.c_str() );
2924 if( !needs_padstack && (
pad.x_offset ||
pad.y_offset ) )
2943 if(
pad.drill_size_x ==
pad.drill_size_y )
2948 newpad->SetDrillSize(
VECTOR2I(
pad.drill_size_x,
pad.drill_size_y ) );
2956 else if(
pad.bottom )
2957 newpad->SetLayerSet(
PAD::SMDMask().FlipStandardLayers() );
2962 newpad->SetOrientation(
EDA_ANGLE( -src->rotate +
pin->rotation,
2965 newpad->SetOrientation(
EDA_ANGLE( src->rotate -
pin->rotation,
2968 if( newpad->GetSizeX() > 0 || newpad->GetSizeY() > 0 )
2974 wxLogError(
_(
"Invalid zero-sized pad ignored in\nfile: %s" ),
3001 for(
auto& layer :
layers )
3006 layer_set.
set( layer.second.layerid );
3011 for(
auto& layer :
layers )
3013 if( layer.second.conductive )
3016 layer.second.name );
3030 std::vector<const FABMASTER_LAYER*> conductiveLayers;
3032 for(
const auto& layer :
layers )
3034 if( layer.second.conductive )
3035 conductiveLayers.push_back( &layer.second );
3044 auto net_it = netinfo.find(
via->net );
3045 auto padstack =
pads.find(
via->padstack );
3051 if( net_it != netinfo.end() )
3052 new_via->
SetNet( net_it->second );
3054 if( padstack ==
pads.end() )
3058 if( !ds.m_ViasDimensionsList.empty() )
3061 new_via->
SetDrill( ds.m_ViasDimensionsList[0].m_Drill );
3071 new_via->
SetDrill( padstack->second.drill_size_x );
3074 const std::set<std::string>& viaLayers = padstack->second.copper_layers;
3076 if( viaLayers.size() >= 2 )
3084 if( viaLayers.count( layer->name ) )
3093 if( topLayer && botLayer && topLayer != botLayer )
3099 bool isThrough = ( topLayerId ==
F_Cu && botLayerId ==
B_Cu );
3105 if( topLayerId ==
F_Cu || botLayerId ==
B_Cu )
3140 auto net_it = netinfo.find( aLine->netname );
3142 for(
const auto& seg : aLine->segment )
3148 switch( seg->shape )
3161 if( net_it != netinfo.end() )
3162 trk->
SetNet( net_it->second );
3176 if( net_it != netinfo.end() )
3177 trk->
SetNet( net_it->second );
3185 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3193 wxLogError(
_(
"Expecting etch data to be on copper layer. Row found on layer '%s'" ),
3194 seg->layer.c_str() );
3205 int last_subseq = 0;
3210 for(
const auto& seg : aElement )
3212 if( seg->subseq > 0 || seg->subseq != last_subseq )
3219 if( poly_outline.
VertexCount( 0, hole_idx ) == 0 )
3233 return poly_outline;
3260 if( aLine.
segment.size() == 0 )
3266 int first_subseq = -1;
3267 bool have_multiple_subseqs =
false;
3269 for(
const std::unique_ptr<GRAPHIC_ITEM>& gr_item : aLine.
segment )
3271 if( first ==
nullptr )
3273 first = gr_item.get();
3274 first_subseq = gr_item->subseq;
3276 else if( gr_item->subseq == first_subseq )
3278 last = gr_item.get();
3282 have_multiple_subseqs =
true;
3288 wxCHECK( first,
true );
3306 switch( last->
shape )
3328 if(
end.has_value() && start ==
end )
3336std::vector<std::unique_ptr<BOARD_ITEM>>
3339 std::vector<std::unique_ptr<BOARD_ITEM>> new_items;
3344 const auto setShapeParameters = [&](
PCB_SHAPE& aShape )
3348 if( aShape.GetWidth() == 0 )
3349 aShape.SetStroke( defaultStroke );
3352 switch( aGraphic.
shape )
3358 auto new_text = std::make_unique<PCB_TEXT>( &aBoard );
3362 new_text->SetMirrored(
true );
3365 setupText( src, aLayer, *new_text, aBoard, std::nullopt );
3367 new_items.emplace_back( std::move( new_text ) );
3380 for(
const SEG& seg : segs )
3382 auto line = std::make_unique<PCB_SHAPE>( &aBoard );
3384 line->SetStart( seg.A );
3385 line->SetEnd( seg.B );
3387 setShapeParameters( *line );
3388 new_items.emplace_back( std::move( line ) );
3396 auto new_shape = std::make_unique<PCB_SHAPE>( &aBoard );
3398 setShapeParameters( *new_shape );
3400 switch( aGraphic.
shape )
3429 new_shape->SetRadius( src.
radius );
3441 new_shape->SetFilled( src.
fill );
3449 new_shape->SetPolyPoints( src.
m_pts );
3479 new_shape->SetPolyShape( poly );
3484 wxLogError(
_(
"Unhandled shape type %d in polygon on layer %s, seq %d %d" ),
3488 new_items.emplace_back( std::move( new_shape ) );
3492 for( std::unique_ptr<BOARD_ITEM>& new_item : new_items )
3494 new_item->SetLayer( aLayer );
3498 if( new_items.size() > 1 )
3500 auto new_group = std::make_unique<PCB_GROUP>( &aBoard );
3501 for( std::unique_ptr<BOARD_ITEM>& new_item : new_items )
3503 new_group->AddItem( new_item.get() );
3505 new_items.emplace_back( std::move( new_group ) );
3514 if( aLine->segment.empty() )
3528 for(
const auto& seg : aLine->segment )
3530 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3577 if( aLine->segment.size() < 3 )
3581 ZONE* zone =
nullptr;
3584 auto net_it = netinfo.find( aLine->netname );
3586 auto new_layer =
getLayer( aLine->layer );
3591 zone =
new ZONE( aBoard );
3594 if( net_it != netinfo.end() )
3595 zone->
SetNet( net_it->second );
3597 if( aLine->layer ==
"ALL" )
3609 if( aLine->lclass ==
"ROUTE KEEPOUT")
3614 else if( aLine->lclass ==
"VIA KEEPOUT")
3629 std::unique_ptr<SHAPE_LINE_CHAIN> pending_hole =
nullptr;
3632 const auto add_hole_if_valid = [&]()
3636 pending_hole->SetClosed(
true );
3642 wxLogMessage(
_(
"Invalid hole with %d points in zone on layer %s with net %s" ),
3647 pending_hole.reset();
3651 int last_subseq = 0;
3652 for(
const auto& seg : aLine->segment )
3654 if( seg->subseq > 0 && seg->subseq != last_subseq )
3658 if( aLine->lclass ==
"BOUNDARY" )
3661 add_hole_if_valid();
3662 pending_hole = std::make_unique<SHAPE_LINE_CHAIN>();
3663 active_chain = pending_hole.get();
3664 last_subseq = seg->subseq;
3675 active_chain->
Append( start );
3684 wxLogError(
_(
"Outline seems discontinuous: last point was %s, "
3685 "start point of next segment is %s" ),
3700 wxLogError(
_(
"Invalid shape type %d in zone outline" ), seg->shape );
3705 add_hole_if_valid();
3714 delete( zone_outline );
3726 if( aLine->lclass ==
"BOARD GEOMETRY" && aLine->layer !=
"DIMENSION" )
3728 else if( aLine->lclass ==
"DRAWING FORMAT" )
3733 for(
auto& seg : aLine->segment )
3735 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3755 if( geom.subclass ==
"PIN_NUMBER" )
3763 if( !geom.elements->empty() )
3766 if( ( *( geom.elements->begin() ) )->width == 0 )
3787 for(
auto& seg : *geom.elements )
3789 for( std::unique_ptr<BOARD_ITEM>& new_item :
createBoardItems( *aBoard, layer, *seg ) )
3830 for(
auto& track :
traces )
3834 if( track->lclass ==
"ETCH" )
3836 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
virtual void SetNet(NETINFO_ITEM *aNetInfo)
Set a NET_INFO object for the item.
void SetLayer(PCB_LAYER_ID aLayer) override
Set the layer this item is on.
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
virtual void SetFilled(bool aFlag)
virtual void SetTextPos(const VECTOR2I &aPoint)
void SetMirrored(bool isMirrored)
void SetVertJustify(GR_TEXT_V_ALIGN_T aType)
virtual void SetTextWidth(int aWidth)
virtual void SetVisible(bool aVisible)
virtual void SetTextHeight(int aHeight)
void SetKeepUpright(bool aKeepUpright)
virtual void SetText(const wxString &aText)
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.
@ FRONT_INNER_BACK
Up to three shapes can be defined (F_Cu, inner copper layers, B_Cu)
static constexpr PCB_LAYER_ID ALL_LAYERS
! Temporary layer identifier to identify code that is not padstack-aware
static constexpr PCB_LAYER_ID INNER_LAYERS
! The layer identifier to use for "inner layers" on top/inner/bottom padstacks
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 SetShape(SHAPE_T aShape) override
void SetEnd(const VECTOR2I &aEnd) override
void SetPolyShape(const SHAPE_POLY_SET &aShape) override
void SetLayer(PCB_LAYER_ID aLayer) override
Set the layer this item is on.
void SetStart(const VECTOR2I &aStart) override
void SetStroke(const STROKE_PARAMS &aStroke) override
void SetTextThickness(int aWidth) override
The TextThickness is that set by the user.
void SetTextAngle(const EDA_ANGLE &aAngle) 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 SetNet(NETINFO_ITEM *aNetInfo) override
Override that drops aNetInfo when this zone is in copper-thieving fill mode.
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!
Per-layer pad geometry within a pad stack.
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