KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
unit_binder.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) 2014-2015 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 * Author: Maciej Suminski <maciej.suminski@cern.ch>
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version 2
11 * of the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, you may find one here:
20 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
21 * or you may search the http://www.gnu.org website for the version 2 license,
22 * or you may write to the Free Software Foundation, Inc.,
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
24 */
25
26#include <wx/clipbrd.h>
27#include <wx/combobox.h>
28#include <wx/stattext.h>
29#include <wx/textentry.h>
30#include <eda_units.h>
31#include <eda_draw_frame.h>
32#include <confirm.h>
33
34#include "widgets/unit_binder.h"
35#include "wx/dcclient.h"
36
37
38wxDEFINE_EVENT( DELAY_FOCUS, wxCommandEvent );
39
40
41UNIT_BINDER::UNIT_BINDER( EDA_DRAW_FRAME* aParent, wxStaticText* aLabel, wxWindow* aValueCtrl,
42 wxStaticText* aUnitLabel, bool allowEval, bool aBindFrameEvents ) :
43 UNIT_BINDER( aParent, aParent, aLabel, aValueCtrl, aUnitLabel, allowEval, aBindFrameEvents )
44{
45}
46
47UNIT_BINDER::UNIT_BINDER( UNITS_PROVIDER* aUnitsProvider, wxWindow* aEventSource,
48 wxStaticText* aLabel, wxWindow* aValueCtrl, wxStaticText* aUnitLabel,
49 bool aAllowEval, bool aBindFocusEvent ) :
50 m_bindFocusEvent( aBindFocusEvent ),
51 m_label( aLabel ),
52 m_valueCtrl( aValueCtrl ),
53 m_eventSource( aEventSource ),
54 m_unitLabel( aUnitLabel ),
55 m_iuScale( &aUnitsProvider->GetIuScale() ),
56 m_negativeZero( false ),
57 m_dataType( EDA_DATA_TYPE::DISTANCE ),
58 m_precision( 0 ),
59 m_eval( aUnitsProvider->GetUserUnits() ),
60 m_unitsInValue( false ),
61 m_originTransforms( aUnitsProvider->GetOriginTransforms() ),
62 m_coordType( ORIGIN_TRANSFORMS::NOT_A_COORD )
63{
64 init( aUnitsProvider );
65 m_allowEval = aAllowEval && ( !m_valueCtrl || dynamic_cast<wxTextEntry*>( m_valueCtrl ) );
66 wxTextEntry* textEntry = dynamic_cast<wxTextEntry*>( m_valueCtrl );
67
68 if( textEntry )
69 {
70 wxClientDC dc( m_valueCtrl );
71
72 // Gives enough room to display a value in inches in textEntry
73 // 3 digits + '.' + 10 digits
74 wxSize minSize = m_valueCtrl->GetMinSize();
75 int minWidth = dc.GetTextExtent( wxT( "XXX.XXXXXXXXXX" ) ).GetWidth();
76
77 if( minSize.GetWidth() < minWidth )
78 m_valueCtrl->SetMinSize( wxSize( minWidth, minSize.GetHeight() ) );
79
80 // Use ChangeValue() instead of SetValue() so we don't generate events.
81 if( m_negativeZero )
82 textEntry->ChangeValue( wxT( "-0" ) );
83 else
84 textEntry->ChangeValue( wxT( "0" ) );
85 }
86
87 if( m_unitLabel )
89
90 if( m_valueCtrl )
91 {
92 m_valueCtrl->Connect( wxEVT_SET_FOCUS, wxFocusEventHandler( UNIT_BINDER::onSetFocus ), nullptr, this );
93 m_valueCtrl->Connect( wxEVT_KILL_FOCUS, wxFocusEventHandler( UNIT_BINDER::onKillFocus ), nullptr, this );
94 m_valueCtrl->Connect( wxEVT_LEFT_UP, wxMouseEventHandler( UNIT_BINDER::onClick ), nullptr, this );
95 m_valueCtrl->Connect( wxEVT_COMBOBOX, wxCommandEventHandler( UNIT_BINDER::onComboBox ), nullptr, this );
96 }
97
99 Connect( DELAY_FOCUS, wxCommandEventHandler( UNIT_BINDER::delayedFocusHandler ), nullptr, this );
100
101 if( m_eventSource )
102 {
103 m_eventSource->Connect( EDA_EVT_UNITS_CHANGED, wxCommandEventHandler( UNIT_BINDER::onUnitsChanged ),
104 nullptr, this );
105 }
106}
107
108
110{
111 if( m_valueCtrl )
112 {
113 m_valueCtrl->Disconnect( wxEVT_SET_FOCUS, wxFocusEventHandler( UNIT_BINDER::onSetFocus ), nullptr, this );
114 m_valueCtrl->Disconnect( wxEVT_KILL_FOCUS, wxFocusEventHandler( UNIT_BINDER::onKillFocus ), nullptr, this );
115 m_valueCtrl->Disconnect( wxEVT_LEFT_UP, wxMouseEventHandler( UNIT_BINDER::onClick ), nullptr, this );
116 m_valueCtrl->Disconnect( wxEVT_COMBOBOX, wxCommandEventHandler( UNIT_BINDER::onComboBox ), nullptr, this );
117 }
118
119 if( m_bindFocusEvent )
120 Disconnect( DELAY_FOCUS, wxCommandEventHandler( UNIT_BINDER::delayedFocusHandler ), nullptr, this );
121
122 if( m_eventSource )
123 {
124 m_eventSource->Disconnect( EDA_EVT_UNITS_CHANGED, wxCommandEventHandler( UNIT_BINDER::onUnitsChanged ),
125 nullptr, this );
126 }
127}
128
129
131{
132 m_units = aProvider->GetUserUnits();
133 m_needsEval = false;
134 m_selStart = 0;
135 m_selEnd = 0;
136}
137
138
140{
141 m_units = aUnits;
142
144 m_eval.LocaleChanged(); // In case locale changed since last run
145
146 if( m_unitLabel )
148}
149
150
151void UNIT_BINDER::SetPrecision( int aLength )
152{
153 m_precision = std::min( aLength, 6 );
154}
155
156
158{
159 m_dataType = aDataType;
160
161 if( m_unitLabel )
163}
164
165
166void UNIT_BINDER::onUnitsChanged( wxCommandEvent& aEvent )
167{
168 EDA_BASE_FRAME* provider = static_cast<EDA_BASE_FRAME*>( aEvent.GetClientData() );
169
170 if( m_units != EDA_UNITS::UNSCALED
171 && m_units != EDA_UNITS::DEGREES
172 && m_units != EDA_UNITS::PERCENT )
173 {
174 int temp = GetIntValue();
175
176 wxComboBox* const combo = dynamic_cast<wxComboBox*>( m_valueCtrl );
177 std::vector<long long int> comboValues;
178
179 // Read out the current values
180 if( combo )
181 {
182 for( unsigned int i = 0; i < combo->GetCount(); i++ )
183 {
184 const wxString value = combo->GetString( i );
186 value, m_dataType );
187 comboValues.push_back( conv );
188 }
189 }
190
191 SetUnits( provider->GetUserUnits() );
192 m_iuScale = &provider->GetIuScale();
193
194 // Re-populate the combo box with updated values
195 if( combo )
196 {
197 SetOptionsList( comboValues );
198 }
199
200 if( !IsIndeterminate() )
201 SetValue( temp );
202 }
203
204 aEvent.Skip();
205}
206
207
208void UNIT_BINDER::onClick( wxMouseEvent& aEvent )
209{
210 wxTextEntry* textEntry = dynamic_cast<wxTextEntry*>( m_valueCtrl );
211
212 if( textEntry && ( textEntry->GetValue() == INDETERMINATE_ACTION
213 || textEntry->GetValue() == INDETERMINATE_STATE ) )
214 {
215 // These are tokens, not strings, so do a select all
216 textEntry->SelectAll();
217 }
218
219 // Needed at least on Windows to avoid hanging
220 aEvent.Skip();
221}
222
223
224void UNIT_BINDER::onComboBox( wxCommandEvent& aEvent )
225{
226 wxComboBox* combo = dynamic_cast<wxComboBox*>( m_valueCtrl );
227 wxCHECK( combo, /*void*/ );
228
229 const wxString value = combo->GetStringSelection();
230 const long long int conv =
232
233 SetValue( conv );
234
235 aEvent.Skip();
236}
237
238
239void UNIT_BINDER::onSetFocus( wxFocusEvent& aEvent )
240{
241 wxTextEntry* textEntry = dynamic_cast<wxTextEntry*>( m_valueCtrl );
242
243 if( textEntry )
244 {
245 if( m_allowEval )
246 {
247 wxString oldStr = m_eval.OriginalText();
248
249 if( oldStr.length() && oldStr != textEntry->GetValue() )
250 {
251 textEntry->ChangeValue( oldStr );
252 textEntry->SetSelection( m_selStart, m_selEnd );
253 }
254
255 m_needsEval = true;
256 }
257
258 if( textEntry->GetValue() == INDETERMINATE_ACTION
259 || textEntry->GetValue() == INDETERMINATE_STATE )
260 {
261 // These are tokens, not strings, so do a select all
262 textEntry->SelectAll();
263 }
264 }
265
266 aEvent.Skip();
267}
268
269
270void UNIT_BINDER::onKillFocus( wxFocusEvent& aEvent )
271{
272 wxTextEntry* textEntry = dynamic_cast<wxTextEntry*>( m_valueCtrl );
273
274 if( m_allowEval && textEntry )
275 {
276 wxString value = textEntry->GetValue();
277 bool success = m_eval.Process( value );
278
279 if( success && !value.IsEmpty() )
280 {
281 textEntry->GetSelection( &m_selStart, &m_selEnd );
282
283 value = m_eval.Result();
284
285 if( m_unitsInValue && !value.IsEmpty() )
286 {
287 if( !( m_units == EDA_UNITS::DEGREES || m_units == EDA_UNITS::PERCENT ) )
288 value += wxT( " " );
289
291 }
292
293 textEntry->ChangeValue( value );
294
295#ifdef __WXGTK__
296 // Manually copy the selected text to the primary selection clipboard
297 if( wxTheClipboard->Open() )
298 {
299 wxString sel = textEntry->GetStringSelection();
300 bool clipTarget = wxTheClipboard->IsUsingPrimarySelection();
301 wxTheClipboard->UsePrimarySelection( true );
302 wxTheClipboard->SetData( new wxTextDataObject( sel ) );
303 wxTheClipboard->UsePrimarySelection( clipTarget );
304 wxTheClipboard->Close();
305 }
306#endif
307 }
308
309 m_needsEval = false;
310 }
311
312 aEvent.Skip();
313}
314
315
316wxString valueDescriptionFromLabel( wxStaticText* aLabel )
317{
318 wxString desc = aLabel->GetLabel();
319
320 desc.EndsWith( wxT( ":" ), &desc );
321 return desc;
322}
323
324
325void UNIT_BINDER::delayedFocusHandler( wxCommandEvent& )
326{
327 if( !m_errorMessage.IsEmpty() )
329
330 m_errorMessage = wxEmptyString;
331 m_valueCtrl->SetFocus();
332}
333
334
335bool UNIT_BINDER::Validate( double aMin, double aMax, EDA_UNITS aUnits )
336{
337 wxTextEntry* textEntry = dynamic_cast<wxTextEntry*>( m_valueCtrl );
338
339 if( !textEntry
340 || textEntry->GetValue() == INDETERMINATE_ACTION
341 || textEntry->GetValue() == INDETERMINATE_STATE )
342 {
343 return true;
344 }
345
346 // TODO: Validate() does not currently support m_dataType being anything other than DISTANCE
347 // Note: aMin and aMax are not always given in internal units
348 if( GetValue() < EDA_UNIT_UTILS::UI::FromUserUnit( *m_iuScale, aUnits, aMin ) )
349 {
350 double val_min_iu = EDA_UNIT_UTILS::UI::FromUserUnit( *m_iuScale, aUnits, aMin );
351 m_errorMessage = wxString::Format( _( "%s must be at least %s." ),
354 val_min_iu,
355 true ) );
356
357 textEntry->SelectAll();
358
359 // Don't focus directly; we might be inside a KillFocus event handler
360 wxPostEvent( this, wxCommandEvent( DELAY_FOCUS ) );
361
362 return false;
363 }
364
365 if( GetValue() > EDA_UNIT_UTILS::UI::FromUserUnit( *m_iuScale, aUnits, aMax ) )
366 {
367 double val_max_iu = EDA_UNIT_UTILS::UI::FromUserUnit( *m_iuScale, aUnits, aMax );
368 m_errorMessage = wxString::Format( _( "%s must be less than %s." ),
371 val_max_iu,
372 true ) );
373
374 textEntry->SelectAll();
375
376 // Don't focus directly; we might be inside a KillFocus event handler
377 wxPostEvent( this, wxCommandEvent( DELAY_FOCUS ) );
378
379 return false;
380 }
381
382 return true;
383}
384
385
386void UNIT_BINDER::SetValue( long long int aValue )
387{
388 double displayValue = m_originTransforms.ToDisplay( aValue, m_coordType );
389 wxString textValue = EDA_UNIT_UTILS::UI::StringFromValue( *m_iuScale, m_units, displayValue,
390 false, m_dataType );
391
392 if( displayValue == 0 && m_negativeZero )
393 SetValue( wxT( "-" ) + textValue );
394 else
395 SetValue( textValue );
396}
397
398
399void UNIT_BINDER::SetDoubleValue( double aValue )
400{
401 double displayValue = m_originTransforms.ToDisplay( aValue, m_coordType );
403 setPrecision( displayValue, false ),
404 false, m_dataType );
405
406 if( displayValue == 0 && !std::signbit( displayValue ) && m_negativeZero )
407 SetValue( wxT( "-" ) + textValue );
408 else
409 SetValue( textValue );
410}
411
412
414{
415 SetDoubleValue( aValue.AsDegrees() );
416}
417
418
419void UNIT_BINDER::SetValue( const wxString& aValue )
420{
421 wxTextEntry* textEntry = dynamic_cast<wxTextEntry*>( m_valueCtrl );
422 wxStaticText* staticText = dynamic_cast<wxStaticText*>( m_valueCtrl );
423
424 wxString value = aValue;
425
426 if( m_unitsInValue && !value.IsEmpty() )
427 {
428 if( !( m_units == EDA_UNITS::DEGREES || m_units == EDA_UNITS::PERCENT ) )
429 value += wxT( " " );
430
432 }
433
434 if( textEntry )
435 textEntry->SetValue( value );
436 else if( staticText )
437 staticText->SetLabel( value );
438
439 if( m_allowEval )
440 m_eval.Clear();
441
442 if( m_unitLabel )
444
445}
446
447
448wxString UNIT_BINDER::getTextForValue( long long int aValue ) const
449{
450 const double displayValue = m_originTransforms.ToDisplay( aValue, m_coordType );
451 wxString textValue = EDA_UNIT_UTILS::UI::StringFromValue(
452 *m_iuScale, m_units, setPrecision( displayValue, false ), false, m_dataType );
453
454 if( displayValue == 0 && m_negativeZero )
455 textValue = wxT( "-" ) + textValue;
456
457 return textValue;
458}
459
460
461wxString UNIT_BINDER::getTextForDoubleValue( double aValue ) const
462{
463 const double displayValue = m_originTransforms.ToDisplay( aValue, m_coordType );
464 wxString textValue = EDA_UNIT_UTILS::UI::StringFromValue(
465 *m_iuScale, m_units, setPrecision( displayValue, false ), false, m_dataType );
466
467 if( displayValue == 0 && !std::signbit( displayValue ) && m_negativeZero )
468 textValue = wxT( "-" ) + textValue;
469
470 return textValue;
471}
472
473
474void UNIT_BINDER::ChangeValue( int aValue )
475{
476 ChangeValue( getTextForValue( aValue ) );
477}
478
479
481{
483}
484
485
487{
488 ChangeDoubleValue( aValue.AsDegrees() );
489}
490
491
492void UNIT_BINDER::ChangeValue( const wxString& aValue )
493{
494 wxTextEntry* textEntry = dynamic_cast<wxTextEntry*>( m_valueCtrl );
495 wxStaticText* staticText = dynamic_cast<wxStaticText*>( m_valueCtrl );
496
497 wxString value = aValue;
498
499 if( m_unitsInValue && !value.IsEmpty() )
500 {
501 if( !( m_units == EDA_UNITS::DEGREES || m_units == EDA_UNITS::PERCENT ) )
502 value += wxT( " " );
503
505 }
506
507 if( textEntry )
508 textEntry->ChangeValue( value );
509 else if( staticText )
510 staticText->SetLabel( value );
511
512 if( m_allowEval )
513 m_eval.Clear();
514
515 if( m_unitLabel )
517}
518
519
521{
522 wxTextEntry* textEntry = dynamic_cast<wxTextEntry*>( m_valueCtrl );
523 wxStaticText* staticText = dynamic_cast<wxStaticText*>( m_valueCtrl );
524 wxString value;
525
526 if( textEntry )
527 {
528 value = textEntry->GetValue();
529
530 if( m_needsEval && !value.IsEmpty() && m_eval.Process( value ) )
531 value = m_eval.Result();
532 else
533 value = textEntry->GetValue();
534 }
535 else if( staticText )
536 {
537 value = staticText->GetLabel();
538 }
539 else
540 {
541 return 0;
542 }
543
544 long long int displayValue = EDA_UNIT_UTILS::UI::ValueFromString( *m_iuScale, m_units, value,
545 m_dataType );
546 return m_originTransforms.FromDisplay( displayValue, m_coordType );
547}
548
549
550double UNIT_BINDER::setPrecision( double aValue, bool aValueUsesUserUnits ) const
551{
552 if( m_precision > 1 )
553 {
554 int scale = pow( 10, m_precision );
555 int64_t tmp = aValue;
556
557 if( !aValueUsesUserUnits )
558 {
560 }
561
562 aValue = static_cast<double>( tmp ) / scale;
563
564 if( !aValueUsesUserUnits )
566 }
567
568 return aValue;
569}
570
571
573{
574 wxTextEntry* textEntry = dynamic_cast<wxTextEntry*>( m_valueCtrl );
575 wxStaticText* staticText = dynamic_cast<wxStaticText*>( m_valueCtrl );
576 wxString value;
577
578 if( textEntry )
579 {
580 value = textEntry->GetValue();
581
582 if( m_needsEval && !value.IsEmpty() && m_eval.Process( value ) )
583 value = m_eval.Result();
584 else
585 value = textEntry->GetValue();
586 }
587 else if( staticText )
588 {
589 value = staticText->GetLabel();
590 }
591 else
592 {
593 return 0.0;
594 }
595
597 value, m_dataType );
598 displayValue = setPrecision( displayValue, false );
599
600 return m_originTransforms.FromDisplay( displayValue, m_coordType );
601}
602
603
605{
607}
608
609
610void UNIT_BINDER::SetOptionsList( std::span<const long long int> aOptions )
611{
612 wxComboBox* cb = dynamic_cast<wxComboBox*>( m_valueCtrl );
613 wxCHECK( cb, /* void */ );
614
615 cb->Clear();
616
617 for( long long int value : aOptions )
618 {
619 cb->Append( getTextForValue( value ) );
620 }
621}
622
623
624void UNIT_BINDER::SetDoubleOptionsList( std::span<const double> aOptions )
625{
626 wxComboBox* cb = dynamic_cast<wxComboBox*>( m_valueCtrl );
627 wxCHECK( cb, /* void */ );
628
629 cb->Clear();
630
631 for( double value : aOptions )
632 {
633 cb->Append( getTextForDoubleValue( value ) );
634 }
635}
636
637
639{
640 wxTextEntry* te = dynamic_cast<wxTextEntry*>( m_valueCtrl );
641
642 if( te )
643 return te->GetValue() == INDETERMINATE_STATE || te->GetValue() == INDETERMINATE_ACTION;
644
645 return false;
646}
647
648
650{
651 if( wxTextEntry* te = dynamic_cast<wxTextEntry*>( m_valueCtrl ) )
652 return te->GetValue().IsEmpty();
653
654 return false;
655}
656
657
659{
660 if( wxTextEntry* te = dynamic_cast<wxTextEntry*>( m_valueCtrl ) )
661 return te->SetValue( wxEmptyString );
662}
663
664
665void UNIT_BINDER::SetLabel( const wxString& aLabel )
666{
667 m_label->SetLabel( aLabel );
668}
669
670
671void UNIT_BINDER::Enable( bool aEnable )
672{
673 if( m_label )
674 m_label->Enable( aEnable );
675
676 m_valueCtrl->Enable( aEnable );
677
678 if( m_unitLabel )
679 m_unitLabel->Enable( aEnable );
680}
681
682
683void UNIT_BINDER::Show( bool aShow, bool aResize )
684{
685 m_label->Show( aShow );
686 m_valueCtrl->Show( aShow );
687
688 if( m_unitLabel )
689 m_unitLabel->Show( aShow );
690
691 if( aResize )
692 {
693 if( aShow )
694 {
695 m_label->SetSize( -1, -1 );
696 m_valueCtrl->SetSize( -1, -1 );
697
698 if( m_unitLabel )
699 m_unitLabel->SetSize( -1, -1 );
700 }
701 else
702 {
703 m_label->SetSize( 0, 0 );
704 m_valueCtrl->SetSize( 0, 0 );
705
706 if( m_unitLabel )
707 m_unitLabel->SetSize( 0, 0 );
708 }
709 }
710}
711
712
714 UNIT_BINDER( aParent, nullptr, nullptr, nullptr, true, false )
715{
716 m_unitsInValue = true;
717}
718
719
721{
722}
723
725{
726 m_valueCtrl = aControl;
727
728 if( m_valueCtrl )
729 {
730 m_valueCtrl->Bind( wxEVT_SET_FOCUS, &PROPERTY_EDITOR_UNIT_BINDER::onSetFocus, this );
731 m_valueCtrl->Bind( wxEVT_KILL_FOCUS, &PROPERTY_EDITOR_UNIT_BINDER::onKillFocus, this );
732 m_valueCtrl->Bind( wxEVT_LEFT_UP, &PROPERTY_EDITOR_UNIT_BINDER::onClick, this );
733
734 m_valueCtrl->Bind( wxEVT_SHOW,
735 [&]( wxShowEvent& e )
736 {
737 if( !e.IsShown() )
738 SetControl( nullptr );
739 } );
740 }
741}
double AsDegrees() const
Definition: eda_angle.h:113
The base frame for deriving all KiCad main window classes.
The base class for create windows for drawing purpose.
wxString OriginalText() const
wxString Result() const
void SetDefaultUnits(EDA_UNITS aUnits)
bool Process(const wxString &aString)
A class to perform either relative or absolute display origin transforms for a single axis of a point...
virtual int FromDisplay(int aValue, COORD_TYPES_T aCoordType) const
virtual int ToDisplay(int aValue, COORD_TYPES_T aCoordType) const
void SetControl(wxWindow *aControl)
PROPERTY_EDITOR_UNIT_BINDER(EDA_DRAW_FRAME *aParent)
const EDA_IU_SCALE & GetIuScale() const
EDA_UNITS GetUserUnits() const
double setPrecision(double aValue, bool aValueUsesUserUnits) const
When m_precision > 0 truncate the value aValue to show only m_precision digits in mantissa.
ORIGIN_TRANSFORMS::COORD_TYPES_T m_coordType
Type of coordinate for display origin transforms.
Definition: unit_binder.h:271
wxString getTextForValue(long long int aValue) const
int GetIntValue()
Definition: unit_binder.h:129
virtual void ChangeDoubleValue(double aValue)
Set new value (in Internal Units) for the text field, taking care of units conversion WITHOUT trigger...
void onKillFocus(wxFocusEvent &aEvent)
virtual long long int GetValue()
Return the current value in Internal Units.
void Enable(bool aEnable)
Enable/disable the label, widget and units label.
void onClick(wxMouseEvent &aEvent)
virtual void SetPrecision(int aLength)
Normally not needed, but can be used to set the precision when using internal units that are floats (...
virtual void SetUnits(EDA_UNITS aUnits)
Normally not needed (as the UNIT_BINDER inherits from the parent frame), but can be used to set to DE...
bool m_allowEval
Definition: unit_binder.h:259
wxString getTextForDoubleValue(double aValue) const
const EDA_IU_SCALE * m_iuScale
Currently used units.
Definition: unit_binder.h:250
wxString m_errorMessage
Definition: unit_binder.h:256
virtual EDA_ANGLE GetAngleValue()
bool m_negativeZero
Indicates "-0" should be displayed for 0.
Definition: unit_binder.h:252
int m_precision
0 to 6.
Definition: unit_binder.h:254
wxStaticText * m_unitLabel
Can be nullptr.
Definition: unit_binder.h:247
bool m_unitsInValue
Units label should be included in value text.
Definition: unit_binder.h:265
wxWindow * m_valueCtrl
Definition: unit_binder.h:245
wxWindow * m_eventSource
Definition: unit_binder.h:246
void onSetFocus(wxFocusEvent &aEvent)
void onComboBox(wxCommandEvent &aEvent)
UNIT_BINDER(EDA_DRAW_FRAME *aParent, wxStaticText *aLabel, wxWindow *aValueCtrl, wxStaticText *aUnitLabel, bool aAllowEval=true, bool aBindFocusEvent=true)
Definition: unit_binder.cpp:41
wxStaticText * m_label
The bound widgets.
Definition: unit_binder.h:244
ORIGIN_TRANSFORMS & m_originTransforms
A reference to an ORIGIN_TRANSFORMS object.
Definition: unit_binder.h:268
bool m_bindFocusEvent
Definition: unit_binder.h:241
long m_selStart
Selection start and end of the original text.
Definition: unit_binder.h:262
virtual void SetDoubleOptionsList(std::span< const double > aOptions)
EDA_UNITS m_units
Definition: unit_binder.h:251
virtual double GetDoubleValue()
Return the current value in Internal Units.
virtual void SetOptionsList(std::span< const long long int > aOptions)
Set the list of options for a combobox control.
bool IsIndeterminate() const
Return true if the control holds the indeterminate value (for instance, if it represents a multiple s...
bool m_needsEval
Definition: unit_binder.h:260
void SetDataType(EDA_DATA_TYPE aDataType)
Used to override the datatype of the displayed property (default is DISTANCE)
void delayedFocusHandler(wxCommandEvent &aEvent)
EDA_DATA_TYPE m_dataType
Definition: unit_binder.h:253
virtual void SetAngleValue(const EDA_ANGLE &aValue)
void SetLabel(const wxString &aLabel)
void init(UNITS_PROVIDER *aProvider)
virtual void ChangeAngleValue(const EDA_ANGLE &aValue)
virtual void SetDoubleValue(double aValue)
Set new value (in Internal Units) for the text field, taking care of units conversion.
void SetNull()
virtual ~UNIT_BINDER() override
virtual bool Validate(double aMin, double aMax, EDA_UNITS aUnits=EDA_UNITS::UNSCALED)
Validate the control against the given range, informing the user of any errors found.
virtual void ChangeValue(int aValue)
Set new value (in Internal Units) for the text field, taking care of units conversion WITHOUT trigger...
virtual void SetValue(long long int aValue)
Set new value (in Internal Units) for the text field, taking care of units conversion.
void Show(bool aShow, bool aResize=false)
Show/hide the label, widget and units label.
NUMERIC_EVALUATOR m_eval
Definition: unit_binder.h:258
bool IsNull() const
Return true if the control holds no value (ie: empty string, not 0).
void onUnitsChanged(wxCommandEvent &aEvent)
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.
const int minSize
Push and Shove router track width and via size dialog.
#define _(s)
@ DEGREES_T
Definition: eda_angle.h:31
EDA_DATA_TYPE
The type of unit.
Definition: eda_units.h:38
EDA_UNITS
Definition: eda_units.h:46
KICOMMON_API double FromUserUnit(const EDA_IU_SCALE &aIuScale, EDA_UNITS aUnit, double aValue)
Return in internal units the value aValue given in a real unit such as "in", "mm",...
Definition: eda_units.cpp:442
KICOMMON_API long long int ValueFromString(const EDA_IU_SCALE &aIuScale, EDA_UNITS aUnits, const wxString &aTextValue, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
Convert aTextValue in aUnits to internal units used by the application.
Definition: eda_units.cpp:592
KICOMMON_API wxString StringFromValue(const EDA_IU_SCALE &aIuScale, EDA_UNITS aUnits, double aValue, bool aAddUnitsText=false, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
Return the string from aValue according to aUnits (inch, mm ...) for display.
Definition: eda_units.cpp:277
KICOMMON_API double DoubleValueFromString(const EDA_IU_SCALE &aIuScale, EDA_UNITS aUnits, const wxString &aTextValue, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
Convert aTextValue to a double.
Definition: eda_units.cpp:497
KICOMMON_API double ToUserUnit(const EDA_IU_SCALE &aIuScale, EDA_UNITS aUnit, double aValue)
Convert aValue in internal units to the appropriate user units defined by aUnit.
Definition: eda_units.cpp:261
KICOMMON_API wxString GetLabel(EDA_UNITS aUnits, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
Get the units string for a given units type.
Definition: eda_units.cpp:156
int GetUserUnits()
Return the currently selected user unit value for the interface.
const int scale
#define INDETERMINATE_ACTION
Definition: ui_common.h:47
#define INDETERMINATE_STATE
Used for holding indeterminate values, such as with multiple selections holding different values or c...
Definition: ui_common.h:46
wxDEFINE_EVENT(DELAY_FOCUS, wxCommandEvent)
wxString valueDescriptionFromLabel(wxStaticText *aLabel)