KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
gbr_metadata.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2019 Jean-Pierre Charras, jp.charras at wanadoo.fr
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, you may find one here:
19 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
20 * or you may search the http://www.gnu.org website for the version 2 license,
21 * or you may write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 */
24
30#include <wx/string.h>
31#include <wx/datetime.h>
32
33#include <gbr_metadata.h>
34#include <core/utf8.h>
35
36
38{
39 // creates the CreationDate attribute:
40 // The attribute value must conform to the full version of the ISO 8601
41 // date and time format, including time and time zone. Note that this is
42 // the date the Gerber file was effectively created,
43 // not the time the project of PCB was started
44 wxDateTime date( wxDateTime::GetTimeNow() );
45
46 // Date format: see http://www.cplusplus.com/reference/ctime/strftime
47 wxString timezone_offset; // ISO 8601 offset from UTC in timezone
48 timezone_offset = date.Format( "%z" ); // Extract the time zone offset
49
50 // The time zone offset format is +mm or +hhmm (or -mm or -hhmm)
51 // (mm = number of minutes, hh = number of hours. 1h00mn is returned as +0100)
52 // we want +(or -) hh:mm
53 if( timezone_offset.Len() > 3 ) // format +hhmm or -hhmm found
54 // Add separator between hours and minutes
55 timezone_offset.insert( 3, ":", 1 );
56
57 wxString msg;
58
59 switch( aFormat )
60 {
62 msg.Printf( wxS( "%%TF.CreationDate,%s%s*%%" ), date.FormatISOCombined(), timezone_offset );
63 break;
64
66 msg.Printf( wxS( "G04 #@! TF.CreationDate,%s%s*" ), date.FormatISOCombined(),
67 timezone_offset );
68 break;
69
71 msg.Printf( wxS( "%s%s" ), date.FormatISOCombined(), timezone_offset );
72 break;
73
75 msg.Printf( wxS( "; #@! TF.CreationDate,%s%s" ), date.FormatISOCombined(),
76 timezone_offset );
77 break;
78 }
79
80 return msg;
81}
82
83
84wxString GbrMakeProjectGUIDfromString( const wxString& aText )
85{
86 /* Gerber GUID format should be RFC4122 Version 1 or 4.
87 * See en.wikipedia.org/wiki/Universally_unique_identifier
88 * The format is:
89 * xxxxxxxx-xxxx-Mxxx-Nxxx-xxxxxxxxxxxx
90 * with
91 * x = hexDigit lower/upper case
92 * and
93 * M = '1' or '4' (UUID version: 1 (basic) or 4 (random)) (we use 4: UUID random)
94 * and
95 * N = '8' or '9' or 'A|a' or 'B|b' : UUID variant 1: 2 MSB bits have meaning) (we use N = 9)
96 * N = 1000 or 1001 or 1010 or 1011 : 10xx means Variant 1 (Variant2: 110x and 111x are
97 * reserved)
98 */
99
100 wxString guid;
101
102 // Build a 32 digits GUID from the board name:
103 // guid has 32 digits, so add chars in name to be sure we can build a 32 digits guid
104 // (i.e. from a 16 char string name)
105 // In fact only 30 digits are used, and 2 UID id
106 wxString bname = aText;
107 int cnt = 16 - bname.Len();
108
109 if( cnt > 0 )
110 bname.Append( 'X', cnt );
111
112 int chr_idx = 0;
113
114 // Output the 8 first hex digits:
115 for( unsigned ii = 0; ii < 4; ii++ )
116 {
117 int cc = int( bname[chr_idx++] ) & 0xFF;
118 guid << wxString::Format( "%2.2x", cc );
119 }
120
121 // Output the 4 next hex digits:
122 guid << '-';
123
124 for( unsigned ii = 0; ii < 2; ii++ )
125 {
126 int cc = int( bname[chr_idx++] ) & 0xFF;
127 guid << wxString::Format( "%2.2x", cc );
128 }
129
130 // Output the 4 next hex digits (UUID version and 3 digits):
131 guid << "-4"; // first digit: UUID version 4 (M = 4)
132 {
133 int cc = int( bname[chr_idx++] ) << 4 & 0xFF0;
134 cc += int( bname[chr_idx] ) >> 4 & 0x0F;
135 guid << wxString::Format( "%3.3x", cc );
136 }
137
138 // Output the 4 next hex digits (UUID variant and 3 digits):
139 guid << "-9"; // first digit: UUID variant 1 (N = 9)
140 {
141 int cc = (int( bname[chr_idx++] ) & 0x0F) << 8;
142 cc += int( bname[chr_idx++] ) & 0xFF;
143 guid << wxString::Format( "%3.3x", cc );
144 }
145
146 // Output the 12 last hex digits:
147 guid << '-';
148
149 for( unsigned ii = 0; ii < 6; ii++ )
150 {
151 int cc = int( bname[chr_idx++] ) & 0xFF;
152 guid << wxString::Format( "%2.2x", cc );
153 }
154
155 return guid;
156}
157
158
160 bool aUseX1StructuredComment,
161 const std::string& aCustomAttribute )
162{
163 std::string attribute_string; // the specific aperture attribute (TA.xxx)
164 std::string comment_string; // a optional G04 comment line to write before the TA. line
165
166 // generate a string to print a Gerber Aperture attribute
167 switch( aAttribute )
168 {
169 // Dummy value (aAttribute must be < GBR_APERTURE_ATTRIB_END).
171 case GBR_APERTURE_ATTRIB_NONE: // idle command: do nothing
172 break;
173
174 case GBR_APERTURE_ATTRIB_ETCHEDCMP: // print info associated to an item
175 // which connects 2 different nets
176 // (Net tees, microwave component)
177 attribute_string = "TA.AperFunction,EtchedComponent";
178 break;
179
180 case GBR_APERTURE_ATTRIB_CONDUCTOR: // print info associated to a track
181 attribute_string = "TA.AperFunction,Conductor";
182 break;
183
184 case GBR_APERTURE_ATTRIB_EDGECUT: // print info associated to a board outline
185 attribute_string = "TA.AperFunction,Profile";
186 break;
187
188 case GBR_APERTURE_ATTRIB_VIAPAD: // print info associated to a flashed via
189 attribute_string = "TA.AperFunction,ViaPad";
190 break;
191
192 case GBR_APERTURE_ATTRIB_NONCONDUCTOR: // print info associated to a item on a copper layer
193 // which is not a track (for instance a text)
194 attribute_string = "TA.AperFunction,NonConductor";
195 break;
196
197 case GBR_APERTURE_ATTRIB_COMPONENTPAD: // print info associated to a flashed
198 // through hole component on outer layer
199 attribute_string = "TA.AperFunction,ComponentPad";
200 break;
201
202 case GBR_APERTURE_ATTRIB_SMDPAD_SMDEF: // print info associated to a flashed for SMD pad.
203 // with solder mask defined from the copper shape
204 // Excluded BGA pads which have their own type
205 attribute_string = "TA.AperFunction,SMDPad,SMDef";
206 break;
207
208 case GBR_APERTURE_ATTRIB_SMDPAD_CUDEF: // print info associated to a flashed SMD pad with
209 // a solder mask defined by the solder mask
210 attribute_string = "TA.AperFunction,SMDPad,CuDef";
211 break;
212
213 case GBR_APERTURE_ATTRIB_BGAPAD_SMDEF: // print info associated to flashed BGA pads with
214 // a solder mask defined by the copper shape
215 attribute_string = "TA.AperFunction,BGAPad,SMDef";
216 break;
217
218 case GBR_APERTURE_ATTRIB_BGAPAD_CUDEF: // print info associated to a flashed BGA pad with
219 // a solder mask defined by the solder mask
220 attribute_string = "TA.AperFunction,BGAPad,CuDef";
221 break;
222
224 // print info associated to a flashed edge connector pad (outer layers)
225 attribute_string = "TA.AperFunction,ConnectorPad";
226 break;
227
229 // print info associated to flashed mechanical pads (NPTH)
230 attribute_string = "TA.AperFunction,WasherPad";
231 break;
232
233 case GBR_APERTURE_ATTRIB_HEATSINKPAD: // print info associated to a flashed heat sink pad
234 // (typically for SMDs)
235 attribute_string = "TA.AperFunction,HeatsinkPad";
236 break;
237
238 case GBR_APERTURE_ATTRIB_TESTPOINT: // print info associated to a flashed test point pad
239 // (typically for SMDs)
240 attribute_string = "TA.AperFunction,TestPad";
241 break;
242
243 case GBR_APERTURE_ATTRIB_FIDUCIAL_GLBL: // print info associated to a flashed fiducial pad
244 // (typically for SMDs)
245 attribute_string = "TA.AperFunction,FiducialPad,Global";
246 break;
247
248 case GBR_APERTURE_ATTRIB_FIDUCIAL_LOCAL: // print info associated to a flashed fiducial pad
249 // (typically for SMDs)
250 attribute_string = "TA.AperFunction,FiducialPad,Local";
251 break;
252
254 // print info associated to a flashed castellated pad (typically for SMDs)
255 attribute_string = "TA.AperFunction,CastellatedPad";
256 break;
257
259 // print info associated to a flashed castellated pad in drill files
260 attribute_string = "TA.AperFunction,CastellatedDrill";
261 break;
262
263 case GBR_APERTURE_ATTRIB_VIADRILL: // print info associated to a via hole in drill files
264 attribute_string = "TA.AperFunction,ViaDrill";
265 break;
266
267 case GBR_APERTURE_ATTRIB_CMP_DRILL: // print info associated to a component
268 // round pad hole in drill files
269 attribute_string = "TA.AperFunction,ComponentDrill";
270 break;
271
272 // print info associated to a component oblong pad hole in drill files
273 // Same as a round pad hole, but is a specific aperture in drill file and
274 // a G04 comment is added to the aperture function
276 comment_string = "aperture for slot hole";
277 attribute_string = "TA.AperFunction,ComponentDrill";
278 break;
279
280 case GBR_APERTURE_ATTRIB_CMP_POSITION: // print info associated to a component
281 // flashed shape at the component position
282 // in placement files
283 attribute_string = "TA.AperFunction,ComponentMain";
284 break;
285
286 case GBR_APERTURE_ATTRIB_PAD1_POS: // print info associated to a component
287 // flashed shape at pad 1 position
288 // (pad 1 is also pad A1 or pad AA1)
289 // in placement files
290 attribute_string = "TA.AperFunction,ComponentPin";
291 break;
292
293 case GBR_APERTURE_ATTRIB_PADOTHER_POS: // print info associated to a component
294 // flashed shape at pads position (all but pad 1)
295 // in placement files
296 // Currently: (could be changed later) same as
297 // GBR_APERTURE_ATTRIB_PADOTHER_POS
298 attribute_string = "TA.AperFunction,ComponentPin";
299 break;
300
301 case GBR_APERTURE_ATTRIB_CMP_BODY: // print info associated to a component
302 // print the component physical body
303 // polygon in placement files
304 attribute_string = "TA.AperFunction,ComponentOutline,Body";
305 break;
306
307 case GBR_APERTURE_ATTRIB_CMP_LEAD2LEAD: // print info associated to a component
308 // print the component physical lead to lead
309 // polygon in placement files
310 attribute_string = "TA.AperFunction,ComponentOutline,Lead2Lead";
311 break;
312
313 case GBR_APERTURE_ATTRIB_CMP_FOOTPRINT: // print info associated to a component
314 // print the component footprint bounding box
315 // polygon in placement files
316 attribute_string = "TA.AperFunction,ComponentOutline,Footprint";
317 break;
318
319 case GBR_APERTURE_ATTRIB_CMP_COURTYARD: // print info associated to a component
320 // print the component courtyard
321 // polygon in placement files
322 attribute_string = "TA.AperFunction,ComponentOutline,Courtyard";
323 break;
324
326 attribute_string = "TA.AperFunction,Other," + aCustomAttribute;
327 break;
328 }
329
330 std::string full_attribute_string;
331 wxString eol_string;
332
333 if( !attribute_string.empty() )
334 {
335 if( !comment_string.empty() )
336 {
337 full_attribute_string = "G04 " + comment_string + "*\n";
338 }
339
340 if( aUseX1StructuredComment )
341 {
342 full_attribute_string += "G04 #@! ";
343 eol_string = "*\n";
344 }
345 else
346 {
347 full_attribute_string += "%";
348 eol_string = "*%\n";
349 }
350 }
351
352 full_attribute_string += attribute_string + eol_string;
353
354 return full_attribute_string;
355}
356
357
358// Helper function to convert a ascii hex char to its integer value
359// If the char is not a hexa char, return -1
360int char2Hex( unsigned aCode )
361{
362 if( aCode >= '0' && aCode <= '9' )
363 return aCode - '0';
364
365 if( aCode >= 'A' && aCode <= 'F' )
366 return aCode - 'A' + 10;
367
368 if( aCode >= 'a' && aCode <= 'f' )
369 return aCode - 'a' + 10;
370
371 return -1;
372}
373
374
375wxString FormatStringFromGerber( const wxString& aString )
376{
377 // make the inverse conversion of FormatStringToGerber()
378 // It converts a "normalized" gerber string containing escape sequences
379 // and convert it to a 16 bits Unicode char
380 // and return a wxString (Unicode 16) from the gerber string
381 // Note the initial gerber string can already contain Unicode chars.
382 wxString txt; // The string converted from Gerber string
383
384 unsigned count = aString.Length();
385
386 for( unsigned ii = 0; ii < count; ++ii )
387 {
388 unsigned code = aString[ii];
389
390 if( code == '\\' && ii < count-5 && aString[ii+1] == 'u' )
391 {
392 // Note the latest Gerber X2 spec (2019 06) uses \uXXXX to encode
393 // the Unicode XXXX hexadecimal value
394 // If 4 chars next to 'u' are hexadecimal chars,
395 // Convert these 4 hexadecimal digits to a 16 bit Unicode
396 // (Gerber allows only 4 hexadecimal digits)
397 // If an error occurs, the escape sequence is not translated,
398 // and used "as this"
399 long value = 0;
400 bool error = false;
401
402 for( int jj = 0; jj < 4; jj++ )
403 {
404 value <<= 4;
405 code = aString[ii+jj+2];
406
407 int hexa = char2Hex( code );
408
409 if( hexa >= 0 )
410 value += hexa;
411 else
412 {
413 error = true;
414 break;
415 }
416 }
417
418 if( !error )
419 {
420 if( value >= ' ' ) // Is a valid wxChar ?
421 txt.Append( wxChar( value ) );
422
423 ii += 5;
424 }
425 else
426 {
427 txt.Append( aString[ii] );
428 continue;
429 }
430 }
431 else
432 {
433 txt.Append( aString[ii] );
434 }
435 }
436
437 return txt;
438}
439
440
441wxString ConvertNotAllowedCharsInGerber( const wxString& aString, bool aAllowUtf8Chars,
442 bool aQuoteString )
443{
444 /* format string means convert any code > 0x7E and unauthorized codes to a hexadecimal
445 * 16 bits sequence Unicode
446 * However if aAllowUtf8Chars is true only unauthorized codes will be escaped, because some
447 * Gerber files accept UTF8 chars.
448 * unauthorized codes are ',' '*' '%' '\' '"' and are used as separators in Gerber files
449 */
450 wxString txt;
451
452 if( aQuoteString )
453 txt << "\"";
454
455 for( unsigned ii = 0; ii < aString.Length(); ++ii )
456 {
457 wxChar code = aString[ii];
458 bool convert = false;
459
460 switch( code )
461 {
462 case '\\':
463 case '%':
464 case '*':
465 case ',':
466 convert = true;
467 break;
468
469 case '"':
470 if( aQuoteString )
471 convert = true;
472 break;
473
474 default:
475 break;
476 }
477
478 if( !aAllowUtf8Chars && code > 0x7F )
479 convert = true;
480
481 if( convert )
482 {
483 // Convert code to 4 hexadecimal digit
484 // (Gerber allows only 4 hexadecimal digit) in escape seq:
485 // "\uXXXX", XXXX is the Unicode 16 bits hexa value
486 char hexa[32];
487 std::snprintf( hexa, sizeof( hexa ), "\\u%4.4X", code & 0xFFFF );
488 txt += hexa;
489 }
490 else
491 {
492 txt += code;
493 }
494 }
495
496 if( aQuoteString )
497 txt << "\"";
498
499 return txt;
500}
501
502
504{
505 wxString converted;
506
507 if( !m_field.IsEmpty() )
509
510 // Convert the char string to std::string. Be careful when converting a wxString to
511 // a std::string: using static_cast<const char*> is mandatory
512 std::string txt = static_cast<const char*>( converted.utf8_str() );
513
514 return txt;
515}
516
517
518std::string FormatStringToGerber( const wxString& aString )
519{
520 wxString converted;
521
522 /* format string means convert any code > 0x7E and unauthorized codes to a hexadecimal
523 * 16 bits sequence Unicode
524 * unauthorized codes are ',' '*' '%' '\'
525 * This conversion is not made for quoted strings, because if the string is
526 * quoted, the conversion is expected to be already made, and the returned string must use
527 * UTF8 encoding
528 */
529 if( !aString.IsEmpty() && ( aString[0] != '\"' || aString[aString.Len()-1] != '\"' ) )
530 converted = ConvertNotAllowedCharsInGerber( aString, false, false );
531 else
532 converted = aString;
533
534 // Convert the char string to std::string. Be careful when converting a wxString to
535 // a std::string: using static_cast<const char*> is mandatory
536 std::string txt = static_cast<const char*>( converted.utf8_str() );
537
538 return txt;
539}
540
541
542// Netname and Pan num fields cannot be empty in Gerber files
543// Normalized names must be used, if any
544#define NO_NET_NAME wxT( "N/C" ) // net name of not connected pads (one pad net) (normalized)
545#define NO_PAD_NAME wxT( "" ) // pad name of pads without pad name/number (not normalized)
546
547
548bool FormatNetAttribute( std::string& aPrintedText, std::string& aLastNetAttributes,
549 const GBR_NETLIST_METADATA* aData, bool& aClearPreviousAttributes,
550 bool aUseX1StructuredComment )
551{
552 aClearPreviousAttributes = false;
553 wxString prepend_string;
554 wxString eol_string;
555
556 if( aUseX1StructuredComment )
557 {
558 prepend_string = "G04 #@! ";
559 eol_string = "*\n";
560 }
561 else
562 {
563 prepend_string = "%";
564 eol_string = "*%\n";
565 }
566
567 // print a Gerber net attribute record.
568 // it is added to the object attributes dictionary
569 // On file, only modified or new attributes are printed.
570 if( aData == nullptr )
571 return false;
572
573 std::string pad_attribute_string;
574 std::string net_attribute_string;
575 std::string cmp_attribute_string;
576
578 return false; // idle command: do nothing
579
581 {
582 // print info associated to a flashed pad (cmpref, pad name, and optionally pin function)
583 // example1: %TO.P,R5,3*%
584 // example2: %TO.P,R5,3,reset*%
585 pad_attribute_string = prepend_string + "TO.P,";
586 pad_attribute_string += FormatStringToGerber( aData->m_Cmpref ) + ",";
587
588 if( aData->m_Padname.IsEmpty() )
589 {
590 // Happens for "mechanical" or never connected pads
591 pad_attribute_string += FormatStringToGerber( NO_PAD_NAME );
592 }
593 else
594 {
595 pad_attribute_string += aData->m_Padname.GetGerberString();
596
597 // In Pcbnew, the pin function comes from the schematic.
598 // so it exists only for named pads
599 if( !aData->m_PadPinFunction.IsEmpty() )
600 {
601 pad_attribute_string += ',';
602 pad_attribute_string += aData->m_PadPinFunction.GetGerberString();
603 }
604 }
605
606 pad_attribute_string += eol_string;
607 }
608
610 {
611 // print info associated to a net
612 // example: %TO.N,Clk3*%
613 net_attribute_string = prepend_string + "TO.N,";
614
615 if( aData->m_Netname.IsEmpty() )
616 {
617 if( aData->m_NotInNet )
618 {
619 // Happens for not connectable pads: mechanical pads
620 // and pads with no padname/num
621 // In this case the net name must be left empty
622 }
623 else
624 {
625 // Happens for not connected pads: use a normalized
626 // dummy name
627 net_attribute_string += FormatStringToGerber( NO_NET_NAME );
628 }
629 }
630 else
631 {
632 net_attribute_string += FormatStringToGerber( aData->m_Netname );
633 }
634
635 net_attribute_string += eol_string;
636 }
637
640 {
641 // print info associated to a footprint
642 // example: %TO.C,R2*%
643 // Because GBR_NETINFO_PAD option already contains this info, it is not
644 // created here for a GBR_NETINFO_PAD attribute
645 cmp_attribute_string = prepend_string + "TO.C,";
646 cmp_attribute_string += FormatStringToGerber( aData->m_Cmpref ) + eol_string;
647 }
648
649 // the full list of requested attributes:
650 std::string full_attribute_string = pad_attribute_string + net_attribute_string
651 + cmp_attribute_string;
652 // the short list of requested attributes
653 // (only modified or new attributes are stored here):
654 std::string short_attribute_string;
655
656 // Attributes have changed: update attribute string, and see if the previous attribute
657 // list (dictionary in Gerber language) must be cleared
658 if( aLastNetAttributes != full_attribute_string )
659 {
660 // first, remove no longer existing attributes.
661 // Because in KiCad the full attribute list is evaluated for each object,
662 // the entire dictionary is cleared
663 // If m_TryKeepPreviousAttributes is true, only the no longer existing attribute
664 // is cleared.
665 // Note: to avoid interaction between clear attributes and set attributes
666 // the clear attribute is inserted first.
667 bool clearDict = false;
668
669 if( aLastNetAttributes.find( "TO.P," ) != std::string::npos )
670 {
671 if( pad_attribute_string.empty() ) // No more this attribute
672 {
673 if( aData->m_TryKeepPreviousAttributes ) // Clear only this attribute
674 short_attribute_string.insert( 0, prepend_string + "TO.P" + eol_string );
675 else
676 clearDict = true;
677 }
678 else if( aLastNetAttributes.find( pad_attribute_string ) == std::string::npos )
679 {
680 // This attribute has changed
681 short_attribute_string += pad_attribute_string;
682 }
683 }
684 else // New attribute
685 {
686 short_attribute_string += pad_attribute_string;
687 }
688
689 if( aLastNetAttributes.find( "TO.N," ) != std::string::npos )
690 {
691 if( net_attribute_string.empty() ) // No more this attribute
692 {
693 if( aData->m_TryKeepPreviousAttributes ) // Clear only this attribute
694 short_attribute_string.insert( 0, prepend_string + "TO.N" + eol_string );
695 else
696 clearDict = true;
697 }
698 else if( aLastNetAttributes.find( net_attribute_string ) == std::string::npos )
699 {
700 // This attribute has changed.
701 short_attribute_string += net_attribute_string;
702 }
703 }
704 else // New attribute
705 {
706 short_attribute_string += net_attribute_string;
707 }
708
709 if( aLastNetAttributes.find( "TO.C," ) != std::string::npos )
710 {
711 if( cmp_attribute_string.empty() ) // No more this attribute
712 {
713 if( aData->m_TryKeepPreviousAttributes ) // Clear only this attribute
714 {
715 // Refinement:
716 // the attribute will be cleared only if there is no pad attribute.
717 // If a pad attribute exists, the component name exists so the old
718 // TO.C value will be updated, therefore no need to clear it before updating
719 if( pad_attribute_string.empty() )
720 short_attribute_string.insert( 0, prepend_string + "TO.C" + eol_string );
721 }
722 else
723 {
724 clearDict = true;
725 }
726 }
727 else if( aLastNetAttributes.find( cmp_attribute_string ) == std::string::npos )
728 {
729 // This attribute has changed.
730 short_attribute_string += cmp_attribute_string;
731 }
732 }
733 else // New attribute
734 {
735 short_attribute_string += cmp_attribute_string;
736 }
737
738 aClearPreviousAttributes = clearDict;
739
740 aLastNetAttributes = full_attribute_string;
741
742 if( clearDict )
743 aPrintedText = full_attribute_string;
744 else
745 aPrintedText = short_attribute_string;
746 }
747
748 return true;
749}
750
751
753{
754 // Clear all strings
755 m_Orientation = 0.0;
756 m_Manufacturer.Clear();
757 m_MPN.Clear();
758 m_Package.Clear();
759 m_Value.Clear();
761}
762
763
765{
766 wxString text;
767 wxString start_of_line( "%TO." );
768 wxString end_of_line( "*%\n" );
769
770 wxString mountTypeStrings[] =
771 {
772 "Other", "SMD", "TH"
773 };
774
775 if( !m_Manufacturer.IsEmpty() )
776 text << start_of_line << "CMfr," << m_Manufacturer << end_of_line;
777
778 if( !m_MPN.IsEmpty() )
779 text << start_of_line << "CMPN," << m_MPN << end_of_line;
780
781 if( !m_Package.IsEmpty() )
782 text << start_of_line << "Cpkg," << m_Package << end_of_line;
783
784 if( !m_Footprint.IsEmpty() )
785 text << start_of_line << "CFtp," << m_Footprint << end_of_line;
786
787 if( !m_Value.IsEmpty() )
788 text << start_of_line << "CVal," << m_Value << end_of_line;
789
790 if( !m_LibraryName.IsEmpty() )
791 text << start_of_line << "CLbN," << m_LibraryName << end_of_line;
792
793 if( !m_LibraryDescr.IsEmpty() )
794 text << start_of_line << "CLbD," << m_LibraryDescr << end_of_line;
795
796 text << start_of_line << "CMnt," << mountTypeStrings[m_MountType] << end_of_line;
797 text << start_of_line << "CRot," << m_Orientation << end_of_line;
798
799 return text;
800}
@ GBR_APERTURE_ATTRIB_CMP_DRILL
Aperture used for pad holes in drill files.
Definition: gbr_metadata.h:141
@ GBR_APERTURE_ATTRIB_CONDUCTOR
Aperture used for connected items like tracks (not vias).
Definition: gbr_metadata.h:98
@ GBR_APERTURE_ATTRIB_VIAPAD
Aperture used for vias.
Definition: gbr_metadata.h:103
@ GBR_APERTURE_ATTRIB_ETCHEDCMP
Aperture used for etched components.
Definition: gbr_metadata.h:95
@ GBR_APERTURE_ATTRIB_BGAPAD_CUDEF
Aperture used for BGA pad with a solder mask defined by the solder mask.
Definition: gbr_metadata.h:118
@ GBR_APERTURE_ATTRIB_PAD1_POS
Aperture used for flashed pin 1 (or A1 or AA1) position in placement files.
Definition: gbr_metadata.h:150
@ GBR_APERTURE_ATTRIB_HEATSINKPAD
Aperture used for heat sink pad (typically for SMDs).
Definition: gbr_metadata.h:132
@ GBR_APERTURE_ATTRIB_CMP_OBLONG_DRILL
Aperture used for pads oblong holes in drill files.
Definition: gbr_metadata.h:144
@ GBR_APERTURE_ATTRIB_CMP_COURTYARD
Aperture used to draw component outline courtyard in placement files.
Definition: gbr_metadata.h:165
@ GBR_APERTURE_ATTRIB_TESTPOINT
Aperture used for test point pad (outer layers).
Definition: gbr_metadata.h:123
@ GBR_APERTURE_ATTRIB_SMDPAD_SMDEF
Aperture used for SMD pad. Excluded BGA pads which have their own type.
Definition: gbr_metadata.h:109
@ GBR_APERTURE_ATTRIB_SMDPAD_CUDEF
Aperture used for SMD pad with a solder mask defined by the solder mask.
Definition: gbr_metadata.h:112
@ GBR_APERTURE_ATTRIB_NONE
uninitialized attribute.
Definition: gbr_metadata.h:94
@ GBR_APERTURE_ATTRIB_CONNECTORPAD
Aperture used for edge connector pad (outer layers).
Definition: gbr_metadata.h:121
@ GBR_APERTURE_ATTRIB_CASTELLATEDDRILL
Aperture used for castellated pads in drill files.
Definition: gbr_metadata.h:138
@ GBR_APERTURE_ATTRIB_NONCONDUCTOR
Aperture used for not connected items (texts, outlines on copper).
Definition: gbr_metadata.h:102
@ GBR_APERTURE_ATTRIB_END
sentinel: max value
Definition: gbr_metadata.h:169
@ GBR_APERTURE_ATTRIB_CMP_POSITION
Aperture used for flashed cmp position in placement files.
Definition: gbr_metadata.h:147
@ GBR_APERTURE_ATTRIB_CMP_BODY
Aperture used to draw component physical body outline without pins in placement files.
Definition: gbr_metadata.h:156
@ GBR_APERTURE_ATTRIB_BGAPAD_SMDEF
Aperture used for BGA pads with a solder mask defined by the copper shape.
Definition: gbr_metadata.h:115
@ GBR_APERTURE_ATTRIB_WASHERPAD
Aperture used for mechanical pads (NPTH).
Definition: gbr_metadata.h:122
@ GBR_APERTURE_ATTRIB_PADOTHER_POS
Aperture used for flashed pads position in placement files.
Definition: gbr_metadata.h:153
@ GBR_APERTURE_ATTRIB_COMPONENTPAD
Aperture used for through hole component on outer layer.
Definition: gbr_metadata.h:106
@ GBR_APERTURE_ATTRIB_VIADRILL
Aperture used for via holes in drill files.
Definition: gbr_metadata.h:140
@ GBR_APERTURE_ATTRIB_CMP_FOOTPRINT
Aperture used to draw component footprint bounding box in placement files.
Definition: gbr_metadata.h:162
@ GBR_APERTURE_ATTRIB_CMP_LEAD2LEAD
Aperture used to draw component physical body outline with pins in placement files.
Definition: gbr_metadata.h:159
@ GBR_APERTURE_ATTRIB_FIDUCIAL_GLBL
Aperture used for fiducial pad (outer layers), at board level.
Definition: gbr_metadata.h:126
@ GBR_APERTURE_ATTRIB_CASTELLATEDPAD
Aperture used for castellated pads in copper layer files.
Definition: gbr_metadata.h:135
@ GBR_APERTURE_ATTRIB_FIDUCIAL_LOCAL
Aperture used for fiducial pad (outer layers), at footprint level.
Definition: gbr_metadata.h:129
@ GBR_APERTURE_ATTRIB_EDGECUT
Aperture used for board cutout,.
Definition: gbr_metadata.h:99
static std::string FormatAttribute(GBR_APERTURE_ATTRIB aAttribute, bool aUseX1StructuredComment, const std::string &aCustomAttribute)
wxString FormatCmpPnPMetadata()
One line by non empty data the orientation (.CRot) and mount type (.CMnt) are always generated.
std::string GetGerberString() const
wxString m_field
the Unicode text to print in Gbr file (after escape and quoting)
bool m_useUTF8
true to use UTF8, false to escape non ASCII7 chars
bool m_escapeString
true to quote the field in gbr file
Information which can be added in a gerber file as attribute of an object.
int m_NetAttribType
the type of net info (used to define the gerber string to create)
wxString m_Cmpref
the component reference parent of the data
@ GBR_NETINFO_NET
print info associated to a net (TO.N attribute)
@ GBR_NETINFO_UNSPECIFIED
idle command (no command)
@ GBR_NETINFO_CMP
print info associated to a component (TO.C attribute)
@ GBR_NETINFO_PAD
print info associated to a flashed pad (TO.P attribute)
GBR_DATA_FIELD m_PadPinFunction
for a pad: the pin function (defined in schematic)
bool m_TryKeepPreviousAttributes
If true, do not clear all attributes when a attribute has changed.
bool m_NotInNet
true if a pad of a footprint cannot be connected (for instance a mechanical NPTH, ot a not named pad)...
GBR_DATA_FIELD m_Padname
for a flashed pad: the pad name ((TO.P attribute)
wxString m_Netname
for items associated to a net: the netname
wxString GbrMakeProjectGUIDfromString(const wxString &aText)
Build a project GUID using format RFC4122 Version 1 or 4 from the project name, because a KiCad proje...
#define NO_NET_NAME
#define NO_PAD_NAME
wxString GbrMakeCreationDateAttributeString(GBR_NC_STRING_FORMAT aFormat)
std::string FormatStringToGerber(const wxString &aString)
Normalize aString and convert it to a Gerber std::string.
wxString ConvertNotAllowedCharsInGerber(const wxString &aString, bool aAllowUtf8Chars, bool aQuoteString)
Normalize aString and convert it to a Gerber compatible wxString.
wxString FormatStringFromGerber(const wxString &aString)
Convert a gerber string into a 16 bit Unicode string.
int char2Hex(unsigned aCode)
bool FormatNetAttribute(std::string &aPrintedText, std::string &aLastNetAttributes, const GBR_NETLIST_METADATA *aData, bool &aClearPreviousAttributes, bool aUseX1StructuredComment)
Generate the string to set a net attribute for a graphic object to print to a gerber file.
Handle special data (items attributes) during plot.
GBR_NC_STRING_FORMAT
Create a gerber TF.CreationDate attribute.
Definition: gbr_metadata.h:60
@ GBR_NC_STRING_FORMAT_X1
Definition: gbr_metadata.h:61
@ GBR_NC_STRING_FORMAT_NCDRILL
Definition: gbr_metadata.h:64
@ GBR_NC_STRING_FORMAT_X2
Definition: gbr_metadata.h:62
@ GBR_NC_STRING_FORMAT_GBRJOB
Definition: gbr_metadata.h:63