Browse Source
Add a class to govern an "incrementing" text control
Add a class to govern an "incrementing" text control
This class, INCREMENTAL_TEXT_CTRL, is used to provide a frameworks for classes that want to control a text control that can be incremented according to some scheme. Also provided is a wxWidgets implementation with spin buttons for control, as well as mousewheel support.pull/3/merge
committed by
Maciej Suminski
5 changed files with 375 additions and 1 deletions
-
1common/CMakeLists.txt
-
184common/incremental_text_ctrl.cpp
-
188include/incremental_text_ctrl.h
-
1pcbnew/dialogs/dialog_display_options.cpp
-
2pcbnew/dialogs/dialog_display_options.h
@ -0,0 +1,184 @@ |
|||
/*
|
|||
* This program source code file is part of KiCad, a free EDA CAD application. |
|||
* |
|||
* Copyright (C) 2015 Jean-Pierre Charras, jean-pierre.charras at wanadoo.fr |
|||
* Copyright (C) 1992-2016 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 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 |
|||
*/ |
|||
|
|||
#include <incremental_text_ctrl.h>
|
|||
|
|||
/**
|
|||
* Check that a string looks like a floating point number that can |
|||
* be dealt with. |
|||
*/ |
|||
static bool validateFloatField( const wxString& aStr ) |
|||
{ |
|||
// Skip empty fields
|
|||
if( aStr.size() == 0 ) |
|||
return false; |
|||
|
|||
// a single . or , doesn't count as number, although valid in a float
|
|||
if( aStr.size() == 1 ) |
|||
{ |
|||
if( (aStr.compare( "." ) == 0) || |
|||
(aStr.compare( "," ) == 0) ) |
|||
return false; |
|||
} |
|||
|
|||
return true; |
|||
} |
|||
|
|||
|
|||
INCREMENTAL_TEXT_CTRL::INCREMENTAL_TEXT_CTRL() : |
|||
m_minVal( 0.0 ), |
|||
m_maxVal( 1.0 ), |
|||
m_currentValue( 0.0 ), |
|||
m_precision( 4 ) |
|||
{} |
|||
|
|||
|
|||
void INCREMENTAL_TEXT_CTRL::SetStep( double aMin, double aMax, |
|||
STEP_FUNCTION aStepFunc ) |
|||
{ |
|||
wxASSERT( aMin <= aMax ); |
|||
|
|||
m_minVal = std::min( aMin, aMax ); |
|||
m_maxVal = std::max( aMin, aMax ); |
|||
m_stepFunc = aStepFunc; |
|||
|
|||
// finally, clamp the current value and re-display
|
|||
updateTextValue(); |
|||
} |
|||
|
|||
|
|||
void INCREMENTAL_TEXT_CTRL::updateTextValue() |
|||
{ |
|||
if( m_currentValue > m_maxVal ) |
|||
m_currentValue = m_maxVal; |
|||
|
|||
if( m_currentValue < m_minVal ) |
|||
m_currentValue = m_minVal; |
|||
|
|||
wxString fmt = wxString::Format( "%%.%df", m_precision ); |
|||
setTextCtrl( wxString::Format( fmt, m_currentValue ) ); |
|||
} |
|||
|
|||
|
|||
void INCREMENTAL_TEXT_CTRL::incrementCtrlBy( double aInc ) |
|||
{ |
|||
const wxString txt = getCtrlText(); |
|||
if( !validateFloatField( txt ) ) |
|||
return; |
|||
|
|||
txt.ToDouble( &m_currentValue ); |
|||
m_currentValue += aInc; |
|||
|
|||
updateTextValue(); |
|||
} |
|||
|
|||
|
|||
void INCREMENTAL_TEXT_CTRL::incrementCtrl( bool aUp ) |
|||
{ |
|||
incrementCtrlBy( m_stepFunc( aUp, m_currentValue ) ); |
|||
} |
|||
|
|||
|
|||
void INCREMENTAL_TEXT_CTRL::SetPrecision( int aPrecision ) |
|||
{ |
|||
m_precision = aPrecision; |
|||
} |
|||
|
|||
|
|||
void INCREMENTAL_TEXT_CTRL::SetValue( double aValue ) |
|||
{ |
|||
m_currentValue = aValue; |
|||
updateTextValue(); |
|||
} |
|||
|
|||
|
|||
double INCREMENTAL_TEXT_CTRL::GetValue() |
|||
{ |
|||
// sanitise before handing the value - if the user did something
|
|||
// like close a window with outstanding text changes, we need
|
|||
// to clamp the value and re-interpret the text
|
|||
incrementCtrlBy( 0.0 ); |
|||
|
|||
return m_currentValue; |
|||
} |
|||
|
|||
|
|||
SPIN_INCREMENTAL_TEXT_CTRL::SPIN_INCREMENTAL_TEXT_CTRL( wxSpinButton& aSpinBtn, |
|||
wxTextCtrl& aTextCtrl ): |
|||
m_spinBtn( aSpinBtn ), |
|||
m_textCtrl( aTextCtrl ) |
|||
{ |
|||
// set always enabled, otherwise it's very hard to keep in sync
|
|||
aSpinBtn.SetRange( -INT_MAX, INT_MAX ); |
|||
|
|||
auto spinUpHandler = [this] ( wxSpinEvent& event ) |
|||
{ |
|||
incrementCtrl( true ); |
|||
}; |
|||
|
|||
// spin up/down if a single step of the field
|
|||
auto spinDownHandler = [this] ( wxSpinEvent& event ) |
|||
{ |
|||
incrementCtrl( false ); |
|||
}; |
|||
|
|||
auto mouseWheelHandler = [this] ( wxMouseEvent& aEvent ) |
|||
{ |
|||
incrementCtrl( aEvent.GetWheelRotation() >= 0 ); |
|||
}; |
|||
|
|||
aSpinBtn.Bind( wxEVT_SPIN_UP, spinUpHandler ); |
|||
aSpinBtn.Bind( wxEVT_SPIN_DOWN, spinDownHandler ); |
|||
|
|||
m_textCtrl.Bind( wxEVT_MOUSEWHEEL, mouseWheelHandler ); |
|||
|
|||
m_textCtrl.Bind( wxEVT_KILL_FOCUS, &SPIN_INCREMENTAL_TEXT_CTRL::onFocusLoss, this ); |
|||
} |
|||
|
|||
SPIN_INCREMENTAL_TEXT_CTRL::~SPIN_INCREMENTAL_TEXT_CTRL() |
|||
{ |
|||
// this must be unbound, as kill focus can arrive after the
|
|||
// text control is gone
|
|||
m_textCtrl.Unbind( wxEVT_KILL_FOCUS, &SPIN_INCREMENTAL_TEXT_CTRL::onFocusLoss, this ); |
|||
} |
|||
|
|||
|
|||
void SPIN_INCREMENTAL_TEXT_CTRL::onFocusLoss( wxFocusEvent& aEvent ) |
|||
{ |
|||
// re-read the input and sanitize any user changes
|
|||
incrementCtrlBy( 0.0 ); |
|||
} |
|||
|
|||
|
|||
void SPIN_INCREMENTAL_TEXT_CTRL::setTextCtrl( const wxString& val ) |
|||
{ |
|||
m_textCtrl.SetValue( val ); |
|||
} |
|||
|
|||
|
|||
wxString SPIN_INCREMENTAL_TEXT_CTRL::getCtrlText() const |
|||
{ |
|||
return m_textCtrl.GetValue(); |
|||
} |
|||
|
|||
@ -0,0 +1,188 @@ |
|||
/* |
|||
* This program source code file is part of KiCad, a free EDA CAD application. |
|||
* |
|||
* Copyright (C) 2013 CERN |
|||
* @author Maciej Suminski <maciej.suminski@cern.ch> |
|||
* |
|||
* 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 |
|||
*/ |
|||
|
|||
#ifndef INCREMENTAL_TEXT_CTRL__H_ |
|||
#define INCREMENTAL_TEXT_CTRL__H_ |
|||
|
|||
#include <wx/wx.h> |
|||
#include <wx/spinbutt.h> |
|||
|
|||
#include <functional> |
|||
|
|||
/** |
|||
* Class that governs a textual control holding a number that can |
|||
* be incremented/decremented according to some scheme (often just |
|||
* a constant step). |
|||
*/ |
|||
class INCREMENTAL_TEXT_CTRL |
|||
{ |
|||
public: |
|||
|
|||
/** |
|||
* A callable object type that can be used to provide a step |
|||
* value. Client can provide one of these to use for implementing |
|||
* non-linear stepping, or stepping based on external parameters, |
|||
* such as unit selection. |
|||
* |
|||
* @param aUp true if the next step is upwards |
|||
* @param aCurrVal the current value of the control |
|||
*/ |
|||
using STEP_FUNCTION = std::function<double(bool aUp, double aCurrVal)>; |
|||
|
|||
INCREMENTAL_TEXT_CTRL(); |
|||
|
|||
virtual ~INCREMENTAL_TEXT_CTRL() {} |
|||
|
|||
/** |
|||
* Set the value of the text control, but obey the limits |
|||
* currently set. |
|||
* |
|||
* @param aValue the control value to set |
|||
*/ |
|||
void SetValue( double aValue ); |
|||
|
|||
/** |
|||
* Get the current value of the control |
|||
*/ |
|||
double GetValue(); |
|||
|
|||
/** |
|||
* Function SetStep() |
|||
* |
|||
* Set the stepping parameters of the control. The range is |
|||
* enforced by not allowing the scroll to exceed it, and on |
|||
* loss of focus, the control is also clamped to the range. |
|||
* |
|||
* @param aMin the minium value allowed |
|||
* @param aMax the maximum value allows |
|||
* @param aNewfunc the step function used to calculate the next step |
|||
*/ |
|||
void SetStep( double aMin, double aMax, STEP_FUNCTION aNewFunc ); |
|||
|
|||
/** |
|||
* Function SetStep() |
|||
* |
|||
* Shortcut method to set step parameters when the step is constant |
|||
* |
|||
* @param aMin the minium value allowed |
|||
* @param aMax the maximum value allows |
|||
* @param aStep the constant step size |
|||
*/ |
|||
void SetStep( double aMin, double aMax, double aStep ) |
|||
{ |
|||
SetStep( aMin, aMax, |
|||
[aStep] ( bool aUp, double aCurrent ) { return aUp ? aStep : -aStep; } ); |
|||
} |
|||
|
|||
/** |
|||
* Set the number of decimal places to display |
|||
*/ |
|||
void SetPrecision( int aPrecision ); |
|||
|
|||
protected: |
|||
|
|||
/** |
|||
* Increment the control by the given amount |
|||
*/ |
|||
void incrementCtrlBy( double aInc); |
|||
|
|||
/** |
|||
* Single increment up or down by one step |
|||
*/ |
|||
void incrementCtrl( bool aUp ); |
|||
|
|||
/** |
|||
* Update the text control value with the current value, |
|||
* clamping to limits as needed |
|||
*/ |
|||
void updateTextValue(); |
|||
|
|||
/* |
|||
* Implementation-specific interfaces |
|||
*/ |
|||
|
|||
/** |
|||
* Set the text control string value after an increment. |
|||
*/ |
|||
virtual void setTextCtrl( const wxString& aVal ) = 0; |
|||
|
|||
/** |
|||
* @return the current string value of the text control |
|||
*/ |
|||
virtual wxString getCtrlText() const = 0; |
|||
|
|||
private: |
|||
|
|||
double m_minVal; |
|||
double m_maxVal; |
|||
|
|||
///< Current value of the control |
|||
double m_currentValue; |
|||
|
|||
///< Precision to display |
|||
int m_precision; |
|||
|
|||
///< The function used to determine the step |
|||
STEP_FUNCTION m_stepFunc; |
|||
}; |
|||
|
|||
|
|||
/** |
|||
* Class SPIN_INCREMENTING_TEXT_CTRL |
|||
* |
|||
* An incrementable text control, with WX spin buttons for clickable |
|||
* control. |
|||
*/ |
|||
class SPIN_INCREMENTAL_TEXT_CTRL: public INCREMENTAL_TEXT_CTRL |
|||
{ |
|||
public: |
|||
|
|||
/** |
|||
* Constructor |
|||
* |
|||
* @param aSpinBtn the spin button to control the value |
|||
* @param aTextCtrl the text control that will display the value |
|||
*/ |
|||
SPIN_INCREMENTAL_TEXT_CTRL( wxSpinButton& aSpinBtn, |
|||
wxTextCtrl& aTextCtrl ); |
|||
|
|||
~SPIN_INCREMENTAL_TEXT_CTRL(); |
|||
|
|||
protected: |
|||
|
|||
///> @copydoc INCREMENTAL_TEXT_CTRL::setTextCtrl() |
|||
void setTextCtrl( const wxString& val ) override; |
|||
|
|||
///> @copydoc INCREMENTAL_TEXT_CTRL::getCtrlText() |
|||
wxString getCtrlText() const override; |
|||
|
|||
private: |
|||
|
|||
void onFocusLoss( wxFocusEvent& aEvent ); |
|||
|
|||
wxSpinButton& m_spinBtn; |
|||
wxTextCtrl& m_textCtrl; |
|||
}; |
|||
|
|||
#endif /* INCREMENTAL_TEXT_CTRL__H_ */ |
|||
Write
Preview
Loading…
Cancel
Save
Reference in new issue