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