842 lines
24 KiB
C++
842 lines
24 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2015 Chris Pavlina <pavlina.chris@gmail.com>
|
|
* Copyright (C) 2015-2018 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 <class_drawpanel.h>
|
|
#include <class_library.h>
|
|
#include <confirm.h>
|
|
#include <invoke_sch_dialog.h>
|
|
#include <kicad_device_context.h>
|
|
#include <kiway.h>
|
|
#include <project_rescue.h>
|
|
#include <sch_component.h>
|
|
#include <sch_sheet.h>
|
|
#include <schframe.h>
|
|
#include <symbol_lib_table.h>
|
|
#include <viewlib_frame.h>
|
|
#include <wildcards_and_files_ext.h>
|
|
|
|
#include <cctype>
|
|
#include <map>
|
|
|
|
|
|
typedef std::pair<SCH_COMPONENT*, wxString> COMPONENT_NAME_PAIR;
|
|
|
|
|
|
// Helper sort function, used in get_components, to sort a component list by lib_id
|
|
static bool sort_by_libid( const SCH_COMPONENT* ref, SCH_COMPONENT* cmp )
|
|
{
|
|
return ref->GetLibId() < cmp->GetLibId();
|
|
}
|
|
|
|
|
|
/**
|
|
* Fill a vector with all of the project's symbols, to ease iterating over them.
|
|
*
|
|
* The list is sorted by #LIB_ID, therefore components using the same library
|
|
* symbol are grouped, allowing later faster calculations (one library search by group
|
|
* of symbols)
|
|
*
|
|
* @param aComponents - a vector that will take the symbols
|
|
*/
|
|
static void get_components( std::vector<SCH_COMPONENT*>& aComponents )
|
|
{
|
|
SCH_SCREENS screens;
|
|
|
|
// Get the full list
|
|
for( SCH_SCREEN* screen = screens.GetFirst(); screen; screen = screens.GetNext() )
|
|
{
|
|
for( SCH_ITEM* item = screen->GetDrawItems(); item; item = item->Next() )
|
|
{
|
|
if( item->Type() != SCH_COMPONENT_T )
|
|
continue;
|
|
|
|
SCH_COMPONENT* component = static_cast<SCH_COMPONENT*>( item );
|
|
aComponents.push_back( component );
|
|
}
|
|
}
|
|
|
|
if( aComponents.empty() )
|
|
return;
|
|
|
|
// sort aComponents by lib part. Components will be grouped by same lib part.
|
|
std::sort( aComponents.begin(), aComponents.end(), sort_by_libid );
|
|
}
|
|
|
|
|
|
/**
|
|
* Search the libraries for the first component with a given name.
|
|
*
|
|
* @param aName - name to search for
|
|
* @param aLibs - the loaded PART_LIBS
|
|
* @param aCached - whether we are looking for the cached part
|
|
*/
|
|
static LIB_PART* find_component( wxString aName, PART_LIBS* aLibs, bool aCached )
|
|
{
|
|
LIB_PART *part = NULL;
|
|
|
|
for( PART_LIB& each_lib : *aLibs )
|
|
{
|
|
if( aCached && !each_lib.IsCache() )
|
|
continue;
|
|
|
|
if( !aCached && each_lib.IsCache() )
|
|
continue;
|
|
|
|
part = each_lib.FindPart( aName );
|
|
|
|
if( part )
|
|
break;
|
|
}
|
|
|
|
return part;
|
|
}
|
|
|
|
|
|
static wxFileName GetRescueLibraryFileName()
|
|
{
|
|
wxFileName fn( g_RootSheet->GetScreen()->GetFileName() );
|
|
fn.SetName( fn.GetName() + wxT( "-rescue" ) );
|
|
fn.SetExt( SchematicLibraryFileExtension );
|
|
return fn;
|
|
}
|
|
|
|
|
|
RESCUE_CASE_CANDIDATE::RESCUE_CASE_CANDIDATE( const wxString& aRequestedName,
|
|
const wxString& aNewName,
|
|
LIB_PART* aLibCandidate )
|
|
{
|
|
m_requested_name = aRequestedName;
|
|
m_new_name = aNewName;
|
|
m_lib_candidate = aLibCandidate;
|
|
}
|
|
|
|
|
|
void RESCUE_CASE_CANDIDATE::FindRescues( RESCUER& aRescuer,
|
|
boost::ptr_vector<RESCUE_CANDIDATE>& aCandidates )
|
|
{
|
|
typedef std::map<wxString, RESCUE_CASE_CANDIDATE> candidate_map_t;
|
|
candidate_map_t candidate_map;
|
|
|
|
// Remember the list of components is sorted by part name.
|
|
// So a search in libraries is made only once by group
|
|
LIB_ALIAS* case_sensitive_match = nullptr;
|
|
std::vector<LIB_ALIAS*> case_insensitive_matches;
|
|
|
|
wxString last_part_name;
|
|
|
|
for( SCH_COMPONENT* each_component : *( aRescuer.GetComponents() ) )
|
|
{
|
|
wxString part_name = each_component->GetLibId().GetLibItemName();
|
|
|
|
if( last_part_name != part_name )
|
|
{
|
|
// A new part name is found (a new group starts here).
|
|
// Search the symbol names candidates only once for this group:
|
|
last_part_name = part_name;
|
|
case_insensitive_matches.clear();
|
|
|
|
LIB_ID id( wxEmptyString, part_name );
|
|
|
|
case_sensitive_match = aRescuer.GetPrj()->SchLibs()->FindLibraryAlias( id );
|
|
|
|
if( !case_sensitive_match )
|
|
// the case sensitive match failed. Try a case insensitive match
|
|
aRescuer.GetPrj()->SchLibs()->FindLibraryNearEntries( case_insensitive_matches,
|
|
part_name );
|
|
}
|
|
|
|
if( case_sensitive_match || !( case_insensitive_matches.size() ) )
|
|
continue;
|
|
|
|
RESCUE_CASE_CANDIDATE candidate( part_name, case_insensitive_matches[0]->GetName(),
|
|
case_insensitive_matches[0]->GetPart() );
|
|
|
|
candidate_map[part_name] = candidate;
|
|
}
|
|
|
|
// Now, dump the map into aCandidates
|
|
for( const candidate_map_t::value_type& each_pair : candidate_map )
|
|
{
|
|
aCandidates.push_back( new RESCUE_CASE_CANDIDATE( each_pair.second ) );
|
|
}
|
|
}
|
|
|
|
|
|
wxString RESCUE_CASE_CANDIDATE::GetActionDescription() const
|
|
{
|
|
wxString action;
|
|
action.Printf( _( "Rename to %s" ), m_new_name );
|
|
return action;
|
|
}
|
|
|
|
|
|
bool RESCUE_CASE_CANDIDATE::PerformAction( RESCUER* aRescuer )
|
|
{
|
|
for( SCH_COMPONENT* each_component : *aRescuer->GetComponents() )
|
|
{
|
|
if( each_component->GetLibId().GetLibItemName() != UTF8( m_requested_name ) )
|
|
continue;
|
|
|
|
LIB_ID libId;
|
|
|
|
libId.SetLibItemName( m_new_name, false );
|
|
each_component->SetLibId( libId );
|
|
each_component->ClearFlags();
|
|
aRescuer->LogRescue( each_component, m_requested_name, m_new_name );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
RESCUE_CACHE_CANDIDATE::RESCUE_CACHE_CANDIDATE( const wxString& aRequestedName,
|
|
const wxString& aNewName,
|
|
LIB_PART* aCacheCandidate,
|
|
LIB_PART* aLibCandidate )
|
|
{
|
|
m_requested_name = aRequestedName;
|
|
m_new_name = aNewName;
|
|
m_cache_candidate = aCacheCandidate;
|
|
m_lib_candidate = aLibCandidate;
|
|
}
|
|
|
|
|
|
RESCUE_CACHE_CANDIDATE::RESCUE_CACHE_CANDIDATE()
|
|
{
|
|
m_cache_candidate = NULL;
|
|
m_lib_candidate = NULL;
|
|
}
|
|
|
|
|
|
void RESCUE_CACHE_CANDIDATE::FindRescues( RESCUER& aRescuer,
|
|
boost::ptr_vector<RESCUE_CANDIDATE>& aCandidates )
|
|
{
|
|
typedef std::map<wxString, RESCUE_CACHE_CANDIDATE> candidate_map_t;
|
|
candidate_map_t candidate_map;
|
|
|
|
// Remember the list of components is sorted by part name.
|
|
// So a search in libraries is made only once by group
|
|
LIB_PART* cache_match = nullptr;
|
|
LIB_PART* lib_match = nullptr;
|
|
wxString old_part_name;
|
|
|
|
wxString part_name_suffix = aRescuer.GetPartNameSuffix();
|
|
|
|
for( SCH_COMPONENT* each_component : *( aRescuer.GetComponents() ) )
|
|
{
|
|
wxString part_name = each_component->GetLibId().GetLibItemName();
|
|
|
|
if( old_part_name != part_name )
|
|
{
|
|
// A new part name is found (a new group starts here).
|
|
// Search the symbol names candidates only once for this group:
|
|
old_part_name = part_name;
|
|
cache_match = find_component( part_name, aRescuer.GetPrj()->SchLibs(), true );
|
|
lib_match = find_component( part_name, aRescuer.GetPrj()->SchLibs(), false );
|
|
|
|
if( !cache_match && !lib_match )
|
|
continue;
|
|
|
|
// Test whether there is a conflict or if the symbol can only be found in the cache
|
|
// and the symbol name does not have any illegal characters.
|
|
if( ( ( cache_match && lib_match
|
|
&& !cache_match->PinsConflictWith( *lib_match, true, true, true, true, false ) )
|
|
|| (!cache_match && lib_match ) ) && !LIB_ID::HasIllegalChars( part_name ) )
|
|
continue;
|
|
|
|
// Check if the symbol has already been rescued.
|
|
wxString new_name = LIB_ID::FixIllegalChars( part_name );
|
|
|
|
if( new_name.Find( part_name_suffix ) == wxNOT_FOUND )
|
|
new_name += part_name_suffix;
|
|
|
|
RESCUE_CACHE_CANDIDATE candidate( part_name, new_name, cache_match, lib_match );
|
|
|
|
candidate_map[part_name] = candidate;
|
|
}
|
|
}
|
|
|
|
// Now, dump the map into aCandidates
|
|
for( const candidate_map_t::value_type& each_pair : candidate_map )
|
|
{
|
|
aCandidates.push_back( new RESCUE_CACHE_CANDIDATE( each_pair.second ) );
|
|
}
|
|
}
|
|
|
|
|
|
wxString RESCUE_CACHE_CANDIDATE::GetActionDescription() const
|
|
{
|
|
wxString action;
|
|
action.Printf( _( "Rescue %s as %s" ), m_requested_name, m_new_name );
|
|
return action;
|
|
}
|
|
|
|
|
|
bool RESCUE_CACHE_CANDIDATE::PerformAction( RESCUER* aRescuer )
|
|
{
|
|
LIB_PART* tmp = ( m_cache_candidate ) ? m_cache_candidate : m_lib_candidate;
|
|
|
|
wxCHECK_MSG( tmp, false, "Both cache and library symbols undefined." );
|
|
|
|
LIB_PART new_part( *tmp );
|
|
new_part.SetName( m_new_name );
|
|
new_part.RemoveAllAliases();
|
|
aRescuer->AddPart( &new_part );
|
|
|
|
for( SCH_COMPONENT* each_component : *aRescuer->GetComponents() )
|
|
{
|
|
if( each_component->GetLibId().GetLibItemName() != UTF8( m_requested_name ) )
|
|
continue;
|
|
|
|
LIB_ID libId;
|
|
|
|
libId.SetLibItemName( m_new_name, false );
|
|
each_component->SetLibId( libId );
|
|
each_component->ClearFlags();
|
|
aRescuer->LogRescue( each_component, m_requested_name, m_new_name );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
RESCUE_SYMBOL_LIB_TABLE_CANDIDATE::RESCUE_SYMBOL_LIB_TABLE_CANDIDATE(
|
|
const LIB_ID& aRequestedId,
|
|
const LIB_ID& aNewId,
|
|
LIB_PART* aCacheCandidate,
|
|
LIB_PART* aLibCandidate ) : RESCUE_CANDIDATE()
|
|
{
|
|
m_requested_id = aRequestedId;
|
|
m_requested_name = aRequestedId.GetLibItemName();
|
|
m_new_id = aNewId;
|
|
m_lib_candidate = aLibCandidate;
|
|
m_cache_candidate = aCacheCandidate;
|
|
}
|
|
|
|
|
|
RESCUE_SYMBOL_LIB_TABLE_CANDIDATE::RESCUE_SYMBOL_LIB_TABLE_CANDIDATE()
|
|
{
|
|
m_cache_candidate = NULL;
|
|
m_lib_candidate = NULL;
|
|
}
|
|
|
|
|
|
void RESCUE_SYMBOL_LIB_TABLE_CANDIDATE::FindRescues(
|
|
RESCUER& aRescuer,
|
|
boost::ptr_vector<RESCUE_CANDIDATE>& aCandidates )
|
|
{
|
|
typedef std::map<LIB_ID, RESCUE_SYMBOL_LIB_TABLE_CANDIDATE> candidate_map_t;
|
|
|
|
candidate_map_t candidate_map;
|
|
|
|
// Remember the list of components is sorted by LIB_ID.
|
|
// So a search in libraries is made only once by group
|
|
LIB_PART* cache_match = nullptr;
|
|
LIB_PART* lib_match = nullptr;
|
|
LIB_ID old_part_id;
|
|
|
|
wxString part_name_suffix = aRescuer.GetPartNameSuffix();
|
|
|
|
for( SCH_COMPONENT* each_component : *( aRescuer.GetComponents() ) )
|
|
{
|
|
LIB_ID part_id = each_component->GetLibId();
|
|
|
|
if( old_part_id != part_id )
|
|
{
|
|
// A new part name is found (a new group starts here).
|
|
// Search the symbol names candidates only once for this group:
|
|
old_part_id = part_id;
|
|
cache_match = find_component( part_id.GetLibItemName(), aRescuer.GetPrj()->SchLibs(),
|
|
true );
|
|
|
|
lib_match = aRescuer.GetFrame()->GetLibPart( part_id );
|
|
|
|
if( !cache_match && !lib_match )
|
|
continue;
|
|
|
|
// Test whether there is a conflict or if the symbol can only be found in the cache.
|
|
if( ( ( cache_match && lib_match
|
|
&& !cache_match->PinsConflictWith( *lib_match, true, true, true, true, false ) )
|
|
|| (!cache_match && lib_match ) )
|
|
&& !LIB_ID::HasIllegalChars( part_id.GetLibItemName() ) )
|
|
continue;
|
|
|
|
// May have been rescued already.
|
|
wxString new_name = LIB_ID::FixIllegalChars( part_id.GetLibItemName() );
|
|
|
|
if( new_name.Find( part_name_suffix ) == wxNOT_FOUND )
|
|
new_name += part_name_suffix;
|
|
|
|
LIB_ID new_id( GetRescueLibraryFileName().GetName(), new_name );
|
|
|
|
RESCUE_SYMBOL_LIB_TABLE_CANDIDATE candidate( part_id, new_id, cache_match, lib_match );
|
|
|
|
candidate_map[part_id] = candidate;
|
|
}
|
|
}
|
|
|
|
// Now, dump the map into aCandidates
|
|
for( const candidate_map_t::value_type& each_pair : candidate_map )
|
|
{
|
|
aCandidates.push_back( new RESCUE_SYMBOL_LIB_TABLE_CANDIDATE( each_pair.second ) );
|
|
}
|
|
}
|
|
|
|
|
|
wxString RESCUE_SYMBOL_LIB_TABLE_CANDIDATE::GetActionDescription() const
|
|
{
|
|
wxString action;
|
|
action.Printf( _( "Rescue to %s" ), m_new_id.Format().wx_str() );
|
|
return action;
|
|
}
|
|
|
|
|
|
bool RESCUE_SYMBOL_LIB_TABLE_CANDIDATE::PerformAction( RESCUER* aRescuer )
|
|
{
|
|
LIB_PART* tmp = ( m_cache_candidate ) ? m_cache_candidate : m_lib_candidate;
|
|
|
|
wxCHECK_MSG( tmp, false, "Both cache and library symbols undefined." );
|
|
|
|
LIB_PART new_part( *tmp );
|
|
new_part.SetName( m_new_id.GetLibItemName() );
|
|
new_part.RemoveAllAliases();
|
|
aRescuer->AddPart( &new_part );
|
|
|
|
for( SCH_COMPONENT* each_component : *aRescuer->GetComponents() )
|
|
{
|
|
if( each_component->GetLibId() != m_requested_id )
|
|
continue;
|
|
|
|
each_component->SetLibId( m_new_id );
|
|
each_component->ClearFlags();
|
|
aRescuer->LogRescue( each_component, m_requested_id.Format(), m_new_id.Format() );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
RESCUER::RESCUER( SCH_EDIT_FRAME& aEditFrame, PROJECT& aProject )
|
|
{
|
|
get_components( m_components );
|
|
m_prj = &aProject;
|
|
m_edit_frame = &aEditFrame;
|
|
}
|
|
|
|
|
|
void RESCUER::LogRescue( SCH_COMPONENT *aComponent, const wxString &aOldName,
|
|
const wxString &aNewName )
|
|
{
|
|
RESCUE_LOG logitem;
|
|
logitem.component = aComponent;
|
|
logitem.old_name = aOldName;
|
|
logitem.new_name = aNewName;
|
|
m_rescue_log.push_back( logitem );
|
|
}
|
|
|
|
|
|
bool RESCUER::DoRescues()
|
|
{
|
|
for( RESCUE_CANDIDATE* each_candidate : m_chosen_candidates )
|
|
{
|
|
if( ! each_candidate->PerformAction( this ) )
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void RESCUER::UndoRescues()
|
|
{
|
|
for( RESCUE_LOG& each_logitem : m_rescue_log )
|
|
{
|
|
LIB_ID libId;
|
|
|
|
libId.SetLibItemName( each_logitem.old_name, false );
|
|
each_logitem.component->SetLibId( libId );
|
|
each_logitem.component->ClearFlags();
|
|
}
|
|
}
|
|
|
|
|
|
wxString RESCUER::GetPartNameSuffix()
|
|
{
|
|
wxString suffix = wxT( "-RESCUE-" );
|
|
wxString pname = GetPrj()->GetProjectName();
|
|
|
|
for( size_t i = 0; i < pname.Len(); ++i )
|
|
{
|
|
if( isspace( pname[i].GetValue() ) )
|
|
suffix.Append( '_' );
|
|
else
|
|
suffix.Append( pname[i] );
|
|
}
|
|
|
|
return suffix;
|
|
}
|
|
|
|
|
|
bool SCH_EDIT_FRAME::RescueLegacyProject( bool aRunningOnDemand )
|
|
{
|
|
LEGACY_RESCUER rescuer( *this, Prj() );
|
|
|
|
return rescueProject( rescuer, aRunningOnDemand );
|
|
}
|
|
|
|
|
|
bool SCH_EDIT_FRAME::RescueSymbolLibTableProject( bool aRunningOnDemand )
|
|
{
|
|
SYMBOL_LIB_TABLE_RESCUER rescuer( *this, Prj() );
|
|
|
|
return rescueProject( rescuer, aRunningOnDemand );
|
|
}
|
|
|
|
|
|
bool SCH_EDIT_FRAME::rescueProject( RESCUER& aRescuer, bool aRunningOnDemand )
|
|
{
|
|
aRescuer.FindCandidates();
|
|
|
|
if( ! aRescuer.GetCandidateCount() )
|
|
{
|
|
if( aRunningOnDemand )
|
|
{
|
|
wxMessageDialog dlg( this, _( "This project has nothing to rescue." ),
|
|
_( "Project Rescue Helper" ) );
|
|
dlg.ShowModal();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
aRescuer.RemoveDuplicates();
|
|
|
|
aRescuer.InvokeDialog( !aRunningOnDemand );
|
|
|
|
// If no symbols were rescued, let the user know what's going on. He might
|
|
// have clicked cancel by mistake, and should have some indication of that.
|
|
if( !aRescuer.GetChosenCandidateCount() )
|
|
{
|
|
wxMessageDialog dlg( this, _( "No symbols were rescued." ),
|
|
_( "Project Rescue Helper" ) );
|
|
dlg.ShowModal();
|
|
|
|
// Set the modified flag even on Cancel. Many users seem to instinctively want to Save at
|
|
// this point, due to the reloading of the symbols, so we'll make the save button active.
|
|
OnModify();
|
|
return true;
|
|
}
|
|
|
|
aRescuer.OpenRescueLibrary();
|
|
|
|
if( !aRescuer.DoRescues() )
|
|
{
|
|
aRescuer.UndoRescues();
|
|
return false;
|
|
}
|
|
|
|
aRescuer.WriteRescueLibrary( this );
|
|
|
|
LIB_VIEW_FRAME* viewer = (LIB_VIEW_FRAME*) Kiway().Player( FRAME_SCH_VIEWER, false );
|
|
|
|
if( viewer )
|
|
viewer->ReCreateListLib();
|
|
|
|
// Clean up wire ends
|
|
SchematicCleanUp();
|
|
GetScreen()->ClearUndoORRedoList( GetScreen()->m_UndoList, 1 );
|
|
m_canvas->Refresh( true );
|
|
OnModify();
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void RESCUER::RemoveDuplicates()
|
|
{
|
|
std::vector<wxString> names_seen;
|
|
|
|
for( boost::ptr_vector<RESCUE_CANDIDATE>::iterator it = m_all_candidates.begin();
|
|
it != m_all_candidates.end(); )
|
|
{
|
|
bool seen_already = false;
|
|
|
|
for( wxString& name_seen : names_seen )
|
|
{
|
|
if( name_seen == it->GetRequestedName() )
|
|
{
|
|
seen_already = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if( seen_already )
|
|
{
|
|
it = m_all_candidates.erase( it );
|
|
}
|
|
else
|
|
{
|
|
names_seen.push_back( it->GetRequestedName() );
|
|
++it;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void LEGACY_RESCUER::FindCandidates()
|
|
{
|
|
RESCUE_CASE_CANDIDATE::FindRescues( *this, m_all_candidates );
|
|
RESCUE_CACHE_CANDIDATE::FindRescues( *this, m_all_candidates );
|
|
}
|
|
|
|
|
|
void LEGACY_RESCUER::InvokeDialog( bool aAskShowAgain )
|
|
{
|
|
InvokeDialogRescueEach( m_edit_frame, static_cast< RESCUER& >( *this ), aAskShowAgain );
|
|
}
|
|
|
|
|
|
void LEGACY_RESCUER::OpenRescueLibrary()
|
|
{
|
|
wxFileName fn = GetRescueLibraryFileName();
|
|
|
|
std::unique_ptr<PART_LIB> rescue_lib( new PART_LIB( LIBRARY_TYPE_EESCHEMA, fn.GetFullPath() ) );
|
|
|
|
m_rescue_lib = std::move( rescue_lib );
|
|
m_rescue_lib->EnableBuffering();
|
|
|
|
// If a rescue library already exists copy the contents of that library so we do not
|
|
// lose an previous rescues.
|
|
PART_LIB* rescueLib = m_prj->SchLibs()->FindLibrary( fn.GetName() );
|
|
|
|
if( rescueLib )
|
|
{
|
|
// For items in the rescue library, aliases are the root symbol.
|
|
std::vector< LIB_ALIAS* > aliases;
|
|
|
|
rescueLib->GetAliases( aliases );
|
|
|
|
for( auto alias : aliases )
|
|
{
|
|
LIB_PART* part = alias->GetPart();
|
|
|
|
wxCHECK2( part, continue );
|
|
|
|
m_rescue_lib->AddPart( new LIB_PART( *part, m_rescue_lib.get() ) );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
bool LEGACY_RESCUER::WriteRescueLibrary( SCH_EDIT_FRAME *aEditFrame )
|
|
{
|
|
try
|
|
{
|
|
m_rescue_lib->Save( false );
|
|
}
|
|
catch( ... /* IO_ERROR ioe */ )
|
|
{
|
|
wxString msg;
|
|
|
|
msg.Printf( _( "Failed to create symbol library file \"%s\"" ),
|
|
m_rescue_lib->GetFullFileName() );
|
|
DisplayError( aEditFrame, msg );
|
|
return false;
|
|
}
|
|
|
|
wxArrayString libNames;
|
|
wxString libPaths;
|
|
|
|
wxString libName = m_rescue_lib->GetName();
|
|
PART_LIBS *libs = dynamic_cast<PART_LIBS*>( m_prj->GetElem( PROJECT::ELEM_SCH_PART_LIBS ) );
|
|
|
|
if( !libs )
|
|
{
|
|
libs = new PART_LIBS();
|
|
m_prj->SetElem( PROJECT::ELEM_SCH_PART_LIBS, libs );
|
|
}
|
|
|
|
try
|
|
{
|
|
PART_LIBS::LibNamesAndPaths( m_prj, false, &libPaths, &libNames );
|
|
|
|
// Make sure the library is not already in the list
|
|
while( libNames.Index( libName ) != wxNOT_FOUND )
|
|
libNames.Remove( libName );
|
|
|
|
// Add the library to the top of the list and save.
|
|
libNames.Insert( libName, 0 );
|
|
PART_LIBS::LibNamesAndPaths( m_prj, true, &libPaths, &libNames );
|
|
}
|
|
catch( const IO_ERROR& )
|
|
{
|
|
// Could not get or save the current libraries.
|
|
return false;
|
|
}
|
|
|
|
// Save the old libraries in case there is a problem after clear(). We'll
|
|
// put them back in.
|
|
boost::ptr_vector<PART_LIB> libsSave;
|
|
libsSave.transfer( libsSave.end(), libs->begin(), libs->end(), *libs );
|
|
|
|
m_prj->SetElem( PROJECT::ELEM_SCH_PART_LIBS, NULL );
|
|
|
|
libs = new PART_LIBS();
|
|
|
|
try
|
|
{
|
|
libs->LoadAllLibraries( m_prj );
|
|
}
|
|
catch( const PARSE_ERROR& )
|
|
{
|
|
// Some libraries were not found. There's no point in showing the error,
|
|
// because it was already shown. Just don't do anything.
|
|
}
|
|
catch( const IO_ERROR& )
|
|
{
|
|
// Restore the old list
|
|
libs->clear();
|
|
libs->transfer( libs->end(), libsSave.begin(), libsSave.end(), libsSave );
|
|
return false;
|
|
}
|
|
|
|
m_prj->SetElem( PROJECT::ELEM_SCH_PART_LIBS, libs );
|
|
|
|
// Update the schematic symbol library links since the library list has changed.
|
|
SCH_SCREENS schematic;
|
|
|
|
schematic.UpdateSymbolLinks();
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void LEGACY_RESCUER::AddPart( LIB_PART* aNewPart )
|
|
{
|
|
wxCHECK_RET( aNewPart, "Invalid LIB_PART pointer." );
|
|
|
|
aNewPart->SetLib( m_rescue_lib.get() );
|
|
m_rescue_lib->AddPart( aNewPart );
|
|
}
|
|
|
|
|
|
SYMBOL_LIB_TABLE_RESCUER::SYMBOL_LIB_TABLE_RESCUER( SCH_EDIT_FRAME& aEditFrame,
|
|
PROJECT& aProject ) :
|
|
RESCUER( aEditFrame, aProject )
|
|
{
|
|
m_properties = std::make_unique<PROPERTIES>();
|
|
}
|
|
|
|
|
|
void SYMBOL_LIB_TABLE_RESCUER::FindCandidates()
|
|
{
|
|
RESCUE_SYMBOL_LIB_TABLE_CANDIDATE::FindRescues( *this, m_all_candidates );
|
|
}
|
|
|
|
|
|
void SYMBOL_LIB_TABLE_RESCUER::InvokeDialog( bool aAskShowAgain )
|
|
{
|
|
InvokeDialogRescueEach( m_edit_frame, static_cast< RESCUER& >( *this ), aAskShowAgain );
|
|
}
|
|
|
|
|
|
void SYMBOL_LIB_TABLE_RESCUER::OpenRescueLibrary()
|
|
{
|
|
m_pi.set( SCH_IO_MGR::FindPlugin( SCH_IO_MGR::SCH_LEGACY ) );
|
|
(*m_properties)[ SCH_LEGACY_PLUGIN::PropBuffering ] = "";
|
|
}
|
|
|
|
|
|
bool SYMBOL_LIB_TABLE_RESCUER::WriteRescueLibrary( SCH_EDIT_FRAME *aEditFrame )
|
|
{
|
|
wxString msg;
|
|
wxFileName fn = GetRescueLibraryFileName();
|
|
|
|
// If the rescue library already exists in the symbol library table no need save it to add
|
|
// it to the table.
|
|
if( !m_prj->SchSymbolLibTable()->HasLibrary( fn.GetName() ) )
|
|
{
|
|
try
|
|
{
|
|
m_pi->SaveLibrary( fn.GetFullPath() );
|
|
}
|
|
catch( const IO_ERROR& ioe )
|
|
{
|
|
msg.Printf( _( "Failed to save rescue library %s." ), fn.GetFullPath() );
|
|
DisplayErrorMessage( aEditFrame, msg, ioe.What() );
|
|
return false;
|
|
}
|
|
|
|
wxString uri = "${KIPRJMOD}/" + fn.GetFullName();
|
|
SYMBOL_LIB_TABLE_ROW* row = new SYMBOL_LIB_TABLE_ROW( fn.GetName(), uri,
|
|
wxString( "Legacy" ) );
|
|
m_prj->SchSymbolLibTable()->InsertRow( row );
|
|
|
|
fn = wxFileName( m_prj->GetProjectPath(), SYMBOL_LIB_TABLE::GetSymbolLibTableFileName() );
|
|
|
|
try
|
|
{
|
|
m_prj->SchSymbolLibTable()->Save( fn.GetFullPath() );
|
|
}
|
|
catch( const IO_ERROR& ioe )
|
|
{
|
|
msg.Printf( _( "Error occurred saving project specific symbol library table." ) );
|
|
DisplayErrorMessage( aEditFrame, msg, ioe.What() );
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Relaod the symbol library table.
|
|
m_prj->SetElem( PROJECT::ELEM_SYMBOL_LIB_TABLE, NULL );
|
|
|
|
// This can only happen if the symbol library table file was currupted on write.
|
|
if( !m_prj->SchSymbolLibTable() )
|
|
return false;
|
|
|
|
// Update the schematic symbol library links since the library list has changed.
|
|
SCH_SCREENS schematic;
|
|
|
|
schematic.UpdateSymbolLinks( true );
|
|
return true;
|
|
}
|
|
|
|
|
|
void SYMBOL_LIB_TABLE_RESCUER::AddPart( LIB_PART* aNewPart )
|
|
{
|
|
wxCHECK_RET( aNewPart, "Invalid LIB_PART pointer." );
|
|
|
|
wxFileName fn = GetRescueLibraryFileName();
|
|
|
|
try
|
|
{
|
|
if( !m_prj->SchSymbolLibTable()->HasLibrary( fn.GetName() ) )
|
|
m_pi->SaveSymbol( fn.GetFullPath(), new LIB_PART( *aNewPart ), m_properties.get() );
|
|
else
|
|
m_prj->SchSymbolLibTable()->SaveSymbol( fn.GetName(), new LIB_PART( *aNewPart ) );
|
|
}
|
|
catch( ... /* IO_ERROR */ )
|
|
{
|
|
}
|
|
}
|