mirror of
https://gitlab.com/kicad/code/kicad.git
synced 2025-09-15 02:33:15 +02:00
473 lines
14 KiB
C++
473 lines
14 KiB
C++
|
/*************************************************/
|
||
|
/* Routines d'edition de symboles de composants */
|
||
|
/*************************************************/
|
||
|
|
||
|
/* fichier symbedit.cpp */
|
||
|
|
||
|
#include "fctsys.h"
|
||
|
#include "gr_basic.h"
|
||
|
|
||
|
#include "common.h"
|
||
|
#include "program.h"
|
||
|
#include "libcmp.h"
|
||
|
#include "general.h"
|
||
|
|
||
|
#include "protos.h"
|
||
|
|
||
|
|
||
|
/* Routines locales */
|
||
|
static bool CompareSymbols(LibEDA_BaseStruct *DEntryRef,
|
||
|
LibEDA_BaseStruct *DEntryCompare);
|
||
|
|
||
|
/* Variables locales */
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
/***************************************************/
|
||
|
void WinEDA_LibeditFrame::LoadOneSymbol(wxDC * DC)
|
||
|
/***************************************************/
|
||
|
/* Cette routine lit un fichier type librairie symbole et
|
||
|
ajoute au symbole courant les elements de trace du nouveau symbole graphique
|
||
|
Si il n'y a pas de symbole courant, le nouveau symbole devient le
|
||
|
symbole courant
|
||
|
*/
|
||
|
{
|
||
|
int NumOfParts;
|
||
|
PriorQue *Entries;
|
||
|
EDA_LibComponentStruct *LibEntry = NULL;
|
||
|
LibEDA_BaseStruct *DrawEntry;
|
||
|
wxString FullFileName, mask;
|
||
|
FILE * ImportFile;
|
||
|
wxString msg;
|
||
|
|
||
|
if(CurrentDrawItem) return;
|
||
|
if( CurrentLibEntry == NULL) return;
|
||
|
|
||
|
DrawPanel->m_IgnoreMouseEvents = TRUE;
|
||
|
|
||
|
mask = wxT("*") + g_SymbolExtBuffer;
|
||
|
FullFileName = EDA_FileSelector( _("Import symbol drawings:"),
|
||
|
g_RealLibDirBuffer, /* Chemin par defaut */
|
||
|
wxEmptyString, /* nom fichier par defaut */
|
||
|
g_SymbolExtBuffer, /* extension par defaut */
|
||
|
mask, /* Masque d'affichage */
|
||
|
this,
|
||
|
0,
|
||
|
TRUE
|
||
|
);
|
||
|
|
||
|
GetScreen()->m_Curseur = wxPoint(0,0);
|
||
|
DrawPanel->MouseToCursorSchema();
|
||
|
DrawPanel->m_IgnoreMouseEvents = FALSE;
|
||
|
|
||
|
if ( FullFileName.IsEmpty() ) return;
|
||
|
|
||
|
|
||
|
/* Chargement de 1 symbole */
|
||
|
ImportFile = wxFopen(FullFileName, wxT("rt"));
|
||
|
if (ImportFile == NULL)
|
||
|
{
|
||
|
msg.Printf( _("Failed to open Symbol File <%s>"), FullFileName.GetData());
|
||
|
DisplayError(this, msg, 20);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
|
||
|
Entries = LoadLibraryAux(this, NULL, ImportFile, &NumOfParts);
|
||
|
fclose(ImportFile);
|
||
|
|
||
|
if( Entries == NULL) return;
|
||
|
|
||
|
if(NumOfParts > 1 )
|
||
|
DisplayError(this, _("Warning: more than 1 part in Symbol File"), 20);
|
||
|
|
||
|
LibEntry = (EDA_LibComponentStruct *)PQFirst(&Entries,FALSE);
|
||
|
|
||
|
if(LibEntry == NULL )
|
||
|
DisplayError(this, _("Symbol File is void"), 20);
|
||
|
|
||
|
else /* Ajout des elements graphiques */
|
||
|
{
|
||
|
DrawEntry = LibEntry->m_Drawings;
|
||
|
while(DrawEntry)
|
||
|
{
|
||
|
if(DrawEntry->m_Unit) DrawEntry->m_Unit = CurrentUnit;
|
||
|
if(DrawEntry->m_Convert) DrawEntry->m_Convert = CurrentConvert;
|
||
|
DrawEntry->m_Flags = IS_NEW;
|
||
|
DrawEntry->m_Selected = IS_SELECTED;
|
||
|
|
||
|
if(DrawEntry->Pnext == NULL)
|
||
|
{ /* Fin de liste trouvee */
|
||
|
DrawEntry->Pnext = CurrentLibEntry->m_Drawings;
|
||
|
CurrentLibEntry->m_Drawings = LibEntry->m_Drawings;
|
||
|
LibEntry->m_Drawings = NULL;
|
||
|
break;
|
||
|
}
|
||
|
DrawEntry = DrawEntry->Next();
|
||
|
}
|
||
|
SuppressDuplicateDrawItem(CurrentLibEntry);
|
||
|
m_CurrentScreen->SetModify();
|
||
|
// Move (and place ) the new draw items:
|
||
|
HandleBlockBegin(DC, -1, GetScreen()->m_Curseur);
|
||
|
HandleBlockEnd(DC);
|
||
|
RedrawActiveWindow( DC,TRUE);
|
||
|
}
|
||
|
|
||
|
PQFreeFunc(Entries, (void(*)(void*))FreeLibraryEntry);
|
||
|
}
|
||
|
|
||
|
|
||
|
/********************************************/
|
||
|
void WinEDA_LibeditFrame::SaveOneSymbol(void)
|
||
|
/********************************************/
|
||
|
/* Routine de sauvegarde du symbole courant edite
|
||
|
Le format est identique aux librairies standards
|
||
|
Les pins invisibles et les elements
|
||
|
non relativ a la forme courante ne sont pas sauves
|
||
|
*/
|
||
|
{
|
||
|
EDA_LibComponentStruct *LibEntry = CurrentLibEntry;
|
||
|
int Unit = CurrentUnit, convert = CurrentConvert;
|
||
|
int SymbUnit, SymbConvert;
|
||
|
LibEDA_BaseStruct *DrawEntry;
|
||
|
wxString FullFileName, mask;
|
||
|
wxString msg;
|
||
|
FILE * ExportFile;
|
||
|
|
||
|
if( LibEntry->m_Drawings == NULL ) return;
|
||
|
|
||
|
/* Creation du fichier symbole */
|
||
|
mask = wxT("*") + g_SymbolExtBuffer;
|
||
|
FullFileName = EDA_FileSelector( _("Export symbol drawings:"),
|
||
|
g_RealLibDirBuffer, /* Chemin par defaut */
|
||
|
wxEmptyString, /* nom fichier par defaut */
|
||
|
g_SymbolExtBuffer, /* extension par defaut */
|
||
|
mask, /* Masque d'affichage */
|
||
|
this,
|
||
|
wxFD_SAVE,
|
||
|
TRUE
|
||
|
);
|
||
|
if ( FullFileName.IsEmpty() ) return;
|
||
|
|
||
|
ExportFile = wxFopen(FullFileName, wxT("wt") );
|
||
|
if ( ExportFile == NULL )
|
||
|
{
|
||
|
msg.Printf(_("Unable to create <%s>"), FullFileName.GetData());
|
||
|
DisplayError(this, msg);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
msg.Printf(_("Save Symbol in [%s]"), FullFileName.GetData());
|
||
|
Affiche_Message(msg);
|
||
|
|
||
|
/* Creation de l'entete de la librairie */
|
||
|
char Line[256];
|
||
|
fprintf(ExportFile,"%s %d.%d %s Date: %s\n",LIBFILE_IDENT,
|
||
|
LIB_VERSION_MAJOR, LIB_VERSION_MINOR,
|
||
|
"SYMBOL",DateAndTime(Line));
|
||
|
|
||
|
/* Creation du commentaire donnant le nom du composant */
|
||
|
fprintf(ExportFile,"# SYMBOL %s\n#\n",
|
||
|
(const char*) LibEntry->m_Name.m_Text.GetData());
|
||
|
|
||
|
/* Generation des lignes utiles */
|
||
|
fprintf(ExportFile,"DEF %s", (const char*)LibEntry->m_Name.m_Text.GetData());
|
||
|
if( ! LibEntry->m_Prefix.m_Text.IsEmpty())
|
||
|
fprintf(ExportFile," %s", (const char*)LibEntry->m_Prefix.m_Text.GetData());
|
||
|
else fprintf(ExportFile," ~");
|
||
|
|
||
|
fprintf(ExportFile," %d %d %c %c %d %d %c\n",
|
||
|
0, /* unused */
|
||
|
LibEntry->m_TextInside,
|
||
|
LibEntry->m_DrawPinNum ? 'Y' : 'N',
|
||
|
LibEntry->m_DrawPinName ? 'Y' : 'N',
|
||
|
1, 0 /* unused */, 'N');
|
||
|
|
||
|
/* Position / orientation / visibilite des champs */
|
||
|
LibEntry->m_Prefix.WriteDescr( ExportFile );
|
||
|
LibEntry->m_Name.WriteDescr( ExportFile );
|
||
|
|
||
|
DrawEntry = LibEntry->m_Drawings;
|
||
|
if(DrawEntry)
|
||
|
{
|
||
|
fprintf(ExportFile,"DRAW\n");
|
||
|
for( ; DrawEntry != NULL ; DrawEntry = DrawEntry->Next())
|
||
|
{
|
||
|
/* Elimination des elements non relatifs a l'unite */
|
||
|
if( Unit && DrawEntry->m_Unit && (DrawEntry->m_Unit != Unit) )
|
||
|
continue;
|
||
|
if( convert && DrawEntry->m_Convert && (DrawEntry->m_Convert != convert) )
|
||
|
continue;
|
||
|
|
||
|
/* .Unit , . Convert est laisse a 0 ou mis a 1 */
|
||
|
SymbUnit = DrawEntry->m_Unit; if ( SymbUnit > 1) SymbUnit = 1;
|
||
|
SymbConvert = DrawEntry->m_Convert;
|
||
|
if ( SymbConvert > 1) SymbConvert = 1;
|
||
|
switch( DrawEntry->m_StructType)
|
||
|
{
|
||
|
case COMPONENT_ARC_DRAW_TYPE:
|
||
|
#define DRAWSTRUCT ((LibDrawArc *) DrawEntry)
|
||
|
DRAWSTRUCT->WriteDescr( ExportFile );
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_CIRCLE_DRAW_TYPE:
|
||
|
#undef DRAWSTRUCT
|
||
|
#define DRAWSTRUCT ((LibDrawCircle *) DrawEntry)
|
||
|
DRAWSTRUCT->WriteDescr( ExportFile );
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_GRAPHIC_TEXT_DRAW_TYPE:
|
||
|
#undef DRAWSTRUCT
|
||
|
#define DRAWSTRUCT ((LibDrawText *) DrawEntry)
|
||
|
DRAWSTRUCT->WriteDescr( ExportFile );
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_RECT_DRAW_TYPE:
|
||
|
#undef DRAWSTRUCT
|
||
|
#define DRAWSTRUCT ((LibDrawSquare *) DrawEntry)
|
||
|
DRAWSTRUCT->WriteDescr( ExportFile );
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_PIN_DRAW_TYPE:
|
||
|
#undef DRAWSTRUCT
|
||
|
#define DRAWSTRUCT ((LibDrawPin *) DrawEntry)
|
||
|
if(DRAWSTRUCT->m_Attributs & PINNOTDRAW) break;
|
||
|
DRAWSTRUCT->WriteDescr( ExportFile );
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_POLYLINE_DRAW_TYPE:
|
||
|
#undef DRAWSTRUCT
|
||
|
#define DRAWSTRUCT ((LibDrawPolyline *) DrawEntry)
|
||
|
DRAWSTRUCT->WriteDescr( ExportFile );
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
fprintf(ExportFile,"ENDDRAW\n");
|
||
|
}
|
||
|
fprintf(ExportFile,"ENDDEF\n");
|
||
|
fclose(ExportFile);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
/*****************************************************************/
|
||
|
void SuppressDuplicateDrawItem(EDA_LibComponentStruct * LibEntry)
|
||
|
/*****************************************************************/
|
||
|
/* Routine de suppression des elements de trace dupliques, situation
|
||
|
frequente lorsque l'on charge des symboles predessines plusieurs fois
|
||
|
pour definir un composant
|
||
|
*/
|
||
|
{
|
||
|
LibEDA_BaseStruct *DEntryRef, *DEntryCompare;
|
||
|
bool deleted;
|
||
|
wxDC * DC = NULL;
|
||
|
|
||
|
DEntryRef = LibEntry->m_Drawings;
|
||
|
while( DEntryRef)
|
||
|
{
|
||
|
if( DEntryRef->Pnext == NULL ) return;
|
||
|
DEntryCompare = DEntryRef->Next();
|
||
|
if( DEntryCompare == NULL ) return;
|
||
|
deleted = 0;
|
||
|
while( DEntryCompare )
|
||
|
{
|
||
|
if( CompareSymbols(DEntryRef,DEntryCompare) == TRUE)
|
||
|
{
|
||
|
DeleteOneLibraryDrawStruct(NULL, DC, LibEntry,DEntryRef, 1);
|
||
|
deleted = TRUE;
|
||
|
break;
|
||
|
}
|
||
|
DEntryCompare = DEntryCompare->Next();
|
||
|
}
|
||
|
if ( ! deleted ) DEntryRef = DEntryRef->Next();
|
||
|
else DEntryRef = LibEntry->m_Drawings;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
/********************************************************************/
|
||
|
static bool CompareSymbols(LibEDA_BaseStruct *DEntryRef,
|
||
|
LibEDA_BaseStruct *DEntryCompare)
|
||
|
/********************************************************************/
|
||
|
/* Routine de comparaison de 2 DrawEntryStruct.
|
||
|
retourne FALSE si differentes
|
||
|
TRUE si egales
|
||
|
*/
|
||
|
{
|
||
|
int ii;
|
||
|
int * ptref, *ptcomp;
|
||
|
|
||
|
/* Comparaison des proprietes generales */
|
||
|
if( DEntryRef->m_StructType != DEntryCompare->m_StructType) return(FALSE);
|
||
|
if( DEntryRef->m_Unit != DEntryCompare->m_Unit) return(FALSE);
|
||
|
if( DEntryRef->m_Convert != DEntryCompare->m_Convert) return(FALSE);
|
||
|
|
||
|
switch( DEntryRef->m_StructType)
|
||
|
{
|
||
|
case COMPONENT_ARC_DRAW_TYPE:
|
||
|
#undef REFSTRUCT
|
||
|
#undef CMPSTRUCT
|
||
|
#define REFSTRUCT ((LibDrawArc *) DEntryRef)
|
||
|
#define CMPSTRUCT ((LibDrawArc *) DEntryCompare)
|
||
|
if( REFSTRUCT->m_Pos.x != CMPSTRUCT->m_Pos.x) return(FALSE);
|
||
|
if( REFSTRUCT->m_Pos.y != CMPSTRUCT->m_Pos.y) return(FALSE);
|
||
|
if( REFSTRUCT->t1 != CMPSTRUCT->t1) return(FALSE);
|
||
|
if( REFSTRUCT->t2 != CMPSTRUCT->t2) return(FALSE);
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_CIRCLE_DRAW_TYPE:
|
||
|
#undef REFSTRUCT
|
||
|
#undef CMPSTRUCT
|
||
|
#define REFSTRUCT ((LibDrawCircle *) DEntryRef)
|
||
|
#define CMPSTRUCT ((LibDrawCircle *) DEntryCompare)
|
||
|
if( REFSTRUCT->m_Pos.x != CMPSTRUCT->m_Pos.x) return(FALSE);
|
||
|
if( REFSTRUCT->m_Pos.y != CMPSTRUCT->m_Pos.y) return(FALSE);
|
||
|
if( REFSTRUCT->m_Rayon != CMPSTRUCT->m_Rayon) return(FALSE);
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_GRAPHIC_TEXT_DRAW_TYPE:
|
||
|
#undef REFSTRUCT
|
||
|
#undef CMPSTRUCT
|
||
|
#define REFSTRUCT ((LibDrawText *) DEntryRef)
|
||
|
#define CMPSTRUCT ((LibDrawText *) DEntryCompare)
|
||
|
if( REFSTRUCT->m_Pos.x != CMPSTRUCT->m_Pos.x) return(FALSE);
|
||
|
if( REFSTRUCT->m_Pos.y != CMPSTRUCT->m_Pos.y) return(FALSE);
|
||
|
if( REFSTRUCT->m_Size != CMPSTRUCT->m_Size) return(FALSE);
|
||
|
if( REFSTRUCT->m_Text != CMPSTRUCT->m_Text )
|
||
|
return(FALSE);
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_RECT_DRAW_TYPE:
|
||
|
#undef REFSTRUCT
|
||
|
#undef CMPSTRUCT
|
||
|
#define REFSTRUCT ((LibDrawSquare *) DEntryRef)
|
||
|
#define CMPSTRUCT ((LibDrawSquare *) DEntryCompare)
|
||
|
if( REFSTRUCT->m_Start.x != CMPSTRUCT->m_Start.x) return(FALSE);
|
||
|
if( REFSTRUCT->m_Start.y != CMPSTRUCT->m_Start.y) return(FALSE);
|
||
|
if( REFSTRUCT->m_End.x != CMPSTRUCT->m_End.x) return(FALSE);
|
||
|
if( REFSTRUCT->m_End.y != CMPSTRUCT->m_End.y) return(FALSE);
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_PIN_DRAW_TYPE:
|
||
|
#undef REFSTRUCT
|
||
|
#undef CMPSTRUCT
|
||
|
#define REFSTRUCT ((LibDrawPin *) DEntryRef)
|
||
|
#define CMPSTRUCT ((LibDrawPin *) DEntryCompare)
|
||
|
if( REFSTRUCT->m_Pos.x != CMPSTRUCT->m_Pos.x) return(FALSE);
|
||
|
if( REFSTRUCT->m_Pos.y != CMPSTRUCT->m_Pos.y) return(FALSE);
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_POLYLINE_DRAW_TYPE:
|
||
|
#undef REFSTRUCT
|
||
|
#undef CMPSTRUCT
|
||
|
#define REFSTRUCT ((LibDrawPolyline *) DEntryRef)
|
||
|
#define CMPSTRUCT ((LibDrawPolyline *) DEntryCompare)
|
||
|
if( REFSTRUCT->n != CMPSTRUCT->n) return(FALSE);
|
||
|
ptref = REFSTRUCT->PolyList;
|
||
|
ptcomp = CMPSTRUCT->PolyList;
|
||
|
for( ii = 2 * REFSTRUCT->n ; ii > 0; ii-- )
|
||
|
{
|
||
|
if( *ptref != *ptcomp) return(FALSE);
|
||
|
ptref++; ptcomp++;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
return(TRUE);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
/***************************************************************************/
|
||
|
/* Routine de placement du point d'ancrage ( reference des coordonnes pour */
|
||
|
/* le trace) du composant courant */
|
||
|
/* Toutes les coord apparaissant dans les structures sont modifiees */
|
||
|
/* pour repositionner le point repere par le curseur souris au point */
|
||
|
/* d'ancrage ( coord 0,0 ). */
|
||
|
/***************************************************************************/
|
||
|
|
||
|
void WinEDA_LibeditFrame::PlaceAncre(void)
|
||
|
{
|
||
|
int ii, *ptsegm;
|
||
|
int dx, dy; /* Offsets de deplacement */
|
||
|
EDA_LibComponentStruct * LibEntry;
|
||
|
LibEDA_BaseStruct * DrawEntry;
|
||
|
|
||
|
dx = - m_CurrentScreen->m_Curseur.x;
|
||
|
dy = m_CurrentScreen->m_Curseur.y;
|
||
|
|
||
|
LibEntry = CurrentLibEntry;
|
||
|
if( LibEntry == NULL ) return;
|
||
|
|
||
|
m_CurrentScreen->SetModify();
|
||
|
|
||
|
LibEntry->m_Name.m_Pos.x += dx; LibEntry->m_Name.m_Pos.y += dy;
|
||
|
LibEntry->m_Prefix.m_Pos.x += dx; LibEntry->m_Prefix.m_Pos.y += dy;
|
||
|
|
||
|
DrawEntry = LibEntry->m_Drawings;
|
||
|
while(DrawEntry)
|
||
|
{
|
||
|
switch(DrawEntry->m_StructType)
|
||
|
{
|
||
|
case COMPONENT_ARC_DRAW_TYPE:
|
||
|
#undef STRUCT
|
||
|
#define STRUCT ((LibDrawArc *) DrawEntry)
|
||
|
STRUCT->m_Pos.x += dx;
|
||
|
STRUCT->m_Pos.y += dy;
|
||
|
STRUCT->m_Start.x += dx;
|
||
|
STRUCT->m_Start.y += dy;
|
||
|
STRUCT->m_End.x += dx;
|
||
|
STRUCT->m_End.y += dy;
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_CIRCLE_DRAW_TYPE:
|
||
|
#undef STRUCT
|
||
|
#define STRUCT ((LibDrawCircle *) DrawEntry)
|
||
|
STRUCT->m_Pos.x += dx;
|
||
|
STRUCT->m_Pos.y += dy;
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_GRAPHIC_TEXT_DRAW_TYPE:
|
||
|
#undef STRUCT
|
||
|
#define STRUCT ((LibDrawText *) DrawEntry)
|
||
|
STRUCT->m_Pos.x += dx;
|
||
|
STRUCT->m_Pos.y += dy;
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_RECT_DRAW_TYPE:
|
||
|
#undef STRUCT
|
||
|
#define STRUCT ((LibDrawSquare *) DrawEntry)
|
||
|
STRUCT->m_Start.x += dx;
|
||
|
STRUCT->m_Start.y += dy;
|
||
|
STRUCT->m_End.x += dx;
|
||
|
STRUCT->m_End.y += dy;
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_PIN_DRAW_TYPE:
|
||
|
#undef STRUCT
|
||
|
#define STRUCT ((LibDrawPin *) DrawEntry)
|
||
|
STRUCT->m_Pos.x += dx;
|
||
|
STRUCT->m_Pos.y += dy;
|
||
|
break;
|
||
|
|
||
|
case COMPONENT_POLYLINE_DRAW_TYPE:
|
||
|
#undef STRUCT
|
||
|
#define STRUCT ((LibDrawPolyline *) DrawEntry)
|
||
|
ptsegm = STRUCT->PolyList;
|
||
|
for( ii = STRUCT->n ; ii > 0; ii-- )
|
||
|
{
|
||
|
*ptsegm += dx; ptsegm++;
|
||
|
*ptsegm += dy;ptsegm++;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
DrawEntry = DrawEntry->Next();
|
||
|
}
|
||
|
/* Reaffichage du symbole */
|
||
|
m_CurrentScreen->m_Curseur.x = m_CurrentScreen->m_Curseur.y = 0;
|
||
|
Recadre_Trace(TRUE);
|
||
|
m_CurrentScreen->SetRefreshReq();
|
||
|
}
|
||
|
|