KiCad PCB EDA Suite
Loading...
Searching...
No Matches
footprint_library_adapter.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
5 * @author Jon Evans <[email protected]>
6 *
7 * This program is free software: you can redistribute it and/or modify it
8 * under the terms of the GNU General Public License as published by the
9 * Free Software Foundation, either version 3 of the License, or (at your
10 * option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program. If not, see <https://www.gnu.org/licenses/>.
19 */
20
22
23#include <chrono>
24#include <env_vars.h>
25#include <footprint_info_impl.h>
26#include <thread_pool.h>
27#include <trace_helpers.h>
28#include <footprint.h>
29
30#include <magic_enum.hpp>
31#include <wx/hash.h>
32#include <wx/log.h>
33
34using namespace std::chrono_literals;
35
36
38
40
42
44
45
50
51
56
57
59{
60 return ENV_VAR::GetVersionedEnvVarName( wxS( "FOOTPRINT_DIR" ) );
61}
62
63
64void FOOTPRINT_LIBRARY_ADAPTER::enumerateLibrary( LIB_DATA* aLib, const wxString& aUri )
65{
66 wxArrayString namesAS;
67 std::map<std::string, UTF8> options = aLib->row->GetOptionsMap();
68 PCB_IO* plugin = pcbplugin( aLib );
69 wxString nickname = aLib->row->Nickname();
70
71 // FootprintEnumerate populates the plugin's internal FP_CACHE with parsed footprints
72 plugin->FootprintEnumerate( namesAS, aUri, false, &options );
73
74 std::vector<std::unique_ptr<FOOTPRINT>> footprints;
75 footprints.reserve( namesAS.size() );
76
77 // For plugins with internal caches (like kicad_sexpr), GetEnumeratedFootprint returns
78 // a borrowed pointer and ClearCachedFootprints handles cleanup. For other plugins,
79 // GetEnumeratedFootprint allocates new memory that we must delete after cloning.
80 const bool pluginCaches = plugin->CachesEnumeratedFootprints();
81
82 for( const wxString& footprintName : namesAS )
83 {
84 try
85 {
86 const FOOTPRINT* cached = plugin->GetEnumeratedFootprint( aUri, footprintName, &options );
87
88 if( !cached )
89 continue;
90
91 FOOTPRINT* footprint = static_cast<FOOTPRINT*>( cached->Duplicate( IGNORE_PARENT_GROUP ) );
92 footprint->SetParent( nullptr );
93
94 // For non-caching plugins, delete the allocated footprint now that we've cloned it
95 if( !pluginCaches )
96 delete cached;
97
98 LIB_ID id = footprint->GetFPID();
99 id.SetLibNickname( nickname );
100 footprint->SetFPID( id );
101 footprints.emplace_back( footprint );
102 }
103 catch( IO_ERROR& e )
104 {
105 wxLogTrace( traceLibraries, "FP: %s:%s enumeration error: %s",
106 nickname, footprintName, e.What() );
107 }
108 }
109
110 // GetLibraryTimestamp() reads the filesystem, so do it before taking the lock.
111 long long timestamp = plugin->GetLibraryTimestamp( aUri );
112
113 {
114 std::unique_lock lock( PreloadedFootprintsMutex );
115 PreloadedFootprints.Get()[nickname] = std::move( footprints );
116 m_preloadedTimestamps[nickname] = timestamp;
117 }
118
119 // Clear the plugin's FP_CACHE now that we've copied footprints to PreloadedFootprints.
120 // This eliminates the double-caching that was consuming ~1.2GB of extra RAM.
121 plugin->ClearCachedFootprints( aUri );
122}
123
124
125std::optional<LIB_STATUS> FOOTPRINT_LIBRARY_ADAPTER::LoadOne( LIB_DATA* aLib )
126{
128
129 std::map<std::string, UTF8> options = aLib->row->GetOptionsMap();
130
131 try
132 {
133 wxArrayString dummyList;
134 pcbplugin( aLib )->FootprintEnumerate( dummyList, getUri( aLib->row ), false, &options );
136 }
137 catch( IO_ERROR& e )
138 {
140 aLib->status.error = LIBRARY_ERROR( { e.What() } );
141 wxLogTrace( traceLibraries, "FP: %s: plugin threw exception: %s", aLib->row->Nickname(), e.What() );
142 }
143
144 return aLib->status;
145}
146
147
148std::optional<LIB_STATUS> FOOTPRINT_LIBRARY_ADAPTER::LoadOne( const wxString& nickname )
149{
151
152 if( result.has_value() )
153 return LoadOne( *result );
154
155 return LIB_STATUS{
156 .load_status = LOAD_STATUS::LOAD_ERROR,
157 .error = LIBRARY_ERROR( { result.error() } )
158 };
159}
160
161
162std::vector<FOOTPRINT*> FOOTPRINT_LIBRARY_ADAPTER::GetFootprints( const wxString& aNickname, bool aBestEfforts )
163{
164 std::vector<FOOTPRINT*> footprints;
165
166 std::shared_lock lock( PreloadedFootprintsMutex );
167 auto it = PreloadedFootprints.Get().find( aNickname );
168
169 if( it == PreloadedFootprints.Get().end() )
170 return footprints;
171
172 footprints.reserve( it->second.size() );
173
174 for( const auto& fp : it->second )
175 footprints.push_back( fp.get() );
176
177 return footprints;
178}
179
180
181
182std::vector<wxString> FOOTPRINT_LIBRARY_ADAPTER::GetFootprintNames( const wxString& aNickname, bool aBestEfforts )
183{
184 // TODO(JE) can we kill wxArrayString in internal API?
185 wxArrayString namesAS;
186 std::vector<wxString> names;
187
188 if( std::optional<const LIB_DATA*> maybeLib = fetchIfLoaded( aNickname ) )
189 {
190 const LIB_DATA* lib = *maybeLib;
191 std::map<std::string, UTF8> options = lib->row->GetOptionsMap();
192
193 try
194 {
195 pcbplugin( lib )->FootprintEnumerate( namesAS, getUri( lib->row ), true, &options );
196 }
197 catch( IO_ERROR& e )
198 {
199 wxLogTrace( traceLibraries, "FP: Exception enumerating library %s: %s", lib->row->Nickname(), e.What() );
200 }
201 }
202
203 for( const wxString& name : namesAS )
204 names.emplace_back( name );
205
206 return names;
207}
208
209
210long long FOOTPRINT_LIBRARY_ADAPTER::GenerateTimestamp( const wxString* aNickname )
211{
212 long long hash = 0;
213
214 if( aNickname )
215 {
216 wxCHECK( HasLibrary( *aNickname, true ), hash );
217
218 if( std::optional<const LIB_DATA*> r = fetchIfLoaded( *aNickname ); r.has_value() )
219 {
220 PCB_IO* plugin = dynamic_cast<PCB_IO*>( ( *r )->plugin.get() );
221 wxCHECK( plugin, hash );
222 return plugin->GetLibraryTimestamp( LIBRARY_MANAGER::GetFullURI( ( *r )->row, true ) )
223 + wxHashTable::MakeKey( *aNickname );
224 }
225 }
226
227 for( const wxString& nickname : GetLibraryNames() )
228 {
229 if( std::optional<const LIB_DATA*> r = fetchIfLoaded( nickname ); r.has_value() )
230 {
231 wxCHECK2( ( *r )->plugin->IsPCB_IO(), continue );
232 PCB_IO* plugin = static_cast<PCB_IO*>( ( *r )->plugin.get() );
233 hash += plugin->GetLibraryTimestamp( LIBRARY_MANAGER::GetFullURI( ( *r )->row, true ) )
234 + wxHashTable::MakeKey( nickname );
235 }
236 }
237
238 return hash;
239}
240
241
243{
244 std::optional<LIB_DATA*> maybeLib = fetchIfLoaded( aNickname );
245
246 if( !maybeLib )
247 return;
248
249 LIB_DATA* lib = *maybeLib;
250 PCB_IO* plugin = dynamic_cast<PCB_IO*>( lib->plugin.get() );
251
252 if( !plugin )
253 return;
254
255 wxString uri = getUri( lib->row );
256 long long currentTimestamp = plugin->GetLibraryTimestamp( uri );
257
258 {
259 std::shared_lock lock( PreloadedFootprintsMutex );
260 auto tsIt = m_preloadedTimestamps.find( aNickname );
261
262 if( tsIt != m_preloadedTimestamps.end() && tsIt->second == currentTimestamp )
263 return;
264
265 wxLogTrace( traceLibraries, "FP: %s changed on disk, re-enumerating", aNickname );
266 }
267
268 enumerateLibrary( lib, uri );
269}
270
271
272bool FOOTPRINT_LIBRARY_ADAPTER::FootprintExists( const wxString& aNickname, const wxString& aName )
273{
274 if( std::optional<const LIB_DATA*> maybeLib = fetchIfLoaded( aNickname ) )
275 {
276 const LIB_DATA* lib = *maybeLib;
277 std::map<std::string, UTF8> options = lib->row->GetOptionsMap();
278 return pcbplugin( lib )->FootprintExists( getUri( lib->row ), aName, &options );
279 }
280
281 return false;
282}
283
284
285FOOTPRINT* FOOTPRINT_LIBRARY_ADAPTER::LoadFootprint( const wxString& aNickname, const wxString& aName, bool aKeepUUID )
286{
287 // First check if the footprint is in PreloadedFootprints and clone from there.
288 // This avoids re-parsing the file and keeps FP_CACHE from being repopulated.
289 {
290 std::shared_lock lock( PreloadedFootprintsMutex );
291 auto libIt = PreloadedFootprints.Get().find( aNickname );
292
293 if( libIt != PreloadedFootprints.Get().end() )
294 {
295 for( const auto& fp : libIt->second )
296 {
297 if( fp->GetFPID().GetLibItemName() == UTF8( aName ) )
298 {
300
301 if( aKeepUUID )
302 copy = static_cast<FOOTPRINT*>( fp->Clone() );
303 else
304 copy = static_cast<FOOTPRINT*>( fp->Duplicate( IGNORE_PARENT_GROUP ) );
305
306 copy->SetParent( nullptr );
307 return copy;
308 }
309 }
310 }
311 }
312
313 // Footprint not found in PreloadedFootprints, fall back to plugin.
314 // This re-parses the file but is needed for footprints not yet enumerated.
315 if( std::optional<const LIB_DATA*> lib = fetchIfLoaded( aNickname ) )
316 {
317 try
318 {
319 if( FOOTPRINT* footprint = pcbplugin( *lib )->FootprintLoad( getUri( ( *lib )->row ), aName, aKeepUUID ) )
320 {
321 LIB_ID id = footprint->GetFPID();
322 id.SetLibNickname( ( *lib )->row->Nickname() );
323 footprint->SetFPID( id );
324 return footprint;
325 }
326 }
327 catch( const IO_ERROR& ioe )
328 {
329 wxLogTrace( traceLibraries, "LoadFootprint: error loading %s:%s: %s", aNickname, aName, ioe.What() );
330 }
331 }
332 else
333 {
334 wxLogTrace( traceLibraries, "LoadFootprint: requested library %s not loaded", aNickname );
335 }
336
337 return nullptr;
338}
339
340
342{
343 wxString nickname = aFootprintId.GetLibNickname();
344 wxString footprintName = aFootprintId.GetLibItemName();
345
346 if( nickname.size() )
347 return LoadFootprint( nickname, footprintName, aKeepUUID );
348
349 // nickname is empty, sequentially search (alphabetically) all libs/nicks for first match:
350 for( const wxString& library : GetLibraryNames() )
351 {
352 // FootprintLoad() returns NULL on not found, does not throw exception
353 // unless there's an IO_ERROR.
354 if( FOOTPRINT* ret = LoadFootprint( library, footprintName, aKeepUUID ) )
355 return ret;
356 }
357
358 return nullptr;
359}
360
361
363 const FOOTPRINT* aFootprint,
364 bool aOverwrite )
365{
366 wxCHECK( aFootprint, SAVE_SKIPPED );
367
368 if( std::optional<const LIB_DATA*> lib = fetchIfLoaded( aNickname ) )
369 {
370 if( !aOverwrite )
371 {
372 wxString fpname = aFootprint->GetFPID().GetLibItemName();
373
374 try
375 {
376 FOOTPRINT* existing = pcbplugin( *lib )->FootprintLoad( getUri( ( *lib )->row ), fpname, false );
377
378 if( existing )
379 {
380 delete existing;
381 return SAVE_SKIPPED;
382 }
383 }
384 catch( IO_ERROR& e )
385 {
386 wxLogTrace( traceLibraries, "SaveFootprint: error checking for existing footprint %s: %s",
387 aFootprint->GetFPIDAsString(), e.What() );
388 return SAVE_SKIPPED;
389 }
390 }
391
392 try
393 {
394 pcbplugin( *lib )->FootprintSave( getUri( ( *lib )->row ), aFootprint );
395 }
396 catch( IO_ERROR& e )
397 {
398 wxLogTrace( traceLibraries, "SaveFootprint: error saving %s: %s",
399 aFootprint->GetFPIDAsString(), e.What() );
400 return SAVE_SKIPPED;
401 }
402
403 {
404 std::unique_lock lock( PreloadedFootprintsMutex );
405 auto it = PreloadedFootprints.Get().find( aNickname );
406
407 if( it != PreloadedFootprints.Get().end() )
408 {
409 wxString fpName = aFootprint->GetFPID().GetLibItemName();
410
411 if( aOverwrite )
412 {
413 auto& footprints = it->second;
414 footprints.erase( std::remove_if( footprints.begin(), footprints.end(),
415 [&fpName]( const std::unique_ptr<FOOTPRINT>& fp )
416 {
417 return fp->GetFPID().GetLibItemName().wx_str() == fpName;
418 } ),
419 footprints.end() );
420 }
421
422 FOOTPRINT* clone = static_cast<FOOTPRINT*>( aFootprint->Duplicate( IGNORE_PARENT_GROUP ) );
423 clone->SetParent( nullptr );
424
425 LIB_ID id = clone->GetFPID();
426 id.SetLibNickname( aNickname );
427 clone->SetFPID( id );
428
429 it->second.emplace_back( clone );
430 }
431 }
432
433 return SAVE_OK;
434 }
435 else
436 {
437 wxLogTrace( traceLibraries, "SaveFootprint: requested library %s not loaded", aNickname );
438 return SAVE_SKIPPED;
439 }
440}
441
442
443void FOOTPRINT_LIBRARY_ADAPTER::DeleteFootprint( const wxString& aNickname, const wxString& aFootprintName )
444{
445 if( std::optional<const LIB_DATA*> lib = fetchIfLoaded( aNickname ) )
446 {
447 try
448 {
449 pcbplugin( *lib )->FootprintDelete( getUri( ( *lib )->row ), aFootprintName );
450 }
451 catch( IO_ERROR& e )
452 {
453 wxLogTrace( traceLibraries, "DeleteFootprint: error deleting %s:%s: %s", aNickname,
454 aFootprintName, e.What() );
455 return;
456 }
457
458 {
459 std::unique_lock lock( PreloadedFootprintsMutex );
460 auto it = PreloadedFootprints.Get().find( aNickname );
461
462 if( it != PreloadedFootprints.Get().end() )
463 {
464 auto& footprints = it->second;
465 footprints.erase( std::remove_if( footprints.begin(), footprints.end(),
466 [&aFootprintName]( const std::unique_ptr<FOOTPRINT>& fp )
467 {
468 return fp->GetFPID().GetLibItemName().wx_str() == aFootprintName;
469 } ),
470 footprints.end() );
471 }
472 }
473 }
474 else
475 {
476 wxLogTrace( traceLibraries, "DeleteFootprint: requested library %s not loaded", aNickname );
477 }
478}
479
480
482{
483 // Route through fetchIfLoaded() so LOAD_ERROR sentinel entries, which carry a null
484 // plugin, are filtered out instead of dereferenced.
485 if( std::optional<const LIB_DATA*> lib = fetchIfLoaded( aLib ) )
486 return ( *lib )->plugin->IsLibraryWritable( getUri( ( *lib )->row ) );
487
488 return false;
489}
490
491
493{
495
496 if( type == PCB_IO_MGR::NESTED_TABLE )
497 {
498 wxString msg;
499 wxFileName fileName( m_manager.GetFullURI( row, true ) );
500
501 if( fileName.FileExists() )
502 return tl::unexpected( LIBRARY_TABLE_OK() );
503 else
504 msg = wxString::Format( _( "Nested table '%s' not found." ), row->URI() );
505
506 return tl::unexpected( LIBRARY_ERROR( msg ) );
507 }
508 else if( type == PCB_IO_MGR::PCB_FILE_UNKNOWN )
509 {
510 wxLogTrace( traceLibraries, "FP: Plugin type %s is unknown!", row->Type() );
511 wxString msg = wxString::Format( _( "Unknown library type %s " ), row->Type() );
512 return tl::unexpected( LIBRARY_ERROR( msg ) );
513 }
514
516 wxCHECK( plugin, tl::unexpected( LIBRARY_ERROR( _( "Internal error" ) ) ) );
517
518 wxLogTrace( traceLibraries, "FP: Library %s (%s) plugin created", row->Nickname(),
519 magic_enum::enum_name( row->Scope() ) );
520
521 return plugin;
522}
523
524
526{
527 // Note: can't use dynamic_cast across compile units on Mac
528 wxCHECK( aRow->plugin->IsPCB_IO(), nullptr );
529 PCB_IO* ret = static_cast<PCB_IO*>( aRow->plugin.get() );
530 return ret;
531}
const char * name
virtual void SetParent(EDA_ITEM *aParent)
Definition eda_item.cpp:89
bool IsFootprintLibWritable(const wxString &aNickname)
Return true if the library given by aNickname is writable.
void DeleteFootprint(const wxString &aNickname, const wxString &aFootprintName)
Deletes the aFootprintName from the library given by aNickname.
std::vector< FOOTPRINT * > GetFootprints(const wxString &aNickname, bool aBestEfforts=false)
Retrieves a list of footprints contained in a given loaded library.
void RefreshLibraryIfChanged(const wxString &aNickname)
Checks whether the library on disk has changed since it was last enumerated into PreloadedFootprints ...
static LEAK_AT_EXIT< std::shared_mutex > GlobalLibraryMutex
void enumerateLibrary(LIB_DATA *aLib, const wxString &aUri) override
Override in derived class to perform library-specific enumeration.
IO_BASE * plugin(const LIB_DATA *aRow) override
SAVE_T SaveFootprint(const wxString &aNickname, const FOOTPRINT *aFootprint, bool aOverwrite=true)
Write aFootprint to an existing library given by aNickname.
std::vector< wxString > GetFootprintNames(const wxString &aNickname, bool aBestEfforts=false)
Retrieves a list of footprint names contained in a given loaded library.
FOOTPRINT * LoadFootprint(const wxString &aNickname, const wxString &aName, bool aKeepUUID)
Load a FOOTPRINT having aName from the library given by aNickname.
static LEAK_AT_EXIT< std::map< wxString, LIB_DATA > > GlobalLibraries
static PCB_IO * pcbplugin(const LIB_DATA *aRow)
FOOTPRINT * LoadFootprintWithOptionalNickname(const LIB_ID &aFootprintId, bool aKeepUUID)
Load a footprint having aFootprintId with possibly an empty nickname.
std::map< wxString, long long > m_preloadedTimestamps
Per-library filesystem timestamps recorded when PreloadedFootprints was last populated.
LIBRARY_RESULT< IO_BASE * > createPlugin(const LIBRARY_TABLE_ROW *row) override
Creates a concrete plugin for the given row.
FOOTPRINT_LIBRARY_ADAPTER(LIBRARY_MANAGER &aManager)
long long GenerateTimestamp(const wxString *aNickname)
Generates a filesystem timestamp / hash value for library(ies)
std::optional< LIB_STATUS > LoadOne(LIB_DATA *aLib) override
Loads or reloads the given library, if it exists.
SAVE_T
The set of return values from SaveSymbol() below.
static LEAK_AT_EXIT< std::map< wxString, std::vector< std::unique_ptr< FOOTPRINT > > > > PreloadedFootprints
Storage for preloaded footprints, indexed by library nickname.
bool FootprintExists(const wxString &aNickname, const wxString &aName)
static std::shared_mutex PreloadedFootprintsMutex
void SetFPID(const LIB_ID &aFPID)
Definition footprint.h:442
wxString GetFPIDAsString() const
Definition footprint.h:447
const LIB_ID & GetFPID() const
Definition footprint.h:441
BOARD_ITEM * Duplicate(bool addToParentGroup, BOARD_COMMIT *aCommit=nullptr) const override
Create a copy of this BOARD_ITEM.
virtual bool IsPCB_IO() const
Work-around for lack of dynamic_cast across compile units on Mac.
Definition io_base.h:84
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
virtual const wxString What() const
A composite of Problem() and Where()
A wrapper for static data that should not be destroyed at program exit.
void evictOwnedGlobalEntries()
Erases this adapter's own entries (LIB_DATA::global_owner == this) from the process-wide globalLibs()...
LIBRARY_MANAGER_ADAPTER(LIBRARY_MANAGER &aManager)
Constructs a type-specific adapter into the library manager.
LIBRARY_RESULT< LIB_DATA * > loadIfNeeded(const wxString &aNickname)
Fetches a loaded library, triggering a load of that library if it isn't loaded yet.
bool HasLibrary(const wxString &aNickname, bool aCheckEnabled=false) const
Test for the existence of aNickname in the library tables.
std::vector< wxString > GetLibraryNames() const
Returns a list of library nicknames that are available (skips any that failed to load)
static wxString getUri(const LIBRARY_TABLE_ROW *aRow)
LIBRARY_MANAGER & m_manager
std::optional< const LIB_DATA * > fetchIfLoaded(const wxString &aNickname) const
std::optional< wxString > GetFullURI(LIBRARY_TABLE_TYPE aType, const wxString &aNickname, bool aSubstituted=false)
Return the full location specifying URI for the LIB, either in original UI form or in environment var...
LIBRARY_TABLE_SCOPE Scope() const
std::map< std::string, UTF8 > GetOptionsMap() const
const wxString & Type() const
const wxString & URI() const
const wxString & Nickname() const
A logical library item identifier and consists of various portions much like a URI.
Definition lib_id.h:45
int SetLibNickname(const UTF8 &aLibNickname)
Override the logical library name portion of the LIB_ID to aLibNickname.
Definition lib_id.cpp:96
const UTF8 & GetLibItemName() const
Definition lib_id.h:98
const UTF8 & GetLibNickname() const
Return the logical library name portion of a LIB_ID.
Definition lib_id.h:83
static PCB_FILE_T EnumFromStr(const wxString &aFileType)
Return the PCB_FILE_T from the corresponding plugin type name: "kicad", "legacy", etc.
PCB_FILE_T
The set of file types that the PCB_IO_MGR knows about, and for which there has been a plugin written,...
Definition pcb_io_mgr.h:52
@ PCB_FILE_UNKNOWN
0 is not a legal menu id on Mac
Definition pcb_io_mgr.h:53
static PCB_IO * FindPlugin(PCB_FILE_T aFileType)
Return a #PLUGIN which the caller can use to import, export, save, or load design documents.
A base class that BOARD loading and saving plugins should derive from.
Definition pcb_io.h:75
virtual void FootprintEnumerate(wxArrayString &aFootprintNames, const wxString &aLibraryPath, bool aBestEfforts, const std::map< std::string, UTF8 > *aProperties=nullptr)
Return a list of footprint names contained within the library at aLibraryPath.
Definition pcb_io.cpp:91
virtual bool FootprintExists(const wxString &aLibraryPath, const wxString &aFootprintName, const std::map< std::string, UTF8 > *aProperties=nullptr)
Check for the existence of a footprint.
Definition pcb_io.cpp:132
virtual void FootprintDelete(const wxString &aLibraryPath, const wxString &aFootprintName, const std::map< std::string, UTF8 > *aProperties=nullptr)
Delete aFootprintName from the library at aLibraryPath.
Definition pcb_io.cpp:156
virtual FOOTPRINT * FootprintLoad(const wxString &aLibraryPath, const wxString &aFootprintName, bool aKeepUUID=false, const std::map< std::string, UTF8 > *aProperties=nullptr)
Load a footprint having aFootprintName from the aLibraryPath containing a library format that this PC...
Definition pcb_io.cpp:140
virtual void FootprintSave(const wxString &aLibraryPath, const FOOTPRINT *aFootprint, const std::map< std::string, UTF8 > *aProperties=nullptr)
Write aFootprint to an existing library located at aLibraryPath.
Definition pcb_io.cpp:148
An 8 bit string that is assuredly encoded in UTF8, and supplies special conversion support to and fro...
Definition utf8.h:67
#define _(s)
#define IGNORE_PARENT_GROUP
Definition eda_item.h:53
Functions related to environment variables, including help functions.
const wxChar *const traceLibraries
Flag to enable library table and library manager tracing.
tl::expected< ResultType, LIBRARY_ERROR > LIBRARY_RESULT
KICOMMON_API wxString GetVersionedEnvVarName(const wxString &aBaseName)
Construct a versioned environment variable based on this KiCad major version.
Definition env_vars.cpp:77
Storage for an actual loaded library (including library content owned by the plugin)
LIB_STATUS status
std::unique_ptr< IO_BASE > plugin
const LIBRARY_TABLE_ROW * row
The overall status of a loaded or loading library.
std::optional< LIBRARY_ERROR > error
LOAD_STATUS load_status
wxString result
Test unit parsing edge cases and error handling.
wxLogTrace helper definitions.