2012-01-09 17:24:01 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
|
2022-03-25 19:51:05 +00:00
|
|
|
* Copyright (C) 2015-2022 KiCad Developers, see change_log.txt for contributors.
|
2012-01-09 17:24:01 +00:00
|
|
|
*
|
|
|
|
* 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 2
|
|
|
|
* 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, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
2023-04-07 17:08:43 +00:00
|
|
|
#include <advanced_config.h>
|
2012-04-13 18:51:24 +00:00
|
|
|
#include <base_units.h>
|
Modular-Kicad milestone B), major portions:
*) Rework the set language support, simplify it by using KIWAY. Now any major
frame with a "change language" menu can change the language for all KIWAY_PLAYERs
in the whole KIWAY. Multiple KIWAYs are not supported yet.
*) Simplify "modal wxFrame" support, and add that support exclusively to
KIWAY_PLAYER where it is inherited by all derivatives. The function
KIWAY_PLAYER::ShowModal() is in the vtable and so is cross module capable.
*) Remove the requirements and assumptions that the wxFrame hierarchy always
had PCB_EDIT_FRAME and SCH_EDIT_FRAME as immediate parents of their viewers
and editors. This is no longer the case, nor required.
*) Use KIWAY::Player() everywhere to make KIWAY_PLAYERs, this registers the
KIWAY_PLAYER within the KIWAY and makes it very easy to find an open frame
quickly. It also gives control to the KIWAY as to frame hierarchical
relationships.
*) Change single_top to use the KIWAY for loading a KIFACE and instantiating
the single KIWAY_PLAYER, see bullet immediately above.
*) Add KIWAY::OnKiwayEnd() and call it from PGM_BASE at program termination, this
gives the KIFACEs a chance to save their final configuration dope to disk.
*) Add dedicated FRAME_T's for the modal frames, so m_Ident can be tested and
these modal frames are distinctly different than their non-modal equivalents.
KIWAY_PLAYER::IsModal() is !not! a valid test during the wxFrame's constructor,
so this is another important reason for having a dedicated FRAME_T for each
modal wxFrame.
On balance, more lines were deleted than were added to achieve all this.
2014-05-03 17:40:19 +00:00
|
|
|
#include <kiway.h>
|
2022-09-16 03:06:23 +00:00
|
|
|
#include <lib_tree_model_adapter.h>
|
2020-02-03 16:46:58 +00:00
|
|
|
#include <pgm_base.h>
|
2022-09-16 03:06:23 +00:00
|
|
|
#include <eda_list_dialog.h>
|
2020-02-03 16:46:58 +00:00
|
|
|
#include <eeschema_settings.h>
|
2022-09-16 03:06:23 +00:00
|
|
|
#include <project/project_file.h>
|
2020-10-31 20:29:54 +00:00
|
|
|
#include <symbol_editor/symbol_editor_settings.h>
|
2018-08-03 12:18:26 +00:00
|
|
|
#include <sch_draw_panel.h>
|
|
|
|
#include <sch_view.h>
|
|
|
|
#include <sch_painter.h>
|
2020-02-03 16:46:58 +00:00
|
|
|
#include <settings/settings_manager.h>
|
2017-10-06 18:07:43 +00:00
|
|
|
#include <confirm.h>
|
2018-09-05 22:17:22 +00:00
|
|
|
#include <preview_items/selection_area.h>
|
2021-06-15 13:24:55 +00:00
|
|
|
#include <symbol_library.h>
|
2017-08-12 12:09:39 +00:00
|
|
|
#include <sch_base_frame.h>
|
|
|
|
#include <symbol_lib_table.h>
|
2019-06-15 00:29:42 +00:00
|
|
|
#include <tool/action_toolbar.h>
|
2019-04-13 17:41:11 +00:00
|
|
|
#include <tool/tool_manager.h>
|
|
|
|
#include <tool/tool_dispatcher.h>
|
2019-05-10 17:19:48 +00:00
|
|
|
#include <tools/ee_actions.h>
|
2019-08-08 09:06:53 +00:00
|
|
|
#include <tools/ee_selection_tool.h>
|
2022-09-16 03:06:23 +00:00
|
|
|
#include <wx/choicdlg.h>
|
2023-03-30 06:49:36 +00:00
|
|
|
#include <wx/log.h>
|
2015-04-22 11:39:00 +00:00
|
|
|
|
2022-03-31 19:03:41 +00:00
|
|
|
#include <navlib/nl_schematic_plugin.h>
|
2019-11-06 19:15:42 +00:00
|
|
|
|
2021-06-17 21:22:10 +00:00
|
|
|
LIB_SYMBOL* SchGetLibSymbol( const LIB_ID& aLibId, SYMBOL_LIB_TABLE* aLibTable,
|
|
|
|
SYMBOL_LIB* aCacheLib, wxWindow* aParent, bool aShowErrorMsg )
|
2017-10-06 18:07:43 +00:00
|
|
|
{
|
2023-01-17 04:14:38 +00:00
|
|
|
wxCHECK_MSG( aLibTable, nullptr, wxS( "Invalid symbol library table." ) );
|
2017-10-06 18:07:43 +00:00
|
|
|
|
2021-06-10 18:51:46 +00:00
|
|
|
LIB_SYMBOL* symbol = nullptr;
|
2017-10-06 18:07:43 +00:00
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2019-11-06 19:15:42 +00:00
|
|
|
symbol = aLibTable->LoadSymbol( aLibId );
|
2017-10-06 18:07:43 +00:00
|
|
|
|
2019-11-06 19:15:42 +00:00
|
|
|
if( !symbol && aCacheLib )
|
2020-04-02 19:24:28 +00:00
|
|
|
{
|
2023-01-17 04:14:38 +00:00
|
|
|
wxCHECK_MSG( aCacheLib->IsCache(), nullptr, wxS( "Invalid cache library." ) );
|
2020-04-02 19:24:28 +00:00
|
|
|
|
|
|
|
wxString cacheName = aLibId.GetLibNickname().wx_str();
|
|
|
|
cacheName += "_" + aLibId.GetLibItemName();
|
2021-06-15 12:31:28 +00:00
|
|
|
symbol = aCacheLib->FindSymbol( cacheName );
|
2020-04-02 19:24:28 +00:00
|
|
|
}
|
2017-10-06 18:07:43 +00:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& ioe )
|
|
|
|
{
|
|
|
|
if( aShowErrorMsg )
|
|
|
|
{
|
2021-06-26 11:33:37 +00:00
|
|
|
wxString msg = wxString::Format( _( "Error loading symbol %s from library '%s'." ),
|
2019-04-17 19:09:48 +00:00
|
|
|
aLibId.GetLibItemName().wx_str(),
|
|
|
|
aLibId.GetLibNickname().wx_str() );
|
2017-10-06 18:07:43 +00:00
|
|
|
DisplayErrorMessage( aParent, msg, ioe.What() );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-06 19:15:42 +00:00
|
|
|
return symbol;
|
2017-10-06 18:07:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-04-17 19:09:48 +00:00
|
|
|
SCH_BASE_FRAME::SCH_BASE_FRAME( KIWAY* aKiway, wxWindow* aParent, FRAME_T aWindowType,
|
|
|
|
const wxString& aTitle, const wxPoint& aPosition,
|
|
|
|
const wxSize& aSize, long aStyle, const wxString& aFrameName ) :
|
2022-03-31 19:03:41 +00:00
|
|
|
EDA_DRAW_FRAME( aKiway, aParent, aWindowType, aTitle, aPosition, aSize, aStyle,
|
2022-09-16 04:38:10 +00:00
|
|
|
aFrameName, schIUScale ),
|
2023-04-07 17:08:43 +00:00
|
|
|
m_base_frame_defaults( nullptr, "base_Frame_defaults" ), m_spaceMouse( nullptr )
|
2012-02-19 19:53:11 +00:00
|
|
|
{
|
2018-09-10 13:37:28 +00:00
|
|
|
createCanvas();
|
2020-11-18 15:23:05 +00:00
|
|
|
|
|
|
|
Bind( wxEVT_IDLE,
|
2021-04-12 10:10:22 +00:00
|
|
|
[this]( wxIdleEvent& aEvent )
|
|
|
|
{
|
|
|
|
// Handle cursor adjustments. While we can get motion and key events through
|
|
|
|
// wxWidgets, we can't get modifier-key-up events.
|
|
|
|
if( m_toolManager )
|
|
|
|
{
|
|
|
|
EE_SELECTION_TOOL* selTool = m_toolManager->GetTool<EE_SELECTION_TOOL>();
|
|
|
|
|
|
|
|
if( selTool )
|
|
|
|
selTool->OnIdle( aEvent );
|
|
|
|
}
|
|
|
|
} );
|
2023-05-31 20:37:58 +00:00
|
|
|
|
|
|
|
m_watcherDebounceTimer.Bind( wxEVT_TIMER, &SCH_BASE_FRAME::OnSymChangeDebounceTimer, this );
|
2012-02-19 19:53:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-08-20 13:48:48 +00:00
|
|
|
SCH_BASE_FRAME::~SCH_BASE_FRAME()
|
|
|
|
{
|
2023-04-07 17:08:43 +00:00
|
|
|
delete m_spaceMouse;
|
2022-08-20 13:48:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-09 17:24:01 +00:00
|
|
|
SCH_SCREEN* SCH_BASE_FRAME::GetScreen() const
|
|
|
|
{
|
|
|
|
return (SCH_SCREEN*) EDA_DRAW_FRAME::GetScreen();
|
|
|
|
}
|
|
|
|
|
2017-10-06 18:07:43 +00:00
|
|
|
|
2020-05-20 03:34:55 +00:00
|
|
|
EESCHEMA_SETTINGS* SCH_BASE_FRAME::eeconfig() const
|
|
|
|
{
|
|
|
|
return dynamic_cast<EESCHEMA_SETTINGS*>( config() );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-10-31 01:27:16 +00:00
|
|
|
SYMBOL_EDITOR_SETTINGS* SCH_BASE_FRAME::libeditconfig() const
|
2020-05-20 03:34:55 +00:00
|
|
|
{
|
2020-10-31 01:27:16 +00:00
|
|
|
return dynamic_cast<SYMBOL_EDITOR_SETTINGS*>( config() );
|
2020-05-20 03:34:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-09 17:24:01 +00:00
|
|
|
void SCH_BASE_FRAME::SetPageSettings( const PAGE_INFO& aPageSettings )
|
|
|
|
{
|
|
|
|
GetScreen()->SetPageSettings( aPageSettings );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const PAGE_INFO& SCH_BASE_FRAME::GetPageSettings () const
|
|
|
|
{
|
|
|
|
return GetScreen()->GetPageSettings();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-02-19 03:40:07 +00:00
|
|
|
const VECTOR2I SCH_BASE_FRAME::GetPageSizeIU() const
|
2012-01-09 17:24:01 +00:00
|
|
|
{
|
|
|
|
// GetSizeIU is compile time dependent:
|
2022-09-16 23:25:07 +00:00
|
|
|
return GetScreen()->GetPageSettings().GetSizeIU( schIUScale.IU_PER_MILS );
|
2012-01-09 17:24:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const TITLE_BLOCK& SCH_BASE_FRAME::GetTitleBlock() const
|
|
|
|
{
|
|
|
|
wxASSERT( GetScreen() );
|
|
|
|
return GetScreen()->GetTitleBlock();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SCH_BASE_FRAME::SetTitleBlock( const TITLE_BLOCK& aTitleBlock )
|
|
|
|
{
|
|
|
|
wxASSERT( GetScreen() );
|
|
|
|
GetScreen()->SetTitleBlock( aTitleBlock );
|
|
|
|
}
|
2012-04-13 18:51:24 +00:00
|
|
|
|
|
|
|
|
|
|
|
void SCH_BASE_FRAME::UpdateStatusBar()
|
|
|
|
{
|
|
|
|
wxString line;
|
|
|
|
BASE_SCREEN* screen = GetScreen();
|
|
|
|
|
|
|
|
if( !screen )
|
|
|
|
return;
|
|
|
|
|
|
|
|
EDA_DRAW_FRAME::UpdateStatusBar();
|
|
|
|
|
2020-10-04 15:29:18 +00:00
|
|
|
// Display absolute and relative coordinates
|
2019-06-13 17:28:55 +00:00
|
|
|
VECTOR2D cursorPos = GetCanvas()->GetViewControls()->GetCursorPosition();
|
2020-10-04 15:29:18 +00:00
|
|
|
VECTOR2D d = cursorPos - screen->m_LocalOrigin;
|
2012-04-13 18:51:24 +00:00
|
|
|
|
2023-01-17 04:14:38 +00:00
|
|
|
line.Printf( wxS( "X %s Y %s" ),
|
2022-09-19 09:25:20 +00:00
|
|
|
MessageTextFromValue( cursorPos.x, false ),
|
|
|
|
MessageTextFromValue( cursorPos.y, false ) );
|
2012-04-13 18:51:24 +00:00
|
|
|
SetStatusText( line, 2 );
|
|
|
|
|
2023-01-17 04:14:38 +00:00
|
|
|
line.Printf( wxS( "dx %s dy %s dist %s" ),
|
2022-09-19 09:25:20 +00:00
|
|
|
MessageTextFromValue( d.x, false ),
|
|
|
|
MessageTextFromValue( d.y, false ),
|
|
|
|
MessageTextFromValue( hypot( d.x, d.y ), false ) );
|
2012-04-13 18:51:24 +00:00
|
|
|
SetStatusText( line, 3 );
|
|
|
|
|
2019-05-24 13:55:33 +00:00
|
|
|
DisplayGridMsg();
|
2012-04-13 18:51:24 +00:00
|
|
|
DisplayUnitsMsg();
|
|
|
|
}
|
2017-08-12 12:09:39 +00:00
|
|
|
|
|
|
|
|
2021-06-17 21:22:10 +00:00
|
|
|
LIB_SYMBOL* SCH_BASE_FRAME::GetLibSymbol( const LIB_ID& aLibId, bool aUseCacheLib,
|
|
|
|
bool aShowErrorMsg )
|
2017-10-06 18:07:43 +00:00
|
|
|
{
|
2021-07-16 20:13:26 +00:00
|
|
|
SYMBOL_LIB* cache = ( aUseCacheLib ) ? Prj().SchLibs()->GetCacheLibrary() : nullptr;
|
2017-10-06 18:07:43 +00:00
|
|
|
|
2021-06-17 21:22:10 +00:00
|
|
|
return SchGetLibSymbol( aLibId, Prj().SchSymbolLibTable(), cache, this, aShowErrorMsg );
|
2017-10-06 18:07:43 +00:00
|
|
|
}
|
2017-11-13 10:53:19 +00:00
|
|
|
|
|
|
|
|
|
|
|
bool SCH_BASE_FRAME::saveSymbolLibTables( bool aGlobal, bool aProject )
|
|
|
|
{
|
2018-08-06 18:33:28 +00:00
|
|
|
wxString msg;
|
2017-11-13 10:53:19 +00:00
|
|
|
bool success = true;
|
|
|
|
|
|
|
|
if( aGlobal )
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
2018-08-06 18:33:28 +00:00
|
|
|
SYMBOL_LIB_TABLE::GetGlobalLibTable().Save( SYMBOL_LIB_TABLE::GetGlobalTableFileName() );
|
2017-11-13 10:53:19 +00:00
|
|
|
}
|
|
|
|
catch( const IO_ERROR& ioe )
|
|
|
|
{
|
|
|
|
success = false;
|
2019-04-17 19:09:48 +00:00
|
|
|
msg.Printf( _( "Error saving global symbol library table:\n%s" ), ioe.What() );
|
2023-01-19 00:53:16 +00:00
|
|
|
DisplayErrorMessage( this, msg );
|
2017-11-13 10:53:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( aProject && !Prj().GetProjectName().IsEmpty() )
|
|
|
|
{
|
|
|
|
wxFileName fn( Prj().GetProjectPath(), SYMBOL_LIB_TABLE::GetSymbolLibTableFileName() );
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
Prj().SchSymbolLibTable()->Save( fn.GetFullPath() );
|
|
|
|
}
|
|
|
|
catch( const IO_ERROR& ioe )
|
|
|
|
{
|
|
|
|
success = false;
|
2021-06-15 12:31:28 +00:00
|
|
|
msg.Printf( _( "Error saving project-specific symbol library table:\n%s" ),
|
|
|
|
ioe.What() );
|
2023-01-19 00:53:16 +00:00
|
|
|
DisplayErrorMessage( this, msg );
|
2017-11-13 10:53:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return success;
|
|
|
|
}
|
2018-08-03 12:18:26 +00:00
|
|
|
|
2018-09-08 20:57:56 +00:00
|
|
|
|
2022-09-16 03:06:23 +00:00
|
|
|
SYMBOL_LIB_TABLE* SCH_BASE_FRAME::SelectSymLibTable( bool aOptional )
|
|
|
|
{
|
|
|
|
// If no project is loaded, always work with the global table
|
|
|
|
if( Prj().IsNullProject() )
|
|
|
|
{
|
|
|
|
SYMBOL_LIB_TABLE* ret = &SYMBOL_LIB_TABLE::GetGlobalLibTable();
|
|
|
|
|
|
|
|
if( aOptional )
|
|
|
|
{
|
|
|
|
wxMessageDialog dlg( this, _( "Add the library to the global library table?" ),
|
|
|
|
_( "Add To Global Library Table" ), wxYES_NO );
|
|
|
|
|
|
|
|
if( dlg.ShowModal() != wxID_OK )
|
|
|
|
ret = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
wxArrayString libTableNames;
|
|
|
|
libTableNames.Add( _( "Global" ) );
|
|
|
|
libTableNames.Add( _( "Project" ) );
|
|
|
|
|
|
|
|
wxSingleChoiceDialog dlg( this, _( "Choose the Library Table to add the library to:" ),
|
|
|
|
_( "Add To Library Table" ), libTableNames );
|
|
|
|
|
|
|
|
if( aOptional )
|
|
|
|
{
|
|
|
|
dlg.FindWindow( wxID_CANCEL )->SetLabel( _( "Skip" ) );
|
|
|
|
dlg.FindWindow( wxID_OK )->SetLabel( _( "Add" ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
if( dlg.ShowModal() != wxID_OK )
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
switch( dlg.GetSelection() )
|
|
|
|
{
|
|
|
|
case 0: return &SYMBOL_LIB_TABLE::GetGlobalLibTable();
|
|
|
|
case 1: return Prj().SchSymbolLibTable();
|
|
|
|
default: return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-01 06:04:08 +00:00
|
|
|
void SCH_BASE_FRAME::RedrawScreen( const VECTOR2I& aCenterPoint, bool aWarpPointer )
|
2018-08-03 12:18:26 +00:00
|
|
|
{
|
2019-03-11 21:32:05 +00:00
|
|
|
GetCanvas()->GetView()->SetCenter( aCenterPoint );
|
|
|
|
|
2018-09-03 13:58:47 +00:00
|
|
|
if( aWarpPointer )
|
2018-09-14 08:15:10 +00:00
|
|
|
GetCanvas()->GetViewControls()->CenterOnCursor();
|
2018-09-03 13:58:47 +00:00
|
|
|
|
|
|
|
GetCanvas()->Refresh();
|
2018-08-03 12:18:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-05 22:17:22 +00:00
|
|
|
void SCH_BASE_FRAME::HardRedraw()
|
|
|
|
{
|
2021-11-22 19:47:17 +00:00
|
|
|
if( GetCanvas() && GetCanvas()->GetView() )
|
|
|
|
{
|
|
|
|
GetCanvas()->GetView()->UpdateAllItems( KIGFX::ALL );
|
|
|
|
GetCanvas()->ForceRefresh();
|
|
|
|
}
|
2018-09-05 22:17:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-02 20:19:22 +00:00
|
|
|
SCH_DRAW_PANEL* SCH_BASE_FRAME::GetCanvas() const
|
2018-08-03 12:18:26 +00:00
|
|
|
{
|
2019-06-13 17:28:55 +00:00
|
|
|
return static_cast<SCH_DRAW_PANEL*>( EDA_DRAW_FRAME::GetCanvas() );
|
2018-09-02 20:19:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
KIGFX::SCH_RENDER_SETTINGS* SCH_BASE_FRAME::GetRenderSettings()
|
|
|
|
{
|
2021-11-22 19:47:17 +00:00
|
|
|
if( GetCanvas() && GetCanvas()->GetView() )
|
|
|
|
{
|
|
|
|
if( KIGFX::PAINTER* painter = GetCanvas()->GetView()->GetPainter() )
|
|
|
|
return static_cast<KIGFX::SCH_RENDER_SETTINGS*>( painter->GetSettings() );
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
2018-08-03 12:18:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-10 13:37:28 +00:00
|
|
|
void SCH_BASE_FRAME::createCanvas()
|
2018-08-03 12:18:26 +00:00
|
|
|
{
|
2020-11-16 11:16:44 +00:00
|
|
|
m_canvasType = loadCanvasTypeSetting();
|
2018-09-10 13:37:28 +00:00
|
|
|
|
2020-11-16 11:16:44 +00:00
|
|
|
SetCanvas( new SCH_DRAW_PANEL( this, wxID_ANY, wxPoint( 0, 0 ), m_frameSize,
|
2020-11-24 22:16:41 +00:00
|
|
|
GetGalDisplayOptions(), m_canvasType ) );
|
2019-05-30 12:25:08 +00:00
|
|
|
ActivateGalCanvas();
|
2018-08-03 12:18:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-03-31 19:03:41 +00:00
|
|
|
void SCH_BASE_FRAME::ActivateGalCanvas()
|
|
|
|
{
|
|
|
|
EDA_DRAW_FRAME::ActivateGalCanvas();
|
|
|
|
|
2023-06-13 16:44:22 +00:00
|
|
|
try
|
2022-03-31 19:03:41 +00:00
|
|
|
{
|
2023-06-13 16:44:22 +00:00
|
|
|
if( !m_spaceMouse )
|
|
|
|
m_spaceMouse = new NL_SCHEMATIC_PLUGIN();
|
2022-03-31 19:03:41 +00:00
|
|
|
|
2023-06-13 16:44:22 +00:00
|
|
|
m_spaceMouse->SetCanvas( GetCanvas() );
|
|
|
|
}
|
|
|
|
catch( const std::system_error& e )
|
|
|
|
{
|
|
|
|
wxLogTrace( wxT( "KI_TRACE_NAVLIB" ), e.what() );
|
2022-03-31 19:03:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-10-06 03:10:20 +00:00
|
|
|
void SCH_BASE_FRAME::UpdateItem( EDA_ITEM* aItem, bool isAddOrDelete, bool aUpdateRtree )
|
2018-09-10 10:58:50 +00:00
|
|
|
{
|
|
|
|
EDA_ITEM* parent = aItem->GetParent();
|
|
|
|
|
|
|
|
if( aItem->Type() == SCH_SHEET_PIN_T )
|
|
|
|
{
|
|
|
|
// Sheet pins aren't in the view. Refresh their parent.
|
|
|
|
if( parent )
|
|
|
|
GetCanvas()->GetView()->Update( parent );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if( !isAddOrDelete )
|
|
|
|
GetCanvas()->GetView()->Update( aItem );
|
|
|
|
|
2020-11-17 16:02:47 +00:00
|
|
|
// Some children are drawn from their parents. Mark them for re-paint.
|
2022-08-20 09:27:35 +00:00
|
|
|
if( parent && parent->IsType( { SCH_SYMBOL_T, SCH_SHEET_T, SCH_LABEL_LOCATE_ANY_T } ) )
|
2020-03-12 13:56:44 +00:00
|
|
|
GetCanvas()->GetView()->Update( parent, KIGFX::REPAINT );
|
2018-09-10 10:58:50 +00:00
|
|
|
}
|
|
|
|
|
2021-10-06 03:10:20 +00:00
|
|
|
/**
|
|
|
|
* Be careful when calling this. Update will invalidate RTree iterators, so you cannot call this
|
|
|
|
* while doing things like `for( SCH_ITEM* item : screen->Items() )`
|
|
|
|
*/
|
2023-04-11 14:25:11 +00:00
|
|
|
if( aUpdateRtree && dynamic_cast<SCH_ITEM*>( aItem ) )
|
2021-10-06 03:10:20 +00:00
|
|
|
GetScreen()->Update( static_cast<SCH_ITEM*>( aItem ) );
|
2021-09-30 14:09:32 +00:00
|
|
|
|
2020-08-10 11:40:58 +00:00
|
|
|
// Calling Refresh() here introduces a bi-stable state: when doing operations on a
|
|
|
|
// large number of items if at some point the refresh timer times out and does a
|
|
|
|
// refresh it will take long enough that the next item will also time out, and the
|
|
|
|
// next, and the next, etc.
|
|
|
|
// GetCanvas()->Refresh();
|
2018-09-10 10:58:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-05-13 21:36:20 +00:00
|
|
|
void SCH_BASE_FRAME::RefreshZoomDependentItems()
|
2019-08-08 09:06:53 +00:00
|
|
|
{
|
2022-05-13 21:36:20 +00:00
|
|
|
// We currently have two zoom-dependent renderings: text, which is rendered as bitmap text
|
|
|
|
// when too small to see the difference, and selection shadows.
|
|
|
|
//
|
|
|
|
// Because non-selected text is cached by OpenGL, we only apply the bitmap performance hack
|
|
|
|
// to selected text items.
|
|
|
|
//
|
|
|
|
// Thus, as it currently stands, all zoom-dependent items can be found in the list of selected
|
|
|
|
// items.
|
|
|
|
|
2019-08-08 09:06:53 +00:00
|
|
|
if( m_toolManager )
|
|
|
|
{
|
|
|
|
EE_SELECTION_TOOL* selectionTool = m_toolManager->GetTool<EE_SELECTION_TOOL>();
|
|
|
|
SELECTION& selection = selectionTool->GetSelection();
|
|
|
|
KIGFX::SCH_VIEW* view = GetCanvas()->GetView();
|
|
|
|
|
|
|
|
for( EDA_ITEM* item : selection )
|
2019-12-10 12:07:11 +00:00
|
|
|
{
|
2022-05-13 21:36:20 +00:00
|
|
|
if( item->RenderAsBitmap( view->GetGAL()->GetWorldScale() ) != item->IsShownAsBitmap()
|
|
|
|
|| item->IsType( KIGFX::SCH_PAINTER::g_ScaledSelectionTypes ) )
|
2019-12-10 12:07:11 +00:00
|
|
|
{
|
|
|
|
view->Update( item, KIGFX::REPAINT );
|
|
|
|
|
2022-05-13 21:36:20 +00:00
|
|
|
EDA_ITEM* parent = item->GetParent();
|
|
|
|
|
2021-06-10 14:10:55 +00:00
|
|
|
// Symbol children are drawn from their parents. Mark them for re-paint.
|
|
|
|
if( parent && parent->Type() == SCH_SYMBOL_T )
|
2019-12-10 12:07:11 +00:00
|
|
|
GetCanvas()->GetView()->Update( parent, KIGFX::REPAINT );
|
|
|
|
}
|
|
|
|
}
|
2019-08-08 09:06:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-05-06 15:30:29 +00:00
|
|
|
void SCH_BASE_FRAME::AddToScreen( EDA_ITEM* aItem, SCH_SCREEN* aScreen )
|
2018-08-03 12:18:26 +00:00
|
|
|
{
|
2022-03-25 19:51:05 +00:00
|
|
|
// Null pointers will cause boost::ptr_vector to raise a boost::bad_pointer exception which
|
|
|
|
// will be unhandled. There is no valid reason to pass an invalid EDA_ITEM pointer to the
|
|
|
|
// screen append function.
|
|
|
|
wxCHECK( aItem != nullptr, /* voide */ );
|
|
|
|
|
2019-03-11 21:32:05 +00:00
|
|
|
auto screen = aScreen;
|
|
|
|
|
|
|
|
if( aScreen == nullptr )
|
|
|
|
screen = GetScreen();
|
|
|
|
|
2019-05-06 15:30:29 +00:00
|
|
|
screen->Append( (SCH_ITEM*) aItem );
|
2019-03-11 21:32:05 +00:00
|
|
|
|
|
|
|
if( screen == GetScreen() )
|
|
|
|
{
|
|
|
|
GetCanvas()->GetView()->Add( aItem );
|
2020-08-10 11:40:58 +00:00
|
|
|
UpdateItem( aItem, true ); // handle any additional parent semantics
|
2019-03-11 21:32:05 +00:00
|
|
|
}
|
2018-08-03 12:18:26 +00:00
|
|
|
}
|
|
|
|
|
2018-09-08 18:07:41 +00:00
|
|
|
|
2019-05-06 15:30:29 +00:00
|
|
|
void SCH_BASE_FRAME::RemoveFromScreen( EDA_ITEM* aItem, SCH_SCREEN* aScreen )
|
2018-08-03 12:18:26 +00:00
|
|
|
{
|
2019-03-11 21:32:05 +00:00
|
|
|
auto screen = aScreen;
|
|
|
|
|
|
|
|
if( aScreen == nullptr )
|
|
|
|
screen = GetScreen();
|
|
|
|
|
|
|
|
if( screen == GetScreen() )
|
|
|
|
GetCanvas()->GetView()->Remove( aItem );
|
|
|
|
|
2019-05-06 15:30:29 +00:00
|
|
|
screen->Remove( (SCH_ITEM*) aItem );
|
2019-03-11 21:32:05 +00:00
|
|
|
|
|
|
|
if( screen == GetScreen() )
|
2020-08-10 11:40:58 +00:00
|
|
|
UpdateItem( aItem, true ); // handle any additional parent semantics
|
2018-08-03 12:18:26 +00:00
|
|
|
}
|
|
|
|
|
2018-09-08 18:07:41 +00:00
|
|
|
|
2018-08-03 12:18:26 +00:00
|
|
|
void SCH_BASE_FRAME::SyncView()
|
|
|
|
{
|
2023-02-16 13:25:10 +00:00
|
|
|
// Let tools add things to the view if necessary
|
|
|
|
if( m_toolManager )
|
|
|
|
m_toolManager->ResetTools( TOOL_BASE::MODEL_RELOAD );
|
|
|
|
|
2019-06-13 17:28:55 +00:00
|
|
|
GetCanvas()->GetView()->UpdateAllItems( KIGFX::ALL );
|
2018-08-03 12:18:26 +00:00
|
|
|
}
|
2019-06-15 00:29:42 +00:00
|
|
|
|
|
|
|
|
2020-02-03 16:46:58 +00:00
|
|
|
COLOR4D SCH_BASE_FRAME::GetLayerColor( SCH_LAYER_ID aLayer )
|
|
|
|
{
|
|
|
|
return GetColorSettings()->GetColor( aLayer );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-07-17 20:04:14 +00:00
|
|
|
void SCH_BASE_FRAME::CommonSettingsChanged( bool aEnvVarsChanged, bool aTextVarsChanged )
|
2020-04-09 16:15:57 +00:00
|
|
|
{
|
2020-07-17 20:04:14 +00:00
|
|
|
EDA_DRAW_FRAME::CommonSettingsChanged( aEnvVarsChanged, aTextVarsChanged );
|
2020-04-09 16:15:57 +00:00
|
|
|
|
2021-09-07 20:34:10 +00:00
|
|
|
COLOR_SETTINGS* colorSettings = GetColorSettings( true );
|
2021-08-29 23:33:08 +00:00
|
|
|
|
|
|
|
GetCanvas()->GetView()->GetPainter()->GetSettings()->LoadColors( colorSettings );
|
|
|
|
GetCanvas()->GetGAL()->SetAxesColor( colorSettings->GetColor( LAYER_SCHEMATIC_GRID_AXES ) );
|
2020-11-29 02:04:34 +00:00
|
|
|
|
|
|
|
GetCanvas()->GetView()->UpdateAllItems( KIGFX::ALL );
|
2021-08-29 23:33:08 +00:00
|
|
|
GetCanvas()->GetView()->RecacheAllItems();
|
|
|
|
GetCanvas()->GetView()->MarkTargetDirty( KIGFX::TARGET_NONCACHED );
|
2020-04-09 16:15:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-09-07 20:34:10 +00:00
|
|
|
COLOR_SETTINGS* SCH_BASE_FRAME::GetColorSettings( bool aForceRefresh ) const
|
2020-02-03 16:46:58 +00:00
|
|
|
{
|
2021-09-07 20:34:10 +00:00
|
|
|
if( !m_colorSettings || aForceRefresh )
|
2020-02-03 16:46:58 +00:00
|
|
|
{
|
2021-08-29 23:33:08 +00:00
|
|
|
SETTINGS_MANAGER& mgr = Pgm().GetSettingsManager();
|
|
|
|
EESCHEMA_SETTINGS* cfg = mgr.GetAppSettings<EESCHEMA_SETTINGS>();
|
|
|
|
wxString colorTheme = cfg->m_ColorTheme;
|
|
|
|
|
|
|
|
if( IsType( FRAME_SCH_SYMBOL_EDITOR ) )
|
|
|
|
{
|
|
|
|
SYMBOL_EDITOR_SETTINGS* symCfg = mgr.GetAppSettings<SYMBOL_EDITOR_SETTINGS>();
|
|
|
|
|
|
|
|
if( !symCfg->m_UseEeschemaColorSettings )
|
|
|
|
colorTheme = symCfg->m_ColorTheme;
|
|
|
|
}
|
|
|
|
|
|
|
|
COLOR_SETTINGS* colorSettings = mgr.GetColorSettings( colorTheme );
|
2020-12-23 23:18:02 +00:00
|
|
|
|
|
|
|
const_cast<SCH_BASE_FRAME*>( this )->m_colorSettings = colorSettings;
|
2020-02-03 16:46:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return m_colorSettings;
|
2020-04-02 19:24:28 +00:00
|
|
|
}
|
2020-12-23 23:18:02 +00:00
|
|
|
|
|
|
|
|
|
|
|
COLOR4D SCH_BASE_FRAME::GetDrawBgColor() const
|
|
|
|
{
|
|
|
|
return GetColorSettings()->GetColor( LAYER_SCHEMATIC_BACKGROUND );
|
|
|
|
}
|
|
|
|
|
2022-03-31 19:03:41 +00:00
|
|
|
|
|
|
|
void SCH_BASE_FRAME::handleActivateEvent( wxActivateEvent& aEvent )
|
|
|
|
{
|
|
|
|
EDA_DRAW_FRAME::handleActivateEvent( aEvent );
|
|
|
|
|
2023-06-13 16:44:22 +00:00
|
|
|
if( m_spaceMouse )
|
2022-03-31 19:03:41 +00:00
|
|
|
m_spaceMouse->SetFocus( aEvent.GetActive() );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SCH_BASE_FRAME::handleIconizeEvent( wxIconizeEvent& aEvent )
|
|
|
|
{
|
|
|
|
EDA_DRAW_FRAME::handleIconizeEvent( aEvent );
|
|
|
|
|
2023-06-13 16:44:22 +00:00
|
|
|
if( m_spaceMouse )
|
2022-03-31 19:03:41 +00:00
|
|
|
m_spaceMouse->SetFocus( false );
|
|
|
|
}
|
2022-09-16 03:06:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
wxString SCH_BASE_FRAME::SelectLibraryFromList()
|
|
|
|
{
|
|
|
|
COMMON_SETTINGS* cfg = Pgm().GetCommonSettings();
|
|
|
|
PROJECT& prj = Prj();
|
|
|
|
|
|
|
|
if( prj.SchSymbolLibTable()->IsEmpty() )
|
|
|
|
{
|
|
|
|
ShowInfoBarError( _( "No symbol libraries are loaded." ) );
|
|
|
|
return wxEmptyString;
|
|
|
|
}
|
|
|
|
|
|
|
|
wxArrayString headers;
|
|
|
|
|
|
|
|
headers.Add( _( "Library" ) );
|
|
|
|
|
|
|
|
std::vector< wxArrayString > itemsToDisplay;
|
|
|
|
std::vector< wxString > libNicknames = prj.SchSymbolLibTable()->GetLogicalLibs();
|
|
|
|
|
|
|
|
for( const wxString& name : libNicknames )
|
|
|
|
{
|
|
|
|
// Exclude read only libraries.
|
|
|
|
if( !prj.SchSymbolLibTable()->IsSymbolLibWritable( name ) )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if( alg::contains( prj.GetProjectFile().m_PinnedSymbolLibs, name )
|
|
|
|
|| alg::contains( cfg->m_Session.pinned_symbol_libs, name ) )
|
|
|
|
{
|
|
|
|
wxArrayString item;
|
|
|
|
|
|
|
|
item.Add( LIB_TREE_MODEL_ADAPTER::GetPinningSymbol() + name );
|
|
|
|
itemsToDisplay.push_back( item );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for( const wxString& name : libNicknames )
|
|
|
|
{
|
|
|
|
// Exclude read only libraries.
|
|
|
|
if( !prj.SchSymbolLibTable()->IsSymbolLibWritable( name ) )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if( !alg::contains( prj.GetProjectFile().m_PinnedSymbolLibs, name )
|
|
|
|
&& !alg::contains( cfg->m_Session.pinned_symbol_libs, name ) )
|
|
|
|
{
|
|
|
|
wxArrayString item;
|
|
|
|
|
|
|
|
item.Add( name );
|
|
|
|
itemsToDisplay.push_back( item );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
wxString oldLibName = prj.GetRString( PROJECT::SCH_LIB_SELECT );
|
|
|
|
|
|
|
|
EDA_LIST_DIALOG dlg( this, _( "Select Symbol Library" ), headers, itemsToDisplay, oldLibName,
|
|
|
|
false );
|
|
|
|
|
|
|
|
if( dlg.ShowModal() != wxID_OK )
|
|
|
|
return wxEmptyString;
|
|
|
|
|
|
|
|
wxString libName = dlg.GetTextSelection();
|
|
|
|
|
|
|
|
if( !libName.empty() )
|
|
|
|
{
|
|
|
|
if( prj.SchSymbolLibTable()->HasLibrary( libName ) )
|
|
|
|
prj.SetRString( PROJECT::SCH_LIB_SELECT, libName );
|
|
|
|
else
|
|
|
|
libName = wxEmptyString;
|
|
|
|
}
|
|
|
|
|
|
|
|
return libName;
|
|
|
|
}
|
2023-05-31 20:37:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
void SCH_BASE_FRAME::setSymWatcher( const LIB_ID* aID )
|
|
|
|
{
|
|
|
|
Unbind( wxEVT_FSWATCHER, &SCH_BASE_FRAME::OnSymChange, this );
|
|
|
|
|
2023-07-14 18:29:14 +00:00
|
|
|
if( m_watcher )
|
2023-05-31 20:37:58 +00:00
|
|
|
{
|
2023-07-14 18:29:14 +00:00
|
|
|
wxLogTrace( "KICAD_LIB_WATCH", "Remove watch" );
|
|
|
|
m_watcher->RemoveAll();
|
|
|
|
m_watcher->SetOwner( nullptr );
|
2023-05-31 20:37:58 +00:00
|
|
|
m_watcher.reset();
|
|
|
|
}
|
|
|
|
|
|
|
|
wxString libfullname;
|
|
|
|
SYMBOL_LIB_TABLE* tbl = Prj().SchSymbolLibTable();
|
|
|
|
|
2023-07-14 18:48:28 +00:00
|
|
|
if( !aID || !tbl )
|
2023-05-31 20:37:58 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
const SYMBOL_LIB_TABLE_ROW* row = tbl->FindRow( aID->GetLibNickname() );
|
|
|
|
|
|
|
|
if( !row )
|
|
|
|
return;
|
|
|
|
|
|
|
|
libfullname = row->GetFullURI( true );
|
|
|
|
}
|
|
|
|
catch( const std::exception& e )
|
|
|
|
{
|
|
|
|
DisplayInfoMessage( this, e.what() );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
catch( const IO_ERROR& error )
|
|
|
|
{
|
|
|
|
wxLogTrace( "KICAD_LIB_WATCH", "Error: %s", error.What() );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wxLogTrace( "KICAD_LIB_WATCH", "Setting up watcher for %s", libfullname );
|
|
|
|
m_watcherFileName.Assign( libfullname );
|
|
|
|
|
|
|
|
if( !m_watcherFileName.FileExists() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
wxLog::EnableLogging( false );
|
|
|
|
m_watcherLastModified = m_watcherFileName.GetModificationTime();
|
|
|
|
wxLog::EnableLogging( true );
|
|
|
|
|
|
|
|
Bind( wxEVT_FSWATCHER, &SCH_BASE_FRAME::OnSymChange, this );
|
|
|
|
m_watcher = std::make_unique<wxFileSystemWatcher>();
|
|
|
|
m_watcher->SetOwner( this );
|
|
|
|
|
|
|
|
wxFileName fn;
|
|
|
|
fn.AssignDir( m_watcherFileName.GetPath() );
|
|
|
|
fn.DontFollowLink();
|
|
|
|
|
|
|
|
m_watcher->AddTree( fn );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SCH_BASE_FRAME::OnSymChange( wxFileSystemWatcherEvent& aEvent )
|
|
|
|
{
|
|
|
|
SYMBOL_LIBS* libs = Prj().SchLibs();
|
|
|
|
|
|
|
|
wxLogTrace( "KICAD_LIB_WATCH", "OnSymChange: %s, watcher file: %s",
|
|
|
|
aEvent.GetPath().GetFullPath(), m_watcherFileName.GetFullPath() );
|
|
|
|
|
|
|
|
if( !libs || !m_watcher || !m_watcher.get() || m_watcherFileName.GetPath().IsEmpty() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
if( aEvent.GetPath() != m_watcherFileName )
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Start the debounce timer (set to 1 second)
|
|
|
|
if( !m_watcherDebounceTimer.StartOnce( 1000 ) )
|
|
|
|
{
|
|
|
|
wxLogTrace( "KICAD_LIB_WATCH", "Failed to start the debounce timer" );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SCH_BASE_FRAME::OnSymChangeDebounceTimer( wxTimerEvent& aEvent )
|
|
|
|
{
|
|
|
|
wxLogTrace( "KICAD_LIB_WATCH", "OnSymChangeDebounceTimer" );
|
|
|
|
// Disable logging to avoid spurious messages and check if the file has changed
|
|
|
|
wxLog::EnableLogging( false );
|
|
|
|
wxDateTime lastModified = m_watcherFileName.GetModificationTime();
|
|
|
|
wxLog::EnableLogging( true );
|
|
|
|
|
|
|
|
if( lastModified == m_watcherLastModified || !lastModified.IsValid() )
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_watcherLastModified = lastModified;
|
|
|
|
|
|
|
|
if( !GetScreen()->IsContentModified() || IsOK( this, _( "The library containing the current symbol has changed.\n"
|
|
|
|
"Do you want to reload the library?" ) ) )
|
|
|
|
{
|
|
|
|
wxLogTrace( "KICAD_LIB_WATCH", "Sending refresh symbol mail" );
|
|
|
|
std::string libName = m_watcherFileName.GetFullPath().ToStdString();
|
|
|
|
Kiway().ExpressMail( FRAME_SCH_VIEWER, MAIL_REFRESH_SYMBOL, libName );
|
|
|
|
Kiway().ExpressMail( FRAME_SCH_SYMBOL_EDITOR, MAIL_REFRESH_SYMBOL, libName );
|
|
|
|
}
|
|
|
|
}
|