kicad/eeschema/sch_io/easyedapro/sch_easyedapro_parser.cpp

1483 lines
51 KiB
C++

/*
* This program source code file is part of KiCad, a free EDA CAD application.
*
* Copyright (C) 2023 Alex Shvartzkop <dudesuchamazing@gmail.com>
* Copyright (C) 2023-2024 KiCad Developers, see AUTHORS.txt for contributors.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, you may find one here:
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
* or you may search the http://www.gnu.org website for the version 2 license,
* or you may write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
*/
#include "sch_easyedapro_parser.h"
#include <io/easyedapro/easyedapro_import_utils.h>
#include <core/map_helpers.h>
#include <sch_io/sch_io_mgr.h>
#include <schematic.h>
#include <sch_sheet.h>
#include <sch_sheet_pin.h>
#include <sch_line.h>
#include <sch_bitmap.h>
#include <lib_shape.h>
#include <lib_text.h>
#include <sch_no_connect.h>
#include <sch_label.h>
#include <sch_junction.h>
#include <sch_edit_frame.h>
#include <sch_shape.h>
#include <sch_bus_entry.h>
#include <string_utils.h>
#include <bezier_curves.h>
#include <wx/base64.h>
#include <wx/url.h>
#include <wx/mstream.h>
#include <gfx_import_utils.h>
#include <import_gfx/svg_import_plugin.h>
#include <import_gfx/graphics_importer_lib_symbol.h>
#include <import_gfx/graphics_importer_sch.h>
// clang-format off
static const std::vector<wxString> c_attributesWhitelist = { "Value",
"Datasheet",
"Manufacturer Part",
"Manufacturer",
"BOM_Manufacturer Part",
"BOM_Manufacturer",
"Supplier Part",
"Supplier",
"BOM_Supplier Part",
"BOM_Supplier",
"LCSC Part Name" };
// clang-format on
SCH_EASYEDAPRO_PARSER::SCH_EASYEDAPRO_PARSER( SCHEMATIC* aSchematic,
PROGRESS_REPORTER* aProgressReporter )
{
m_schematic = aSchematic;
}
SCH_EASYEDAPRO_PARSER::~SCH_EASYEDAPRO_PARSER()
{
}
double SCH_EASYEDAPRO_PARSER::Convert( wxString aValue )
{
double value = 0;
if( !aValue.ToCDouble( &value ) )
THROW_IO_ERROR( wxString::Format( _( "Failed to parse value: '%s'" ), aValue ) );
return value;
}
double SCH_EASYEDAPRO_PARSER::SizeToKi( wxString aValue )
{
return ScaleSize( Convert( aValue ) );
}
static LINE_STYLE ConvertStrokeStyle( int aStyle )
{
if( aStyle == 0 )
return LINE_STYLE::SOLID;
else if( aStyle == 1 )
return LINE_STYLE::DASH;
else if( aStyle == 2 )
return LINE_STYLE::DOT;
else if( aStyle == 3 )
return LINE_STYLE::DASHDOT;
return LINE_STYLE::DEFAULT;
}
template <typename T>
void SCH_EASYEDAPRO_PARSER::ApplyFontStyle( const std::map<wxString, nlohmann::json>& fontStyles,
T& text, const wxString& styleStr )
{
auto it = fontStyles.find( styleStr );
if( it == fontStyles.end() )
return;
nlohmann::json style = it->second;
if( !style.is_array() )
return;
if( style.size() < 12 )
return;
if( style.at( 3 ).is_string() )
{
COLOR4D color( style.at( 3 ).get<wxString>() );
text->SetTextColor( color );
}
if( style.at( 4 ).is_string() )
{
wxString fontname = ( style.at( 4 ) );
if( !fontname.IsSameAs( wxS( "default" ), false ) )
text->SetFont( KIFONT::FONT::GetFont( fontname ) );
}
if( style.at( 5 ).is_number() )
{
double size = style.at( 5 ).get<double>() * 0.5;
text->SetTextSize( VECTOR2I( ScaleSize( size ), ScaleSize( size ) ) );
}
if( style.at( 10 ).is_number() )
{
int valign = style.at( 10 );
if( !text->GetText().Contains( wxS( "\n" ) ) )
{
if( valign == 0 )
text->SetVertJustify( GR_TEXT_V_ALIGN_TOP );
else if( valign == 1 )
text->SetVertJustify( GR_TEXT_V_ALIGN_CENTER );
else if( valign == 2 )
text->SetVertJustify( GR_TEXT_V_ALIGN_BOTTOM );
}
else
{
text->SetVertJustify( GR_TEXT_V_ALIGN_TOP );
// TODO: align by first line
}
}
else
{
text->SetVertJustify( GR_TEXT_V_ALIGN_BOTTOM );
}
if( style.at( 11 ).is_number() )
{
int halign = style.at( 11 );
if( halign == 0 )
text->SetHorizJustify( GR_TEXT_H_ALIGN_LEFT );
else if( halign == 1 )
text->SetHorizJustify( GR_TEXT_H_ALIGN_CENTER );
else if( halign == 2 )
text->SetHorizJustify( GR_TEXT_H_ALIGN_RIGHT );
}
else
{
text->SetHorizJustify( GR_TEXT_H_ALIGN_LEFT );
}
}
template <typename T>
void SCH_EASYEDAPRO_PARSER::ApplyLineStyle( const std::map<wxString, nlohmann::json>& lineStyles,
T& shape, const wxString& styleStr )
{
auto it = lineStyles.find( styleStr );
if( it == lineStyles.end() )
return;
nlohmann::json style = it->second;
if( !style.is_array() )
return;
if( style.size() < 6 )
return;
STROKE_PARAMS stroke = shape->GetStroke();
if( style.at( 2 ).is_string() )
{
wxString colorStr = style.at( 2 ).get<wxString>();
if( !colorStr.empty() && colorStr.starts_with( wxS( "#" ) ) )
{
COLOR4D color( colorStr );
stroke.SetColor( color );
}
}
if( style.at( 3 ).is_number() )
{
int dashStyle = style.at( 3 );
stroke.SetLineStyle( ConvertStrokeStyle( dashStyle ) );
}
if( style.at( 5 ).is_number() )
{
double thickness = style.at( 5 );
stroke.SetWidth( ScaleSize( thickness ) );
}
shape->SetStroke( stroke );
}
wxString SCH_EASYEDAPRO_PARSER::ResolveFieldVariables(
const wxString aInput, const std::map<wxString, wxString>& aDeviceAttributes )
{
wxString inputText = aInput;
wxString resolvedText;
int variableCount = 0;
// Resolve variables
// ={Variable1}text{Variable2}
do
{
if( !inputText.StartsWith( wxS( "={" ) ) )
return inputText;
resolvedText.Clear();
variableCount = 0;
for( size_t i = 1; i < inputText.size(); )
{
wxUniChar c = inputText[i++];
if( c == '{' )
{
wxString varName;
bool endFound = false;
while( i < inputText.size() )
{
c = inputText[i++];
if( c == '}' )
{
endFound = true;
break;
}
varName << c;
}
if( !endFound )
return inputText;
wxString varValue =
get_def( aDeviceAttributes, varName, wxString::Format( "{%s!}", varName ) );
resolvedText << varValue;
variableCount++;
}
else
{
resolvedText << c;
}
}
inputText = resolvedText;
} while( variableCount > 0 );
return resolvedText;
}
template <typename T>
void SCH_EASYEDAPRO_PARSER::ApplyAttrToField( const std::map<wxString, nlohmann::json>& fontStyles,
T* field, const EASYEDAPRO::SCH_ATTR& aAttr,
bool aIsSym, bool aToSym,
const std::map<wxString, wxString>& aDeviceAttributes,
SCH_SYMBOL* aParent )
{
EDA_TEXT* text = static_cast<EDA_TEXT*>( field );
text->SetText( ResolveFieldVariables( aAttr.value, aDeviceAttributes ) );
text->SetVisible( aAttr.keyVisible || aAttr.valVisible );
field->SetNameShown( aAttr.keyVisible );
if( aAttr.position )
{
field->SetPosition( !aIsSym ? ScalePos( *aAttr.position )
: ScalePosSym( *aAttr.position ) );
}
ApplyFontStyle( fontStyles, text, aAttr.fontStyle );
auto parent = aParent;
if( parent && parent->Type() == SCH_SYMBOL_T )
{
int orient = static_cast<SCH_SYMBOL*>( parent )->GetOrientation();
if( orient == SYM_ORIENT_180 )
{
text->SetVertJustify( static_cast<GR_TEXT_V_ALIGN_T>( -text->GetVertJustify() ) );
text->SetHorizJustify( static_cast<GR_TEXT_H_ALIGN_T>( -text->GetHorizJustify() ) );
}
else if( orient == SYM_MIRROR_X + SYM_ORIENT_0 )
{
text->SetVertJustify( static_cast<GR_TEXT_V_ALIGN_T>( -text->GetVertJustify() ) );
}
else if( orient == SYM_MIRROR_Y + SYM_ORIENT_0 )
{
text->SetHorizJustify( static_cast<GR_TEXT_H_ALIGN_T>( -text->GetHorizJustify() ) );
}
else if( orient == SYM_MIRROR_Y + SYM_ORIENT_180 )
{
text->SetHorizJustify( static_cast<GR_TEXT_H_ALIGN_T>( text->GetHorizJustify() ) );
}
else if( orient == SYM_ORIENT_90 )
{
text->SetTextAngle( ANGLE_VERTICAL );
text->SetVertJustify( static_cast<GR_TEXT_V_ALIGN_T>( -text->GetVertJustify() ) );
text->SetHorizJustify( static_cast<GR_TEXT_H_ALIGN_T>( -text->GetHorizJustify() ) );
}
if( orient == SYM_ORIENT_270 )
{
text->SetTextAngle( ANGLE_VERTICAL );
}
else if( orient == SYM_MIRROR_X + SYM_ORIENT_90 )
{
text->SetTextAngle( ANGLE_VERTICAL );
text->SetVertJustify( static_cast<GR_TEXT_V_ALIGN_T>( -text->GetVertJustify() ) );
text->SetHorizJustify( static_cast<GR_TEXT_H_ALIGN_T>( -text->GetHorizJustify() ) );
}
else if( orient == SYM_MIRROR_X + SYM_ORIENT_270 )
{
text->SetTextAngle( ANGLE_VERTICAL );
text->SetVertJustify( static_cast<GR_TEXT_V_ALIGN_T>( -text->GetVertJustify() ) );
}
else if( orient == SYM_MIRROR_Y + SYM_ORIENT_90 )
{
text->SetTextAngle( ANGLE_VERTICAL );
text->SetHorizJustify( static_cast<GR_TEXT_H_ALIGN_T>( -text->GetHorizJustify() ) );
}
else if( orient == SYM_MIRROR_Y + SYM_ORIENT_270 )
{
text->SetHorizJustify( static_cast<GR_TEXT_H_ALIGN_T>( text->GetHorizJustify() ) );
}
if( aAttr.rotation == 90 )
{
if( text->GetTextAngle() == ANGLE_HORIZONTAL )
text->SetTextAngle( ANGLE_VERTICAL );
else
text->SetTextAngle( ANGLE_HORIZONTAL );
if( orient == SYM_ORIENT_90 )
{
text->SetVertJustify( static_cast<GR_TEXT_V_ALIGN_T>( -text->GetVertJustify() ) );
text->SetHorizJustify( static_cast<GR_TEXT_H_ALIGN_T>( -text->GetHorizJustify() ) );
}
if( orient == SYM_ORIENT_270 )
{
text->SetVertJustify( static_cast<GR_TEXT_V_ALIGN_T>( -text->GetVertJustify() ) );
text->SetHorizJustify( static_cast<GR_TEXT_H_ALIGN_T>( -text->GetHorizJustify() ) );
}
else if( orient == SYM_MIRROR_X + SYM_ORIENT_90 )
{
text->SetVertJustify( static_cast<GR_TEXT_V_ALIGN_T>( -text->GetVertJustify() ) );
}
}
}
}
EASYEDAPRO::SYM_INFO
SCH_EASYEDAPRO_PARSER::ParseSymbol( const std::vector<nlohmann::json>& aLines,
const std::map<wxString, wxString>& aDeviceAttributes )
{
EASYEDAPRO::SYM_INFO symInfo;
std::unique_ptr<LIB_SYMBOL> ksymbolPtr = std::make_unique<LIB_SYMBOL>( wxEmptyString );
LIB_SYMBOL* ksymbol = ksymbolPtr.get();
std::map<wxString, nlohmann::json> lineStyles;
std::map<wxString, nlohmann::json> fontStyles;
std::map<wxString, int> partUnits;
std::map<int, std::map<wxString, EASYEDAPRO::SCH_ATTR>> unitAttributes;
std::map<int, std::map<wxString, std::vector<nlohmann::json>>> unitParentedLines;
int totalUnits = 0;
for( const nlohmann::json& line : aLines )
{
wxString type = line.at( 0 );
if( type == wxS( "LINESTYLE" ) )
lineStyles[line.at( 1 )] = line;
else if( type == wxS( "FONTSTYLE" ) )
fontStyles[line.at( 1 )] = line;
else if( type == wxS( "PART" ) )
partUnits[line.at( 1 )] = ++totalUnits;
}
symInfo.partUnits = partUnits;
ksymbol->SetUnitCount( totalUnits, false );
int currentUnit = 1;
for( const nlohmann::json& line : aLines )
{
wxString type = line.at( 0 );
if( type == wxS( "PART" ) )
{
currentUnit = partUnits.at( line.at( 1 ) );
}
else if( type == wxS( "RECT" ) )
{
VECTOR2D start( line.at( 2 ), line.at( 3 ) );
VECTOR2D end( line.at( 4 ), line.at( 5 ) );
wxString styleStr = line.at( 9 );
std::unique_ptr<LIB_SHAPE> rect =
std::make_unique<LIB_SHAPE>( ksymbol, SHAPE_T::RECTANGLE );
rect->SetStart( ScalePosSym( start ) );
rect->SetEnd( ScalePosSym( end ) );
rect->SetUnit( currentUnit );
ApplyLineStyle( lineStyles, rect, styleStr );
ksymbol->AddDrawItem( rect.release() );
}
else if( type == wxS( "CIRCLE" ) )
{
VECTOR2D center( line.at( 2 ), line.at( 3 ) );
double radius = line.at( 4 );
wxString styleStr = line.at( 5 );
std::unique_ptr<LIB_SHAPE> circle =
std::make_unique<LIB_SHAPE>( ksymbol, SHAPE_T::CIRCLE );
circle->SetCenter( ScalePosSym( center ) );
circle->SetEnd( circle->GetCenter() + VECTOR2I( ScaleSize( radius ), 0 ) );
circle->SetUnit( currentUnit );
ApplyLineStyle( lineStyles, circle, styleStr );
ksymbol->AddDrawItem( circle.release() );
}
else if( type == wxS( "ARC" ) )
{
VECTOR2D start( line.at( 2 ), line.at( 3 ) );
VECTOR2D mid( line.at( 4 ), line.at( 5 ) );
VECTOR2D end( line.at( 6 ), line.at( 7 ) );
wxString styleStr = line.at( 8 );
VECTOR2D kstart = ScalePosSym( start );
VECTOR2D kmid = ScalePosSym( mid );
VECTOR2D kend = ScalePosSym( end );
VECTOR2D kcenter = CalcArcCenter( kstart, kmid, kend );
std::unique_ptr<LIB_SHAPE> shape = std::make_unique<LIB_SHAPE>( ksymbol, SHAPE_T::ARC );
shape->SetStart( kstart );
shape->SetEnd( kend );
shape->SetCenter( kcenter );
if( SEG( start, end ).Side( mid ) != SEG( kstart, kend ).Side( shape->GetArcMid() ) )
{
shape->SetStart( kend );
shape->SetEnd( kstart );
}
shape->SetUnit( currentUnit );
ApplyLineStyle( lineStyles, shape, styleStr );
ksymbol->AddDrawItem( shape.release() );
}
else if( type == wxS( "POLY" ) )
{
std::vector<double> points = line.at( 2 );
wxString styleStr = line.at( 4 );
std::unique_ptr<LIB_SHAPE> shape =
std::make_unique<LIB_SHAPE>( ksymbol, SHAPE_T::POLY );
for( size_t i = 1; i < points.size(); i += 2 )
{
shape->AddPoint( ScalePosSym( VECTOR2D( points[i - 1], points[i] ) ) );
}
shape->SetUnit( currentUnit );
ApplyLineStyle( lineStyles, shape, styleStr );
ksymbol->AddDrawItem( shape.release() );
}
else if( type == wxS( "TEXT" ) )
{
VECTOR2D pos( line.at( 2 ), line.at( 3 ) );
double angle = line.at( 4 );
wxString textStr = line.at( 5 );
wxString fontStyleStr = line.at( 6 );
std::unique_ptr<LIB_TEXT> text = std::make_unique<LIB_TEXT>( ksymbol );
text->SetPosition( ScalePosSym( pos ) );
text->SetText( UnescapeHTML( textStr ) );
text->SetHorizJustify( GR_TEXT_H_ALIGN_LEFT );
text->SetVertJustify( GR_TEXT_V_ALIGN_TOP );
text->SetTextAngleDegrees( angle );
text->SetUnit( currentUnit );
ApplyFontStyle( fontStyles, text, fontStyleStr );
ksymbol->AddDrawItem( text.release() );
}
else if( type == wxS( "OBJ" ) )
{
VECTOR2D start, size;
wxString mimeType, data;
//double angle = 0;
int upsideDown = 0;
if( line.at( 3 ).is_number() )
{
start = VECTOR2D( line.at( 3 ), line.at( 4 ) );
size = VECTOR2D( line.at( 5 ), line.at( 6 ) );
//angle = line.at( 7 );
upsideDown = line.at( 8 );
wxString imageUrl = line.at( 9 );
if( imageUrl.BeforeFirst( ':' ) == wxS( "data" ) )
{
wxArrayString paramsArr =
wxSplit( imageUrl.AfterFirst( ':' ).BeforeFirst( ',' ), ';', '\0' );
data = imageUrl.AfterFirst( ',' );
if( paramsArr.size() > 0 )
{
mimeType = paramsArr[0];
}
}
}
else if( line.at( 3 ).is_string() )
{
mimeType = line.at( 3 ).get<wxString>().BeforeFirst( ';' );
start = VECTOR2D( line.at( 4 ), line.at( 5 ) );
size = VECTOR2D( line.at( 6 ), line.at( 7 ) );
//angle = line.at( 8 );
data = line.at( 9 ).get<wxString>();
}
if( mimeType.empty() || data.empty() )
continue;
wxMemoryBuffer buf = wxBase64Decode( data );
if( mimeType == wxS( "image/svg+xml" ) )
{
VECTOR2D offset = ScalePosSym( start );
SVG_IMPORT_PLUGIN svgImportPlugin;
GRAPHICS_IMPORTER_LIB_SYMBOL libsymImporter( ksymbol, 0 );
svgImportPlugin.SetImporter( &libsymImporter );
svgImportPlugin.LoadFromMemory( buf );
VECTOR2D imSize( svgImportPlugin.GetImageWidth(),
svgImportPlugin.GetImageHeight() );
VECTOR2D pixelScale( schIUScale.IUTomm( ScaleSize( size.x ) ) / imSize.x,
schIUScale.IUTomm( -ScaleSize( size.y ) ) / imSize.y );
if( upsideDown )
pixelScale.y *= -1;
libsymImporter.SetScale( pixelScale );
VECTOR2D offsetMM( schIUScale.IUTomm( offset.x ), schIUScale.IUTomm( offset.y ) );
libsymImporter.SetImportOffsetMM( offsetMM );
svgImportPlugin.Import();
// TODO: rotation
for( std::unique_ptr<EDA_ITEM>& item : libsymImporter.GetItems() )
ksymbol->AddDrawItem( static_cast<LIB_ITEM*>( item.release() ) );
}
else
{
wxMemoryInputStream memis( buf.GetData(), buf.GetDataLen() );
wxImage::SetDefaultLoadFlags( wxImage::GetDefaultLoadFlags()
& ~wxImage::Load_Verbose );
wxImage img;
if( img.LoadFile( memis, mimeType ) )
{
int dimMul = img.GetWidth() * img.GetHeight();
double maxPixels = 30000;
if( dimMul > maxPixels )
{
double scale = sqrt( maxPixels / dimMul );
img.Rescale( img.GetWidth() * scale, img.GetHeight() * scale );
}
VECTOR2D pixelScale( ScaleSize( size.x ) / img.GetWidth(),
-ScaleSize( size.y ) / img.GetHeight() );
// TODO: rotation
ConvertImageToLibShapes( ksymbol, 0, img, pixelScale, ScalePosSym( start ) );
}
}
}
else if( type == wxS( "HEAD" ) )
{
symInfo.head = line;
}
else if( type == wxS( "PIN" ) )
{
wxString pinId = line.at( 1 );
unitParentedLines[currentUnit][pinId].push_back( line );
}
else if( type == wxS( "ATTR" ) )
{
wxString parentId = line.at( 2 );
if( parentId.empty() )
{
EASYEDAPRO::SCH_ATTR attr = line;
unitAttributes[currentUnit].emplace( attr.key, attr );
}
else
{
unitParentedLines[currentUnit][parentId].push_back( line );
}
}
}
if( symInfo.head.symbolType == EASYEDAPRO::SYMBOL_TYPE::POWER_PORT
|| symInfo.head.symbolType == EASYEDAPRO::SYMBOL_TYPE::NETPORT )
{
ksymbol->SetPower();
ksymbol->GetReferenceField().SetText( wxS( "#PWR" ) );
ksymbol->GetReferenceField().SetVisible( false );
ksymbol->SetKeyWords( wxS( "power-flag" ) );
ksymbol->SetShowPinNames( false );
ksymbol->SetShowPinNumbers( false );
if( auto globalNetAttr = get_opt( unitAttributes[1], wxS( "Global Net Name" ) ) )
{
ApplyAttrToField( fontStyles, &ksymbol->GetValueField(), *globalNetAttr, true, true );
wxString globalNetname = globalNetAttr->value;
if( !globalNetname.empty() )
{
ksymbol->SetDescription( wxString::Format(
_( "Power symbol creates a global label with name '%s'" ),
globalNetname ) );
}
}
}
else
{
auto designatorAttr = get_opt( unitAttributes[1], wxS( "Designator" ) );
if( designatorAttr && !designatorAttr->value.empty() )
{
wxString symbolPrefix = designatorAttr->value;
if( symbolPrefix.EndsWith( wxS( "?" ) ) )
symbolPrefix.RemoveLast();
ksymbol->GetReferenceField().SetText( symbolPrefix );
}
for( const wxString& attrName : c_attributesWhitelist )
{
if( auto valOpt = get_opt( aDeviceAttributes, attrName ) )
{
if( valOpt->empty() )
continue;
LIB_FIELD* fd = ksymbol->FindField( attrName, true );
if( !fd )
{
fd = new LIB_FIELD( ksymbol->GetNextAvailableFieldId(), attrName );
ksymbol->AddField( fd );
}
wxString value = *valOpt;
value.Replace( wxS( "\u2103" ), wxS( "\u00B0C" ), true ); // ℃ -> °C
fd->SetText( value );
fd->SetVisible( false );
}
}
}
for( auto& [unitId, parentedLines] : unitParentedLines )
{
for( auto& [pinId, lines] : parentedLines )
{
std::optional<EASYEDAPRO::SYM_PIN> epin;
std::map<wxString, EASYEDAPRO::SCH_ATTR> pinAttributes;
for( const nlohmann::json& line : lines )
{
wxString type = line.at( 0 );
if( type == wxS( "ATTR" ) )
{
EASYEDAPRO::SCH_ATTR attr = line;
pinAttributes.emplace( attr.key, attr );
}
else if( type == wxS( "PIN" ) )
{
epin = line;
}
}
if( !epin )
continue;
EASYEDAPRO::PIN_INFO pinInfo;
pinInfo.pin = *epin;
std::unique_ptr<LIB_PIN> pin = std::make_unique<LIB_PIN>( ksymbol );
pin->SetUnit( unitId );
pin->SetLength( ScaleSize( epin->length ) );
pin->SetPosition( ScalePosSym( epin->position ) );
PIN_ORIENTATION orient = PIN_ORIENTATION::PIN_RIGHT;
if( epin->rotation == 0 )
orient = PIN_ORIENTATION::PIN_RIGHT;
if( epin->rotation == 90 )
orient = PIN_ORIENTATION::PIN_UP;
if( epin->rotation == 180 )
orient = PIN_ORIENTATION::PIN_LEFT;
if( epin->rotation == 270 )
orient = PIN_ORIENTATION::PIN_DOWN;
pin->SetOrientation( orient );
if( symInfo.head.symbolType == EASYEDAPRO::SYMBOL_TYPE::POWER_PORT )
{
pin->SetName( ksymbol->GetName() );
//pin->SetVisible( false );
}
else if( auto pinNameAttr = get_opt( pinAttributes, "NAME" ) )
{
pin->SetName( pinNameAttr->value );
pinInfo.name = pinNameAttr->value;
if( !pinNameAttr->valVisible )
pin->SetNameTextSize( schIUScale.MilsToIU( 1 ) );
}
if( auto pinNumAttr = get_opt( pinAttributes, "NUMBER" ) )
{
pin->SetNumber( pinNumAttr->value );
pinInfo.number = pinNumAttr->value;
if( !pinNumAttr->valVisible )
pin->SetNumberTextSize( schIUScale.MilsToIU( 1 ) );
}
if( symInfo.head.symbolType == EASYEDAPRO::SYMBOL_TYPE::POWER_PORT )
{
pin->SetType( ELECTRICAL_PINTYPE::PT_POWER_IN );
}
else if( auto pinTypeAttr = get_opt( pinAttributes, "Pin Type" ) )
{
if( pinTypeAttr->value == wxS( "IN" ) )
pin->SetType( ELECTRICAL_PINTYPE::PT_INPUT );
if( pinTypeAttr->value == wxS( "OUT" ) )
pin->SetType( ELECTRICAL_PINTYPE::PT_OUTPUT );
if( pinTypeAttr->value == wxS( "BI" ) )
pin->SetType( ELECTRICAL_PINTYPE::PT_BIDI );
}
if( get_opt( pinAttributes, "NO_CONNECT" ) )
pin->SetType( ELECTRICAL_PINTYPE::PT_NC );
if( pin->GetNumberTextSize() * int( pin->GetNumber().size() ) > pin->GetLength() )
pin->SetNumberTextSize( pin->GetLength() / pin->GetNumber().size() );
symInfo.pins.push_back( pinInfo );
ksymbol->AddDrawItem( pin.release() );
}
}
symInfo.symbolAttr = get_opt( unitAttributes[1], "Symbol" ); // TODO: per-unit
/*BOX2I bbox = ksymbol->GetBodyBoundingBox( 0, 0, true, true );
bbox.Inflate( schIUScale.MilsToIU( 10 ) );*/
/*ksymbol->GetReferenceField().SetVertJustify( GR_TEXT_V_ALIGN_BOTTOM );
ksymbol->GetReferenceField().SetHorizJustify( GR_TEXT_H_ALIGN_CENTER );
ksymbol->GetReferenceField().SetPosition( VECTOR2I( bbox.GetCenter().x, -bbox.GetTop() ) );
ksymbol->GetValueField().SetVertJustify( GR_TEXT_V_ALIGN_TOP );
ksymbol->GetValueField().SetHorizJustify( GR_TEXT_H_ALIGN_CENTER );
ksymbol->GetValueField().SetPosition( VECTOR2I( bbox.GetCenter().x, -bbox.GetBottom() ) );*/
symInfo.libSymbol = std::move( ksymbolPtr );
return symInfo;
}
void SCH_EASYEDAPRO_PARSER::ParseSchematic( SCHEMATIC* aSchematic, SCH_SHEET* aRootSheet,
const nlohmann::json& aProject,
std::map<wxString, EASYEDAPRO::SYM_INFO>& aSymbolMap,
const std::map<wxString, EASYEDAPRO::BLOB>& aBlobMap,
const std::vector<nlohmann::json>& aLines,
const wxString& aLibName )
{
std::vector<std::unique_ptr<SCH_ITEM>> createdItems;
std::map<wxString, std::vector<nlohmann::json>> parentedLines;
std::map<wxString, std::vector<nlohmann::json>> ruleLines;
std::map<wxString, nlohmann::json> lineStyles;
std::map<wxString, nlohmann::json> fontStyles;
for( const nlohmann::json& line : aLines )
{
wxString type = line.at( 0 );
if( type == wxS( "LINESTYLE" ) )
lineStyles[line.at( 1 )] = line;
else if( type == wxS( "FONTSTYLE" ) )
fontStyles[line.at( 1 )] = line;
}
for( const nlohmann::json& line : aLines )
{
wxString type = line.at( 0 );
if( type == wxS( "RECT" ) )
{
VECTOR2D start( line.at( 2 ), line.at( 3 ) );
VECTOR2D end( line.at( 4 ), line.at( 5 ) );
wxString styleStr = line.at( 9 );
std::unique_ptr<SCH_SHAPE> rect = std::make_unique<SCH_SHAPE>( SHAPE_T::RECTANGLE );
rect->SetStart( ScalePos( start ) );
rect->SetEnd( ScalePos( end ) );
ApplyLineStyle( lineStyles, rect, styleStr );
createdItems.push_back( std::move( rect ) );
}
else if( type == wxS( "CIRCLE" ) )
{
VECTOR2D center( line.at( 2 ), line.at( 3 ) );
double radius = line.at( 4 );
wxString styleStr = line.at( 5 );
std::unique_ptr<SCH_SHAPE> circle = std::make_unique<SCH_SHAPE>( SHAPE_T::CIRCLE );
circle->SetCenter( ScalePos( center ) );
circle->SetEnd( circle->GetCenter() + VECTOR2I( ScaleSize( radius ), 0 ) );
ApplyLineStyle( lineStyles, circle, styleStr );
createdItems.push_back( std::move( circle ) );
}
else if( type == wxS( "POLY" ) )
{
std::vector<double> points = line.at( 2 );
wxString styleStr = line.at( 4 );
SHAPE_LINE_CHAIN chain;
for( size_t i = 1; i < points.size(); i += 2 )
chain.Append( ScalePos( VECTOR2D( points[i - 1], points[i] ) ) );
for( int segId = 0; segId < chain.SegmentCount(); segId++ )
{
const SEG& seg = chain.CSegment( segId );
std::unique_ptr<SCH_LINE> schLine =
std::make_unique<SCH_LINE>( seg.A, LAYER_NOTES );
schLine->SetEndPoint( seg.B );
ApplyLineStyle( lineStyles, schLine, styleStr );
createdItems.push_back( std::move( schLine ) );
}
}
else if( type == wxS( "TEXT" ) )
{
VECTOR2D pos( line.at( 2 ), line.at( 3 ) );
double angle = line.at( 4 );
wxString textStr = line.at( 5 );
wxString fontStyleStr = line.at( 6 );
std::unique_ptr<SCH_TEXT> text =
std::make_unique<SCH_TEXT>( ScalePos( pos ), UnescapeHTML( textStr ) );
text->SetHorizJustify( GR_TEXT_H_ALIGN_LEFT );
text->SetVertJustify( GR_TEXT_V_ALIGN_TOP );
text->SetTextAngleDegrees( angle );
ApplyFontStyle( fontStyles, text, fontStyleStr );
createdItems.push_back( std::move( text ) );
}
else if( type == wxS( "OBJ" ) )
{
VECTOR2D start, size;
wxString mimeType, base64Data;
double angle = 0;
int flipped = 0;
if( line.at( 3 ).is_number() )
{
start = VECTOR2D( line.at( 3 ), line.at( 4 ) );
size = VECTOR2D( line.at( 5 ), line.at( 6 ) );
angle = line.at( 7 );
flipped = line.at( 8 );
wxString imageUrl = line.at( 9 );
if( imageUrl.BeforeFirst( ':' ) == wxS( "data" ) )
{
wxArrayString paramsArr =
wxSplit( imageUrl.AfterFirst( ':' ).BeforeFirst( ',' ), ';', '\0' );
base64Data = imageUrl.AfterFirst( ',' );
if( paramsArr.size() > 0 )
mimeType = paramsArr[0];
}
else if( imageUrl.BeforeFirst( ':' ) == wxS( "blob" ) )
{
wxString objectId = imageUrl.AfterLast( ':' );
if( auto blob = get_opt( aBlobMap, objectId ) )
{
wxString blobUrl = blob->url;
if( blobUrl.BeforeFirst( ':' ) == wxS( "data" ) )
{
wxArrayString paramsArr = wxSplit(
blobUrl.AfterFirst( ':' ).BeforeFirst( ',' ), ';', '\0' );
base64Data = blobUrl.AfterFirst( ',' );
if( paramsArr.size() > 0 )
mimeType = paramsArr[0];
}
}
}
}
else if( line.at( 3 ).is_string() )
{
mimeType = line.at( 3 ).get<wxString>().BeforeFirst( ';' );
start = VECTOR2D( line.at( 4 ), line.at( 5 ) );
size = VECTOR2D( line.at( 6 ), line.at( 7 ) );
angle = line.at( 8 );
base64Data = line.at( 9 ).get<wxString>();
}
VECTOR2D kstart = ScalePos( start );
VECTOR2D ksize = ScaleSize( size );
if( mimeType.empty() || base64Data.empty() )
continue;
wxMemoryBuffer buf = wxBase64Decode( base64Data );
if( mimeType == wxS( "image/svg+xml" ) )
{
SVG_IMPORT_PLUGIN svgImportPlugin;
GRAPHICS_IMPORTER_SCH schImporter;
svgImportPlugin.SetImporter( &schImporter );
svgImportPlugin.LoadFromMemory( buf );
VECTOR2D imSize( svgImportPlugin.GetImageWidth(),
svgImportPlugin.GetImageHeight() );
VECTOR2D pixelScale( schIUScale.IUTomm( ScaleSize( size.x ) ) / imSize.x,
schIUScale.IUTomm( ScaleSize( size.y ) ) / imSize.y );
schImporter.SetScale( pixelScale );
VECTOR2D offsetMM( schIUScale.IUTomm( kstart.x ), schIUScale.IUTomm( kstart.y ) );
schImporter.SetImportOffsetMM( offsetMM );
svgImportPlugin.Import();
for( std::unique_ptr<EDA_ITEM>& item : schImporter.GetItems() )
{
SCH_ITEM* schItem = static_cast<SCH_ITEM*>( item.release() );
for( double i = angle; i > 0; i -= 90 )
{
if( schItem->Type() == SCH_LINE_T )
{
// Lines need special handling for some reason
schItem->SetFlags( STARTPOINT );
schItem->Rotate( kstart );
schItem->ClearFlags( STARTPOINT );
schItem->SetFlags( ENDPOINT );
schItem->Rotate( kstart );
schItem->ClearFlags( ENDPOINT );
}
else
{
schItem->Rotate( kstart );
}
}
if( flipped )
{
// Lines need special handling for some reason
if( schItem->Type() == SCH_LINE_T )
schItem->SetFlags( STARTPOINT | ENDPOINT );
schItem->MirrorHorizontally( kstart.x );
if( schItem->Type() == SCH_LINE_T )
schItem->ClearFlags( STARTPOINT | ENDPOINT );
}
createdItems.emplace_back( schItem );
}
}
else
{
std::unique_ptr<SCH_BITMAP> bitmap = std::make_unique<SCH_BITMAP>();
wxImage::SetDefaultLoadFlags( wxImage::GetDefaultLoadFlags()
& ~wxImage::Load_Verbose );
if( bitmap->ReadImageFile( buf ) )
{
VECTOR2D kcenter = kstart + ksize / 2;
double scaleFactor = ScaleSize( size.x ) / bitmap->GetSize().x;
bitmap->SetImageScale( scaleFactor );
bitmap->SetPosition( kcenter );
for( double i = angle; i > 0; i -= 90 )
bitmap->Rotate( kstart );
if( flipped )
bitmap->MirrorHorizontally( kstart.x );
createdItems.push_back( std::move( bitmap ) );
}
}
}
if( type == wxS( "WIRE" ) )
{
wxString wireId = line.at( 1 );
parentedLines[wireId].push_back( line );
}
else if( type == wxS( "COMPONENT" ) )
{
wxString compId = line.at( 1 );
parentedLines[compId].push_back( line );
}
else if( type == wxS( "ATTR" ) )
{
wxString compId = line.at( 2 );
parentedLines[compId].push_back( line );
}
}
for( auto& [parentId, lines] : parentedLines )
{
std::optional<EASYEDAPRO::SCH_COMPONENT> component;
std::optional<EASYEDAPRO::SCH_WIRE> wire;
std::map<wxString, EASYEDAPRO::SCH_ATTR> attributes;
for( const nlohmann::json& line : lines )
{
if( line.at( 0 ) == "COMPONENT" )
{
component = line;
}
else if( line.at( 0 ) == "WIRE" )
{
wire = line;
}
else if( line.at( 0 ) == "ATTR" )
{
EASYEDAPRO::SCH_ATTR attr = line;
attributes.emplace( attr.key, attr );
}
}
if( component )
{
auto deviceAttr = get_opt( attributes, "Device" );
auto symbolAttr = get_opt( attributes, "Symbol" );
if( !deviceAttr )
continue;
std::map<wxString, wxString> compAttrs =
aProject.at( "devices" ).at( deviceAttr->value ).at( "attributes" );
wxString symbolId;
if( symbolAttr && !symbolAttr->value.IsEmpty() )
symbolId = symbolAttr->value;
else
symbolId = compAttrs.at( "Symbol" );
auto it = aSymbolMap.find( symbolId );
if( it == aSymbolMap.end() )
{
wxLogError( "Symbol of '%s' with uuid '%s' not found.", component->name, symbolId );
continue;
}
EASYEDAPRO::SYM_INFO& esymInfo = it->second;
LIB_SYMBOL newLibSymbol = *esymInfo.libSymbol.get();
wxString unitName = component->name;
LIB_ID libId =
EASYEDAPRO::ToKiCadLibID( aLibName, newLibSymbol.GetLibId().GetLibItemName() );
std::unique_ptr<SCH_SYMBOL> schSym =
std::make_unique<SCH_SYMBOL>( newLibSymbol, libId, &aSchematic->CurrentSheet(),
esymInfo.partUnits[unitName] );
schSym->SetFootprintFieldText( newLibSymbol.GetFootprintField().GetText() );
for( double i = component->rotation; i > 0; i -= 90 )
schSym->Rotate( VECTOR2I() );
if( component->mirror )
schSym->MirrorHorizontally( 0 );
schSym->SetPosition( ScalePos( component->position ) );
if( esymInfo.head.symbolType == EASYEDAPRO::SYMBOL_TYPE::POWER_PORT )
{
if( auto globalNetAttr = get_opt( attributes, "Global Net Name" ) )
{
ApplyAttrToField( fontStyles, schSym->GetField( VALUE_FIELD ), *globalNetAttr,
false, true, compAttrs, schSym.get() );
for( LIB_PIN* pin : schSym->GetAllLibPins() )
pin->SetName( globalNetAttr->value );
}
else
{
schSym->GetField( VALUE_FIELD )
->SetText( newLibSymbol.GetValueField().GetText() );
}
schSym->SetRef( &aSchematic->CurrentSheet(), wxS( "#PWR?" ) );
schSym->GetField( REFERENCE_FIELD )->SetVisible( false );
}
else if( esymInfo.head.symbolType == EASYEDAPRO::SYMBOL_TYPE::NETPORT )
{
auto nameAttr = get_opt( attributes, "Name" );
if( nameAttr )
{
std::unique_ptr<SCH_GLOBALLABEL> label = std::make_unique<SCH_GLOBALLABEL>(
ScalePos( component->position ), nameAttr->value );
SPIN_STYLE spin = SPIN_STYLE::LEFT;
if( esymInfo.symbolAttr )
{
wxString symStr = esymInfo.symbolAttr->value;
if( symStr == wxS( "Netport-IN" ) )
{
spin = SPIN_STYLE::LEFT;
label->SetShape( LABEL_FLAG_SHAPE::L_INPUT );
}
if( symStr == wxS( "Netport-OUT" ) )
{
spin = SPIN_STYLE::RIGHT;
label->SetShape( LABEL_FLAG_SHAPE::L_OUTPUT );
}
if( symStr == wxS( "Netport-BI" ) )
{
spin = SPIN_STYLE::RIGHT;
label->SetShape( LABEL_FLAG_SHAPE::L_BIDI );
}
}
for( double i = component->rotation; i > 0; i -= 90 )
spin = spin.RotateCCW();
label->SetSpinStyle( spin );
nlohmann::json style = fontStyles[nameAttr->fontStyle];
if( !style.is_null() && style.at( 5 ).is_number() )
{
double size = style.at( 5 ).get<double>() * 0.5;
label->SetTextSize( VECTOR2I( ScaleSize( size ), ScaleSize( size ) ) );
}
createdItems.push_back( std::move( label ) );
}
continue;
}
else
{
for( const wxString& attrKey : c_attributesWhitelist )
{
if( auto valOpt = get_opt( compAttrs, attrKey ) )
{
if( valOpt->empty() )
continue;
SCH_FIELD* text = schSym->FindField( attrKey, true );
if( !text )
{
text = schSym->AddField(
SCH_FIELD( schSym.get(), schSym->GetFieldCount(), attrKey ) );
}
wxString value = *valOpt;
value.Replace( wxS( "\u2103" ), wxS( "\u00B0C" ), true ); // ℃ -> °C
text->SetText( value );
text->SetVisible( false );
}
}
auto nameAttr = get_opt( attributes, "Name" );
auto valueAttr = get_opt( attributes, "Value" );
std::optional<EASYEDAPRO::SCH_ATTR> targetValueAttr;
if( valueAttr && !valueAttr->value.empty() && valueAttr->valVisible )
targetValueAttr = valueAttr;
else if( nameAttr && !nameAttr->value.empty() && nameAttr->valVisible )
targetValueAttr = nameAttr;
else if( valueAttr && !valueAttr->value.empty() )
targetValueAttr = valueAttr;
else if( nameAttr && !nameAttr->value.empty() )
targetValueAttr = nameAttr;
if( targetValueAttr )
{
ApplyAttrToField( fontStyles, schSym->GetField( VALUE_FIELD ), *targetValueAttr,
false, true, compAttrs, schSym.get() );
}
if( auto descrAttr = get_opt( attributes, "Description" ) )
{
ApplyAttrToField( fontStyles, schSym->GetField( DESCRIPTION_FIELD ), *descrAttr,
false, true, compAttrs, schSym.get() );
}
if( auto designatorAttr = get_opt( attributes, "Designator" ) )
{
ApplyAttrToField( fontStyles, schSym->GetField( REFERENCE_FIELD ),
*designatorAttr, false, true, compAttrs, schSym.get() );
schSym->SetRef( &aSchematic->CurrentSheet(), designatorAttr->value );
}
for( auto& [attrKey, attr] : attributes )
{
if( attrKey == wxS( "Name" ) || attrKey == wxS( "Value" )
|| attrKey == wxS( "Global Net Name" ) || attrKey == wxS( "Designator" )
|| attrKey == wxS( "Description" ) || attrKey == wxS( "Device" )
|| attrKey == wxS( "Footprint" ) || attrKey == wxS( "Symbol" )
|| attrKey == wxS( "Unique ID" ) )
{
continue;
}
if( attr.value.IsEmpty() )
continue;
SCH_FIELD* text = schSym->FindField( attrKey, true );
if( !text )
{
text = schSym->AddField(
SCH_FIELD( schSym.get(), schSym->GetFieldCount(), attrKey ) );
}
text->SetPosition( schSym->GetPosition() );
ApplyAttrToField( fontStyles, text, attr, false, true, compAttrs,
schSym.get() );
}
}
for( const EASYEDAPRO::PIN_INFO& pinInfo : esymInfo.pins )
{
wxString pinKey = parentId + pinInfo.pin.id;
auto pinLines = get_opt( parentedLines, pinKey );
if( !pinLines )
continue;
for( const nlohmann::json& pinLine : *pinLines )
{
if( pinLine.at( 0 ) != "ATTR" )
continue;
EASYEDAPRO::SCH_ATTR attr = pinLine;
if( attr.key != wxS( "NO_CONNECT" ) )
continue;
if( SCH_PIN* schPin = schSym->GetPin( pinInfo.number ) )
{
VECTOR2I pos = schSym->GetPinPhysicalPosition( schPin->GetLibPin() );
std::unique_ptr<SCH_NO_CONNECT> noConn =
std::make_unique<SCH_NO_CONNECT>( pos );
createdItems.push_back( std::move( noConn ) );
}
}
}
createdItems.push_back( std::move( schSym ) );
}
else // Not component
{
std::vector<SHAPE_LINE_CHAIN> wireLines;
if( wire )
{
for( const std::vector<double>& ptArr : wire->geometry )
{
SHAPE_LINE_CHAIN chain;
for( size_t i = 1; i < ptArr.size(); i += 2 )
chain.Append( ScalePos( VECTOR2D( ptArr[i - 1], ptArr[i] ) ) );
if( chain.PointCount() < 2 )
continue;
wireLines.push_back( chain );
for( int segId = 0; segId < chain.SegmentCount(); segId++ )
{
const SEG& seg = chain.CSegment( segId );
std::unique_ptr<SCH_LINE> schLine =
std::make_unique<SCH_LINE>( seg.A, LAYER_WIRE );
schLine->SetEndPoint( seg.B );
createdItems.push_back( std::move( schLine ) );
}
}
}
auto netAttr = get_opt( attributes, "NET" );
if( netAttr )
{
if( !netAttr->valVisible )
continue;
VECTOR2I kpos = ScalePos( *netAttr->position );
VECTOR2I nearestPos = kpos;
SEG::ecoord min_dist_sq = VECTOR2I::ECOORD_MAX;
for( const SHAPE_LINE_CHAIN& chain : wireLines )
{
VECTOR2I nearestPt = chain.NearestPoint( kpos, false );
SEG::ecoord dist_sq = ( nearestPt - kpos ).SquaredEuclideanNorm();
if( dist_sq < min_dist_sq )
{
min_dist_sq = dist_sq;
nearestPos = nearestPt;
}
}
std::unique_ptr<SCH_LABEL> label = std::make_unique<SCH_LABEL>();
label->SetHorizJustify( GR_TEXT_H_ALIGN_LEFT );
label->SetVertJustify( GR_TEXT_V_ALIGN_BOTTOM );
for( double i = netAttr->rotation; i > 0; i -= 90 )
label->Rotate90( true );
label->SetPosition( nearestPos );
label->SetText( netAttr->value );
ApplyFontStyle( fontStyles, label, netAttr->fontStyle );
createdItems.push_back( std::move( label ) );
}
}
}
// Adjust page to content
BOX2I sheetBBox;
for( std::unique_ptr<SCH_ITEM>& ptr : createdItems )
{
if( ptr->Type() == SCH_SYMBOL_T )
sheetBBox.Merge( static_cast<SCH_SYMBOL*>( ptr.get() )->GetBodyAndPinsBoundingBox() );
else
sheetBBox.Merge( ptr->GetBoundingBox() );
}
SCH_SCREEN* screen = aRootSheet->GetScreen();
PAGE_INFO pageInfo = screen->GetPageSettings();
int alignGrid = schIUScale.MilsToIU( 50 );
VECTOR2D offset( -sheetBBox.GetLeft(), -sheetBBox.GetTop() );
offset.x = KiROUND( offset.x / alignGrid ) * alignGrid;
offset.y = KiROUND( offset.y / alignGrid ) * alignGrid;
pageInfo.SetWidthMils( schIUScale.IUToMils( sheetBBox.GetWidth() ) );
pageInfo.SetHeightMils( schIUScale.IUToMils( sheetBBox.GetHeight() ) );
screen->SetPageSettings( pageInfo );
for( std::unique_ptr<SCH_ITEM>& ptr : createdItems )
{
ptr->Move( offset );
screen->Append( ptr.release() );
}
}