25#include <fmt/format.h>
29#include <wx/mstream.h>
41#include <schematic_lexer.h>
68using namespace TSCHEMATIC_T;
71#define SCH_PARSE_ERROR( text, reader, pos ) \
72 THROW_PARSE_ERROR( text, reader.GetSource(), reader.Line(), \
73 reader.LineNumber(), pos - reader.Line() )
89 const std::map<std::string, UTF8>* aProperties )
105 const std::map<std::string, UTF8>* aProperties )
107 wxASSERT( !aFileName || aSchematic !=
nullptr );
111 wxFileName fn = aFileName;
119 wxASSERT( fn.IsAbsolute() );
129 if( !normedFn.IsAbsolute() )
131 if( aFileName.Right( normedFn.GetFullPath().Length() ) == normedFn.GetFullPath() )
132 m_path = aFileName.Left( aFileName.Length() - normedFn.GetFullPath().Length() );
146 init( aSchematic, aProperties );
148 if( aAppendToMe ==
nullptr )
151 std::unique_ptr<SCH_SHEET> newSheet = std::make_unique<SCH_SHEET>( aSchematic );
153 wxFileName relPath( aFileName );
159 newSheet->SetFileName( relPath.GetFullPath() );
164 sheet = newSheet.release();
170 wxCHECK_MSG( aSchematic->
IsValid(),
nullptr,
"Can't append to a schematic with no root!" );
199 if( !fileName.IsAbsolute() )
208 wxLogTrace(
traceSchPlugin,
"Loading '%s'", fileName.GetFullPath() );
212 while( !ancestorSheetPath.
empty() )
219 m_error += wxString::Format(
_(
"Could not load sheet '%s' because it already "
220 "appears as a direct ancestor in the schematic "
222 fileName.GetFullPath() );
224 fileName = wxEmptyString;
232 if( ancestorSheetPath.
empty() )
236 if( !
m_rootSheet->SearchHierarchy( fileName.GetFullPath(), &screen ) )
245 if( prevRoot->SearchHierarchy( fileName.GetFullPath(), &screen ) )
264 loadFile( fileName.GetFullPath(), aSheet );
279 if( fileName.FileExists() )
317 size_t lineCount = 0;
351 const std::map<std::string, UTF8>* aProperties )
353 wxCHECK_RET( aSheet !=
nullptr,
"NULL SCH_SHEET object." );
354 wxCHECK_RET( !aFileName.IsEmpty(),
"No schematic file name defined." );
361 wxString::Format(
_(
"Please report this bug. Error validating group "
362 "structure: %s\n\nSave anyway?" ),
364 _(
"Save Anyway" ) ) )
370 wxFileName fn = aFileName;
374 wxASSERT( fn.IsAbsolute() );
387 const std::map<std::string, UTF8>* aProperties )
389 wxCHECK_RET( aSheet !=
nullptr,
"NULL SCH_SHEET object." );
391 init( aSchematic, aProperties );
403 wxCHECK_RET( aSheet !=
nullptr,
"NULL SCH_SHEET* object." );
404 wxCHECK_RET(
m_schematic !=
nullptr,
"NULL SCHEMATIC* object." );
417 m_schematic->GetEmbeddedFiles()->ClearEmbeddedFonts();
419 m_out->Print(
"(kicad_sch (version %d) (generator \"eeschema\") (generator_version %s)",
429 m_out->Print(
"(lib_symbols" );
431 for(
const auto& [ libItemName, libSymbol ] : screen->
GetLibSymbols() )
440 if( a->
Type() != b->Type() )
441 return a->
Type() < b->Type();
443 return a->
m_Uuid < b->m_Uuid;
446 std::multiset<
SCH_ITEM*,
decltype( cmp )> save_map( cmp );
452 save_map.insert( item );
457 switch( item->Type() )
517 wxASSERT(
"Unexpected schematic object type in SCH_IO_KICAD_SEXPR::Format()" );
523 std::vector< SCH_SHEET_INSTANCE> instances;
543 wxCHECK( aSelection && aSelectionPath && aFormatter, );
550 std::map<wxString, LIB_SYMBOL*> libSymbols;
552 std::set<SCH_TABLE*> promotedTables;
569 libSymbols[ libSymbolLookup ] = it->second;
572 if( !libSymbols.empty() )
574 m_out->Print(
"(lib_symbols" );
576 for(
const auto& [
name, libSymbol] : libSymbols )
582 for(
EDA_ITEM* edaItem : *aSelection )
584 if( !edaItem->IsSCH_ITEM() )
589 switch( item->
Type() )
592 saveSymbol(
static_cast<SCH_SYMBOL*
>( item ), aSchematic, sheets, aForClipboard, aSelectionPath );
644 if( promotedTables.count(
table ) )
650 promotedTables.insert(
table );
664 wxASSERT(
"Unexpected schematic object type in SCH_IO_KICAD_SEXPR::Format()" );
674 wxCHECK_RET( aSymbol !=
nullptr &&
m_out !=
nullptr,
"" );
680 if( symbol_name.size() )
686 libName =
"_NONAME_";
701 m_out->Print(
"(symbol" );
705 m_out->Print(
"(lib_name %s)",
709 m_out->Print(
"(lib_id %s) (at %s %s %s)",
720 if( mirrorX || mirrorY )
722 m_out->Print(
"(mirror %s %s)",
724 mirrorY ?
"y" :
"" );
735 wxASSERT( parentScreen );
739 std::optional<SCH_SHEET_PATH> ordinalPath =
740 m_schematic->Hierarchy().GetOrdinalPath( parentScreen );
747 aSymbol->
GetInstance( ordinalInstance, ordinalPath->Path() );
752 int unit = ordinalInstance.
m_Unit;
754 if( aForClipboard && aRelativePath )
759 unit = unitInstance.
m_Unit;
762 m_out->Print(
"(unit %d)", unit );
781 std::vector<SCH_FIELD*> orderedFields;
782 aSymbol->
GetFields( orderedFields,
false );
787 wxString value = field->GetText();
796 else if( aForClipboard && aSymbol->
GetInstances().size() && aRelativePath
813 field->SetText( value );
819 field->SetText( value );
822 for(
const std::unique_ptr<SCH_PIN>&
pin : aSymbol->
GetRawPins() )
827 if(
pin->GetAlt().IsEmpty() || (
pin->GetAlt() ==
pin->GetBaseName() ) )
829 m_out->Print(
"(pin %s",
m_out->Quotew(
pin->GetNumber() ).c_str() );
835 m_out->Print(
"(pin %s",
m_out->Quotew(
pin->GetNumber() ).c_str() );
837 m_out->Print(
"(alternate %s))",
m_out->Quotew(
pin->GetAlt() ).c_str() );
843 std::map<KIID, std::vector<SCH_SYMBOL_INSTANCE>> projectInstances;
844 std::set<KIID> currentProjectKeys;
846 m_out->Print(
"(instances" );
848 wxString projectName;
858 currentProjectKeys.insert( sheet->m_Uuid );
862 currentProjectKeys.insert( rootSheetUuid );
868 wxCHECK2( inst.m_Path.size(),
continue );
880 if( rootSheetUuid ==
niluuid && !pathToCheck.empty() && pathToCheck[0] ==
niluuid )
882 if( pathToCheck.size() > 1 )
884 pathToCheck.erase( pathToCheck.begin() );
896 bool belongsToThisProject = currentProjectKeys.count( pathToCheck[0] );
901 if( !aForClipboard && isOrphaned )
905 KIID projectKey = pathToCheck[0];
906 auto it = projectInstances.find( projectKey );
908 if( it == projectInstances.end() )
909 projectInstances[ projectKey ] = { inst };
911 it->second.emplace_back( inst );
914 for(
auto& [uuid, instances] : projectInstances )
916 wxCHECK2( instances.size(),
continue );
919 std::sort( instances.begin(), instances.end(),
922 return aLhs.m_Path < aRhs.m_Path;
925 if( currentProjectKeys.count( uuid ) )
926 projectName =
m_schematic->Project().GetProjectName();
928 projectName = instances[0].m_ProjectName;
930 m_out->Print(
"(project %s",
m_out->Quotew( projectName ).c_str() );
937 if( aForClipboard && aRelativePath )
942 m_out->Print(
"(path %s (reference %s) (unit %d)",
944 m_out->Quotew( instance.m_Reference ).c_str(),
947 if( !instance.m_Variants.empty() )
949 for(
const auto&[
name, variant] : instance.m_Variants )
951 m_out->Print(
"(variant (name %s)",
m_out->Quotew(
name ).c_str() );
953 if( variant.m_DNP != aSymbol->
GetDNP() )
968 for(
const auto&[fname, fvalue] : variant.m_Fields )
970 m_out->Print(
"(field (name %s) (value %s))",
971 m_out->Quotew( fname ).c_str(),
m_out->Quotew( fvalue ).c_str() );
993 wxCHECK_RET( aField !=
nullptr &&
m_out !=
nullptr,
"" );
1000 fieldName = aField->
GetName();
1002 m_out->Print(
"(property %s %s %s (at %s %s %s)",
1004 m_out->Quotew( fieldName ).c_str(),
1025 m_out->Print(
")" );
1031 wxCHECK_RET(
m_out !=
nullptr,
"" );
1038 wxCHECK_RET(
image !=
nullptr,
"wxImage* is NULL" );
1040 m_out->Print(
"(image (at %s %s)",
1061 wxMemoryOutputStream stream;
1066 m_out->Print(
")" );
1072 wxCHECK_RET( aSheet !=
nullptr &&
m_out !=
nullptr,
"" );
1074 m_out->Print(
"(sheet (at %s %s) (size %s %s)",
1102 m_out->Print(
"(fill (color %d %d %d %s))",
1115 m_out->Print(
"(pin %s %s (at %s %s %s)",
1119 pin->GetPosition().x ).c_str(),
1121 pin->GetPosition().y ).c_str(),
1128 m_out->Print(
")" );
1132 std::vector< SCH_SHEET_INSTANCE > sheetInstances = aSheet->
GetInstances();
1134 auto it = sheetInstances.begin();
1136 while( it != sheetInstances.end() )
1138 if( it->m_Path.size() == 0 )
1139 it = sheetInstances.erase( it );
1144 if( !sheetInstances.empty() )
1146 m_out->Print(
"(instances" );
1148 KIID lastProjectUuid;
1150 bool inProjectClause =
false;
1152 std::set<KIID> currentProjectKeys;
1154 if( rootSheetUuid ==
niluuid )
1157 currentProjectKeys.insert( sheet->m_Uuid );
1161 currentProjectKeys.insert( rootSheetUuid );
1164 for(
size_t i = 0; i < sheetInstances.size(); i++ )
1171 bool belongsToThisProject =
1172 !sheetInstances[i].m_Path.empty() && currentProjectKeys.count( sheetInstances[i].m_Path[0] );
1176 if( inProjectClause && ( ( i + 1 == sheetInstances.size() )
1177 || lastProjectUuid != sheetInstances[i+1].m_Path[0] ) )
1179 m_out->Print(
")" );
1180 inProjectClause =
false;
1186 if( lastProjectUuid != sheetInstances[i].m_Path[0] )
1188 wxString projectName;
1190 if( belongsToThisProject )
1191 projectName =
m_schematic->Project().GetProjectName();
1193 projectName = sheetInstances[i].m_ProjectName;
1195 lastProjectUuid = sheetInstances[i].m_Path[0];
1196 m_out->Print(
"(project %s",
m_out->Quotew( projectName ).c_str() );
1197 inProjectClause =
true;
1200 wxString
path = sheetInstances[i].m_Path.AsString();
1202 m_out->Print(
"(path %s (page %s)",
1204 m_out->Quotew( sheetInstances[i].m_PageNumber ).c_str() );
1206 if( !sheetInstances[i].m_Variants.empty() )
1208 for(
const auto&[
name, variant] : sheetInstances[i].m_Variants )
1210 m_out->Print(
"(variant (name %s)",
m_out->Quotew(
name ).c_str() );
1212 if( variant.m_DNP != aSheet->
GetDNP() )
1221 for(
const auto&[fname, fvalue] : variant.m_Fields )
1223 m_out->Print(
"(field (name %s) (value %s))",
1224 m_out->Quotew( fname ).c_str(),
m_out->Quotew( fvalue ).c_str() );
1227 m_out->Print(
")" );
1231 m_out->Print(
")" );
1233 if( inProjectClause && ( ( i + 1 == sheetInstances.size() )
1234 || lastProjectUuid != sheetInstances[i+1].m_Path[0] ) )
1236 m_out->Print(
")" );
1237 inProjectClause =
false;
1241 m_out->Print(
")" );
1244 m_out->Print(
")" );
1250 wxCHECK_RET( aJunction !=
nullptr &&
m_out !=
nullptr,
"" );
1252 m_out->Print(
"(junction (at %s %s) (diameter %s) (color %d %d %d %s)",
1269 m_out->Print(
")" );
1275 wxCHECK_RET( aNoConnect !=
nullptr &&
m_out !=
nullptr,
"" );
1277 m_out->Print(
"(no_connect (at %s %s)",
1288 m_out->Print(
")" );
1294 wxCHECK_RET( aBusEntry !=
nullptr &&
m_out !=
nullptr,
"" );
1297 if( aBusEntry->
GetClass() ==
"SCH_BUS_BUS_ENTRY" )
1306 m_out->Print(
"(bus_entry (at %s %s) (size %s %s)",
1314 aBusEntry->
GetSize().
y ).c_str() );
1322 m_out->Print(
")" );
1328 wxCHECK_RET( aShape !=
nullptr &&
m_out !=
nullptr,
"" );
1369 wxCHECK_RET( aRuleArea !=
nullptr &&
m_out !=
nullptr,
"" );
1371 m_out->Print(
"(rule_area " );
1383 m_out->Print(
")" );
1389 wxCHECK_RET( aLine !=
nullptr &&
m_out !=
nullptr,
"" );
1397 case LAYER_BUS: lineType =
"bus";
break;
1404 m_out->Print(
"(%s (pts (xy %s %s) (xy %s %s))",
1421 m_out->Print(
")" );
1427 wxCHECK_RET( aText !=
nullptr &&
m_out !=
nullptr,
"" );
1432 m_out->Print(
"(%s %s",
1443 m_out->Print(
"(length %s)",
1445 flag->GetPinLength() ).c_str() );
1471 m_out->Print(
"(at %s %s %s)",
1478 if( label && !label->
GetFields().empty() )
1486 aText->EDA_TEXT::Format(
m_out, 0 );
1498 m_out->Print(
")" );
1504 wxCHECK_RET( aTextBox !=
nullptr &&
m_out !=
nullptr,
"" );
1506 m_out->Print(
"(%s %s",
1515 m_out->Print(
"(at %s %s %s) (size %s %s) (margins %s %s %s %s)",
1527 m_out->Print(
"(span %d %d)", cell->GetColSpan(), cell->GetRowSpan() );
1533 aTextBox->EDA_TEXT::Format(
m_out, 0 );
1539 m_out->Print(
")" );
1554 for(
int row = 0; row < aTable->
GetRowCount(); ++row )
1556 for(
int col = 0; col < aTable->
GetColCount(); ++col )
1562 minRow = std::min( minRow, row );
1563 maxRow = std::max( maxRow, row );
1564 minCol = std::min( minCol, col );
1565 maxCol = std::max( maxCol, col );
1574 wxCHECK_MSG( maxCol >= minCol && maxRow >= minRow, , wxT(
"No selected cells!" ) );
1578 for(
int row = minRow; row <= maxRow; row++ )
1583 for(
int col = minCol; col <= maxCol; col++ )
1590 wxCHECK_RET( aTable !=
nullptr &&
m_out !=
nullptr,
"" );
1594 m_out->Print(
"(border" );
1601 m_out->Print(
")" );
1603 m_out->Print(
"(separators" );
1610 m_out->Print(
")" );
1612 m_out->Print(
"(column_widths" );
1614 for(
int col = 0; col < aTable->
GetColCount(); ++col )
1616 m_out->Print(
" %s",
1620 m_out->Print(
")" );
1622 m_out->Print(
"(row_heights" );
1624 for(
int row = 0; row < aTable->
GetRowCount(); ++row )
1626 m_out->Print(
" %s",
1630 m_out->Print(
")" );
1637 m_out->Print(
"(cells" );
1642 m_out->Print(
")" );
1643 m_out->Print(
")" );
1666 wxArrayString memberIds;
1669 memberIds.Add( member->m_Uuid.AsString() );
1673 m_out->Print(
"(members" );
1675 for(
const wxString& memberId : memberIds )
1676 m_out->Print(
" %s",
m_out->Quotew( memberId ).c_str() );
1678 m_out->Print(
")" );
1679 m_out->Print(
")" );
1685 if( aInstances.size() )
1687 m_out->Print(
"(sheet_instances" );
1691 wxString
path = instance.m_Path.AsString();
1693 if(
path.IsEmpty() )
1696 m_out->Print(
"(path %s (page %s))",
1698 m_out->Quotew( instance.m_PageNumber ).c_str() );
1701 m_out->Print(
")" );
1707 const std::map<std::string, UTF8>* aProperties )
1714 int oldModifyHash = 1;
1715 bool isNewCache =
false;
1718 oldModifyHash =
m_cache->m_modHash;
1729 m_cache->m_modHash = oldModifyHash + 1;
1744 return m_cache->GetModifyHash();
1752 const wxString& aLibraryPath,
1753 const std::map<std::string, UTF8>* aProperties )
1757 cacheLib( aLibraryPath, aProperties );
1761 for( LIB_SYMBOL_MAP::const_iterator it = symbols.begin(); it != symbols.end(); ++it )
1763 if( !powerSymbolsOnly || it->second->IsPower() )
1764 aSymbolNameList.Add( it->first );
1770 const wxString& aLibraryPath,
1771 const std::map<std::string, UTF8>* aProperties )
1775 cacheLib( aLibraryPath, aProperties );
1779 for( LIB_SYMBOL_MAP::const_iterator it = symbols.begin(); it != symbols.end(); ++it )
1781 if( !powerSymbolsOnly || it->second->IsPower() )
1782 aSymbolList.push_back( it->second );
1788 const wxString& aSymbolName,
1789 const std::map<std::string, UTF8>* aProperties )
1791 cacheLib( aLibraryPath, aProperties );
1793 LIB_SYMBOL_MAP::const_iterator it =
m_cache->m_symbols.find( aSymbolName );
1796 if( it ==
m_cache->m_symbols.end() && aSymbolName.Contains(
'/' ) )
1799 if( it ==
m_cache->m_symbols.end() && aSymbolName.Contains( wxT(
"{slash}" ) ) )
1801 wxString unescaped = aSymbolName;
1802 unescaped.Replace( wxT(
"{slash}" ), wxT(
"/" ) );
1803 it =
m_cache->m_symbols.find( unescaped );
1806 if( it ==
m_cache->m_symbols.end() )
1814 const std::map<std::string, UTF8>* aProperties )
1816 cacheLib( aLibraryPath, aProperties );
1818 m_cache->AddSymbol( aSymbol );
1826 const std::map<std::string, UTF8>* aProperties )
1828 cacheLib( aLibraryPath, aProperties );
1830 m_cache->DeleteSymbol( aSymbolName );
1838 const std::map<std::string, UTF8>* aProperties )
1840 wxFileName fn( aLibraryPath );
1845 if( !fn.IsDir() && wxFileName::DirExists( fn.GetFullPath() ) )
1846 fn.AssignDir( fn.GetFullPath() );
1850 if( fn.FileExists() )
1851 THROW_IO_ERROR( wxString::Format(
_(
"Symbol library file '%s' already exists." ), fn.GetFullPath() ) );
1855 if( fn.DirExists() )
1856 THROW_IO_ERROR( wxString::Format(
_(
"Symbol library path '%s' already exists." ), fn.GetPath() ) );
1868 const std::map<std::string, UTF8>* aProperties )
1870 wxFileName fn = aLibraryPath;
1874 if( !fn.IsDir() && wxFileName::DirExists( fn.GetFullPath() ) )
1875 fn.AssignDir( fn.GetFullPath() );
1877 if( !fn.FileExists() && !fn.DirExists() )
1884 if( wxRemove( aLibraryPath ) )
1886 THROW_IO_ERROR( wxString::Format(
_(
"Symbol library file '%s' cannot be deleted." ),
1887 aLibraryPath.GetData() ) );
1894 if( !fn.Rmdir( wxPATH_RMDIR_RECURSIVE ) )
1896 THROW_IO_ERROR( wxString::Format(
_(
"Symbol library folder '%s' cannot be deleted." ),
1916 wxString oldFileName =
m_cache->GetFileName();
1918 if( !
m_cache->IsFile( aLibraryPath ) )
1919 m_cache->SetFileName( aLibraryPath );
1925 m_cache->SetFileName( oldFileName );
1932 if( wxFileName::DirExists( aLibraryPath ) )
1934 wxDir dir( aLibraryPath );
1936 if( dir.IsOpened() )
1941 if( dir.GetFirst( &filename, filespec, wxDIR_FILES ) )
1953 wxFileName fn( aLibraryPath );
1954 return fn.IsOk() && fn.FileExists();
1960 wxFileName fn( aLibraryPath );
1962 if( fn.FileExists() )
1963 return fn.IsFileWritable();
1965 return fn.IsDirWritable();
1976 std::set<wxString> fieldNames;
1978 for( LIB_SYMBOL_MAP::const_iterator it = symbols.begin(); it != symbols.end(); ++it )
1980 std::map<wxString, wxString> chooserFields;
1981 it->second->GetChooserFields( chooserFields );
1983 for(
const auto& [
name, value] : chooserFields )
1984 fieldNames.insert(
name );
1987 std::copy( fieldNames.begin(), fieldNames.end(), std::back_inserter( aNames ) );
2003 std::vector<LIB_SYMBOL*> newSymbols;
2004 std::unique_ptr<STRING_LINE_READER> reader = std::make_unique<STRING_LINE_READER>( aSymbolText,
2011 newSymbol = parser.
ParseSymbol( map, aFileVersion );
2014 newSymbols.emplace_back( newSymbol );
constexpr EDA_IU_SCALE schIUScale
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
wxString GetMajorMinorVersion()
Get only the major and minor version in a string major.minor.
This class handle bitmap images in KiCad.
bool SaveImageData(wxOutputStream &aOutStream) const
Write the bitmap data to aOutStream.
const LIB_ID & GetDesignBlockLibId() const
std::unordered_set< EDA_ITEM * > & GetItems()
bool HasDesignBlockLink() const
A base class for most all the KiCad significant classes used in schematics and boards.
void SetFlags(EDA_ITEM_FLAGS aMask)
KICAD_T Type() const
Returns the type of object.
void ClearFlags(EDA_ITEM_FLAGS aMask=EDA_ITEM_ALL_FLAGS)
EDA_ITEM * GetParent() const
virtual void SetParent(EDA_ITEM *aParent)
EDA_ITEM_FLAGS GetFlags() const
FILL_T GetFillMode() const
const VECTOR2I & GetEnd() const
Return the ending point of the graphic.
const VECTOR2I & GetStart() const
Return the starting point of the graphic.
COLOR4D GetFillColor() const
wxString SHAPE_T_asString() const
int GetTextHeight() const
bool IsDefaultFormatting() const
const EDA_ANGLE & GetTextAngle() const
virtual const wxString & GetText() const
Return the string associated with the text object.
virtual bool IsVisible() const
virtual void Format(OUTPUTFORMATTER *aFormatter, int aControlBits) const
Output the object to aFormatter in s-expression form.
EE_TYPE OfType(KICAD_T aType) const
A LINE_READER that reads from an open file.
void Rewind()
Rewind the file and resets the line number back to zero.
char * ReadLine() override
Read a line of text into the buffer and increments the line number counter.
RAII class to set and restore the fontconfig reporter.
PROGRESS_REPORTER * m_progressReporter
Progress reporter to track the progress of the operation, may be nullptr.
virtual bool CanReadLibrary(const wxString &aFileName) const
Checks if this IO object can read the specified library file/directory.
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()
bool MakeRelativeTo(const KIID_PATH &aPath)
wxString AsString() const
Define a library symbol object.
An abstract class from which implementation specific LINE_READERs may be derived to read single lines...
static LOAD_INFO_REPORTER & GetInstance()
void Format(OUTPUTFORMATTER *aFormatter) const
Output the page class to aFormatter in s-expression form.
virtual const wxString GetProjectPath() const
Return the full path of the project.
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.
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.
bool IsValid() const
A simple test if the schematic is loaded, not a complete one.
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.
REFERENCE_IMAGE & GetReferenceImage()
Base class for a bus or wire entry.
VECTOR2I GetPosition() const override
virtual STROKE_PARAMS GetStroke() const override
VECTOR2I GetPosition() const override
virtual const wxString & GetText() const override
Return the string associated with the text object.
wxString GetCanonicalName() const
Get a non-language-specific name for a field which can be used for storage, variable look-up,...
wxString GetName(bool aUseDefaultName=true) const
Return the field name (not translated).
bool CanAutoplace() const
A set of SCH_ITEMs (i.e., without duplicates).
A cache assistant for the KiCad s-expression symbol libraries.
static void SaveSymbol(LIB_SYMBOL *aSymbol, OUTPUTFORMATTER &aFormatter, const wxString &aLibName=wxEmptyString, bool aIncludeData=true)
Object to parser s-expression symbol library and schematic file formats.
void ParseSchematic(SCH_SHEET *aSheet, bool aIsCopyablyOnly=false, int aFileVersion=SEXPR_SCHEMATIC_FILE_VERSION)
Parse the internal LINE_READER object into aSheet.
LIB_SYMBOL * ParseSymbol(LIB_SYMBOL_MAP &aSymbolLibMap, int aFileVersion=SEXPR_SYMBOL_LIB_FILE_VERSION)
Parse internal LINE_READER object into symbols and return all found.
wxString m_path
Root project path for loading child sheets.
void GetDefaultSymbolFields(std::vector< wxString > &aNames) override
Retrieves a list of (custom) field names that should be shown by default for this library in the symb...
void saveShape(SCH_SHAPE *aShape)
void SaveSchematicFile(const wxString &aFileName, SCH_SHEET *aSheet, SCHEMATIC *aSchematic, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Write aSchematic to a storage file in a format that this SCH_IO implementation knows about,...
void SaveLibrary(const wxString &aLibraryPath, const std::map< std::string, UTF8 > *aProperties=nullptr) override
SCH_SHEET_PATH m_currentSheetPath
void saveGroup(SCH_GROUP *aGroup)
void LoadContent(LINE_READER &aReader, SCH_SHEET *aSheet, int aVersion=SEXPR_SCHEMATIC_FILE_VERSION)
void EnumerateSymbolLib(wxArrayString &aSymbolNameList, const wxString &aLibraryPath, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Populate a list of LIB_SYMBOL alias names contained within the library aLibraryPath.
void FormatSchematicToFormatter(OUTPUTFORMATTER *aOut, SCH_SHEET *aSheet, SCHEMATIC *aSchematic, const std::map< std::string, UTF8 > *aProperties=nullptr)
Serialize a schematic sheet to an OUTPUTFORMATTER without file I/O or Prettify.
bool m_appending
Schematic load append status.
std::vector< SCH_SHEET * > m_loadedRootSheets
Root sheets from previous LoadSchematicFile() calls, enabling screen reuse across top-level sheets th...
int m_version
Version of file being loaded.
void loadFile(const wxString &aFileName, SCH_SHEET *aSheet)
static void FormatLibSymbol(LIB_SYMBOL *aPart, OUTPUTFORMATTER &aFormatter)
void DeleteSymbol(const wxString &aLibraryPath, const wxString &aSymbolName, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Delete the entire LIB_SYMBOL associated with aAliasName from the library aLibraryPath.
void loadHierarchy(const SCH_SHEET_PATH &aParentSheetPath, SCH_SHEET *aSheet)
void SaveSymbol(const wxString &aLibraryPath, const LIB_SYMBOL *aSymbol, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Write aSymbol to an existing library located at aLibraryPath.
static std::vector< LIB_SYMBOL * > ParseLibSymbols(std::string &aSymbolText, std::string aSource, int aFileVersion=SEXPR_SCHEMATIC_FILE_VERSION)
OUTPUTFORMATTER * m_out
The formatter for saving SCH_SCREEN objects.
SCH_SHEET * LoadSchematicFile(const wxString &aFileName, SCHEMATIC *aSchematic, SCH_SHEET *aAppendToMe=nullptr, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Load information from some input file format that this SCH_IO implementation knows about,...
LIB_SYMBOL * LoadSymbol(const wxString &aLibraryPath, const wxString &aAliasName, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Load a LIB_SYMBOL object having aPartName from the aLibraryPath containing a library format that this...
wxString m_error
For throwing exceptions or errors on partial loads.
void saveInstances(const std::vector< SCH_SHEET_INSTANCE > &aSheets)
bool isBuffering(const std::map< std::string, UTF8 > *aProperties)
static const char * PropBuffering
The property used internally by the plugin to enable cache buffering which prevents the library file ...
SCH_SHEET * m_rootSheet
The root sheet of the schematic being loaded.
void cacheLib(const wxString &aLibraryFileName, const std::map< std::string, UTF8 > *aProperties)
void saveRuleArea(SCH_RULE_AREA *aRuleArea)
void saveField(SCH_FIELD *aField)
SCH_IO_KICAD_SEXPR_LIB_CACHE * m_cache
void Format(SCH_SHEET *aSheet)
bool IsLibraryWritable(const wxString &aLibraryPath) override
Return true if the library at aLibraryPath is writable.
void init(SCHEMATIC *aSchematic, const std::map< std::string, UTF8 > *aProperties=nullptr)
initialize PLUGIN like a constructor would.
bool DeleteLibrary(const wxString &aLibraryPath, const std::map< std::string, UTF8 > *aProperties=nullptr) override
Delete an existing library and returns true, or if library does not exist returns false,...
void saveBitmap(const SCH_BITMAP &aBitmap)
void saveText(SCH_TEXT *aText)
void saveSheet(SCH_SHEET *aSheet, const SCH_SHEET_LIST &aSheetList)
int GetModifyHash() const override
Return the modification hash from the library cache.
bool CanReadLibrary(const wxString &aLibraryPath) const override
Checks if this IO object can read the specified library file/directory.
virtual ~SCH_IO_KICAD_SEXPR()
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)
Base class for any item which can be embedded within the SCHEMATIC container class,...
bool IsLocked() const override
SCH_LAYER_ID GetLayer() const
Return the layer this item is on.
AUTOPLACE_ALGO GetFieldsAutoplaced() const
Return whether the fields have been automatically placed.
wxString GetClass() const override
Return the class name.
VECTOR2I GetPosition() const override
SPIN_STYLE GetSpinStyle() const
LABEL_FLAG_SHAPE GetShape() const
std::vector< SCH_FIELD > & GetFields()
Segment description base class to describe items which have 2 end points (track, wire,...
virtual STROKE_PARAMS GetStroke() const override
VECTOR2I GetEndPoint() const
VECTOR2I GetStartPoint() const
void SetEndPoint(const VECTOR2I &aPosition)
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
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...
EE_RTREE & Items()
Get the full RTree, usually for iterating.
const wxString & GetFileName() const
void SetFileName(const wxString &aFileName)
Set the file name for this screen to aFileName.
TITLE_BLOCK & GetTitleBlock()
wxString GroupsSanityCheck(bool repair=false)
Consistency check of internal m_groups structure.
KIID m_uuid
A unique identifier for each schematic file.
void SetFileReadOnly(bool aIsReadOnly)
void SetFileExists(bool aFileExists)
STROKE_PARAMS GetStroke() const override
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.
wxString GetFileName() const
Return the filename corresponding to this sheet.
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.
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
SCH_SCREEN * GetScreen() const
VECTOR2I GetPosition() const override
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
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
int GetBorderWidth() const
std::vector< SCH_SHEET_PIN * > & GetPins()
const std::vector< SCH_SHEET_INSTANCE > & GetInstances() const
KIGFX::COLOR4D GetBackgroundColor() const
bool GetExcludedFromSim(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
std::vector< std::unique_ptr< SCH_PIN > > & GetRawPins()
const std::vector< SCH_SYMBOL_INSTANCE > & GetInstances() const
bool UseLibIdLookup() const
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
const LIB_ID & GetLibId() const override
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
void SetRowHeight(int aRow, int aHeight)
const STROKE_PARAMS & GetSeparatorsStroke() const
void SetColCount(int aCount)
bool StrokeExternal() const
int GetRowHeight(int aRow) const
void SetColWidth(int aCol, int aWidth)
std::vector< SCH_TABLECELL * > GetCells() const
int GetColWidth(int aCol) const
const STROKE_PARAMS & GetBorderStroke() const
bool StrokeHeaderSeparator() const
SCH_TABLECELL * GetCell(int aRow, int aCol) const
bool StrokeColumns() const
EDA_ITEM * Clone() const override
Create a duplicate of this item with linked list members set to NULL.
int GetMarginBottom() const
int GetMarginLeft() const
bool GetExcludedFromSim(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
int GetMarginRight() const
bool GetExcludedFromSim(const SCH_SHEET_PATH *aInstance=nullptr, const wxString &aVariantName=wxEmptyString) const override
VECTOR2I GetPosition() const override
Is a LINE_READER that reads from a multiline 8 bit wide std::string.
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
static constexpr EDA_ANGLE ANGLE_0
static constexpr EDA_ANGLE ANGLE_90
static constexpr EDA_ANGLE ANGLE_270
static constexpr EDA_ANGLE ANGLE_180
#define STRUCT_DELETED
flag indication structures to be erased
#define SKIP_STRUCT
flag indicating that the structure should be ignored
@ RECTANGLE
Use RECTANGLE instead of RECT to avoid collision in a Windows header.
#define DEFAULT_SIZE_TEXT
This is the "default-of-the-default" hardcoded text size; individual application define their own def...
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
wxString LayerName(int aLayer)
Returns the default display name for a given layer.
#define UNIMPLEMENTED_FOR(type)
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.
#define SEXPR_SCHEMATIC_FILE_VERSION
Schematic file version.
Class to handle a set of SCH_ITEMs.
void formatArc(OUTPUTFORMATTER *aFormatter, EDA_SHAPE *aArc, bool aIsPrivate, const STROKE_PARAMS &aStroke, FILL_T aFillMode, const COLOR4D &aFillColor, bool aInvertY, const KIID &aUuid, bool aLocked)
void formatCircle(OUTPUTFORMATTER *aFormatter, EDA_SHAPE *aCircle, bool aIsPrivate, const STROKE_PARAMS &aStroke, FILL_T aFillMode, const COLOR4D &aFillColor, bool aInvertY, const KIID &aUuid, bool aLocked)
const char * getSheetPinShapeToken(LABEL_FLAG_SHAPE aShape)
void formatRect(OUTPUTFORMATTER *aFormatter, EDA_SHAPE *aRect, bool aIsPrivate, const STROKE_PARAMS &aStroke, FILL_T aFillMode, const COLOR4D &aFillColor, bool aInvertY, const KIID &aUuid, bool aLocked)
const char * getTextTypeToken(KICAD_T aType)
void formatBezier(OUTPUTFORMATTER *aFormatter, EDA_SHAPE *aBezier, bool aIsPrivate, const STROKE_PARAMS &aStroke, FILL_T aFillMode, const COLOR4D &aFillColor, bool aInvertY, const KIID &aUuid, bool aLocked)
void formatPoly(OUTPUTFORMATTER *aFormatter, EDA_SHAPE *aPolyLine, bool aIsPrivate, const STROKE_PARAMS &aStroke, FILL_T aFillMode, const COLOR4D &aFillColor, bool aInvertY, const KIID &aUuid, bool aLocked)
EDA_ANGLE getSheetPinAngle(SHEET_SIDE aSide)
void formatFill(OUTPUTFORMATTER *aFormatter, FILL_T aFillMode, const COLOR4D &aFillColor)
Fill token formatting helper.
std::string toUTFTildaText(const wxString &txt)
Convert a wxString to UTF8 and replace any control characters with a ~, where a control character is ...
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.
A simple container for sheet instance information.
A simple container for schematic symbol instance information.
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::vector< std::vector< std::string > > table
wxLogTrace helper definitions.
VECTOR2< int32_t > VECTOR2I
Definition of file extensions used in Kicad.