fixed bug 2934980 (Eeschema crashes when alias already exists).

This commit is contained in:
charras 2010-02-13 13:16:09 +00:00
parent 4f947e861e
commit 636eace77c
2 changed files with 111 additions and 34 deletions

View File

@ -215,6 +215,15 @@ bool CMP_LIBRARY::AddAlias( LIB_ALIAS* aAlias )
} }
/**
* Add /a aComponent entry to library.
* Note a component can have an alias list,
* so these alias will be added in library.
* Conflicts can happen if aliases are already existing.
* User is asked to choose what alias is removed (existing, or new)
* @param aComponent - Component to add.
* @return Added component if successful.
*/
LIB_COMPONENT* CMP_LIBRARY::AddComponent( LIB_COMPONENT* aComponent ) LIB_COMPONENT* CMP_LIBRARY::AddComponent( LIB_COMPONENT* aComponent )
{ {
wxASSERT( aComponent != NULL ); wxASSERT( aComponent != NULL );
@ -224,41 +233,78 @@ LIB_COMPONENT* CMP_LIBRARY::AddComponent( LIB_COMPONENT* aComponent )
if( newCmp == NULL ) if( newCmp == NULL )
return NULL; return NULL;
entries.push_back( (CMP_LIB_ENTRY*) newCmp ); // Conflict detection: See if already existing aliases exist,
SetModifyFlags( ); // and if yes, ask user for continue or abort
wxString msg;
/* Cache libraries are component only libraries. Do not create alias int conflict_count = 0;
* entries. */ for( size_t i = 0; i < newCmp->m_AliasList.GetCount(); i++ )
if( !isCache )
{ {
for( size_t i = 0; i < newCmp->m_AliasList.GetCount(); i++ ) LIB_ALIAS* alias = FindAlias( newCmp->m_AliasList[ i ] );
{
LIB_ALIAS* alias = FindAlias( newCmp->m_AliasList[ i ] );
if( alias == NULL ) if( alias->GetComponent()->GetName().CmpNoCase( newCmp->GetName() ) != 0 )
{ {
alias = new LIB_ALIAS( newCmp->m_AliasList[ i ], newCmp ); wxString msg1;
entries.push_back( alias ); msg1.Printf( _("alias <%s> already exists and has root name<%s>"),
} GetChars( alias->GetName() ),
else if( alias->GetComponent()->GetName().CmpNoCase( newCmp->GetName() ) != 0 ) GetChars( alias->GetComponent()->GetName() ) );
{ msg << msg1 << wxT("\n");
wxLogError( _( "Conflict in library <%s>: alias <%s> already has root name \ conflict_count++;
<%s> and will not be assigned to root name <%s>." ), }
GetChars( fileName.GetName() ),
GetChars( alias->GetComponent()->GetName() ), if( conflict_count > 20 )
GetChars( newCmp->GetName() ) ); break;
} }
if( conflict_count ) // Conflict: ask user what he wants remove all aliases or abort:
{
wxString title;
wxString msg1;
title.Printf( _( "Conflict in library <%s>"), GetChars( fileName.GetName()));
msg1.Printf( _("and appears in alias list of current component <%s>." ),
GetChars( newCmp->GetName() ) );
msg << wxT("\n\n") << msg1;
msg << wxT("\n\n") << _("All old aliases will be removed. Continue ?");
int diag = wxMessageBox(msg, title, wxYES | wxICON_QUESTION);
if( diag != wxYES )
return NULL;
}
for( size_t i = 0; i < newCmp->m_AliasList.GetCount(); i++ )
{
LIB_ALIAS* alias = FindAlias( newCmp->m_AliasList[ i ] );
if( alias == NULL )
{
alias = new LIB_ALIAS( newCmp->m_AliasList[ i ], newCmp );
entries.push_back( alias );
}
else if( alias->GetComponent()->GetName().CmpNoCase( newCmp->GetName() ) != 0 )
{
// Remove alias from library and alias list of its root component
RemoveEntry( alias );
alias = new LIB_ALIAS( newCmp->m_AliasList[ i ], newCmp );
entries.push_back( alias );
} }
} }
entries.push_back( (CMP_LIB_ENTRY*) newCmp );
SetModifyFlags( );
entries.sort(); entries.sort();
entries.unique( DuplicateEntryName ); entries.unique( DuplicateEntryName );
return newCmp; return newCmp;
} }
/** function RemoveEntryName
void CMP_LIBRARY::RemoveEntry( const wxString& aName ) * Remove an /a aName entry from the library list names.
* Warning: this is a partiel remove, because if aname is an alias
* it is not removed from its root component.
* this is for internal use only
* Use RemoveEntry( CMP_LIB_ENTRY* aEntry ) to remove safely an entry in library.
* @param aName - Entry name to remove from library.
*/
void CMP_LIBRARY::RemoveEntryName( const wxString& aName )
{ {
LIB_ENTRY_LIST::iterator i; LIB_ENTRY_LIST::iterator i;
@ -273,7 +319,19 @@ void CMP_LIBRARY::RemoveEntry( const wxString& aName )
} }
void CMP_LIBRARY::RemoveEntry( CMP_LIB_ENTRY* aEntry ) /**
* Remove safely an /a aEntry from the library.
*
* If the entry is an alias, the alias is removed from the library and from
* the alias list of the root component. If the entry is a root component
* with no aliases, it is removed from the library. If the entry is a root
* component with aliases, the root component is renamed to the name of
* the first alias and the root name for all remaining aliases are updated
* to reflect the new root name.
*
* @param aEntry - Entry to remove from library.
*/
void CMP_LIBRARY::RemoveEntry( CMP_LIB_ENTRY* aEntry )
{ {
wxASSERT( aEntry != NULL ); wxASSERT( aEntry != NULL );
@ -308,7 +366,7 @@ library <%s>" ),
root->m_AliasList.RemoveAt( index ); root->m_AliasList.RemoveAt( index );
} }
RemoveEntry( alias->GetName() ); RemoveEntryName( alias->GetName() );
return; return;
} }
@ -318,7 +376,7 @@ library <%s>" ),
/* Entry is a component with no aliases so removal is simple. */ /* Entry is a component with no aliases so removal is simple. */
if( root->m_AliasList.GetCount() == 0 ) if( root->m_AliasList.GetCount() == 0 )
{ {
RemoveEntry( root->GetName() ); RemoveEntryName( root->GetName() );
return; return;
} }
@ -346,13 +404,20 @@ library <%s>" ),
root->SetKeyWords( alias->GetKeyWords() ); root->SetKeyWords( alias->GetKeyWords() );
/* Remove the first alias from library. */ /* Remove the first alias from library. */
RemoveEntry( aliasName ); RemoveEntryName( aliasName );
/* Change the root name. */ /* Change the root name. */
root->SetName( aliasName ); root->SetName( aliasName );
} }
/**
* Replace an existing component entry in the library.
*
* @param aOldComponent - The component to replace.
* @param aNewComponent - The new component.
* the new component and the old component are expected having the same name.
*/
LIB_COMPONENT* CMP_LIBRARY::ReplaceComponent( LIB_COMPONENT* aOldComponent, LIB_COMPONENT* CMP_LIBRARY::ReplaceComponent( LIB_COMPONENT* aOldComponent,
LIB_COMPONENT* aNewComponent ) LIB_COMPONENT* aNewComponent )
{ {
@ -374,7 +439,7 @@ LIB_COMPONENT* CMP_LIBRARY::ReplaceComponent( LIB_COMPONENT* aOldComponent,
GetChars( aOldComponent->GetName() ), GetChars( aOldComponent->GetName() ),
GetChars( fileName.GetName() ) ); GetChars( fileName.GetName() ) );
*/ */
RemoveEntry( aOldComponent->m_AliasList[ i ] ); RemoveEntryName( aOldComponent->m_AliasList[ i ] );
} }
/* Now, add current aliases. */ /* Now, add current aliases. */
@ -389,7 +454,7 @@ LIB_COMPONENT* CMP_LIBRARY::ReplaceComponent( LIB_COMPONENT* aOldComponent,
entries.push_back( alias ); entries.push_back( alias );
} }
RemoveEntry( aOldComponent->GetName() ); RemoveEntryName( aOldComponent->GetName() );
entries.push_back( (CMP_LIB_ENTRY*) newCmp ); entries.push_back( (CMP_LIB_ENTRY*) newCmp );
entries.sort(); entries.sort();

