KiCad PCB EDA Suite
Loading...
Searching...
No Matches
sch_io_kicad_sexpr.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) 2020 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * @author Wayne Stambaugh <[email protected]>
8 *
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version 2
12 * of the License, or (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License along
20 * with this program. If not, see <http://www.gnu.org/licenses/>.
21 */
22
23#include <algorithm>
24
25#include <fmt/format.h>
26
27#include <wx/dir.h>
28#include <wx/log.h>
29#include <wx/mstream.h>
30
31#include <base_units.h>
32#include <bitmap_base.h>
34#include <build_version.h>
35#include <sch_selection.h>
36#include <font/fontconfig.h>
39#include <progress_reporter.h>
40#include <schematic.h>
41#include <schematic_lexer.h>
42#include <sch_bitmap.h>
43#include <sch_bus_entry.h>
44#include <sch_edit_frame.h> // SYMBOL_ORIENTATION_T
45#include <sch_group.h>
50#include <sch_junction.h>
51#include <sch_line.h>
52#include <sch_no_connect.h>
53#include <sch_pin.h>
54#include <sch_rule_area.h>
55#include <sch_screen.h>
56#include <sch_shape.h>
57#include <sch_sheet.h>
58#include <sch_sheet_pin.h>
59#include <sch_symbol.h>
60#include <sch_table.h>
61#include <sch_tablecell.h>
62#include <sch_text.h>
63#include <sch_textbox.h>
64#include <string_utils.h>
65#include <trace_helpers.h>
66#include <reporter.h>
67
68using namespace TSCHEMATIC_T;
69
70
71#define SCH_PARSE_ERROR( text, reader, pos ) \
72 THROW_PARSE_ERROR( text, reader.GetSource(), reader.Line(), \
73 reader.LineNumber(), pos - reader.Line() )
74
75
76SCH_IO_KICAD_SEXPR::SCH_IO_KICAD_SEXPR() : SCH_IO( wxS( "Eeschema s-expression" ) )
77{
78 init( nullptr );
79}
80
81
86
87
89 const std::map<std::string, UTF8>* aProperties )
90{
91 if( m_schematic != aSchematic )
92 m_loadedRootSheets.clear();
93
94 m_version = 0;
95 m_appending = false;
96 m_rootSheet = nullptr;
97 m_schematic = aSchematic;
98 m_cache = nullptr;
99 m_out = nullptr;
100}
101
102
103SCH_SHEET* SCH_IO_KICAD_SEXPR::LoadSchematicFile( const wxString& aFileName, SCHEMATIC* aSchematic,
104 SCH_SHEET* aAppendToMe,
105 const std::map<std::string, UTF8>* aProperties )
106{
107 wxASSERT( !aFileName || aSchematic != nullptr );
108
109 SCH_SHEET* sheet;
110
111 wxFileName fn = aFileName;
112
113 // Collect the font substitution warnings (RAII - automatically reset on scope exit)
115
116 // Unfortunately child sheet file names the legacy schematic file format are not fully
117 // qualified and are always appended to the project path. The aFileName attribute must
118 // always be an absolute path so the project path can be used for load child sheet files.
119 wxASSERT( fn.IsAbsolute() );
120
121 if( aAppendToMe )
122 {
123 m_appending = true;
124 wxLogTrace( traceSchPlugin, "Append \"%s\" to sheet \"%s\".",
125 aFileName, aAppendToMe->GetFileName() );
126
127 wxFileName normedFn = aAppendToMe->GetFileName();
128
129 if( !normedFn.IsAbsolute() )
130 {
131 if( aFileName.Right( normedFn.GetFullPath().Length() ) == normedFn.GetFullPath() )
132 m_path = aFileName.Left( aFileName.Length() - normedFn.GetFullPath().Length() );
133 }
134
135 if( m_path.IsEmpty() )
136 m_path = aSchematic->Project().GetProjectPath();
137
138 wxLogTrace( traceSchPlugin, "Normalized append path \"%s\".", m_path );
139 }
140 else
141 {
142 m_path = aSchematic->Project().GetProjectPath();
143 }
144
145 m_currentPath.push( m_path );
146 init( aSchematic, aProperties );
147
148 if( aAppendToMe == nullptr )
149 {
150 // Clean up any allocated memory if an exception occurs loading the schematic.
151 std::unique_ptr<SCH_SHEET> newSheet = std::make_unique<SCH_SHEET>( aSchematic );
152
153 wxFileName relPath( aFileName );
154
155 // Do not use wxPATH_UNIX as option in MakeRelativeTo(). It can create incorrect
156 // relative paths on Windows, because paths have a disk identifier (C:, D: ...)
157 relPath.MakeRelativeTo( aSchematic->Project().GetProjectPath() );
158
159 newSheet->SetFileName( relPath.GetFullPath() );
160 m_rootSheet = newSheet.get();
161 loadHierarchy( SCH_SHEET_PATH(), newSheet.get() );
162
163 // If we got here, the schematic loaded successfully.
164 sheet = newSheet.release();
165 m_rootSheet = nullptr; // Quiet Coverity warning.
166 m_loadedRootSheets.push_back( sheet );
167 }
168 else
169 {
170 wxCHECK_MSG( aSchematic->IsValid(), nullptr, "Can't append to a schematic with no root!" );
171 m_rootSheet = &aSchematic->Root();
172 sheet = aAppendToMe;
173 loadHierarchy( SCH_SHEET_PATH(), sheet );
174 }
175
176 wxASSERT( m_currentPath.size() == 1 ); // only the project path should remain
177
178 m_currentPath.pop(); // Clear the path stack for next call to Load
179
180 return sheet;
181}
182
183
184// Everything below this comment is recursive. Modify with care.
185
186void SCH_IO_KICAD_SEXPR::loadHierarchy( const SCH_SHEET_PATH& aParentSheetPath, SCH_SHEET* aSheet )
187{
188 m_currentSheetPath.push_back( aSheet );
189
190 SCH_SCREEN* screen = nullptr;
191
192 if( !aSheet->GetScreen() )
193 {
194 // SCH_SCREEN objects store the full path and file name where the SCH_SHEET object only
195 // stores the file name and extension. Add the project path to the file name and
196 // extension to compare when calling SCH_SHEET::SearchHierarchy().
197 wxFileName fileName = aSheet->GetFileName();
198
199 if( !fileName.IsAbsolute() )
200 fileName.MakeAbsolute( m_currentPath.top() );
201
202 // Save the current path so that it gets restored when descending and ascending the
203 // sheet hierarchy which allows for sheet schematic files to be nested in folders
204 // relative to the last path a schematic was loaded from.
205 wxLogTrace( traceSchPlugin, "Saving path '%s'", m_currentPath.top() );
206 m_currentPath.push( fileName.GetPath() );
207 wxLogTrace( traceSchPlugin, "Current path '%s'", m_currentPath.top() );
208 wxLogTrace( traceSchPlugin, "Loading '%s'", fileName.GetFullPath() );
209
210 SCH_SHEET_PATH ancestorSheetPath = aParentSheetPath;
211
212 while( !ancestorSheetPath.empty() )
213 {
214 if( ancestorSheetPath.LastScreen()->GetFileName() == fileName.GetFullPath() )
215 {
216 if( !m_error.IsEmpty() )
217 m_error += "\n";
218
219 m_error += wxString::Format( _( "Could not load sheet '%s' because it already "
220 "appears as a direct ancestor in the schematic "
221 "hierarchy." ),
222 fileName.GetFullPath() );
223
224 fileName = wxEmptyString;
225
226 break;
227 }
228
229 ancestorSheetPath.pop_back();
230 }
231
232 if( ancestorSheetPath.empty() )
233 {
234 // Existing schematics could be either in the root sheet path or the current sheet
235 // load path so we have to check both.
236 if( !m_rootSheet->SearchHierarchy( fileName.GetFullPath(), &screen ) )
237 m_currentSheetPath.at( 0 )->SearchHierarchy( fileName.GetFullPath(), &screen );
238
239 // When loading multiple top-level sheets that reference the same sub-sheet file,
240 // the screen may have already been loaded by a previous top-level sheet.
241 if( !screen )
242 {
243 for( SCH_SHEET* prevRoot : m_loadedRootSheets )
244 {
245 if( prevRoot->SearchHierarchy( fileName.GetFullPath(), &screen ) )
246 break;
247 }
248 }
249 }
250
251 if( screen )
252 {
253 aSheet->SetScreen( screen );
254 aSheet->GetScreen()->SetParent( m_schematic );
255 // Do not need to load the sub-sheets - this has already been done.
256 }
257 else
258 {
259 aSheet->SetScreen( new SCH_SCREEN( m_schematic ) );
260 aSheet->GetScreen()->SetFileName( fileName.GetFullPath() );
261
262 try
263 {
264 loadFile( fileName.GetFullPath(), aSheet );
265 }
266 catch( const IO_ERROR& ioe )
267 {
268 // If there is a problem loading the root sheet, there is no recovery.
269 if( aSheet == m_rootSheet )
270 throw;
271
272 // For all subsheets, queue up the error message for the caller.
273 if( !m_error.IsEmpty() )
274 m_error += "\n";
275
276 m_error += ioe.What();
277 }
278
279 if( fileName.FileExists() )
280 {
281 aSheet->GetScreen()->SetFileReadOnly( !fileName.IsFileWritable() );
282 aSheet->GetScreen()->SetFileExists( true );
283 }
284 else
285 {
286 aSheet->GetScreen()->SetFileReadOnly( !fileName.IsDirWritable() );
287 aSheet->GetScreen()->SetFileExists( false );
288 }
289
290 SCH_SHEET_PATH currentSheetPath = aParentSheetPath;
291 currentSheetPath.push_back( aSheet );
292
293 // This was moved out of the try{} block so that any sheet definitions that
294 // the plugin fully parsed before the exception was raised will be loaded.
295 for( SCH_ITEM* aItem : aSheet->GetScreen()->Items().OfType( SCH_SHEET_T ) )
296 {
297 wxCHECK2( aItem->Type() == SCH_SHEET_T, /* do nothing */ );
298 SCH_SHEET* sheet = static_cast<SCH_SHEET*>( aItem );
299
300 // Recursion starts here.
301 loadHierarchy( currentSheetPath, sheet );
302 }
303 }
304
305 m_currentPath.pop();
306 wxLogTrace( traceSchPlugin, "Restoring path \"%s\"", m_currentPath.top() );
307 }
308
309 m_currentSheetPath.pop_back();
310}
311
312
313void SCH_IO_KICAD_SEXPR::loadFile( const wxString& aFileName, SCH_SHEET* aSheet )
314{
315 FILE_LINE_READER reader( aFileName );
316
317 size_t lineCount = 0;
318
320 {
321 m_progressReporter->Report( wxString::Format( _( "Loading %s..." ), aFileName ) );
322
323 if( !m_progressReporter->KeepRefreshing() )
324 THROW_IO_ERROR( _( "Open canceled by user." ) );
325
326 while( reader.ReadLine() )
327 lineCount++;
328
329 reader.Rewind();
330 }
331
332 SCH_IO_KICAD_SEXPR_PARSER parser( &reader, m_progressReporter, lineCount, m_rootSheet,
333 m_appending );
334
335 parser.ParseSchematic( aSheet );
336}
337
338
339void SCH_IO_KICAD_SEXPR::LoadContent( LINE_READER& aReader, SCH_SHEET* aSheet, int aFileVersion )
340{
341 wxCHECK( aSheet, /* void */ );
342
343 SCH_IO_KICAD_SEXPR_PARSER parser( &aReader );
344
345 parser.ParseSchematic( aSheet, true, aFileVersion );
346}
347
348
349void SCH_IO_KICAD_SEXPR::SaveSchematicFile( const wxString& aFileName, SCH_SHEET* aSheet,
350 SCHEMATIC* aSchematic,
351 const std::map<std::string, UTF8>* aProperties )
352{
353 wxCHECK_RET( aSheet != nullptr, "NULL SCH_SHEET object." );
354 wxCHECK_RET( !aFileName.IsEmpty(), "No schematic file name defined." );
355
356 wxString sanityResult = aSheet->GetScreen()->GroupsSanityCheck();
357
358 if( sanityResult != wxEmptyString && m_queryUserCallback )
359 {
360 if( !m_queryUserCallback( _( "Internal Group Data Error" ), wxICON_ERROR,
361 wxString::Format( _( "Please report this bug. Error validating group "
362 "structure: %s\n\nSave anyway?" ),
363 sanityResult ),
364 _( "Save Anyway" ) ) )
365 {
366 return;
367 }
368 }
369
370 wxFileName fn = aFileName;
371
372 // File names should be absolute. Don't assume everything relative to the project path
373 // works properly.
374 wxASSERT( fn.IsAbsolute() );
375
376 PRETTIFIED_FILE_OUTPUTFORMATTER formatter( fn.GetFullPath() );
377 FormatSchematicToFormatter( &formatter, aSheet, aSchematic, aProperties );
378 formatter.Finish();
379
380 if( aSheet->GetScreen() )
381 aSheet->GetScreen()->SetFileExists( true );
382}
383
384
386 SCHEMATIC* aSchematic,
387 const std::map<std::string, UTF8>* aProperties )
388{
389 wxCHECK_RET( aSheet != nullptr, "NULL SCH_SHEET object." );
390
391 init( aSchematic, aProperties );
392
393 m_out = aOut;
394
395 Format( aSheet );
396
397 m_out = nullptr;
398}
399
400
402{
403 wxCHECK_RET( aSheet != nullptr, "NULL SCH_SHEET* object." );
404 wxCHECK_RET( m_schematic != nullptr, "NULL SCHEMATIC* object." );
405
406 SCH_SHEET_LIST sheets = m_schematic->Hierarchy();
407 SCH_SCREEN* screen = aSheet->GetScreen();
408
409 wxCHECK( screen, /* void */ );
410
411 // If we've requested to embed the fonts in the schematic, do so.
412 // Otherwise, clear the embedded fonts from the schematic. Embedded
413 // fonts will be used if available
414 if( m_schematic->GetAreFontsEmbedded() )
415 m_schematic->EmbedFonts();
416 else
417 m_schematic->GetEmbeddedFiles()->ClearEmbeddedFonts();
418
419 m_out->Print( "(kicad_sch (version %d) (generator \"eeschema\") (generator_version %s)",
421 m_out->Quotew( GetMajorMinorVersion() ).c_str() );
422
424
425 screen->GetPageSettings().Format( m_out );
426 screen->GetTitleBlock().Format( m_out );
427
428 // Save cache library.
429 m_out->Print( "(lib_symbols" );
430
431 for( const auto& [ libItemName, libSymbol ] : screen->GetLibSymbols() )
432 SCH_IO_KICAD_SEXPR_LIB_CACHE::SaveSymbol( libSymbol, *m_out, libItemName );
433
434 m_out->Print( ")" );
435
436 // Enforce item ordering
437 auto cmp =
438 []( const SCH_ITEM* a, const SCH_ITEM* b )
439 {
440 if( a->Type() != b->Type() )
441 return a->Type() < b->Type();
442
443 return a->m_Uuid < b->m_Uuid;
444 };
445
446 std::multiset<SCH_ITEM*, decltype( cmp )> save_map( cmp );
447
448 for( SCH_ITEM* item : screen->Items() )
449 {
450 // Markers are not saved, so keep them from being considered below
451 if( item->Type() != SCH_MARKER_T )
452 save_map.insert( item );
453 }
454
455 for( SCH_ITEM* item : save_map )
456 {
457 switch( item->Type() )
458 {
459 case SCH_SYMBOL_T:
460 saveSymbol( static_cast<SCH_SYMBOL*>( item ), *m_schematic, sheets, false );
461 break;
462
463 case SCH_BITMAP_T:
464 saveBitmap( static_cast<SCH_BITMAP&>( *item ) );
465 break;
466
467 case SCH_SHEET_T:
468 saveSheet( static_cast<SCH_SHEET*>( item ), sheets );
469 break;
470
471 case SCH_JUNCTION_T:
472 saveJunction( static_cast<SCH_JUNCTION*>( item ) );
473 break;
474
475 case SCH_NO_CONNECT_T:
476 saveNoConnect( static_cast<SCH_NO_CONNECT*>( item ) );
477 break;
478
481 saveBusEntry( static_cast<SCH_BUS_ENTRY_BASE*>( item ) );
482 break;
483
484 case SCH_LINE_T:
485 saveLine( static_cast<SCH_LINE*>( item ) );
486 break;
487
488 case SCH_SHAPE_T:
489 saveShape( static_cast<SCH_SHAPE*>( item ) );
490 break;
491
492 case SCH_RULE_AREA_T:
493 saveRuleArea( static_cast<SCH_RULE_AREA*>( item ) );
494 break;
495
496 case SCH_TEXT_T:
497 case SCH_LABEL_T:
499 case SCH_HIER_LABEL_T:
501 saveText( static_cast<SCH_TEXT*>( item ) );
502 break;
503
504 case SCH_TEXTBOX_T:
505 saveTextBox( static_cast<SCH_TEXTBOX*>( item ) );
506 break;
507
508 case SCH_TABLE_T:
509 saveTable( static_cast<SCH_TABLE*>( item ) );
510 break;
511
512 case SCH_GROUP_T:
513 saveGroup( static_cast<SCH_GROUP*>( item ) );
514 break;
515
516 default:
517 wxASSERT( "Unexpected schematic object type in SCH_IO_KICAD_SEXPR::Format()" );
518 }
519 }
520
521 if( aSheet->HasRootInstance() )
522 {
523 std::vector< SCH_SHEET_INSTANCE> instances;
524
525 instances.emplace_back( aSheet->GetRootInstance() );
526 saveInstances( instances );
527
528 KICAD_FORMAT::FormatBool( m_out, "embedded_fonts", m_schematic->GetAreFontsEmbedded() );
529
530 // Save any embedded files
531 if( !m_schematic->GetEmbeddedFiles()->IsEmpty() )
532 m_schematic->WriteEmbeddedFiles( *m_out, true );
533 }
534
535 m_out->Print( ")" );
536}
537
538
539void SCH_IO_KICAD_SEXPR::Format( SCH_SELECTION* aSelection, SCH_SHEET_PATH* aSelectionPath,
540 SCHEMATIC& aSchematic, OUTPUTFORMATTER* aFormatter,
541 bool aForClipboard )
542{
543 wxCHECK( aSelection && aSelectionPath && aFormatter, /* void */ );
544
545 SCH_SHEET_LIST sheets = aSchematic.Hierarchy();
546
547 m_schematic = &aSchematic;
548 m_out = aFormatter;
549
550 std::map<wxString, LIB_SYMBOL*> libSymbols;
551 SCH_SCREEN* screen = aSelection->GetScreen();
552 std::set<SCH_TABLE*> promotedTables;
553
554 for( EDA_ITEM* item : *aSelection )
555 {
556 if( item->Type() != SCH_SYMBOL_T )
557 continue;
558
559 SCH_SYMBOL* symbol = static_cast<SCH_SYMBOL*>( item );
560
561 wxString libSymbolLookup = symbol->GetLibId().Format().wx_str();
562
563 if( !symbol->UseLibIdLookup() )
564 libSymbolLookup = symbol->GetSchSymbolLibraryName();
565
566 auto it = screen->GetLibSymbols().find( libSymbolLookup );
567
568 if( it != screen->GetLibSymbols().end() )
569 libSymbols[ libSymbolLookup ] = it->second;
570 }
571
572 if( !libSymbols.empty() )
573 {
574 m_out->Print( "(lib_symbols" );
575
576 for( const auto& [name, libSymbol] : libSymbols )
578
579 m_out->Print( ")" );
580 }
581
582 for( EDA_ITEM* edaItem : *aSelection )
583 {
584 if( !edaItem->IsSCH_ITEM() )
585 continue;
586
587 SCH_ITEM* item = static_cast<SCH_ITEM*>( edaItem );
588
589 switch( item->Type() )
590 {
591 case SCH_SYMBOL_T:
592 saveSymbol( static_cast<SCH_SYMBOL*>( item ), aSchematic, sheets, aForClipboard, aSelectionPath );
593 break;
594
595 case SCH_BITMAP_T:
596 saveBitmap( static_cast<SCH_BITMAP&>( *item ) );
597 break;
598
599 case SCH_SHEET_T:
600 saveSheet( static_cast<SCH_SHEET*>( item ), sheets );
601 break;
602
603 case SCH_JUNCTION_T:
604 saveJunction( static_cast<SCH_JUNCTION*>( item ) );
605 break;
606
607 case SCH_NO_CONNECT_T:
608 saveNoConnect( static_cast<SCH_NO_CONNECT*>( item ) );
609 break;
610
613 saveBusEntry( static_cast<SCH_BUS_ENTRY_BASE*>( item ) );
614 break;
615
616 case SCH_LINE_T:
617 saveLine( static_cast<SCH_LINE*>( item ) );
618 break;
619
620 case SCH_SHAPE_T:
621 saveShape( static_cast<SCH_SHAPE*>( item ) );
622 break;
623
624 case SCH_RULE_AREA_T:
625 saveRuleArea( static_cast<SCH_RULE_AREA*>( item ) );
626 break;
627
628 case SCH_TEXT_T:
629 case SCH_LABEL_T:
631 case SCH_HIER_LABEL_T:
633 saveText( static_cast<SCH_TEXT*>( item ) );
634 break;
635
636 case SCH_TEXTBOX_T:
637 saveTextBox( static_cast<SCH_TEXTBOX*>( item ) );
638 break;
639
640 case SCH_TABLECELL_T:
641 {
642 SCH_TABLE* table = static_cast<SCH_TABLE*>( item->GetParent() );
643
644 if( promotedTables.count( table ) )
645 break;
646
647 table->SetFlags( SKIP_STRUCT );
648 saveTable( table );
649 table->ClearFlags( SKIP_STRUCT );
650 promotedTables.insert( table );
651 break;
652 }
653
654 case SCH_TABLE_T:
655 item->ClearFlags( SKIP_STRUCT );
656 saveTable( static_cast<SCH_TABLE*>( item ) );
657 break;
658
659 case SCH_GROUP_T:
660 saveGroup( static_cast<SCH_GROUP*>( item ) );
661 break;
662
663 default:
664 wxASSERT( "Unexpected schematic object type in SCH_IO_KICAD_SEXPR::Format()" );
665 }
666 }
667}
668
669
670void SCH_IO_KICAD_SEXPR::saveSymbol( SCH_SYMBOL* aSymbol, const SCHEMATIC& aSchematic,
671 const SCH_SHEET_LIST& aSheetList, bool aForClipboard,
672 const SCH_SHEET_PATH* aRelativePath )
673{
674 wxCHECK_RET( aSymbol != nullptr && m_out != nullptr, "" );
675
676 std::string libName;
677
678 wxString symbol_name = aSymbol->GetLibId().Format();
679
680 if( symbol_name.size() )
681 {
682 libName = toUTFTildaText( symbol_name );
683 }
684 else
685 {
686 libName = "_NONAME_";
687 }
688
689 EDA_ANGLE angle;
690 int orientation = aSymbol->GetOrientation() & ~( SYM_MIRROR_X | SYM_MIRROR_Y );
691
692 if( orientation == SYM_ORIENT_90 )
693 angle = ANGLE_90;
694 else if( orientation == SYM_ORIENT_180 )
695 angle = ANGLE_180;
696 else if( orientation == SYM_ORIENT_270 )
697 angle = ANGLE_270;
698 else
699 angle = ANGLE_0;
700
701 m_out->Print( "(symbol" );
702
703 if( !aSymbol->UseLibIdLookup() )
704 {
705 m_out->Print( "(lib_name %s)",
706 m_out->Quotew( aSymbol->GetSchSymbolLibraryName() ).c_str() );
707 }
708
709 m_out->Print( "(lib_id %s) (at %s %s %s)",
710 m_out->Quotew( aSymbol->GetLibId().Format().wx_str() ).c_str(),
712 aSymbol->GetPosition().x ).c_str(),
714 aSymbol->GetPosition().y ).c_str(),
715 EDA_UNIT_UTILS::FormatAngle( angle ).c_str() );
716
717 bool mirrorX = aSymbol->GetOrientation() & SYM_MIRROR_X;
718 bool mirrorY = aSymbol->GetOrientation() & SYM_MIRROR_Y;
719
720 if( mirrorX || mirrorY )
721 {
722 m_out->Print( "(mirror %s %s)",
723 mirrorX ? "x" : "",
724 mirrorY ? "y" : "" );
725 }
726
727 // The symbol unit is always set to the ordianal instance regardless of the current sheet
728 // instance to prevent file churn.
729 SCH_SYMBOL_INSTANCE ordinalInstance;
730
731 ordinalInstance.m_Reference = aSymbol->GetPrefix();
732
733 const SCH_SCREEN* parentScreen = static_cast<const SCH_SCREEN*>( aSymbol->GetParent() );
734
735 wxASSERT( parentScreen );
736
737 if( parentScreen && m_schematic )
738 {
739 std::optional<SCH_SHEET_PATH> ordinalPath =
740 m_schematic->Hierarchy().GetOrdinalPath( parentScreen );
741
742 // Design blocks are saved from a temporary sheet & screen which will not be found in
743 // the schematic, and will therefore have no ordinal path.
744 // wxASSERT( ordinalPath );
745
746 if( ordinalPath )
747 aSymbol->GetInstance( ordinalInstance, ordinalPath->Path() );
748 else if( aSymbol->GetInstances().size() )
749 ordinalInstance = aSymbol->GetInstances()[0];
750 }
751
752 int unit = ordinalInstance.m_Unit;
753
754 if( aForClipboard && aRelativePath )
755 {
756 SCH_SYMBOL_INSTANCE unitInstance;
757
758 if( aSymbol->GetInstance( unitInstance, aRelativePath->Path() ) )
759 unit = unitInstance.m_Unit;
760 }
761
762 m_out->Print( "(unit %d)", unit );
763 m_out->Print( "(body_style %d)", aSymbol->GetBodyStyle() );
764
765 KICAD_FORMAT::FormatBool( m_out, "exclude_from_sim", aSymbol->GetExcludedFromSim() );
766 KICAD_FORMAT::FormatBool( m_out, "in_bom", !aSymbol->GetExcludedFromBOM() );
767 KICAD_FORMAT::FormatBool( m_out, "on_board", !aSymbol->GetExcludedFromBoard() );
768 KICAD_FORMAT::FormatBool( m_out, "in_pos_files", !aSymbol->GetExcludedFromPosFiles() );
769 KICAD_FORMAT::FormatBool( m_out, "dnp", aSymbol->GetDNP() );
770
771 if( aSymbol->IsLocked() )
772 KICAD_FORMAT::FormatBool( m_out, "locked", true );
773
774 AUTOPLACE_ALGO fieldsAutoplaced = aSymbol->GetFieldsAutoplaced();
775
776 if( fieldsAutoplaced == AUTOPLACE_AUTO || fieldsAutoplaced == AUTOPLACE_MANUAL )
777 KICAD_FORMAT::FormatBool( m_out, "fields_autoplaced", true );
778
780
781 std::vector<SCH_FIELD*> orderedFields;
782 aSymbol->GetFields( orderedFields, false );
783
784 for( SCH_FIELD* field : orderedFields )
785 {
786 FIELD_T id = field->GetId();
787 wxString value = field->GetText();
788
789 if( !aForClipboard && aSymbol->GetInstances().size() )
790 {
791 // The instance fields are always set to the default instance regardless of the
792 // sheet instance to prevent file churn.
793 if( id == FIELD_T::REFERENCE )
794 field->SetText( ordinalInstance.m_Reference );
795 }
796 else if( aForClipboard && aSymbol->GetInstances().size() && aRelativePath
797 && ( id == FIELD_T::REFERENCE ) )
798 {
799 SCH_SYMBOL_INSTANCE instance;
800
801 if( aSymbol->GetInstance( instance, aRelativePath->Path() ) )
802 field->SetText( instance.m_Reference );
803 }
804
805 try
806 {
807 saveField( field );
808 }
809 catch( ... )
810 {
811 // Restore the changed field text on write error.
812 if( id == FIELD_T::REFERENCE )
813 field->SetText( value );
814
815 throw;
816 }
817
818 if( id == FIELD_T::REFERENCE )
819 field->SetText( value );
820 }
821
822 for( const std::unique_ptr<SCH_PIN>& pin : aSymbol->GetRawPins() )
823 {
824 // There was a bug introduced somewhere in the original alternated pin code that would
825 // set the alternate pin to the default pin name which caused a number of library symbol
826 // comparison issues. Clearing the alternate pin resolves this issue.
827 if( pin->GetAlt().IsEmpty() || ( pin->GetAlt() == pin->GetBaseName() ) )
828 {
829 m_out->Print( "(pin %s", m_out->Quotew( pin->GetNumber() ).c_str() );
831 m_out->Print( ")" );
832 }
833 else
834 {
835 m_out->Print( "(pin %s", m_out->Quotew( pin->GetNumber() ).c_str() );
837 m_out->Print( "(alternate %s))", m_out->Quotew( pin->GetAlt() ).c_str() );
838 }
839 }
840
841 if( !aSymbol->GetInstances().empty() )
842 {
843 std::map<KIID, std::vector<SCH_SYMBOL_INSTANCE>> projectInstances;
844 std::set<KIID> currentProjectKeys;
845
846 m_out->Print( "(instances" );
847
848 wxString projectName;
849 KIID rootSheetUuid = aSchematic.Root().m_Uuid;
850
851 // Collect top-level sheet UUIDs to identify current project instances.
852 // When root is virtual (niluuid), Path() skips it, so instance paths
853 // start with the real top-level sheet UUID, not niluuid.
854
855 if( rootSheetUuid == niluuid )
856 {
857 for( const SCH_SHEET* sheet : aSchematic.GetTopLevelSheets() )
858 currentProjectKeys.insert( sheet->m_Uuid );
859 }
860 else
861 {
862 currentProjectKeys.insert( rootSheetUuid );
863 }
864
865 for( const SCH_SYMBOL_INSTANCE& inst : aSymbol->GetInstances() )
866 {
867 // Zero length KIID_PATH objects are not valid and will cause a crash below.
868 wxCHECK2( inst.m_Path.size(), continue );
869
870 // If the instance data is part of this design but no longer has an associated sheet
871 // path, don't save it. This prevents large amounts of orphaned instance data for the
872 // current project from accumulating in the schematic files.
873 //
874 // The root sheet UUID can be niluuid for the virtual root. In that case, instance
875 // paths may include the virtual root, but SCH_SHEET_PATH::Path() skips it. We need
876 // to normalize the path by removing the virtual root before comparison.
877 KIID_PATH pathToCheck = inst.m_Path;
878
879 // If root is virtual (niluuid) and path starts with virtual root, strip it
880 if( rootSheetUuid == niluuid && !pathToCheck.empty() && pathToCheck[0] == niluuid )
881 {
882 if( pathToCheck.size() > 1 )
883 {
884 pathToCheck.erase( pathToCheck.begin() );
885 }
886 else
887 {
888 // Path only contains virtual root, skip it
889 continue;
890 }
891 }
892
893 // Check if this instance is orphaned (no matching sheet path)
894 // For virtual root, we check if the first real sheet matches one of the top-level sheets
895 // For non-virtual root, we check if it matches the root sheet UUID
896 bool belongsToThisProject = currentProjectKeys.count( pathToCheck[0] );
897
898 bool isOrphaned = belongsToThisProject && !aSheetList.GetSheetPathByKIIDPath( pathToCheck );
899
900 // Keep all instance data when copying to the clipboard. They may be needed on paste.
901 if( !aForClipboard && isOrphaned )
902 continue;
903
904 // Group by project - use the first real sheet KIID (after stripping virtual root)
905 KIID projectKey = pathToCheck[0];
906 auto it = projectInstances.find( projectKey );
907
908 if( it == projectInstances.end() )
909 projectInstances[ projectKey ] = { inst };
910 else
911 it->second.emplace_back( inst );
912 }
913
914 for( auto& [uuid, instances] : projectInstances )
915 {
916 wxCHECK2( instances.size(), continue );
917
918 // Sort project instances by KIID_PATH.
919 std::sort( instances.begin(), instances.end(),
921 {
922 return aLhs.m_Path < aRhs.m_Path;
923 } );
924
925 if( currentProjectKeys.count( uuid ) )
926 projectName = m_schematic->Project().GetProjectName();
927 else
928 projectName = instances[0].m_ProjectName;
929
930 m_out->Print( "(project %s", m_out->Quotew( projectName ).c_str() );
931
932 for( const SCH_SYMBOL_INSTANCE& instance : instances )
933 {
934 wxString path;
935 KIID_PATH tmp = instance.m_Path;
936
937 if( aForClipboard && aRelativePath )
938 tmp.MakeRelativeTo( aRelativePath->Path() );
939
940 path = tmp.AsString();
941
942 m_out->Print( "(path %s (reference %s) (unit %d)",
943 m_out->Quotew( path ).c_str(),
944 m_out->Quotew( instance.m_Reference ).c_str(),
945 instance.m_Unit );
946
947 if( !instance.m_Variants.empty() )
948 {
949 for( const auto&[name, variant] : instance.m_Variants )
950 {
951 m_out->Print( "(variant (name %s)", m_out->Quotew( name ).c_str() );
952
953 if( variant.m_DNP != aSymbol->GetDNP() )
954 KICAD_FORMAT::FormatBool( m_out, "dnp", variant.m_DNP );
955
956 if( variant.m_ExcludedFromSim != aSymbol->GetExcludedFromSim() )
957 KICAD_FORMAT::FormatBool( m_out, "exclude_from_sim", variant.m_ExcludedFromSim );
958
959 if( variant.m_ExcludedFromBOM != aSymbol->GetExcludedFromBOM() )
960 KICAD_FORMAT::FormatBool( m_out, "in_bom", !variant.m_ExcludedFromBOM );
961
962 if( variant.m_ExcludedFromBoard != aSymbol->GetExcludedFromBoard() )
963 KICAD_FORMAT::FormatBool( m_out, "on_board", !variant.m_ExcludedFromBoard );
964
965 if( variant.m_ExcludedFromPosFiles != aSymbol->GetExcludedFromPosFiles() )
966 KICAD_FORMAT::FormatBool( m_out, "in_pos_files", !variant.m_ExcludedFromPosFiles );
967
968 for( const auto&[fname, fvalue] : variant.m_Fields )
969 {
970 m_out->Print( "(field (name %s) (value %s))",
971 m_out->Quotew( fname ).c_str(), m_out->Quotew( fvalue ).c_str() );
972 }
973
974 m_out->Print( ")" ); // Closes `variant` token.
975 }
976 }
977
978 m_out->Print( ")" ); // Closes `path` token.
979 }
980
981 m_out->Print( ")" ); // Closes `project`.
982 }
983
984 m_out->Print( ")" ); // Closes `instances`.
985 }
986
987 m_out->Print( ")" ); // Closes `symbol`.
988}
989
990
992{
993 wxCHECK_RET( aField != nullptr && m_out != nullptr, "" );
994
995 wxString fieldName;
996
997 if( aField->IsMandatory() )
998 fieldName = aField->GetCanonicalName();
999 else
1000 fieldName = aField->GetName();
1001
1002 m_out->Print( "(property %s %s %s (at %s %s %s)",
1003 aField->IsPrivate() ? "private" : "",
1004 m_out->Quotew( fieldName ).c_str(),
1005 m_out->Quotew( aField->GetText() ).c_str(),
1007 aField->GetPosition().x ).c_str(),
1009 aField->GetPosition().y ).c_str(),
1010 EDA_UNIT_UTILS::FormatAngle( aField->GetTextAngle() ).c_str() );
1011
1012 if( !aField->IsVisible() )
1013 KICAD_FORMAT::FormatBool( m_out, "hide", true );
1014
1015 KICAD_FORMAT::FormatBool( m_out, "show_name", aField->IsNameShown() );
1016
1017 KICAD_FORMAT::FormatBool( m_out, "do_not_autoplace", !aField->CanAutoplace() );
1018
1019 if( !aField->IsDefaultFormatting()
1020 || ( aField->GetTextHeight() != schIUScale.MilsToIU( DEFAULT_SIZE_TEXT ) ) )
1021 {
1022 aField->Format( m_out, 0 );
1023 }
1024
1025 m_out->Print( ")" ); // Closes `property` token
1026}
1027
1028
1030{
1031 wxCHECK_RET( m_out != nullptr, "" );
1032
1033 const REFERENCE_IMAGE& refImage = aBitmap.GetReferenceImage();
1034 const BITMAP_BASE& bitmapBase = refImage.GetImage();
1035
1036 const wxImage* image = bitmapBase.GetImageData();
1037
1038 wxCHECK_RET( image != nullptr, "wxImage* is NULL" );
1039
1040 m_out->Print( "(image (at %s %s)",
1042 refImage.GetPosition().x ).c_str(),
1044 refImage.GetPosition().y ).c_str() );
1045
1046 double scale = refImage.GetImageScale();
1047
1048 // 20230121 or older file format versions assumed 300 image PPI at load/save.
1049 // Let's keep compatibility by changing image scale.
1050 if( SEXPR_SCHEMATIC_FILE_VERSION <= 20230121 )
1051 scale = scale * 300.0 / bitmapBase.GetPPI();
1052
1053 if( scale != 1.0 )
1054 m_out->Print( "%s", fmt::format("(scale {:g})", refImage.GetImageScale()).c_str() );
1055
1057
1058 if( aBitmap.IsLocked() )
1059 KICAD_FORMAT::FormatBool( m_out, "locked", true );
1060
1061 wxMemoryOutputStream stream;
1062 bitmapBase.SaveImageData( stream );
1063
1064 KICAD_FORMAT::FormatStreamData( *m_out, *stream.GetOutputStreamBuffer() );
1065
1066 m_out->Print( ")" ); // Closes image token.
1067}
1068
1069
1071{
1072 wxCHECK_RET( aSheet != nullptr && m_out != nullptr, "" );
1073
1074 m_out->Print( "(sheet (at %s %s) (size %s %s)",
1076 aSheet->GetPosition().x ).c_str(),
1078 aSheet->GetPosition().y ).c_str(),
1080 aSheet->GetSize().x ).c_str(),
1082 aSheet->GetSize().y ).c_str() );
1083
1084 KICAD_FORMAT::FormatBool( m_out, "exclude_from_sim", aSheet->GetExcludedFromSim() );
1085 KICAD_FORMAT::FormatBool( m_out, "in_bom", !aSheet->GetExcludedFromBOM() );
1086 KICAD_FORMAT::FormatBool( m_out, "on_board", !aSheet->GetExcludedFromBoard() );
1087 KICAD_FORMAT::FormatBool( m_out, "dnp", aSheet->GetDNP() );
1088
1089 if( aSheet->IsLocked() )
1090 KICAD_FORMAT::FormatBool( m_out, "locked", true );
1091
1092 AUTOPLACE_ALGO fieldsAutoplaced = aSheet->GetFieldsAutoplaced();
1093
1094 if( fieldsAutoplaced == AUTOPLACE_AUTO || fieldsAutoplaced == AUTOPLACE_MANUAL )
1095 KICAD_FORMAT::FormatBool( m_out, "fields_autoplaced", true );
1096
1097 STROKE_PARAMS stroke( aSheet->GetBorderWidth(), LINE_STYLE::SOLID, aSheet->GetBorderColor() );
1098
1099 stroke.SetWidth( aSheet->GetBorderWidth() );
1100 stroke.Format( m_out, schIUScale );
1101
1102 m_out->Print( "(fill (color %d %d %d %s))",
1103 KiROUND( aSheet->GetBackgroundColor().r * 255.0 ),
1104 KiROUND( aSheet->GetBackgroundColor().g * 255.0 ),
1105 KiROUND( aSheet->GetBackgroundColor().b * 255.0 ),
1106 FormatDouble2Str( aSheet->GetBackgroundColor().a ).c_str() );
1107
1109
1110 for( SCH_FIELD& field : aSheet->GetFields() )
1111 saveField( &field );
1112
1113 for( const SCH_SHEET_PIN* pin : aSheet->GetPins() )
1114 {
1115 m_out->Print( "(pin %s %s (at %s %s %s)",
1116 EscapedUTF8( pin->GetText() ).c_str(),
1117 getSheetPinShapeToken( pin->GetShape() ),
1119 pin->GetPosition().x ).c_str(),
1121 pin->GetPosition().y ).c_str(),
1122 EDA_UNIT_UTILS::FormatAngle( getSheetPinAngle( pin->GetSide() ) ).c_str() );
1123
1125
1126 pin->Format( m_out, 0 );
1127
1128 m_out->Print( ")" ); // Closes pin token.
1129 }
1130
1131 // Save all sheet instances here except the root sheet instance.
1132 std::vector< SCH_SHEET_INSTANCE > sheetInstances = aSheet->GetInstances();
1133
1134 auto it = sheetInstances.begin();
1135
1136 while( it != sheetInstances.end() )
1137 {
1138 if( it->m_Path.size() == 0 )
1139 it = sheetInstances.erase( it );
1140 else
1141 it++;
1142 }
1143
1144 if( !sheetInstances.empty() )
1145 {
1146 m_out->Print( "(instances" );
1147
1148 KIID lastProjectUuid;
1149 KIID rootSheetUuid = m_schematic->Root().m_Uuid;
1150 bool inProjectClause = false;
1151
1152 std::set<KIID> currentProjectKeys;
1153
1154 if( rootSheetUuid == niluuid )
1155 {
1156 for( const SCH_SHEET* sheet : m_schematic->GetTopLevelSheets() )
1157 currentProjectKeys.insert( sheet->m_Uuid );
1158 }
1159 else
1160 {
1161 currentProjectKeys.insert( rootSheetUuid );
1162 }
1163
1164 for( size_t i = 0; i < sheetInstances.size(); i++ )
1165 {
1166 // If the instance data is part of this design but no longer has an associated sheet
1167 // path, don't save it. This prevents large amounts of orphaned instance data for the
1168 // current project from accumulating in the schematic files.
1169 //
1170 // Keep all instance data when copying to the clipboard. It may be needed on paste.
1171 bool belongsToThisProject =
1172 !sheetInstances[i].m_Path.empty() && currentProjectKeys.count( sheetInstances[i].m_Path[0] );
1173
1174 if( belongsToThisProject && !aSheetList.GetSheetPathByKIIDPath( sheetInstances[i].m_Path, false ) )
1175 {
1176 if( inProjectClause && ( ( i + 1 == sheetInstances.size() )
1177 || lastProjectUuid != sheetInstances[i+1].m_Path[0] ) )
1178 {
1179 m_out->Print( ")" ); // Closes `project` token.
1180 inProjectClause = false;
1181 }
1182
1183 continue;
1184 }
1185
1186 if( lastProjectUuid != sheetInstances[i].m_Path[0] )
1187 {
1188 wxString projectName;
1189
1190 if( belongsToThisProject )
1191 projectName = m_schematic->Project().GetProjectName();
1192 else
1193 projectName = sheetInstances[i].m_ProjectName;
1194
1195 lastProjectUuid = sheetInstances[i].m_Path[0];
1196 m_out->Print( "(project %s", m_out->Quotew( projectName ).c_str() );
1197 inProjectClause = true;
1198 }
1199
1200 wxString path = sheetInstances[i].m_Path.AsString();
1201
1202 m_out->Print( "(path %s (page %s)",
1203 m_out->Quotew( path ).c_str(),
1204 m_out->Quotew( sheetInstances[i].m_PageNumber ).c_str() );
1205
1206 if( !sheetInstances[i].m_Variants.empty() )
1207 {
1208 for( const auto&[name, variant] : sheetInstances[i].m_Variants )
1209 {
1210 m_out->Print( "(variant (name %s)", m_out->Quotew( name ).c_str() );
1211
1212 if( variant.m_DNP != aSheet->GetDNP() )
1213 KICAD_FORMAT::FormatBool( m_out, "dnp", variant.m_DNP );
1214
1215 if( variant.m_ExcludedFromSim != aSheet->GetExcludedFromSim() )
1216 KICAD_FORMAT::FormatBool( m_out, "exclude_from_sim", variant.m_ExcludedFromSim );
1217
1218 if( variant.m_ExcludedFromBOM != aSheet->GetExcludedFromBOM() )
1219 KICAD_FORMAT::FormatBool( m_out, "in_bom", !variant.m_ExcludedFromBOM );
1220
1221 for( const auto&[fname, fvalue] : variant.m_Fields )
1222 {
1223 m_out->Print( "(field (name %s) (value %s))",
1224 m_out->Quotew( fname ).c_str(), m_out->Quotew( fvalue ).c_str() );
1225 }
1226
1227 m_out->Print( ")" ); // Closes `variant` token.
1228 }
1229 }
1230
1231 m_out->Print( ")" ); // Closes `path` token.
1232
1233 if( inProjectClause && ( ( i + 1 == sheetInstances.size() )
1234 || lastProjectUuid != sheetInstances[i+1].m_Path[0] ) )
1235 {
1236 m_out->Print( ")" ); // Closes `project` token.
1237 inProjectClause = false;
1238 }
1239 }
1240
1241 m_out->Print( ")" ); // Closes `instances` token.
1242 }
1243
1244 m_out->Print( ")" ); // Closes sheet token.
1245}
1246
1247
1249{
1250 wxCHECK_RET( aJunction != nullptr && m_out != nullptr, "" );
1251
1252 m_out->Print( "(junction (at %s %s) (diameter %s) (color %d %d %d %s)",
1254 aJunction->GetPosition().x ).c_str(),
1256 aJunction->GetPosition().y ).c_str(),
1258 aJunction->GetDiameter() ).c_str(),
1259 KiROUND( aJunction->GetColor().r * 255.0 ),
1260 KiROUND( aJunction->GetColor().g * 255.0 ),
1261 KiROUND( aJunction->GetColor().b * 255.0 ),
1262 FormatDouble2Str( aJunction->GetColor().a ).c_str() );
1263
1264 KICAD_FORMAT::FormatUuid( m_out, aJunction->m_Uuid );
1265
1266 if( aJunction->IsLocked() )
1267 KICAD_FORMAT::FormatBool( m_out, "locked", true );
1268
1269 m_out->Print( ")" );
1270}
1271
1272
1274{
1275 wxCHECK_RET( aNoConnect != nullptr && m_out != nullptr, "" );
1276
1277 m_out->Print( "(no_connect (at %s %s)",
1279 aNoConnect->GetPosition().x ).c_str(),
1281 aNoConnect->GetPosition().y ).c_str() );
1282
1283 KICAD_FORMAT::FormatUuid( m_out, aNoConnect->m_Uuid );
1284
1285 if( aNoConnect->IsLocked() )
1286 KICAD_FORMAT::FormatBool( m_out, "locked", true );
1287
1288 m_out->Print( ")" );
1289}
1290
1291
1293{
1294 wxCHECK_RET( aBusEntry != nullptr && m_out != nullptr, "" );
1295
1296 // Bus to bus entries are converted to bus line segments.
1297 if( aBusEntry->GetClass() == "SCH_BUS_BUS_ENTRY" )
1298 {
1299 SCH_LINE busEntryLine( aBusEntry->GetPosition(), LAYER_BUS );
1300
1301 busEntryLine.SetEndPoint( aBusEntry->GetEnd() );
1302 saveLine( &busEntryLine );
1303 return;
1304 }
1305
1306 m_out->Print( "(bus_entry (at %s %s) (size %s %s)",
1308 aBusEntry->GetPosition().x ).c_str(),
1310 aBusEntry->GetPosition().y ).c_str(),
1312 aBusEntry->GetSize().x ).c_str(),
1314 aBusEntry->GetSize().y ).c_str() );
1315
1316 aBusEntry->GetStroke().Format( m_out, schIUScale );
1317 KICAD_FORMAT::FormatUuid( m_out, aBusEntry->m_Uuid );
1318
1319 if( aBusEntry->IsLocked() )
1320 KICAD_FORMAT::FormatBool( m_out, "locked", true );
1321
1322 m_out->Print( ")" );
1323}
1324
1325
1327{
1328 wxCHECK_RET( aShape != nullptr && m_out != nullptr, "" );
1329
1330 // Rule areas handle locked at their own level via saveRuleArea(), so don't duplicate it
1331 // inside the shape sub-expression.
1332 bool writeLocked = aShape->Type() != SCH_RULE_AREA_T && aShape->IsLocked();
1333
1334 switch( aShape->GetShape() )
1335 {
1336 case SHAPE_T::ARC:
1337 formatArc( m_out, aShape, false, aShape->GetStroke(), aShape->GetFillMode(),
1338 aShape->GetFillColor(), false, aShape->m_Uuid, writeLocked );
1339 break;
1340
1341 case SHAPE_T::CIRCLE:
1342 formatCircle( m_out, aShape, false, aShape->GetStroke(), aShape->GetFillMode(),
1343 aShape->GetFillColor(), false, aShape->m_Uuid, writeLocked );
1344 break;
1345
1346 case SHAPE_T::RECTANGLE:
1347 formatRect( m_out, aShape, false, aShape->GetStroke(), aShape->GetFillMode(),
1348 aShape->GetFillColor(), false, aShape->m_Uuid, writeLocked );
1349 break;
1350
1351 case SHAPE_T::BEZIER:
1352 formatBezier( m_out, aShape, false, aShape->GetStroke(), aShape->GetFillMode(),
1353 aShape->GetFillColor(), false, aShape->m_Uuid, writeLocked );
1354 break;
1355
1356 case SHAPE_T::POLY:
1357 formatPoly( m_out, aShape, false, aShape->GetStroke(), aShape->GetFillMode(),
1358 aShape->GetFillColor(), false, aShape->m_Uuid, writeLocked );
1359 break;
1360
1361 default:
1363 }
1364}
1365
1366
1368{
1369 wxCHECK_RET( aRuleArea != nullptr && m_out != nullptr, "" );
1370
1371 m_out->Print( "(rule_area " );
1372
1373 if( aRuleArea->IsLocked() )
1374 KICAD_FORMAT::FormatBool( m_out, "locked", true );
1375
1376 KICAD_FORMAT::FormatBool( m_out, "exclude_from_sim", aRuleArea->GetExcludedFromSim() );
1377 KICAD_FORMAT::FormatBool( m_out, "in_bom", !aRuleArea->GetExcludedFromBOM() );
1378 KICAD_FORMAT::FormatBool( m_out, "on_board", !aRuleArea->GetExcludedFromBoard() );
1379 KICAD_FORMAT::FormatBool( m_out, "dnp", aRuleArea->GetDNP() );
1380
1381 saveShape( aRuleArea );
1382
1383 m_out->Print( ")" );
1384}
1385
1386
1388{
1389 wxCHECK_RET( aLine != nullptr && m_out != nullptr, "" );
1390
1391 wxString lineType;
1392
1393 STROKE_PARAMS line_stroke = aLine->GetStroke();
1394
1395 switch( aLine->GetLayer() )
1396 {
1397 case LAYER_BUS: lineType = "bus"; break;
1398 case LAYER_WIRE: lineType = "wire"; break;
1399 case LAYER_NOTES: lineType = "polyline"; break;
1400 default:
1401 UNIMPLEMENTED_FOR( LayerName( aLine->GetLayer() ) );
1402 }
1403
1404 m_out->Print( "(%s (pts (xy %s %s) (xy %s %s))",
1405 TO_UTF8( lineType ),
1407 aLine->GetStartPoint().x ).c_str(),
1409 aLine->GetStartPoint().y ).c_str(),
1411 aLine->GetEndPoint().x ).c_str(),
1413 aLine->GetEndPoint().y ).c_str() );
1414
1415 line_stroke.Format( m_out, schIUScale );
1417
1418 if( aLine->IsLocked() )
1419 KICAD_FORMAT::FormatBool( m_out, "locked", true );
1420
1421 m_out->Print( ")" );
1422}
1423
1424
1426{
1427 wxCHECK_RET( aText != nullptr && m_out != nullptr, "" );
1428
1429 // Note: label is nullptr SCH_TEXT, but not for SCH_LABEL_XXX,
1430 SCH_LABEL_BASE* label = dynamic_cast<SCH_LABEL_BASE*>( aText );
1431
1432 m_out->Print( "(%s %s",
1433 getTextTypeToken( aText->Type() ),
1434 m_out->Quotew( aText->GetText() ).c_str() );
1435
1436 if( aText->Type() == SCH_TEXT_T )
1437 KICAD_FORMAT::FormatBool( m_out, "exclude_from_sim", aText->GetExcludedFromSim() );
1438
1439 if( aText->Type() == SCH_DIRECTIVE_LABEL_T )
1440 {
1441 SCH_DIRECTIVE_LABEL* flag = static_cast<SCH_DIRECTIVE_LABEL*>( aText );
1442
1443 m_out->Print( "(length %s)",
1445 flag->GetPinLength() ).c_str() );
1446 }
1447
1448 EDA_ANGLE angle = aText->GetTextAngle();
1449
1450 if( label )
1451 {
1452 if( label->Type() == SCH_GLOBAL_LABEL_T
1453 || label->Type() == SCH_HIER_LABEL_T
1454 || label->Type() == SCH_DIRECTIVE_LABEL_T )
1455 {
1456 m_out->Print( "(shape %s)", getSheetPinShapeToken( label->GetShape() ) );
1457 }
1458
1459 // The angle of the text is always 0 or 90 degrees for readibility reasons,
1460 // but the item itself can have more rotation (-90 and 180 deg)
1461 switch( label->GetSpinStyle() )
1462 {
1463 default:
1464 case SPIN_STYLE::LEFT: angle += ANGLE_180; break;
1465 case SPIN_STYLE::UP: break;
1466 case SPIN_STYLE::RIGHT: break;
1467 case SPIN_STYLE::BOTTOM: angle += ANGLE_180; break;
1468 }
1469 }
1470
1471 m_out->Print( "(at %s %s %s)",
1473 aText->GetPosition().x ).c_str(),
1475 aText->GetPosition().y ).c_str(),
1476 EDA_UNIT_UTILS::FormatAngle( angle ).c_str() );
1477
1478 if( label && !label->GetFields().empty() )
1479 {
1480 AUTOPLACE_ALGO fieldsAutoplaced = label->GetFieldsAutoplaced();
1481
1482 if( fieldsAutoplaced == AUTOPLACE_AUTO || fieldsAutoplaced == AUTOPLACE_MANUAL )
1483 KICAD_FORMAT::FormatBool( m_out, "fields_autoplaced", true );
1484 }
1485
1486 aText->EDA_TEXT::Format( m_out, 0 );
1488
1489 if( aText->IsLocked() )
1490 KICAD_FORMAT::FormatBool( m_out, "locked", true );
1491
1492 if( label )
1493 {
1494 for( SCH_FIELD& field : label->GetFields() )
1495 saveField( &field );
1496 }
1497
1498 m_out->Print( ")" ); // Closes text token.
1499}
1500
1501
1503{
1504 wxCHECK_RET( aTextBox != nullptr && m_out != nullptr, "" );
1505
1506 m_out->Print( "(%s %s",
1507 aTextBox->Type() == SCH_TABLECELL_T ? "table_cell" : "text_box",
1508 m_out->Quotew( aTextBox->GetText() ).c_str() );
1509
1510 KICAD_FORMAT::FormatBool( m_out, "exclude_from_sim", aTextBox->GetExcludedFromSim() );
1511
1512 VECTOR2I pos = aTextBox->GetStart();
1513 VECTOR2I size = aTextBox->GetEnd() - pos;
1514
1515 m_out->Print( "(at %s %s %s) (size %s %s) (margins %s %s %s %s)",
1518 EDA_UNIT_UTILS::FormatAngle( aTextBox->GetTextAngle() ).c_str(),
1525
1526 if( SCH_TABLECELL* cell = dynamic_cast<SCH_TABLECELL*>( aTextBox ) )
1527 m_out->Print( "(span %d %d)", cell->GetColSpan(), cell->GetRowSpan() );
1528
1529 if( aTextBox->Type() != SCH_TABLECELL_T )
1530 aTextBox->GetStroke().Format( m_out, schIUScale );
1531
1532 formatFill( m_out, aTextBox->GetFillMode(), aTextBox->GetFillColor() );
1533 aTextBox->EDA_TEXT::Format( m_out, 0 );
1535
1536 if( aTextBox->IsLocked() )
1537 KICAD_FORMAT::FormatBool( m_out, "locked", true );
1538
1539 m_out->Print( ")" );
1540}
1541
1542
1544{
1545 if( aTable->GetFlags() & SKIP_STRUCT )
1546 {
1547 aTable = static_cast<SCH_TABLE*>( aTable->Clone() );
1548
1549 int minCol = aTable->GetColCount();
1550 int maxCol = -1;
1551 int minRow = aTable->GetRowCount();
1552 int maxRow = -1;
1553
1554 for( int row = 0; row < aTable->GetRowCount(); ++row )
1555 {
1556 for( int col = 0; col < aTable->GetColCount(); ++col )
1557 {
1558 SCH_TABLECELL* cell = aTable->GetCell( row, col );
1559
1560 if( cell->IsSelected() )
1561 {
1562 minRow = std::min( minRow, row );
1563 maxRow = std::max( maxRow, row );
1564 minCol = std::min( minCol, col );
1565 maxCol = std::max( maxCol, col );
1566 }
1567 else
1568 {
1569 cell->SetFlags( STRUCT_DELETED );
1570 }
1571 }
1572 }
1573
1574 wxCHECK_MSG( maxCol >= minCol && maxRow >= minRow, /*void*/, wxT( "No selected cells!" ) );
1575
1576 int destRow = 0;
1577
1578 for( int row = minRow; row <= maxRow; row++ )
1579 aTable->SetRowHeight( destRow++, aTable->GetRowHeight( row ) );
1580
1581 int destCol = 0;
1582
1583 for( int col = minCol; col <= maxCol; col++ )
1584 aTable->SetColWidth( destCol++, aTable->GetColWidth( col ) );
1585
1586 aTable->DeleteMarkedCells();
1587 aTable->SetColCount( ( maxCol - minCol ) + 1 );
1588 }
1589
1590 wxCHECK_RET( aTable != nullptr && m_out != nullptr, "" );
1591
1592 m_out->Print( "(table (column_count %d)", aTable->GetColCount() );
1593
1594 m_out->Print( "(border" );
1595 KICAD_FORMAT::FormatBool( m_out, "external", aTable->StrokeExternal() );
1597
1598 if( aTable->StrokeExternal() || aTable->StrokeHeaderSeparator() )
1599 aTable->GetBorderStroke().Format( m_out, schIUScale );
1600
1601 m_out->Print( ")" ); // Close `border` token.
1602
1603 m_out->Print( "(separators" );
1604 KICAD_FORMAT::FormatBool( m_out, "rows", aTable->StrokeRows() );
1605 KICAD_FORMAT::FormatBool( m_out, "cols", aTable->StrokeColumns() );
1606
1607 if( aTable->StrokeRows() || aTable->StrokeColumns() )
1609
1610 m_out->Print( ")" ); // Close `separators` token.
1611
1612 m_out->Print( "(column_widths" );
1613
1614 for( int col = 0; col < aTable->GetColCount(); ++col )
1615 {
1616 m_out->Print( " %s",
1617 EDA_UNIT_UTILS::FormatInternalUnits( schIUScale, aTable->GetColWidth( col ) ).c_str() );
1618 }
1619
1620 m_out->Print( ")" );
1621
1622 m_out->Print( "(row_heights" );
1623
1624 for( int row = 0; row < aTable->GetRowCount(); ++row )
1625 {
1626 m_out->Print( " %s",
1627 EDA_UNIT_UTILS::FormatInternalUnits( schIUScale, aTable->GetRowHeight( row ) ).c_str() );
1628 }
1629
1630 m_out->Print( ")" );
1631
1633
1634 if( aTable->IsLocked() )
1635 KICAD_FORMAT::FormatBool( m_out, "locked", true );
1636
1637 m_out->Print( "(cells" );
1638
1639 for( SCH_TABLECELL* cell : aTable->GetCells() )
1640 saveTextBox( cell );
1641
1642 m_out->Print( ")" ); // Close `cells` token.
1643 m_out->Print( ")" ); // Close `table` token.
1644
1645 if( aTable->GetFlags() & SKIP_STRUCT )
1646 delete aTable;
1647}
1648
1649
1651{
1652 // Don't write empty groups
1653 if( aGroup->GetItems().empty() )
1654 return;
1655
1656 m_out->Print( "(group %s", m_out->Quotew( aGroup->GetName() ).c_str() );
1657
1659
1660 if( aGroup->IsLocked() )
1661 KICAD_FORMAT::FormatBool( m_out, "locked", true );
1662
1663 if( aGroup->HasDesignBlockLink() )
1664 m_out->Print( "(lib_id \"%s\")", aGroup->GetDesignBlockLibId().Format().c_str() );
1665
1666 wxArrayString memberIds;
1667
1668 for( EDA_ITEM* member : aGroup->GetItems() )
1669 memberIds.Add( member->m_Uuid.AsString() );
1670
1671 memberIds.Sort();
1672
1673 m_out->Print( "(members" );
1674
1675 for( const wxString& memberId : memberIds )
1676 m_out->Print( " %s", m_out->Quotew( memberId ).c_str() );
1677
1678 m_out->Print( ")" ); // Close `members` token.
1679 m_out->Print( ")" ); // Close `group` token.
1680}
1681
1682
1683void SCH_IO_KICAD_SEXPR::saveInstances( const std::vector<SCH_SHEET_INSTANCE>& aInstances )
1684{
1685 if( aInstances.size() )
1686 {
1687 m_out->Print( "(sheet_instances" );
1688
1689 for( const SCH_SHEET_INSTANCE& instance : aInstances )
1690 {
1691 wxString path = instance.m_Path.AsString();
1692
1693 if( path.IsEmpty() )
1694 path = wxT( "/" ); // Root path
1695
1696 m_out->Print( "(path %s (page %s))",
1697 m_out->Quotew( path ).c_str(),
1698 m_out->Quotew( instance.m_PageNumber ).c_str() );
1699 }
1700
1701 m_out->Print( ")" ); // Close sheet instances token.
1702 }
1703}
1704
1705
1706void SCH_IO_KICAD_SEXPR::cacheLib( const wxString& aLibraryFileName,
1707 const std::map<std::string, UTF8>* aProperties )
1708{
1709 // Suppress font substitution warnings (RAII - automatically restored on scope exit)
1710 FONTCONFIG_REPORTER_SCOPE fontconfigScope( nullptr );
1711
1712 if( !m_cache || !m_cache->IsFile( aLibraryFileName ) || m_cache->IsFileChanged() )
1713 {
1714 int oldModifyHash = 1;
1715 bool isNewCache = false;
1716
1717 if( m_cache )
1718 oldModifyHash = m_cache->m_modHash;
1719 else
1720 isNewCache = true;
1721
1722 // a spectacular episode in memory management:
1723 delete m_cache;
1724 m_cache = new SCH_IO_KICAD_SEXPR_LIB_CACHE( aLibraryFileName );
1725
1726 if( !isBuffering( aProperties ) || ( isNewCache && m_cache->isLibraryPathValid() ) )
1727 {
1728 m_cache->Load();
1729 m_cache->m_modHash = oldModifyHash + 1;
1730 }
1731 }
1732}
1733
1734
1735bool SCH_IO_KICAD_SEXPR::isBuffering( const std::map<std::string, UTF8>* aProperties )
1736{
1737 return ( aProperties && aProperties->contains( SCH_IO_KICAD_SEXPR::PropBuffering ) );
1738}
1739
1740
1742{
1743 if( m_cache )
1744 return m_cache->GetModifyHash();
1745
1746 // If the cache hasn't been loaded, it hasn't been modified.
1747 return 0;
1748}
1749
1750
1751void SCH_IO_KICAD_SEXPR::EnumerateSymbolLib( wxArrayString& aSymbolNameList,
1752 const wxString& aLibraryPath,
1753 const std::map<std::string, UTF8>* aProperties )
1754{
1755 bool powerSymbolsOnly = ( aProperties && aProperties->contains( SYMBOL_LIBRARY_ADAPTER::PropPowerSymsOnly ) );
1756
1757 cacheLib( aLibraryPath, aProperties );
1758
1759 const LIB_SYMBOL_MAP& symbols = m_cache->m_symbols;
1760
1761 for( LIB_SYMBOL_MAP::const_iterator it = symbols.begin(); it != symbols.end(); ++it )
1762 {
1763 if( !powerSymbolsOnly || it->second->IsPower() )
1764 aSymbolNameList.Add( it->first );
1765 }
1766}
1767
1768
1769void SCH_IO_KICAD_SEXPR::EnumerateSymbolLib( std::vector<LIB_SYMBOL*>& aSymbolList,
1770 const wxString& aLibraryPath,
1771 const std::map<std::string, UTF8>* aProperties )
1772{
1773 bool powerSymbolsOnly = ( aProperties && aProperties->contains( SYMBOL_LIBRARY_ADAPTER::PropPowerSymsOnly ) );
1774
1775 cacheLib( aLibraryPath, aProperties );
1776
1777 const LIB_SYMBOL_MAP& symbols = m_cache->m_symbols;
1778
1779 for( LIB_SYMBOL_MAP::const_iterator it = symbols.begin(); it != symbols.end(); ++it )
1780 {
1781 if( !powerSymbolsOnly || it->second->IsPower() )
1782 aSymbolList.push_back( it->second );
1783 }
1784}
1785
1786
1787LIB_SYMBOL* SCH_IO_KICAD_SEXPR::LoadSymbol( const wxString& aLibraryPath,
1788 const wxString& aSymbolName,
1789 const std::map<std::string, UTF8>* aProperties )
1790{
1791 cacheLib( aLibraryPath, aProperties );
1792
1793 LIB_SYMBOL_MAP::const_iterator it = m_cache->m_symbols.find( aSymbolName );
1794
1795 // We no longer escape '/' in symbol names, but we used to.
1796 if( it == m_cache->m_symbols.end() && aSymbolName.Contains( '/' ) )
1797 it = m_cache->m_symbols.find( EscapeString( aSymbolName, CTX_LEGACY_LIBID ) );
1798
1799 if( it == m_cache->m_symbols.end() && aSymbolName.Contains( wxT( "{slash}" ) ) )
1800 {
1801 wxString unescaped = aSymbolName;
1802 unescaped.Replace( wxT( "{slash}" ), wxT( "/" ) );
1803 it = m_cache->m_symbols.find( unescaped );
1804 }
1805
1806 if( it == m_cache->m_symbols.end() )
1807 return nullptr;
1808
1809 return it->second;
1810}
1811
1812
1813void SCH_IO_KICAD_SEXPR::SaveSymbol( const wxString& aLibraryPath, const LIB_SYMBOL* aSymbol,
1814 const std::map<std::string, UTF8>* aProperties )
1815{
1816 cacheLib( aLibraryPath, aProperties );
1817
1818 m_cache->AddSymbol( aSymbol );
1819
1820 if( !isBuffering( aProperties ) )
1821 m_cache->Save();
1822}
1823
1824
1825void SCH_IO_KICAD_SEXPR::DeleteSymbol( const wxString& aLibraryPath, const wxString& aSymbolName,
1826 const std::map<std::string, UTF8>* aProperties )
1827{
1828 cacheLib( aLibraryPath, aProperties );
1829
1830 m_cache->DeleteSymbol( aSymbolName );
1831
1832 if( !isBuffering( aProperties ) )
1833 m_cache->Save();
1834}
1835
1836
1837void SCH_IO_KICAD_SEXPR::CreateLibrary( const wxString& aLibraryPath,
1838 const std::map<std::string, UTF8>* aProperties )
1839{
1840 wxFileName fn( aLibraryPath );
1841
1842 // Normalize the path: if it's a directory on the filesystem, ensure fn is marked as a
1843 // directory so that IsDir() checks work correctly. wxFileName::IsDir() only checks if
1844 // the path string ends with a separator, not if the path is actually a directory.
1845 if( !fn.IsDir() && wxFileName::DirExists( fn.GetFullPath() ) )
1846 fn.AssignDir( fn.GetFullPath() );
1847
1848 if( !fn.IsDir() )
1849 {
1850 if( fn.FileExists() )
1851 THROW_IO_ERROR( wxString::Format( _( "Symbol library file '%s' already exists." ), fn.GetFullPath() ) );
1852 }
1853 else
1854 {
1855 if( fn.DirExists() )
1856 THROW_IO_ERROR( wxString::Format( _( "Symbol library path '%s' already exists." ), fn.GetPath() ) );
1857 }
1858
1859 delete m_cache;
1860 m_cache = new SCH_IO_KICAD_SEXPR_LIB_CACHE( aLibraryPath );
1861 m_cache->SetModified();
1862 m_cache->Save();
1863 m_cache->Load(); // update m_writable and m_timestamp
1864}
1865
1866
1867bool SCH_IO_KICAD_SEXPR::DeleteLibrary( const wxString& aLibraryPath,
1868 const std::map<std::string, UTF8>* aProperties )
1869{
1870 wxFileName fn = aLibraryPath;
1871
1872 // Normalize the path: if it's a directory on the filesystem, ensure fn is marked as a
1873 // directory so that IsDir() checks work correctly.
1874 if( !fn.IsDir() && wxFileName::DirExists( fn.GetFullPath() ) )
1875 fn.AssignDir( fn.GetFullPath() );
1876
1877 if( !fn.FileExists() && !fn.DirExists() )
1878 return false;
1879
1880 // Some of the more elaborate wxRemoveFile() crap puts up its own wxLog dialog
1881 // we don't want that. we want bare metal portability with no UI here.
1882 if( !fn.IsDir() )
1883 {
1884 if( wxRemove( aLibraryPath ) )
1885 {
1886 THROW_IO_ERROR( wxString::Format( _( "Symbol library file '%s' cannot be deleted." ),
1887 aLibraryPath.GetData() ) );
1888 }
1889 }
1890 else
1891 {
1892 // This may be overly agressive. Perhaps in the future we should remove all of the *.kicad_sym
1893 // files and only delete the folder if it's empty.
1894 if( !fn.Rmdir( wxPATH_RMDIR_RECURSIVE ) )
1895 {
1896 THROW_IO_ERROR( wxString::Format( _( "Symbol library folder '%s' cannot be deleted." ),
1897 fn.GetPath() ) );
1898 }
1899 }
1900
1901 if( m_cache && m_cache->IsFile( aLibraryPath ) )
1902 {
1903 delete m_cache;
1904 m_cache = nullptr;
1905 }
1906
1907 return true;
1908}
1909
1910
1911void SCH_IO_KICAD_SEXPR::SaveLibrary( const wxString& aLibraryPath, const std::map<std::string, UTF8>* aProperties )
1912{
1913 if( !m_cache )
1914 m_cache = new SCH_IO_KICAD_SEXPR_LIB_CACHE( aLibraryPath );
1915
1916 wxString oldFileName = m_cache->GetFileName();
1917
1918 if( !m_cache->IsFile( aLibraryPath ) )
1919 m_cache->SetFileName( aLibraryPath );
1920
1921 // This is a forced save.
1922 m_cache->SetModified();
1923 m_cache->Save();
1924
1925 m_cache->SetFileName( oldFileName );
1926}
1927
1928
1929bool SCH_IO_KICAD_SEXPR::CanReadLibrary( const wxString& aLibraryPath ) const
1930{
1931 // Check if the path is a directory containing at least one .kicad_sym file
1932 if( wxFileName::DirExists( aLibraryPath ) )
1933 {
1934 wxDir dir( aLibraryPath );
1935
1936 if( dir.IsOpened() )
1937 {
1938 wxString filename;
1939 wxString filespec = wxT( "*." ) + wxString( FILEEXT::KiCadSymbolLibFileExtension );
1940
1941 if( dir.GetFirst( &filename, filespec, wxDIR_FILES ) )
1942 return true;
1943 }
1944
1945 return false;
1946 }
1947
1948 // Check for proper extension
1949 if( !SCH_IO::CanReadLibrary( aLibraryPath ) )
1950 return false;
1951
1952 // Above just checks for proper extension; now check that it actually exists
1953 wxFileName fn( aLibraryPath );
1954 return fn.IsOk() && fn.FileExists();
1955}
1956
1957
1958bool SCH_IO_KICAD_SEXPR::IsLibraryWritable( const wxString& aLibraryPath )
1959{
1960 wxFileName fn( aLibraryPath );
1961
1962 if( fn.FileExists() )
1963 return fn.IsFileWritable();
1964
1965 return fn.IsDirWritable();
1966}
1967
1968
1969void SCH_IO_KICAD_SEXPR::GetAvailableSymbolFields( std::vector<wxString>& aNames )
1970{
1971 if( !m_cache )
1972 return;
1973
1974 const LIB_SYMBOL_MAP& symbols = m_cache->m_symbols;
1975
1976 std::set<wxString> fieldNames;
1977
1978 for( LIB_SYMBOL_MAP::const_iterator it = symbols.begin(); it != symbols.end(); ++it )
1979 {
1980 std::map<wxString, wxString> chooserFields;
1981 it->second->GetChooserFields( chooserFields );
1982
1983 for( const auto& [name, value] : chooserFields )
1984 fieldNames.insert( name );
1985 }
1986
1987 std::copy( fieldNames.begin(), fieldNames.end(), std::back_inserter( aNames ) );
1988}
1989
1990
1991void SCH_IO_KICAD_SEXPR::GetDefaultSymbolFields( std::vector<wxString>& aNames )
1992{
1993 GetAvailableSymbolFields( aNames );
1994}
1995
1996
1997std::vector<LIB_SYMBOL*> SCH_IO_KICAD_SEXPR::ParseLibSymbols( std::string& aSymbolText, std::string aSource,
1998 int aFileVersion )
1999{
2000 LIB_SYMBOL* newSymbol = nullptr;
2001 LIB_SYMBOL_MAP map;
2002
2003 std::vector<LIB_SYMBOL*> newSymbols;
2004 std::unique_ptr<STRING_LINE_READER> reader = std::make_unique<STRING_LINE_READER>( aSymbolText,
2005 aSource );
2006
2007 do
2008 {
2009 SCH_IO_KICAD_SEXPR_PARSER parser( reader.get() );
2010
2011 newSymbol = parser.ParseSymbol( map, aFileVersion );
2012
2013 if( newSymbol )
2014 newSymbols.emplace_back( newSymbol );
2015
2016 reader.reset( new STRING_LINE_READER( *reader ) );
2017 }
2018 while( newSymbol );
2019
2020 return newSymbols;
2021}
2022
2023
2025{
2026 SCH_IO_KICAD_SEXPR_LIB_CACHE::SaveSymbol( symbol, formatter );
2027}
2028
2029
2030const char* SCH_IO_KICAD_SEXPR::PropBuffering = "buffering";
const char * name
constexpr EDA_IU_SCALE schIUScale
Definition base_units.h:127
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition box2.h:990
wxString GetMajorMinorVersion()
Get only the major and minor version in a string major.minor.
This class handle bitmap images in KiCad.
Definition bitmap_base.h:49
bool SaveImageData(wxOutputStream &aOutStream) const
Write the bitmap data to aOutStream.
int GetPPI() const
wxImage * GetImageData()
Definition bitmap_base.h:68
const LIB_ID & GetDesignBlockLibId() const
Definition eda_group.h:73
std::unordered_set< EDA_ITEM * > & GetItems()
Definition eda_group.h:54
wxString GetName() const
Definition eda_group.h:51
bool HasDesignBlockLink() const
Definition eda_group.h:70
A base class for most all the KiCad significant classes used in schematics and boards.
Definition eda_item.h:100
void SetFlags(EDA_ITEM_FLAGS aMask)
Definition eda_item.h:149
const KIID m_Uuid
Definition eda_item.h:528
KICAD_T Type() const
Returns the type of object.
Definition eda_item.h:112
void ClearFlags(EDA_ITEM_FLAGS aMask=EDA_ITEM_ALL_FLAGS)
Definition eda_item.h:151
bool IsSelected() const
Definition eda_item.h:129
EDA_ITEM * GetParent() const
Definition eda_item.h:114
virtual void SetParent(EDA_ITEM *aParent)
Definition eda_item.cpp:93
EDA_ITEM_FLAGS GetFlags() const
Definition eda_item.h:152
FILL_T GetFillMode() const
Definition eda_shape.h:158
SHAPE_T GetShape() const
Definition eda_shape.h:185
const VECTOR2I & GetEnd() const
Return the ending point of the graphic.
Definition eda_shape.h:232
const VECTOR2I & GetStart() const
Return the starting point of the graphic.
Definition eda_shape.h:190
COLOR4D GetFillColor() const
Definition eda_shape.h:169
wxString SHAPE_T_asString() const
int GetTextHeight() const
Definition eda_text.h:292
bool IsDefaultFormatting() const
const EDA_ANGLE & GetTextAngle() const
Definition eda_text.h:172
virtual const wxString & GetText() const
Return the string associated with the text object.
Definition eda_text.h:114
virtual bool IsVisible() const
Definition eda_text.h:212
virtual void Format(OUTPUTFORMATTER *aFormatter, int aControlBits) const
Output the object to aFormatter in s-expression form.
EE_TYPE OfType(KICAD_T aType) const
Definition sch_rtree.h:225
A LINE_READER that reads from an open file.
Definition richio.h:158
void Rewind()
Rewind the file and resets the line number back to zero.
Definition richio.h:207
char * ReadLine() override
Read a line of text into the buffer and increments the line number counter.
Definition richio.cpp:212
RAII class to set and restore the fontconfig reporter.
Definition reporter.h:336
PROGRESS_REPORTER * m_progressReporter
Progress reporter to track the progress of the operation, may be nullptr.
Definition io_base.h:240
virtual bool CanReadLibrary(const wxString &aFileName) const
Checks if this IO object can read the specified library file/directory.
Definition io_base.cpp:71
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
virtual const wxString What() const
A composite of Problem() and Where()
double r
Red component.
Definition color4d.h:393
double g
Green component.
Definition color4d.h:394
double a
Alpha component.
Definition color4d.h:396
double b
Blue component.
Definition color4d.h:395
bool MakeRelativeTo(const KIID_PATH &aPath)
Definition kiid.cpp:320
wxString AsString() const
Definition kiid.cpp:365
Definition kiid.h:48
UTF8 Format() const
Definition lib_id.cpp:119
Define a library symbol object.
Definition lib_symbol.h:83
An abstract class from which implementation specific LINE_READERs may be derived to read single lines...
Definition richio.h:66
static LOAD_INFO_REPORTER & GetInstance()
Definition reporter.cpp:249
An interface used to output 8 bit text in a convenient way.
Definition richio.h:295
void Format(OUTPUTFORMATTER *aFormatter) const
Output the page class to aFormatter in s-expression form.
bool Finish() override
Runs prettification over the buffered bytes, writes them to the sibling temp file,...
Definition richio.cpp:700
virtual const wxString GetProjectPath() const
Return the full path of the project.
Definition project.cpp:187
A REFERENCE_IMAGE is a wrapper around a BITMAP_IMAGE that is displayed in an editor as a reference fo...
VECTOR2I GetPosition() const
const BITMAP_BASE & GetImage() const
Get the underlying image.
double GetImageScale() const
Holds all the data relating to one schematic.
Definition schematic.h:89
SCH_SHEET_LIST Hierarchy() const
Return the full schematic flattened hierarchical sheet list.
PROJECT & Project() const
Return a reference to the project this schematic is part of.
Definition schematic.h:104
bool IsValid() const
A simple test if the schematic is loaded, not a complete one.
Definition schematic.h:173
SCH_SHEET & Root() const
Definition schematic.h:133
std::vector< SCH_SHEET * > GetTopLevelSheets() const
Get the list of top-level sheets.
Object to handle a bitmap image that can be inserted in a schematic.
Definition sch_bitmap.h:40
REFERENCE_IMAGE & GetReferenceImage()
Definition sch_bitmap.h:54
Base class for a bus or wire entry.
VECTOR2I GetSize() const
VECTOR2I GetPosition() const override
virtual STROKE_PARAMS GetStroke() const override
VECTOR2I GetEnd() const
bool IsMandatory() const
VECTOR2I GetPosition() const override
bool IsNameShown() const
Definition sch_field.h:216
virtual const wxString & GetText() const override
Return the string associated with the text object.
Definition sch_field.h:126
wxString GetCanonicalName() const
Get a non-language-specific name for a field which can be used for storage, variable look-up,...
wxString GetName(bool aUseDefaultName=true) const
Return the field name (not translated).
bool CanAutoplace() const
Definition sch_field.h:227
A set of SCH_ITEMs (i.e., without duplicates).
Definition sch_group.h:52
A cache assistant for the KiCad s-expression symbol libraries.
static void SaveSymbol(LIB_SYMBOL *aSymbol, OUTPUTFORMATTER &aFormatter, const wxString &aLibName=wxEmptyString, bool aIncludeData=true)
Object to parser s-expression symbol library and schematic file formats.
void ParseSchematic(SCH_SHEET *aSheet, bool aIsCopyablyOnly=false, int aFileVersion=SEXPR_SCHEMATIC_FILE_VERSION)
Parse the internal LINE_READER object into aSheet.
LIB_SYMBOL * ParseSymbol(LIB_SYMBOL_MAP &aSymbolLibMap, int aFileVersion=SEXPR_SYMBOL_LIB_FILE_VERSION)
Parse internal LINE_READER object into symbols and return all found.
wxString m_path
Root project path for loading child sheets.
void GetDefaultSymbolFields(std::vector< wxString > &aNames) override
Retrieves a list of (custom) field names that should be shown by default for this library in the symb...
void saveShape(SCH_SHAPE *aShape)
void SaveSchematicFile(const wxString &aFileName, SCH_SHEET *aSheet, SCHEMATIC *aSchematic, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Write aSchematic to a storage file in a format that this SCH_IO implementation knows about,...
void SaveLibrary(const wxString &aLibraryPath, const std::map< std::string, UTF8 > *aProperties=nullptr) override
SCH_SHEET_PATH m_currentSheetPath
void saveGroup(SCH_GROUP *aGroup)
void LoadContent(LINE_READER &aReader, SCH_SHEET *aSheet, int aVersion=SEXPR_SCHEMATIC_FILE_VERSION)
void EnumerateSymbolLib(wxArrayString &aSymbolNameList, const wxString &aLibraryPath, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Populate a list of LIB_SYMBOL alias names contained within the library aLibraryPath.
void FormatSchematicToFormatter(OUTPUTFORMATTER *aOut, SCH_SHEET *aSheet, SCHEMATIC *aSchematic, const std::map< std::string, UTF8 > *aProperties=nullptr)
Serialize a schematic sheet to an OUTPUTFORMATTER without file I/O or Prettify.
bool m_appending
Schematic load append status.
std::vector< SCH_SHEET * > m_loadedRootSheets
Root sheets from previous LoadSchematicFile() calls, enabling screen reuse across top-level sheets th...
int m_version
Version of file being loaded.
void loadFile(const wxString &aFileName, SCH_SHEET *aSheet)
static void FormatLibSymbol(LIB_SYMBOL *aPart, OUTPUTFORMATTER &aFormatter)
void DeleteSymbol(const wxString &aLibraryPath, const wxString &aSymbolName, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Delete the entire LIB_SYMBOL associated with aAliasName from the library aLibraryPath.
void loadHierarchy(const SCH_SHEET_PATH &aParentSheetPath, SCH_SHEET *aSheet)
void SaveSymbol(const wxString &aLibraryPath, const LIB_SYMBOL *aSymbol, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Write aSymbol to an existing library located at aLibraryPath.
static std::vector< LIB_SYMBOL * > ParseLibSymbols(std::string &aSymbolText, std::string aSource, int aFileVersion=SEXPR_SCHEMATIC_FILE_VERSION)
OUTPUTFORMATTER * m_out
The formatter for saving SCH_SCREEN objects.
SCH_SHEET * LoadSchematicFile(const wxString &aFileName, SCHEMATIC *aSchematic, SCH_SHEET *aAppendToMe=nullptr, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Load information from some input file format that this SCH_IO implementation knows about,...
LIB_SYMBOL * LoadSymbol(const wxString &aLibraryPath, const wxString &aAliasName, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Load a LIB_SYMBOL object having aPartName from the aLibraryPath containing a library format that this...
wxString m_error
For throwing exceptions or errors on partial loads.
void saveInstances(const std::vector< SCH_SHEET_INSTANCE > &aSheets)
bool isBuffering(const std::map< std::string, UTF8 > *aProperties)
static const char * PropBuffering
The property used internally by the plugin to enable cache buffering which prevents the library file ...
SCH_SHEET * m_rootSheet
The root sheet of the schematic being loaded.
void cacheLib(const wxString &aLibraryFileName, const std::map< std::string, UTF8 > *aProperties)
void saveRuleArea(SCH_RULE_AREA *aRuleArea)
void saveField(SCH_FIELD *aField)
SCH_IO_KICAD_SEXPR_LIB_CACHE * m_cache
void Format(SCH_SHEET *aSheet)
bool IsLibraryWritable(const wxString &aLibraryPath) override
Return true if the library at aLibraryPath is writable.
void init(SCHEMATIC *aSchematic, const std::map< std::string, UTF8 > *aProperties=nullptr)
initialize PLUGIN like a constructor would.
bool DeleteLibrary(const wxString &aLibraryPath, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Delete an existing library and returns true, or if library does not exist returns false,...
void saveBitmap(const SCH_BITMAP &aBitmap)
void saveText(SCH_TEXT *aText)
void saveSheet(SCH_SHEET *aSheet, const SCH_SHEET_LIST &aSheetList)
int GetModifyHash() const override
Return the modification hash from the library cache.
bool CanReadLibrary(const wxString &aLibraryPath) const override
Checks if this IO object can read the specified library file/directory.
void saveLine(SCH_LINE *aLine)
void saveNoConnect(SCH_NO_CONNECT *aNoConnect)
void saveTable(SCH_TABLE *aTable)
std::stack< wxString > m_currentPath
Stack to maintain nested sheet paths.
void CreateLibrary(const wxString &aLibraryPath, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Create a new empty library at aLibraryPath empty.
void saveJunction(SCH_JUNCTION *aJunction)
std::function< bool(wxString aTitle, int aIcon, wxString aMsg, wxString aAction)> m_queryUserCallback
void saveTextBox(SCH_TEXTBOX *aText)
void saveSymbol(SCH_SYMBOL *aSymbol, const SCHEMATIC &aSchematic, const SCH_SHEET_LIST &aSheetList, bool aForClipboard, const SCH_SHEET_PATH *aRelativePath=nullptr)
void saveBusEntry(SCH_BUS_ENTRY_BASE *aBusEntry)
void GetAvailableSymbolFields(std::vector< wxString > &aNames) override
Retrieves a list of (custom) field names that are present on symbols in this library.
SCH_IO(const wxString &aName)
Definition sch_io.h:375
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition sch_item.h:168
int GetBodyStyle() const
Definition sch_item.h:248
bool IsLocked() const override
Definition sch_item.cpp:152
bool IsPrivate() const
Definition sch_item.h:254
SCH_LAYER_ID GetLayer() const
Return the layer this item is on.
Definition sch_item.h:344
AUTOPLACE_ALGO GetFieldsAutoplaced() const
Return whether the fields have been automatically placed.
Definition sch_item.h:629
wxString GetClass() const override
Return the class name.
Definition sch_item.h:178
COLOR4D GetColor() const
int GetDiameter() const
VECTOR2I GetPosition() const override
SPIN_STYLE GetSpinStyle() const
LABEL_FLAG_SHAPE GetShape() const
Definition sch_label.h:182
std::vector< SCH_FIELD > & GetFields()
Definition sch_label.h:214
Segment description base class to describe items which have 2 end points (track, wire,...
Definition sch_line.h:42
virtual STROKE_PARAMS GetStroke() const override
Definition sch_line.h:201
VECTOR2I GetEndPoint() const
Definition sch_line.h:148
VECTOR2I GetStartPoint() const
Definition sch_line.h:139
void SetEndPoint(const VECTOR2I &aPosition)
Definition sch_line.h:149
VECTOR2I GetPosition() const override
bool GetExcludedFromSim(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
bool GetExcludedFromBOM(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
bool GetDNP(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
Set or clear the 'Do Not Populate' flag.
bool GetExcludedFromBoard(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
const PAGE_INFO & GetPageSettings() const
Definition sch_screen.h:141
const std::map< wxString, LIB_SYMBOL * > & GetLibSymbols() const
Fetch a list of unique LIB_SYMBOL object pointers required to properly render each SCH_SYMBOL in this...
Definition sch_screen.h:498
EE_RTREE & Items()
Get the full RTree, usually for iterating.
Definition sch_screen.h:119
const wxString & GetFileName() const
Definition sch_screen.h:154
void SetFileName(const wxString &aFileName)
Set the file name for this screen to aFileName.
TITLE_BLOCK & GetTitleBlock()
Definition sch_screen.h:165
wxString GroupsSanityCheck(bool repair=false)
Consistency check of internal m_groups structure.
KIID m_uuid
A unique identifier for each schematic file.
Definition sch_screen.h:736
void SetFileReadOnly(bool aIsReadOnly)
Definition sch_screen.h:156
void SetFileExists(bool aFileExists)
Definition sch_screen.h:159
SCH_SCREEN * GetScreen()
STROKE_PARAMS GetStroke() const override
Definition sch_shape.h:61
A container for handling SCH_SHEET_PATH objects in a flattened hierarchy.
std::optional< SCH_SHEET_PATH > GetSheetPathByKIIDPath(const KIID_PATH &aPath, bool aIncludeLastSheet=true) const
Finds a SCH_SHEET_PATH that matches the provided KIID_PATH.
Handle access to a stack of flattened SCH_SHEET objects by way of a path for creating a flattened sch...
bool empty() const
Forwarded method from std::vector.
KIID_PATH Path() const
Get the sheet path as an KIID_PATH.
SCH_SCREEN * LastScreen()
void push_back(SCH_SHEET *aSheet)
Forwarded method from std::vector.
void pop_back()
Forwarded method from std::vector.
Define a sheet pin (label) used in sheets to create hierarchical schematics.
Sheet symbol placed in a schematic, and is the entry point for a sub schematic.
Definition sch_sheet.h:48
wxString GetFileName() const
Return the filename corresponding to this sheet.
Definition sch_sheet.h:374
bool HasRootInstance() const
Check to see if this sheet has a root sheet instance.
std::vector< SCH_FIELD > & GetFields()
Return a reference to the vector holding the sheet's fields.
Definition sch_sheet.h:91
bool GetExcludedFromBOM(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
bool GetExcludedFromSim(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
VECTOR2I GetSize() const
Definition sch_sheet.h:145
SCH_SCREEN * GetScreen() const
Definition sch_sheet.h:143
VECTOR2I GetPosition() const override
Definition sch_sheet.h:494
void SetScreen(SCH_SCREEN *aScreen)
Set the SCH_SCREEN associated with this sheet to aScreen.
const SCH_SHEET_INSTANCE & GetRootInstance() const
Return the root sheet instance data.
KIGFX::COLOR4D GetBorderColor() const
Definition sch_sheet.h:151
bool GetDNP(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
Set or clear the 'Do Not Populate' flags.
bool GetExcludedFromBoard(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
Definition sch_sheet.h:465
int GetBorderWidth() const
Definition sch_sheet.h:148
std::vector< SCH_SHEET_PIN * > & GetPins()
Definition sch_sheet.h:231
const std::vector< SCH_SHEET_INSTANCE > & GetInstances() const
Definition sch_sheet.h:509
KIGFX::COLOR4D GetBackgroundColor() const
Definition sch_sheet.h:154
Schematic symbol object.
Definition sch_symbol.h:76
bool GetExcludedFromSim(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
std::vector< std::unique_ptr< SCH_PIN > > & GetRawPins()
Definition sch_symbol.h:667
const std::vector< SCH_SYMBOL_INSTANCE > & GetInstances() const
Definition sch_symbol.h:135
bool UseLibIdLookup() const
Definition sch_symbol.h:182
wxString GetSchSymbolLibraryName() const
bool GetExcludedFromBOM(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
void GetFields(std::vector< SCH_FIELD * > &aVector, bool aVisibleOnly) const override
Populate a std::vector with SCH_FIELDs, sorted in ordinal order.
VECTOR2I GetPosition() const override
Definition sch_symbol.h:871
const LIB_ID & GetLibId() const override
Definition sch_symbol.h:165
bool GetExcludedFromPosFiles(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
bool GetInstance(SCH_SYMBOL_INSTANCE &aInstance, const KIID_PATH &aSheetPath, bool aTestFromEnd=false) const
bool GetExcludedFromBoard(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
int GetOrientation() const override
Get the display symbol orientation.
virtual bool GetDNP(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
Set or clear the 'Do Not Populate' flag.
wxString GetPrefix() const
Definition sch_symbol.h:238
void SetRowHeight(int aRow, int aHeight)
Definition sch_table.h:137
const STROKE_PARAMS & GetSeparatorsStroke() const
Definition sch_table.h:77
void SetColCount(int aCount)
Definition sch_table.h:119
bool StrokeExternal() const
Definition sch_table.h:53
int GetRowHeight(int aRow) const
Definition sch_table.h:139
void SetColWidth(int aCol, int aWidth)
Definition sch_table.h:127
std::vector< SCH_TABLECELL * > GetCells() const
Definition sch_table.h:157
int GetColWidth(int aCol) const
Definition sch_table.h:129
const STROKE_PARAMS & GetBorderStroke() const
Definition sch_table.h:59
int GetColCount() const
Definition sch_table.h:120
bool StrokeHeaderSeparator() const
Definition sch_table.h:56
void DeleteMarkedCells()
Definition sch_table.h:182
SCH_TABLECELL * GetCell(int aRow, int aCol) const
Definition sch_table.h:147
bool StrokeColumns() const
Definition sch_table.h:99
EDA_ITEM * Clone() const override
Create a duplicate of this item with linked list members set to NULL.
Definition sch_table.h:231
bool StrokeRows() const
Definition sch_table.h:102
int GetRowCount() const
Definition sch_table.h:122
int GetMarginBottom() const
Definition sch_textbox.h:76
int GetMarginLeft() const
Definition sch_textbox.h:73
bool GetExcludedFromSim(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
int GetMarginRight() const
Definition sch_textbox.h:75
int GetMarginTop() const
Definition sch_textbox.h:74
bool GetExcludedFromSim(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
Definition sch_text.h:93
VECTOR2I GetPosition() const override
Definition sch_text.h:150
Is a LINE_READER that reads from a multiline 8 bit wide std::string.
Definition richio.h:226
Simple container to manage line stroke parameters.
void SetWidth(int aWidth)
void Format(OUTPUTFORMATTER *out, const EDA_IU_SCALE &aIuScale) const
static const char * PropPowerSymsOnly
virtual void Format(OUTPUTFORMATTER *aFormatter) const
Output the object to aFormatter in s-expression form.
const char * c_str() const
Definition utf8.h:108
wxString wx_str() const
Definition utf8.cpp:45
#define _(s)
static constexpr EDA_ANGLE ANGLE_0
Definition eda_angle.h:411
static constexpr EDA_ANGLE ANGLE_90
Definition eda_angle.h:413
static constexpr EDA_ANGLE ANGLE_270
Definition eda_angle.h:416
static constexpr EDA_ANGLE ANGLE_180
Definition eda_angle.h:415
#define STRUCT_DELETED
flag indication structures to be erased
#define SKIP_STRUCT
flag indicating that the structure should be ignored
@ RECTANGLE
Use RECTANGLE instead of RECT to avoid collision in a Windows header.
Definition eda_shape.h:49
#define DEFAULT_SIZE_TEXT
This is the "default-of-the-default" hardcoded text size; individual application define their own def...
Definition eda_text.h:83
static const std::string KiCadSymbolLibFileExtension
const wxChar *const traceSchPlugin
Flag to enable legacy schematic plugin debug output.
#define THROW_IO_ERROR(msg)
macro which captures the "call site" values of FILE_, __FUNCTION & LINE
KIID niluuid(0)
wxString LayerName(int aLayer)
Returns the default display name for a given layer.
Definition layer_id.cpp:31
@ LAYER_WIRE
Definition layer_ids.h:454
@ LAYER_NOTES
Definition layer_ids.h:469
@ LAYER_BUS
Definition layer_ids.h:455
#define UNIMPLEMENTED_FOR(type)
Definition macros.h:96
KICOMMON_API std::string FormatAngle(const EDA_ANGLE &aAngle)
Convert aAngle from board units to a string appropriate for writing to file.
KICOMMON_API std::string FormatInternalUnits(const EDA_IU_SCALE &aIuScale, int aValue, EDA_DATA_TYPE aDataType=EDA_DATA_TYPE::DISTANCE)
Converts aValue from internal units to a string appropriate for writing to file.
void FormatUuid(OUTPUTFORMATTER *aOut, const KIID &aUuid)
void FormatStreamData(OUTPUTFORMATTER &aOut, const wxStreamBuffer &aStream)
Write binary data to the formatter as base 64 encoded string.
void FormatBool(OUTPUTFORMATTER *aOut, const wxString &aKey, bool aValue)
Writes a boolean to the formatter, in the style (aKey [yes|no])
#define SEXPR_SCHEMATIC_FILE_VERSION
Schematic file version.
Class to handle a set of SCH_ITEMs.
void formatArc(OUTPUTFORMATTER *aFormatter, EDA_SHAPE *aArc, bool aIsPrivate, const STROKE_PARAMS &aStroke, FILL_T aFillMode, const COLOR4D &aFillColor, bool aInvertY, const KIID &aUuid, bool aLocked)
void formatCircle(OUTPUTFORMATTER *aFormatter, EDA_SHAPE *aCircle, bool aIsPrivate, const STROKE_PARAMS &aStroke, FILL_T aFillMode, const COLOR4D &aFillColor, bool aInvertY, const KIID &aUuid, bool aLocked)
const char * getSheetPinShapeToken(LABEL_FLAG_SHAPE aShape)
void formatRect(OUTPUTFORMATTER *aFormatter, EDA_SHAPE *aRect, bool aIsPrivate, const STROKE_PARAMS &aStroke, FILL_T aFillMode, const COLOR4D &aFillColor, bool aInvertY, const KIID &aUuid, bool aLocked)
const char * getTextTypeToken(KICAD_T aType)
void formatBezier(OUTPUTFORMATTER *aFormatter, EDA_SHAPE *aBezier, bool aIsPrivate, const STROKE_PARAMS &aStroke, FILL_T aFillMode, const COLOR4D &aFillColor, bool aInvertY, const KIID &aUuid, bool aLocked)
void formatPoly(OUTPUTFORMATTER *aFormatter, EDA_SHAPE *aPolyLine, bool aIsPrivate, const STROKE_PARAMS &aStroke, FILL_T aFillMode, const COLOR4D &aFillColor, bool aInvertY, const KIID &aUuid, bool aLocked)
EDA_ANGLE getSheetPinAngle(SHEET_SIDE aSide)
void formatFill(OUTPUTFORMATTER *aFormatter, FILL_T aFillMode, const COLOR4D &aFillColor)
Fill token formatting helper.
AUTOPLACE_ALGO
Definition sch_item.h:69
@ AUTOPLACE_MANUAL
Definition sch_item.h:72
@ AUTOPLACE_AUTO
Definition sch_item.h:71
std::string toUTFTildaText(const wxString &txt)
Convert a wxString to UTF8 and replace any control characters with a ~, where a control character is ...
const int scale
std::string FormatDouble2Str(double aValue)
Print a float number without using scientific notation and no trailing 0 This function is intended in...
std::string EscapedUTF8(const wxString &aString)
Return an 8 bit UTF8 string given aString in Unicode form.
wxString EscapeString(const wxString &aSource, ESCAPE_CONTEXT aContext)
The Escape/Unescape routines use HTML-entity-reference-style encoding to handle characters which are:...
#define TO_UTF8(wxstring)
Convert a wxString to a UTF8 encoded C string for all wxWidgets build modes.
@ CTX_LEGACY_LIBID
A simple container for sheet instance information.
A simple container for schematic symbol instance information.
@ SYM_ORIENT_270
Definition symbol.h:42
@ SYM_MIRROR_Y
Definition symbol.h:44
@ SYM_ORIENT_180
Definition symbol.h:41
@ SYM_MIRROR_X
Definition symbol.h:43
@ SYM_ORIENT_90
Definition symbol.h:40
std::map< wxString, LIB_SYMBOL *, LibSymbolMapSort > LIB_SYMBOL_MAP
FIELD_T
The set of all field indices assuming an array like sequence that a SCH_COMPONENT or LIB_PART can hol...
@ REFERENCE
Field Reference of part, i.e. "IC21".
std::string path
KIBIS_PIN * pin
std::vector< std::vector< std::string > > table
wxLogTrace helper definitions.
@ SCH_GROUP_T
Definition typeinfo.h:174
@ SCH_TABLE_T
Definition typeinfo.h:166
@ SCH_LINE_T
Definition typeinfo.h:164
@ SCH_NO_CONNECT_T
Definition typeinfo.h:161
@ SCH_SYMBOL_T
Definition typeinfo.h:173
@ SCH_TABLECELL_T
Definition typeinfo.h:167
@ SCH_DIRECTIVE_LABEL_T
Definition typeinfo.h:172
@ SCH_LABEL_T
Definition typeinfo.h:168
@ SCH_SHEET_T
Definition typeinfo.h:176
@ SCH_MARKER_T
Definition typeinfo.h:159
@ SCH_SHAPE_T
Definition typeinfo.h:150
@ SCH_RULE_AREA_T
Definition typeinfo.h:171
@ SCH_HIER_LABEL_T
Definition typeinfo.h:170
@ SCH_BUS_BUS_ENTRY_T
Definition typeinfo.h:163
@ SCH_TEXT_T
Definition typeinfo.h:152
@ SCH_BUS_WIRE_ENTRY_T
Definition typeinfo.h:162
@ SCH_BITMAP_T
Definition typeinfo.h:165
@ SCH_TEXTBOX_T
Definition typeinfo.h:153
@ SCH_GLOBAL_LABEL_T
Definition typeinfo.h:169
@ SCH_JUNCTION_T
Definition typeinfo.h:160
VECTOR2< int32_t > VECTOR2I
Definition vector2d.h:687
Definition of file extensions used in Kicad.