KiCad PCB EDA Suite
Loading...
Searching...
No Matches
library_manager.h
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 along
18 * with this program. If not, see <http://www.gnu.org/licenses/>.
19 */
20
21#ifndef LIBRARY_MANAGER_H
22#define LIBRARY_MANAGER_H
23
24#include <future>
25#include <memory>
26#include <shared_mutex>
27
28#include <kicommon.h>
30#include <io/io_base.h>
31
32
33class LIBRARY_MANAGER;
34class PROJECT;
35
36
45
48{
50 std::optional<LIBRARY_ERROR> error;
51};
52
53
56{
57 std::unique_ptr<IO_BASE> plugin;
58 const LIBRARY_TABLE_ROW* row = nullptr;
60
61 int modify_hash = -1;
62 std::vector<wxString> available_fields_cache;
63};
64
65
71{
72public:
81
83
84 LIBRARY_MANAGER& Manager() const;
85
87 virtual LIBRARY_TABLE_TYPE Type() const = 0;
88
91
93 std::optional<LIBRARY_TABLE*> ProjectTable() const;
94
95 std::optional<wxString> FindLibraryByURI( const wxString& aURI ) const;
96
98 std::vector<wxString> GetLibraryNames() const;
99
106 bool HasLibrary( const wxString& aNickname, bool aCheckEnabled = false ) const;
107
109 bool DeleteLibrary( const wxString& aNickname );
110
111 std::optional<wxString> GetLibraryDescription( const wxString& aNickname ) const;
112
114 std::vector<LIBRARY_TABLE_ROW*> Rows( LIBRARY_TABLE_SCOPE aScope = LIBRARY_TABLE_SCOPE::BOTH,
115 bool aIncludeInvalid = false ) const;
116
118 std::optional<LIBRARY_TABLE_ROW*> GetRow( const wxString& aNickname,
120
122 std::optional<LIBRARY_TABLE_ROW*> FindRowByURI( const wxString& aUri,
124
126 virtual void ProjectChanged();
127
129 void GlobalTablesChanged( std::initializer_list<LIBRARY_TABLE_TYPE> aChangedTables = {} );
130
137 void ProjectTablesChanged( std::initializer_list<LIBRARY_TABLE_TYPE> aChangedTables = {} );
138
143 void ProjectTablesReloaded( std::initializer_list<LIBRARY_TABLE_TYPE> aChangedTables = {} );
144
145 void CheckTableRow( LIBRARY_TABLE_ROW& aRow );
146
148 void AsyncLoad();
149
150 virtual std::optional<LIB_STATUS> LoadOne( LIB_DATA* aLib ) = 0;
151
153 std::optional<float> AsyncLoadProgress() const;
154
155 void BlockUntilLoaded();
156
159 void AbortAsyncLoad();
160
161 bool IsLibraryLoaded( const wxString& aNickname );
162
164 std::optional<LIB_STATUS> GetLibraryStatus( const wxString& aNickname ) const;
165
167 std::vector<std::pair<wxString, LIB_STATUS>> GetLibraryStatuses() const;
168
170 wxString GetLibraryLoadErrors() const;
171
172 void ReloadLibraryEntry( const wxString& aNickname,
174
177 std::optional<LIB_STATUS> LoadLibraryEntry( const wxString& aNickname );
178
180 virtual bool IsWritable( const wxString& aNickname ) const;
181
183 bool CreateLibrary( const wxString& aNickname );
184
185 virtual bool SupportsConfigurationDialog( const wxString& aNickname ) const { return false; }
186
187 virtual void ShowConfigurationDialog( const wxString& aNickname, wxWindow* aParent ) const {};
188
189 virtual std::optional<LIBRARY_ERROR> LibraryError( const wxString& aNickname ) const;
190
191protected:
192 virtual std::map<wxString, LIB_DATA>& globalLibs() = 0;
193 virtual std::map<wxString, LIB_DATA>& globalLibs() const = 0;
194 virtual std::shared_mutex& globalLibsMutex() = 0;
195 virtual std::shared_mutex& globalLibsMutex() const = 0;
196
200 virtual void enumerateLibrary( LIB_DATA* aLib, const wxString& aUri ) = 0;
201
202 static wxString getUri( const LIBRARY_TABLE_ROW* aRow );
203
204 std::optional<const LIB_DATA*> fetchIfLoaded( const wxString& aNickname ) const;
205
206 std::optional<LIB_DATA*> fetchIfLoaded( const wxString& aNickname );
207
209 LIBRARY_RESULT<LIB_DATA*> loadIfNeeded( const wxString& aNickname );
210
211 LIBRARY_RESULT<LIB_DATA*> loadFromScope( const wxString& aNickname,
212 LIBRARY_TABLE_SCOPE aScope,
213 std::map<wxString, LIB_DATA>& aTarget,
214 std::shared_mutex& aMutex );
215
217 void abortLoad();
218
226 void resetProjectCache();
227
230
231 virtual IO_BASE* plugin( const LIB_DATA* aRow ) = 0;
232
234
235 // The actual library content is held in an associated IO plugin
236 std::map<wxString, LIB_DATA> m_libraries;
237
238 mutable std::shared_mutex m_librariesMutex;
239
240 std::atomic_bool m_abort;
241 std::vector<std::future<void>> m_futures;
242
243 std::atomic<size_t> m_loadCount{ 0 };
244 std::atomic<size_t> m_loadTotal{ 0 };
245 std::mutex m_loadMutex;
246};
247
248
250{
251public:
253
255
258
259 static wxString DefaultGlobalTablePath( LIBRARY_TABLE_TYPE aType );
260
261 static wxString StockTablePath( LIBRARY_TABLE_TYPE aType );
262
263 static bool IsTableValid( const wxString& aPath );
264
266 static bool GlobalTablesValid();
267
269 static std::vector<LIBRARY_TABLE_TYPE> InvalidGlobalTables();
270
271 static bool CreateGlobalTable( LIBRARY_TABLE_TYPE aType, bool aPopulateDefaultLibraries );
272
274 void LoadGlobalTables( std::initializer_list<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
275
277 void LoadProjectTables( std::initializer_list<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
278
279 void ReloadTables( LIBRARY_TABLE_SCOPE aScope, std::initializer_list<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
280
282 void ProjectChanged();
283
287 void AbortAsyncLoads();
288
289 void RegisterAdapter( LIBRARY_TABLE_TYPE aType,
290 std::unique_ptr<LIBRARY_MANAGER_ADAPTER>&& aAdapter );
291
292 bool RemoveAdapter( LIBRARY_TABLE_TYPE aType, LIBRARY_MANAGER_ADAPTER* aAdapter );
293
294 std::optional<LIBRARY_MANAGER_ADAPTER*> Adapter( LIBRARY_TABLE_TYPE aType ) const;
295
304 std::optional<LIBRARY_TABLE*> Table( LIBRARY_TABLE_TYPE aType,
305 LIBRARY_TABLE_SCOPE aScope );
306
314 std::vector<LIBRARY_TABLE_ROW*> Rows( LIBRARY_TABLE_TYPE aType,
316 bool aIncludeInvalid = false ) const;
317
325 std::optional<LIBRARY_TABLE_ROW*> GetRow( LIBRARY_TABLE_TYPE aType, const wxString &aNickname,
327
328 std::optional<LIBRARY_TABLE_ROW*> FindRowByURI( LIBRARY_TABLE_TYPE aType, const wxString &aUri,
330
331 void ReloadLibraryEntry( LIBRARY_TABLE_TYPE aType, const wxString& aNickname,
333
336 std::optional<LIB_STATUS> LoadLibraryEntry( LIBRARY_TABLE_TYPE aType,
337 const wxString& aNickname );
338
339 void LoadProjectTables( const wxString& aProjectPath,
340 std::initializer_list<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
341
351 std::optional<wxString> GetFullURI( LIBRARY_TABLE_TYPE aType, const wxString& aNickname,
352 bool aSubstituted = false );
353
354 static wxString GetFullURI( const LIBRARY_TABLE_ROW* aRow, bool aSubstituted = false );
355
356 static wxString ExpandURI( const wxString& aShortURI, const PROJECT& aProject );
357
358 static bool UrisAreEquivalent( const wxString& aURI1, const wxString& aURI2 );
359
369 static bool IsPcmManagedRow( const LIBRARY_TABLE_ROW& aRow );
370
371private:
372 void loadTables( const wxString& aTablePath, LIBRARY_TABLE_SCOPE aScope,
373 std::vector<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
374
375 void loadNestedTables( LIBRARY_TABLE& aTable );
376
377 static wxString tableFileName( LIBRARY_TABLE_TYPE aType );
378
379 void createEmptyTable( LIBRARY_TABLE_TYPE aType, LIBRARY_TABLE_SCOPE aScope );
380
381 std::map<LIBRARY_TABLE_TYPE, std::unique_ptr<LIBRARY_TABLE>> m_tables;
382
384 std::map<wxString, std::unique_ptr<LIBRARY_TABLE>> m_childTables;
385
386 // TODO: support multiple projects
387 std::map<LIBRARY_TABLE_TYPE, std::unique_ptr<LIBRARY_TABLE>> m_projectTables;
388
389 std::map<LIBRARY_TABLE_TYPE, std::unique_ptr<LIBRARY_MANAGER_ADAPTER>> m_adapters;
390
391 mutable std::mutex m_adaptersMutex;
392
393 typedef std::tuple<LIBRARY_TABLE_TYPE, LIBRARY_TABLE_SCOPE, wxString> ROW_CACHE_KEY;
394
395 std::map<ROW_CACHE_KEY, LIBRARY_TABLE_ROW*> m_rowCache;
396 mutable std::mutex m_rowCacheMutex;
397};
398
399#endif //LIBRARY_MANAGER_H
std::optional< float > AsyncLoadProgress() const
Returns async load progress between 0.0 and 1.0, or nullopt if load is not in progress.
virtual std::optional< LIB_STATUS > LoadOne(LIB_DATA *aLib)=0
void resetProjectCache()
Aborts pending loads and resets every project-scope cache entry in place (plugin and row cleared,...
void ReloadLibraryEntry(const wxString &aNickname, LIBRARY_TABLE_SCOPE aScope=LIBRARY_TABLE_SCOPE::BOTH)
virtual std::shared_mutex & globalLibsMutex() const =0
std::optional< LIB_STATUS > GetLibraryStatus(const wxString &aNickname) const
Returns the status of a loaded library, or nullopt if the library hasn't been loaded (yet)
std::optional< LIBRARY_TABLE * > ProjectTable() const
Retrieves the project library table for this adapter type, or nullopt if one doesn't exist.
void AbortAsyncLoad()
Aborts any async load in progress; blocks until fully done aborting.
virtual void enumerateLibrary(LIB_DATA *aLib, const wxString &aUri)=0
Override in derived class to perform library-specific enumeration.
LIBRARY_TABLE * GlobalTable() const
Retrieves the global library table for this adapter type.
virtual std::shared_mutex & globalLibsMutex()=0
LIBRARY_MANAGER_ADAPTER(LIBRARY_MANAGER &aManager)
Constructs a type-specific adapter into the library manager.
bool IsLibraryLoaded(const wxString &aNickname)
std::vector< LIBRARY_TABLE_ROW * > Rows(LIBRARY_TABLE_SCOPE aScope=LIBRARY_TABLE_SCOPE::BOTH, bool aIncludeInvalid=false) const
Like LIBRARY_MANAGER::Rows but filtered to the LIBRARY_TABLE_TYPE of this adapter.
std::optional< LIBRARY_TABLE_ROW * > FindRowByURI(const wxString &aUri, LIBRARY_TABLE_SCOPE aScope=LIBRARY_TABLE_SCOPE::BOTH) const
Like LIBRARY_MANAGER::FindRowByURI but filtered to the LIBRARY_TABLE_TYPE of this adapter.
virtual std::map< wxString, LIB_DATA > & globalLibs()=0
virtual LIBRARY_TABLE_TYPE Type() const =0
The type of library table this adapter works with.
bool DeleteLibrary(const wxString &aNickname)
Deletes the given library from disk if it exists; returns true if deleted.
LIBRARY_RESULT< LIB_DATA * > loadIfNeeded(const wxString &aNickname)
Fetches a loaded library, triggering a load of that library if it isn't loaded yet.
wxString GetLibraryLoadErrors() const
Returns all library load errors as newline-separated strings for display.
std::optional< wxString > FindLibraryByURI(const wxString &aURI) const
std::shared_mutex m_librariesMutex
LIBRARY_MANAGER & Manager() const
void abortLoad()
Aborts any async load in progress; blocks until fully done aborting.
std::optional< wxString > GetLibraryDescription(const wxString &aNickname) const
virtual LIBRARY_RESULT< IO_BASE * > createPlugin(const LIBRARY_TABLE_ROW *row)=0
Creates a concrete plugin for the given row.
void GlobalTablesChanged(std::initializer_list< LIBRARY_TABLE_TYPE > aChangedTables={})
Notify the adapter that the global library tables have changed.
bool HasLibrary(const wxString &aNickname, bool aCheckEnabled=false) const
Test for the existence of aNickname in the library tables.
std::optional< LIBRARY_TABLE_ROW * > GetRow(const wxString &aNickname, LIBRARY_TABLE_SCOPE aScope=LIBRARY_TABLE_SCOPE::BOTH) const
Like LIBRARY_MANAGER::GetRow but filtered to the LIBRARY_TABLE_TYPE of this adapter.
virtual void ShowConfigurationDialog(const wxString &aNickname, wxWindow *aParent) const
std::map< wxString, LIB_DATA > m_libraries
virtual IO_BASE * plugin(const LIB_DATA *aRow)=0
std::vector< wxString > GetLibraryNames() const
Returns a list of library nicknames that are available (skips any that failed to load)
LIBRARY_RESULT< LIB_DATA * > loadFromScope(const wxString &aNickname, LIBRARY_TABLE_SCOPE aScope, std::map< wxString, LIB_DATA > &aTarget, std::shared_mutex &aMutex)
std::vector< std::future< void > > m_futures
virtual bool SupportsConfigurationDialog(const wxString &aNickname) const
static wxString getUri(const LIBRARY_TABLE_ROW *aRow)
bool CreateLibrary(const wxString &aNickname)
Creates the library (i.e. saves to disk) for the given row if it exists.
virtual bool IsWritable(const wxString &aNickname) const
Return true if the given nickname exists and is not a read-only library.
std::vector< std::pair< wxString, LIB_STATUS > > GetLibraryStatuses() const
Returns a list of all library nicknames and their status (even if they failed to load)
std::atomic< size_t > m_loadTotal
virtual std::map< wxString, LIB_DATA > & globalLibs() const =0
std::atomic< size_t > m_loadCount
LIBRARY_MANAGER & m_manager
std::optional< LIB_STATUS > LoadLibraryEntry(const wxString &aNickname)
Synchronously loads the named library to LOADED state.
std::optional< const LIB_DATA * > fetchIfLoaded(const wxString &aNickname) const
std::mutex m_rowCacheMutex
std::map< wxString, std::unique_ptr< LIBRARY_TABLE > > m_childTables
Map of full URI to table object for tables that are referenced by global or project tables.
std::map< ROW_CACHE_KEY, LIBRARY_TABLE_ROW * > m_rowCache
std::mutex m_adaptersMutex
static wxString DefaultGlobalTablePath(LIBRARY_TABLE_TYPE aType)
static std::vector< LIBRARY_TABLE_TYPE > InvalidGlobalTables()
static bool GlobalTablesValid()
std::map< LIBRARY_TABLE_TYPE, std::unique_ptr< LIBRARY_TABLE > > m_projectTables
LIBRARY_MANAGER(const LIBRARY_MANAGER &)=delete
static bool CreateGlobalTable(LIBRARY_TABLE_TYPE aType, bool aPopulateDefaultLibraries)
void LoadGlobalTables(std::initializer_list< LIBRARY_TABLE_TYPE > aTablesToLoad={})
(Re)loads the global library tables in the given list, or all tables if no list is given
std::map< LIBRARY_TABLE_TYPE, std::unique_ptr< LIBRARY_TABLE > > m_tables
LIBRARY_MANAGER & operator=(const LIBRARY_MANAGER &)=delete
std::tuple< LIBRARY_TABLE_TYPE, LIBRARY_TABLE_SCOPE, wxString > ROW_CACHE_KEY
static wxString StockTablePath(LIBRARY_TABLE_TYPE aType)
static bool IsTableValid(const wxString &aPath)
std::map< LIBRARY_TABLE_TYPE, std::unique_ptr< LIBRARY_MANAGER_ADAPTER > > m_adapters
Container for project specific data.
Definition project.h:66
void ProjectChanged() override
#define KICOMMON_API
Definition kicommon.h:27
LOAD_STATUS
Status of a library load managed by a library adapter.
tl::expected< ResultType, LIBRARY_ERROR > LIBRARY_RESULT
LIBRARY_TABLE_TYPE
LIBRARY_TABLE_SCOPE
Storage for an actual loaded library (including library content owned by the plugin)
std::vector< wxString > available_fields_cache
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