2022-03-09 02:40:59 +01:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2022 Mikolaj Wielgus
|
|
|
|
* Copyright (C) 2022 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, you may find one here:
|
|
|
|
* https://www.gnu.org/licenses/gpl-3.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 3 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <sim/sim_property.h>
|
|
|
|
#include <sim/sim_value.h>
|
|
|
|
#include <ki_exception.h>
|
|
|
|
#include <wx/combo.h>
|
|
|
|
|
|
|
|
|
|
|
|
wxBEGIN_EVENT_TABLE( SIM_VALIDATOR, wxValidator )
|
2022-06-12 05:39:13 +02:00
|
|
|
EVT_KEY_DOWN( SIM_VALIDATOR::onKeyDown )
|
2022-03-09 02:40:59 +01:00
|
|
|
wxEND_EVENT_TABLE()
|
|
|
|
|
|
|
|
|
2022-06-12 05:39:13 +02:00
|
|
|
void SIM_VALIDATOR::navigate( int flags )
|
|
|
|
{
|
|
|
|
wxWindow* textCtrl = GetWindow();
|
|
|
|
if( !textCtrl )
|
|
|
|
return;
|
|
|
|
|
|
|
|
wxPropertyGrid* paramGrid = dynamic_cast<wxPropertyGrid*>( textCtrl->GetParent() );
|
|
|
|
if( !paramGrid )
|
|
|
|
return;
|
|
|
|
|
|
|
|
wxPropertyGridManager* paramGridMgr =
|
|
|
|
dynamic_cast<wxPropertyGridManager*>( paramGrid->GetParent() );
|
|
|
|
if( !paramGridMgr )
|
|
|
|
return;
|
|
|
|
|
|
|
|
#ifdef __WXGTK__
|
|
|
|
// Workaround for wxWindow::Navigate() working differently on GTK. Same workaround is
|
|
|
|
// in the WxWidgets source code.
|
|
|
|
if( flags == wxNavigationKeyEvent::IsBackward )
|
|
|
|
{
|
|
|
|
if( wxWindow* sibling = paramGridMgr->GetPrevSibling() )
|
|
|
|
{
|
|
|
|
sibling->SetFocusFromKbd();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if( flags == wxNavigationKeyEvent::IsForward )
|
|
|
|
{
|
|
|
|
if( wxWindow* sibling = paramGridMgr->GetNextSibling() )
|
|
|
|
{
|
|
|
|
sibling->SetFocusFromKbd();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We didn't find the sibling, so instead we try another workaround by by finding the notebook
|
|
|
|
// we are in, and jumping out of it.
|
|
|
|
for( wxWindow* window = paramGridMgr; window; window = window->GetParent() )
|
|
|
|
{
|
|
|
|
if( wxNotebook* notebook = dynamic_cast<wxNotebook*>( window ) )
|
|
|
|
{
|
|
|
|
if( flags == wxNavigationKeyEvent::IsBackward )
|
|
|
|
{
|
|
|
|
for( wxWindow* sibling = notebook->GetNextSibling();
|
|
|
|
sibling;
|
|
|
|
sibling = sibling->GetNextSibling() )
|
|
|
|
{
|
|
|
|
if( sibling->IsFocusable() )
|
|
|
|
{
|
|
|
|
sibling->SetFocusFromKbd();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if( flags == wxNavigationKeyEvent::IsForward )
|
|
|
|
{
|
|
|
|
for( wxWindow* sibling = notebook->GetNextSibling();
|
|
|
|
sibling;
|
|
|
|
sibling = sibling->GetNextSibling() )
|
|
|
|
{
|
|
|
|
if( sibling->IsFocusable() )
|
|
|
|
{
|
|
|
|
sibling->SetFocusFromKbd();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
paramGridMgr->Navigate( flags );
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SIM_VALIDATOR::onKeyDown( wxKeyEvent& aEvent )
|
|
|
|
{
|
|
|
|
// Because wxPropertyGrid has special handling for the tab key, wxPropertyGrid::DedicateKey()
|
|
|
|
// and wxPropertyGrid::AddActionTrigger() don't work for it. So instead we translate it to an
|
|
|
|
// (up or down) arrow key, which has proper handling (select next or previous property) defined
|
|
|
|
// by the aforementioned functions.
|
2022-08-12 10:22:33 +02:00
|
|
|
|
2022-06-12 05:39:13 +02:00
|
|
|
if( aEvent.GetKeyCode() == WXK_TAB )
|
|
|
|
{
|
|
|
|
// However, before that, if this is the first or last property, we instead want to navigate
|
|
|
|
// to the next or previous widget.
|
|
|
|
wxWindow* textCtrl = GetWindow();
|
|
|
|
if( !textCtrl )
|
|
|
|
{
|
|
|
|
aEvent.Skip();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wxPropertyGrid* paramGrid = dynamic_cast<wxPropertyGrid*>( textCtrl->GetParent() );
|
|
|
|
if( !paramGrid )
|
|
|
|
{
|
|
|
|
aEvent.Skip();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wxPropertyGridIterator it = paramGrid->GetIterator( wxPG_ITERATE_VISIBLE,
|
|
|
|
paramGrid->GetSelection() );
|
|
|
|
if( !it.AtEnd() )
|
|
|
|
it.Next();
|
|
|
|
|
|
|
|
bool isFirst = paramGrid->GetSelection() == paramGrid->wxPropertyGridInterface::GetFirst();
|
|
|
|
bool isLast = it.AtEnd();
|
|
|
|
|
|
|
|
if( isFirst && aEvent.ShiftDown() )
|
|
|
|
{
|
|
|
|
navigate( wxNavigationKeyEvent::IsBackward );
|
|
|
|
return;
|
|
|
|
}
|
2022-08-12 10:22:33 +02:00
|
|
|
|
2022-06-12 05:39:13 +02:00
|
|
|
if( isLast && !aEvent.ShiftDown() )
|
|
|
|
{
|
|
|
|
navigate( wxNavigationKeyEvent::IsForward );
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if( aEvent.GetModifiers() == wxMOD_SHIFT )
|
|
|
|
{
|
|
|
|
aEvent.m_shiftDown = false;
|
|
|
|
aEvent.m_keyCode = WXK_UP;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
aEvent.m_keyCode = WXK_DOWN;
|
|
|
|
}
|
|
|
|
|
|
|
|
aEvent.Skip();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-10-31 22:31:29 +00:00
|
|
|
bool SIM_VALIDATOR::Validate( wxWindow* aParent )
|
2022-06-21 04:22:52 +02:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-09-11 15:22:45 +02:00
|
|
|
SIM_PROPERTY::SIM_PROPERTY( std::shared_ptr<SIM_LIBRARY> aLibrary,
|
|
|
|
std::shared_ptr<SIM_MODEL> aModel,
|
|
|
|
int aParamIndex )
|
|
|
|
: m_library( std::move( aLibrary ) ),
|
|
|
|
m_model( std::move( aModel ) ),
|
|
|
|
m_paramIndex( aParamIndex )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-06-21 04:22:52 +02:00
|
|
|
SIM_BOOL_PROPERTY::SIM_BOOL_PROPERTY( const wxString& aLabel, const wxString& aName,
|
|
|
|
std::shared_ptr<SIM_LIBRARY> aLibrary,
|
|
|
|
std::shared_ptr<SIM_MODEL> aModel,
|
|
|
|
int aParamIndex )
|
|
|
|
: wxBoolProperty( aLabel, aName ),
|
2022-09-11 15:22:45 +02:00
|
|
|
SIM_PROPERTY( aLibrary, aModel, aParamIndex )
|
2022-06-21 04:22:52 +02:00
|
|
|
{
|
|
|
|
auto simValue = dynamic_cast<SIM_VALUE_INST<bool>*>(
|
|
|
|
m_model->GetParam( m_paramIndex ).value.get() );
|
|
|
|
|
|
|
|
wxCHECK( simValue, /*void*/ );
|
|
|
|
|
|
|
|
SetValue( *simValue == true );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
wxValidator* SIM_BOOL_PROPERTY::DoGetValidator() const
|
|
|
|
{
|
2022-10-31 22:31:29 +00:00
|
|
|
return new SIM_VALIDATOR();
|
2022-06-21 04:22:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SIM_BOOL_PROPERTY::OnSetValue()
|
|
|
|
{
|
|
|
|
wxPGProperty::OnSetValue();
|
|
|
|
|
|
|
|
auto simValue = dynamic_cast<SIM_VALUE_INST<bool>*>(
|
|
|
|
m_model->GetParam( m_paramIndex ).value.get() );
|
|
|
|
|
|
|
|
wxCHECK( simValue, /*void*/ );
|
|
|
|
|
|
|
|
if( m_model->GetBaseModel() && *simValue == m_value.GetBool() )
|
|
|
|
m_model->SetParamValue( m_paramIndex, "" );
|
|
|
|
else
|
|
|
|
m_model->SetParamValue( m_paramIndex, m_value.GetBool() ? "1" : "0" );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SIM_STRING_PROPERTY::SIM_STRING_PROPERTY( const wxString& aLabel, const wxString& aName,
|
|
|
|
std::shared_ptr<SIM_LIBRARY> aLibrary,
|
|
|
|
std::shared_ptr<SIM_MODEL> aModel,
|
2022-10-31 22:31:29 +00:00
|
|
|
int aParamIndex )
|
2022-04-01 06:30:50 +02:00
|
|
|
: wxStringProperty( aLabel, aName ),
|
2022-10-31 22:31:29 +00:00
|
|
|
SIM_PROPERTY( aLibrary, aModel, aParamIndex )
|
2022-03-09 02:40:59 +01:00
|
|
|
{
|
2022-10-31 22:31:29 +00:00
|
|
|
const SIM_MODEL::PARAM& param = GetParam();
|
|
|
|
|
|
|
|
wxASSERT( !param.resolved );
|
|
|
|
|
|
|
|
SetValueFromString( param.source );
|
2022-03-09 02:40:59 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-06-21 04:22:52 +02:00
|
|
|
wxValidator* SIM_STRING_PROPERTY::DoGetValidator() const
|
2022-03-09 02:40:59 +01:00
|
|
|
{
|
2022-10-31 22:31:29 +00:00
|
|
|
return new SIM_VALIDATOR();
|
2022-03-09 02:40:59 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-08-30 09:00:52 +02:00
|
|
|
bool SIM_STRING_PROPERTY::StringToValue( wxVariant& aVariant, const wxString& aText,
|
|
|
|
int aArgFlags ) const
|
2022-03-09 02:40:59 +01:00
|
|
|
{
|
2022-08-30 09:00:52 +02:00
|
|
|
wxString baseParamValue = m_model->GetBaseParam( m_paramIndex ).value->ToString();
|
2022-04-12 16:37:06 +02:00
|
|
|
aVariant = aText;
|
|
|
|
|
|
|
|
// TODO: Don't use string comparison.
|
2022-08-30 09:00:52 +02:00
|
|
|
if( m_model->GetBaseModel() && ( aText == "" || aText == baseParamValue ) )
|
2022-03-09 02:40:59 +01:00
|
|
|
{
|
2022-10-31 22:31:29 +00:00
|
|
|
// TODO: do we want this magic of clearing overrides?
|
|
|
|
// Consider the case where someone uses a library model set to 220u and overrides it to
|
|
|
|
// 330u. Someone then modifies the library to use 330u. But that doesn't work either,
|
|
|
|
// so they modify the library again to 470u. If the overridden symbol was edited in the
|
|
|
|
// middle (to set some other parameter perhaps), it suddenly gets changed to 470u, which
|
|
|
|
// will be a surprise to the user.
|
|
|
|
// NOTE: other properties also contain this magic.
|
2022-10-12 04:26:16 +02:00
|
|
|
try
|
|
|
|
{
|
|
|
|
m_model->SetParamValue( m_paramIndex, "" ); // Nullify.
|
|
|
|
}
|
|
|
|
catch( const IO_ERROR& )
|
|
|
|
{
|
2022-04-12 16:37:06 +02:00
|
|
|
return false;
|
2022-10-12 04:26:16 +02:00
|
|
|
}
|
2022-04-12 16:37:06 +02:00
|
|
|
|
2022-08-30 09:00:52 +02:00
|
|
|
aVariant = baseParamValue; // Use the inherited value (if it exists) if null.
|
2022-03-09 02:40:59 +01:00
|
|
|
}
|
2022-04-12 16:37:06 +02:00
|
|
|
else
|
2022-03-09 02:40:59 +01:00
|
|
|
{
|
2022-10-31 22:31:29 +00:00
|
|
|
m_model->SetParamSource( m_paramIndex, aText );
|
|
|
|
aVariant = aText;
|
2022-03-09 02:40:59 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2022-08-30 09:00:52 +02:00
|
|
|
|
|
|
|
|
2022-09-22 07:38:45 +02:00
|
|
|
static wxArrayString convertStringsToWx( const std::vector<std::string>& aStrings )
|
|
|
|
{
|
|
|
|
wxArrayString result;
|
|
|
|
|
|
|
|
for( const std::string& string : aStrings )
|
|
|
|
result.Add( string );
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-08-30 09:00:52 +02:00
|
|
|
SIM_ENUM_PROPERTY::SIM_ENUM_PROPERTY( const wxString& aLabel, const wxString& aName,
|
|
|
|
std::shared_ptr<SIM_LIBRARY> aLibrary,
|
|
|
|
std::shared_ptr<SIM_MODEL> aModel,
|
2022-10-31 22:31:29 +00:00
|
|
|
int aParamIndex )
|
2022-08-30 09:00:52 +02:00
|
|
|
: wxEnumProperty( aLabel, aName,
|
2022-09-22 07:38:45 +02:00
|
|
|
convertStringsToWx( aModel->GetParam( aParamIndex ).info.enumValues ) ),
|
2022-09-11 15:22:45 +02:00
|
|
|
SIM_PROPERTY( aLibrary, aModel, aParamIndex )
|
2022-08-30 09:00:52 +02:00
|
|
|
{
|
|
|
|
auto it = std::find( GetParam().info.enumValues.begin(), GetParam().info.enumValues.end(),
|
|
|
|
GetParam().value->ToString() );
|
2022-08-30 07:24:58 -04:00
|
|
|
|
|
|
|
// we need the force cast for msvc because wxVariant lacks 64-bit methods due to `long`
|
|
|
|
SetValue( static_cast<int>( std::distance( GetParam().info.enumValues.begin(), it ) ) );
|
2022-08-30 09:00:52 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool SIM_ENUM_PROPERTY::IntToValue( wxVariant& aVariant, int aNumber, int aArgFlags ) const
|
|
|
|
{
|
|
|
|
m_model->SetParamValue( m_paramIndex, GetParam().info.enumValues.at( aNumber ) );
|
|
|
|
return wxEnumProperty::IntToValue( aVariant, aNumber, aArgFlags );
|
|
|
|
}
|