KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
settings_manager.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 (C) 2020 Jon Evans <jon@craftyjon.com>
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
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
22#include <regex>
23#include <wx/debug.h>
24#include <wx/dir.h>
25#include <wx/filename.h>
26#include <wx/snglinst.h>
27#include <wx/stdpaths.h>
28#include <wx/utils.h>
29
30#include <build_version.h>
31#include <confirm.h>
33#include <gestfich.h>
35#include <kiplatform/io.h>
36#include <kiway.h>
37#include <lockfile.h>
38#include <macros.h>
39#include <pgm_base.h>
40#include <paths.h>
41#include <project.h>
50#include <env_vars.h>
51
52
54 m_headless( aHeadless ),
55 m_kiway( nullptr ),
56 m_common_settings( nullptr ),
57 m_migration_source(),
58 m_migrateLibraryTables( true )
59{
60 // Check if the settings directory already exists, and if not, perform a migration if possible
61 if( !MigrateIfNeeded() )
62 {
63 m_ok = false;
64 return;
65 }
66
67 m_ok = true;
68
69 // create the common settings shared by all applications. Not loaded immediately
71
72 // Create the built-in color settings
73 // Here to allow the Python API to access the built-in colors
75
76 wxFileName commonSettings( GetPathForSettingsFile( m_common_settings ),
78
79 if( !wxFileExists( commonSettings.GetFullPath() ) )
80 {
83 }
84}
85
86
88{
89 for( std::unique_ptr<PROJECT>& project : m_projects_list )
90 project.reset();
91
92 m_projects.clear();
93
94 for( std::unique_ptr<JSON_SETTINGS>& settings : m_settings )
95 settings.reset();
96
97 m_settings.clear();
98
99 m_color_settings.clear();
100}
101
102
104{
105 std::unique_ptr<JSON_SETTINGS> ptr( aSettings );
106
107 ptr->SetManager( this );
108
109 wxLogTrace( traceSettings, wxT( "Registered new settings object <%s>" ),
110 ptr->GetFullFilename() );
111
112 if( aLoadNow )
113 ptr->LoadFromFile( GetPathForSettingsFile( ptr.get() ) );
114
115 m_settings.push_back( std::move( ptr ) );
116 return m_settings.back().get();
117}
118
119
121{
122 // TODO(JE) We should check for dirty settings here and write them if so, because
123 // Load() could be called late in the application lifecycle
124 std::vector<JSON_SETTINGS*> toLoad;
125
126 // Cache a copy of raw pointers; m_settings may be modified during the load loop
127 std::transform( m_settings.begin(), m_settings.end(), std::back_inserter( toLoad ),
128 []( std::unique_ptr<JSON_SETTINGS>& aSettings )
129 {
130 return aSettings.get();
131 } );
132
133 for( JSON_SETTINGS* settings : toLoad )
134 settings->LoadFromFile( GetPathForSettingsFile( settings ) );
135}
136
137
139{
140 auto it = std::find_if( m_settings.begin(), m_settings.end(),
141 [&aSettings]( const std::unique_ptr<JSON_SETTINGS>& aPtr )
142 {
143 return aPtr.get() == aSettings;
144 } );
145
146 if( it != m_settings.end() )
147 ( *it )->LoadFromFile( GetPathForSettingsFile( it->get() ) );
148}
149
150
152{
153 for( auto&& settings : m_settings )
154 {
155 // Never automatically save color settings, caller should use SaveColorSettings
156 if( dynamic_cast<COLOR_SETTINGS*>( settings.get() ) )
157 continue;
158
159 // Never automatically save project settings, caller should use SaveProject or UnloadProject
160 if( dynamic_cast<PROJECT_FILE*>( settings.get() )
161 || dynamic_cast<PROJECT_LOCAL_SETTINGS*>( settings.get() ) )
162 {
163 continue;
164 }
165
166 settings->SaveToFile( GetPathForSettingsFile( settings.get() ) );
167 }
168}
169
170
172{
173 auto it = std::find_if( m_settings.begin(), m_settings.end(),
174 [&aSettings]( const std::unique_ptr<JSON_SETTINGS>& aPtr )
175 {
176 return aPtr.get() == aSettings;
177 } );
178
179 if( it != m_settings.end() )
180 {
181 wxLogTrace( traceSettings, wxT( "Saving %s" ), ( *it )->GetFullFilename() );
182 ( *it )->SaveToFile( GetPathForSettingsFile( it->get() ) );
183 }
184}
185
186
188{
189 auto it = std::find_if( m_settings.begin(), m_settings.end(),
190 [&aSettings]( const std::unique_ptr<JSON_SETTINGS>& aPtr )
191 {
192 return aPtr.get() == aSettings;
193 } );
194
195 if( it != m_settings.end() )
196 {
197 wxLogTrace( traceSettings, wxT( "Flush and release %s" ), ( *it )->GetFullFilename() );
198
199 if( aSave )
200 ( *it )->SaveToFile( GetPathForSettingsFile( it->get() ) );
201
202 JSON_SETTINGS* tmp = it->get(); // We use a temporary to suppress a Clang warning
203 size_t typeHash = typeid( *tmp ).hash_code();
204
205 if( m_app_settings_cache.count( typeHash ) )
206 m_app_settings_cache.erase( typeHash );
207
208 m_settings.erase( it );
209 }
210}
211
212
214{
215 // Find settings the fast way
216 if( m_color_settings.count( aName ) )
217 return m_color_settings.at( aName );
218
219 // Maybe it's the display name (cli is one method of invoke)
220 auto it = std::find_if( m_color_settings.begin(), m_color_settings.end(),
221 [&aName]( const std::pair<wxString, COLOR_SETTINGS*>& p )
222 {
223 return p.second->GetName().Lower() == aName.Lower();
224 } );
225
226 if( it != m_color_settings.end() )
227 {
228 return it->second;
229 }
230
231 // No match? See if we can load it
232 if( !aName.empty() )
233 {
235
236 if( !ret )
237 {
238 ret = registerColorSettings( aName );
240 ret->SetFilename( wxT( "user" ) );
241 ret->SetReadOnly( false );
242 }
243
244 return ret;
245 }
246
247 // This had better work
249}
250
251
253{
254 wxLogTrace( traceSettings, wxT( "Attempting to load color theme %s" ), aName );
255
256 wxFileName fn( GetColorSettingsPath(), aName, wxS( "json" ) );
257
258 if( !fn.IsOk() || !fn.Exists() )
259 {
260 wxLogTrace( traceSettings, wxT( "Theme file %s.json not found, falling back to user" ),
261 aName );
262 return nullptr;
263 }
264
265 COLOR_SETTINGS* settings = RegisterSettings( new COLOR_SETTINGS( aName ) );
266
267 if( settings->GetFilename() != aName.ToStdString() )
268 {
269 wxLogTrace( traceSettings, wxT( "Warning: stored filename is actually %s, " ),
270 settings->GetFilename() );
271 }
272
273 m_color_settings[aName] = settings;
274
275 return settings;
276}
277
278
279class JSON_DIR_TRAVERSER : public wxDirTraverser
280{
281private:
282 std::function<void( const wxFileName& )> m_action;
283
284public:
285 explicit JSON_DIR_TRAVERSER( std::function<void( const wxFileName& )> aAction )
286 : m_action( std::move( aAction ) )
287 {
288 }
289
290 wxDirTraverseResult OnFile( const wxString& aFilePath ) override
291 {
292 wxFileName file( aFilePath );
293
294 if( file.GetExt() == wxS( "json" ) )
295 m_action( file );
296
297 return wxDIR_CONTINUE;
298 }
299
300 wxDirTraverseResult OnDir( const wxString& dirPath ) override
301 {
302 return wxDIR_CONTINUE;
303 }
304};
305
306
307COLOR_SETTINGS* SETTINGS_MANAGER::registerColorSettings( const wxString& aName, bool aAbsolutePath )
308{
309 if( !m_color_settings.count( aName ) )
310 {
311 COLOR_SETTINGS* colorSettings = RegisterSettings( new COLOR_SETTINGS( aName,
312 aAbsolutePath ) );
313 m_color_settings[aName] = colorSettings;
314 }
315
316 return m_color_settings.at( aName );
317}
318
319
321{
322 if( aName.EndsWith( wxT( ".json" ) ) )
323 return registerColorSettings( aName.BeforeLast( '.' ) );
324 else
325 return registerColorSettings( aName );
326}
327
328
330{
331 if( !m_color_settings.count( "user" ) )
332 {
333 COLOR_SETTINGS* settings = registerColorSettings( wxT( "user" ) );
334 settings->SetName( wxT( "User" ) );
335 Save( settings );
336 }
337
338 return m_color_settings.at( "user" );
339}
340
341
343{
345 m_color_settings[settings->GetFilename()] = RegisterSettings( settings, false );
346}
347
348
350{
351 // Create the built-in color settings
353
354 wxFileName third_party_path;
355 const ENV_VAR_MAP& env = Pgm().GetLocalEnvVariables();
356 auto it = env.find( ENV_VAR::GetVersionedEnvVarName( wxS( "3RD_PARTY" ) ) );
357
358 if( it != env.end() && !it->second.GetValue().IsEmpty() )
359 third_party_path.SetPath( it->second.GetValue() );
360 else
361 third_party_path.SetPath( PATHS::GetDefault3rdPartyPath() );
362
363 third_party_path.AppendDir( wxS( "colors" ) );
364
365 // PCM-managed themes
366 wxDir third_party_colors_dir( third_party_path.GetFullPath() );
367
368 // System-installed themes
369 wxDir system_colors_dir( PATHS::GetStockDataPath( false ) + "/colors" );
370
371 // User-created themes
372 wxDir colors_dir( GetColorSettingsPath() );
373
374 // Search for and load any other settings
375 JSON_DIR_TRAVERSER loader( [&]( const wxFileName& aFilename )
376 {
377 registerColorSettings( aFilename.GetName() );
378 } );
379
380 JSON_DIR_TRAVERSER readOnlyLoader(
381 [&]( const wxFileName& aFilename )
382 {
383 COLOR_SETTINGS* settings = registerColorSettings( aFilename.GetFullPath(), true );
384 settings->SetReadOnly( true );
385 } );
386
387 if( system_colors_dir.IsOpened() )
388 system_colors_dir.Traverse( readOnlyLoader );
389
390 if( third_party_colors_dir.IsOpened() )
391 third_party_colors_dir.Traverse( readOnlyLoader );
392
393 if( colors_dir.IsOpened() )
394 colors_dir.Traverse( loader );
395}
396
397
399{
400 m_color_settings.clear();
402}
403
404
405void SETTINGS_MANAGER::SaveColorSettings( COLOR_SETTINGS* aSettings, const std::string& aNamespace )
406{
407 // The passed settings should already be managed
408 wxASSERT( std::find_if( m_color_settings.begin(), m_color_settings.end(),
409 [aSettings] ( const std::pair<wxString, COLOR_SETTINGS*>& el )
410 {
411 return el.second->GetFilename() == aSettings->GetFilename();
412 }
413 ) != m_color_settings.end() );
414
415 if( aSettings->IsReadOnly() )
416 return;
417
418 if( !aSettings->Store() )
419 {
420 wxLogTrace( traceSettings, wxT( "Color scheme %s not modified; skipping save" ),
421 aNamespace );
422 return;
423 }
424
425 wxASSERT( aSettings->Contains( aNamespace ) );
426
427 wxLogTrace( traceSettings, wxT( "Saving color scheme %s, preserving %s" ),
428 aSettings->GetFilename(),
429 aNamespace );
430
431 std::optional<nlohmann::json> backup = aSettings->GetJson( aNamespace );
432 wxString path = GetColorSettingsPath();
433
434 aSettings->LoadFromFile( path );
435
436 if( backup )
437 ( *aSettings->Internals() )[aNamespace].update( *backup );
438
439 aSettings->Load();
440
441 aSettings->SaveToFile( path, true );
442}
443
444
446{
447 wxASSERT( aSettings );
448
449 switch( aSettings->GetLocation() )
450 {
451 case SETTINGS_LOC::USER:
453
454 case SETTINGS_LOC::PROJECT:
455 // TODO: MDI support
456 return Prj().GetProjectPath();
457
458 case SETTINGS_LOC::COLORS:
459 return GetColorSettingsPath();
460
461 case SETTINGS_LOC::TOOLBARS:
462 return GetToolbarSettingsPath();
463
464 case SETTINGS_LOC::NONE:
465 return "";
466
467 default:
468 wxASSERT_MSG( false, wxT( "Unknown settings location!" ) );
469 }
470
471 return "";
472}
473
474
475class MIGRATION_TRAVERSER : public wxDirTraverser
476{
477private:
478 wxString m_src;
479 wxString m_dest;
480 wxString m_errors;
482
483public:
484 MIGRATION_TRAVERSER( const wxString& aSrcDir, const wxString& aDestDir, bool aMigrateTables ) :
485 m_src( aSrcDir ),
486 m_dest( aDestDir ),
487 m_migrateTables( aMigrateTables )
488 {
489 }
490
491 wxString GetErrors() { return m_errors; }
492
493 wxDirTraverseResult OnFile( const wxString& aSrcFilePath ) override
494 {
495 wxFileName file( aSrcFilePath );
496
497 if( !m_migrateTables && ( file.GetName() == FILEEXT::SymbolLibraryTableFileName ||
498 file.GetName() == FILEEXT::FootprintLibraryTableFileName ) )
499 {
500 return wxDIR_CONTINUE;
501 }
502
503 // Skip migrating PCM installed packages as packages themselves are not moved
504 if( file.GetFullName() == wxT( "installed_packages.json" ) )
505 return wxDIR_CONTINUE;
506
507 // Don't migrate hotkeys config files; we don't have a reasonable migration handler for them
508 // and so there is no way to resolve conflicts at the moment
509 if( file.GetExt() == wxT( "hotkeys" ) )
510 return wxDIR_CONTINUE;
511
512 wxString path = file.GetPath();
513
514 path.Replace( m_src, m_dest, false );
515 file.SetPath( path );
516
517 wxLogTrace( traceSettings, wxT( "Copying %s to %s" ), aSrcFilePath, file.GetFullPath() );
518
519 // For now, just copy everything
520 KiCopyFile( aSrcFilePath, file.GetFullPath(), m_errors );
521
522 return wxDIR_CONTINUE;
523 }
524
525 wxDirTraverseResult OnDir( const wxString& dirPath ) override
526 {
527 wxFileName dir( dirPath );
528
529 // Whitelist of directories to migrate
530 if( dir.GetName() == wxS( "colors" ) ||
531 dir.GetName() == wxS( "3d" ) )
532 {
533
534 wxString path = dir.GetPath();
535
536 path.Replace( m_src, m_dest, false );
537 dir.SetPath( path );
538
539 wxMkdir( dir.GetFullPath() );
540
541 return wxDIR_CONTINUE;
542 }
543 else
544 {
545 return wxDIR_IGNORE;
546 }
547 }
548};
549
550
552{
553 wxFileName path( PATHS::GetUserSettingsPath(), wxS( "" ) );
554 wxLogTrace( traceSettings, wxT( "Using settings path %s" ), path.GetFullPath() );
555
556 if( m_headless )
557 {
558 // Special case namely for cli
559 // Ensure the settings directory at least exists to prevent additional loading errors
560 // from subdirectories.
561 // TODO review headless (unit tests) vs cli needs, this should be fine for unit tests though
562 if( !path.DirExists() )
563 {
564 wxLogTrace( traceSettings, wxT( "Path didn't exist; creating it" ) );
565 path.Mkdir( wxS_DIR_DEFAULT, wxPATH_MKDIR_FULL );
566 }
567
568 wxLogTrace( traceSettings, wxT( "Settings migration not checked; running headless" ) );
569 return true;
570 }
571
572 if( path.DirExists() )
573 {
574 wxFileName common = path;
575 common.SetName( wxS( "kicad_common" ) );
576 common.SetExt( wxS( "json" ) );
577
578 if( common.Exists() )
579 {
580 wxLogTrace( traceSettings, wxT( "Path exists and has a kicad_common, continuing!" ) );
581 return true;
582 }
583 }
584
585 // Now we have an empty path, let's figure out what to put in it
586 DIALOG_MIGRATE_SETTINGS dlg( this );
587
588 if( dlg.ShowModal() != wxID_OK )
589 {
590 wxLogTrace( traceSettings, wxT( "Migration dialog canceled; exiting" ) );
591 return false;
592 }
593
594 if( !path.DirExists() )
595 {
596 wxLogTrace( traceSettings, wxT( "Path didn't exist; creating it" ) );
597 path.Mkdir( wxS_DIR_DEFAULT, wxPATH_MKDIR_FULL );
598 }
599
600 if( m_migration_source.IsEmpty() )
601 {
602 wxLogTrace( traceSettings, wxT( "No migration source given; starting with defaults" ) );
603 return true;
604 }
605
606 wxLogTrace( traceSettings, wxT( "Migrating from path %s" ), m_migration_source );
607
609 wxDir source_dir( m_migration_source );
610
611 source_dir.Traverse( traverser );
612
613 if( !traverser.GetErrors().empty() )
614 DisplayErrorMessage( nullptr, traverser.GetErrors() );
615
616 // Remove any library configuration if we didn't choose to import
618 {
619 COMMON_SETTINGS common;
620 wxString commonPath = GetPathForSettingsFile( &common );
621 common.LoadFromFile( commonPath );
622
623 const std::vector<wxString> libKeys = {
624 wxT( "KICAD6_SYMBOL_DIR" ),
625 wxT( "KICAD6_3DMODEL_DIR" ),
626 wxT( "KICAD6_FOOTPRINT_DIR" ),
627 wxT( "KICAD6_TEMPLATE_DIR" ), // Stores the default library table to be copied
628 wxT( "KICAD7_SYMBOL_DIR" ),
629 wxT( "KICAD7_3DMODEL_DIR" ),
630 wxT( "KICAD7_FOOTPRINT_DIR" ),
631 wxT( "KICAD7_TEMPLATE_DIR" ),
632 wxT( "KICAD8_SYMBOL_DIR" ),
633 wxT( "KICAD8_3DMODEL_DIR" ),
634 wxT( "KICAD8_FOOTPRINT_DIR" ),
635 wxT( "KICAD8_TEMPLATE_DIR" ),
636
637 // Deprecated keys
638 wxT( "KICAD_PTEMPLATES" ),
639 wxT( "KISYS3DMOD" ),
640 wxT( "KISYSMOD" ),
641 wxT( "KICAD_SYMBOL_DIR" ),
642 };
643
644 for( const wxString& key : libKeys )
645 common.m_Env.vars.erase( key );
646
647 common.SaveToFile( commonPath );
648 }
649
650 return true;
651}
652
653
654bool SETTINGS_MANAGER::GetPreviousVersionPaths( std::vector<wxString>* aPaths )
655{
656 wxASSERT( aPaths );
657
658 aPaths->clear();
659
660 wxDir dir;
661 std::vector<wxFileName> base_paths;
662
663 base_paths.emplace_back( wxFileName( PATHS::CalculateUserSettingsPath( false ), wxS( "" ) ) );
664
665 // If the env override is set, also check the default paths
666 if( wxGetEnv( wxT( "KICAD_CONFIG_HOME" ), nullptr ) )
667 base_paths.emplace_back( wxFileName( PATHS::CalculateUserSettingsPath( false, false ),
668 wxS( "" ) ) );
669
670#ifdef __WXGTK__
671 // When running inside FlatPak, KIPLATFORM::ENV::GetUserConfigPath() will return a sandboxed
672 // path. In case the user wants to move from non-FlatPak KiCad to FlatPak KiCad, let's add our
673 // best guess as to the non-FlatPak config path. Unfortunately FlatPak also hides the host
674 // XDG_CONFIG_HOME, so if the user customizes their config path, they will have to browse
675 // for it.
676 {
677 wxFileName wxGtkPath;
678 wxGtkPath.AssignDir( wxS( "~/.config/kicad" ) );
679 wxGtkPath.MakeAbsolute();
680 base_paths.emplace_back( wxGtkPath );
681
682 // We also want to pick up regular flatpak if we are nightly
683 wxGtkPath.AssignDir( wxS( "~/.var/app/org.kicad.KiCad/config/kicad" ) );
684 wxGtkPath.MakeAbsolute();
685 base_paths.emplace_back( wxGtkPath );
686 }
687#endif
688
689 wxString subdir;
690 std::string mine = GetSettingsVersion();
691
692 auto check_dir = [&] ( const wxString& aSubDir )
693 {
694 // Only older versions are valid for migration
695 if( compareVersions( aSubDir.ToStdString(), mine ) <= 0 )
696 {
697 wxString sub_path = dir.GetNameWithSep() + aSubDir;
698
699 if( IsSettingsPathValid( sub_path ) )
700 {
701 aPaths->push_back( sub_path );
702 wxLogTrace( traceSettings, wxT( "GetPreviousVersionName: %s is valid" ), sub_path );
703 }
704 }
705 };
706
707 std::set<wxString> checkedPaths;
708
709 for( const wxFileName& base_path : base_paths )
710 {
711 if( checkedPaths.count( base_path.GetFullPath() ) )
712 continue;
713
714 checkedPaths.insert( base_path.GetFullPath() );
715
716 if( !dir.Open( base_path.GetFullPath() ) )
717 {
718 wxLogTrace( traceSettings, wxT( "GetPreviousVersionName: could not open base path %s" ),
719 base_path.GetFullPath() );
720 continue;
721 }
722
723 wxLogTrace( traceSettings, wxT( "GetPreviousVersionName: checking base path %s" ),
724 base_path.GetFullPath() );
725
726 if( dir.GetFirst( &subdir, wxEmptyString, wxDIR_DIRS ) )
727 {
728 if( subdir != mine )
729 check_dir( subdir );
730
731 while( dir.GetNext( &subdir ) )
732 {
733 if( subdir != mine )
734 check_dir( subdir );
735 }
736 }
737
738 // If we didn't find one yet, check for legacy settings without a version directory
739 if( IsSettingsPathValid( dir.GetNameWithSep() ) )
740 {
741 wxLogTrace( traceSettings,
742 wxT( "GetPreviousVersionName: root path %s is valid" ), dir.GetName() );
743 aPaths->push_back( dir.GetName() );
744 }
745 }
746
747 alg::delete_if( *aPaths, []( const wxString& aPath ) -> bool
748 {
749 wxFileName fulldir = wxFileName::DirName( aPath );
750 const wxArrayString& dirs = fulldir.GetDirs();
751
752 if( dirs.empty() || !fulldir.IsDirReadable() )
753 return true;
754
755 std::string ver = dirs.back().ToStdString();
756
757 if( !extractVersion( ver ) )
758 return true;
759
760 return false;
761 } );
762
763 std::sort( aPaths->begin(), aPaths->end(),
764 [&]( const wxString& a, const wxString& b ) -> bool
765 {
766 wxFileName aPath = wxFileName::DirName( a );
767 wxFileName bPath = wxFileName::DirName( b );
768
769 const wxArrayString& aDirs = aPath.GetDirs();
770 const wxArrayString& bDirs = bPath.GetDirs();
771
772 if( aDirs.empty() )
773 return false;
774
775 if( bDirs.empty() )
776 return true;
777
778 std::string verA = aDirs.back().ToStdString();
779 std::string verB = bDirs.back().ToStdString();
780
781 if( !extractVersion( verA ) )
782 return false;
783
784 if( !extractVersion( verB ) )
785 return true;
786
787 return compareVersions( verA, verB ) >= 0;
788 } );
789
790 return aPaths->size() > 0;
791}
792
793
794bool SETTINGS_MANAGER::IsSettingsPathValid( const wxString& aPath )
795{
796 wxFileName test( aPath, wxS( "kicad_common" ) );
797
798 if( test.Exists() )
799 return true;
800
801 test.SetExt( "json" );
802
803 return test.Exists();
804}
805
806
808{
809 wxFileName path;
810
811 path.AssignDir( PATHS::GetUserSettingsPath() );
812 path.AppendDir( wxS( "colors" ) );
813
814 if( !path.DirExists() )
815 {
816 if( !wxMkdir( path.GetPath() ) )
817 {
818 wxLogTrace( traceSettings,
819 wxT( "GetColorSettingsPath(): Path %s missing and could not be created!" ),
820 path.GetPath() );
821 }
822 }
823
824 return path.GetPath();
825}
826
827
829{
830 wxFileName path;
831
832 path.AssignDir( PATHS::GetUserSettingsPath() );
833 path.AppendDir( wxS( "toolbars" ) );
834
835 if( !path.DirExists() )
836 {
837 if( !wxMkdir( path.GetPath() ) )
838 {
839 wxLogTrace( traceSettings,
840 wxT( "GetToolbarSettingsPath(): Path %s missing and could not be created!" ),
841 path.GetPath() );
842 }
843 }
844
845 return path.GetPath();
846}
847
848
850{
851 // CMake computes the major.minor string for us.
852 return GetMajorMinorVersion().ToStdString();
853}
854
855
856int SETTINGS_MANAGER::compareVersions( const std::string& aFirst, const std::string& aSecond )
857{
858 int a_maj = 0;
859 int a_min = 0;
860 int b_maj = 0;
861 int b_min = 0;
862
863 if( !extractVersion( aFirst, &a_maj, &a_min ) || !extractVersion( aSecond, &b_maj, &b_min ) )
864 {
865 wxLogTrace( traceSettings, wxT( "compareSettingsVersions: bad input (%s, %s)" ),
866 aFirst, aSecond );
867 return -1;
868 }
869
870 if( a_maj < b_maj )
871 {
872 return -1;
873 }
874 else if( a_maj > b_maj )
875 {
876 return 1;
877 }
878 else
879 {
880 if( a_min < b_min )
881 {
882 return -1;
883 }
884 else if( a_min > b_min )
885 {
886 return 1;
887 }
888 else
889 {
890 return 0;
891 }
892 }
893}
894
895
896bool SETTINGS_MANAGER::extractVersion( const std::string& aVersionString, int* aMajor, int* aMinor )
897{
898 std::regex re_version( "(\\d+)\\.(\\d+)" );
899 std::smatch match;
900
901 if( std::regex_match( aVersionString, match, re_version ) )
902 {
903 try
904 {
905 int major = std::stoi( match[1].str() );
906 int minor = std::stoi( match[2].str() );
907
908 if( aMajor )
909 *aMajor = major;
910
911 if( aMinor )
912 *aMinor = minor;
913 }
914 catch( ... )
915 {
916 return false;
917 }
918
919 return true;
920 }
921
922 return false;
923}
924
925
926bool SETTINGS_MANAGER::LoadProject( const wxString& aFullPath, bool aSetActive )
927{
928 // Normalize path to new format even if migrating from a legacy file
929 wxFileName path( aFullPath );
930
933
934 wxString fullPath = path.GetFullPath();
935
936 // If already loaded, we are all set. This might be called more than once over a project's
937 // lifetime in case the project is first loaded by the KiCad manager and then Eeschema or
938 // Pcbnew try to load it again when they are launched.
939 if( m_projects.count( fullPath ) )
940 return true;
941
942 bool readOnly = false;
943 LOCKFILE lockFile( fullPath );
944
945 if( !lockFile.Valid() )
946 {
947 wxLogTrace( traceSettings, wxT( "Project %s is locked; opening read-only" ), fullPath );
948 readOnly = true;
949 }
950
951 // No MDI yet
952 if( aSetActive && !m_projects.empty() )
953 {
954 PROJECT* oldProject = m_projects.begin()->second;
955 unloadProjectFile( oldProject, false );
956 m_projects.erase( m_projects.begin() );
957
958 auto it = std::find_if( m_projects_list.begin(), m_projects_list.end(),
959 [&]( const std::unique_ptr<PROJECT>& ptr )
960 {
961 return ptr.get() == oldProject;
962 } );
963
964 wxASSERT( it != m_projects_list.end() );
965 m_projects_list.erase( it );
966 }
967
968 wxLogTrace( traceSettings, wxT( "Load project %s" ), fullPath );
969
970 std::unique_ptr<PROJECT> project = std::make_unique<PROJECT>();
971 project->setProjectFullName( fullPath );
972
973 if( aSetActive )
974 {
975 // until multiple projects are in play, set an environment variable for the
976 // the project pointer.
977 wxFileName projectPath( fullPath );
978 wxSetEnv( PROJECT_VAR_NAME, projectPath.GetPath() );
979
980 // set the cwd but don't impact kicad-cli
981 if( !projectPath.GetPath().IsEmpty() && wxTheApp && wxTheApp->IsGUI() )
982 wxSetWorkingDirectory( projectPath.GetPath() );
983 }
984
985 bool success = loadProjectFile( *project );
986
987 if( success )
988 {
989 project->SetReadOnly( readOnly || project->GetProjectFile().IsReadOnly() );
990
991 if( lockFile && aSetActive )
992 m_project_lock.reset( new LOCKFILE( std::move( lockFile ) ) );
993 }
994
995 m_projects_list.push_back( std::move( project ) );
996 m_projects[fullPath] = m_projects_list.back().get();
997
998 wxString fn( path.GetName() );
999
1000 PROJECT_LOCAL_SETTINGS* settings = new PROJECT_LOCAL_SETTINGS( m_projects[fullPath], fn );
1001
1002 if( aSetActive )
1003 settings = RegisterSettings( settings );
1004 else
1005 settings->LoadFromFile( path.GetPath() );
1006
1007 m_projects[fullPath]->setLocalSettings( settings );
1008
1009 if( aSetActive && m_kiway )
1011
1012 return success;
1013}
1014
1015
1016bool SETTINGS_MANAGER::UnloadProject( PROJECT* aProject, bool aSave )
1017{
1018 if( !aProject || !m_projects.count( aProject->GetProjectFullName() ) )
1019 return false;
1020
1021 if( !unloadProjectFile( aProject, aSave ) )
1022 return false;
1023
1024 wxString projectPath = aProject->GetProjectFullName();
1025 wxLogTrace( traceSettings, wxT( "Unload project %s" ), projectPath );
1026
1027 PROJECT* toRemove = m_projects.at( projectPath );
1028 bool wasActiveProject = m_projects_list.begin()->get() == toRemove;
1029
1030 auto it = std::find_if( m_projects_list.begin(), m_projects_list.end(),
1031 [&]( const std::unique_ptr<PROJECT>& ptr )
1032 {
1033 return ptr.get() == toRemove;
1034 } );
1035
1036 wxASSERT( it != m_projects_list.end() );
1037 m_projects_list.erase( it );
1038
1039 m_projects.erase( projectPath );
1040
1041 if( wasActiveProject )
1042 {
1043 // Immediately reload a null project; this is required until the rest of the application
1044 // is refactored to not assume that Prj() always works
1045 if( m_projects_list.empty() )
1046 LoadProject( "" );
1047
1048 // Remove the reference in the environment to the previous project
1049 wxSetEnv( PROJECT_VAR_NAME, wxS( "" ) );
1050
1051 // Release lock on the file, in case we had one
1052 m_project_lock = nullptr;
1053
1054 if( m_kiway )
1056 }
1057
1058 return true;
1059}
1060
1061
1063{
1064 // No MDI yet: First project in the list is the active project
1065 wxASSERT_MSG( m_projects_list.size(), wxT( "no project in list" ) );
1066 return *m_projects_list.begin()->get();
1067}
1068
1069
1071{
1072 return !m_projects.empty();
1073}
1074
1075
1077{
1078 return m_projects.size() > 1 || ( m_projects.size() == 1
1079 && !m_projects.begin()->second->GetProjectFullName().IsEmpty() );
1080}
1081
1082
1083PROJECT* SETTINGS_MANAGER::GetProject( const wxString& aFullPath ) const
1084{
1085 if( m_projects.count( aFullPath ) )
1086 return m_projects.at( aFullPath );
1087
1088 return nullptr;
1089}
1090
1091
1092std::vector<wxString> SETTINGS_MANAGER::GetOpenProjects() const
1093{
1094 std::vector<wxString> ret;
1095
1096 for( const std::pair<const wxString, PROJECT*>& pair : m_projects )
1097 {
1098 // Don't save empty projects (these are the default project settings)
1099 if( !pair.first.IsEmpty() )
1100 ret.emplace_back( pair.first );
1101 }
1102
1103 return ret;
1104}
1105
1106
1107bool SETTINGS_MANAGER::SaveProject( const wxString& aFullPath, PROJECT* aProject )
1108{
1109 if( !aProject )
1110 aProject = &Prj();
1111
1112 wxString path = aFullPath;
1113
1114 if( path.empty() )
1115 path = aProject->GetProjectFullName();
1116
1117 // TODO: refactor for MDI
1118 if( aProject->IsReadOnly() )
1119 return false;
1120
1121 if( !m_project_files.count( path ) )
1122 return false;
1123
1125 wxString projectPath = aProject->GetProjectPath();
1126
1127 project->SaveToFile( projectPath );
1128 aProject->GetLocalSettings().SaveToFile( projectPath );
1129
1130 return true;
1131}
1132
1133
1134void SETTINGS_MANAGER::SaveProjectAs( const wxString& aFullPath, PROJECT* aProject )
1135{
1136 if( !aProject )
1137 aProject = &Prj();
1138
1139 wxString oldName = aProject->GetProjectFullName();
1140
1141 if( aFullPath.IsSameAs( oldName ) )
1142 {
1143 SaveProject( aFullPath, aProject );
1144 return;
1145 }
1146
1147 // Changing this will cause UnloadProject to not save over the "old" project when loading below
1148 aProject->setProjectFullName( aFullPath );
1149
1150 wxFileName fn( aFullPath );
1151
1152 PROJECT_FILE* project = m_project_files.at( oldName );
1153
1154 // Ensure read-only flags are copied; this allows doing a "Save As" on a standalone board/sch
1155 // without creating project files if the checkbox is turned off
1156 project->SetReadOnly( aProject->IsReadOnly() );
1157 aProject->GetLocalSettings().SetReadOnly( aProject->IsReadOnly() );
1158
1159 project->SetFilename( fn.GetName() );
1160 project->SaveToFile( fn.GetPath() );
1161
1162 aProject->GetLocalSettings().SetFilename( fn.GetName() );
1163 aProject->GetLocalSettings().SaveToFile( fn.GetPath() );
1164
1165 m_project_files[fn.GetFullPath()] = project;
1166 m_project_files.erase( oldName );
1167
1168 m_projects[fn.GetFullPath()] = m_projects[oldName];
1169 m_projects.erase( oldName );
1170}
1171
1172
1173void SETTINGS_MANAGER::SaveProjectCopy( const wxString& aFullPath, PROJECT* aProject )
1174{
1175 if( !aProject )
1176 aProject = &Prj();
1177
1179 wxString oldName = project->GetFilename();
1180 wxFileName fn( aFullPath );
1181
1182 bool readOnly = project->IsReadOnly();
1183 project->SetReadOnly( false );
1184
1185 project->SetFilename( fn.GetName() );
1186 project->SaveToFile( fn.GetPath() );
1187 project->SetFilename( oldName );
1188
1189 PROJECT_LOCAL_SETTINGS& localSettings = aProject->GetLocalSettings();
1190
1191 localSettings.SetFilename( fn.GetName() );
1192 localSettings.SaveToFile( fn.GetPath() );
1193 localSettings.SetFilename( oldName );
1194
1195 project->SetReadOnly( readOnly );
1196}
1197
1198
1200{
1201 wxFileName fullFn( aProject.GetProjectFullName() );
1202 wxString fn( fullFn.GetName() );
1203
1204 PROJECT_FILE* file = RegisterSettings( new PROJECT_FILE( fn ), false );
1205
1206 m_project_files[aProject.GetProjectFullName()] = file;
1207
1208 aProject.setProjectFile( file );
1209 file->SetProject( &aProject );
1210
1211 wxString path( fullFn.GetPath() );
1212
1213 return file->LoadFromFile( path );
1214}
1215
1216
1218{
1219 if( !aProject )
1220 return false;
1221
1222 wxString name = aProject->GetProjectFullName();
1223
1224 if( !m_project_files.count( name ) )
1225 return false;
1226
1228
1229 if( !file->ShouldAutoSave() )
1230 aSave = false;
1231
1232 auto it = std::find_if( m_settings.begin(), m_settings.end(),
1233 [&file]( const std::unique_ptr<JSON_SETTINGS>& aPtr )
1234 {
1235 return aPtr.get() == file;
1236 } );
1237
1238 if( it != m_settings.end() )
1239 {
1240 wxString projectPath = GetPathForSettingsFile( it->get() );
1241
1242 bool saveLocalSettings = aSave && aProject->GetLocalSettings().ShouldAutoSave();
1243
1244 FlushAndRelease( &aProject->GetLocalSettings(), saveLocalSettings );
1245
1246 if( aSave )
1247 ( *it )->SaveToFile( projectPath );
1248
1249 m_settings.erase( it );
1250 }
1251
1252 m_project_files.erase( name );
1253
1254 return true;
1255}
1256
1257
1259{
1261}
1262
1263
1264wxString SETTINGS_MANAGER::backupDateTimeFormat = wxT( "%Y-%m-%d_%H%M%S" );
1265
1266
1267bool SETTINGS_MANAGER::BackupProject( REPORTER& aReporter, wxFileName& aTarget ) const
1268{
1269 wxDateTime timestamp = wxDateTime::Now();
1270
1271 wxString fileName = wxString::Format( wxT( "%s-%s" ), Prj().GetProjectName(),
1272 timestamp.Format( backupDateTimeFormat ) );
1273
1274 if( !aTarget.IsOk() )
1275 {
1276 aTarget.SetPath( GetProjectBackupsPath() );
1277 aTarget.SetName( fileName );
1278 aTarget.SetExt( FILEEXT::ArchiveFileExtension );
1279 }
1280
1281 wxString test = aTarget.GetPath();
1282
1283 if( !aTarget.DirExists() && !wxMkdir( aTarget.GetPath() ) )
1284 {
1285 wxLogTrace( traceSettings, wxT( "Could not create project backup path %s" ),
1286 aTarget.GetPath() );
1287 return false;
1288 }
1289
1290 if( !aTarget.IsDirWritable() )
1291 {
1292 wxLogTrace( traceSettings, wxT( "Backup directory %s is not writable" ),
1293 aTarget.GetPath() );
1294 return false;
1295 }
1296
1297 wxLogTrace( traceSettings, wxT( "Backing up project to %s" ), aTarget.GetPath() );
1298
1299 return PROJECT_ARCHIVER::Archive( Prj().GetProjectPath(), aTarget.GetFullPath(), aReporter );
1300}
1301
1302
1303class VECTOR_INSERT_TRAVERSER : public wxDirTraverser
1304{
1305public:
1306 VECTOR_INSERT_TRAVERSER( std::vector<wxString>& aVec,
1307 std::function<bool( const wxString& )> aCond ) :
1308 m_files( aVec ),
1309 m_condition( aCond )
1310 {
1311 }
1312
1313 wxDirTraverseResult OnFile( const wxString& aFile ) override
1314 {
1315 if( m_condition( aFile ) )
1316 m_files.emplace_back( aFile );
1317
1318 return wxDIR_CONTINUE;
1319 }
1320
1321 wxDirTraverseResult OnDir( const wxString& aDirName ) override
1322 {
1323 return wxDIR_CONTINUE;
1324 }
1325
1326private:
1327 std::vector<wxString>& m_files;
1328
1329 std::function<bool( const wxString& )> m_condition;
1330};
1331
1332
1334{
1336
1337 if( !settings.enabled )
1338 return true;
1339
1340 wxString prefix = Prj().GetProjectName() + '-';
1341
1342 auto modTime =
1343 [&prefix]( const wxString& aFile )
1344 {
1345 wxDateTime dt;
1346 wxString fn( wxFileName( aFile ).GetName() );
1347 fn.Replace( prefix, wxS( "" ) );
1348 dt.ParseFormat( fn, backupDateTimeFormat );
1349 return dt;
1350 };
1351
1352 wxFileName projectPath( Prj().GetProjectPath() );
1353
1354 // Skip backup if project path isn't valid or writable
1355 if( !projectPath.IsOk() || !projectPath.Exists() || !projectPath.IsDirWritable() )
1356 return true;
1357
1358 wxString backupPath = GetProjectBackupsPath();
1359
1360 if( !wxDirExists( backupPath ) )
1361 {
1362 wxLogTrace( traceSettings, wxT( "Backup path %s doesn't exist, creating it" ), backupPath );
1363
1364 if( !wxMkdir( backupPath ) )
1365 {
1366 wxLogTrace( traceSettings, wxT( "Could not create backups path! Skipping backup" ) );
1367 return false;
1368 }
1369 }
1370
1371 wxDir dir( backupPath );
1372
1373 if( !dir.IsOpened() )
1374 {
1375 wxLogTrace( traceSettings, wxT( "Could not open project backups path %s" ), dir.GetName() );
1376 return false;
1377 }
1378
1379 std::vector<wxString> files;
1380
1381 VECTOR_INSERT_TRAVERSER traverser( files,
1382 [&modTime]( const wxString& aFile )
1383 {
1384 return modTime( aFile ).IsValid();
1385 } );
1386
1387 dir.Traverse( traverser, wxT( "*.zip" ) );
1388
1389 // Sort newest-first
1390 std::sort( files.begin(), files.end(),
1391 [&]( const wxString& aFirst, const wxString& aSecond ) -> bool
1392 {
1393 wxDateTime first = modTime( aFirst );
1394 wxDateTime second = modTime( aSecond );
1395
1396 return first.GetTicks() > second.GetTicks();
1397 } );
1398
1399 // Do we even need to back up?
1400 if( !files.empty() )
1401 {
1402 wxDateTime lastTime = modTime( files[0] );
1403
1404 if( lastTime.IsValid() )
1405 {
1406 wxTimeSpan delta = wxDateTime::Now() - modTime( files[0] );
1407
1408 if( delta.IsShorterThan( wxTimeSpan::Seconds( settings.min_interval ) ) )
1409 return true;
1410 }
1411 }
1412
1413 // Backup
1414 wxFileName target;
1415 bool backupSuccessful = BackupProject( aReporter, target );
1416
1417 if( !backupSuccessful )
1418 return false;
1419
1420 // Update the file list
1421 files.insert( files.begin(), target.GetFullPath() );
1422
1423 // Are there any changes since the last backup?
1424 if( files.size() >= 2
1425 && PROJECT_ARCHIVER::AreZipArchivesIdentical( files[0], files[1], aReporter ) )
1426 {
1427 wxRemoveFile( files[0] );
1428 return true;
1429 }
1430
1431 // Now that we know a backup is needed, apply the retention policy
1432
1433 // Step 1: if we're over the total file limit, remove the oldest
1434 if( !files.empty() && settings.limit_total_files > 0 )
1435 {
1436 while( files.size() > static_cast<size_t>( settings.limit_total_files ) )
1437 {
1438 wxRemoveFile( files.back() );
1439 files.pop_back();
1440 }
1441 }
1442
1443 // Step 2: Stay under the total size limit
1444 if( settings.limit_total_size > 0 )
1445 {
1446 wxULongLong totalSize = 0;
1447
1448 for( const wxString& file : files )
1449 totalSize += wxFileName::GetSize( file );
1450
1451 while( !files.empty() && totalSize > static_cast<wxULongLong>( settings.limit_total_size ) )
1452 {
1453 totalSize -= wxFileName::GetSize( files.back() );
1454 wxRemoveFile( files.back() );
1455 files.pop_back();
1456 }
1457 }
1458
1459 // Step 3: Stay under the daily limit
1460 if( settings.limit_daily_files > 0 && files.size() > 1 )
1461 {
1462 wxDateTime day = modTime( files[0] );
1463 int num = 1;
1464
1465 wxASSERT( day.IsValid() );
1466
1467 std::vector<wxString> filesToDelete;
1468
1469 for( size_t i = 1; i < files.size(); i++ )
1470 {
1471 wxDateTime dt = modTime( files[i] );
1472
1473 if( dt.IsSameDate( day ) )
1474 {
1475 num++;
1476
1477 if( num > settings.limit_daily_files )
1478 filesToDelete.emplace_back( files[i] );
1479 }
1480 else
1481 {
1482 day = dt;
1483 num = 1;
1484 }
1485 }
1486
1487 for( const wxString& file : filesToDelete )
1488 wxRemoveFile( file );
1489 }
1490
1491 return true;
1492}
1493
1494
1496{
1498}
const char * name
Definition: DXF_plotter.cpp:59
wxString GetMajorMinorVersion()
Get only the major and minor version in a string major.minor.
Color settings are a bit different than most of the settings objects in that there can be more than o...
void SetName(const wxString &aName)
static std::vector< COLOR_SETTINGS * > CreateBuiltinColorSettings()
Constructs and returns a list of color settings objects based on the built-in color themes.
static const wxString COLOR_BUILTIN_DEFAULT
AUTO_BACKUP m_Backup
ENVIRONMENT m_Env
int ShowModal() override
std::function< void(const wxFileName &)> m_action
JSON_DIR_TRAVERSER(std::function< void(const wxFileName &)> aAction)
wxDirTraverseResult OnDir(const wxString &dirPath) override
wxDirTraverseResult OnFile(const wxString &aFilePath) override
std::optional< nlohmann::json > GetJson(const std::string &aPath) const
Fetches a JSON object that is a subset of this JSON_SETTINGS object, using a path of the form "key1....
bool Contains(const std::string &aPath) const
SETTINGS_LOC GetLocation() const
Definition: json_settings.h:88
virtual bool LoadFromFile(const wxString &aDirectory="")
Loads the backing file from disk and then calls Load()
virtual void Load()
Updates the parameters of this object based on the current JSON document contents.
bool IsReadOnly() const
Definition: json_settings.h:92
void SetReadOnly(bool aReadOnly)
Definition: json_settings.h:93
wxString GetFullFilename() const
JSON_SETTINGS_INTERNALS * Internals()
void SetFilename(const wxString &aFilename)
Definition: json_settings.h:85
virtual bool SaveToFile(const wxString &aDirectory="", bool aForce=false)
Calls Store() and then writes the contents of the JSON document to a file.
virtual bool Store()
Stores the current parameters into the JSON document represented by this object Note: this doesn't do...
wxString GetFilename() const
Definition: json_settings.h:81
virtual void ProjectChanged()
Calls ProjectChanged() on all KIWAY_PLAYERs.
Definition: kiway.cpp:640
bool Valid() const
Definition: lockfile.h:240
wxDirTraverseResult OnDir(const wxString &dirPath) override
MIGRATION_TRAVERSER(const wxString &aSrcDir, const wxString &aDestDir, bool aMigrateTables)
wxDirTraverseResult OnFile(const wxString &aSrcFilePath) override
static wxString CalculateUserSettingsPath(bool aIncludeVer=true, bool aUseEnv=true)
Determines the base path for user settings files.
Definition: paths.cpp:608
static wxString GetDefault3rdPartyPath()
Gets the default path for PCM packages.
Definition: paths.cpp:132
static wxString GetStockDataPath(bool aRespectRunFromBuildDir=true)
Gets the stock (install) data path, which is the base path for things like scripting,...
Definition: paths.cpp:196
static wxString GetUserSettingsPath()
Return the user configuration path used to store KiCad's configuration files.
Definition: paths.cpp:597
virtual ENV_VAR_MAP & GetLocalEnvVariables() const
Definition: pgm_base.cpp:933
static bool Archive(const wxString &aSrcDir, const wxString &aDestFile, REPORTER &aReporter, bool aVerbose=true, bool aIncludeExtraFiles=false)
Create an archive of the project.
static bool AreZipArchivesIdentical(const wxString &aZipFileA, const wxString &aZipFileB, REPORTER &aReporter)
Compare the CRCs of all the files in zip archive to determine whether the archives are identical.
The backing store for a PROJECT, in JSON format.
Definition: project_file.h:73
bool ShouldAutoSave() const
Definition: project_file.h:117
void SetProject(PROJECT *aProject)
Definition: project_file.h:89
The project local settings are things that are attached to a particular project, but also might be pa...
bool SaveToFile(const wxString &aDirectory="", bool aForce=false) override
Calls Store() and then writes the contents of the JSON document to a file.
Container for project specific data.
Definition: project.h:64
virtual void setProjectFile(PROJECT_FILE *aFile)
Set the backing store file for this project.
Definition: project.h:326
virtual bool IsReadOnly() const
Definition: project.h:165
virtual const wxString GetProjectFullName() const
Return the full path and name of the project.
Definition: project.cpp:140
virtual const wxString GetProjectPath() const
Return the full path of the project.
Definition: project.cpp:146
virtual const wxString GetProjectName() const
Return the short name of the project.
Definition: project.cpp:158
virtual PROJECT_LOCAL_SETTINGS & GetLocalSettings() const
Definition: project.h:209
virtual void setProjectFullName(const wxString &aFullPathAndName)
Set the full directory, basename, and extension of the project.
Definition: project.cpp:116
A pure virtual class used to derive REPORTER objects from.
Definition: reporter.h:73
static int compareVersions(const std::string &aFirst, const std::string &aSecond)
Compare two settings versions, like "5.99" and "6.0".
std::unique_ptr< LOCKFILE > m_project_lock
Lock for loaded project (expand to multiple once we support MDI).
wxString GetPathForSettingsFile(JSON_SETTINGS *aSettings)
Return the path a given settings file should be loaded from / stored to.
static std::string GetSettingsVersion()
Parse the current KiCad build version and extracts the major and minor revision to use as the name of...
void SaveProjectAs(const wxString &aFullPath, PROJECT *aProject=nullptr)
Set the currently loaded project path and saves it (pointers remain valid).
JSON_SETTINGS * registerSettings(JSON_SETTINGS *aSettings, bool aLoadNow=true)
static wxString GetUserSettingsPath()
A proxy for PATHS::GetUserSettingsPath() rather than fighting swig.
COLOR_SETTINGS * GetColorSettings(const wxString &aName="user")
Retrieve a color settings object that applications can read colors from.
T * RegisterSettings(T *aSettings, bool aLoadNow=true)
Take ownership of the pointer passed in.
void SaveProjectCopy(const wxString &aFullPath, PROJECT *aProject=nullptr)
Save a copy of the current project under the given path.
bool MigrateIfNeeded()
Handle the initialization of the user settings directory and migration from previous KiCad versions a...
wxString m_migration_source
void SaveColorSettings(COLOR_SETTINGS *aSettings, const std::string &aNamespace="")
Safely save a COLOR_SETTINGS to disk, preserving any changes outside the given namespace.
std::map< wxString, PROJECT * > m_projects
Loaded projects, mapped according to project full name.
static bool extractVersion(const std::string &aVersionString, int *aMajor=nullptr, int *aMinor=nullptr)
Extract the numeric version from a given settings string.
COLOR_SETTINGS * registerColorSettings(const wxString &aFilename, bool aAbsolutePath=false)
bool m_headless
True if running outside a UI context.
SETTINGS_MANAGER(bool aHeadless=false)
static wxString GetColorSettingsPath()
Return the path where color scheme files are stored; creating it if missing (normally .
COMMON_SETTINGS * GetCommonSettings() const
Retrieve the common settings shared by all applications.
bool SaveProject(const wxString &aFullPath=wxEmptyString, PROJECT *aProject=nullptr)
Save a loaded project.
wxString GetProjectBackupsPath() const
bool LoadProject(const wxString &aFullPath, bool aSetActive=true)
Load a project or sets up a new project with a specified path.
std::map< wxString, PROJECT_FILE * > m_project_files
Loaded project files, mapped according to project full name.
std::unordered_map< wxString, COLOR_SETTINGS * > m_color_settings
COLOR_SETTINGS * loadColorSettingsByName(const wxString &aName)
Attempt to load a color theme by name (the color theme directory and .json ext are assumed).
bool IsProjectOpen() const
Helper for checking if we have a project open.
static wxString GetToolbarSettingsPath()
Return the path where toolbar configuration files are stored; creating it if missing (normally .
bool GetPreviousVersionPaths(std::vector< wxString > *aName=nullptr)
Retrieve the name of the most recent previous KiCad version that can be found in the user settings di...
static bool IsSettingsPathValid(const wxString &aPath)
Check if a given path is probably a valid KiCad configuration directory.
bool BackupProject(REPORTER &aReporter, wxFileName &aTarget) const
Create a backup archive of the current project.
std::vector< std::unique_ptr< PROJECT > > m_projects_list
Loaded projects (ownership here).
PROJECT * GetProject(const wxString &aFullPath) const
Retrieve a loaded project by name.
bool UnloadProject(PROJECT *aProject, bool aSave=true)
Save, unload and unregister the given PROJECT.
std::vector< wxString > GetOpenProjects() const
std::vector< std::unique_ptr< JSON_SETTINGS > > m_settings
bool TriggerBackupIfNeeded(REPORTER &aReporter) const
Call BackupProject() if a new backup is needed according to the current backup policy.
bool m_migrateLibraryTables
If true, the symbol and footprint library tables will be migrated from the previous version.
bool unloadProjectFile(PROJECT *aProject, bool aSave)
Optionally save, unload and unregister the given PROJECT_FILE.
COLOR_SETTINGS * GetMigratedColorSettings()
Return a color theme for storing colors migrated from legacy (5.x and earlier) settings,...
std::unordered_map< size_t, JSON_SETTINGS * > m_app_settings_cache
Cache for app settings.
COLOR_SETTINGS * AddNewColorSettings(const wxString &aFilename)
Register a new color settings object with the given filename.
bool m_ok
True if settings loaded successfully at construction.
void registerBuiltinColorSettings()
PROJECT & Prj() const
A helper while we are not MDI-capable – return the one and only project.
bool loadProjectFile(PROJECT &aProject)
Register a PROJECT_FILE and attempt to load it from disk.
bool IsProjectOpenNotDummy() const
Helper for checking if we have a project open that is not a dummy project.
static wxString backupDateTimeFormat
void ReloadColorSettings()
Re-scan the color themes directory, reloading any changes it finds.
COMMON_SETTINGS * m_common_settings
KIWAY * m_kiway
The kiway this settings manager interacts with.
void FlushAndRelease(JSON_SETTINGS *aSettings, bool aSave=true)
If the given settings object is registered, save it to disk and unregister it.
wxDirTraverseResult OnFile(const wxString &aFile) override
wxDirTraverseResult OnDir(const wxString &aDirName) override
std::vector< wxString > & m_files
VECTOR_INSERT_TRAVERSER(std::vector< wxString > &aVec, std::function< bool(const wxString &)> aCond)
std::function< bool(const wxString &)> m_condition
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
Definition: confirm.cpp:194
This file is part of the common library.
Functions related to environment variables, including help functions.
void KiCopyFile(const wxString &aSrcPath, const wxString &aDestPath, wxString &aErrors)
Definition: gestfich.cpp:290
static const std::string SymbolLibraryTableFileName
static const std::string ProjectFileExtension
static const std::string LegacyProjectFileExtension
static const std::string FootprintLibraryTableFileName
static const std::string ArchiveFileExtension
std::map< wxString, ENV_VAR_ITEM > ENV_VAR_MAP
#define traceSettings
Definition: json_settings.h:52
File locking utilities.
This file contains miscellaneous commonly used macros and functions.
KICOMMON_API wxString GetVersionedEnvVarName(const wxString &aBaseName)
Construct a versioned environment variable based on this KiCad major version.
Definition: env_vars.cpp:83
void delete_if(_Container &__c, _Function &&__f)
Deletes all values from __c for which __f returns true.
Definition: kicad_algo.h:174
STL namespace.
PGM_BASE & Pgm()
The global program "get" accessor.
Definition: pgm_base.cpp:1071
see class PGM_BASE
#define PROJECT_VAR_NAME
A variable name whose value holds the current project directory.
Definition: project.h:40
#define PROJECT_BACKUPS_DIR_SUFFIX
Project settings path will be <projectname> + this.
int min_interval
Minimum time, in seconds, between subsequent backups.
unsigned long long limit_total_size
Maximum total size of backups (bytes), 0 for unlimited.
int limit_total_files
Maximum number of backup archives to retain.
int limit_daily_files
Maximum files to keep per day, 0 for unlimited.
bool enabled
Automatically back up the project when files are saved.
constexpr int delta
Definition of file extensions used in Kicad.