2017-11-12 18:55:20 +01:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2017 CERN
|
2025-01-01 13:30:11 -08:00
|
|
|
* Copyright The KiCad Developers, see AUTHORS.txt for contributors.
|
2017-11-12 18:55:20 +01:00
|
|
|
* @author Maciej Suminski <maciej.suminski@cern.ch>
|
|
|
|
*
|
|
|
|
* 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 3
|
|
|
|
* 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:
|
|
|
|
* https://www.gnu.org/licenses/gpl-3.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 3 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
2024-10-04 15:49:31 +01:00
|
|
|
#include "symbol_library_manager.h"
|
|
|
|
|
2021-06-15 09:24:55 -04:00
|
|
|
#include <symbol_library.h>
|
2021-09-14 19:26:03 +01:00
|
|
|
#include <dialogs/html_message_box.h>
|
2020-10-31 01:27:16 +00:00
|
|
|
#include <symbol_edit_frame.h>
|
2024-10-04 15:49:31 +01:00
|
|
|
#include <symbol_lib_table.h>
|
2018-09-15 01:37:40 +01:00
|
|
|
#include <env_paths.h>
|
|
|
|
#include <pgm_base.h>
|
2023-09-27 23:04:53 -04:00
|
|
|
#include <project_sch.h>
|
2017-11-12 18:55:20 +01:00
|
|
|
#include <kiway.h>
|
2023-09-07 07:22:10 -04:00
|
|
|
#include <core/profile.h>
|
2022-08-14 13:36:40 -04:00
|
|
|
#include <wx_filename.h>
|
2023-12-24 00:31:24 +00:00
|
|
|
#include <sch_io/kicad_legacy/sch_io_kicad_legacy.h>
|
2021-03-19 20:23:54 -04:00
|
|
|
#include <symbol_async_loader.h>
|
2021-08-14 21:05:21 +01:00
|
|
|
#include <progress_reporter.h>
|
2017-11-12 18:55:20 +01:00
|
|
|
#include <list>
|
2021-03-19 20:23:54 -04:00
|
|
|
#include <locale_io.h>
|
2023-08-20 12:05:31 -04:00
|
|
|
#include <confirm.h>
|
2021-07-29 10:56:22 +01:00
|
|
|
#include <string_utils.h>
|
2021-06-01 01:17:57 -04:00
|
|
|
#include "lib_logger.h"
|
2017-11-12 18:55:20 +01:00
|
|
|
|
|
|
|
|
2022-09-15 23:06:23 -04:00
|
|
|
SYMBOL_LIBRARY_MANAGER::SYMBOL_LIBRARY_MANAGER( SCH_BASE_FRAME& aFrame ) :
|
|
|
|
m_frame( aFrame )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2021-06-01 01:17:57 -04:00
|
|
|
m_logger = new LIB_LOGGER();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SYMBOL_LIBRARY_MANAGER::~SYMBOL_LIBRARY_MANAGER()
|
|
|
|
{
|
|
|
|
delete m_logger;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-19 20:23:54 -04:00
|
|
|
void SYMBOL_LIBRARY_MANAGER::Preload( PROGRESS_REPORTER& aReporter )
|
|
|
|
{
|
|
|
|
SYMBOL_ASYNC_LOADER loader( symTable()->GetLogicalLibs(), symTable(), false, nullptr,
|
|
|
|
&aReporter );
|
|
|
|
|
|
|
|
LOCALE_IO toggle;
|
|
|
|
|
|
|
|
loader.Start();
|
|
|
|
|
|
|
|
while( !loader.Done() )
|
|
|
|
{
|
2022-02-27 11:54:12 +00:00
|
|
|
if( !aReporter.KeepRefreshing() )
|
|
|
|
break;
|
2021-03-19 20:23:54 -04:00
|
|
|
|
2022-02-27 11:54:12 +00:00
|
|
|
wxMilliSleep( 33 /* 30 FPS refresh rate */ );
|
2021-03-19 20:23:54 -04:00
|
|
|
}
|
|
|
|
|
2022-02-27 11:54:12 +00:00
|
|
|
loader.Join();
|
2021-03-19 20:23:54 -04:00
|
|
|
|
|
|
|
if( !loader.GetErrors().IsEmpty() )
|
|
|
|
{
|
2021-04-02 19:50:50 -04:00
|
|
|
HTML_MESSAGE_BOX dlg( &m_frame, _( "Load Error" ) );
|
|
|
|
|
2021-06-26 12:33:37 +01:00
|
|
|
dlg.MessageSet( _( "Errors loading symbols:" ) );
|
2021-04-02 19:50:50 -04:00
|
|
|
|
|
|
|
wxString msg = loader.GetErrors();
|
|
|
|
msg.Replace( "\n", "<BR>" );
|
|
|
|
|
|
|
|
dlg.AddHTML_Text( msg );
|
|
|
|
dlg.ShowModal();
|
2021-03-19 20:23:54 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::HasModifications() const
|
2019-06-13 07:44:12 -04:00
|
|
|
{
|
2024-06-16 11:41:01 +01:00
|
|
|
for( const auto& [name, buffer] : m_libs )
|
2019-06-13 07:44:12 -04:00
|
|
|
{
|
2024-06-16 11:41:01 +01:00
|
|
|
if( buffer.IsModified() )
|
2019-06-13 07:44:12 -04:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
int SYMBOL_LIBRARY_MANAGER::GetHash() const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2018-01-08 15:18:08 +01:00
|
|
|
int hash = symTable()->GetModifyHash();
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2024-06-16 11:41:01 +01:00
|
|
|
for( const auto& [name, buffer] : m_libs )
|
|
|
|
hash += buffer.GetHash();
|
2017-11-12 18:55:20 +01:00
|
|
|
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
int SYMBOL_LIBRARY_MANAGER::GetLibraryHash( const wxString& aLibrary ) const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
const auto libBufIt = m_libs.find( aLibrary );
|
2017-11-12 14:24:55 +01:00
|
|
|
|
|
|
|
if( libBufIt != m_libs.end() )
|
|
|
|
return libBufIt->second.GetHash();
|
|
|
|
|
2022-02-27 11:54:12 +00:00
|
|
|
SYMBOL_LIB_TABLE_ROW* row = GetLibrary( aLibrary );
|
2017-11-12 14:24:55 +01:00
|
|
|
|
|
|
|
// return -1 if library does not exist or 0 if not modified
|
2019-11-06 14:15:42 -05:00
|
|
|
return row ? std::hash<std::string>{}( aLibrary.ToStdString() +
|
|
|
|
row->GetFullURI( true ).ToStdString() ) : -1;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
wxArrayString SYMBOL_LIBRARY_MANAGER::GetLibraryNames() const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
wxArrayString res;
|
|
|
|
|
2022-02-27 11:54:12 +00:00
|
|
|
for( const wxString& libName : symTable()->GetLogicalLibs() )
|
2023-01-22 20:49:41 -05:00
|
|
|
{
|
|
|
|
// Database libraries are hidden from the symbol editor at the moment
|
2024-12-02 17:16:18 -08:00
|
|
|
SYMBOL_LIB_TABLE_ROW* row = GetLibrary( libName );
|
|
|
|
|
|
|
|
if( !row || row->SchLibType() == SCH_IO_MGR::SCH_DATABASE )
|
2023-01-22 20:49:41 -05:00
|
|
|
continue;
|
|
|
|
|
2017-11-12 18:55:20 +01:00
|
|
|
res.Add( libName );
|
2023-01-22 20:49:41 -05:00
|
|
|
}
|
2017-11-12 18:55:20 +01:00
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
SYMBOL_LIB_TABLE_ROW* SYMBOL_LIBRARY_MANAGER::GetLibrary( const wxString& aLibrary ) const
|
2018-02-06 15:14:01 +00:00
|
|
|
{
|
2018-02-16 11:59:22 +01:00
|
|
|
SYMBOL_LIB_TABLE_ROW* row = nullptr;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2021-01-10 17:55:30 -05:00
|
|
|
row = symTable()->FindRow( aLibrary, true );
|
2018-02-16 11:59:22 +01:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& e )
|
|
|
|
{
|
2023-08-20 12:05:31 -04:00
|
|
|
wxString msg;
|
|
|
|
|
|
|
|
msg.Printf( _( "Library '%s' not found in the Symbol Library Table." ), aLibrary );
|
|
|
|
DisplayErrorMessage( &m_frame, msg, e.What() );
|
2018-02-16 11:59:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return row;
|
2018-02-06 15:14:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::SaveLibrary( const wxString& aLibrary, const wxString& aFileName,
|
|
|
|
SCH_IO_MGR::SCH_FILE_T aFileType )
|
2017-11-13 14:58:39 +01:00
|
|
|
{
|
2024-10-14 00:47:23 +08:00
|
|
|
wxCHECK( aFileType != SCH_IO_MGR::SCH_FILE_T::SCH_LEGACY, false );
|
|
|
|
wxCHECK_MSG( LibraryExists( aLibrary ), false,
|
|
|
|
wxString::Format( "Library missing: %s", aLibrary ) );
|
|
|
|
|
2017-11-13 14:58:39 +01:00
|
|
|
wxFileName fn( aFileName );
|
|
|
|
wxCHECK( !fn.FileExists() || fn.IsFileWritable(), false );
|
2023-12-27 20:39:29 +00:00
|
|
|
|
|
|
|
IO_RELEASER<SCH_IO> pi( SCH_IO_MGR::FindPlugin( aFileType ) );
|
2024-06-16 11:41:01 +01:00
|
|
|
bool res = true; // assume all libraries are successfully saved
|
2024-08-19 11:47:02 -07:00
|
|
|
std::map<std::string, UTF8> properties;
|
2017-11-13 14:58:39 +01:00
|
|
|
|
2023-12-24 00:31:24 +00:00
|
|
|
properties.emplace( SCH_IO_KICAD_LEGACY::PropBuffering, "" );
|
2021-03-02 19:19:08 -05:00
|
|
|
|
2017-11-13 14:58:39 +01:00
|
|
|
auto it = m_libs.find( aLibrary );
|
|
|
|
|
|
|
|
if( it != m_libs.end() )
|
|
|
|
{
|
|
|
|
// Handle buffered library
|
|
|
|
LIB_BUFFER& libBuf = it->second;
|
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
const auto& symbolBuffers = libBuf.GetBuffers();
|
2017-11-13 14:58:39 +01:00
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
for( const std::shared_ptr<SYMBOL_BUFFER>& symbolBuf : symbolBuffers )
|
2017-11-13 14:58:39 +01:00
|
|
|
{
|
2024-09-17 20:23:07 +01:00
|
|
|
wxCHECK2( symbolBuf, continue );
|
|
|
|
if( !libBuf.SaveBuffer( *symbolBuf, aFileName, &*pi, true ) )
|
2017-11-13 14:58:39 +01:00
|
|
|
{
|
|
|
|
// Something went wrong, but try to save other libraries
|
|
|
|
res = false;
|
|
|
|
}
|
|
|
|
}
|
2017-11-14 14:40:37 +01:00
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
// clear the deleted symbols buffer only if data is saved to the original file
|
2018-02-05 16:51:42 +01:00
|
|
|
wxFileName original, destination( aFileName );
|
2022-10-12 15:56:21 +02:00
|
|
|
SYMBOL_LIB_TABLE_ROW* row = GetLibrary( aLibrary );
|
2017-11-14 14:40:37 +01:00
|
|
|
|
2018-02-05 16:51:42 +01:00
|
|
|
if( row )
|
|
|
|
{
|
2022-10-12 15:56:21 +02:00
|
|
|
original = row->GetFullURI();
|
|
|
|
original.Normalize( FN_NORMALIZE_FLAGS | wxPATH_NORM_ENV_VARS );
|
2018-02-05 16:51:42 +01:00
|
|
|
}
|
|
|
|
|
2022-08-14 13:36:40 -04:00
|
|
|
destination.Normalize( FN_NORMALIZE_FLAGS | wxPATH_NORM_ENV_VARS );
|
2018-02-05 16:51:42 +01:00
|
|
|
|
|
|
|
if( res && original == destination )
|
2017-11-14 14:40:37 +01:00
|
|
|
libBuf.ClearDeletedBuffer();
|
2017-11-13 14:58:39 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Handle original library
|
2021-06-15 08:31:28 -04:00
|
|
|
for( LIB_SYMBOL* symbol : getOriginalSymbols( aLibrary ) )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2021-06-10 14:51:46 -04:00
|
|
|
LIB_SYMBOL* newSymbol;
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2022-05-06 09:47:06 -07:00
|
|
|
try
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2022-05-06 09:47:06 -07:00
|
|
|
if( symbol->IsAlias() )
|
|
|
|
{
|
|
|
|
std::shared_ptr< LIB_SYMBOL > oldParent = symbol->GetParent().lock();
|
|
|
|
|
|
|
|
wxCHECK_MSG( oldParent, false,
|
2023-08-20 12:05:31 -04:00
|
|
|
wxString::Format( wxT( "Derived symbol '%s' found with "
|
|
|
|
"undefined parent." ),
|
2022-05-06 09:47:06 -07:00
|
|
|
symbol->GetName() ) );
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2022-05-06 09:47:06 -07:00
|
|
|
LIB_SYMBOL* libParent = pi->LoadSymbol( aLibrary, oldParent->GetName(),
|
|
|
|
&properties );
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2022-05-06 09:47:06 -07:00
|
|
|
if( !libParent )
|
|
|
|
{
|
|
|
|
libParent = new LIB_SYMBOL( *oldParent.get() );
|
|
|
|
pi->SaveSymbol( aLibrary, libParent, &properties );
|
|
|
|
}
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2022-05-06 09:47:06 -07:00
|
|
|
newSymbol = new LIB_SYMBOL( *symbol );
|
|
|
|
newSymbol->SetParent( libParent );
|
|
|
|
pi->SaveSymbol( aLibrary, newSymbol, &properties );
|
|
|
|
}
|
|
|
|
else if( !pi->LoadSymbol( aLibrary, symbol->GetName(), &properties ) )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2022-05-06 09:47:06 -07:00
|
|
|
pi->SaveSymbol( aLibrary, new LIB_SYMBOL( *symbol ), &properties );
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
}
|
2022-05-06 09:47:06 -07:00
|
|
|
catch( ... )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2022-05-06 09:47:06 -07:00
|
|
|
res = false;
|
|
|
|
break;
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
}
|
2017-11-13 14:58:39 +01:00
|
|
|
}
|
|
|
|
|
2020-07-19 18:27:19 +02:00
|
|
|
try
|
|
|
|
{
|
|
|
|
pi->SaveLibrary( aFileName );
|
|
|
|
}
|
|
|
|
catch( ... )
|
|
|
|
{
|
|
|
|
// return false because something happens.
|
|
|
|
// The library is not successfully saved
|
|
|
|
res = false;
|
|
|
|
}
|
|
|
|
|
2017-11-13 14:58:39 +01:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::IsLibraryModified( const wxString& aLibrary ) const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
auto it = m_libs.find( aLibrary );
|
|
|
|
return it != m_libs.end() ? it->second.IsModified() : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::IsSymbolModified( const wxString& aAlias,
|
|
|
|
const wxString& aLibrary ) const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
auto libIt = m_libs.find( aLibrary );
|
|
|
|
|
|
|
|
if( libIt == m_libs.end() )
|
|
|
|
return false;
|
|
|
|
|
2024-06-16 11:41:01 +01:00
|
|
|
const LIB_BUFFER& buf = libIt->second;
|
2023-10-09 14:58:49 -04:00
|
|
|
const std::shared_ptr<SYMBOL_BUFFER> symbolBuf = buf.GetBuffer( aAlias );
|
2024-06-16 11:41:01 +01:00
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
return symbolBuf ? symbolBuf->IsModified() : false;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
void SYMBOL_LIBRARY_MANAGER::SetSymbolModified( const wxString& aAlias, const wxString& aLibrary )
|
2022-04-21 11:57:15 +01:00
|
|
|
{
|
|
|
|
auto libIt = m_libs.find( aLibrary );
|
|
|
|
|
|
|
|
if( libIt == m_libs.end() )
|
|
|
|
return;
|
|
|
|
|
2024-06-16 11:41:01 +01:00
|
|
|
const LIB_BUFFER& buf = libIt->second;
|
2023-10-09 14:58:49 -04:00
|
|
|
std::shared_ptr<SYMBOL_BUFFER> symbolBuf = buf.GetBuffer( aAlias );
|
2022-10-17 09:19:39 -04:00
|
|
|
|
|
|
|
wxCHECK( symbolBuf, /* void */ );
|
|
|
|
|
2022-04-21 11:57:15 +01:00
|
|
|
symbolBuf->GetScreen()->SetContentModified();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::ClearLibraryModified( const wxString& aLibrary ) const
|
2017-11-13 23:13:10 +01:00
|
|
|
{
|
|
|
|
auto libIt = m_libs.find( aLibrary );
|
|
|
|
|
|
|
|
if( libIt == m_libs.end() )
|
|
|
|
return false;
|
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
for( auto& symbolBuf : libIt->second.GetBuffers() )
|
2017-11-13 23:13:10 +01:00
|
|
|
{
|
2021-06-10 14:51:46 -04:00
|
|
|
SCH_SCREEN* screen = symbolBuf->GetScreen();
|
2017-11-13 23:13:10 +01:00
|
|
|
|
|
|
|
if( screen )
|
2021-05-28 12:07:04 -07:00
|
|
|
screen->SetContentModified( false );
|
2017-11-13 23:13:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::ClearSymbolModified( const wxString& aAlias,
|
|
|
|
const wxString& aLibrary ) const
|
2017-11-13 23:13:10 +01:00
|
|
|
{
|
2024-06-16 11:41:01 +01:00
|
|
|
auto libIt = m_libs.find( aLibrary );
|
2017-11-13 23:13:10 +01:00
|
|
|
|
2024-06-16 11:41:01 +01:00
|
|
|
if( libIt == m_libs.end() )
|
2017-11-13 23:13:10 +01:00
|
|
|
return false;
|
|
|
|
|
2024-06-16 11:41:01 +01:00
|
|
|
auto symbolBuf = libIt->second.GetBuffer( aAlias );
|
2021-06-10 14:51:46 -04:00
|
|
|
wxCHECK( symbolBuf, false );
|
2017-11-13 23:13:10 +01:00
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
symbolBuf->GetScreen()->SetContentModified( false );
|
2017-11-13 23:13:10 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::IsLibraryReadOnly( const wxString& aLibrary ) const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-10-14 00:47:23 +08:00
|
|
|
wxCHECK_MSG( LibraryExists( aLibrary ), true,
|
|
|
|
wxString::Format( "Library missing: %s", aLibrary ) );
|
2020-05-15 09:25:11 -04:00
|
|
|
|
2020-05-22 10:12:44 -04:00
|
|
|
return !symTable()->IsSymbolLibWritable( aLibrary );
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::IsLibraryLoaded( const wxString& aLibrary ) const
|
2020-08-30 00:19:05 +01:00
|
|
|
{
|
2024-10-14 00:47:23 +08:00
|
|
|
wxCHECK_MSG( LibraryExists( aLibrary ), false,
|
|
|
|
wxString::Format( "Library missing: %s", aLibrary ) );
|
2020-08-30 00:19:05 +01:00
|
|
|
|
|
|
|
return symTable()->IsSymbolLibLoaded( aLibrary );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
std::list<LIB_SYMBOL*> SYMBOL_LIBRARY_MANAGER::GetAliases( const wxString& aLibrary ) const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2021-06-10 14:51:46 -04:00
|
|
|
std::list<LIB_SYMBOL*> ret;
|
2024-10-14 00:47:23 +08:00
|
|
|
wxCHECK_MSG( LibraryExists( aLibrary ), ret,
|
|
|
|
wxString::Format( "Library missing: %s", aLibrary ) );
|
2017-11-12 18:55:20 +01:00
|
|
|
|
|
|
|
auto libIt = m_libs.find( aLibrary );
|
|
|
|
|
|
|
|
if( libIt != m_libs.end() )
|
|
|
|
{
|
2024-06-16 11:41:01 +01:00
|
|
|
for( const std::shared_ptr<SYMBOL_BUFFER>& symbolBuf : libIt->second.GetBuffers() )
|
2024-09-17 19:53:56 +01:00
|
|
|
ret.push_back( &symbolBuf->GetSymbol() );
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-06-10 14:51:46 -04:00
|
|
|
std::vector<LIB_SYMBOL*> aliases;
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2018-01-25 23:49:04 +00:00
|
|
|
try
|
|
|
|
{
|
|
|
|
symTable()->LoadSymbolLib( aliases, aLibrary );
|
|
|
|
}
|
2018-02-13 08:50:51 +01:00
|
|
|
catch( const IO_ERROR& e )
|
|
|
|
{
|
2019-07-10 18:59:39 +01:00
|
|
|
wxLogWarning( e.Problem() );
|
2018-02-13 08:50:51 +01:00
|
|
|
}
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2018-01-25 23:49:04 +00:00
|
|
|
std::copy( aliases.begin(), aliases.end(), std::back_inserter( ret ) );
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
LIB_SYMBOL* SYMBOL_LIBRARY_MANAGER::GetBufferedSymbol( const wxString& aAlias,
|
|
|
|
const wxString& aLibrary )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-10-14 00:47:23 +08:00
|
|
|
wxCHECK_MSG( LibraryExists( aLibrary ), nullptr,
|
|
|
|
wxString::Format( "Library missing: %s, for alias %s", aLibrary, aAlias ) );
|
2017-11-12 18:55:20 +01:00
|
|
|
|
|
|
|
// try the library buffers first
|
|
|
|
LIB_BUFFER& libBuf = getLibraryBuffer( aLibrary );
|
2021-06-15 08:31:28 -04:00
|
|
|
LIB_SYMBOL* bufferedSymbol = libBuf.GetSymbol( aAlias );
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
if( !bufferedSymbol ) // no buffer symbol found
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2021-06-10 14:51:46 -04:00
|
|
|
// create a copy of the symbol
|
2018-02-13 08:50:51 +01:00
|
|
|
try
|
|
|
|
{
|
2021-06-10 14:51:46 -04:00
|
|
|
LIB_SYMBOL* symbol = symTable()->LoadSymbol( aLibrary, aAlias );
|
2018-11-21 19:50:26 +00:00
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
if( symbol == nullptr )
|
2018-11-21 19:50:26 +00:00
|
|
|
THROW_IO_ERROR( _( "Symbol not found." ) );
|
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
LIB_SYMBOL* bufferedParent = nullptr;
|
2019-11-06 14:15:42 -05:00
|
|
|
|
|
|
|
// Create parent symbols on demand so parent symbol can be set.
|
2021-06-10 14:51:46 -04:00
|
|
|
if( symbol->IsAlias() )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2024-06-16 11:41:01 +01:00
|
|
|
std::shared_ptr<LIB_SYMBOL> parent = symbol->GetParent().lock();
|
2019-11-06 14:15:42 -05:00
|
|
|
wxCHECK_MSG( parent, nullptr,
|
|
|
|
wxString::Format( "Derived symbol '%s' found with undefined parent.",
|
2021-06-10 14:51:46 -04:00
|
|
|
symbol->GetName() ) );
|
2019-11-06 14:15:42 -05:00
|
|
|
|
|
|
|
// Check if the parent symbol buffer has already be created.
|
2021-06-15 08:31:28 -04:00
|
|
|
bufferedParent = libBuf.GetSymbol( parent->GetName() );
|
2019-11-06 14:15:42 -05:00
|
|
|
|
|
|
|
if( !bufferedParent )
|
|
|
|
{
|
2024-09-17 19:29:17 +01:00
|
|
|
auto newParent = std::make_unique<LIB_SYMBOL>( *parent.get() );
|
|
|
|
bufferedParent = newParent.get();
|
|
|
|
libBuf.CreateBuffer( std::move( newParent ), std::make_unique<SCH_SCREEN>() );
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-17 19:29:17 +01:00
|
|
|
auto newSymbol = std::make_unique<LIB_SYMBOL>( *symbol );
|
|
|
|
bufferedSymbol = newSymbol.get();
|
2019-11-06 14:15:42 -05:00
|
|
|
|
|
|
|
if( bufferedParent )
|
2024-09-17 19:29:17 +01:00
|
|
|
newSymbol->SetParent( bufferedParent );
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2024-09-17 19:29:17 +01:00
|
|
|
libBuf.CreateBuffer( std::move( newSymbol ), std::make_unique<SCH_SCREEN>() );
|
2018-01-10 12:04:34 +01:00
|
|
|
}
|
2018-02-13 08:50:51 +01:00
|
|
|
catch( const IO_ERROR& e )
|
2018-01-10 12:04:34 +01:00
|
|
|
{
|
2023-08-20 12:05:31 -04:00
|
|
|
wxString msg;
|
|
|
|
|
|
|
|
msg.Printf( _( "Error loading symbol %s from library '%s'." ), aAlias, aLibrary );
|
|
|
|
DisplayErrorMessage( &m_frame, msg, e.What() );
|
2021-06-15 08:31:28 -04:00
|
|
|
bufferedSymbol = nullptr;
|
2018-01-10 12:04:34 +01:00
|
|
|
}
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
return bufferedSymbol;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
SCH_SCREEN* SYMBOL_LIBRARY_MANAGER::GetScreen( const wxString& aAlias, const wxString& aLibrary )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-10-14 00:47:23 +08:00
|
|
|
wxCHECK_MSG( LibraryExists( aLibrary ), nullptr,
|
|
|
|
wxString::Format( "Library missing: %s, for alias %s", aLibrary, aAlias ) );
|
|
|
|
wxCHECK_MSG( !aAlias.IsEmpty(), nullptr,
|
|
|
|
wxString::Format( "Alias missing in library: %s", aLibrary ) );
|
2017-11-12 18:55:20 +01:00
|
|
|
auto it = m_libs.find( aLibrary );
|
|
|
|
wxCHECK( it != m_libs.end(), nullptr );
|
|
|
|
|
2024-06-16 11:41:01 +01:00
|
|
|
LIB_BUFFER& buf = it->second;
|
|
|
|
std::shared_ptr<SYMBOL_BUFFER> symbolBuf = buf.GetBuffer( aAlias );
|
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
return symbolBuf ? symbolBuf->GetScreen() : nullptr;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::UpdateSymbol( LIB_SYMBOL* aSymbol, const wxString& aLibrary )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-10-14 00:47:23 +08:00
|
|
|
wxCHECK_MSG( aSymbol, false, wxString::Format( "Null symbol in library: %s", aLibrary ) );
|
|
|
|
wxCHECK_MSG( LibraryExists( aLibrary ), false,
|
2025-02-06 09:47:00 -05:00
|
|
|
wxString::Format( "Library missing: %s, for symbol %s", aLibrary,
|
2024-10-14 00:47:23 +08:00
|
|
|
aSymbol->GetName() ) );
|
2024-06-16 11:41:01 +01:00
|
|
|
|
|
|
|
LIB_BUFFER& libBuf = getLibraryBuffer( aLibrary );
|
|
|
|
std::shared_ptr<SYMBOL_BUFFER> symbolBuf = libBuf.GetBuffer( aSymbol->GetName() );
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
if( symbolBuf ) // Existing symbol.
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
LIB_SYMBOL& bufferedSymbol = symbolBuf->GetSymbol();
|
2019-12-13 16:51:59 -05:00
|
|
|
|
2024-07-16 11:30:48 -07:00
|
|
|
// If we are coming from a different library, the library ID needs to be preserved
|
2024-09-17 19:53:56 +01:00
|
|
|
const LIB_ID libId = bufferedSymbol.GetLibId();
|
|
|
|
bufferedSymbol = *aSymbol;
|
|
|
|
bufferedSymbol.SetLibId( libId );
|
2024-07-16 11:30:48 -07:00
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
symbolBuf->GetScreen()->SetContentModified();
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
2024-06-16 11:41:01 +01:00
|
|
|
else // New symbol
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-09-17 19:29:17 +01:00
|
|
|
auto symbolCopy = std::make_unique<LIB_SYMBOL>( *aSymbol, nullptr );
|
2019-12-26 08:00:59 -05:00
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
symbolCopy->SetLibId( LIB_ID( aLibrary, aSymbol->GetLibId().GetLibItemName() ) );
|
2019-12-26 08:00:59 -05:00
|
|
|
|
2024-09-17 19:29:17 +01:00
|
|
|
auto newScreen = std::make_unique<SCH_SCREEN>();
|
|
|
|
SCH_SCREEN& screen = *newScreen;
|
|
|
|
libBuf.CreateBuffer( std::move( symbolCopy ), std::move( newScreen ) );
|
|
|
|
screen.SetContentModified();
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::UpdateSymbolAfterRename( LIB_SYMBOL* aSymbol, const wxString& aOldName,
|
|
|
|
const wxString& aLibrary )
|
2018-01-19 18:56:01 +00:00
|
|
|
{
|
2024-06-16 11:41:01 +01:00
|
|
|
LIB_BUFFER& libBuf = getLibraryBuffer( aLibrary );
|
|
|
|
std::shared_ptr<SYMBOL_BUFFER> symbolBuf = libBuf.GetBuffer( aOldName );
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2024-09-18 22:53:46 +01:00
|
|
|
wxCHECK( symbolBuf && aSymbol, false );
|
2018-01-19 18:56:01 +00:00
|
|
|
|
2024-09-18 22:53:46 +01:00
|
|
|
libBuf.UpdateBuffer( *symbolBuf, *aSymbol );
|
2022-09-15 23:06:23 -04:00
|
|
|
OnDataChanged();
|
2018-08-25 13:21:09 +01:00
|
|
|
|
2018-01-19 18:56:01 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
LIB_ID SYMBOL_LIBRARY_MANAGER::RevertSymbol( const wxString& aAlias, const wxString& aLibrary )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
auto it = m_libs.find( aLibrary );
|
|
|
|
|
|
|
|
if( it == m_libs.end() ) // no items to flush
|
2018-01-19 18:56:01 +00:00
|
|
|
return LIB_ID( aLibrary, aAlias );
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2024-06-16 11:41:01 +01:00
|
|
|
std::shared_ptr<SYMBOL_BUFFER> symbolBuf = it->second.GetBuffer( aAlias );
|
2021-06-10 14:51:46 -04:00
|
|
|
wxCHECK( symbolBuf, LIB_ID( aLibrary, aAlias ) );
|
2024-09-17 19:53:56 +01:00
|
|
|
LIB_SYMBOL original( symbolBuf->GetOriginal() );
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2018-01-19 18:56:01 +00:00
|
|
|
if( original.GetName() != aAlias )
|
|
|
|
{
|
2021-06-15 08:31:28 -04:00
|
|
|
UpdateSymbolAfterRename( &original, aAlias, aLibrary );
|
2018-01-19 18:56:01 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-10-08 15:56:07 +02:00
|
|
|
// copy the initial data to the current symbol to restore
|
2024-09-17 19:53:56 +01:00
|
|
|
symbolBuf->GetSymbol() = original;
|
2022-09-15 23:06:23 -04:00
|
|
|
OnDataChanged();
|
2018-01-19 18:56:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return LIB_ID( aLibrary, original.GetName() );
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::RevertLibrary( const wxString& aLibrary )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
auto it = m_libs.find( aLibrary );
|
|
|
|
|
|
|
|
if( it == m_libs.end() ) // nothing to reverse
|
|
|
|
return false;
|
|
|
|
|
|
|
|
m_libs.erase( it );
|
2022-09-15 23:06:23 -04:00
|
|
|
OnDataChanged();
|
2017-11-12 18:55:20 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::RevertAll()
|
2019-06-13 07:44:12 -04:00
|
|
|
{
|
|
|
|
bool retv = true;
|
|
|
|
|
|
|
|
// Nothing to revert.
|
|
|
|
if( GetHash() == 0 )
|
|
|
|
return true;
|
|
|
|
|
2019-12-05 07:20:59 -08:00
|
|
|
for( const auto& lib : m_libs )
|
2019-06-13 07:44:12 -04:00
|
|
|
{
|
|
|
|
if( !lib.second.IsModified() )
|
|
|
|
continue;
|
|
|
|
|
2024-06-16 11:41:01 +01:00
|
|
|
for( const std::shared_ptr<SYMBOL_BUFFER>& buffer : lib.second.GetBuffers() )
|
2019-06-13 07:44:12 -04:00
|
|
|
{
|
|
|
|
if( !buffer->IsModified() )
|
|
|
|
continue;
|
|
|
|
|
2024-09-17 19:53:56 +01:00
|
|
|
RevertSymbol( lib.first, buffer->GetOriginal().GetName() );
|
2019-06-13 07:44:12 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return retv;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::RemoveSymbol( const wxString& aAlias, const wxString& aLibrary )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-06-16 11:41:01 +01:00
|
|
|
LIB_BUFFER& libBuf = getLibraryBuffer( aLibrary );
|
|
|
|
std::shared_ptr<SYMBOL_BUFFER> symbolBuf = libBuf.GetBuffer( aAlias );
|
2021-06-10 14:51:46 -04:00
|
|
|
wxCHECK( symbolBuf, false );
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2019-11-06 14:15:42 -05:00
|
|
|
bool retv = true;
|
|
|
|
|
2024-09-18 22:53:46 +01:00
|
|
|
retv &= libBuf.DeleteBuffer( *symbolBuf );
|
2022-09-15 23:06:23 -04:00
|
|
|
OnDataChanged();
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2019-11-06 14:15:42 -05:00
|
|
|
return retv;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
LIB_SYMBOL* SYMBOL_LIBRARY_MANAGER::GetAlias( const wxString& aAlias,
|
2021-06-15 08:31:28 -04:00
|
|
|
const wxString& aLibrary ) const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
// Try the library buffers first
|
|
|
|
auto libIt = m_libs.find( aLibrary );
|
|
|
|
|
|
|
|
if( libIt != m_libs.end() )
|
|
|
|
{
|
2021-06-15 08:31:28 -04:00
|
|
|
LIB_SYMBOL* symbol = libIt->second.GetSymbol( aAlias );
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
if( symbol )
|
|
|
|
return symbol;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
// Get the original symbol
|
|
|
|
LIB_SYMBOL* alias = nullptr;
|
2017-11-13 15:25:55 +01:00
|
|
|
|
2018-02-13 08:50:51 +01:00
|
|
|
try
|
|
|
|
{
|
2018-01-08 15:18:08 +01:00
|
|
|
alias = symTable()->LoadSymbol( aLibrary, aAlias );
|
2018-02-13 08:50:51 +01:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& e )
|
|
|
|
{
|
2023-08-20 12:05:31 -04:00
|
|
|
wxString msg;
|
|
|
|
|
|
|
|
msg.Printf( _( "Cannot load symbol '%s' from library '%s'." ), aAlias, aLibrary );
|
|
|
|
DisplayErrorMessage( &m_frame, msg, e.What() );
|
2018-02-13 08:50:51 +01:00
|
|
|
}
|
2017-11-13 15:25:55 +01:00
|
|
|
|
|
|
|
return alias;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::SymbolExists( const wxString& aAlias, const wxString& aLibrary ) const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
auto libBufIt = m_libs.find( aLibrary );
|
2021-06-10 14:51:46 -04:00
|
|
|
LIB_SYMBOL* alias = nullptr;
|
2017-11-12 18:55:20 +01:00
|
|
|
|
|
|
|
if( libBufIt != m_libs.end() )
|
|
|
|
return !!libBufIt->second.GetBuffer( aAlias );
|
|
|
|
|
2018-02-13 08:50:51 +01:00
|
|
|
try
|
|
|
|
{
|
2018-01-08 15:18:08 +01:00
|
|
|
alias = symTable()->LoadSymbol( aLibrary, aAlias );
|
2019-06-13 07:44:12 -04:00
|
|
|
}
|
2019-06-05 20:15:57 +01:00
|
|
|
catch( IO_ERROR& )
|
2018-02-13 08:50:51 +01:00
|
|
|
{
|
|
|
|
// checking if certain symbol exists, so its absence is perfectly fine
|
|
|
|
}
|
2017-11-13 15:25:55 +01:00
|
|
|
|
|
|
|
return alias != nullptr;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2025-01-25 18:50:09 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::SymbolNameInUse( const wxString& aName, const wxString& aLibrary )
|
|
|
|
{
|
|
|
|
wxArrayString existing;
|
|
|
|
|
|
|
|
GetSymbolNames( aLibrary, existing );
|
|
|
|
|
|
|
|
for( wxString& candidate : existing )
|
|
|
|
{
|
|
|
|
if( candidate.CmpNoCase( aName ) )
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::LibraryExists( const wxString& aLibrary, bool aCheckEnabled ) const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2017-11-12 14:24:55 +01:00
|
|
|
if( aLibrary.IsEmpty() )
|
|
|
|
return false;
|
|
|
|
|
2017-11-12 18:55:20 +01:00
|
|
|
if( m_libs.count( aLibrary ) > 0 )
|
|
|
|
return true;
|
|
|
|
|
2018-04-28 10:43:41 +01:00
|
|
|
return symTable()->HasLibrary( aLibrary, aCheckEnabled );
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
2019-06-13 07:44:12 -04:00
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
wxString SYMBOL_LIBRARY_MANAGER::GetUniqueLibraryName() const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
wxString name = "New_Library";
|
|
|
|
|
|
|
|
if( !LibraryExists( name ) )
|
|
|
|
return name;
|
|
|
|
|
|
|
|
name += "_";
|
|
|
|
|
|
|
|
for( unsigned int i = 0; i < std::numeric_limits<unsigned int>::max(); ++i )
|
|
|
|
{
|
|
|
|
if( !LibraryExists( name + wxString::Format( "%u", i ) ) )
|
|
|
|
return name + wxString::Format( "%u", i );
|
|
|
|
}
|
|
|
|
|
|
|
|
wxFAIL;
|
|
|
|
return wxEmptyString;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-08-20 12:05:31 -04:00
|
|
|
void SYMBOL_LIBRARY_MANAGER::GetSymbolNames( const wxString& aLibraryName,
|
|
|
|
wxArrayString& aSymbolNames,
|
|
|
|
SYMBOL_NAME_FILTER aFilter )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
|
|
|
LIB_BUFFER& libBuf = getLibraryBuffer( aLibraryName );
|
|
|
|
|
2023-08-20 12:05:31 -04:00
|
|
|
libBuf.GetSymbolNames( aSymbolNames, aFilter );
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER:: HasDerivedSymbols( const wxString& aSymbolName,
|
|
|
|
const wxString& aLibraryName )
|
2019-12-12 10:44:53 -05:00
|
|
|
{
|
|
|
|
LIB_BUFFER& libBuf = getLibraryBuffer( aLibraryName );
|
|
|
|
|
|
|
|
return libBuf.HasDerivedSymbols( aSymbolName );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-03-19 20:23:54 -04:00
|
|
|
size_t SYMBOL_LIBRARY_MANAGER::GetLibraryCount() const
|
|
|
|
{
|
|
|
|
return symTable()->GetLogicalLibs().size();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
wxString SYMBOL_LIBRARY_MANAGER::getLibraryName( const wxString& aFilePath )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
wxFileName fn( aFilePath );
|
|
|
|
return fn.GetName();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::addLibrary( const wxString& aFilePath, bool aCreate,
|
2024-09-17 13:12:21 +01:00
|
|
|
SYMBOL_LIB_TABLE& aTable )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
wxString libName = getLibraryName( aFilePath );
|
|
|
|
wxCHECK( !LibraryExists( libName ), false ); // either create or add an existing one
|
|
|
|
|
2018-09-15 01:37:40 +01:00
|
|
|
// try to use path normalized to an environmental variable or project path
|
|
|
|
wxString relPath = NormalizePath( aFilePath, &Pgm().GetLocalEnvVariables(), &m_frame.Prj() );
|
|
|
|
|
2020-05-07 11:25:48 -04:00
|
|
|
SCH_IO_MGR::SCH_FILE_T schFileType = SCH_IO_MGR::GuessPluginTypeFromLibPath( aFilePath );
|
2023-08-26 22:28:53 +03:00
|
|
|
|
|
|
|
if( schFileType == SCH_IO_MGR::SCH_FILE_UNKNOWN )
|
|
|
|
schFileType = SCH_IO_MGR::SCH_LEGACY;
|
|
|
|
|
2020-05-07 11:25:48 -04:00
|
|
|
wxString typeName = SCH_IO_MGR::ShowType( schFileType );
|
2019-08-15 08:19:15 +01:00
|
|
|
SYMBOL_LIB_TABLE_ROW* libRow = new SYMBOL_LIB_TABLE_ROW( libName, relPath, typeName );
|
2024-09-17 13:12:21 +01:00
|
|
|
aTable.InsertRow( libRow );
|
2017-11-12 18:55:20 +01:00
|
|
|
|
|
|
|
if( aCreate )
|
2017-11-22 13:20:26 +01:00
|
|
|
{
|
2020-05-15 09:25:11 -04:00
|
|
|
wxCHECK( schFileType != SCH_IO_MGR::SCH_FILE_T::SCH_LEGACY, false );
|
|
|
|
|
2019-04-18 22:00:04 +01:00
|
|
|
try
|
2017-11-22 13:20:26 +01:00
|
|
|
{
|
2024-09-17 13:12:21 +01:00
|
|
|
aTable.CreateSymbolLib( libName );
|
2019-04-18 22:00:04 +01:00
|
|
|
}
|
2020-02-29 22:53:51 +01:00
|
|
|
catch( const IO_ERROR& )
|
2019-04-18 22:00:04 +01:00
|
|
|
{
|
2024-09-17 13:12:21 +01:00
|
|
|
aTable.RemoveRow( libRow );
|
2019-04-18 22:00:04 +01:00
|
|
|
return false;
|
2017-11-22 13:20:26 +01:00
|
|
|
}
|
|
|
|
}
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2022-09-15 23:06:23 -04:00
|
|
|
OnDataChanged();
|
2017-11-12 18:55:20 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
SYMBOL_LIB_TABLE* SYMBOL_LIBRARY_MANAGER::symTable() const
|
2018-01-08 15:18:08 +01:00
|
|
|
{
|
2023-09-27 23:04:53 -04:00
|
|
|
return PROJECT_SCH::SchSymbolLibTable( &m_frame.Prj() );
|
2018-01-08 15:18:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
std::set<LIB_SYMBOL*> SYMBOL_LIBRARY_MANAGER::getOriginalSymbols( const wxString& aLibrary )
|
2017-11-13 14:58:39 +01:00
|
|
|
{
|
2021-06-10 14:51:46 -04:00
|
|
|
std::set<LIB_SYMBOL*> symbols;
|
|
|
|
wxCHECK( LibraryExists( aLibrary ), symbols );
|
2017-11-13 14:58:39 +01:00
|
|
|
|
2018-02-13 08:50:51 +01:00
|
|
|
try
|
|
|
|
{
|
2017-11-13 15:25:55 +01:00
|
|
|
wxArrayString aliases;
|
2018-01-08 15:18:08 +01:00
|
|
|
symTable()->EnumerateSymbolLib( aLibrary, aliases );
|
2017-11-13 15:25:55 +01:00
|
|
|
|
|
|
|
for( const auto& aliasName : aliases )
|
|
|
|
{
|
2021-06-10 14:51:46 -04:00
|
|
|
LIB_SYMBOL* alias = symTable()->LoadSymbol( aLibrary, aliasName );
|
|
|
|
symbols.insert( alias );
|
2017-11-13 15:25:55 +01:00
|
|
|
}
|
2019-04-18 22:00:04 +01:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& e )
|
2018-02-13 08:50:51 +01:00
|
|
|
{
|
2023-08-20 12:05:31 -04:00
|
|
|
wxString msg;
|
|
|
|
|
|
|
|
msg.Printf( _( "Cannot enumerate library '%s'." ), aLibrary );
|
|
|
|
DisplayErrorMessage( &m_frame, msg, e.What() );
|
2018-02-13 08:50:51 +01:00
|
|
|
}
|
2017-11-13 14:58:39 +01:00
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
return symbols;
|
2017-11-13 14:58:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
LIB_BUFFER& SYMBOL_LIBRARY_MANAGER::getLibraryBuffer( const wxString& aLibrary )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
auto it = m_libs.find( aLibrary );
|
|
|
|
|
|
|
|
if( it != m_libs.end() )
|
|
|
|
return it->second;
|
|
|
|
|
2018-01-10 12:04:34 +01:00
|
|
|
// The requested buffer does not exist yet, so create one
|
2017-11-12 18:55:20 +01:00
|
|
|
auto ret = m_libs.emplace( aLibrary, LIB_BUFFER( aLibrary ) );
|
|
|
|
LIB_BUFFER& buf = ret.first->second;
|
|
|
|
|
2024-06-16 11:41:01 +01:00
|
|
|
for( LIB_SYMBOL* symbol : getOriginalSymbols( aLibrary ) )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2021-06-10 14:51:46 -04:00
|
|
|
if( symbol->IsAlias() )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2024-09-17 19:29:17 +01:00
|
|
|
std::shared_ptr<LIB_SYMBOL> oldParent = symbol->GetParent().lock();
|
2019-11-06 14:15:42 -05:00
|
|
|
|
|
|
|
wxCHECK_MSG( oldParent, buf,
|
|
|
|
wxString::Format( "Derived symbol '%s' found with undefined parent.",
|
2021-06-10 14:51:46 -04:00
|
|
|
symbol->GetName() ) );
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
LIB_SYMBOL* libParent = buf.GetSymbol( oldParent->GetName() );
|
2019-11-06 14:15:42 -05:00
|
|
|
|
|
|
|
if( !libParent )
|
|
|
|
{
|
2024-09-17 19:29:17 +01:00
|
|
|
auto newParent = std::make_unique<LIB_SYMBOL>( *oldParent.get() );
|
|
|
|
libParent = newParent.get();
|
|
|
|
buf.CreateBuffer( std::move( newParent ), std::make_unique<SCH_SCREEN>() );
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
|
2024-09-17 19:29:17 +01:00
|
|
|
auto newSymbol = std::make_unique<LIB_SYMBOL>( *symbol );
|
2019-11-06 14:15:42 -05:00
|
|
|
newSymbol->SetParent( libParent );
|
2024-09-17 19:29:17 +01:00
|
|
|
buf.CreateBuffer( std::move( newSymbol ), std::make_unique<SCH_SCREEN>() );
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
2021-06-15 08:31:28 -04:00
|
|
|
else if( !buf.GetSymbol( symbol->GetName() ) )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2024-09-17 19:29:17 +01:00
|
|
|
buf.CreateBuffer( std::make_unique<LIB_SYMBOL>( *symbol ),
|
|
|
|
std::make_unique<SCH_SCREEN>() );
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
}
|
2017-11-12 18:55:20 +01:00
|
|
|
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-05-31 13:37:58 -07:00
|
|
|
bool SYMBOL_LIBRARY_MANAGER::UpdateLibraryBuffer( const wxString& aLibrary )
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
m_libs.erase( aLibrary );
|
|
|
|
getLibraryBuffer( aLibrary );
|
|
|
|
}
|
|
|
|
catch(const std::exception& e)
|
|
|
|
{
|
|
|
|
wxLogError( _( "Error updating library buffer: %s" ), e.what() );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
catch( const IO_ERROR& e )
|
|
|
|
{
|
|
|
|
wxLogError( _( "Error updating library buffer: %s" ), e.What() );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
wxLogError( _( "Error updating library buffer." ) );
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
getLibraryBuffer( aLibrary );
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-09-17 19:29:17 +01:00
|
|
|
SYMBOL_BUFFER::SYMBOL_BUFFER( std::unique_ptr<LIB_SYMBOL> aSymbol,
|
|
|
|
std::unique_ptr<SCH_SCREEN> aScreen ) :
|
|
|
|
m_screen( std::move( aScreen ) ), m_symbol( std::move( aSymbol ) )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-09-17 19:29:17 +01:00
|
|
|
wxASSERT( m_symbol );
|
|
|
|
m_original = std::make_unique<LIB_SYMBOL>( *m_symbol );
|
2024-09-17 19:53:56 +01:00
|
|
|
wxASSERT( m_original );
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
SYMBOL_BUFFER::~SYMBOL_BUFFER()
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-09-17 19:29:17 +01:00
|
|
|
void SYMBOL_BUFFER::SetSymbol( std::unique_ptr<LIB_SYMBOL> aSymbol )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2021-06-15 08:31:28 -04:00
|
|
|
wxCHECK( m_symbol != aSymbol, /* void */ );
|
2021-06-10 14:51:46 -04:00
|
|
|
wxASSERT( aSymbol );
|
2024-09-17 19:29:17 +01:00
|
|
|
m_symbol = std::move( aSymbol );
|
2018-11-21 19:50:26 +00:00
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
// If the symbol moves libraries then the original moves with it
|
|
|
|
if( m_original->GetLibId().GetLibNickname() != m_symbol->GetLibId().GetLibNickname() )
|
2018-11-21 19:50:26 +00:00
|
|
|
{
|
2021-06-15 08:31:28 -04:00
|
|
|
m_original->SetLibId( LIB_ID( m_symbol->GetLibId().GetLibNickname(),
|
2018-11-21 19:50:26 +00:00
|
|
|
m_original->GetLibId().GetLibItemName() ) );
|
|
|
|
}
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-09-17 19:29:17 +01:00
|
|
|
void SYMBOL_BUFFER::SetOriginal( std::unique_ptr<LIB_SYMBOL> aSymbol )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2021-06-10 14:51:46 -04:00
|
|
|
wxCHECK( m_original != aSymbol, /* void */ );
|
|
|
|
wxASSERT( aSymbol );
|
2024-09-17 19:29:17 +01:00
|
|
|
m_original = std::move( aSymbol );
|
2018-11-21 19:50:26 +00:00
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
// The original is not allowed to have a different library than its symbol
|
|
|
|
if( m_original->GetLibId().GetLibNickname() != m_symbol->GetLibId().GetLibNickname() )
|
2018-11-21 19:50:26 +00:00
|
|
|
{
|
2021-06-15 08:31:28 -04:00
|
|
|
m_original->SetLibId( LIB_ID( m_symbol->GetLibId().GetLibNickname(),
|
2018-11-21 19:50:26 +00:00
|
|
|
m_original->GetLibId().GetLibItemName() ) );
|
|
|
|
}
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
bool SYMBOL_BUFFER::IsModified() const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2021-05-28 12:07:04 -07:00
|
|
|
return m_screen && m_screen->IsContentModified();
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
LIB_SYMBOL* LIB_BUFFER::GetSymbol( const wxString& aAlias ) const
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
|
|
|
auto buf = GetBuffer( aAlias );
|
|
|
|
|
|
|
|
if( !buf )
|
|
|
|
return nullptr;
|
|
|
|
|
2024-09-17 19:53:56 +01:00
|
|
|
LIB_SYMBOL& symbol = buf->GetSymbol();
|
|
|
|
return &symbol;
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-09-17 19:29:17 +01:00
|
|
|
bool LIB_BUFFER::CreateBuffer( std::unique_ptr<LIB_SYMBOL> aCopy,
|
|
|
|
std::unique_ptr<SCH_SCREEN> aScreen )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2019-11-06 14:15:42 -05:00
|
|
|
wxASSERT( aCopy );
|
2017-11-12 18:55:20 +01:00
|
|
|
wxASSERT( aCopy->GetLib() == nullptr );
|
|
|
|
|
|
|
|
// Set the parent library name,
|
|
|
|
// otherwise it is empty as no library has been given as the owner during object construction
|
2019-11-06 14:15:42 -05:00
|
|
|
LIB_ID libId = aCopy->GetLibId();
|
2017-11-12 18:55:20 +01:00
|
|
|
libId.SetLibNickname( m_libName );
|
2019-11-06 14:15:42 -05:00
|
|
|
aCopy->SetLibId( libId );
|
2024-09-17 19:29:17 +01:00
|
|
|
|
|
|
|
auto symbolBuf = std::make_shared<SYMBOL_BUFFER>( std::move( aCopy ), std::move( aScreen ) );
|
|
|
|
m_symbols.push_back( symbolBuf );
|
|
|
|
|
2017-11-12 18:55:20 +01:00
|
|
|
++m_hash;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-09-18 22:53:46 +01:00
|
|
|
bool LIB_BUFFER::UpdateBuffer( SYMBOL_BUFFER& aSymbolBuf, const LIB_SYMBOL& aCopy )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
LIB_SYMBOL& bufferedSymbol = aSymbolBuf.GetSymbol();
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2024-09-17 19:53:56 +01:00
|
|
|
bufferedSymbol = aCopy;
|
2017-11-12 18:55:20 +01:00
|
|
|
++m_hash;
|
|
|
|
|
2019-12-26 08:00:59 -05:00
|
|
|
return true;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-09-18 22:53:46 +01:00
|
|
|
bool LIB_BUFFER::DeleteBuffer( const SYMBOL_BUFFER& aSymbolBuf )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-09-18 22:53:46 +01:00
|
|
|
const auto sameBufferPredicate = [&]( const std::shared_ptr<SYMBOL_BUFFER>& aBuf )
|
|
|
|
{
|
|
|
|
return aBuf.get() == &aSymbolBuf;
|
|
|
|
};
|
|
|
|
|
|
|
|
auto symbolBufIt = std::find_if( m_symbols.begin(), m_symbols.end(), sameBufferPredicate );
|
2021-06-15 08:31:28 -04:00
|
|
|
wxCHECK( symbolBufIt != m_symbols.end(), false );
|
2019-11-06 14:15:42 -05:00
|
|
|
|
|
|
|
bool retv = true;
|
|
|
|
|
|
|
|
// Remove all derived symbols to prevent broken inheritance.
|
2024-09-17 19:53:56 +01:00
|
|
|
if( HasDerivedSymbols( aSymbolBuf.GetSymbol().GetName() )
|
2024-09-18 22:53:46 +01:00
|
|
|
&& ( removeChildSymbols( aSymbolBuf ) == 0 ) )
|
2020-09-26 12:40:32 +01:00
|
|
|
{
|
2019-12-12 14:36:29 -05:00
|
|
|
retv = false;
|
2020-09-26 12:40:32 +01:00
|
|
|
}
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
m_deleted.emplace_back( *symbolBufIt );
|
|
|
|
m_symbols.erase( symbolBufIt );
|
2017-11-12 18:55:20 +01:00
|
|
|
++m_hash;
|
|
|
|
|
2019-11-06 14:15:42 -05:00
|
|
|
return retv;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-09-17 20:23:07 +01:00
|
|
|
bool LIB_BUFFER::SaveBuffer( SYMBOL_BUFFER& aSymbolBuf, const wxString& aFileName, SCH_IO* aPlugin,
|
|
|
|
bool aBuffer )
|
2017-11-13 14:58:39 +01:00
|
|
|
{
|
2021-03-13 08:28:52 -05:00
|
|
|
wxCHECK( !aFileName.IsEmpty(), false );
|
2017-11-13 14:58:39 +01:00
|
|
|
|
2021-06-26 20:21:30 +01:00
|
|
|
wxString errorMsg = _( "Error saving symbol %s to library '%s'." ) + wxS( "\n%s" );
|
2020-01-09 11:23:46 -05:00
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
// Set properties to prevent saving the file on every symbol save.
|
2024-08-19 11:47:02 -07:00
|
|
|
std::map<std::string, UTF8> properties;
|
2023-12-24 00:31:24 +00:00
|
|
|
properties.emplace( SCH_IO_KICAD_LEGACY::PropBuffering, "" );
|
2017-11-13 14:58:39 +01:00
|
|
|
|
2024-09-17 20:23:07 +01:00
|
|
|
LIB_SYMBOL& libSymbol = aSymbolBuf.GetSymbol();
|
2023-10-09 14:58:49 -04:00
|
|
|
|
2021-03-02 19:19:08 -05:00
|
|
|
{
|
2024-09-17 20:23:07 +01:00
|
|
|
LIB_SYMBOL& originalSymbol = aSymbolBuf.GetOriginal();
|
2024-09-17 19:53:56 +01:00
|
|
|
const wxString originalName = originalSymbol.GetName();
|
2024-09-17 19:29:17 +01:00
|
|
|
|
|
|
|
// Delete the original symbol if the symbol name has been changed.
|
2024-09-17 19:53:56 +01:00
|
|
|
if( libSymbol.GetName() != originalSymbol.GetName() )
|
2022-05-06 16:12:25 -07:00
|
|
|
{
|
2024-09-17 19:29:17 +01:00
|
|
|
try
|
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
if( aPlugin->LoadSymbol( aFileName, originalName ) )
|
|
|
|
aPlugin->DeleteSymbol( aFileName, originalName, &properties );
|
2024-09-17 19:29:17 +01:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& ioe )
|
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
wxLogError( errorMsg, UnescapeString( originalName ), aFileName, ioe.What() );
|
2024-09-17 19:29:17 +01:00
|
|
|
return false;
|
|
|
|
}
|
2022-05-06 16:12:25 -07:00
|
|
|
}
|
2021-03-02 19:19:08 -05:00
|
|
|
}
|
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
LIB_SYMBOL* parentSymbol = nullptr;
|
|
|
|
|
2024-09-17 19:53:56 +01:00
|
|
|
if( libSymbol.IsAlias() )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
LIB_SYMBOL* newCachedSymbol = new LIB_SYMBOL( libSymbol );
|
|
|
|
std::shared_ptr<LIB_SYMBOL> bufferedParent = libSymbol.GetParent().lock();
|
2023-10-09 14:58:49 -04:00
|
|
|
parentSymbol = newCachedSymbol;
|
2019-11-06 14:15:42 -05:00
|
|
|
|
|
|
|
wxCHECK( bufferedParent, false );
|
|
|
|
|
2021-06-10 14:51:46 -04:00
|
|
|
LIB_SYMBOL* cachedParent = nullptr;
|
2020-01-09 11:23:46 -05:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2021-03-13 08:28:52 -05:00
|
|
|
cachedParent = aPlugin->LoadSymbol( aFileName, bufferedParent->GetName() );
|
2020-01-09 11:23:46 -05:00
|
|
|
}
|
2020-02-29 22:53:51 +01:00
|
|
|
catch( const IO_ERROR& )
|
2020-01-09 11:23:46 -05:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2019-11-06 14:15:42 -05:00
|
|
|
|
|
|
|
if( !cachedParent )
|
|
|
|
{
|
2021-06-10 14:51:46 -04:00
|
|
|
cachedParent = new LIB_SYMBOL( *bufferedParent.get() );
|
2021-06-15 08:31:28 -04:00
|
|
|
newCachedSymbol->SetParent( cachedParent );
|
2020-01-09 11:23:46 -05:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2021-03-13 08:28:52 -05:00
|
|
|
aPlugin->SaveSymbol( aFileName, cachedParent, aBuffer ? &properties : nullptr );
|
2020-01-09 11:23:46 -05:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& ioe )
|
|
|
|
{
|
2021-06-30 11:53:04 +01:00
|
|
|
wxLogError( errorMsg, UnescapeString( cachedParent->GetName() ), aFileName,
|
|
|
|
ioe.What() );
|
2020-01-09 11:23:46 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2021-06-15 08:31:28 -04:00
|
|
|
aPlugin->SaveSymbol( aFileName, newCachedSymbol, aBuffer ? &properties : nullptr );
|
2020-01-09 11:23:46 -05:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& ioe )
|
|
|
|
{
|
2021-06-30 11:53:04 +01:00
|
|
|
wxLogError( errorMsg, UnescapeString( newCachedSymbol->GetName() ), aFileName,
|
|
|
|
ioe.What() );
|
2020-01-09 11:23:46 -05:00
|
|
|
return false;
|
|
|
|
}
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2024-09-17 19:29:17 +01:00
|
|
|
auto originalParent = std::make_unique<LIB_SYMBOL>( *bufferedParent.get() );
|
|
|
|
LIB_SYMBOL& parentRef = *originalParent;
|
2024-09-17 20:23:07 +01:00
|
|
|
aSymbolBuf.SetOriginal( std::move( originalParent ) );
|
2024-09-17 19:29:17 +01:00
|
|
|
|
2024-09-17 19:53:56 +01:00
|
|
|
auto newSymbol = std::make_unique<LIB_SYMBOL>( libSymbol );
|
2024-09-17 19:29:17 +01:00
|
|
|
newSymbol->SetParent( &parentRef );
|
2024-09-17 20:23:07 +01:00
|
|
|
aSymbolBuf.SetOriginal( std::move( newSymbol ) );
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2021-06-15 08:31:28 -04:00
|
|
|
newCachedSymbol->SetParent( cachedParent );
|
2020-01-09 11:23:46 -05:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2021-06-15 08:31:28 -04:00
|
|
|
aPlugin->SaveSymbol( aFileName, newCachedSymbol, aBuffer ? &properties : nullptr );
|
2020-01-09 11:23:46 -05:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& ioe )
|
|
|
|
{
|
2021-06-30 11:53:04 +01:00
|
|
|
wxLogError( errorMsg, UnescapeString( newCachedSymbol->GetName() ), aFileName,
|
|
|
|
ioe.What() );
|
2020-01-09 11:23:46 -05:00
|
|
|
return false;
|
|
|
|
}
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2021-07-28 21:04:53 +01:00
|
|
|
auto originalBufferedParent = GetBuffer( bufferedParent->GetName() );
|
2019-11-06 14:15:42 -05:00
|
|
|
wxCHECK( originalBufferedParent, false );
|
2024-09-17 19:29:17 +01:00
|
|
|
|
2024-09-17 19:53:56 +01:00
|
|
|
auto newSymbol = std::make_unique<LIB_SYMBOL>( libSymbol );
|
|
|
|
newSymbol->SetParent( &originalBufferedParent->GetSymbol() );
|
2024-09-17 20:23:07 +01:00
|
|
|
aSymbolBuf.SetOriginal( std::move( newSymbol ) );
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
}
|
2020-01-09 11:23:46 -05:00
|
|
|
else
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
parentSymbol = new LIB_SYMBOL( libSymbol );
|
2020-01-09 11:23:46 -05:00
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
try
|
2020-01-09 11:23:46 -05:00
|
|
|
{
|
2023-10-09 14:58:49 -04:00
|
|
|
aPlugin->SaveSymbol( aFileName, parentSymbol, aBuffer ? &properties : nullptr );
|
2020-01-09 11:23:46 -05:00
|
|
|
}
|
2023-10-09 14:58:49 -04:00
|
|
|
catch( const IO_ERROR& ioe )
|
2020-01-09 11:23:46 -05:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
wxLogError( errorMsg, UnescapeString( libSymbol.GetName() ), aFileName, ioe.What() );
|
2023-10-09 14:58:49 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-09-17 20:23:07 +01:00
|
|
|
aSymbolBuf.SetOriginal( std::make_unique<LIB_SYMBOL>( libSymbol ) );
|
2023-10-09 14:58:49 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
wxArrayString derivedSymbols;
|
|
|
|
|
|
|
|
// Reparent all symbols derived from the saved symbol.
|
2024-09-17 19:53:56 +01:00
|
|
|
if( GetDerivedSymbolNames( libSymbol.GetName(), derivedSymbols ) != 0 )
|
2023-10-09 14:58:49 -04:00
|
|
|
{
|
|
|
|
// Save the derived symbols.
|
|
|
|
for( const wxString& entry : derivedSymbols )
|
|
|
|
{
|
|
|
|
std::shared_ptr<SYMBOL_BUFFER> symbol = GetBuffer( entry );
|
|
|
|
|
|
|
|
wxCHECK2( symbol, continue );
|
|
|
|
|
2024-09-17 19:53:56 +01:00
|
|
|
LIB_SYMBOL* derivedSymbol = new LIB_SYMBOL( symbol->GetSymbol() );
|
2023-10-09 14:58:49 -04:00
|
|
|
derivedSymbol->SetParent( parentSymbol );
|
2020-01-09 11:23:46 -05:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2023-10-09 14:58:49 -04:00
|
|
|
aPlugin->SaveSymbol( aFileName, new LIB_SYMBOL( *derivedSymbol ),
|
|
|
|
aBuffer ? &properties : nullptr );
|
2020-01-09 11:23:46 -05:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& ioe )
|
|
|
|
{
|
2023-10-09 14:58:49 -04:00
|
|
|
wxLogError( errorMsg, UnescapeString( derivedSymbol->GetName() ), aFileName,
|
2021-06-30 11:53:04 +01:00
|
|
|
ioe.What() );
|
2020-01-09 11:23:46 -05:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
|
2017-11-13 14:58:39 +01:00
|
|
|
++m_hash;
|
|
|
|
return true;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
std::shared_ptr<SYMBOL_BUFFER> LIB_BUFFER::GetBuffer( const wxString& aAlias ) const
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2023-10-09 14:58:49 -04:00
|
|
|
for( std::shared_ptr<SYMBOL_BUFFER> entry : m_symbols )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
if( entry->GetSymbol().GetName() == aAlias )
|
2019-11-06 14:15:42 -05:00
|
|
|
return entry;
|
|
|
|
}
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2021-07-28 21:04:53 +01:00
|
|
|
return std::shared_ptr<SYMBOL_BUFFER>( nullptr );
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
bool LIB_BUFFER::HasDerivedSymbols( const wxString& aParentName ) const
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2023-10-09 14:58:49 -04:00
|
|
|
for( const std::shared_ptr<SYMBOL_BUFFER>& entry : m_symbols )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
if( entry->GetSymbol().IsAlias() )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
LIB_SYMBOL_SPTR parent = entry->GetSymbol().GetParent().lock();
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
// Check for inherited symbol without a valid parent.
|
2019-11-06 14:15:42 -05:00
|
|
|
wxCHECK( parent, false );
|
|
|
|
|
|
|
|
if( parent->GetName() == aParentName )
|
|
|
|
return true;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-06 14:15:42 -05:00
|
|
|
return false;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
void LIB_BUFFER::GetSymbolNames( wxArrayString& aSymbolNames, SYMBOL_NAME_FILTER aFilter )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2023-10-09 14:58:49 -04:00
|
|
|
for( std::shared_ptr<SYMBOL_BUFFER>& entry : m_symbols )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
const LIB_SYMBOL& symbol = entry->GetSymbol();
|
|
|
|
if( ( symbol.IsAlias() && ( aFilter == SYMBOL_NAME_FILTER::ROOT_ONLY ) )
|
|
|
|
|| ( symbol.IsRoot() && ( aFilter == SYMBOL_NAME_FILTER::DERIVED_ONLY ) ) )
|
|
|
|
{
|
2019-11-06 14:15:42 -05:00
|
|
|
continue;
|
2024-09-17 19:53:56 +01:00
|
|
|
}
|
|
|
|
aSymbolNames.Add( UnescapeString( symbol.GetName() ) );
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
size_t LIB_BUFFER::GetDerivedSymbolNames( const wxString& aSymbolName, wxArrayString& aList )
|
2020-01-09 11:23:46 -05:00
|
|
|
{
|
|
|
|
wxCHECK( !aSymbolName.IsEmpty(), 0 );
|
|
|
|
|
2023-10-09 14:58:49 -04:00
|
|
|
for( std::shared_ptr<SYMBOL_BUFFER>& entry : m_symbols )
|
2020-01-09 11:23:46 -05:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
const LIB_SYMBOL& symbol = entry->GetSymbol();
|
|
|
|
if( symbol.IsAlias() )
|
2020-01-09 11:23:46 -05:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
LIB_SYMBOL_SPTR parent = symbol.GetParent().lock();
|
2020-01-09 11:23:46 -05:00
|
|
|
|
2021-06-15 08:31:28 -04:00
|
|
|
// Check for inherited symbol without a valid parent.
|
2023-08-21 11:13:50 -04:00
|
|
|
wxCHECK2( parent, continue );
|
2020-01-09 11:23:46 -05:00
|
|
|
|
|
|
|
if( parent->GetName() == aSymbolName )
|
2023-08-21 11:13:50 -04:00
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
aList.Add( symbol.GetName() );
|
2023-08-21 11:13:50 -04:00
|
|
|
|
2024-09-17 19:53:56 +01:00
|
|
|
GetDerivedSymbolNames( symbol.GetName(), aList );
|
2023-08-21 11:13:50 -04:00
|
|
|
}
|
2020-01-09 11:23:46 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return aList.GetCount();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-09-18 22:53:46 +01:00
|
|
|
int LIB_BUFFER::removeChildSymbols( const SYMBOL_BUFFER& aSymbolBuf )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2024-06-16 11:41:01 +01:00
|
|
|
int cnt = 0;
|
|
|
|
wxArrayString derivedSymbolNames;
|
|
|
|
std::deque<std::shared_ptr<SYMBOL_BUFFER>>::iterator it;
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2024-09-17 19:53:56 +01:00
|
|
|
if( GetDerivedSymbolNames( aSymbolBuf.GetSymbol().GetName(), derivedSymbolNames ) )
|
2017-11-12 18:55:20 +01:00
|
|
|
{
|
2023-08-21 11:13:50 -04:00
|
|
|
for( const wxString& symbolName : derivedSymbolNames )
|
2019-11-06 14:15:42 -05:00
|
|
|
{
|
2023-08-21 11:13:50 -04:00
|
|
|
it = std::find_if( m_symbols.begin(), m_symbols.end(),
|
|
|
|
[symbolName]( std::shared_ptr<SYMBOL_BUFFER>& buf )
|
|
|
|
{
|
2024-09-17 19:53:56 +01:00
|
|
|
return buf->GetSymbol().GetName() == symbolName;
|
2023-08-21 11:13:50 -04:00
|
|
|
} );
|
2017-11-12 18:55:20 +01:00
|
|
|
|
2023-08-21 11:13:50 -04:00
|
|
|
wxCHECK2( it != m_symbols.end(), continue );
|
2019-11-06 14:15:42 -05:00
|
|
|
|
2023-08-21 11:13:50 -04:00
|
|
|
m_deleted.emplace_back( *it );
|
|
|
|
m_symbols.erase( it );
|
|
|
|
cnt += 1;
|
2019-11-06 14:15:42 -05:00
|
|
|
}
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|
|
|
|
|
2019-11-06 14:15:42 -05:00
|
|
|
return cnt;
|
2017-11-12 18:55:20 +01:00
|
|
|
}
|