diff options
Diffstat (limited to 'new/sch_part.cpp')
-rw-r--r-- | new/sch_part.cpp | 675 |
1 files changed, 675 insertions, 0 deletions
diff --git a/new/sch_part.cpp b/new/sch_part.cpp new file mode 100644 index 0000000..730ded9 --- /dev/null +++ b/new/sch_part.cpp @@ -0,0 +1,675 @@ +/* + * This program source code file is part of KiCad, a free EDA CAD application. + * + * Copyright (C) 2011 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com> + * Copyright (C) 2010 KiCad Developers, see change_log.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 <wx/wx.h> // _() + +#include <sch_part.h> +#include <sch_sweet_parser.h> +#include <sch_lpid.h> +#include <sch_lib_table.h> +#include <macros.h> + + + +/** + * Function formatAt + * outputs a formatted "(at X Y [ANGLE])" s-expression + */ + static void formatAt( OUTPUTFORMATTER* out, const SCH::POINT& aPos, ANGLE aAngle, int indent=0 ) + throw( IO_ERROR ) +{ + // if( aPos.x || aPos.y || aAngle ) + { + out->Print( indent, aAngle!=0.0 ? "(at %.6g %.6g %.6g)" : "(at %.6g %.6g)", + InternalToLogical( aPos.x ), InternalToLogical( aPos.y ), + double( aAngle ) ); + } +} + +static void formatStroke( OUTPUTFORMATTER* out, STROKE aStroke, int indent=0 ) + throw( IO_ERROR ) +{ + if( aStroke == STROKE_DEFAULT ) + out->Print( indent, "(stroke %.6g)", InternalToWidth( aStroke ) ); +} + + +using namespace SCH; + + +PART::PART( LIB* aOwner, const STRING& aPartNameAndRev ) : + owner( aOwner ), + contains( 0 ), + partNameAndRev( aPartNameAndRev ), + extends( 0 ), + base( 0 ) +{ + // Our goal is to have class LIB only instantiate what is needed, so print here + // what it is doing. It is the only class where PART can be instantiated. + D(printf("PART::PART(%s)\n", aPartNameAndRev.c_str() );) + + for( int i=REFERENCE; i<END; ++i ) + mandatory[i] = 0; +} + + +void PART::clear() +{ + if( extends ) + { + delete extends; + extends = 0; + } + + // clear the mandatory fields + for( int ndx = REFERENCE; ndx < END; ++ndx ) + { + delete mandatory[ndx]; + mandatory[ndx] = 0; + } + + // delete properties I own, since their container will not destroy them: + for( PROPERTIES::iterator it = properties.begin(); it != properties.end(); ++it ) + delete *it; + properties.clear(); + + // delete graphics I own, since their container will not destroy them: + for( GRAPHICS::iterator it = graphics.begin(); it != graphics.end(); ++it ) + delete *it; + graphics.clear(); + + // delete PINs I own, since their container will not destroy them. + for( PINS::iterator it = pins.begin(); it != pins.end(); ++it ) + delete *it; + pins.clear(); + + alternates.clear(); + + keywords.clear(); + + pin_merges.clear(); + + contains = 0; +} + + +PROPERTY* PART::FieldLookup( PROP_ID aPropertyId ) +{ + wxASSERT( unsigned(aPropertyId) < unsigned(END) ); + + PROPERTY* p = mandatory[aPropertyId]; + + if( !p ) + { + switch( aPropertyId ) + { + case REFERENCE: + p = new PROPERTY( this, wxT( "reference" ) ); + p->text = wxT( "U?" ); + break; + + case VALUE: + p = new PROPERTY( this, wxT( "value" ) ); + break; + + case FOOTPRINT: + p = new PROPERTY( this, wxT( "footprint" ) ); + break; + + case DATASHEET: + p = new PROPERTY( this, wxT( "datasheet" ) ); + break; + + case MODEL: + p = new PROPERTY( this, wxT( "model" ) ); + break; + + default: + ; + } + + mandatory[aPropertyId] = p; + } + + return p; +} + + +PROPERTY& PROPERTY::operator = ( const PROPERTY& r ) +{ + *(BASE_GRAPHIC*) this = (BASE_GRAPHIC&) r; + + name = r.name; + text = r.text; + + delete effects; + + if( r.effects ) + effects = new TEXT_EFFECTS( *r.effects ); + else + effects = 0; + + return *this; +} + + +PINS::iterator PART::pinFindByPad( const wxString& aPad ) +{ + PINS::iterator it; + + for( it = pins.begin(); it != pins.end(); ++it ) + { + if( (*it)->pad.text == aPad ) + break; + } + + return it; +} + + +void PART::PinsFindBySignal( PIN_LIST* aResults, const wxString& aSignal ) +{ + for( PINS::const_iterator it = pins.begin(); it != pins.end(); ++it ) + { + if( (*it)->signal.text == aSignal ) + { + aResults->push_back( *it ); + } + } +} + + +bool PART::PinDelete( const wxString& aPad ) +{ + PINS::iterator it = pinFindByPad( aPad ); + if( it != pins.end() ) + { + delete *it; + pins.erase( it ); + return true; + } + + // there is only one reason this can fail: not found: + return false; +} + + +PART::~PART() +{ + clear(); +} + + +void PART::setExtends( LPID* aLPID ) +{ + delete extends; + extends = aLPID; +} + + +void PART::inherit( const PART& r ) +{ + // Inherit can be called at any time, even from an interactive text + // editor, so cannot assume 'this' object is new. Clear it. + clear(); + + // copy anything inherited, such as drawables, properties, pins, etc. here + contains = r.contains; + + base = &r; + + anchor = r.anchor; + + for( int i=REFERENCE; i<END; ++i ) + { + if( r.mandatory[i] ) + mandatory[i] = (PROPERTY*) r.mandatory[i]->Clone( this ); + } + + for( PROPERTIES::const_iterator it = r.properties.begin(); it != r.properties.end(); ++it ) + properties.push_back( (PROPERTY*) (*it)->Clone( this ) ); + + for( GRAPHICS::const_iterator it = r.graphics.begin(); it != r.graphics.end(); ++it ) + graphics.push_back( (*it)->Clone( this ) ); + + for( PINS::const_iterator it = r.pins.begin(); it != r.pins.end(); ++it ) + pins.push_back( (PIN*) (*it)->Clone( this ) ); + + /* not sure about this concept yet: + for( PART_REFS::const_iterator it = r.alternates.begin(); it != r.alternates.end(); ++it ) + alternates.push_back( *it ); + */ + + for( KEYWORDS::const_iterator it = r.keywords.begin(); it != r.keywords.end(); ++it ) + keywords.insert( *it ); + + for( MERGE_SETS::const_iterator it = r.pin_merges.begin(); it != r.pin_merges.end(); ++it ) + { + pin_merges[ *it->first ] = * new MERGE_SET( *it->second ); + } +} + + +PART& PART::operator=( const PART& r ) +{ + // maintain in concert with inherit(), which is a partial assignment operator. + inherit( r ); + + owner = r.owner; + partNameAndRev = r.partNameAndRev; + body = r.body; + base = r.base; + + setExtends( r.extends ? new LPID( *r.extends ) : 0 ); + + return *this; +} + + +void PART::Parse( SWEET_PARSER* aParser , LIB_TABLE* aTable ) throw( IO_ERROR, PARSE_ERROR ) +{ + aParser->Parse( this, aTable ); +} + + +bool PART::PropDelete( const wxString& aPropertyName ) +{ + PROPERTIES::iterator it = propertyFind( aPropertyName ); + if( it != properties.end() ) + { + delete *it; + properties.erase( it ); + return true; + } + + return false; +} + + +PROPERTIES::iterator PART::propertyFind( const wxString& aPropertyName ) +{ + PROPERTIES::iterator it; + for( it = properties.begin(); it!=properties.end(); ++it ) + if( (*it)->name == aPropertyName ) + break; + return it; +} + + +void PART::Format( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + out->Print( indent, "(part %s", partNameAndRev.c_str() ); + + if( extends ) + out->Print( 0, " inherits %s", extends->Format().c_str() ); + + out->Print( 0, "\n" ); + + for( int i = REFERENCE; i < END; ++i ) + { + PROPERTY* prop = Field( PROP_ID( i ) ); + if( prop ) + prop->Format( out, indent+1, ctl ); + } + + for( PROPERTIES::const_iterator it = properties.begin(); it != properties.end(); ++it ) + { + (*it)->Format( out, indent+1, ctl ); + } + + if( anchor.x || anchor.y ) + { + out->Print( indent+1, "(anchor (at %.6g %.6g))\n", + InternalToLogical( anchor.x ), + InternalToLogical( anchor.y ) ); + } + + if( keywords.size() ) + { + out->Print( indent+1, "(keywords" ); + for( KEYWORDS::iterator it = keywords.begin(); it != keywords.end(); ++it ) + out->Print( 0, " %s", out->Quotew( *it ).c_str() ); + out->Print( 0, ")\n" ); + } + + for( GRAPHICS::const_iterator it = graphics.begin(); it != graphics.end(); ++it ) + { + (*it)->Format( out, indent+1, ctl ); + } + + for( PINS::const_iterator it = pins.begin(); it != pins.end(); ++it ) + { + (*it)->Format( out, indent+1, ctl ); + } + + if( alternates.size() ) + { + out->Print( indent+1, "(alternates" ); + for( PART_REFS::const_iterator it = alternates.begin(); it!=alternates.end(); ++it ) + out->Print( 0, " %s", out->Quotes( it->Format() ).c_str() ); + out->Print( 0, ")\n" ); + } + + for( MERGE_SETS::const_iterator mit = pin_merges.begin(); mit != pin_merges.end(); ++mit ) + { + out->Print( indent+1, "(pin_merge %s (pads", out->Quotew( mit->first ).c_str() ); + + const MERGE_SET& mset = *mit->second; + for( MERGE_SET::const_iterator pit = mset.begin(); pit != mset.end(); ++pit ) + { + out->Print( 0, " %s", out->Quotew( *pit ).c_str() ); + } + out->Print( 0, "))\n" ); + } + + out->Print( indent, ")\n" ); +} + + +//-----< PART objects >------------------------------------------------------ + + +void PROPERTY::Format( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + wxASSERT( owner ); // all PROPERTYs should have an owner. + + int i; + for( i = PART::REFERENCE; i < PART::END; ++i ) + { + if( owner->Field( PART::PROP_ID(i) ) == this ) + break; + } + + if( i < PART::END ) // is a field not a property + out->Print( indent, "(%s", TO_UTF8( name ) ); + else + out->Print( indent, "(property %s", out->Quotew( name ).c_str() ); + + if( effects ) + { + out->Print( 0, " %s\n", out->Quotew( text ).c_str() ); + effects->Format( out, indent+1, ctl | CTL_OMIT_NL ); + out->Print( 0, ")\n" ); + } + else + { + out->Print( 0, " %s)\n", out->Quotew( text ).c_str() ); + } +} + + +TEXT_EFFECTS* PROPERTY::EffectsLookup() +{ + if( !effects ) + { + effects = new TEXT_EFFECTS(); + } + + return effects; +} + + +void TEXT_EFFECTS::Format( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + if( propName.IsEmpty() ) + out->Print( indent, "(effects " ); + else + out->Print( indent, "(effects %s ", out->Quotew( propName ).c_str() ); + + formatAt( out, pos, angle ); + + font.Format( out, 0, ctl | CTL_OMIT_NL ); + + out->Print( 0, "(visible %s))%s", + isVisible ? "yes" : "no", + ctl & CTL_OMIT_NL ? "" : "\n" ); +} + + +void FONT::Format( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + if( italic || bold || !name.IsEmpty() || size.height != FONTZ_DEFAULT || size.width != FONTZ_DEFAULT ) + { + if( name.IsEmpty() ) + out->Print( indent, "(font " ); + else + out->Print( indent, "(font %s ", out->Quotew( name ).c_str() ); + + out->Print( 0, "(size %.6g %.6g)", + InternalToFontz( size.height ), + InternalToFontz( size.width ) ); + + if( italic ) + out->Print( 0, " italic" ); + + if( bold ) + out->Print( 0, " bold" ); + + out->Print( 0, ")%s", (ctl & CTL_OMIT_NL) ? "" : "\n" ); + } +} + + +void PIN::Format( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + bool hasSignal = !signal.text.IsEmpty(); + bool hasPad = !pad.text.IsEmpty(); + + out->Print( indent, "(pin" ); + + if( connectionType != PIN_CONN_DEFAULT ) + out->Print( 0, " %s", ShowType() ); + + if( shape != PIN_SHAPE_DEFAULT ) + out->Print( 0, " %s", ShowShape() ); + + out->Print( 0, " " ); + + if( pos.x || pos.y || angle ) + formatAt( out, pos, angle ); + + if( length != PIN_LEN_DEFAULT ) + out->Print( 0, "(length %.6g)", InternalToLogical( length ) ); + + if( !isVisible ) + out->Print( 0, "(visible %s)", isVisible ? "yes" : "no" ); + + if( hasSignal ) + signal.Format( out, "signal", 0, CTL_OMIT_NL ); + + if( hasPad ) + pad.Format( out, "pad", 0, CTL_OMIT_NL ); + + out->Print( 0, ")\n" ); +} + + +PIN::~PIN() +{ +} + + +void PINTEXT::Format( OUTPUTFORMATTER* out, const char* aElement, int indent, int ctl ) const + throw( IO_ERROR ) +{ + out->Print( indent, "(%s %s", aElement, out->Quotew( text ).c_str() ); + + font.Format( out, 0, CTL_OMIT_NL ); + + if( !isVisible ) + out->Print( 0, " (visible %s)", isVisible ? "yes" : "no" ); + + out->Print( 0, ")%s", ctl & CTL_OMIT_NL ? "" : "\n" ); +} + + +void POLY_LINE::Format( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + out->Print( indent, "(%s ", pts.size() == 2 ? "line" : "polyline" ); + formatContents( out, indent, ctl ); +} + + +void POLY_LINE::formatContents( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + formatStroke( out, stroke ); + + if( fillType != PR::T_none ) + out->Print( 0, "(fill %s)", ShowFill( fillType ) ); + + out->Print( 0, "\n" ); + + if( pts.size() ) + { + const int maxLength = 75; + int len = 10; + + len += out->Print( indent+1, "(pts " ); + + for( POINTS::const_iterator it = pts.begin(); it != pts.end(); ++it ) + { + if( len > maxLength ) + { + len = 10; + out->Print( 0, "\n" ); + out->Print( indent+2, "(xy %.6g %.6g)", + InternalToLogical( it->x ), InternalToLogical( it->y ) ); + } + else + out->Print( 0, "(xy %.6g %.6g)", + InternalToLogical( it->x ), InternalToLogical( it->y ) ); + } + + out->Print( 0, ")" ); + } + + out->Print( 0, ")\n" ); +} + + +void BEZIER::Format( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + out->Print( indent, "(bezier " ); + formatContents( out, indent, ctl ); // inherited from POLY_LINE +} + + +void RECTANGLE::Format( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + // (rectangle (start X Y) (end X Y) [(stroke WIDTH)] (fill FILL_TYPE)) + + out->Print( indent, "(rectangle (start %.6g %.6g)(end %.6g %.6g)", + InternalToLogical( start.x ), InternalToLogical( start.y ), + InternalToLogical( end.x ), InternalToLogical( end.y ) + ); + + formatStroke( out, stroke ); + + if( fillType != PR::T_none ) + out->Print( 0, "(fill %s)", ShowFill( fillType ) ); + + out->Print( 0, ")\n" ); +} + + +void CIRCLE::Format( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + /* + (circle (center X Y)(radius LENGTH) [(stroke WIDTH)] (fill FILL_TYPE)) + */ + + out->Print( indent, "(circle (center %.6g %.6g)(radius %.6g)", + InternalToLogical( center.x ), InternalToLogical( center.y ), + InternalToLogical( radius) ); + + formatStroke( out, stroke ); + + if( fillType != PR::T_none ) + out->Print( 0, "(fill %s)", ShowFill( fillType ) ); + + out->Print( 0, ")\n" ); +} + + +void ARC::Format( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + /* + (arc (pos X Y)(radius RADIUS)(start X Y)(end X Y) [(stroke WIDTH)] (fill FILL_TYPE)) + */ + + out->Print( indent, "(arc (pos %.6g %.6g)(radius %.6g)(start %.6g %.6g)(end %.6g %.6g)", + InternalToLogical( pos.x ), InternalToLogical( pos.y ), + InternalToLogical( radius), + InternalToLogical( start.x ), InternalToLogical( start.y ), + InternalToLogical( end.x ), InternalToLogical( end.y ) + ); + + formatStroke( out, stroke ); + + if( fillType != PR::T_none ) + out->Print( 0, "(fill %s)", ShowFill( fillType ) ); + + out->Print( 0, ")\n" ); +} + + +void GR_TEXT::Format( OUTPUTFORMATTER* out, int indent, int ctl ) const + throw( IO_ERROR ) +{ + /* + (text "This is the text that gets drawn." + (at X Y [ANGLE])(justify HORIZONTAL_JUSTIFY VERTICAL_JUSTIFY)(visible YES)(fill FILL_TYPE) + (font [FONT] (size HEIGHT WIDTH) [italic] [bold]) + ) + */ + + out->Print( indent, "(text %s\n", out->Quotew( text ).c_str() ); + + formatAt( out, pos, angle, indent+1 ); + + if( hjustify != PR::T_left || vjustify != PR::T_bottom ) + out->Print( 0, "(justify %s %s)", + ShowJustify( hjustify ), ShowJustify( vjustify ) ); + + if( !isVisible ) + out->Print( 0, "(visible %s)", isVisible ? "yes" : "no" ); + + if( fillType != PR::T_filled ) + out->Print( 0, "(fill %s)", ShowFill( fillType ) ); + + font.Format( out, 0, CTL_OMIT_NL ); + out->Print( 0, ")\n" ); +} + |