/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2014-2023 KiCad Developers, see AUTHORS.txt for contributors. * * This program is free software: you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation, either version 3 of the License, or (at your * option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program. If not, see . */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /// Helper for all the old plotting/printing code while it still exists COLOR4D GetLayerColor( SCH_LAYER_ID aLayer ) { return Pgm().GetSettingsManager().GetColorSettings()->GetColor( aLayer ); } bool SCH_EDIT_FRAME::LoadProjectSettings() { SCHEMATIC_SETTINGS& settings = Schematic().Settings(); settings.m_JunctionSize = GetSchematicJunctionSize(); GetRenderSettings()->SetDefaultPenWidth( settings.m_DefaultLineWidth ); GetRenderSettings()->m_LabelSizeRatio = settings.m_LabelSizeRatio; GetRenderSettings()->m_TextOffsetRatio = settings.m_TextOffsetRatio; GetRenderSettings()->m_PinSymbolSize = settings.m_PinSymbolSize; GetRenderSettings()->SetDashLengthRatio( settings.m_DashedLineDashRatio ); GetRenderSettings()->SetGapLengthRatio( settings.m_DashedLineGapRatio ); // Verify some values, because the config file can be edited by hand, and have bad values: LIB_SYMBOL::SetSubpartIdNotation( LIB_SYMBOL::GetSubpartIdSeparator(), LIB_SYMBOL::GetSubpartFirstId() ); BASE_SCREEN::m_DrawingSheetFileName = settings.m_SchDrawingSheetFileName; // Load the drawing sheet from the filename stored in BASE_SCREEN::m_DrawingSheetFileName. // If empty, or not existing, the default drawing sheet is loaded. wxString filename = DS_DATA_MODEL::ResolvePath( BASE_SCREEN::m_DrawingSheetFileName, Prj().GetProjectPath() ); if( !DS_DATA_MODEL::GetTheInstance().LoadDrawingSheet( filename ) ) ShowInfoBarError( _( "Error loading drawing sheet." ), true ); return true; } void SCH_EDIT_FRAME::ShowSchematicSetupDialog( const wxString& aInitialPage ) { DIALOG_SCHEMATIC_SETUP dlg( this ); if( !aInitialPage.IsEmpty() ) dlg.SetInitialPage( aInitialPage, wxEmptyString ); if( dlg.ShowQuasiModal() == wxID_OK ) { // Mark document as modified so that project settings can be saved as part of doc save OnModify(); Kiway().CommonSettingsChanged( false, true ); GetRenderSettings()->SetDefaultPenWidth( Schematic().Settings().m_DefaultLineWidth ); GetRenderSettings()->m_LabelSizeRatio = Schematic().Settings().m_LabelSizeRatio; GetRenderSettings()->m_TextOffsetRatio = Schematic().Settings().m_TextOffsetRatio; GetRenderSettings()->m_PinSymbolSize = Schematic().Settings().m_PinSymbolSize; GetRenderSettings()->SetDashLengthRatio( Schematic().Settings().m_DashedLineDashRatio ); GetRenderSettings()->SetGapLengthRatio( Schematic().Settings().m_DashedLineGapRatio ); GetCanvas()->GetView()->MarkDirty(); GetCanvas()->GetView()->UpdateAllItems( KIGFX::REPAINT ); RefreshOperatingPointDisplay(); GetCanvas()->Refresh(); } } int SCH_EDIT_FRAME::GetSchematicJunctionSize() { std::vector& sizeMultipliers = eeconfig()->m_Drawing.junction_size_mult_list; const std::shared_ptr& netSettings = Prj().GetProjectFile().NetSettings(); int sizeChoice = Schematic().Settings().m_JunctionSizeChoice; int dotSize = netSettings->m_DefaultNetClass->GetWireWidth() * sizeMultipliers[ sizeChoice ]; return std::max( dotSize, 1 ); } void SCH_EDIT_FRAME::saveProjectSettings() { wxFileName fn = Schematic().RootScreen()->GetFileName(); //ConfigFileName fn.SetExt( ProjectFileExtension ); if( !fn.HasName() || !IsWritable( fn, false ) ) return; Schematic().RecordERCExclusions(); if( Kiway().Player( FRAME_SIMULATOR, false ) ) Prj().GetProjectFile().m_SchematicSettings->m_NgspiceSettings->SaveToFile(); // Save the page layout file if doesn't exist yet (e.g. if we opened a non-kicad schematic) // TODO: We need to remove dependence on BASE_SCREEN Prj().GetProjectFile().m_SchematicSettings->m_SchDrawingSheetFileName = BASE_SCREEN::m_DrawingSheetFileName; if( !BASE_SCREEN::m_DrawingSheetFileName.IsEmpty() ) { wxFileName layoutfn( DS_DATA_MODEL::ResolvePath( BASE_SCREEN::m_DrawingSheetFileName, Prj().GetProjectPath() ) ); bool success = true; if( !layoutfn.IsAbsolute() ) success = layoutfn.MakeAbsolute( Prj().GetProjectPath() ); if( success && layoutfn.IsOk() && !layoutfn.FileExists() ) { if( layoutfn.DirExists() && layoutfn.IsDirWritable() ) DS_DATA_MODEL::GetTheInstance().Save( layoutfn.GetFullPath() ); } } GetSettingsManager()->SaveProject( fn.GetFullPath() ); } void SCH_EDIT_FRAME::SaveProjectLocalSettings() { // No schematic local settings yet } void SCH_EDIT_FRAME::LoadSettings( APP_SETTINGS_BASE* aCfg ) { // For now, axes are forced off in Eeschema even if turned on in config EESCHEMA_SETTINGS* cfg = eeconfig(); cfg->m_Window.grid.axes_enabled = false; SCH_BASE_FRAME::LoadSettings( cfg ); SCH_SEARCH_DATA* searchData = dynamic_cast( m_findReplaceData.get() ); if( searchData ) { searchData->replaceReferences = cfg->m_FindReplaceExtra.replace_references; searchData->searchAllFields = cfg->m_FindReplaceExtra.search_all_fields; searchData->searchAllPins = cfg->m_FindReplaceExtra.search_all_pins; searchData->searchCurrentSheetOnly = cfg->m_FindReplaceExtra.search_current_sheet_only; searchData->searchSelectedOnly = cfg->m_FindReplaceExtra.search_selected_only; } m_show_search = cfg->m_AuiPanels.show_search; GetRenderSettings()->m_ShowPinsElectricalType = false; GetRenderSettings()->m_ShowPinNumbers = false; GetRenderSettings()->SetDefaultFont( cfg->m_Appearance.default_font ); } void SCH_EDIT_FRAME::SaveSettings( APP_SETTINGS_BASE* aCfg ) { EESCHEMA_SETTINGS* cfg = eeconfig(); SCH_BASE_FRAME::SaveSettings( cfg ); wxAuiPaneInfo& hierarchy_pane = m_auimgr.GetPane( SchematicHierarchyPaneName() ); if( cfg ) { cfg->m_System.units = static_cast( GetUserUnits() ); cfg->m_AuiPanels.show_schematic_hierarchy = hierarchy_pane.IsShown(); cfg->m_AuiPanels.schematic_hierarchy_float = hierarchy_pane.IsFloating(); // Other parameters (hierarchy_panel_float_width, hierarchy_panel_float_height, // and hierarchy_panel_docked_width should have been updated when resizing the // hierarchy panel SCH_SEARCH_DATA* searchData = dynamic_cast( m_findReplaceData.get() ); if( searchData ) { cfg->m_FindReplaceExtra.replace_references = searchData->replaceReferences; cfg->m_FindReplaceExtra.search_all_fields = searchData->searchAllFields; cfg->m_FindReplaceExtra.search_all_pins = searchData->searchAllPins; cfg->m_FindReplaceExtra.search_current_sheet_only = searchData->searchCurrentSheetOnly; cfg->m_FindReplaceExtra.search_selected_only = searchData->searchSelectedOnly; } wxAuiPaneInfo& searchPaneInfo = m_auimgr.GetPane( SearchPaneName() ); m_show_search = searchPaneInfo.IsShown(); cfg->m_AuiPanels.show_search = m_show_search; cfg->m_AuiPanels.search_panel_height = m_searchPane->GetSize().y; cfg->m_AuiPanels.search_panel_width = m_searchPane->GetSize().x; cfg->m_AuiPanels.search_panel_dock_direction = searchPaneInfo.dock_direction; wxAuiPaneInfo& propertiesPane = m_auimgr.GetPane( PropertiesPaneName() ); cfg->m_AuiPanels.show_properties = propertiesPane.IsShown(); } } void SCH_BASE_FRAME::LoadSettings( APP_SETTINGS_BASE* aCfg ) { wxCHECK_RET( aCfg, "Call to SCH_BASE_FRAME::LoadSettings with null settings" ); EDA_DRAW_FRAME::LoadSettings( aCfg ); if( aCfg->m_Window.grid.sizes.empty() ) aCfg->m_Window.grid.sizes = aCfg->DefaultGridSizeList(); // Move legacy user grids to grid list if( !aCfg->m_Window.grid.user_grid_x.empty() ) { aCfg->m_Window.grid.sizes.emplace_back( aCfg->m_Window.grid.user_grid_x ); aCfg->m_Window.grid.user_grid_x = wxEmptyString; aCfg->m_Window.grid.user_grid_y = wxEmptyString; } if( aCfg->m_Window.grid.last_size_idx > (int) aCfg->m_Window.grid.sizes.size() ) aCfg->m_Window.grid.last_size_idx = 1; if( aCfg->m_Window.grid.fast_grid_1 > (int) aCfg->m_Window.grid.sizes.size() ) aCfg->m_Window.grid.fast_grid_1 = 1; if( aCfg->m_Window.grid.fast_grid_2 > (int) aCfg->m_Window.grid.sizes.size() ) aCfg->m_Window.grid.fast_grid_2 = 2; if( aCfg->m_Window.zoom_factors.empty() ) { aCfg->m_Window.zoom_factors = { ZOOM_LIST_EESCHEMA }; } } void SCH_BASE_FRAME::SaveSettings( APP_SETTINGS_BASE* aCfg ) { wxCHECK_RET( aCfg, wxS( "Call to SCH_BASE_FRAME::SaveSettings with null settings" ) ); EDA_DRAW_FRAME::SaveSettings( aCfg ); } static std::mutex s_symbolTableMutex; SYMBOL_LIB_TABLE* PROJECT::SchSymbolLibTable() { std::lock_guard lock( s_symbolTableMutex ); // This is a lazy loading function, it loads the project specific table when // that table is asked for, not before. SYMBOL_LIB_TABLE* tbl = (SYMBOL_LIB_TABLE*) GetElem( ELEM_SYMBOL_LIB_TABLE ); // its gotta be NULL or a SYMBOL_LIB_TABLE, or a bug. wxASSERT( !tbl || tbl->Type() == SYMBOL_LIB_TABLE_T ); if( !tbl ) { // Stack the project specific SYMBOL_LIB_TABLE overlay on top of the global table. // ~SYMBOL_LIB_TABLE() will not touch the fallback table, so multiple projects may // stack this way, all using the same global fallback table. tbl = new SYMBOL_LIB_TABLE( &SYMBOL_LIB_TABLE::GetGlobalLibTable() ); SetElem( ELEM_SYMBOL_LIB_TABLE, tbl ); wxString prjPath; wxGetEnv( PROJECT_VAR_NAME, &prjPath ); if( !prjPath.IsEmpty() ) { wxFileName fn( prjPath, SYMBOL_LIB_TABLE::GetSymbolLibTableFileName() ); try { tbl->Load( fn.GetFullPath() ); } catch( const IO_ERROR& ioe ) { wxString msg; msg.Printf( _( "Error loading the symbol library table '%s'." ), fn.GetFullPath() ); DisplayErrorMessage( nullptr, msg, ioe.What() ); } } } return tbl; }