KiCad PCB EDA Suite
Loading...
Searching...
No Matches
drc_re_matched_length_diff_pair_overlay_panel.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) 2024 KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License
8 * as published by the Free Software Foundation; either version 2
9 * of the License, or (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with this program. If not, see <https://www.gnu.org/licenses/>.
18 */
19
24
25#include <base_units.h>
26#include <eda_base_frame.h>
27#include <widgets/unit_binder.h>
28
29#include <wx/textctrl.h>
30#include <wx/stattext.h>
31#include <wx/checkbox.h>
33
35 wxWindow* aParent, DRC_RE_MATCHED_LENGTH_DIFF_PAIR_CONSTRAINT_DATA* aData, EDA_UNITS aUnits ) :
37 m_data( aData ),
39{
41
42 std::vector<DRC_RE_FIELD_POSITION> positions = m_data->GetFieldPositions();
43
44 wxWindow* eventSource = nullptr;
45
46 for( wxWindow* win = aParent; win; win = win->GetParent() )
47 {
48 if( dynamic_cast<EDA_BASE_FRAME*>( win ) )
49 {
50 eventSource = win;
51 break;
52 }
53 }
54
55 // Create opt length field
56 auto* optLengthField = AddField<wxTextCtrl>( wxS( "opt_length" ), positions[0], wxTE_CENTRE | wxTE_PROCESS_ENTER );
58 std::make_unique<UNIT_BINDER>( &m_unitsProvider, eventSource, nullptr, optLengthField->GetControl(),
59 optLengthField->GetLabel(), false, false );
60 optLengthField->SetUnitBinder( m_optLengthBinder.get() );
61 optLengthField->GetControl()->SetValidator( VALIDATOR_NUMERIC_CTRL( false, false ) );
62
63 // Create tolerance field
64 auto* toleranceField = AddField<wxTextCtrl>( wxS( "tolerance" ), positions[1], wxTE_CENTRE | wxTE_PROCESS_ENTER );
66 std::make_unique<UNIT_BINDER>( &m_unitsProvider, eventSource, nullptr, toleranceField->GetControl(),
67 toleranceField->GetLabel(), false, false );
68 toleranceField->SetUnitBinder( m_toleranceBinder.get() );
69 toleranceField->GetControl()->SetValidator( VALIDATOR_NUMERIC_CTRL( false, false ) );
70
71 // Add +/- decoration between opt and tolerance fields
72 {
73 const DRC_RE_FIELD_POSITION& optPos = positions[0];
74 const DRC_RE_FIELD_POSITION& tolPos = positions[1];
75 int fieldHeight = optLengthField->GetControl()->GetBestSize().GetHeight();
76
77 auto* plusMinus = new wxStaticText( this, wxID_ANY, wxS( "\u00B1" ) );
78 wxSize pmSize = plusMinus->GetBestSize();
79 wxStaticText* optMmLabel = optLengthField->GetLabel();
80 int afterOptLabel = optMmLabel->GetPosition().x + optMmLabel->GetBestSize().GetWidth();
81 int gapMid = ( afterOptLabel + tolPos.xStart ) / 2;
82 plusMinus->SetPosition(
83 wxPoint( gapMid - pmSize.GetWidth() / 2, optPos.yTop + ( fieldHeight - pmSize.GetHeight() ) / 2 ) );
84 }
85
86 // Create max skew field
87 auto* maxSkewField = AddField<wxTextCtrl>( wxS( "max_skew" ), positions[2], wxTE_CENTRE | wxTE_PROCESS_ENTER );
88 m_maxSkewBinder = std::make_unique<UNIT_BINDER>( &m_unitsProvider, eventSource, nullptr, maxSkewField->GetControl(),
89 maxSkewField->GetLabel(), false, false );
90 maxSkewField->SetUnitBinder( m_maxSkewBinder.get() );
91 maxSkewField->GetControl()->SetValidator( VALIDATOR_NUMERIC_CTRL( false, false ) );
92
93 {
94 const DRC_RE_FIELD_POSITION& skewPos = positions[2];
95 wxStaticText* skewMmLabel = maxSkewField->GetLabel();
96 int afterSkewLabel = skewMmLabel->GetPosition().x + skewMmLabel->GetBestSize().GetWidth();
97 int fieldHeight = maxSkewField->GetControl()->GetBestSize().GetHeight();
98
99 m_withinDiffPairsCheckbox = new wxCheckBox( this, wxID_ANY, _( "Within diff pairs" ) );
100 wxSize cbSize = m_withinDiffPairsCheckbox->GetBestSize();
101 m_withinDiffPairsCheckbox->SetPosition(
102 wxPoint( afterSkewLabel + 8, skewPos.yTop + ( fieldHeight - cbSize.GetHeight() ) / 2 ) );
103 }
104
105 auto notifyModified = [this]( wxCommandEvent& )
106 {
108 if( dlg )
109 dlg->SetModified();
110 };
111
112 optLengthField->GetControl()->Bind( wxEVT_TEXT, notifyModified );
113 toleranceField->GetControl()->Bind( wxEVT_TEXT, notifyModified );
114 maxSkewField->GetControl()->Bind( wxEVT_TEXT, notifyModified );
115 m_withinDiffPairsCheckbox->Bind( wxEVT_CHECKBOX, notifyModified );
116
117 auto notifySave = [this]( wxCommandEvent& aEvent )
118 {
120 if( dlg )
121 dlg->OnSave( aEvent );
122 };
123
124 optLengthField->GetControl()->Bind( wxEVT_TEXT_ENTER, notifySave );
125 toleranceField->GetControl()->Bind( wxEVT_TEXT_ENTER, notifySave );
126 maxSkewField->GetControl()->Bind( wxEVT_TEXT_ENTER, notifySave );
127 m_withinDiffPairsCheckbox->Bind( wxEVT_CHECKBOX, notifySave );
128
129 // Position all fields and update the panel layout
132}
133
134
136{
137 if( !m_data )
138 return false;
139
140 m_optLengthBinder->ChangeDoubleValue( pcbIUScale.mmToIU( m_data->GetOptimumLength() ) );
141 m_toleranceBinder->ChangeDoubleValue( pcbIUScale.mmToIU( m_data->GetTolerance() ) );
142 m_maxSkewBinder->ChangeDoubleValue( pcbIUScale.mmToIU( m_data->GetMaxSkew() ) );
143 m_withinDiffPairsCheckbox->SetValue( m_data->GetWithinDiffPairs() );
144
145 return true;
146}
147
148
150{
151 if( !m_data )
152 return false;
153
154 m_data->SetOptimumLength( pcbIUScale.IUTomm( m_optLengthBinder->GetDoubleValue() ) );
155 m_data->SetTolerance( pcbIUScale.IUTomm( m_toleranceBinder->GetDoubleValue() ) );
156 m_data->SetMaxSkew( pcbIUScale.IUTomm( m_maxSkewBinder->GetDoubleValue() ) );
157 m_data->SetWithinDiffPairs( m_withinDiffPairsCheckbox->GetValue() );
158
159 return true;
160}
161
162
163bool DRC_RE_MATCHED_LENGTH_DIFF_PAIR_OVERLAY_PANEL::ValidateInputs( int* aErrorCount, wxString* aValidationMessage )
164{
166
167 VALIDATION_RESULT result = m_data->Validate();
168
169 if( !result.isValid )
170 {
171 *aErrorCount = result.errors.size();
172
173 for( size_t i = 0; i < result.errors.size(); i++ )
174 *aValidationMessage += DRC_RULE_EDITOR_UTILS::FormatErrorMessage( i + 1, result.errors[i] );
175
176 return false;
177 }
178
179 return true;
180}
181
182
184{
185 if( !m_data )
186 return wxEmptyString;
187
188 return m_data->GenerateRule( aContext );
189}
constexpr EDA_IU_SCALE pcbIUScale
Definition base_units.h:121
DRC_RE_OVERLAY_FIELD * AddField(const wxString &aId, const DRC_RE_FIELD_POSITION &aPosition, long aStyle=0)
Create and position a field control on the bitmap overlay.
void SetBackgroundBitmap(BITMAPS aBitmap)
Set the background bitmap for this panel.
DRC_RE_BITMAP_OVERLAY_PANEL(wxWindow *aParent, wxWindowID aId=wxID_ANY)
void PositionFields()
Position all fields based on the current scale factor.
BITMAPS GetOverlayBitmap() const override
Returns the bitmap to use for the overlay panel background.
bool ValidateInputs(int *aErrorCount, wxString *aValidationMessage) override
wxString GenerateRule(const RULE_GENERATION_CONTEXT &aContext) override
DRC_RE_MATCHED_LENGTH_DIFF_PAIR_OVERLAY_PANEL(wxWindow *aParent, DRC_RE_MATCHED_LENGTH_DIFF_PAIR_CONSTRAINT_DATA *aData, EDA_UNITS aUnits)
static wxString FormatErrorMessage(int aErrorCount, const wxString &aErrorMessage)
The base frame for deriving all KiCad main window classes.
void SetModified()
Marks the dialog as modified, indicating unsaved changes.
static RULE_EDITOR_DIALOG_BASE * GetDialog(wxWindow *aWindow)
Static method to retrieve the rule editor dialog instance associated with a given window.
virtual void OnSave(wxCommandEvent &aEvent)=0
#define _(s)
Base window classes and related definitions.
EDA_UNITS
Definition eda_units.h:44
Specifies the position and size of a field overlaid on a constraint bitmap.
int xStart
Left edge X coordinate where the field starts.
int yTop
Top edge Y coordinate of the field.
Result of a validation operation.
wxString result
Test unit parsing edge cases and error handling.