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
27#include <kicommon.h>
29#include <io/io_base.h>
30
31
32class LIBRARY_MANAGER;
33class PROJECT;
34
35
44
47{
49 std::optional<LIBRARY_ERROR> error;
50};
51
52
55{
56 std::unique_ptr<IO_BASE> plugin;
57 const LIBRARY_TABLE_ROW* row = nullptr;
58 std::mutex mutex;
60
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
132 virtual void AsyncLoad() = 0;
133
135 std::optional<float> AsyncLoadProgress() const;
136
137 void BlockUntilLoaded();
138
139 bool IsLibraryLoaded( const wxString& aNickname );
140
142 virtual std::optional<LIB_STATUS> GetLibraryStatus( const wxString& aNickname ) const = 0;
143
145 std::vector<std::pair<wxString, LIB_STATUS>> GetLibraryStatuses() const;
146
148 virtual bool IsWritable( const wxString& aNickname ) const;
149
151 bool CreateLibrary( const wxString& aNickname );
152
153 virtual bool SupportsConfigurationDialog( const wxString& aNickname ) const { return false; }
154
155 virtual void ShowConfigurationDialog( const wxString& aNickname, wxWindow* aParent ) const {};
156
157 virtual std::optional<LIBRARY_ERROR> LibraryError( const wxString& aNickname ) const { return std::nullopt; };
158
159protected:
160 virtual std::map<wxString, LIB_DATA>& globalLibs() = 0;
161 virtual std::map<wxString, LIB_DATA>& globalLibs() const = 0;
162 virtual std::mutex& globalLibsMutex() = 0;
163
164 static wxString getUri( const LIBRARY_TABLE_ROW* aRow );
165
166 std::optional<const LIB_DATA*> fetchIfLoaded( const wxString& aNickname ) const;
167
168 std::optional<LIB_DATA*> fetchIfLoaded( const wxString& aNickname );
169
171 LIBRARY_RESULT<LIB_DATA*> loadIfNeeded( const wxString& aNickname );
172
174 void abortLoad();
175
178
179 virtual IO_BASE* plugin( const LIB_DATA* aRow ) = 0;
180
182
183 // The actual library content is held in an associated IO plugin
184 // TODO(JE) should this be an expected<LIB_ROW> so we can store the
185 // error result if a lib can't be loaded instead of retrying the load every time
186 // content is requested?
187 std::map<wxString, LIB_DATA> m_libraries;
188
190
191 std::atomic_bool m_abort;
192 std::vector<std::future<void>> m_futures;
193
194 std::atomic<size_t> m_loadCount;
196};
197
198
200{
201public:
203
205
208
209 static wxString DefaultGlobalTablePath( LIBRARY_TABLE_TYPE aType );
210
211 static bool IsTableValid( const wxString& aPath );
212
214 static bool GlobalTablesValid();
215
217 static std::vector<LIBRARY_TABLE_TYPE> InvalidGlobalTables();
218
219 static bool CreateGlobalTable( LIBRARY_TABLE_TYPE aType, bool aPopulateDefaultLibraries );
220
222 void LoadGlobalTables( std::initializer_list<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
223
225 void ProjectChanged();
226
227 void RegisterAdapter( LIBRARY_TABLE_TYPE aType,
228 std::unique_ptr<LIBRARY_MANAGER_ADAPTER>&& aAdapter );
229
230 std::optional<LIBRARY_MANAGER_ADAPTER*> Adapter( LIBRARY_TABLE_TYPE aType ) const;
231
240 std::optional<LIBRARY_TABLE*> Table( LIBRARY_TABLE_TYPE aType,
241 LIBRARY_TABLE_SCOPE aScope );
242
250 std::vector<LIBRARY_TABLE_ROW*> Rows( LIBRARY_TABLE_TYPE aType,
252 bool aIncludeInvalid = false ) const;
253
261 std::optional<LIBRARY_TABLE_ROW*> GetRow( LIBRARY_TABLE_TYPE aType,
262 const wxString &aNickname,
263 LIBRARY_TABLE_SCOPE aScope =
265
266 std::optional<LIBRARY_TABLE_ROW*> FindRowByURI( LIBRARY_TABLE_TYPE aType,
267 const wxString &aUri,
268 LIBRARY_TABLE_SCOPE aScope =
270
271 void LoadProjectTables( const wxString& aProjectPath );
272
282 std::optional<wxString> GetFullURI( LIBRARY_TABLE_TYPE aType, const wxString& aNickname,
283 bool aSubstituted = false ) const;
284
285 static wxString GetFullURI( const LIBRARY_TABLE_ROW* aRow, bool aSubstituted = false );
286
287 static wxString ExpandURI( const wxString& aShortURI, const PROJECT& aProject );
288
289 static bool UrisAreEquivalent( const wxString& aURI1, const wxString& aURI2 );
290
291private:
292 void loadTables( const wxString& aTablePath, LIBRARY_TABLE_SCOPE aScope,
293 std::vector<LIBRARY_TABLE_TYPE> aTablesToLoad = {} );
294
295 void loadNestedTables( LIBRARY_TABLE& aTable );
296
297 static wxString tableFileName( LIBRARY_TABLE_TYPE aType );
298
299 void createEmptyTable( LIBRARY_TABLE_TYPE aType, LIBRARY_TABLE_SCOPE aScope );
300
301 std::map<LIBRARY_TABLE_TYPE, std::unique_ptr<LIBRARY_TABLE>> m_tables;
302
304 std::map<wxString, std::unique_ptr<LIBRARY_TABLE>> m_childTables;
305
306 // TODO: support multiple projects
307 std::map<LIBRARY_TABLE_TYPE, std::unique_ptr<LIBRARY_TABLE>> m_projectTables;
308
309 std::map<LIBRARY_TABLE_TYPE, std::unique_ptr<LIBRARY_MANAGER_ADAPTER>> m_adapters;
310
311 mutable std::mutex m_adaptersMutex;
312};
313
314#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.
std::optional< LIBRARY_TABLE * > ProjectTable() const
Retrieves the project library table for this adapter type, or nullopt if one doesn't exist.
LIBRARY_TABLE * GlobalTable() const
Retrieves the global library table for this adapter type.
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 std::mutex & globalLibsMutex()=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.
std::optional< wxString > FindLibraryByURI(const wxString &aURI) const
virtual std::optional< LIBRARY_ERROR > LibraryError(const wxString &aNickname) const
virtual std::optional< LIB_STATUS > GetLibraryStatus(const wxString &aNickname) const =0
Returns the status of a loaded library, or nullopt if the library hasn't been loaded (yet)
LIBRARY_MANAGER & Manager() const
virtual void AsyncLoad()=0
Loads all available libraries for this adapter type in the background.
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)
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)
virtual std::map< wxString, LIB_DATA > & globalLibs() const =0
std::atomic< size_t > m_loadCount
LIBRARY_MANAGER & m_manager
std::optional< const LIB_DATA * > fetchIfLoaded(const wxString &aNickname) const
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::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
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:65
void ProjectChanged() override
#define KICOMMON_API
Definition kicommon.h:28
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
std::mutex mutex
const LIBRARY_TABLE_ROW * row
The overall status of a loaded or loading library.
std::optional< LIBRARY_ERROR > error
LOAD_STATUS load_status