KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
pcb_io_ipc2581.cpp
Go to the documentation of this file.
1
20#include "pcb_io_ipc2581.h"
21
22#include <base_units.h>
23#include <bezier_curves.h>
24#include <board.h>
27#include <build_version.h>
28#include <callback_gal.h>
32#include <font/font.h>
33#include <hash_eda.h>
34#include <pcb_dimension.h>
35#include <pcb_textbox.h>
36#include <pgm_base.h>
37#include <progress_reporter.h>
39#include <wx_fstream_progress.h>
40
44
45#include <wx/log.h>
46#include <wx/numformatter.h>
47#include <wx/xml/xml.h>
48
49
55static const wxChar traceIpc2581[] = wxT( "KICAD_IPC_2581" );
56
57
59{
61}
62
63
65{
67 delete fp;
68
69 m_loaded_footprints.clear();
70}
71
72
74{
75 std::vector<FOOTPRINT*> retval;
76
78 retval.push_back( static_cast<FOOTPRINT*>( fp->Clone() ) );
79
80 return retval;
81}
82
83
84void PCB_IO_IPC2581::insertNode( wxXmlNode* aParent, wxXmlNode* aNode )
85{
86 // insertNode places the node at the start of the list of children
87
88 if( aParent->GetChildren() )
89 aNode->SetNext( aParent->GetChildren() );
90 else
91 aNode->SetNext( nullptr );
92
93 aParent->SetChildren( aNode );
94 aNode->SetParent( aParent );
95 m_total_bytes += 2 * aNode->GetName().size() + 5;
96}
97
98
99void PCB_IO_IPC2581::insertNodeAfter( wxXmlNode* aPrev, wxXmlNode* aNode )
100{
101 // insertNode places the node directly after aPrev
102
103 aNode->SetNext( aPrev->GetNext() );
104 aPrev->SetNext( aNode );
105 aNode->SetParent( aPrev->GetParent() );
106 m_total_bytes += 2 * aNode->GetName().size() + 5;
107}
108
109
110wxXmlNode* PCB_IO_IPC2581::insertNode( wxXmlNode* aParent, const wxString& aName )
111{
112 // Opening tag, closing tag, brackets and the closing slash
113 m_total_bytes += 2 * aName.size() + 5;
114 wxXmlNode* node = new wxXmlNode( wxXML_ELEMENT_NODE, aName );
115 insertNode( aParent, node );
116 return node;
117}
118
119
120void PCB_IO_IPC2581::appendNode( wxXmlNode* aParent, wxXmlNode* aNode )
121{
122 // AddChild iterates through the entire list of children, so we want to avoid
123 // that if possible. When we share a parent and our next sibling is null,
124 // then we are the last child and can just append to the end of the list.
125
126 static wxXmlNode* lastNode = nullptr;
127
128 if( lastNode && lastNode->GetParent() == aParent && lastNode->GetNext() == nullptr )
129 {
130 aNode->SetParent( aParent );
131 lastNode->SetNext( aNode );
132 }
133 else
134 {
135 aParent->AddChild( aNode );
136 }
137
138 lastNode = aNode;
139
140 // Opening tag, closing tag, brackets and the closing slash
141 m_total_bytes += 2 * aNode->GetName().size() + 5;
142}
143
144
145wxXmlNode* PCB_IO_IPC2581::appendNode( wxXmlNode* aParent, const wxString& aName )
146{
147 wxXmlNode* node = new wxXmlNode( wxXML_ELEMENT_NODE, aName );
148
149 appendNode( aParent, node );
150 return node;
151}
152
153
154wxString PCB_IO_IPC2581::genString( const wxString& aStr, const char* aPrefix ) const
155{
156 wxString str;
157
158 if( m_version == 'C' )
159 {
160 str = aStr;
161 str.Replace( wxT( ":" ), wxT( "_" ) );
162
163 if( aPrefix )
164 str.Prepend( wxString( aPrefix ) + wxT( ":" ) );
165 else
166 str.Prepend( wxT( "KI:" ) );
167 }
168 else
169 {
170 if( aPrefix )
171 str = wxString::Format( "%s_", aPrefix );
172
173 for( wxString::const_iterator iter = aStr.begin(); iter != aStr.end(); ++iter )
174 {
175 if( !m_acceptable_chars.count( *iter ) )
176 str.Append( '_' );
177 else
178 str.Append( *iter );
179 }
180 }
181
182 return str;
183}
184
185
186wxString PCB_IO_IPC2581::genLayerString( PCB_LAYER_ID aLayer, const char* aPrefix ) const
187{
188 return genString( m_board->GetLayerName( aLayer ), aPrefix );
189}
190
191
193 const char* aPrefix ) const
194{
195 return genString( wxString::Format( wxS( "%s_%s" ),
196 m_board->GetLayerName( aTop ),
197 m_board->GetLayerName( aBottom ) ), aPrefix );
198}
199
200
201wxString PCB_IO_IPC2581::pinName( const PAD* aPad ) const
202{
203 wxString name = aPad->GetNumber();
204
205 FOOTPRINT* fp = aPad->GetParentFootprint();
206 size_t ii = 0;
207
208 if( name.empty() && fp )
209 {
210 for( ii = 0; ii < fp->GetPadCount(); ++ii )
211 {
212 if( fp->Pads()[ii] == aPad )
213 break;
214 }
215 }
216
217 // Pins are required to have names, so if our pad doesn't have a name, we need to
218 // generate one that is unique
219 if( aPad->GetAttribute() == PAD_ATTRIB::NPTH )
220 name = wxString::Format( "NPTH%zu", ii );
221 else if( name.empty() )
222 name = wxString::Format( "PAD%zu", ii );
223
224 return genString( name, "PIN" );
225}
226
227
229{
230 auto tryInsert =
231 [&]( const wxString& aName )
232 {
233 if( m_footprint_refdes_dict.count( aName ) )
234 {
235 if( m_footprint_refdes_dict.at( aName ) != aFootprint )
236 return false;
237 }
238 else
239 {
240 m_footprint_refdes_dict.insert( { aName, aFootprint } );
241 }
242
243 return true;
244 };
245
246 if( m_footprint_refdes_reverse_dict.count( aFootprint ) )
247 return m_footprint_refdes_reverse_dict.at( aFootprint );
248
249 wxString baseName = genString( aFootprint->GetReference(), "CMP" );
250 wxString name = baseName;
251 int suffix = 1;
252
253 while( !tryInsert( name ) )
254 name = wxString::Format( "%s_%d", baseName, suffix++ );
255
257
258 return name;
259}
260
261
262wxString PCB_IO_IPC2581::floatVal( double aVal, int aSigFig ) const
263{
264 wxString str = wxString::FromCDouble( aVal, aSigFig == -1 ? m_sigfig : aSigFig );
265
266 // Remove all but the last trailing zeros from str
267 while( str.EndsWith( wxT( "00" ) ) )
268 str.RemoveLast();
269
270 // We don't want to output -0.0 as this value is just 0 for fabs
271 if( str == wxT( "-0.0" ) )
272 return wxT( "0.0" );
273
274 return str;
275}
276
277
278void PCB_IO_IPC2581::addXY( wxXmlNode* aNode, const VECTOR2I& aVec, const char* aXName,
279 const char* aYName )
280{
281 if( aXName )
282 addAttribute( aNode, aXName, floatVal( m_scale * aVec.x ) );
283 else
284 addAttribute( aNode, "x", floatVal( m_scale * aVec.x ) );
285
286 if( aYName )
287 addAttribute( aNode, aYName, floatVal( -m_scale * aVec.y ) );
288 else
289 addAttribute( aNode, "y", floatVal( -m_scale * aVec.y ) );
290}
291
292
293void PCB_IO_IPC2581::addAttribute( wxXmlNode* aNode, const wxString& aName, const wxString& aValue )
294{
295 m_total_bytes += aName.size() + aValue.size() + 4;
296 aNode->AddAttribute( aName, aValue );
297}
298
299
301{
302 wxXmlNode* xmlHeaderNode = new wxXmlNode(wxXML_ELEMENT_NODE, "IPC-2581");
303 addAttribute( xmlHeaderNode, "revision", m_version);
304 addAttribute( xmlHeaderNode, "xmlns", "http://webstds.ipc.org/2581");
305 addAttribute( xmlHeaderNode, "xmlns:xsi", "http://www.w3.org/2001/XMLSchema-instance");
306 addAttribute( xmlHeaderNode, "xmlns:xsd", "http://www.w3.org/2001/XMLSchema");
307
308 if( m_version == 'B' )
309 {
310 addAttribute( xmlHeaderNode, "xsi:schemaLocation",
311 "http://webstds.ipc.org/2581 http://webstds.ipc.org/2581/IPC-2581B1.xsd" );
312 }
313 else
314 {
315 addAttribute( xmlHeaderNode, "xsi:schemaLocation",
316 "http://webstds.ipc.org/2581 http://webstds.ipc.org/2581/IPC-2581C.xsd" );
317 }
318
319 m_xml_doc->SetRoot( xmlHeaderNode );
320
321 return xmlHeaderNode;
322}
323
324
326{
328 m_progressReporter->AdvancePhase( _( "Generating content section" ) );
329
330 wxXmlNode* contentNode = appendNode( m_xml_root, "Content" );
331 addAttribute( contentNode, "roleRef", "Owner" );
332
333 wxXmlNode* node = appendNode( contentNode, "FunctionMode" );
334 addAttribute( node, "mode", "ASSEMBLY" );
335
336 // This element is deprecated in revision 'C' and later
337 if( m_version == 'B' )
338 addAttribute( node, "level", "3" );
339
340 node = appendNode( contentNode, "StepRef" );
341 wxFileName fn( m_board->GetFileName() );
342 addAttribute( node, "name", genString( fn.GetName(), "BOARD" ) );
343
344 wxXmlNode* color_node = generateContentStackup( contentNode );
345
346 if( m_version == 'C' )
347 {
348 contentNode->AddChild( color_node );
349 m_line_node = appendNode( contentNode, "DictionaryLineDesc" );
351
352 wxXmlNode* fillNode = appendNode( contentNode, "DictionaryFillDesc" );
353 addAttribute( fillNode, "units", m_units_str );
354
355 m_shape_std_node = appendNode( contentNode, "DictionaryStandard" );
357
358 m_shape_user_node = appendNode( contentNode, "DictionaryUser" );
360 }
361 else
362 {
363 m_shape_std_node = appendNode( contentNode, "DictionaryStandard" );
365
366 m_shape_user_node = appendNode( contentNode, "DictionaryUser" );
368
369 m_line_node = appendNode( contentNode, "DictionaryLineDesc" );
371
372 contentNode->AddChild( color_node );
373 }
374
375 return contentNode;
376}
377
378
379void PCB_IO_IPC2581::addLocationNode( wxXmlNode* aNode, double aX, double aY )
380{
381 wxXmlNode* location_node = appendNode( aNode, "Location" );
382 addXY( location_node, VECTOR2I( aX, aY ) );
383}
384
385
386void PCB_IO_IPC2581::addLocationNode( wxXmlNode* aNode, const PAD& aPad, bool aRelative )
387{
388 VECTOR2D pos{};
389
390 if( aRelative )
391 pos = aPad.GetFPRelativePosition();
392 else
393 pos = aPad.GetPosition();
394
395 if( aPad.GetOffset( PADSTACK::ALL_LAYERS ).x != 0 || aPad.GetOffset( PADSTACK::ALL_LAYERS ).y != 0 )
396 pos += aPad.GetOffset( PADSTACK::ALL_LAYERS );
397
398 addLocationNode( aNode, pos.x, pos.y );
399}
400
401
402void PCB_IO_IPC2581::addLocationNode( wxXmlNode* aNode, const PCB_SHAPE& aShape )
403{
404 VECTOR2D pos{};
405
406 switch( aShape.GetShape() )
407 {
408 // Rectangles in KiCad are mapped by their corner while IPC2581 uses the center
409 case SHAPE_T::RECTANGLE:
410 pos = aShape.GetPosition()
411 + VECTOR2I( aShape.GetRectangleWidth() / 2.0, aShape.GetRectangleHeight() / 2.0 );
412 break;
413 // Both KiCad and IPC2581 use the center of the circle
414 case SHAPE_T::CIRCLE:
415 pos = aShape.GetPosition();
416 break;
417
418 // KiCad uses the exact points on the board, so we want the reference location to be 0,0
419 case SHAPE_T::POLY:
420 case SHAPE_T::BEZIER:
421 case SHAPE_T::SEGMENT:
422 case SHAPE_T::ARC:
423 pos = VECTOR2D( 0, 0 );
424 break;
425
426 case SHAPE_T::UNDEFINED:
427 wxFAIL;
428 }
429
430 addLocationNode( aNode, pos.x, pos.y );
431}
432
433
434size_t PCB_IO_IPC2581::lineHash( int aWidth, LINE_STYLE aDashType )
435{
436 size_t hash = hash_val( aWidth );
437 hash_combine( hash, aDashType );
438
439 return hash;
440}
441
442
444{
445 return hash_fp_item( &aShape, HASH_POS | REL_COORD );
446}
447
448
449wxXmlNode* PCB_IO_IPC2581::generateContentStackup( wxXmlNode* aContentNode )
450{
451
453 BOARD_STACKUP& stackup = bds.GetStackupDescriptor();
454 stackup.SynchronizeWithBoard( &bds );
455
456 wxXmlNode* color_node = new wxXmlNode( wxXML_ELEMENT_NODE, "DictionaryColor" );
457
458 for( BOARD_STACKUP_ITEM* item: stackup.GetList() )
459 {
460 wxString layer_name = item->GetLayerName();
461 int sub_layer_count = 1;
462
463 if( layer_name.empty() )
464 layer_name = m_board->GetLayerName( item->GetBrdLayerId() );
465
466 layer_name = genString( layer_name, "LAYER" );
467
468 if( item->GetType() == BS_ITEM_TYPE_DIELECTRIC )
469 {
470 layer_name = genString( wxString::Format( "DIELECTRIC_%d", item->GetDielectricLayerId() ),
471 "LAYER" );
472 sub_layer_count = item->GetSublayersCount();
473 }
474 else
475 {
476 m_layer_name_map.emplace( item->GetBrdLayerId(), layer_name );
477 }
478
479 for( int sub_idx = 0; sub_idx < sub_layer_count; sub_idx++ )
480 {
481 wxString sub_layer_name = layer_name;
482
483 if( sub_idx > 0 )
484 sub_layer_name += wxString::Format( "_%d", sub_idx );
485
486 wxXmlNode* node = appendNode( aContentNode, "LayerRef" );
487 addAttribute( node, "name", sub_layer_name );
488
489 if( !IsPrmSpecified( item->GetColor( sub_idx ) ) )
490 continue;
491
492 wxXmlNode* entry_color = appendNode( color_node, "EntryColor" );
493 addAttribute( entry_color, "id", genString( sub_layer_name, "COLOR" ) );
494 wxXmlNode* color = appendNode( entry_color, "Color" );
495
496 wxString colorName = item->GetColor( sub_idx );
497
498 if( colorName.StartsWith( wxT( "#" ) ) ) // This is a user defined color,
499 // not in standard color list.
500 {
501 COLOR4D layer_color( colorName );
502 addAttribute( color, "r", wxString::Format( "%d",
503 KiROUND( layer_color.r * 255 ) ) );
504 addAttribute( color, "g", wxString::Format( "%d",
505 KiROUND( layer_color.g * 255 ) ) );
506 addAttribute( color, "b", wxString::Format( "%d",
507 KiROUND( layer_color.b * 255 ) ) );
508 }
509 else
510 {
511 for( const FAB_LAYER_COLOR& fab_color : GetStandardColors( item->GetType() ) )
512 {
513 if( fab_color.GetName() == colorName )
514 {
515 addAttribute( color, "r", wxString::Format( "%d", KiROUND( fab_color.GetColor( item->GetType() ).r * 255 ) ) );
516 addAttribute( color, "g", wxString::Format( "%d", KiROUND( fab_color.GetColor( item->GetType() ).g * 255 ) ) );
517 addAttribute( color, "b", wxString::Format( "%d", KiROUND( fab_color.GetColor( item->GetType() ).b * 255 ) ) );
518 break;
519 }
520 }
521 }
522 }
523 }
524
525 return color_node;
526}
527
528
529void PCB_IO_IPC2581::addFillDesc( wxXmlNode* aNode, FILL_T aFill, bool aForce )
530{
531 if( aFill == FILL_T::FILLED_SHAPE )
532 {
533 // By default, we do not fill shapes because FILL is the default value for most.
534 // But for some outlines, we may need to force a fill.
535 if( aForce )
536 {
537 wxXmlNode* fillDesc_node = appendNode( aNode, "FillDesc" );
538 addAttribute( fillDesc_node, "fillProperty", "FILL" );
539 }
540 }
541 else
542 {
543 wxXmlNode* fillDesc_node = appendNode( aNode, "FillDesc" );
544 addAttribute( fillDesc_node, "fillProperty", "HOLLOW" );
545 }
546}
547
548
549void PCB_IO_IPC2581::addLineDesc( wxXmlNode* aNode, int aWidth, LINE_STYLE aDashType, bool aForce )
550{
551 wxCHECK_RET( aNode, "aNode is null" );
552
553 if( aWidth < 0 )
554 return;
555
556 wxXmlNode* entry_node = nullptr;
557
558 if( !aForce )
559 {
560 size_t hash = lineHash( aWidth, aDashType );
561 wxString name = wxString::Format( "LINE_%zu", m_line_dict.size() + 1 );
562 auto[ iter, inserted ] = m_line_dict.emplace( hash, name );
563
564 // Either add a new entry or reference an existing one
565 wxXmlNode* lineDesc_node = appendNode( aNode, "LineDescRef" );
566 addAttribute( lineDesc_node, "id", iter->second );
567
568 if( !inserted )
569 return;
570
571 entry_node = appendNode( m_line_node, "EntryLineDesc" );
572 addAttribute( entry_node, "id", name );
573 }
574 else
575 {
576 // Force the LineDesc to be added directly to the parent node
577 entry_node = aNode;
578 }
579
580 wxXmlNode* line_node = appendNode( entry_node, "LineDesc" );
581 addAttribute( line_node, "lineWidth", floatVal( m_scale * aWidth ) );
582 addAttribute( line_node, "lineEnd", "ROUND" );
583
584 switch( aDashType )
585 {
586 case LINE_STYLE::DOT:
587 addAttribute( line_node, "lineProperty", "DOTTED" );
588 break;
589 case LINE_STYLE::DASH:
590 addAttribute( line_node, "lineProperty", "DASHED" );
591 break;
592 case LINE_STYLE::DASHDOT:
593 addAttribute( line_node, "lineProperty", "CENTER" );
594 break;
595 case LINE_STYLE::DASHDOTDOT:
596 addAttribute( line_node, "lineProperty", "PHANTOM" );
597 break;
598 default:
599 break;
600 }
601}
602
603
604void PCB_IO_IPC2581::addKnockoutText( wxXmlNode* aContentNode, PCB_TEXT* aText )
605{
606 SHAPE_POLY_SET finalPoly;
607
608 aText->TransformTextToPolySet( finalPoly, 0, ARC_HIGH_DEF, ERROR_INSIDE );
609 finalPoly.Fracture();
610
611 for( int ii = 0; ii < finalPoly.OutlineCount(); ++ii )
612 addContourNode( aContentNode, finalPoly, ii );
613}
614
615
616void PCB_IO_IPC2581::addText( wxXmlNode* aContentNode, EDA_TEXT* aText,
617 const KIFONT::METRICS& aFontMetrics )
618{
620 KIFONT::FONT* font = aText->GetFont();
621 TEXT_ATTRIBUTES attrs = aText->GetAttributes();
622
624 attrs.m_Angle = aText->GetDrawRotation();
625 attrs.m_Multiline = false;
626
627 wxXmlNode* text_node = appendNode( aContentNode, "UserSpecial" );
628
629 if( !font )
630 font = KIFONT::FONT::GetFont();
631
632 std::list<VECTOR2I> pts;
633
634 auto push_pts =
635 [&]()
636 {
637 if( pts.size() < 2 )
638 return;
639
640 wxXmlNode* line_node = nullptr;
641
642 // Polylines are only allowed for more than 3 points (in version B).
643 // Otherwise, we have to use a line
644 if( pts.size() < 3 )
645 {
646 line_node = appendNode( text_node, "Line" );
647 addXY( line_node, pts.front(), "startX", "startY" );
648 addXY( line_node, pts.back(), "endX", "endY" );
649 }
650 else
651 {
652 line_node = appendNode( text_node, "Polyline" );
653 wxXmlNode* point_node = appendNode( line_node, "PolyBegin" );
654 addXY( point_node, pts.front() );
655
656 auto iter = pts.begin();
657
658 for( ++iter; iter != pts.end(); ++iter )
659 {
660 wxXmlNode* point_node = appendNode( line_node, "PolyStepSegment" );
661 addXY( point_node, *iter );
662 }
663
664 }
665
666 addLineDesc( line_node, attrs.m_StrokeWidth, LINE_STYLE::SOLID );
667 pts.clear();
668 };
669
670 CALLBACK_GAL callback_gal( empty_opts,
671 // Stroke callback
672 [&]( const VECTOR2I& aPt1, const VECTOR2I& aPt2 )
673 {
674 if( !pts.empty() )
675 {
676 if( aPt1 == pts.back() )
677 pts.push_back( aPt2 );
678 else if( aPt2 == pts.front() )
679 pts.push_front( aPt1 );
680 else if( aPt1 == pts.front() )
681 pts.push_front( aPt2 );
682 else if( aPt2 == pts.back() )
683 pts.push_back( aPt1 );
684 else
685 {
686 push_pts();
687 pts.push_back( aPt1 );
688 pts.push_back( aPt2 );
689 }
690 }
691 else
692 {
693 pts.push_back( aPt1 );
694 pts.push_back( aPt2 );
695 }
696 },
697 // Polygon callback
698 [&]( const SHAPE_LINE_CHAIN& aPoly )
699 {
700 if( aPoly.PointCount() < 3 )
701 return;
702
703 wxXmlNode* outline_node = appendNode( text_node, "Outline" );
704 wxXmlNode* poly_node = appendNode( outline_node, "Polygon" );
705 addLineDesc( outline_node, 0, LINE_STYLE::SOLID );
706
707 const std::vector<VECTOR2I>& pts = aPoly.CPoints();
708 wxXmlNode* point_node = appendNode( poly_node, "PolyBegin" );
709 addXY( point_node, pts.front() );
710
711 for( size_t ii = 1; ii < pts.size(); ++ii )
712 {
713 wxXmlNode* point_node =
714 appendNode( poly_node, "PolyStepSegment" );
715 addXY( point_node, pts[ii] );
716 }
717
718 point_node = appendNode( poly_node, "PolyStepSegment" );
719 addXY( point_node, pts.front() );
720 } );
721
722 //TODO: handle multiline text
723
724 font->Draw( &callback_gal, aText->GetShownText( true ), aText->GetTextPos(), attrs,
725 aFontMetrics );
726
727 if( !pts.empty() )
728 push_pts();
729
730 if( text_node->GetChildren() == nullptr )
731 {
732 aContentNode->RemoveChild( text_node );
733 delete text_node;
734 }
735}
736
737
738void PCB_IO_IPC2581::addShape( wxXmlNode* aContentNode, const PAD& aPad, PCB_LAYER_ID aLayer )
739{
740 size_t hash = hash_fp_item( &aPad, 0 );
741 auto iter = m_std_shape_dict.find( hash );
742
743 if( iter != m_std_shape_dict.end() )
744 {
745 wxXmlNode* shape_node = appendNode( aContentNode, "StandardPrimitiveRef" );
746 addAttribute( shape_node, "id", iter->second );
747 return;
748 }
749
750 int maxError = m_board->GetDesignSettings().m_MaxError;
751 wxString name;
752 VECTOR2I expansion{ 0, 0 };
753
754 if( LSET( { F_Mask, B_Mask } ).Contains( aLayer ) )
755 expansion.x = expansion.y = 2 * aPad.GetSolderMaskExpansion( PADSTACK::ALL_LAYERS );
756
757 if( LSET( { F_Paste, B_Paste } ).Contains( aLayer ) )
758 expansion = 2 * aPad.GetSolderPasteMargin( PADSTACK::ALL_LAYERS );
759
760 switch( aPad.GetShape( PADSTACK::ALL_LAYERS ) )
761 {
762 case PAD_SHAPE::CIRCLE:
763 {
764 name = wxString::Format( "CIRCLE_%zu", m_std_shape_dict.size() + 1 );
765 m_std_shape_dict.emplace( hash, name );
766
767 wxXmlNode* entry_node = appendNode( m_shape_std_node, "EntryStandard" );
768 addAttribute( entry_node, "id", name );
769
770 wxXmlNode* circle_node = appendNode( entry_node, "Circle" );
771 circle_node->AddAttribute( "diameter",
772 floatVal( m_scale * ( expansion.x + aPad.GetSizeX() ) ) );
773 break;
774 }
775
776 case PAD_SHAPE::RECTANGLE:
777 {
778 name = wxString::Format( "RECT_%zu", m_std_shape_dict.size() + 1 );
779 m_std_shape_dict.emplace( hash, name );
780
781 wxXmlNode* entry_node = appendNode( m_shape_std_node, "EntryStandard" );
782 addAttribute( entry_node, "id", name );
783
784 wxXmlNode* rect_node = appendNode( entry_node, "RectCenter" );
785 VECTOR2D pad_size = aPad.GetSize( PADSTACK::ALL_LAYERS ) + expansion;
786 addAttribute( rect_node, "width", floatVal( m_scale * std::abs( pad_size.x ) ) );
787 addAttribute( rect_node, "height", floatVal( m_scale * std::abs( pad_size.y ) ) );
788
789 break;
790
791 }
792 case PAD_SHAPE::OVAL:
793 {
794 name = wxString::Format( "OVAL_%zu", m_std_shape_dict.size() + 1 );
795 m_std_shape_dict.emplace( hash, name );
796
797 wxXmlNode* entry_node = appendNode( m_shape_std_node, "EntryStandard" );
798 addAttribute( entry_node, "id", name );
799
800 wxXmlNode* oval_node = appendNode( entry_node, "Oval" );
801 VECTOR2D pad_size = aPad.GetSize( PADSTACK::ALL_LAYERS ) + expansion;
802 addAttribute( oval_node, "width", floatVal( m_scale * pad_size.x ) );
803 addAttribute( oval_node, "height", floatVal( m_scale * pad_size.y ) );
804
805 break;
806 }
807
808 case PAD_SHAPE::ROUNDRECT:
809 {
810 name = wxString::Format( "ROUNDRECT_%zu", m_std_shape_dict.size() + 1 );
811 m_std_shape_dict.emplace( hash, name );
812
813 wxXmlNode* entry_node = appendNode( m_shape_std_node, "EntryStandard" );
814 addAttribute( entry_node, "id", name );
815
816 wxXmlNode* roundrect_node = appendNode( entry_node, "RectRound" );
817 VECTOR2D pad_size = aPad.GetSize( PADSTACK::ALL_LAYERS ) + expansion;
818 addAttribute( roundrect_node, "width", floatVal( m_scale * pad_size.x ) );
819 addAttribute( roundrect_node, "height", floatVal( m_scale * pad_size.y ) );
820 roundrect_node->AddAttribute( "radius",
822 addAttribute( roundrect_node, "upperRight", "true" );
823 addAttribute( roundrect_node, "upperLeft", "true" );
824 addAttribute( roundrect_node, "lowerRight", "true" );
825 addAttribute( roundrect_node, "lowerLeft", "true" );
826
827 break;
828 }
829
830 case PAD_SHAPE::CHAMFERED_RECT:
831 {
832 name = wxString::Format( "RECTCHAMFERED_%zu", m_std_shape_dict.size() + 1 );
833 m_std_shape_dict.emplace( hash, name );
834
835 wxXmlNode* entry_node = appendNode( m_shape_std_node, "EntryStandard" );
836 addAttribute( entry_node, "id", name );
837
838 wxXmlNode* chamfered_node = appendNode( entry_node, "RectCham" );
839 VECTOR2D pad_size = aPad.GetSize( PADSTACK::ALL_LAYERS ) + expansion;
840 addAttribute( chamfered_node, "width", floatVal( m_scale * pad_size.x ) );
841 addAttribute( chamfered_node, "height", floatVal( m_scale * pad_size.y ) );
842
843 int shorterSide = std::min( pad_size.x, pad_size.y );
844 int chamfer = std::max( 0, KiROUND( aPad.GetChamferRectRatio( PADSTACK::ALL_LAYERS ) * shorterSide ) );
845
846 addAttribute( chamfered_node, "chamfer", floatVal( m_scale * chamfer ) );
847
848 int positions = aPad.GetChamferPositions( PADSTACK::ALL_LAYERS );
849
850 if( positions & RECT_CHAMFER_TOP_LEFT )
851 addAttribute( chamfered_node, "upperLeft", "true" );
852 if( positions & RECT_CHAMFER_TOP_RIGHT )
853 addAttribute( chamfered_node, "upperRight", "true" );
854 if( positions & RECT_CHAMFER_BOTTOM_LEFT )
855 addAttribute( chamfered_node, "lowerLeft", "true" );
856 if( positions & RECT_CHAMFER_BOTTOM_RIGHT )
857 addAttribute( chamfered_node, "lowerRight", "true" );
858
859 break;
860 }
861
862 case PAD_SHAPE::TRAPEZOID:
863 {
864 name = wxString::Format( "TRAPEZOID_%zu", m_std_shape_dict.size() + 1 );
865 m_std_shape_dict.emplace( hash, name );
866
867 wxXmlNode* entry_node = appendNode( m_shape_std_node, "EntryStandard" );
868 addAttribute( entry_node, "id", name );
869
870 VECTOR2I pad_size = aPad.GetSize( PADSTACK::ALL_LAYERS );
871 VECTOR2I trap_delta = aPad.GetDelta( PADSTACK::ALL_LAYERS );
872 SHAPE_POLY_SET outline;
873 outline.NewOutline();
874 int dx = pad_size.x / 2;
875 int dy = pad_size.y / 2;
876 int ddx = trap_delta.x / 2;
877 int ddy = trap_delta.y / 2;
878
879 outline.Append( -dx - ddy, dy + ddx );
880 outline.Append( dx + ddy, dy - ddx );
881 outline.Append( dx - ddy, -dy + ddx );
882 outline.Append( -dx + ddy, -dy - ddx );
883
884 // Shape polygon can have holes so use InflateWithLinkedHoles(), not Inflate()
885 // which can create bad shapes if margin.x is < 0
886 if( expansion.x )
887 {
888 outline.InflateWithLinkedHoles( expansion.x, CORNER_STRATEGY::ROUND_ALL_CORNERS,
889 maxError );
890 }
891
892 addContourNode( entry_node, outline );
893
894 break;
895 }
896 case PAD_SHAPE::CUSTOM:
897 {
898 name = wxString::Format( "CUSTOM_%zu", m_std_shape_dict.size() + 1 );
899 m_std_shape_dict.emplace( hash, name );
900
901 wxXmlNode* entry_node = appendNode( m_shape_std_node, "EntryStandard" );
902 addAttribute( entry_node, "id", name );
903
904 SHAPE_POLY_SET shape;
906
907 if( expansion != VECTOR2I( 0, 0 ) )
908 {
909 shape.InflateWithLinkedHoles( std::max( expansion.x, expansion.y ),
910 CORNER_STRATEGY::ROUND_ALL_CORNERS, maxError );
911 }
912
913 addContourNode( entry_node, shape );
914 break;
915 }
916 default:
917 wxLogError( "Unknown pad type" );
918 break;
919 }
920
921 if( !name.empty() )
922 {
923 m_std_shape_dict.emplace( hash, name );
924 wxXmlNode* shape_node = appendNode( aContentNode, "StandardPrimitiveRef" );
925 addAttribute( shape_node, "id", name );
926 }
927}
928
929
930void PCB_IO_IPC2581::addShape( wxXmlNode* aContentNode, const PCB_SHAPE& aShape )
931{
932 size_t hash = shapeHash( aShape );
933 auto iter = m_user_shape_dict.find( hash );
934 wxString name;
935
936 if( iter != m_user_shape_dict.end() )
937 {
938 wxXmlNode* shape_node = appendNode( aContentNode, "UserPrimitiveRef" );
939 addAttribute( shape_node, "id", iter->second );
940 return;
941 }
942
943 switch( aShape.GetShape() )
944 {
945 case SHAPE_T::CIRCLE:
946 {
947 name = wxString::Format( "UCIRCLE_%zu", m_user_shape_dict.size() + 1 );
948 m_user_shape_dict.emplace( hash, name );
949 int diameter = aShape.GetRadius() * 2.0;
950 int width = aShape.GetStroke().GetWidth();
951 LINE_STYLE dash = aShape.GetStroke().GetLineStyle();
952
953
954 wxXmlNode* entry_node = appendNode( m_shape_user_node, "EntryUser" );
955 addAttribute( entry_node, "id", name );
956 wxXmlNode* special_node = appendNode( entry_node, "UserSpecial" );
957
958 wxXmlNode* circle_node = appendNode( special_node, "Circle" );
959
960 if( aShape.GetFillMode() == FILL_T::NO_FILL )
961 {
962 addAttribute( circle_node, "diameter", floatVal( m_scale * diameter ) );
963 addLineDesc( circle_node, width, dash, true );
964 }
965 else
966 {
967 // IPC2581 does not allow strokes on filled elements
968 addAttribute( circle_node, "diameter", floatVal( m_scale * ( diameter + width ) ) );
969 }
970
971 addFillDesc( circle_node, aShape.GetFillMode() );
972
973 break;
974 }
975
976 case SHAPE_T::RECTANGLE:
977 {
978 name = wxString::Format( "URECT_%zu", m_user_shape_dict.size() + 1 );
979 m_user_shape_dict.emplace( hash, name );
980
981 wxXmlNode* entry_node = appendNode( m_shape_user_node, "EntryUser" );
982 addAttribute( entry_node, "id", name );
983 wxXmlNode* special_node = appendNode( entry_node, "UserSpecial" );
984
985 int width = std::abs( aShape.GetRectangleWidth() );
986 int height = std::abs( aShape.GetRectangleHeight() );
987 int stroke_width = aShape.GetStroke().GetWidth();
988 LINE_STYLE dash = aShape.GetStroke().GetLineStyle();
989
990 wxXmlNode* rect_node = appendNode( special_node, "RectRound" );
991 addLineDesc( rect_node, aShape.GetStroke().GetWidth(), aShape.GetStroke().GetLineStyle(),
992 true );
993
994 if( aShape.GetFillMode() == FILL_T::NO_FILL )
995 {
996 addAttribute( rect_node, "upperRight", "false" );
997 addAttribute( rect_node, "upperLeft", "false" );
998 addAttribute( rect_node, "lowerRight", "false" );
999 addAttribute( rect_node, "lowerLeft", "false" );
1000 }
1001 else
1002 {
1003 addAttribute( rect_node, "upperRight", "true" );
1004 addAttribute( rect_node, "upperLeft", "true" );
1005 addAttribute( rect_node, "lowerRight", "true" );
1006 addAttribute( rect_node, "lowerLeft", "true" );
1007 width += stroke_width;
1008 height += stroke_width;
1009 }
1010
1011 addFillDesc( rect_node, aShape.GetFillMode() );
1012
1013 addAttribute( rect_node, "width", floatVal( m_scale * width ) );
1014 addAttribute( rect_node, "height", floatVal( m_scale * height ) );
1015 addAttribute( rect_node, "radius", floatVal( m_scale * ( stroke_width / 2.0 ) ) );
1016
1017 break;
1018 }
1019
1020 case SHAPE_T::POLY:
1021 {
1022 name = wxString::Format( "UPOLY_%zu", m_user_shape_dict.size() + 1 );
1023 m_user_shape_dict.emplace( hash, name );
1024
1025 wxXmlNode* entry_node = appendNode( m_shape_user_node, "EntryUser" );
1026 addAttribute( entry_node, "id", name );
1027
1028 // If we are stroking a polygon, we need two contours. This is only allowed
1029 // inside a "UserSpecial" shape
1030 wxXmlNode* special_node = appendNode( entry_node, "UserSpecial" );
1031
1032 const SHAPE_POLY_SET& poly_set = aShape.GetPolyShape();
1033
1034 for( int ii = 0; ii < poly_set.OutlineCount(); ++ii )
1035 {
1036 if( aShape.GetFillMode() != FILL_T::NO_FILL )
1037 {
1038 // IPC2581 does not allow strokes on filled elements
1039 addContourNode( special_node, poly_set, ii, FILL_T::FILLED_SHAPE, 0,
1040 LINE_STYLE::SOLID );
1041 }
1042
1043 addContourNode( special_node, poly_set, ii, FILL_T::NO_FILL,
1044 aShape.GetStroke().GetWidth(), aShape.GetStroke().GetLineStyle() );
1045 }
1046
1047 break;
1048 }
1049
1050 case SHAPE_T::ARC:
1051 {
1052 wxXmlNode* arc_node = appendNode( aContentNode, "Arc" );
1053 addXY( arc_node, aShape.GetStart(), "startX", "startY" );
1054 addXY( arc_node, aShape.GetEnd(), "endX", "endY" );
1055 addXY( arc_node, aShape.GetCenter(), "centerX", "centerY" );
1056
1057 //N.B. because our coordinate system is flipped, we need to flip the arc direction
1058 addAttribute( arc_node, "clockwise", !aShape.IsClockwiseArc() ? "true" : "false" );
1059
1060 if( aShape.GetStroke().GetWidth() > 0 )
1061 {
1062 addLineDesc( arc_node, aShape.GetStroke().GetWidth(),
1063 aShape.GetStroke().GetLineStyle(), true );
1064 }
1065
1066 break;
1067 }
1068
1069 case SHAPE_T::BEZIER:
1070 {
1071 wxXmlNode* polyline_node = appendNode( aContentNode, "Polyline" );
1072 std::vector<VECTOR2I> ctrlPoints = { aShape.GetStart(), aShape.GetBezierC1(),
1073 aShape.GetBezierC2(), aShape.GetEnd() };
1074 BEZIER_POLY converter( ctrlPoints );
1075 std::vector<VECTOR2I> points;
1076 converter.GetPoly( points, ARC_HIGH_DEF );
1077
1078 wxXmlNode* point_node = appendNode( polyline_node, "PolyBegin" );
1079 addXY( point_node, points[0] );
1080
1081 for( size_t i = 1; i < points.size(); i++ )
1082 {
1083 wxXmlNode* point_node = appendNode( polyline_node, "PolyStepSegment" );
1084 addXY( point_node, points[i] );
1085 }
1086
1087 if( aShape.GetStroke().GetWidth() > 0 )
1088 {
1089 addLineDesc( polyline_node, aShape.GetStroke().GetWidth(),
1090 aShape.GetStroke().GetLineStyle(), true );
1091 }
1092
1093 break;
1094 }
1095
1096 case SHAPE_T::SEGMENT:
1097 {
1098 wxXmlNode* line_node = appendNode( aContentNode, "Line" );
1099 addXY( line_node, aShape.GetStart(), "startX", "startY" );
1100 addXY( line_node, aShape.GetEnd(), "endX", "endY" );
1101
1102 if( aShape.GetStroke().GetWidth() > 0 )
1103 {
1104 addLineDesc( line_node, aShape.GetStroke().GetWidth(),
1105 aShape.GetStroke().GetLineStyle(), true );
1106 }
1107
1108 break;
1109 }
1110
1111 case SHAPE_T::UNDEFINED:
1112 wxFAIL;
1113 }
1114
1115 if( !name.empty() )
1116 {
1117 wxXmlNode* shape_node = appendNode( aContentNode, "UserPrimitiveRef" );
1118 addAttribute( shape_node, "id", name );
1119 }
1120
1121}
1122
1123
1124void PCB_IO_IPC2581::addSlotCavity( wxXmlNode* aNode, const PAD& aPad, const wxString& aName )
1125{
1126 wxXmlNode* slotNode = appendNode( aNode, "SlotCavity" );
1127 addAttribute( slotNode, "name", aName );
1128 addAttribute( slotNode, "platingStatus", aPad.GetAttribute() == PAD_ATTRIB::PTH ? "PLATED"
1129 : "NONPLATED" );
1130 addAttribute( slotNode, "plusTol", "0.0" );
1131 addAttribute( slotNode, "minusTol", "0.0" );
1132
1133 if( m_version > 'B' )
1134 addLocationNode( slotNode, 0.0, 0.0 );
1135
1136 SHAPE_POLY_SET poly_set;
1137 aPad.GetEffectiveShape( PADSTACK::ALL_LAYERS )->TransformToPolygon( poly_set, 0, ERROR_INSIDE );
1138
1139 addOutlineNode( slotNode, poly_set );
1140}
1141
1142
1144{
1145 wxXmlNode* logisticNode = appendNode( m_xml_root, "LogisticHeader" );
1146
1147 wxXmlNode* roleNode = appendNode( logisticNode, "Role" );
1148 addAttribute( roleNode, "id", "Owner" );
1149 addAttribute( roleNode, "roleFunction", "SENDER" );
1150
1151 m_enterpriseNode = appendNode( logisticNode, "Enterprise" );
1152 addAttribute( m_enterpriseNode, "id", "UNKNOWN" );
1153 addAttribute( m_enterpriseNode, "code", "NONE" );
1154
1155 wxXmlNode* personNode = appendNode( logisticNode, "Person" );
1156 addAttribute( personNode, "name", "UNKNOWN" );
1157 addAttribute( personNode, "enterpriseRef", "UNKNOWN" );
1158 addAttribute( personNode, "roleRef", "Owner" );
1159
1160 return logisticNode;
1161}
1162
1163
1165{
1166 if( m_progressReporter )
1167 m_progressReporter->AdvancePhase( _( "Generating history section" ) );
1168
1169 wxXmlNode* historyNode = appendNode( m_xml_root, "HistoryRecord" );
1170 addAttribute( historyNode, "number", "1" );
1171 addAttribute( historyNode, "origination", wxDateTime::Now().FormatISOCombined() );
1172 addAttribute( historyNode, "software", "KiCad EDA" );
1173 addAttribute( historyNode, "lastChange", wxDateTime::Now().FormatISOCombined() );
1174
1175 wxXmlNode* fileRevisionNode = appendNode( historyNode, "FileRevision" );
1176 addAttribute( fileRevisionNode, "fileRevisionId", "1" );
1177 addAttribute( fileRevisionNode, "comment", "NO COMMENT" );
1178 addAttribute( fileRevisionNode, "label", "NO LABEL" );
1179
1180 wxXmlNode* softwarePackageNode = appendNode( fileRevisionNode, "SoftwarePackage" );
1181 addAttribute( softwarePackageNode, "name", "KiCad" );
1182 addAttribute( softwarePackageNode, "revision", GetMajorMinorPatchVersion() );
1183 addAttribute( softwarePackageNode, "vendor", "KiCad EDA" );
1184
1185 wxXmlNode* certificationNode = appendNode( softwarePackageNode, "Certification" );
1186 addAttribute( certificationNode, "certificationStatus", "SELFTEST" );
1187
1188 return historyNode;
1189}
1190
1191
1192wxXmlNode* PCB_IO_IPC2581::generateBOMSection( wxXmlNode* aEcadNode )
1193{
1194 if( m_progressReporter )
1195 m_progressReporter->AdvancePhase( _( "Generating BOM section" ) );
1196
1197 struct REFDES
1198 {
1199 wxString m_name;
1200 wxString m_pkg;
1201 bool m_populate;
1202 wxString m_layer;
1203 };
1204
1205 struct BOM_ENTRY
1206 {
1207 BOM_ENTRY()
1208 {
1209 m_refdes = new std::vector<REFDES>();
1210 m_props = new std::map<wxString, wxString>();
1211 m_count = 0;
1212 m_pads = 0;
1213 }
1214
1215 ~BOM_ENTRY()
1216 {
1217 delete m_refdes;
1218 delete m_props;
1219 }
1220
1221 wxString m_OEMDesignRef; // String combining LIB+FP+VALUE
1222 int m_count;
1223 int m_pads;
1224 wxString m_type;
1225 std::vector<REFDES>* m_refdes;
1226 std::map<wxString, wxString>* m_props;
1227 };
1228
1229 std::set<std::unique_ptr<struct BOM_ENTRY>,
1230 std::function<bool( const std::unique_ptr<struct BOM_ENTRY>&,
1231 const std::unique_ptr<struct BOM_ENTRY>& )>> bom_entries(
1232 []( const std::unique_ptr<struct BOM_ENTRY>& a,
1233 const std::unique_ptr<struct BOM_ENTRY>& b )
1234 {
1235 return a->m_OEMDesignRef < b->m_OEMDesignRef;
1236 } );
1237
1238 for( FOOTPRINT* fp_it : m_board->Footprints() )
1239 {
1240 std::unique_ptr<FOOTPRINT> fp( static_cast<FOOTPRINT*>( fp_it->Clone() ) );
1241 fp->SetParentGroup( nullptr );
1242 fp->SetPosition( {0, 0} );
1243 fp->SetOrientation( ANGLE_0 );
1244
1245 size_t hash = hash_fp_item( fp.get(), HASH_POS | REL_COORD );
1246 auto iter = m_footprint_dict.find( hash );
1247
1248 if( iter == m_footprint_dict.end() )
1249 {
1250 wxLogError( "Footprint %s not found in dictionary",
1251 fp->GetFPID().GetLibItemName().wx_str() );
1252 continue;
1253 }
1254
1255 auto entry = std::make_unique<struct BOM_ENTRY>();
1256
1259 if( auto it = m_OEMRef_dict.find( fp_it ); it != m_OEMRef_dict.end() )
1260 {
1261 entry->m_OEMDesignRef = it->second;
1262 }
1263 else
1264 {
1265 wxLogError( "Footprint %s not found in OEMRef dictionary",
1266 fp->GetFPID().GetLibItemName().wx_str() );
1267 }
1268
1269 entry->m_OEMDesignRef = genString( entry->m_OEMDesignRef, "REF" );
1270 entry->m_count = 1;
1271 entry->m_pads = fp->GetPadCount();
1272
1273 // TODO: The options are "ELECTRICAL", "MECHANICAL", "PROGRAMMABLE", "DOCUMENT", "MATERIAL"
1274 // We need to figure out how to determine this.
1275 if( entry->m_pads == 0 || fp_it->GetAttributes() & FP_EXCLUDE_FROM_BOM )
1276 entry->m_type = "DOCUMENT";
1277 else
1278 entry->m_type = "ELECTRICAL";
1279
1280 auto[ bom_iter, inserted ] = bom_entries.insert( std::move( entry ) );
1281
1282 if( !inserted )
1283 ( *bom_iter )->m_count++;
1284
1285 REFDES refdes;
1286 refdes.m_name = componentName( fp_it );
1287 refdes.m_pkg = fp->GetFPID().GetLibItemName().wx_str();
1288 refdes.m_populate = !fp->IsDNP() && !( fp->GetAttributes() & FP_EXCLUDE_FROM_BOM );
1289 refdes.m_layer = m_layer_name_map[fp_it->GetLayer()];
1290
1291 ( *bom_iter )->m_refdes->push_back( refdes );
1292
1293 // TODO: This amalgamates all the properties from all the footprints. We need to decide
1294 // if we want to group footprints by their properties
1295 for( PCB_FIELD* prop : fp->GetFields() )
1296 {
1297 // We don't need ref, footprint or datasheet in the BOM characteristics. Just value
1298 // and any additional fields the user has added. Ref and footprint are captured above.
1299 if( prop->IsMandatory() && !prop->IsValue() )
1300 continue;
1301
1302 ( *bom_iter )->m_props->emplace( prop->GetName(), prop->GetText() );
1303 }
1304 }
1305
1306 if( bom_entries.empty() )
1307 return nullptr;
1308
1309 wxFileName fn( m_board->GetFileName() );
1310
1311 wxXmlNode* bomNode = new wxXmlNode( wxXML_ELEMENT_NODE, "Bom" );
1312 m_xml_root->InsertChild( bomNode, aEcadNode );
1313 addAttribute( bomNode, "name", genString( fn.GetName(), "BOM" ) );
1314
1315 wxXmlNode* bomHeaderNode = appendNode( bomNode, "BomHeader" );
1316 addAttribute( bomHeaderNode, "revision", "1.0" );
1317 addAttribute( bomHeaderNode, "assembly", genString( fn.GetName() ) );
1318
1319 wxXmlNode* stepRefNode = appendNode( bomHeaderNode, "StepRef" );
1320 addAttribute( stepRefNode, "name", genString( fn.GetName(), "BOARD" ) );
1321
1322 for( const auto& entry : bom_entries )
1323 {
1324 wxXmlNode* bomEntryNode = appendNode( bomNode, "BomItem" );
1325 addAttribute( bomEntryNode, "OEMDesignNumberRef", entry->m_OEMDesignRef );
1326 addAttribute( bomEntryNode, "quantity", wxString::Format( "%d", entry->m_count ) );
1327 addAttribute( bomEntryNode, "pinCount", wxString::Format( "%d", entry->m_pads ) );
1328 addAttribute( bomEntryNode, "category", entry->m_type );
1329
1330 for( const REFDES& refdes : *( entry->m_refdes ) )
1331 {
1332 wxXmlNode* refdesNode = appendNode( bomEntryNode, "RefDes" );
1333 addAttribute( refdesNode, "name", refdes.m_name );
1334 addAttribute( refdesNode, "packageRef", genString( refdes.m_pkg, "PKG" ) );
1335 addAttribute( refdesNode, "populate", refdes.m_populate ? "true" : "false" );
1336 addAttribute( refdesNode, "layerRef", refdes.m_layer );
1337 }
1338
1339 wxXmlNode* characteristicsNode = appendNode( bomEntryNode, "Characteristics" );
1340 addAttribute( characteristicsNode, "category", entry->m_type );
1341
1342 for( const auto& prop : *( entry->m_props ) )
1343 {
1344 wxXmlNode* textualDefNode = appendNode( characteristicsNode, "Textual" );
1345 addAttribute( textualDefNode, "definitionSource", "KICAD" );
1346 addAttribute( textualDefNode, "textualCharacteristicName", prop.first );
1347 addAttribute( textualDefNode, "textualCharacteristicValue", prop.second );
1348 }
1349 }
1350
1351 return bomNode;
1352}
1353
1354
1356{
1357 if( m_progressReporter )
1358 m_progressReporter->AdvancePhase( _( "Generating CAD data" ) );
1359
1360 wxXmlNode* ecadNode = appendNode( m_xml_root, "Ecad" );
1361 addAttribute( ecadNode, "name", "Design" );
1362
1363 addCadHeader( ecadNode );
1364
1365 wxXmlNode* cadDataNode = appendNode( ecadNode, "CadData" );
1366 generateCadLayers( cadDataNode );
1367 generateDrillLayers( cadDataNode );
1368 generateAuxilliaryLayers( cadDataNode );
1369 generateStackup( cadDataNode );
1370 generateStepSection( cadDataNode );
1371
1372 return ecadNode;
1373}
1374
1375
1376void PCB_IO_IPC2581::generateCadSpecs( wxXmlNode* aCadLayerNode )
1377{
1379 BOARD_STACKUP& stackup = dsnSettings.GetStackupDescriptor();
1380 stackup.SynchronizeWithBoard( &dsnSettings );
1381
1382 std::vector<BOARD_STACKUP_ITEM*> layers = stackup.GetList();
1383 std::set<PCB_LAYER_ID> added_layers;
1384
1385 for( int i = 0; i < stackup.GetCount(); i++ )
1386 {
1387 BOARD_STACKUP_ITEM* stackup_item = layers.at( i );
1388
1389 for( int sublayer_id = 0; sublayer_id < stackup_item->GetSublayersCount(); sublayer_id++ )
1390 {
1391 wxString ly_name = stackup_item->GetLayerName();
1392
1393 if( ly_name.IsEmpty() )
1394 {
1395 if( IsValidLayer( stackup_item->GetBrdLayerId() ) )
1396 ly_name = m_board->GetLayerName( stackup_item->GetBrdLayerId() );
1397
1398 if( ly_name.IsEmpty() && stackup_item->GetType() == BS_ITEM_TYPE_DIELECTRIC )
1399 {
1400 ly_name = wxString::Format( "DIELECTRIC_%d", stackup_item->GetDielectricLayerId() );
1401
1402 if( sublayer_id > 0 )
1403 ly_name += wxString::Format( "_%d", sublayer_id );
1404 }
1405 }
1406
1407 ly_name = genString( ly_name, "SPEC_LAYER" );
1408
1409 wxXmlNode* specNode = appendNode( aCadLayerNode, "Spec" );
1410 addAttribute( specNode, "name", ly_name );
1411 wxXmlNode* generalNode = appendNode( specNode, "General" );
1412 addAttribute( generalNode, "type", "MATERIAL" );
1413 wxXmlNode* propertyNode = appendNode( generalNode, "Property" );
1414
1415 switch ( stackup_item->GetType() )
1416 {
1418 {
1419 addAttribute( propertyNode, "text", "COPPER" );
1420 wxXmlNode* conductorNode = appendNode( specNode, "Conductor" );
1421 addAttribute( conductorNode, "type", "CONDUCTIVITY" );
1422 propertyNode = appendNode( conductorNode, "Property" );
1423 addAttribute( propertyNode, "unit", wxT( "SIEMENS/M" ) );
1424 addAttribute( propertyNode, "value", wxT( "5.959E7" ) );
1425 break;
1426 }
1428 {
1429 addAttribute( propertyNode, "text", stackup_item->GetMaterial() );
1430 propertyNode = appendNode( generalNode, "Property" );
1431 addAttribute( propertyNode, "text", wxString::Format( "Type : %s",
1432 stackup_item->GetTypeName() ) );
1433 wxXmlNode* dielectricNode = appendNode( specNode, "Dielectric" );
1434 addAttribute( dielectricNode, "type", "DIELECTRIC_CONSTANT" );
1435 propertyNode = appendNode( dielectricNode, "Property" );
1436 addAttribute( propertyNode, "value",
1437 floatVal( stackup_item->GetEpsilonR( sublayer_id ) ) );
1438 dielectricNode = appendNode( specNode, "Dielectric" );
1439 addAttribute( dielectricNode, "type", "LOSS_TANGENT" );
1440 propertyNode = appendNode( dielectricNode, "Property" );
1441 addAttribute( propertyNode, "value",
1442 floatVal( stackup_item->GetLossTangent( sublayer_id ) ) );
1443 break;
1444 }
1446 addAttribute( propertyNode, "text", stackup_item->GetTypeName() );
1447 propertyNode = appendNode( generalNode, "Property" );
1448 addAttribute( propertyNode, "text", wxString::Format( "Color : %s",
1449 stackup_item->GetColor() ) );
1450 propertyNode = appendNode( generalNode, "Property" );
1451 addAttribute( propertyNode, "text", wxString::Format( "Type : %s",
1452 stackup_item->GetTypeName() ) );
1453 break;
1455 addAttribute( propertyNode, "text", "SOLDERMASK" );
1456 propertyNode = appendNode( generalNode, "Property" );
1457 addAttribute( propertyNode, "text", wxString::Format( "Color : %s",
1458 stackup_item->GetColor() ) );
1459 propertyNode = appendNode( generalNode, "Property" );
1460 addAttribute( propertyNode, "text", wxString::Format( "Type : %s",
1461 stackup_item->GetTypeName() ) );
1462 break;
1463 default:
1464 break;
1465 }
1466 }
1467 }
1468}
1469
1470
1471void PCB_IO_IPC2581::addCadHeader( wxXmlNode* aEcadNode )
1472{
1473 wxXmlNode* cadHeaderNode = appendNode( aEcadNode, "CadHeader" );
1474 addAttribute( cadHeaderNode, "units", m_units_str );
1475
1476 generateCadSpecs( cadHeaderNode );
1477}
1478
1479
1481{
1482 return ( aLayer >= F_Cu && aLayer <= User_9 ) || aLayer == UNDEFINED_LAYER;
1483}
1484
1485
1486void PCB_IO_IPC2581::addLayerAttributes( wxXmlNode* aNode, PCB_LAYER_ID aLayer )
1487{
1488 switch( aLayer )
1489 {
1490 case F_Adhes:
1491 case B_Adhes:
1492 addAttribute( aNode, "layerFunction", "GLUE" );
1493 addAttribute( aNode, "polarity", "POSITIVE" );
1494 addAttribute( aNode, "side", aLayer == F_Adhes ? "TOP" : "BOTTOM" );
1495 break;
1496 case F_Paste:
1497 case B_Paste:
1498 addAttribute( aNode, "layerFunction", "SOLDERPASTE" );
1499 addAttribute( aNode, "polarity", "POSITIVE" );
1500 addAttribute( aNode, "side", aLayer == F_Paste ? "TOP" : "BOTTOM" );
1501 break;
1502 case F_SilkS:
1503 case B_SilkS:
1504 addAttribute( aNode, "layerFunction", "SILKSCREEN" );
1505 addAttribute( aNode, "polarity", "POSITIVE" );
1506 addAttribute( aNode, "side", aLayer == F_SilkS ? "TOP" : "BOTTOM" );
1507 break;
1508 case F_Mask:
1509 case B_Mask:
1510 addAttribute( aNode, "layerFunction", "SOLDERMASK" );
1511 addAttribute( aNode, "polarity", "POSITIVE" );
1512 addAttribute( aNode, "side", aLayer == F_Mask ? "TOP" : "BOTTOM" );
1513 break;
1514 case Edge_Cuts:
1515 addAttribute( aNode, "layerFunction", "BOARD_OUTLINE" );
1516 addAttribute( aNode, "polarity", "POSITIVE" );
1517 addAttribute( aNode, "side", "ALL" );
1518 break;
1519 case B_CrtYd:
1520 case F_CrtYd:
1521 addAttribute( aNode, "layerFunction", "COURTYARD" );
1522 addAttribute( aNode, "polarity", "POSITIVE" );
1523 addAttribute( aNode, "side", aLayer == F_CrtYd ? "TOP" : "BOTTOM" );
1524 break;
1525 case B_Fab:
1526 case F_Fab:
1527 addAttribute( aNode, "layerFunction", "ASSEMBLY" );
1528 addAttribute( aNode, "polarity", "POSITIVE" );
1529 addAttribute( aNode, "side", aLayer == F_Fab ? "TOP" : "BOTTOM" );
1530 break;
1531 case Dwgs_User:
1532 case Cmts_User:
1533 case Eco1_User:
1534 case Eco2_User:
1535 case Margin:
1536 case User_1:
1537 case User_2:
1538 case User_3:
1539 case User_4:
1540 case User_5:
1541 case User_6:
1542 case User_7:
1543 case User_8:
1544 case User_9:
1545 addAttribute( aNode, "layerFunction", "DOCUMENT" );
1546 addAttribute( aNode, "polarity", "POSITIVE" );
1547 addAttribute( aNode, "side", "NONE" );
1548 break;
1549
1550 default:
1551 if( IsCopperLayer( aLayer ) )
1552 {
1553 addAttribute( aNode, "layerFunction", "CONDUCTOR" );
1554 addAttribute( aNode, "polarity", "POSITIVE" );
1555 addAttribute( aNode, "side",
1556 aLayer == F_Cu ? "TOP"
1557 : aLayer == B_Cu ? "BOTTOM"
1558 : "INTERNAL" );
1559 }
1560
1561 break; // Do not handle other layers
1562 }
1563}
1564
1565
1566void PCB_IO_IPC2581::generateStackup( wxXmlNode* aCadLayerNode )
1567{
1569 BOARD_STACKUP& stackup = dsnSettings.GetStackupDescriptor();
1570 stackup.SynchronizeWithBoard( &dsnSettings );
1571
1572 wxXmlNode* stackupNode = appendNode( aCadLayerNode, "Stackup" );
1573 addAttribute( stackupNode, "name", "Primary_Stackup" );
1574 addAttribute( stackupNode, "overallThickness", floatVal( m_scale * stackup.BuildBoardThicknessFromStackup() ) );
1575 addAttribute( stackupNode, "tolPlus", "0.0" );
1576 addAttribute( stackupNode, "tolMinus", "0.0" );
1577 addAttribute( stackupNode, "whereMeasured", "MASK" );
1578
1579 if( m_version > 'B' )
1580 addAttribute( stackupNode, "stackupStatus", "PROPOSED" );
1581
1582 wxXmlNode* stackupGroup = appendNode( stackupNode, "StackupGroup" );
1583 addAttribute( stackupGroup, "name", "Primary_Stackup_Group" );
1584 addAttribute( stackupGroup, "thickness", floatVal( m_scale * stackup.BuildBoardThicknessFromStackup() ) );
1585 addAttribute( stackupGroup, "tolPlus", "0.0" );
1586 addAttribute( stackupGroup, "tolMinus", "0.0" );
1587
1588 std::vector<BOARD_STACKUP_ITEM*> layers = stackup.GetList();
1589 std::set<PCB_LAYER_ID> added_layers;
1590
1591 for( int i = 0; i < stackup.GetCount(); i++ )
1592 {
1593 BOARD_STACKUP_ITEM* stackup_item = layers.at( i );
1594
1595 for( int sublayer_id = 0; sublayer_id < stackup_item->GetSublayersCount(); sublayer_id++ )
1596 {
1597
1598 wxXmlNode* stackupLayer = appendNode( stackupGroup, "StackupLayer" );
1599 wxString ly_name = stackup_item->GetLayerName();
1600
1601 if( ly_name.IsEmpty() )
1602 {
1603 if( IsValidLayer( stackup_item->GetBrdLayerId() ) )
1604 ly_name = m_board->GetLayerName( stackup_item->GetBrdLayerId() );
1605
1606 if( ly_name.IsEmpty() && stackup_item->GetType() == BS_ITEM_TYPE_DIELECTRIC )
1607 {
1608 ly_name = wxString::Format( "DIELECTRIC_%d", stackup_item->GetDielectricLayerId() );
1609
1610 if( sublayer_id > 0 )
1611 ly_name += wxString::Format( "_%d", sublayer_id );
1612 }
1613 }
1614
1615 ly_name = genString( ly_name, "LAYER" );
1616
1617 addAttribute( stackupLayer, "layerOrGroupRef", ly_name );
1618 addAttribute( stackupLayer, "thickness", floatVal( m_scale * stackup_item->GetThickness() ) );
1619 addAttribute( stackupLayer, "tolPlus", "0.0" );
1620 addAttribute( stackupLayer, "tolMinus", "0.0" );
1621 addAttribute( stackupLayer, "sequence", wxString::Format( "%d", i ) );
1622
1623 wxXmlNode* specLayerNode = appendNode( stackupLayer, "SpecRef" );
1624 addAttribute( specLayerNode, "id", wxString::Format( "SPEC_%s", ly_name ) );
1625 }
1626 }
1627}
1628
1629
1630void PCB_IO_IPC2581::generateCadLayers( wxXmlNode* aCadLayerNode )
1631{
1632
1634 BOARD_STACKUP& stackup = dsnSettings.GetStackupDescriptor();
1635 stackup.SynchronizeWithBoard( &dsnSettings );
1636
1637 std::vector<BOARD_STACKUP_ITEM*> layers = stackup.GetList();
1638 std::set<PCB_LAYER_ID> added_layers;
1639
1640 for( int i = 0; i < stackup.GetCount(); i++ )
1641 {
1642 BOARD_STACKUP_ITEM* stackup_item = layers.at( i );
1643
1644 if( !isValidLayerFor2581( stackup_item->GetBrdLayerId() ) )
1645 continue;
1646
1647 for( int sublayer_id = 0; sublayer_id < stackup_item->GetSublayersCount(); sublayer_id++ )
1648 {
1649 wxXmlNode* cadLayerNode = appendNode( aCadLayerNode, "Layer" );
1650 wxString ly_name = stackup_item->GetLayerName();
1651
1652 if( ly_name.IsEmpty() )
1653 {
1654
1655 if( IsValidLayer( stackup_item->GetBrdLayerId() ) )
1656 ly_name = m_board->GetLayerName( stackup_item->GetBrdLayerId() );
1657
1658 if( ly_name.IsEmpty() && stackup_item->GetType() == BS_ITEM_TYPE_DIELECTRIC )
1659 {
1660 ly_name = wxString::Format( "DIELECTRIC_%d", stackup_item->GetDielectricLayerId() );
1661
1662 if( sublayer_id > 0 )
1663 ly_name += wxString::Format( "_%d", sublayer_id );
1664 }
1665 }
1666
1667 ly_name = genString( ly_name, "LAYER" );
1668
1669 addAttribute( cadLayerNode, "name", ly_name );
1670
1671 if( stackup_item->GetType() == BS_ITEM_TYPE_DIELECTRIC )
1672 {
1673 if( stackup_item->GetTypeName() == KEY_CORE )
1674 addAttribute( cadLayerNode, "layerFunction", "DIELCORE" );
1675 else
1676 addAttribute( cadLayerNode, "layerFunction", "DIELPREG" );
1677
1678 addAttribute( cadLayerNode, "polarity", "POSITIVE" );
1679 addAttribute( cadLayerNode, "side", "INTERNAL" );
1680 continue;
1681 }
1682 else
1683 {
1684 added_layers.insert( stackup_item->GetBrdLayerId() );
1685 addLayerAttributes( cadLayerNode, stackup_item->GetBrdLayerId() );
1686 m_layer_name_map.emplace( stackup_item->GetBrdLayerId(), ly_name );
1687 }
1688 }
1689 }
1690
1691 LSEQ layer_seq = m_board->GetEnabledLayers().Seq();
1692
1693 for( PCB_LAYER_ID layer : layer_seq )
1694 {
1695 if( added_layers.find( layer ) != added_layers.end() || !isValidLayerFor2581( layer ) )
1696 continue;
1697
1698 wxString ly_name = genLayerString( layer, "LAYER" );
1699 m_layer_name_map.emplace( layer, ly_name );
1700 added_layers.insert( layer );
1701 wxXmlNode* cadLayerNode = appendNode( aCadLayerNode, "Layer" );
1702 addAttribute( cadLayerNode, "name", ly_name );
1703
1704 addLayerAttributes( cadLayerNode, layer );
1705 }
1706}
1707
1708
1709void PCB_IO_IPC2581::generateDrillLayers( wxXmlNode* aCadLayerNode )
1710{
1711 for( BOARD_ITEM* item : m_board->Tracks() )
1712 {
1713 if( item->Type() == PCB_VIA_T )
1714 {
1715 PCB_VIA* via = static_cast<PCB_VIA*>( item );
1716 m_drill_layers[std::make_pair( via->TopLayer(), via->BottomLayer() )].push_back( via );
1717 }
1718 }
1719
1720 for( FOOTPRINT* fp : m_board->Footprints() )
1721 {
1722 for( PAD* pad : fp->Pads() )
1723 {
1724 if( pad->HasDrilledHole() )
1725 m_drill_layers[std::make_pair( F_Cu, B_Cu )].push_back( pad );
1726 else if( pad->HasHole() )
1727 m_slot_holes[std::make_pair( F_Cu, B_Cu )].push_back( pad );
1728 }
1729 }
1730
1731 for( const auto& [layers, vec] : m_drill_layers )
1732 {
1733 wxXmlNode* drillNode = appendNode( aCadLayerNode, "Layer" );
1734 drillNode->AddAttribute( "name", genLayersString( layers.first, layers.second, "DRILL" ) );
1735 addAttribute( drillNode, "layerFunction", "DRILL" );
1736 addAttribute( drillNode, "polarity", "POSITIVE" );
1737 addAttribute( drillNode, "side", "ALL" );
1738
1739 wxXmlNode* spanNode = appendNode( drillNode, "Span" );
1740 addAttribute( spanNode, "fromLayer", genLayerString( layers.first, "LAYER" ) );
1741 addAttribute( spanNode, "toLayer", genLayerString( layers.second, "LAYER" ) );
1742 }
1743
1744 for( const auto& [layers, vec] : m_slot_holes )
1745 {
1746 wxXmlNode* drillNode = appendNode( aCadLayerNode, "Layer" );
1747 drillNode->AddAttribute( "name", genLayersString( layers.first, layers.second, "SLOT" ) );
1748
1749 addAttribute( drillNode, "layerFunction", "ROUT" );
1750 addAttribute( drillNode, "polarity", "POSITIVE" );
1751 addAttribute( drillNode, "side", "ALL" );
1752
1753 wxXmlNode* spanNode = appendNode( drillNode, "Span" );
1754 addAttribute( spanNode, "fromLayer", genLayerString( layers.first, "LAYER" ) );
1755 addAttribute( spanNode, "toLayer", genLayerString( layers.second, "LAYER" ) );
1756 }
1757}
1758
1759
1760void PCB_IO_IPC2581::generateAuxilliaryLayers( wxXmlNode* aCadLayerNode )
1761{
1762 for( BOARD_ITEM* item : m_board->Tracks() )
1763 {
1764 if( item->Type() != PCB_VIA_T )
1765 continue;
1766
1767 PCB_VIA* via = static_cast<PCB_VIA*>( item );
1768
1769 std::vector<std::tuple<auxLayerType, PCB_LAYER_ID, PCB_LAYER_ID>> new_layers;
1770
1771 if( via->Padstack().IsFilled().value_or( false ) )
1772 {
1773 new_layers.emplace_back( auxLayerType::FILLING, via->TopLayer(), via->BottomLayer() );
1774 }
1775
1776 if( via->Padstack().IsCapped().value_or( false ) )
1777 {
1778 new_layers.emplace_back( auxLayerType::CAPPING, via->TopLayer(), via->BottomLayer() );
1779 }
1780
1781 for( PCB_LAYER_ID layer : { via->TopLayer(), via->BottomLayer() } )
1782 {
1783 if( via->Padstack().IsPlugged( layer ).value_or( false ) )
1784 {
1785 new_layers.emplace_back( auxLayerType::PLUGGING, layer, UNDEFINED_LAYER );
1786 }
1787
1788 if( via->Padstack().IsCovered( layer ).value_or( false ) )
1789 {
1790 new_layers.emplace_back( auxLayerType::COVERING, layer, UNDEFINED_LAYER );
1791 }
1792
1793 if( via->Padstack().IsTented( layer ).value_or( false ) )
1794 {
1795 new_layers.emplace_back( auxLayerType::TENTING, layer, UNDEFINED_LAYER );
1796 }
1797 }
1798
1799 for( auto& tuple : new_layers )
1800 {
1801 m_auxilliary_Layers[tuple].push_back( via );
1802 }
1803 }
1804
1805 for( const auto& [layers, vec] : m_auxilliary_Layers )
1806 {
1807 bool add_node = true;
1808
1809 wxString name;
1810 wxString layerFunction;
1811
1812 // clang-format off: suggestion is inconsitent
1813 switch( std::get<0>(layers) )
1814 {
1815 case auxLayerType::COVERING:
1816 name = "COVERING";
1817 layerFunction = "COATINGNONCOND";
1818 break;
1819 case auxLayerType::PLUGGING:
1820 name = "PLUGGING";
1821 layerFunction = "HOLEFILL";
1822 break;
1823 case auxLayerType::TENTING:
1824 name = "TENTING";
1825 layerFunction = "COATINGNONCOND";
1826 break;
1827 case auxLayerType::FILLING:
1828 name = "FILLING";
1829 layerFunction = "HOLEFILL";
1830 break;
1831 case auxLayerType::CAPPING:
1832 name = "CAPPING";
1833 layerFunction = "COATINGCOND";
1834 break;
1835 default:
1836 add_node = false;
1837 break;
1838 }
1839 // clang-format on: suggestion is inconsitent
1840
1841 if( add_node && !vec.empty() )
1842 {
1843 wxXmlNode* node = appendNode( aCadLayerNode, "LAYER" );
1844 addAttribute( node, "layerFunction", layerFunction );
1845 addAttribute( node, "polarity", "POSITIVE" );
1846
1847 if( std::get<2>( layers ) == UNDEFINED_LAYER )
1848 {
1849 addAttribute( node, "name", genLayerString( std::get<1>( layers ), TO_UTF8( name ) ) );
1850 addAttribute( node, "side",
1851 IsFrontLayer( std::get<1>( layers ) ) ? "TOP" : "BOTTOM" );
1852 }
1853 else
1854 {
1856 node, "name",
1857 genLayersString( std::get<1>( layers ), std::get<2>( layers ), TO_UTF8( name ) ) );
1858
1859 const bool first_external =
1860 std::get<1>( layers ) == F_Cu || std::get<1>( layers ) == B_Cu;
1861 const bool second_external =
1862 std::get<2>( layers ) == F_Cu || std::get<2>( layers ) == B_Cu;
1863
1864 if( first_external )
1865 {
1866 if( second_external )
1867 addAttribute( node, "side", "ALL" );
1868 else
1869 addAttribute( node, "side", "FRONT" );
1870 }
1871 else
1872 {
1873 if( second_external )
1874 addAttribute( node, "side", "BACK" );
1875 else
1876 addAttribute( node, "side", "INTERNAL" );
1877 }
1878
1879 wxXmlNode* spanNode = appendNode( node, "SPAN" );
1880 addAttribute( spanNode, "fromLayer",
1881 genLayerString( std::get<1>( layers ), "LAYER" ) );
1882 addAttribute( spanNode, "toLayer",
1883 genLayerString( std::get<2>( layers ), "LAYER" ) );
1884 }
1885 }
1886 }
1887}
1888
1889
1890void PCB_IO_IPC2581::generateStepSection( wxXmlNode* aCadNode )
1891{
1892 wxXmlNode* stepNode = appendNode( aCadNode, "Step" );
1893 wxFileName fn( m_board->GetFileName() );
1894 addAttribute( stepNode, "name", genString( fn.GetName(), "BOARD" ) );
1895
1896 if( m_version > 'B' )
1897 addAttribute( stepNode, "type", "BOARD" );
1898
1899 wxXmlNode* datumNode = appendNode( stepNode, "Datum" );
1900 addAttribute( datumNode, "x", "0.0" );
1901 addAttribute( datumNode, "y", "0.0" );
1902
1903 generateProfile( stepNode );
1904 generateComponents( stepNode );
1905
1906 m_last_padstack = insertNode( stepNode, "NonstandardAttribute" );
1907 addAttribute( m_last_padstack, "name", "FOOTPRINT_COUNT" );
1908 addAttribute( m_last_padstack, "type", "INTEGER" );
1909 addAttribute( m_last_padstack, "value",
1910 wxString::Format( "%zu", m_board->Footprints().size() ) );
1911
1912 generateLayerFeatures( stepNode );
1913 generateLayerSetDrill( stepNode );
1914 generateLayerSetAuxilliary( stepNode );
1915}
1916
1917
1918void PCB_IO_IPC2581::addPad( wxXmlNode* aContentNode, const PAD* aPad, PCB_LAYER_ID aLayer )
1919{
1920 wxXmlNode* padNode = appendNode( aContentNode, "Pad" );
1921 FOOTPRINT* fp = aPad->GetParentFootprint();
1922
1923 addPadStack( padNode, aPad );
1924
1925 if( aPad->GetOrientation() != ANGLE_0 )
1926 {
1927 wxXmlNode* xformNode = appendNode( padNode, "Xform" );
1928 EDA_ANGLE angle = aPad->GetOrientation().Normalize();
1929
1930 xformNode->AddAttribute( "rotation", floatVal( angle.AsDegrees() ) );
1931 }
1932
1933 addLocationNode( padNode, *aPad, false );
1934 addShape( padNode, *aPad, aLayer );
1935
1936 if( fp )
1937 {
1938 wxXmlNode* pinRefNode = appendNode( padNode, "PinRef" );
1939
1940 addAttribute( pinRefNode, "componentRef", componentName( fp ) );
1941 addAttribute( pinRefNode, "pin", pinName( aPad ) );
1942 }
1943}
1944
1945
1946void PCB_IO_IPC2581::addVia( wxXmlNode* aContentNode, const PCB_VIA* aVia, PCB_LAYER_ID aLayer )
1947{
1948 if( !aVia->FlashLayer( aLayer ) )
1949 return;
1950
1951 wxXmlNode* padNode = appendNode( aContentNode, "Pad" );
1952
1953 addPadStack( padNode, aVia );
1954 addLocationNode( padNode, aVia->GetPosition().x, aVia->GetPosition().y );
1955
1956 PAD dummy( nullptr );
1957 int hole = aVia->GetDrillValue();
1958 dummy.SetDrillSize( VECTOR2I( hole, hole ) );
1959 dummy.SetPosition( aVia->GetStart() );
1960 dummy.SetSize( aLayer, VECTOR2I( aVia->GetWidth( aLayer ), aVia->GetWidth( aLayer ) ) );
1961
1962 addShape( padNode, dummy, aLayer );
1963}
1964
1965
1966void PCB_IO_IPC2581::addPadStack( wxXmlNode* aPadNode, const PAD* aPad )
1967{
1968 size_t hash = hash_fp_item( aPad, 0 );
1969 wxString name = wxString::Format( "PADSTACK_%zu", m_padstack_dict.size() + 1 );
1970 auto [ th_pair, success ] = m_padstack_dict.emplace( hash, name );
1971
1972 addAttribute( aPadNode, "padstackDefRef", th_pair->second );
1973
1974 // If we did not insert a new padstack, then we have already added it to the XML
1975 // and we don't need to add it again.
1976 if( !success )
1977 return;
1978
1979 wxXmlNode* padStackDefNode = new wxXmlNode( wxXML_ELEMENT_NODE, "PadStackDef" );
1980 addAttribute( padStackDefNode, "name", name );
1981 m_padstacks.push_back( padStackDefNode );
1982
1983 if( m_last_padstack )
1984 {
1985 insertNodeAfter( m_last_padstack, padStackDefNode );
1986 m_last_padstack = padStackDefNode;
1987 }
1988
1989 // Only handle round holes here because IPC2581 does not support non-round holes
1990 // These will be handled in a slot layer
1991 if( aPad->HasDrilledHole() )
1992 {
1993 wxXmlNode* padStackHoleNode = appendNode( padStackDefNode, "PadstackHoleDef" );
1994 padStackHoleNode->AddAttribute( "name",
1995 wxString::Format( "%s%d_%d",
1996 aPad->GetAttribute() == PAD_ATTRIB::PTH ? "PTH" : "NPTH",
1997 aPad->GetDrillSizeX(), aPad->GetDrillSizeY() ) );
1998
1999 addAttribute( padStackHoleNode, "diameter", floatVal( m_scale * aPad->GetDrillSizeX() ) );
2000 addAttribute( padStackHoleNode, "platingStatus",
2001 aPad->GetAttribute() == PAD_ATTRIB::PTH ? "PLATED" : "NONPLATED" );
2002 addAttribute( padStackHoleNode, "plusTol", "0.0" );
2003 addAttribute( padStackHoleNode, "minusTol", "0.0" );
2004 addXY( padStackHoleNode, aPad->GetOffset( PADSTACK::ALL_LAYERS ) );
2005 }
2006
2007 LSEQ layer_seq = aPad->GetLayerSet().Seq();
2008
2009 for( PCB_LAYER_ID layer : layer_seq )
2010 {
2011 FOOTPRINT* fp = aPad->GetParentFootprint();
2012
2013 if( !m_board->IsLayerEnabled( layer ) )
2014 continue;
2015
2016 wxXmlNode* padStackPadDefNode = appendNode( padStackDefNode, "PadstackPadDef" );
2017 addAttribute( padStackPadDefNode, "layerRef", m_layer_name_map[layer] );
2018 addAttribute( padStackPadDefNode, "padUse", "REGULAR" );
2019 addLocationNode( padStackPadDefNode, aPad->GetOffset( PADSTACK::ALL_LAYERS ).x, aPad->GetOffset( PADSTACK::ALL_LAYERS ).y );
2020
2021 if( aPad->HasHole() || !aPad->FlashLayer( layer ) )
2022 {
2023 PCB_SHAPE shape( nullptr, SHAPE_T::CIRCLE );
2024 shape.SetStart( aPad->GetOffset( PADSTACK::ALL_LAYERS ) );
2025 shape.SetEnd( shape.GetStart() + aPad->GetDrillSize() / 2 );
2026 addShape( padStackPadDefNode, shape );
2027 }
2028 else
2029 {
2030 addShape( padStackPadDefNode, *aPad, layer );
2031 }
2032 }
2033}
2034
2035
2036void PCB_IO_IPC2581::addPadStack( wxXmlNode* aContentNode, const PCB_VIA* aVia )
2037{
2038 size_t hash = hash_fp_item( aVia, 0 );
2039 wxString name = wxString::Format( "PADSTACK_%zu", m_padstack_dict.size() + 1 );
2040 auto [ via_pair, success ] = m_padstack_dict.emplace( hash, name );
2041
2042 addAttribute( aContentNode, "padstackDefRef", via_pair->second );
2043
2044 // If we did not insert a new padstack, then we have already added it to the XML
2045 // and we don't need to add it again.
2046 if( !success )
2047 return;
2048
2049 wxXmlNode* padStackDefNode = new wxXmlNode( wxXML_ELEMENT_NODE, "PadStackDef" );
2050 insertNodeAfter( m_last_padstack, padStackDefNode );
2051 m_last_padstack = padStackDefNode;
2052 addAttribute( padStackDefNode, "name", name );
2053
2054 wxXmlNode* padStackHoleNode = appendNode( padStackDefNode, "PadstackHoleDef" );
2055 addAttribute( padStackHoleNode, "name", wxString::Format( "PH%d", aVia->GetDrillValue() ) );
2056 padStackHoleNode->AddAttribute( "diameter", floatVal( m_scale * aVia->GetDrillValue() ) );
2057 addAttribute( padStackHoleNode, "platingStatus", "VIA" );
2058 addAttribute( padStackHoleNode, "plusTol", "0.0" );
2059 addAttribute( padStackHoleNode, "minusTol", "0.0" );
2060 addAttribute( padStackHoleNode, "x", "0.0" );
2061 addAttribute( padStackHoleNode, "y", "0.0" );
2062
2063 LSEQ layer_seq = aVia->GetLayerSet().Seq();
2064
2065 auto addPadShape{ [&]( PCB_LAYER_ID layer, const PCB_VIA* aVia, const wxString& name,
2066 bool drill ) -> void
2067 {
2068 PCB_SHAPE shape( nullptr, SHAPE_T::CIRCLE );
2069
2070 if( drill )
2071 shape.SetEnd( { KiROUND( aVia->GetDrillValue() / 2.0 ), 0 } );
2072 else
2073 shape.SetEnd( { KiROUND( aVia->GetWidth( layer ) / 2.0 ), 0 } );
2074
2075 wxXmlNode* padStackPadDefNode =
2076 appendNode( padStackDefNode, "PadstackPadDef" );
2077 addAttribute( padStackPadDefNode, "layerRef", name );
2078 addAttribute( padStackPadDefNode, "padUse", "REGULAR" );
2079
2080 addLocationNode( padStackPadDefNode, 0.0, 0.0 );
2081 addShape( padStackPadDefNode, shape );
2082 } };
2083
2084 for( PCB_LAYER_ID layer : layer_seq )
2085 {
2086 if( !aVia->FlashLayer( layer ) || !m_board->IsLayerEnabled( layer ) )
2087 continue;
2088
2089 addPadShape( layer, aVia, m_layer_name_map[layer], false );
2090 }
2091
2092 if( aVia->Padstack().IsFilled().value_or( false ) )
2093 addPadShape( UNDEFINED_LAYER, aVia,
2094 genLayersString( aVia->TopLayer(), aVia->BottomLayer(), "FILLING" ), true );
2095
2096 if( aVia->Padstack().IsCapped().value_or( false ) )
2097 addPadShape( UNDEFINED_LAYER, aVia,
2098 genLayersString( aVia->TopLayer(), aVia->BottomLayer(), "CAPPING" ), true );
2099
2100 for( PCB_LAYER_ID layer : { aVia->TopLayer(), aVia->BottomLayer() } )
2101 {
2102 if( aVia->Padstack().IsPlugged( layer ).value_or( false ) )
2103 addPadShape( layer, aVia, genLayerString( layer, "PLUGGING" ), true );
2104
2105 if( aVia->Padstack().IsCovered( layer ).value_or( false ) )
2106 addPadShape( layer, aVia, genLayerString( layer, "COVERING" ), false );
2107
2108 if( aVia->Padstack().IsTented( layer ).value_or( false ) )
2109 addPadShape( layer, aVia, genLayerString( layer, "TENTING" ), false );
2110 }
2111}
2112
2113
2114bool PCB_IO_IPC2581::addPolygonNode( wxXmlNode* aParentNode,
2115 const SHAPE_LINE_CHAIN& aPolygon, FILL_T aFillType,
2116 int aWidth, LINE_STYLE aDashType )
2117{
2118 wxXmlNode* polygonNode = nullptr;
2119
2120 if( aPolygon.PointCount() < 3 )
2121 return false;
2122
2123 auto make_node =
2124 [&]()
2125 {
2126 polygonNode = appendNode( aParentNode, "Polygon" );
2127 wxXmlNode* polybeginNode = appendNode( polygonNode, "PolyBegin" );
2128
2129 const std::vector<VECTOR2I>& pts = aPolygon.CPoints();
2130 addXY( polybeginNode, pts[0] );
2131
2132 for( size_t ii = 1; ii < pts.size(); ++ii )
2133 {
2134 wxXmlNode* polyNode = appendNode( polygonNode, "PolyStepSegment" );
2135 addXY( polyNode, pts[ii] );
2136 }
2137
2138 wxXmlNode* polyendNode = appendNode( polygonNode, "PolyStepSegment" );
2139 addXY( polyendNode, pts[0] );
2140 };
2141
2142 // Allow the case where we don't want line/fill information in the polygon
2143 if( aFillType == FILL_T::NO_FILL )
2144 {
2145 make_node();
2146 // If we specify a line width, we need to add a LineDescRef node and
2147 // since this is only valid for a non-filled polygon, we need to create
2148 // the fillNode as well
2149 if( aWidth > 0 )
2150 addLineDesc( polygonNode, aWidth, aDashType, true );
2151 }
2152 else
2153 {
2154 wxCHECK( aWidth == 0, false );
2155 make_node();
2156 }
2157
2158 addFillDesc( polygonNode, aFillType );
2159
2160 return true;
2161}
2162
2163
2164bool PCB_IO_IPC2581::addPolygonCutouts( wxXmlNode* aParentNode,
2165 const SHAPE_POLY_SET::POLYGON& aPolygon )
2166{
2167 for( size_t ii = 1; ii < aPolygon.size(); ++ii )
2168 {
2169 wxCHECK2( aPolygon[ii].PointCount() >= 3, continue );
2170
2171 wxXmlNode* cutoutNode = appendNode( aParentNode, "Cutout" );
2172 wxXmlNode* polybeginNode = appendNode( cutoutNode, "PolyBegin" );
2173
2174 const std::vector<VECTOR2I>& hole = aPolygon[ii].CPoints();
2175 addXY( polybeginNode, hole[0] );
2176
2177 for( size_t jj = 1; jj < hole.size(); ++jj )
2178 {
2179 wxXmlNode* polyNode = appendNode( cutoutNode, "PolyStepSegment" );
2180 addXY( polyNode, hole[jj] );
2181 }
2182
2183 wxXmlNode* polyendNode = appendNode( cutoutNode, "PolyStepSegment" );
2184 addXY( polyendNode, hole[0] );
2185 }
2186
2187 return true;
2188}
2189
2190
2191bool PCB_IO_IPC2581::addOutlineNode( wxXmlNode* aParentNode, const SHAPE_POLY_SET& aPolySet,
2192 int aWidth, LINE_STYLE aDashType )
2193{
2194 if( aPolySet.OutlineCount() == 0 )
2195 return false;
2196
2197 wxXmlNode* outlineNode = appendNode( aParentNode, "Outline" );
2198
2199 // Outlines can only have one polygon according to the IPC-2581 spec, so
2200 // if there are more than one, we need to combine them into a single polygon
2201 const SHAPE_LINE_CHAIN* outline = &aPolySet.Outline( 0 );
2202 SHAPE_LINE_CHAIN bbox_outline;
2203 BOX2I bbox = outline->BBox();
2204
2205 if( aPolySet.OutlineCount() > 1 )
2206 {
2207 for( int ii = 1; ii < aPolySet.OutlineCount(); ++ii )
2208 {
2209 wxCHECK2( aPolySet.Outline( ii ).PointCount() >= 3, continue );
2210 bbox.Merge( aPolySet.Outline( ii ).BBox() );
2211 }
2212
2213 bbox_outline.Append( bbox.GetLeft(), bbox.GetTop() );
2214 bbox_outline.Append( bbox.GetRight(), bbox.GetTop() );
2215 bbox_outline.Append( bbox.GetRight(), bbox.GetBottom() );
2216 bbox_outline.Append( bbox.GetLeft(), bbox.GetBottom() );
2217 outline = &bbox_outline;
2218 }
2219
2220
2221 if( !addPolygonNode( outlineNode, *outline ) )
2222 wxLogTrace( traceIpc2581, wxS( "Failed to add polygon to outline" ) );
2223
2224 if( !outlineNode->GetChildren() )
2225 {
2226 aParentNode->RemoveChild( outlineNode );
2227 delete outlineNode;
2228 return false;
2229 }
2230
2231 addLineDesc( outlineNode, aWidth, aDashType );
2232
2233 return true;
2234}
2235
2236
2237bool PCB_IO_IPC2581::addContourNode( wxXmlNode* aParentNode, const SHAPE_POLY_SET& aPolySet,
2238 int aOutline, FILL_T aFillType, int aWidth,
2239 LINE_STYLE aDashType )
2240{
2241 if( aPolySet.OutlineCount() < ( aOutline + 1 ) )
2242 return false;
2243
2244 wxXmlNode* contourNode = appendNode( aParentNode, "Contour" );
2245
2246 if( addPolygonNode( contourNode, aPolySet.Outline( aOutline ), aFillType, aWidth, aDashType ) )
2247 {
2248 // Do not attempt to add cutouts to shapes that are already hollow
2249 if( aFillType != FILL_T::NO_FILL )
2250 addPolygonCutouts( contourNode, aPolySet.Polygon( aOutline ) );
2251 }
2252 else
2253 {
2254 aParentNode->RemoveChild( contourNode );
2255 delete contourNode;
2256 return false;
2257 }
2258
2259 return true;
2260}
2261
2262
2263void PCB_IO_IPC2581::generateProfile( wxXmlNode* aStepNode )
2264{
2265 SHAPE_POLY_SET board_outline;
2266
2267 if( ! m_board->GetBoardPolygonOutlines( board_outline ) )
2268 {
2269 wxLogError( "Failed to get board outline" );
2270 return;
2271 }
2272
2273 wxXmlNode* profileNode = appendNode( aStepNode, "Profile" );
2274
2275 if( !addPolygonNode( profileNode, board_outline.Outline( 0 ) ) )
2276 {
2277 wxLogTrace( traceIpc2581, wxS( "Failed to add polygon to profile" ) );
2278 aStepNode->RemoveChild( profileNode );
2279 delete profileNode;
2280 }
2281}
2282
2283
2284wxXmlNode* PCB_IO_IPC2581::addPackage( wxXmlNode* aContentNode, FOOTPRINT* aFp )
2285{
2286 std::unique_ptr<FOOTPRINT> fp( static_cast<FOOTPRINT*>( aFp->Clone() ) );
2287 fp->SetParentGroup( nullptr );
2288 fp->SetPosition( { 0, 0 } );
2289 fp->SetOrientation( ANGLE_0 );
2290
2291 size_t hash = hash_fp_item( fp.get(), HASH_POS | REL_COORD );
2292 wxString name = genString( wxString::Format( "%s_%zu",
2293 fp->GetFPID().GetLibItemName().wx_str(),
2294 m_footprint_dict.size() + 1 ) );
2295
2296 auto [ iter, success ] = m_footprint_dict.emplace( hash, name );
2297 addAttribute( aContentNode, "packageRef", iter->second );
2298
2299 if( !success)
2300 return nullptr;
2301
2302 // Package and Component nodes are at the same level, so we need to find the parent
2303 // which should be the Step node
2304 wxXmlNode* packageNode = new wxXmlNode( wxXML_ELEMENT_NODE, "Package" );
2305 wxXmlNode* otherSideViewNode = nullptr; // Only set this if we have elements on the back side
2306
2307 addAttribute( packageNode, "name", name );
2308 addAttribute( packageNode, "type", "OTHER" ); // TODO: Replace with actual package type once we encode this
2309
2310 // We don't specially identify pin 1 in our footprints, so we need to guess
2311 if( fp->FindPadByNumber( "1" ) )
2312 addAttribute( packageNode, "pinOne", "1" );
2313 else if ( fp->FindPadByNumber( "A1" ) )
2314 addAttribute( packageNode, "pinOne", "A1" );
2315 else if ( fp->FindPadByNumber( "A" ) )
2316 addAttribute( packageNode, "pinOne", "A" );
2317 else if ( fp->FindPadByNumber( "a" ) )
2318 addAttribute( packageNode, "pinOne", "a" );
2319 else if ( fp->FindPadByNumber( "a1" ) )
2320 addAttribute( packageNode, "pinOne", "a1" );
2321 else if ( fp->FindPadByNumber( "Anode" ) )
2322 addAttribute( packageNode, "pinOne", "Anode" );
2323 else if ( fp->FindPadByNumber( "ANODE" ) )
2324 addAttribute( packageNode, "pinOne", "ANODE" );
2325 else
2326 addAttribute( packageNode, "pinOne", "UNKNOWN" );
2327
2328 addAttribute( packageNode, "pinOneOrientation", "OTHER" );
2329
2330 const SHAPE_POLY_SET& courtyard = fp->GetCourtyard( F_CrtYd );
2331 const SHAPE_POLY_SET& courtyard_back = fp->GetCourtyard( B_CrtYd );
2332
2333 if( courtyard.OutlineCount() > 0 )
2334 addOutlineNode( packageNode, courtyard, courtyard.Outline( 0 ).Width(), LINE_STYLE::SOLID );
2335
2336 if( courtyard_back.OutlineCount() > 0 )
2337 {
2338 otherSideViewNode = appendNode( packageNode, "OtherSideView" );
2339 addOutlineNode( otherSideViewNode, courtyard_back, courtyard_back.Outline( 0 ).Width(),
2340 LINE_STYLE::SOLID );
2341 }
2342
2343 if( !courtyard.OutlineCount() && !courtyard_back.OutlineCount() )
2344 {
2345 SHAPE_POLY_SET bbox = fp->GetBoundingHull();
2346 addOutlineNode( packageNode, bbox );
2347 }
2348
2349 wxXmlNode* pickupPointNode = appendNode( packageNode, "PickupPoint" );
2350 addAttribute( pickupPointNode, "x", "0.0" );
2351 addAttribute( pickupPointNode, "y", "0.0" );
2352
2353 std::map<PCB_LAYER_ID, std::map<bool, std::vector<BOARD_ITEM*>>> elements;
2354
2355 for( BOARD_ITEM* item : fp->GraphicalItems() )
2356 {
2357 PCB_LAYER_ID layer = item->GetLayer();
2358
2362 if( layer != F_SilkS && layer != B_SilkS && layer != F_Fab && layer != B_Fab )
2363 continue;
2364
2365 bool is_abs = true;
2366
2367 if( item->Type() == PCB_SHAPE_T )
2368 {
2369 PCB_SHAPE* shape = static_cast<PCB_SHAPE*>( item );
2370
2371 // Circles and Rectanges only have size information so we need to place them in
2372 // a separate node that has a location
2373 if( shape->GetShape() == SHAPE_T::CIRCLE || shape->GetShape() == SHAPE_T::RECTANGLE )
2374 is_abs = false;
2375 }
2376
2377 elements[item->GetLayer()][is_abs].push_back( item );
2378 }
2379
2380 auto add_base_node =
2381 [&]( PCB_LAYER_ID aLayer ) -> wxXmlNode*
2382 {
2383 wxXmlNode* parent = packageNode;
2384 bool is_back = aLayer == B_SilkS || aLayer == B_Fab;
2385
2386 if( is_back )
2387 {
2388 if( !otherSideViewNode )
2389 otherSideViewNode = new wxXmlNode( wxXML_ELEMENT_NODE, "OtherSideView" );
2390
2391 parent = otherSideViewNode;
2392 }
2393
2394 wxString name;
2395
2396 if( aLayer == F_SilkS || aLayer == B_SilkS )
2397 name = "SilkScreen";
2398 else if( aLayer == F_Fab || aLayer == B_Fab )
2399 name = "AssemblyDrawing";
2400 else
2401 wxASSERT( false );
2402
2403 wxXmlNode* new_node = appendNode( parent, name );
2404 return new_node;
2405 };
2406
2407 auto add_marking_node =
2408 [&]( wxXmlNode* aNode ) -> wxXmlNode*
2409 {
2410 wxXmlNode* marking_node = appendNode( aNode, "Marking" );
2411 addAttribute( marking_node, "markingUsage", "NONE" );
2412 return marking_node;
2413 };
2414
2415 std::map<PCB_LAYER_ID, wxXmlNode*> layer_nodes;
2416 std::map<PCB_LAYER_ID, BOX2I> layer_bbox;
2417
2418 for( auto layer : { F_Fab, B_Fab } )
2419 {
2420 if( elements.find( layer ) != elements.end() )
2421 {
2422 if( elements[layer][true].size() > 0 )
2423 layer_bbox[layer] = elements[layer][true][0]->GetBoundingBox();
2424 else if( elements[layer][false].size() > 0 )
2425 layer_bbox[layer] = elements[layer][false][0]->GetBoundingBox();
2426 }
2427 }
2428
2429 for( auto& [layer, map] : elements )
2430 {
2431 wxXmlNode* layer_node = add_base_node( layer );
2432 wxXmlNode* marking_node = add_marking_node( layer_node );
2433 wxXmlNode* group_node = appendNode( marking_node, "UserSpecial" );
2434 bool update_bbox = false;
2435
2436 if( layer == F_Fab || layer == B_Fab )
2437 {
2438 layer_nodes[layer] = layer_node;
2439 update_bbox = true;
2440 }
2441
2442 for( auto& [is_abs, vec] : map )
2443 {
2444 for( BOARD_ITEM* item : vec )
2445 {
2446 wxXmlNode* output_node = nullptr;
2447
2448 if( update_bbox )
2449 layer_bbox[layer].Merge( item->GetBoundingBox() );
2450
2451 if( !is_abs )
2452 output_node = add_marking_node( layer_node );
2453 else
2454 output_node = group_node;
2455
2456 switch( item->Type() )
2457 {
2458 case PCB_TEXT_T:
2459 {
2460 PCB_TEXT* text = static_cast<PCB_TEXT*>( item );
2461
2462 if( text->IsKnockout() )
2463 addKnockoutText( output_node, text );
2464 else
2465 addText( output_node, text, text->GetFontMetrics() );
2466
2467 break;
2468 }
2469
2470 case PCB_TEXTBOX_T:
2471 {
2472 PCB_TEXTBOX* text = static_cast<PCB_TEXTBOX*>( item );
2473 addText( output_node, text, text->GetFontMetrics() );
2474
2475 // We want to force this to be a polygon to get absolute coordinates
2476 if( text->IsBorderEnabled() )
2477 {
2478 SHAPE_POLY_SET poly_set;
2479 text->GetEffectiveShape()->TransformToPolygon( poly_set, 0, ERROR_INSIDE );
2480 addContourNode( output_node, poly_set, 0, FILL_T::NO_FILL,
2481 text->GetBorderWidth() );
2482 }
2483
2484 break;
2485 }
2486
2487 case PCB_SHAPE_T:
2488 {
2489 if( !is_abs )
2490 addLocationNode( output_node, *static_cast<PCB_SHAPE*>( item ) );
2491
2492 addShape( output_node, *static_cast<PCB_SHAPE*>( item ) );
2493
2494 break;
2495 }
2496
2497 default: break;
2498 }
2499 }
2500 }
2501
2502 if( group_node->GetChildren() == nullptr )
2503 {
2504 marking_node->RemoveChild( group_node );
2505 layer_node->RemoveChild( marking_node );
2506 delete group_node;
2507 delete marking_node;
2508 }
2509 }
2510
2511 for( auto&[layer, bbox] : layer_bbox )
2512 {
2513 if( bbox.GetWidth() > 0 )
2514 {
2515 wxXmlNode* outlineNode = insertNode( layer_nodes[layer], "Outline" );
2516
2517 SHAPE_LINE_CHAIN outline;
2518 std::vector<VECTOR2I> points( 4 );
2519 points[0] = bbox.GetPosition();
2520 points[2] = bbox.GetEnd();
2521 points[1].x = points[0].x;
2522 points[1].y = points[2].y;
2523 points[3].x = points[2].x;
2524 points[3].y = points[0].y;
2525
2526 outline.Append( points );
2527 addPolygonNode( outlineNode, outline, FILL_T::NO_FILL, 0 );
2528 addLineDesc( outlineNode, 0, LINE_STYLE::SOLID );
2529 }
2530 }
2531
2532 for( size_t ii = 0; ii < fp->Pads().size(); ++ii )
2533 {
2534 PAD* pad = fp->Pads()[ii];
2535 wxXmlNode* pinNode = appendNode( packageNode, "Pin" );
2536 wxString name = pinName( pad );
2537
2538 addAttribute( pinNode, "number", name );
2539
2540 m_net_pin_dict[pad->GetNetCode()].emplace_back(
2541 genString( fp->GetReference(), "CMP" ), name );
2542
2543 if( pad->GetAttribute() == PAD_ATTRIB::NPTH )
2544 addAttribute( pinNode, "electricalType", "MECHANICAL" );
2545 else if( pad->IsOnCopperLayer() )
2546 addAttribute( pinNode, "electricalType", "ELECTRICAL" );
2547 else
2548 addAttribute( pinNode, "electricalType", "UNDEFINED" );
2549
2550 if( pad->HasHole() )
2551 addAttribute( pinNode, "type", "THRU" );
2552 else
2553 addAttribute( pinNode, "type", "SURFACE" );
2554
2555 if( pad->GetFPRelativeOrientation() != ANGLE_0 )//|| fp->IsFlipped() )
2556 {
2557 wxXmlNode* xformNode = appendNode( pinNode, "Xform" );
2558 EDA_ANGLE pad_angle = pad->GetFPRelativeOrientation().Normalize();
2559
2560 if( fp->IsFlipped() )
2561 pad_angle = ( pad_angle.Invert() - ANGLE_180 ).Normalize();
2562
2563 if( pad_angle != ANGLE_0 )
2564 xformNode->AddAttribute( "rotation", floatVal( pad_angle.AsDegrees() ) );
2565 }
2566
2567 addLocationNode( pinNode, *pad, true );
2568 addShape( pinNode, *pad, pad->GetLayer() );
2569
2570 // We just need the padstack, we don't need the reference here. The reference will be
2571 // created in the LayerFeature set
2572 wxXmlNode dummy;
2573 addPadStack( &dummy, pad );
2574 }
2575
2576 return packageNode;
2577}
2578
2579
2580void PCB_IO_IPC2581::generateComponents( wxXmlNode* aStepNode )
2581{
2582 std::vector<wxXmlNode*> componentNodes;
2583 std::vector<wxXmlNode*> packageNodes;
2584 std::set<wxString> packageNames;
2585
2586 bool generate_unique = m_OEMRef.empty();
2587
2588 for( FOOTPRINT* fp : m_board->Footprints() )
2589 {
2590 wxXmlNode* componentNode = new wxXmlNode( wxXML_ELEMENT_NODE, "Component" );
2591 addAttribute( componentNode, "refDes", componentName( fp ) );
2592 wxXmlNode* pkg = addPackage( componentNode, fp );
2593
2594 if( pkg )
2595 packageNodes.push_back( pkg );
2596
2597 wxString name;
2598
2599 PCB_FIELD* field = nullptr;
2600
2601 if( !generate_unique )
2602 field = fp->GetField( m_OEMRef );
2603
2604 if( field && !field->GetText().empty() )
2605 {
2606 name = field->GetShownText( false );
2607 }
2608 else
2609 {
2610 name = wxString::Format( "%s_%s_%s", fp->GetFPID().GetFullLibraryName(),
2611 fp->GetFPID().GetLibItemName().wx_str(),
2612 fp->GetValue() );
2613 }
2614
2615 if( !m_OEMRef_dict.emplace( fp, name ).second )
2616 wxLogError( "Duplicate footprint pointers. Please report this bug." );
2617
2618 addAttribute( componentNode, "part", genString( name, "REF" ) );
2619 addAttribute( componentNode, "layerRef", m_layer_name_map[fp->GetLayer()] );
2620
2621 if( fp->GetAttributes() & FP_THROUGH_HOLE )
2622 addAttribute( componentNode, "mountType", "THMT" );
2623 else if( fp->GetAttributes() & FP_SMD )
2624 addAttribute( componentNode, "mountType", "SMT" );
2625 else
2626 addAttribute( componentNode, "mountType", "OTHER" );
2627
2628 if( fp->GetOrientation() != ANGLE_0 || fp->IsFlipped() )
2629 {
2630 wxXmlNode* xformNode = appendNode( componentNode, "Xform" );
2631
2632 EDA_ANGLE fp_angle = fp->GetOrientation().Normalize();
2633
2634 if( fp->IsFlipped() )
2635 fp_angle = ( fp_angle.Invert() - ANGLE_180 ).Normalize();
2636
2637 if( fp_angle != ANGLE_0 )
2638 addAttribute( xformNode, "rotation", floatVal( fp_angle.AsDegrees(), 2 ) );
2639
2640 if( fp->IsFlipped() )
2641 addAttribute( xformNode, "mirror", "true" );
2642 }
2643
2644 addLocationNode( componentNode, fp->GetPosition().x, fp->GetPosition().y );
2645
2646 componentNodes.push_back( componentNode );
2647 }
2648
2649 for( wxXmlNode* padstack : m_padstacks )
2650 {
2651 insertNode( aStepNode, padstack );
2652 m_last_padstack = padstack;
2653 }
2654
2655 for( wxXmlNode* pkg : packageNodes )
2656 aStepNode->AddChild( pkg );
2657
2658 for( wxXmlNode* cmp : componentNodes )
2659 aStepNode->AddChild( cmp );
2660}
2661
2662
2663void PCB_IO_IPC2581::generateLogicalNets( wxXmlNode* aStepNode )
2664{
2665 for( auto& [ net, pin_pair] : m_net_pin_dict )
2666 {
2667 wxXmlNode* netNode = appendNode( aStepNode, "LogicalNet" );
2668 addAttribute( netNode, "name",
2669 genString( m_board->GetNetInfo().GetNetItem( net )->GetNetname(), "NET" ) ) ;
2670
2671 for( auto& [cmp, pin] : pin_pair )
2672 {
2673 wxXmlNode* netPinNode = appendNode( netNode, "PinRef" );
2674 addAttribute( netPinNode, "componentRef", cmp );
2675 addAttribute( netPinNode, "pin", pin );
2676 }
2677 //TODO: Finish
2678 }
2679}
2680
2681//TODO: Add PhyNetGroup section
2682
2683void PCB_IO_IPC2581::generateLayerFeatures( wxXmlNode* aStepNode )
2684{
2685 LSEQ layers = m_board->GetEnabledLayers().Seq();
2686 const NETINFO_LIST& nets = m_board->GetNetInfo();
2687 std::vector<std::unique_ptr<FOOTPRINT>> footprints;
2688
2689 // To avoid the overhead of repeatedly cycling through the layers and nets,
2690 // we pre-sort the board items into a map of layer -> net -> items
2691 std::map<PCB_LAYER_ID, std::map<int, std::vector<BOARD_ITEM*>>> elements;
2692
2693 std::for_each( m_board->Tracks().begin(), m_board->Tracks().end(),
2694 [&layers, &elements]( PCB_TRACK* aTrack )
2695 {
2696 if( aTrack->Type() == PCB_VIA_T )
2697 {
2698 PCB_VIA* via = static_cast<PCB_VIA*>( aTrack );
2699
2700 for( PCB_LAYER_ID layer : layers )
2701 {
2702 if( via->FlashLayer( layer ) )
2703 elements[layer][via->GetNetCode()].push_back( via );
2704 }
2705 }
2706 else
2707 {
2708 elements[aTrack->GetLayer()][aTrack->GetNetCode()].push_back( aTrack );
2709 }
2710 } );
2711
2712 std::for_each( m_board->Zones().begin(), m_board->Zones().end(),
2713 [ &elements ]( ZONE* zone )
2714 {
2715 LSEQ zone_layers = zone->GetLayerSet().Seq();
2716
2717 for( PCB_LAYER_ID layer : zone_layers )
2718 elements[layer][zone->GetNetCode()].push_back( zone );
2719 } );
2720
2721 for( BOARD_ITEM* item : m_board->Drawings() )
2722 {
2723 if( BOARD_CONNECTED_ITEM* conn_it = dynamic_cast<BOARD_CONNECTED_ITEM*>( item ) )
2724 elements[conn_it->GetLayer()][conn_it->GetNetCode()].push_back( conn_it );
2725 else
2726 elements[item->GetLayer()][0].push_back( item );
2727 }
2728
2729 for( FOOTPRINT* fp : m_board->Footprints() )
2730 {
2731 for( PCB_FIELD* field : fp->GetFields() )
2732 elements[field->GetLayer()][0].push_back( field );
2733
2734 for( BOARD_ITEM* item : fp->GraphicalItems() )
2735 elements[item->GetLayer()][0].push_back( item );
2736
2737 for( PAD* pad : fp->Pads() )
2738 {
2739 LSEQ pad_layers = pad->GetLayerSet().Seq();
2740
2741 for( PCB_LAYER_ID layer : pad_layers )
2742 {
2743 if( pad->FlashLayer( layer ) )
2744 elements[layer][pad->GetNetCode()].push_back( pad );
2745 }
2746 }
2747 }
2748
2749 for( PCB_LAYER_ID layer : layers )
2750 {
2751 if( m_progressReporter )
2752 m_progressReporter->SetMaxProgress( nets.GetNetCount() * layers.size() );
2753
2754 wxXmlNode* layerNode = appendNode( aStepNode, "LayerFeature" );
2755 addAttribute( layerNode, "layerRef", m_layer_name_map[layer] );
2756
2757 auto process_net = [&] ( int net )
2758 {
2759 std::vector<BOARD_ITEM*>& vec = elements[layer][net];
2760
2761 if( vec.empty() )
2762 return;
2763
2764 std::stable_sort( vec.begin(), vec.end(),
2765 []( BOARD_ITEM* a, BOARD_ITEM* b )
2766 {
2767 if( a->GetParentFootprint() == b->GetParentFootprint() )
2768 return a->Type() < b->Type();
2769
2770 return a->GetParentFootprint() < b->GetParentFootprint();
2771 } );
2772
2773 generateLayerSetNet( layerNode, layer, vec );
2774 };
2775
2776 for( const NETINFO_ITEM* net : nets )
2777 {
2778 if( m_progressReporter )
2779 {
2780 m_progressReporter->Report( wxString::Format( _( "Exporting Layer %s, Net %s" ),
2781 m_board->GetLayerName( layer ),
2782 net->GetNetname() ) );
2783 m_progressReporter->AdvanceProgress();
2784 }
2785
2786 process_net( net->GetNetCode() );
2787 }
2788
2789 if( layerNode->GetChildren() == nullptr )
2790 {
2791 aStepNode->RemoveChild( layerNode );
2792 delete layerNode;
2793 }
2794 }
2795}
2796
2797
2798void PCB_IO_IPC2581::generateLayerSetDrill( wxXmlNode* aLayerNode )
2799{
2800 int hole_count = 1;
2801
2802 for( const auto& [layers, vec] : m_drill_layers )
2803 {
2804 wxXmlNode* layerNode = appendNode( aLayerNode, "LayerFeature" );
2805 layerNode->AddAttribute( "layerRef", genLayersString( layers.first, layers.second, "DRILL" ) );
2806
2807 for( BOARD_ITEM* item : vec )
2808 {
2809 if( item->Type() == PCB_VIA_T )
2810 {
2811 PCB_VIA* via = static_cast<PCB_VIA*>( item );
2812 auto it = m_padstack_dict.find( hash_fp_item( via, 0 ) );
2813
2814 if( it == m_padstack_dict.end() )
2815 {
2816 wxLogError( "Failed to find padstack for via" );
2817 continue;
2818 }
2819
2820 wxXmlNode* padNode = appendNode( layerNode, "Set" );
2821 addAttribute( padNode, "geometry", it->second );
2822
2823 if( via->GetNetCode() > 0 )
2824 addAttribute( padNode, "net", genString( via->GetNetname(), "NET" ) );
2825
2826 wxXmlNode* holeNode = appendNode( padNode, "Hole" );
2827 addAttribute( holeNode, "name", wxString::Format( "H%d", hole_count++ ) );
2828 addAttribute( holeNode, "diameter", floatVal( m_scale * via->GetDrillValue() ) );
2829 addAttribute( holeNode, "platingStatus", "VIA" );
2830 addAttribute( holeNode, "plusTol", "0.0" );
2831 addAttribute( holeNode, "minusTol", "0.0" );
2832 addXY( holeNode, via->GetPosition() );
2833 }
2834 else if( item->Type() == PCB_PAD_T )
2835 {
2836 PAD* pad = static_cast<PAD*>( item );
2837 auto it = m_padstack_dict.find( hash_fp_item( pad, 0 ) );
2838
2839 if( it == m_padstack_dict.end() )
2840 {
2841 wxLogError( "Failed to find padstack for pad" );
2842 continue;
2843 }
2844
2845 wxXmlNode* padNode = appendNode( layerNode, "Set" );
2846 addAttribute( padNode, "geometry", it->second );
2847
2848 if( pad->GetNetCode() > 0 )
2849 addAttribute( padNode, "net", genString( pad->GetNetname(), "NET" ) );
2850
2851 wxXmlNode* holeNode = appendNode( padNode, "Hole" );
2852 addAttribute( holeNode, "name", wxString::Format( "H%d", hole_count++ ) );
2853 addAttribute( holeNode, "diameter", floatVal( m_scale * pad->GetDrillSizeX() ) );
2854 addAttribute( holeNode, "platingStatus",
2855 pad->GetAttribute() == PAD_ATTRIB::PTH ? "PLATED" : "NONPLATED" );
2856 addAttribute( holeNode, "plusTol", "0.0" );
2857 addAttribute( holeNode, "minusTol", "0.0" );
2858 addXY( holeNode, pad->GetPosition() );
2859 }
2860 }
2861 }
2862
2863 hole_count = 1;
2864
2865 for( const auto& [layers, vec] : m_slot_holes )
2866 {
2867 wxXmlNode* layerNode = appendNode( aLayerNode, "LayerFeature" );
2868 layerNode->AddAttribute( "layerRef", genLayersString( layers.first, layers.second, "SLOT" ) );
2869
2870 for( PAD* pad : vec )
2871 {
2872 wxXmlNode* padNode = appendNode( layerNode, "Set" );
2873
2874 if( pad->GetNetCode() > 0 )
2875 addAttribute( padNode, "net", genString( pad->GetNetname(), "NET" ) );
2876
2877 addSlotCavity( padNode, *pad, wxString::Format( "SLOT%d", hole_count++ ) );
2878 }
2879 }
2880}
2881
2882
2883void PCB_IO_IPC2581::generateLayerSetNet( wxXmlNode* aLayerNode, PCB_LAYER_ID aLayer,
2884 std::vector<BOARD_ITEM*>& aItems )
2885{
2886 auto it = aItems.begin();
2887 wxXmlNode* layerSetNode = appendNode( aLayerNode, "Set" );
2888 wxXmlNode* featureSetNode = appendNode( layerSetNode, "Features" );
2889 wxXmlNode* specialNode = appendNode( featureSetNode, "UserSpecial" );
2890
2891 bool has_via = false;
2892 bool has_pad = false;
2893
2894 wxXmlNode* padSetNode = nullptr;
2895
2896 wxXmlNode* viaSetNode = nullptr;
2897
2898 wxXmlNode* teardropLayerSetNode = nullptr;
2899 wxXmlNode* teardropFeatureSetNode = nullptr;
2900
2901 if( BOARD_CONNECTED_ITEM* item = dynamic_cast<BOARD_CONNECTED_ITEM*>( *it );
2902 IsCopperLayer( aLayer ) && item )
2903 {
2904 if( item->GetNetCode() > 0 )
2905 addAttribute( layerSetNode, "net", genString( item->GetNetname(), "NET" ) );
2906 }
2907
2908 auto add_track =
2909 [&]( PCB_TRACK* track )
2910 {
2911 if( track->Type() == PCB_TRACE_T )
2912 {
2913 PCB_SHAPE shape( nullptr, SHAPE_T::SEGMENT );
2914 shape.SetStart( track->GetStart() );
2915 shape.SetEnd( track->GetEnd() );
2916 shape.SetWidth( track->GetWidth() );
2917 addShape( specialNode, shape );
2918 }
2919 else if( track->Type() == PCB_ARC_T )
2920 {
2921 PCB_ARC* arc = static_cast<PCB_ARC*>( track );
2922 PCB_SHAPE shape( nullptr, SHAPE_T::ARC );
2923 shape.SetArcGeometry( arc->GetStart(), arc->GetMid(), arc->GetEnd() );
2924 shape.SetWidth( arc->GetWidth() );
2925 addShape( specialNode, shape );
2926 }
2927 else
2928 {
2929 if( !viaSetNode )
2930 {
2931 if( !has_pad )
2932 {
2933 viaSetNode = layerSetNode;
2934 has_via = true;
2935 }
2936 else
2937 {
2938 viaSetNode = appendNode( layerSetNode, "Set" );
2939
2940 if( track->GetNetCode() > 0 )
2941 addAttribute( viaSetNode, "net", genString( track->GetNetname(), "NET" ) );
2942 }
2943
2944 addAttribute( viaSetNode, "padUsage", "VIA" );
2945 }
2946
2947 addVia( viaSetNode, static_cast<PCB_VIA*>( track ), aLayer );
2948 }
2949 };
2950
2951 auto add_zone =
2952 [&]( ZONE* zone )
2953 {
2954 wxXmlNode* zoneFeatureNode = specialNode;
2955
2956 if( zone->IsTeardropArea() && m_version > 'B' )
2957 {
2958 if( !teardropFeatureSetNode )
2959 {
2960 teardropLayerSetNode = appendNode( aLayerNode, "Set" );
2961 addAttribute( teardropLayerSetNode, "geometryUsage", "TEARDROP" );
2962
2963 if( zone->GetNetCode() > 0 )
2964 {
2965 addAttribute( teardropLayerSetNode, "net",
2966 genString( zone->GetNetname(), "NET" ) );
2967 }
2968
2969 wxXmlNode* new_teardrops = appendNode( teardropLayerSetNode, "Features" );
2970 addLocationNode( new_teardrops, 0.0, 0.0 );
2971 teardropFeatureSetNode = appendNode( new_teardrops, "UserSpecial" );
2972 }
2973
2974 zoneFeatureNode = teardropFeatureSetNode;
2975 }
2976 else
2977 {
2978 if( FOOTPRINT* fp = zone->GetParentFootprint() )
2979 {
2980 wxXmlNode* tempSetNode = appendNode( aLayerNode, "Set" );
2981 wxString refDes = componentName( zone->GetParentFootprint() );
2982 addAttribute( tempSetNode, "componentRef", refDes );
2983 wxXmlNode* newFeatures = appendNode( tempSetNode, "Features" );
2984 addLocationNode( newFeatures, 0.0, 0.0 );
2985 zoneFeatureNode = appendNode( newFeatures, "UserSpecial" );
2986 }
2987 }
2988
2989 SHAPE_POLY_SET& zone_shape = *zone->GetFilledPolysList( aLayer );
2990
2991 for( int ii = 0; ii < zone_shape.OutlineCount(); ++ii )
2992 addContourNode( zoneFeatureNode, zone_shape, ii );
2993 };
2994
2995 auto add_shape =
2996 [&] ( PCB_SHAPE* shape )
2997 {
2998 FOOTPRINT* fp = shape->GetParentFootprint();
2999
3000 if( fp )
3001 {
3002 wxXmlNode* tempSetNode = appendNode( aLayerNode, "Set" );
3003
3004 if( m_version > 'B' )
3005 addAttribute( tempSetNode, "geometryUsage", "GRAPHIC" );
3006
3007 addAttribute( tempSetNode, "componentRef", componentName( fp ) );
3008
3009 wxXmlNode* tempFeature = appendNode( tempSetNode, "Features" );
3010 addLocationNode( tempFeature, *shape );
3011
3012 addShape( tempFeature, *shape );
3013 }
3014 else if( shape->GetShape() == SHAPE_T::CIRCLE
3015 || shape->GetShape() == SHAPE_T::RECTANGLE
3016 || shape->GetShape() == SHAPE_T::POLY )
3017 {
3018 wxXmlNode* tempSetNode = appendNode( aLayerNode, "Set" );
3019
3020 if( shape->GetNetCode() > 0 )
3021 addAttribute( tempSetNode, "net", genString( shape->GetNetname(), "NET" ) );
3022
3023 wxXmlNode* tempFeature = appendNode( tempSetNode, "Features" );
3024 addLocationNode( tempFeature, *shape );
3025 addShape( tempFeature, *shape );
3026 }
3027 else
3028 {
3029 addShape( specialNode, *shape );
3030 }
3031 };
3032
3033 auto add_text =
3034 [&] ( BOARD_ITEM* text )
3035 {
3036 EDA_TEXT* text_item;
3037 FOOTPRINT* fp = text->GetParentFootprint();
3038
3039 if( PCB_TEXT* tmp_text = dynamic_cast<PCB_TEXT*>( text ) )
3040 text_item = static_cast<EDA_TEXT*>( tmp_text );
3041 else if( PCB_TEXTBOX* tmp_text = dynamic_cast<PCB_TEXTBOX*>( text ) )
3042 text_item = static_cast<EDA_TEXT*>( tmp_text );
3043
3044 if( !text_item->IsVisible() || text_item->GetShownText( false ).empty() )
3045 return;
3046
3047 wxXmlNode* tempSetNode = appendNode( aLayerNode, "Set" );
3048
3049 if( m_version > 'B' )
3050 addAttribute( tempSetNode, "geometryUsage", "TEXT" );
3051
3052 if( fp )
3053 addAttribute( tempSetNode, "componentRef", componentName( fp ) );
3054
3055 wxXmlNode* nonStandardAttributeNode = appendNode( tempSetNode, "NonstandardAttribute" );
3056 addAttribute( nonStandardAttributeNode, "name", "TEXT" );
3057 addAttribute( nonStandardAttributeNode, "value", text_item->GetShownText( false ) );
3058 addAttribute( nonStandardAttributeNode, "type", "STRING" );
3059
3060 wxXmlNode* tempFeature = appendNode( tempSetNode, "Features" );
3061 addLocationNode( tempFeature, 0.0, 0.0 );
3062
3063 if( text->Type() == PCB_TEXT_T && static_cast<PCB_TEXT*>( text )->IsKnockout() )
3064 addKnockoutText( tempFeature, static_cast<PCB_TEXT*>( text ) );
3065 else
3066 addText( tempFeature, text_item, text->GetFontMetrics() );
3067
3068 if( text->Type() == PCB_TEXTBOX_T )
3069 {
3070 PCB_TEXTBOX* textbox = static_cast<PCB_TEXTBOX*>( text );
3071
3072 if( textbox->IsBorderEnabled() )
3073 addShape( tempFeature, *static_cast<PCB_SHAPE*>( textbox ) );
3074 }
3075 };
3076
3077 auto add_pad =
3078 [&]( PAD* pad )
3079 {
3080 if( !padSetNode )
3081 {
3082 if( !has_via )
3083 {
3084 padSetNode = layerSetNode;
3085 has_pad = true;
3086 }
3087 else
3088 {
3089 padSetNode = appendNode( aLayerNode, "Set" );
3090
3091 if( pad->GetNetCode() > 0 )
3092 addAttribute( padSetNode, "net", genString( pad->GetNetname(), "NET" ) );
3093 }
3094 }
3095
3096 FOOTPRINT* fp = pad->GetParentFootprint();
3097
3098 if( fp && fp->IsFlipped() )
3099 addPad( padSetNode, pad, FlipLayer( aLayer ) );
3100 else
3101 addPad( padSetNode, pad, aLayer );
3102 };
3103
3104 for( BOARD_ITEM* item : aItems )
3105 {
3106 switch( item->Type() )
3107 {
3108 case PCB_TRACE_T:
3109 case PCB_ARC_T:
3110 case PCB_VIA_T:
3111 add_track( static_cast<PCB_TRACK*>( item ) );
3112 break;
3113
3114 case PCB_ZONE_T:
3115 add_zone( static_cast<ZONE*>( item ) );
3116 break;
3117
3118 case PCB_PAD_T:
3119 add_pad( static_cast<PAD*>( item ) );
3120 break;
3121
3122 case PCB_SHAPE_T:
3123 add_shape( static_cast<PCB_SHAPE*>( item ) );
3124 break;
3125
3126 case PCB_TEXT_T:
3127 case PCB_TEXTBOX_T:
3128 case PCB_FIELD_T:
3129 add_text( item );
3130 break;
3131
3132 case PCB_DIMENSION_T:
3133 case PCB_TARGET_T:
3134 case PCB_DIM_ALIGNED_T:
3135 case PCB_DIM_LEADER_T:
3136 case PCB_DIM_CENTER_T:
3137 case PCB_DIM_RADIAL_T:
3139 //TODO: Add support for dimensions
3140 break;
3141
3142 default:
3143 wxLogTrace( traceIpc2581, wxS( "Unhandled type %s" ),
3144 ENUM_MAP<KICAD_T>::Instance().ToString( item->Type() ) );
3145 }
3146 }
3147
3148 if( specialNode->GetChildren() == nullptr )
3149 {
3150 featureSetNode->RemoveChild( specialNode );
3151 delete specialNode;
3152 }
3153
3154 if( featureSetNode->GetChildren() == nullptr )
3155 {
3156 layerSetNode->RemoveChild( featureSetNode );
3157 delete featureSetNode;
3158 }
3159
3160 if( layerSetNode->GetChildren() == nullptr )
3161 {
3162 aLayerNode->RemoveChild( layerSetNode );
3163 delete layerSetNode;
3164 }
3165}
3166
3168{
3169 int hole_count = 1;
3170
3171 for( const auto& [layers, vec] : m_auxilliary_Layers )
3172 {
3173 hole_count = 1;
3174 bool add_node = true;
3175
3176 wxString name;
3177 bool hole = false;
3178
3179 // clang-format off: suggestion is inconsitent
3180 switch( std::get<0>(layers) )
3181 {
3182 case auxLayerType::COVERING:
3183 name = "COVERING";
3184 break;
3185 case auxLayerType::PLUGGING:
3186 name = "PLUGGING";
3187 hole = true;
3188 break;
3189 case auxLayerType::TENTING:
3190 name = "TENTING";
3191 break;
3192 case auxLayerType::FILLING:
3193 name = "FILLING";
3194 hole = true;
3195 break;
3196 case auxLayerType::CAPPING:
3197 name = "CAPPING";
3198 hole = true;
3199 break;
3200 default:
3201 add_node = false;
3202 break;
3203 }
3204 // clang-format on: suggestion is inconsitent
3205
3206 if( !add_node )
3207 continue;
3208
3209 wxXmlNode* layerNode = appendNode( aStepNode, "LayerFeature" );
3210 if( std::get<2>( layers ) == UNDEFINED_LAYER )
3211 layerNode->AddAttribute( "layerRef", genLayerString( std::get<1>( layers ), TO_UTF8( name ) ) );
3212 else
3213 layerNode->AddAttribute( "layerRef", genLayersString( std::get<1>( layers ),
3214 std::get<2>( layers ), TO_UTF8( name ) ) );
3215
3216 for( BOARD_ITEM* item : vec )
3217 {
3218 if( item->Type() == PCB_VIA_T )
3219 {
3220 PCB_VIA* via = static_cast<PCB_VIA*>( item );
3221
3222 PCB_SHAPE shape( nullptr, SHAPE_T::CIRCLE );
3223
3224 if( hole )
3225 shape.SetEnd( { KiROUND( via->GetDrillValue() / 2.0 ), 0 } );
3226 else
3227 shape.SetEnd( { KiROUND( via->GetWidth( std::get<1>( layers ) ) / 2.0 ), 0 } );
3228
3229 wxXmlNode* padNode = appendNode( layerNode, "Pad" );
3230 addPadStack( padNode, via );
3231
3232 addLocationNode( padNode, 0.0, 0.0 );
3233 addShape( padNode, shape );
3234 }
3235 }
3236 }
3237}
3238
3239
3241{
3242 if( m_progressReporter )
3243 m_progressReporter->AdvancePhase( _( "Generating BOM section" ) );
3244
3245 wxXmlNode* avl = appendNode( m_xml_root, "Avl" );
3246 addAttribute( avl, "name", "Primary_Vendor_List" );
3247
3248 wxXmlNode* header = appendNode( avl, "AvlHeader" );
3249 addAttribute( header, "title", "BOM" );
3250 addAttribute( header, "source", "KiCad" );
3251 addAttribute( header, "author", "OWNER" );
3252 addAttribute( header, "datetime", wxDateTime::Now().FormatISOCombined() );
3253 addAttribute( header, "version", "1" );
3254
3255 std::set<wxString> unique_parts;
3256 std::map<wxString,wxString> unique_vendors;
3257
3258 for( auto& [fp, name] : m_OEMRef_dict )
3259 {
3260 auto [ it, success ] = unique_parts.insert( name );
3261
3262 if( !success )
3263 continue;
3264
3265 wxXmlNode* part = appendNode( avl, "AvlItem" );
3266 addAttribute( part, "OEMDesignNumber", genString( name, "REF" ) );
3267
3268 PCB_FIELD* nums[2] = { fp->GetField( m_mpn ), fp->GetField( m_distpn ) };
3269 PCB_FIELD* company[2] = { fp->GetField( m_mfg ), nullptr };
3270 wxString company_name[2] = { m_mfg, m_dist };
3271
3272 for ( int ii = 0; ii < 2; ++ii )
3273 {
3274 if( nums[ii] )
3275 {
3276 wxString mpn_name = nums[ii]->GetShownText( false );
3277
3278 if( mpn_name.empty() )
3279 continue;
3280
3281 wxXmlNode* vmpn = appendNode( part, "AvlVmpn" );
3282 addAttribute( vmpn, "qualified", "false" );
3283 addAttribute( vmpn, "chosen", "false" );
3284
3285 wxXmlNode* mpn = appendNode( vmpn, "AvlMpn" );
3286 addAttribute( mpn, "name", mpn_name );
3287
3288 wxXmlNode* vendor = appendNode( vmpn, "AvlVendor" );
3289
3290 wxString name = wxT( "UNKNOWN" );
3291
3292 // If the field resolves, then use that field content unless it is empty
3293 if( !ii && company[ii] )
3294 {
3295 wxString tmp = company[ii]->GetShownText( false );
3296
3297 if( !tmp.empty() )
3298 name = tmp;
3299 }
3300 // If it doesn't resolve but there is content from the dialog, use the static content
3301 else if( !ii && !company_name[ii].empty() )
3302 {
3303 name = company_name[ii];
3304 }
3305 else if( ii && !m_dist.empty() )
3306 {
3307 name = m_dist;
3308 }
3309
3310 auto [vendor_id, inserted] = unique_vendors.emplace(
3311 name,
3312 wxString::Format( "VENDOR_%zu", unique_vendors.size() ) );
3313
3314 addAttribute( vendor, "enterpriseRef", vendor_id->second );
3315
3316 if( inserted )
3317 {
3318 wxXmlNode* new_vendor = new wxXmlNode( wxXML_ELEMENT_NODE, "Enterprise" );
3319 addAttribute( new_vendor, "id", vendor_id->second );
3320 addAttribute( new_vendor, "name", name );
3321 addAttribute( new_vendor, "code", "NONE" );
3322 insertNodeAfter( m_enterpriseNode, new_vendor );
3323 m_enterpriseNode = new_vendor;
3324 }
3325 }
3326 }
3327 }
3328
3329 return avl;
3330}
3331
3332
3333void PCB_IO_IPC2581::SaveBoard( const wxString& aFileName, BOARD* aBoard,
3334 const std::map<std::string, UTF8>* aProperties )
3335{
3336 m_board = aBoard;
3337 m_units_str = "MILLIMETER";
3338 m_scale = 1.0 / PCB_IU_PER_MM;
3339 m_sigfig = 6;
3340
3341 if( auto it = aProperties->find( "units" ); it != aProperties->end() )
3342 {
3343 if( it->second == "inch" )
3344 {
3345 m_units_str = "INCH";
3346 m_scale = ( 1.0 / 25.4 ) / PCB_IU_PER_MM;
3347 }
3348 }
3349
3350 if( auto it = aProperties->find( "sigfig" ); it != aProperties->end() )
3351 m_sigfig = std::stoi( it->second );
3352
3353 if( auto it = aProperties->find( "version" ); it != aProperties->end() )
3354 m_version = it->second.c_str()[0];
3355
3356 if( auto it = aProperties->find( "OEMRef" ); it != aProperties->end() )
3357 m_OEMRef = it->second.wx_str();
3358
3359 if( auto it = aProperties->find( "mpn" ); it != aProperties->end() )
3360 m_mpn = it->second.wx_str();
3361
3362 if( auto it = aProperties->find( "mfg" ); it != aProperties->end() )
3363 m_mfg = it->second.wx_str();
3364
3365 if( auto it = aProperties->find( "dist" ); it != aProperties->end() )
3366 m_dist = it->second.wx_str();
3367
3368 if( auto it = aProperties->find( "distpn" ); it != aProperties->end() )
3369 m_distpn = it->second.wx_str();
3370
3371 if( m_version == 'B' )
3372 {
3373 for( char c = 'a'; c <= 'z'; ++c )
3374 m_acceptable_chars.insert( c );
3375
3376 for( char c = 'A'; c <= 'Z'; ++c )
3377 m_acceptable_chars.insert( c );
3378
3379 for( char c = '0'; c <= '9'; ++c )
3380 m_acceptable_chars.insert( c );
3381
3382 // Add special characters
3383 std::string specialChars = "_\\-.+><";
3384
3385 for( char c : specialChars )
3386 m_acceptable_chars.insert( c );
3387 }
3388
3389 m_xml_doc = new wxXmlDocument();
3391
3393
3394 if( m_progressReporter )
3395 {
3398 m_progressReporter->Report( _( "Generating logistic section" ) );
3399 }
3400
3403
3404 wxXmlNode* ecad_node = generateEcadSection();
3405 generateBOMSection( ecad_node );
3407
3408 if( m_progressReporter )
3409 {
3410 m_progressReporter->AdvancePhase( _( "Saving file" ) );
3411 }
3412
3413 wxFileOutputStreamWithProgress out_stream( aFileName );
3414 double written_bytes = 0.0;
3415 double last_yield = 0.0;
3416
3417 // This is a rough estimation of the size of the spaces in the file
3418 // We just need to total to be slightly larger than the value of the
3419 // progress bar, so accurately counting spaces is not terribly important
3421
3422 auto update_progress = [&]( size_t aBytes )
3423 {
3424 written_bytes += aBytes;
3425 double percent = written_bytes / static_cast<double>( m_total_bytes );
3426
3427 if( m_progressReporter )
3428 {
3429 // Only update every percent
3430 if( last_yield + 0.01 < percent )
3431 {
3432 last_yield = percent;
3434 }
3435 }
3436 };
3437
3438 out_stream.SetProgressCallback( update_progress );
3439
3440 if( !m_xml_doc->Save( out_stream ) )
3441 {
3442 wxLogError( _( "Failed to save file to buffer" ) );
3443 return;
3444 }
3445
3446 size_t size = out_stream.GetSize();
3447}
int color
Definition: DXF_plotter.cpp:60
const char * name
Definition: DXF_plotter.cpp:59
@ ERROR_INSIDE
Definition: approximation.h:34
constexpr int ARC_HIGH_DEF
Definition: base_units.h:120
constexpr double PCB_IU_PER_MM
Pcbnew IU is 1 nanometer.
Definition: base_units.h:70
bool IsPrmSpecified(const wxString &aPrmValue)
@ BS_ITEM_TYPE_COPPER
Definition: board_stackup.h:45
@ BS_ITEM_TYPE_SILKSCREEN
Definition: board_stackup.h:51
@ BS_ITEM_TYPE_DIELECTRIC
Definition: board_stackup.h:46
@ BS_ITEM_TYPE_SOLDERMASK
Definition: board_stackup.h:49
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
wxString GetMajorMinorPatchVersion()
Get the major, minor and patch version in a string major.minor.patch This is extracted by CMake from ...
Bezier curves to polygon converter.
Definition: bezier_curves.h:38
void GetPoly(std::vector< VECTOR2I > &aOutput, int aMaxError=10)
Convert a Bezier curve to a polygon.
A base class derived from BOARD_ITEM for items that can be connected and have a net,...
Container for design settings for a BOARD object.
BOARD_STACKUP & GetStackupDescriptor()
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition: board_item.h:78
virtual bool IsKnockout() const
Definition: board_item.h:316
FOOTPRINT * GetParentFootprint() const
Definition: board_item.cpp:299
VECTOR2I GetFPRelativePosition() const
Definition: board_item.cpp:328
Manage one layer needed to make a physical board.
Definition: board_stackup.h:96
wxString GetTypeName() const
int GetSublayersCount() const
double GetEpsilonR(int aDielectricSubLayer=0) const
wxString GetColor(int aDielectricSubLayer=0) const
wxString GetLayerName() const
PCB_LAYER_ID GetBrdLayerId() const
int GetThickness(int aDielectricSubLayer=0) const
BOARD_STACKUP_ITEM_TYPE GetType() const
wxString GetMaterial(int aDielectricSubLayer=0) const
int GetDielectricLayerId() const
double GetLossTangent(int aDielectricSubLayer=0) const
Manage layers needed to make a physical board.
const std::vector< BOARD_STACKUP_ITEM * > & GetList() const
int GetCount() const
bool SynchronizeWithBoard(BOARD_DESIGN_SETTINGS *aSettings)
Synchronize the BOARD_STACKUP_ITEM* list with the board.
int BuildBoardThicknessFromStackup() const
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:297
bool GetBoardPolygonOutlines(SHAPE_POLY_SET &aOutlines, OUTLINE_ERROR_HANDLER *aErrorHandler=nullptr, bool aAllowUseArcsInPolygons=false, bool aIncludeNPTHAsOutlines=false)
Extract the board outlines and build a closed polygon from lines, arcs and circle items on edge cut l...
Definition: board.cpp:2531
const NETINFO_LIST & GetNetInfo() const
Definition: board.h:897
LSET GetEnabledLayers() const
A proxy function that calls the corresponding function in m_BoardSettings.
Definition: board.cpp:829
bool IsLayerEnabled(PCB_LAYER_ID aLayer) const
A proxy function that calls the correspondent function in m_BoardSettings tests whether a given layer...
Definition: board.cpp:855
const FOOTPRINTS & Footprints() const
Definition: board.h:338
const TRACKS & Tracks() const
Definition: board.h:336
const wxString & GetFileName() const
Definition: board.h:334
const wxString GetLayerName(PCB_LAYER_ID aLayer) const
Return the name of a aLayer.
Definition: board.cpp:614
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Definition: board.cpp:946
constexpr BOX2< Vec > & Merge(const BOX2< Vec > &aRect)
Modify the position and size of the rectangle in order to contain aRect.
Definition: box2.h:658
constexpr coord_type GetLeft() const
Definition: box2.h:228
constexpr coord_type GetRight() const
Definition: box2.h:217
constexpr coord_type GetTop() const
Definition: box2.h:229
constexpr coord_type GetBottom() const
Definition: box2.h:222
EDA_ANGLE Normalize()
Definition: eda_angle.h:221
double AsDegrees() const
Definition: eda_angle.h:113
EDA_ANGLE Invert() const
Definition: eda_angle.h:165
const VECTOR2I & GetBezierC2() const
Definition: eda_shape.h:258
FILL_T GetFillMode() const
Definition: eda_shape.h:142
int GetRectangleWidth() const
Definition: eda_shape.cpp:419
SHAPE_POLY_SET & GetPolyShape()
Definition: eda_shape.h:337
int GetRadius() const
Definition: eda_shape.cpp:961
SHAPE_T GetShape() const
Definition: eda_shape.h:168
const VECTOR2I & GetEnd() const
Return the ending point of the graphic.
Definition: eda_shape.h:215
void SetStart(const VECTOR2I &aStart)
Definition: eda_shape.h:177
const VECTOR2I & GetStart() const
Return the starting point of the graphic.
Definition: eda_shape.h:173
void SetEnd(const VECTOR2I &aEnd)
Definition: eda_shape.h:219
void SetArcGeometry(const VECTOR2I &aStart, const VECTOR2I &aMid, const VECTOR2I &aEnd)
Set the three controlling points for an arc.
Definition: eda_shape.cpp:994
const VECTOR2I & GetBezierC1() const
Definition: eda_shape.h:255
int GetRectangleHeight() const
Definition: eda_shape.cpp:405
bool IsClockwiseArc() const
Definition: eda_shape.cpp:1044
void SetWidth(int aWidth)
Definition: eda_shape.cpp:2303
A mix-in class (via multiple inheritance) that handles texts such as labels, parts,...
Definition: eda_text.h:80
virtual const wxString & GetText() const
Return the string associated with the text object.
Definition: eda_text.h:98
virtual bool IsVisible() const
Definition: eda_text.h:174
KIFONT::FONT * GetFont() const
Definition: eda_text.h:234
virtual EDA_ANGLE GetDrawRotation() const
Definition: eda_text.h:365
const TEXT_ATTRIBUTES & GetAttributes() const
Definition: eda_text.h:218
int GetEffectiveTextPenWidth(int aDefaultPenWidth=0) const
The EffectiveTextPenWidth uses the text thickness if > 1 or aDefaultPenWidth.
Definition: eda_text.cpp:458
virtual wxString GetShownText(bool aAllowExtraText, int aDepth=0) const
Return the string actually shown after processing of the base text.
Definition: eda_text.h:109
unsigned GetPadCount(INCLUDE_NPTH_T aIncludeNPTH=INCLUDE_NPTH_T(INCLUDE_NPTH)) const
Return the number of pads.
Definition: footprint.cpp:2007
EDA_ITEM * Clone() const override
Invoke a function on all children.
Definition: footprint.cpp:2178
std::deque< PAD * > & Pads()
Definition: footprint.h:211
bool IsFlipped() const
Definition: footprint.h:396
const wxString & GetReference() const
Definition: footprint.h:619
PROGRESS_REPORTER * m_progressReporter
Progress reporter to track the progress of the operation, may be nullptr.
Definition: io_base.h:223
FONT is an abstract base class for both outline and stroke fonts.
Definition: font.h:131
static FONT * GetFont(const wxString &aFontName=wxEmptyString, bool aBold=false, bool aItalic=false, const std::vector< wxString > *aEmbeddedFiles=nullptr, bool aForDrawingSheet=false)
Definition: font.cpp:147
A color representation with 4 components: red, green, blue, alpha.
Definition: color4d.h:104
double r
Red component.
Definition: color4d.h:392
double g
Green component.
Definition: color4d.h:393
double b
Blue component.
Definition: color4d.h:394
LSEQ is a sequence (and therefore also a set) of PCB_LAYER_IDs.
Definition: lseq.h:47
LSET is a set of PCB_LAYER_IDs.
Definition: lset.h:37
LSEQ Seq(const LSEQ &aSequence) const
Return an LSEQ from the union of this LSET and a desired sequence.
Definition: lset.cpp:297
Handle the data for a net.
Definition: netinfo.h:56
const wxString & GetNetname() const
Definition: netinfo.h:114
Container for NETINFO_ITEM elements, which are the nets.
Definition: netinfo.h:346
NETINFO_ITEM * GetNetItem(int aNetCode) const
std::optional< bool > IsFilled() const
Definition: padstack.cpp:1376
std::optional< bool > IsTented(PCB_LAYER_ID aSide) const
Checks if this padstack is tented (covered in soldermask) on the given side.
Definition: padstack.cpp:1338
std::optional< bool > IsPlugged(PCB_LAYER_ID aSide) const
Definition: padstack.cpp:1360
std::optional< bool > IsCapped() const
Definition: padstack.cpp:1371
std::optional< bool > IsCovered(PCB_LAYER_ID aSide) const
Definition: padstack.cpp:1349
static constexpr PCB_LAYER_ID ALL_LAYERS
! Temporary layer identifier to identify code that is not padstack-aware
Definition: padstack.h:144
Definition: pad.h:54
LSET GetLayerSet() const override
Return a std::bitset of all layers on which the item physically resides.
Definition: pad.h:437
int GetSizeX() const
Definition: pad.h:280
void MergePrimitivesAsPolygon(PCB_LAYER_ID aLayer, SHAPE_POLY_SET *aMergedPolygon, ERROR_LOC aErrorLoc=ERROR_INSIDE) const
Merge all basic shapes to a SHAPE_POLY_SET.
Definition: pad.cpp:2620
int GetRoundRectCornerRadius(PCB_LAYER_ID aLayer) const
Definition: pad.cpp:455
bool FlashLayer(int aLayer, bool aOnlyCheckIfPermitted=false) const
Check to see whether the pad should be flashed on the specific layer.
Definition: pad.cpp:355
virtual std::shared_ptr< SHAPE > GetEffectiveShape(PCB_LAYER_ID aLayer, FLASHING flashPTHPads=FLASHING::DEFAULT) const override
Some pad shapes can be complex (rounded/chamfered rectangle), even without considering custom shapes.
Definition: pad.cpp:525
int GetDrillSizeY() const
Definition: pad.h:309
const VECTOR2I & GetDrillSize() const
Definition: pad.h:305
PAD_ATTRIB GetAttribute() const
Definition: pad.h:440
const wxString & GetNumber() const
Definition: pad.h:136
const VECTOR2I & GetDelta(PCB_LAYER_ID aLayer) const
Definition: pad.h:299
VECTOR2I GetPosition() const override
Definition: pad.h:208
int GetDrillSizeX() const
Definition: pad.h:307
PAD_SHAPE GetShape(PCB_LAYER_ID aLayer) const
Definition: pad.h:195
int GetSolderMaskExpansion(PCB_LAYER_ID aLayer) const
Definition: pad.cpp:1152
const VECTOR2I & GetOffset(PCB_LAYER_ID aLayer) const
Definition: pad.h:317
EDA_ANGLE GetOrientation() const
Return the rotation angle of the pad.
Definition: pad.h:408
int GetChamferPositions(PCB_LAYER_ID aLayer) const
Definition: pad.h:708
double GetChamferRectRatio(PCB_LAYER_ID aLayer) const
Definition: pad.h:691
VECTOR2I GetSolderPasteMargin(PCB_LAYER_ID aLayer) const
Usually < 0 (mask shape smaller than pad)because the margin can be dependent on the pad size,...
Definition: pad.cpp:1198
bool HasDrilledHole() const override
Definition: pad.h:111
bool HasHole() const override
Definition: pad.h:106
const VECTOR2I & GetSize(PCB_LAYER_ID aLayer) const
Definition: pad.h:264
const VECTOR2I & GetMid() const
Definition: pcb_track.h:337
void addText(wxXmlNode *aContentNode, EDA_TEXT *aShape, const KIFONT::METRICS &aFontMetrics)
wxString floatVal(double aVal, int aSigFig=-1) const
void generateLayerSetDrill(wxXmlNode *aStepNode)
wxString genLayerString(PCB_LAYER_ID aLayer, const char *aPrefix) const
wxXmlNode * generateContentSection()
Creates the Content section of the XML file.
wxXmlNode * appendNode(wxXmlNode *aParent, const wxString &aName)
void generateDrillLayers(wxXmlNode *aCadLayerNode)
wxXmlNode * addPackage(wxXmlNode *aStepNode, FOOTPRINT *aFootprint)
void generateComponents(wxXmlNode *aStepNode)
bool addContourNode(wxXmlNode *aParentNode, const SHAPE_POLY_SET &aPolySet, int aOutline=0, FILL_T aFillType=FILL_T::FILLED_SHAPE, int aWidth=0, LINE_STYLE aDashType=LINE_STYLE::SOLID)
wxString componentName(FOOTPRINT *aFootprint)
bool addOutlineNode(wxXmlNode *aParentNode, const SHAPE_POLY_SET &aPolySet, int aWidth=0, LINE_STYLE aDashType=LINE_STYLE::SOLID)
void generateStackup(wxXmlNode *aCadLayerNode)
std::map< std::tuple< auxLayerType, PCB_LAYER_ID, PCB_LAYER_ID >, std::vector< BOARD_ITEM * > > m_auxilliary_Layers
bool addPolygonNode(wxXmlNode *aParentNode, const SHAPE_LINE_CHAIN &aPolygon, FILL_T aFillType=FILL_T::FILLED_SHAPE, int aWidth=0, LINE_STYLE aDashType=LINE_STYLE::SOLID)
void generateLayerSetNet(wxXmlNode *aLayerNode, PCB_LAYER_ID aLayer, std::vector< BOARD_ITEM * > &aItems)
bool isValidLayerFor2581(PCB_LAYER_ID aLayer)
void insertNodeAfter(wxXmlNode *aPrev, wxXmlNode *aNode)
void generateCadLayers(wxXmlNode *aCadLayerNode)
std::vector< wxXmlNode * > m_padstacks
wxString pinName(const PAD *aPad) const
void generateCadSpecs(wxXmlNode *aCadLayerNode)
void addVia(wxXmlNode *aContentNode, const PCB_VIA *aVia, PCB_LAYER_ID aLayer)
void addSlotCavity(wxXmlNode *aContentNode, const PAD &aPad, const wxString &aName)
wxXmlNode * m_last_padstack
size_t lineHash(int aWidth, LINE_STYLE aDashType)
std::map< size_t, wxString > m_std_shape_dict
void addAttribute(wxXmlNode *aNode, const wxString &aName, const wxString &aValue)
wxXmlNode * m_shape_user_node
wxXmlNode * generateAvlSection()
Creates the Approved Vendor List section.
wxXmlNode * generateHistorySection()
Creates the history section.
void addXY(wxXmlNode *aNode, const VECTOR2I &aVec, const char *aXName=nullptr, const char *aYName=nullptr)
wxXmlNode * m_shape_std_node
void addPadStack(wxXmlNode *aContentNode, const PAD *aPad)
std::map< FOOTPRINT *, wxString > m_OEMRef_dict
void clearLoadedFootprints()
Frees the memory allocated for the loaded footprints in m_loaded_footprints.
wxString m_units_str
std::map< std::pair< PCB_LAYER_ID, PCB_LAYER_ID >, std::vector< PAD * > > m_slot_holes
std::map< size_t, wxString > m_footprint_dict
wxXmlNode * generateLogisticSection()
Creates the logistical data header.
std::map< size_t, wxString > m_line_dict
void addLineDesc(wxXmlNode *aNode, int aWidth, LINE_STYLE aDashType, bool aForce=false)
void addKnockoutText(wxXmlNode *aContentNode, PCB_TEXT *aText)
std::map< size_t, wxString > m_padstack_dict
std::map< std::pair< PCB_LAYER_ID, PCB_LAYER_ID >, std::vector< BOARD_ITEM * > > m_drill_layers
std::map< wxString, FOOTPRINT * > m_footprint_refdes_dict
void generateProfile(wxXmlNode *aStepNode)
void generateLayerFeatures(wxXmlNode *aStepNode)
void SaveBoard(const wxString &aFileName, BOARD *aBoard, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Write aBoard to a storage file in a format that this PCB_IO implementation knows about or it can be u...
wxXmlNode * generateBOMSection(wxXmlNode *aEcadNode)
Creates the BOM section.
wxXmlNode * generateContentStackup(wxXmlNode *aContentNode)
void addShape(wxXmlNode *aContentNode, const PCB_SHAPE &aShape)
void generateAuxilliaryLayers(wxXmlNode *aCadLayerNode)
void addCadHeader(wxXmlNode *aEcadNode)
void generateLayerSetAuxilliary(wxXmlNode *aStepNode)
std::vector< FOOTPRINT * > GetImportedCachedLibraryFootprints() override
Return a container with the cached library footprints generated in the last call to Load.
wxString genLayersString(PCB_LAYER_ID aTop, PCB_LAYER_ID aBottom, const char *aPrefix) const
void generateLogicalNets(wxXmlNode *aStepNode)
void addFillDesc(wxXmlNode *aNode, FILL_T aFillType, bool aForce=false)
std::map< size_t, wxString > m_user_shape_dict
size_t shapeHash(const PCB_SHAPE &aShape)
~PCB_IO_IPC2581() override
wxString genString(const wxString &aStr, const char *aPrefix=nullptr) const
wxXmlNode * m_xml_root
std::vector< FOOTPRINT * > m_loaded_footprints
bool addPolygonCutouts(wxXmlNode *aParentNode, const SHAPE_POLY_SET::POLYGON &aPolygon)
std::set< wxUniChar > m_acceptable_chars
void addLayerAttributes(wxXmlNode *aNode, PCB_LAYER_ID aLayer)
wxXmlNode * generateXmlHeader()
Creates the XML header for IPC-2581.
wxXmlNode * generateEcadSection()
Creates the ECAD section.
wxXmlDocument * m_xml_doc
wxXmlNode * insertNode(wxXmlNode *aParent, const wxString &aName)
void addPad(wxXmlNode *aContentNode, const PAD *aPad, PCB_LAYER_ID aLayer)
void generateStepSection(wxXmlNode *aCadNode)
std::map< PCB_LAYER_ID, wxString > m_layer_name_map
void addLocationNode(wxXmlNode *aContentNode, double aX, double aY)
wxXmlNode * m_line_node
std::map< int, std::vector< std::pair< wxString, wxString > > > m_net_pin_dict
std::map< FOOTPRINT *, wxString > m_footprint_refdes_reverse_dict
wxXmlNode * m_enterpriseNode
VECTOR2I GetCenter() const override
This defaults to the center of the bounding box if not overridden.
Definition: pcb_shape.h:81
STROKE_PARAMS GetStroke() const override
Definition: pcb_shape.h:91
VECTOR2I GetPosition() const override
Definition: pcb_shape.h:79
bool IsBorderEnabled() const
Disables the border, this is done by changing the stroke internally.
wxString GetShownText(bool aAllowExtraText, int aDepth=0) const override
Return the string actually shown after processing of the base text.
Definition: pcb_text.cpp:140
void TransformTextToPolySet(SHAPE_POLY_SET &aBuffer, int aClearance, int aMaxError, ERROR_LOC aErrorLoc) const
Function TransformTextToPolySet Convert the text to a polygonSet describing the actual character stro...
Definition: pcb_text.cpp:514
const VECTOR2I & GetStart() const
Definition: pcb_track.h:152
const VECTOR2I & GetEnd() const
Definition: pcb_track.h:149
virtual int GetWidth() const
Definition: pcb_track.h:146
PCB_LAYER_ID BottomLayer() const
Definition: pcb_track.cpp:1349
VECTOR2I GetPosition() const override
Definition: pcb_track.h:543
bool FlashLayer(int aLayer) const
Check to see whether the via should have a pad on the specific layer.
Definition: pcb_track.cpp:1385
const PADSTACK & Padstack() const
Definition: pcb_track.h:446
int GetWidth() const override
Definition: pcb_track.cpp:380
PCB_LAYER_ID TopLayer() const
Definition: pcb_track.cpp:1343
int GetDrillValue() const
Calculate the drill value for vias (m_drill if > 0, or default drill value for the board).
Definition: pcb_track.cpp:634
virtual LSET GetLayerSet() const override
Return a std::bitset of all layers on which the item physically resides.
Definition: pcb_track.cpp:1222
virtual void SetNumPhases(int aNumPhases)=0
Set the number of phases.
virtual void Report(const wxString &aMessage)=0
Display aMessage in the progress bar dialog.
virtual void BeginPhase(int aPhase)=0
Initialize the aPhase virtual zone of the dialog progress bar.
virtual void AdvancePhase()=0
Use the next available virtual zone of the dialog progress bar.
virtual void SetCurrentProgress(double aProgress)=0
Set the progress value to aProgress (0..1).
Represent a polyline containing arcs as well as line segments: A chain of connected line and/or arc s...
int Width() const
Get the current width of the segments in the chain.
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 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 Fracture()
Convert a set of polygons with holes to a single outline with "slits"/"fractures" connecting the oute...
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)
std::vector< SHAPE_LINE_CHAIN > POLYGON
represents a single polygon outline with holes.
SHAPE_LINE_CHAIN & Outline(int aIndex)
Return the reference to aIndex-th outline in the set.
int NewOutline()
Creates a new empty polygon in the set and returns its index.
int OutlineCount() const
Return the number of outlines in the set.
void InflateWithLinkedHoles(int aFactor, CORNER_STRATEGY aCornerStrategy, int aMaxError)
Perform outline inflation/deflation, using round corners.
int GetWidth() const
LINE_STYLE GetLineStyle() const
Handle a list of polygons defining a copper zone.
Definition: zone.h:74
void SetProgressCallback(std::function< void(size_t)> aCallback)
static bool empty(const wxTextEntryBase *aCtrl)
#define _(s)
static constexpr EDA_ANGLE ANGLE_0
Definition: eda_angle.h:401
static constexpr EDA_ANGLE ANGLE_180
Definition: eda_angle.h:405
FILL_T
Definition: eda_shape.h:56
@ FP_SMD
Definition: footprint.h:81
@ FP_EXCLUDE_FROM_BOM
Definition: footprint.h:83
@ FP_THROUGH_HOLE
Definition: footprint.h:80
static const wxChar traceIpc2581[]
This program source code file is part of KiCad, a free EDA CAD application.
static constexpr void hash_combine(std::size_t &seed)
This is a dummy function to take the final case of hash_combine below.
Definition: hash.h:32
static constexpr std::size_t hash_val(const Types &... args)
Definition: hash.h:51
size_t hash_fp_item(const EDA_ITEM *aItem, int aFlags)
Calculate hash of an EDA_ITEM.
Definition: hash_eda.cpp:55
Hashing functions for EDA_ITEMs.
@ HASH_POS
Definition: hash_eda.h:47
@ REL_COORD
Use coordinates relative to the parent object.
Definition: hash_eda.h:50
PCB_LAYER_ID FlipLayer(PCB_LAYER_ID aLayerId, int aCopperLayersCount)
Definition: layer_id.cpp:169
bool IsFrontLayer(PCB_LAYER_ID aLayerId)
Layer classification: check if it's a front layer.
Definition: layer_ids.h:765
bool IsCopperLayer(int aLayerId)
Test whether a layer is a copper layer.
Definition: layer_ids.h:663
PCB_LAYER_ID
A quick note on layer IDs:
Definition: layer_ids.h:60
@ User_8
Definition: layer_ids.h:131
@ F_CrtYd
Definition: layer_ids.h:116
@ B_Adhes
Definition: layer_ids.h:103
@ Edge_Cuts
Definition: layer_ids.h:112
@ Dwgs_User
Definition: layer_ids.h:107
@ F_Paste
Definition: layer_ids.h:104
@ Cmts_User
Definition: layer_ids.h:108
@ User_6
Definition: layer_ids.h:129
@ User_7
Definition: layer_ids.h:130
@ F_Adhes
Definition: layer_ids.h:102
@ B_Mask
Definition: layer_ids.h:98
@ B_Cu
Definition: layer_ids.h:65
@ User_5
Definition: layer_ids.h:128
@ Eco1_User
Definition: layer_ids.h:109
@ F_Mask
Definition: layer_ids.h:97
@ B_Paste
Definition: layer_ids.h:105
@ User_9
Definition: layer_ids.h:132
@ F_Fab
Definition: layer_ids.h:119
@ Margin
Definition: layer_ids.h:113
@ F_SilkS
Definition: layer_ids.h:100
@ B_CrtYd
Definition: layer_ids.h:115
@ UNDEFINED_LAYER
Definition: layer_ids.h:61
@ Eco2_User
Definition: layer_ids.h:110
@ User_3
Definition: layer_ids.h:126
@ User_1
Definition: layer_ids.h:124
@ B_SilkS
Definition: layer_ids.h:101
@ User_4
Definition: layer_ids.h:127
@ User_2
Definition: layer_ids.h:125
@ F_Cu
Definition: layer_ids.h:64
@ B_Fab
Definition: layer_ids.h:118
bool IsValidLayer(int aLayerId)
Test whether a given integer is a valid layer index, i.e.
Definition: layer_ids.h:641
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
Definition: eda_angle.h:390
see class PGM_BASE
std::vector< FAB_LAYER_COLOR > dummy
const std::vector< FAB_LAYER_COLOR > & GetStandardColors(BOARD_STACKUP_ITEM_TYPE aType)
#define KEY_CORE
#define TO_UTF8(wxstring)
Convert a wxString to a UTF8 encoded C string for all wxWidgets build modes.
Definition: string_utils.h:403
LINE_STYLE
Dashed line types.
Definition: stroke_params.h:46
@ PCB_SHAPE_T
class PCB_SHAPE, a segment not on copper layers
Definition: typeinfo.h:88
@ PCB_DIM_ORTHOGONAL_T
class PCB_DIM_ORTHOGONAL, a linear dimension constrained to x/y
Definition: typeinfo.h:105
@ PCB_DIM_LEADER_T
class PCB_DIM_LEADER, a leader dimension (graphic item)
Definition: typeinfo.h:102
@ PCB_VIA_T
class PCB_VIA, a via (like a track segment on a copper layer)
Definition: typeinfo.h:97
@ PCB_DIM_CENTER_T
class PCB_DIM_CENTER, a center point marking (graphic item)
Definition: typeinfo.h:103
@ PCB_TEXTBOX_T
class PCB_TEXTBOX, wrapped text on a layer
Definition: typeinfo.h:93
@ PCB_ZONE_T
class ZONE, a copper pour area
Definition: typeinfo.h:107
@ PCB_TEXT_T
class PCB_TEXT, text on a layer
Definition: typeinfo.h:92
@ PCB_FIELD_T
class PCB_FIELD, text associated with a footprint property
Definition: typeinfo.h:90
@ PCB_TARGET_T
class PCB_TARGET, a target (graphic item)
Definition: typeinfo.h:106
@ PCB_DIM_ALIGNED_T
class PCB_DIM_ALIGNED, a linear dimension (graphic item)
Definition: typeinfo.h:101
@ PCB_PAD_T
class PAD, a pad in a footprint
Definition: typeinfo.h:87
@ PCB_ARC_T
class PCB_ARC, an arc track segment on a copper layer
Definition: typeinfo.h:98
@ PCB_DIMENSION_T
class PCB_DIMENSION_BASE: abstract dimension meta-type
Definition: typeinfo.h:100
@ PCB_TRACE_T
class PCB_TRACK, a track segment (segment on a copper layer)
Definition: typeinfo.h:96
@ PCB_DIM_RADIAL_T
class PCB_DIM_RADIAL, a radius or diameter dimension
Definition: typeinfo.h:104
VECTOR2< int32_t > VECTOR2I
Definition: vector2d.h:695
VECTOR2< double > VECTOR2D
Definition: vector2d.h:694