View File

@ -142,7 +142,15 @@ private:
bool LoadHeader( FILE* aFile, int* aLineNum ); bool LoadHeader( FILE* aFile, int* aLineNum );
void LoadAliases( LIB_COMPONENT* aComponent ); void LoadAliases( LIB_COMPONENT* aComponent );
void RemoveEntry( const wxString& aName ); /** function RemoveEntryName
* Remove an /a aName entry from the library list names.
* Warning: this is a partiel remove, because if aname is an alias
* it is not removed from its root component.
* this is for internal use only
* Use RemoveEntry( CMP_LIB_ENTRY* aEntry ) to remove safely an entry.
* @param aName - Entry name to remove from library.
*/
void RemoveEntryName( const wxString& aName );
public: public:
/** /**
@ -271,14 +279,17 @@ public:
/** /**
* Add /a aComponent entry to library. * Add /a aComponent entry to library.
* * Note a component can have an alias list,
* so these alias will be added in library.
* Conflicts can happen if aliases are already existing.
* User is asked to choose what alias is removed (existing, or new)
* @param aComponent - Component to add. * @param aComponent - Component to add.
* @return Added component if successful. * @return Added component if successful.
*/ */
LIB_COMPONENT* AddComponent( LIB_COMPONENT* aComponent ); LIB_COMPONENT* AddComponent( LIB_COMPONENT* aComponent );
/** /**
* Remove an /a aEntry from the library. * Remove safely an /a aEntry from the library.
* *
* If the entry is an alias, the alias is removed from the library and from * If the entry is an alias, the alias is removed from the library and from
* the alias list of the root component. If the entry is a root component * the alias list of the root component. If the entry is a root component
@ -293,7 +304,8 @@ public:
/** /**
* Replace an existing component entry in the library. * Replace an existing component entry in the library.
* * Note a component can have an alias list,
* so these alias will be added in library (and previously existing alias removed)
* @param aOldComponent - The component to replace. * @param aOldComponent - The component to replace.
* @param aNewComponent - The new component. * @param aNewComponent - The new component.
*/ */