KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
pcb_fields_grid_table.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 The 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, you may find one here:
18 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
19 * or you may search the http://www.gnu.org website for the version 2 license,
20 * or you may write to the Free Software Foundation, Inc.,
21 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
22 */
23
24#include <board.h>
25#include <footprint.h>
27#include <kiway.h>
28#include <kiway_player.h>
30#include <pcb_base_frame.h>
31#include <pcb_edit_frame.h>
32#include <project.h>
33#include <trigo.h>
35
39
40enum
41{
42 MYID_SELECT_FOOTPRINT = 991, // must be within GRID_TRICKS' enum range
44};
45
46
47wxArrayString g_menuOrientations;
48
49
51 EMBEDDED_FILES* aFiles ) :
52 m_frame( aFrame ), m_dialog( aDialog ),
53 m_fieldNameValidator( FIELD_T::USER ),
54 m_referenceValidator( FIELD_T::REFERENCE ),
55 m_valueValidator( FIELD_T::VALUE ),
56 m_urlValidator( FIELD_T::USER ),
57 m_nonUrlValidator( FIELD_T::USER )
58{
59 // Build the column attributes.
60
61 m_readOnlyAttr = new wxGridCellAttr;
62 m_readOnlyAttr->SetReadOnly( true );
63
64 m_boolColAttr = new wxGridCellAttr;
65 m_boolColAttr->SetRenderer( new wxGridCellBoolRenderer() );
66 m_boolColAttr->SetEditor( new wxGridCellBoolEditor() );
67 m_boolColAttr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
68
69 if( g_menuOrientations.IsEmpty() )
70 {
71 g_menuOrientations.push_back( "0" + EDA_UNIT_UTILS::GetText( EDA_UNITS::DEGREES ) );
72 g_menuOrientations.push_back( "90" + EDA_UNIT_UTILS::GetText( EDA_UNITS::DEGREES ) );
73 g_menuOrientations.push_back( "-90" + EDA_UNIT_UTILS::GetText( EDA_UNITS::DEGREES ) );
74 g_menuOrientations.push_back( "180" + EDA_UNIT_UTILS::GetText( EDA_UNITS::DEGREES ) );
75 }
76
77 m_orientationColAttr = new wxGridCellAttr;
79
80 m_layerColAttr = new wxGridCellAttr;
81 m_layerColAttr->SetRenderer( new GRID_CELL_LAYER_RENDERER( m_frame ) );
82 m_layerColAttr->SetEditor( new GRID_CELL_LAYER_SELECTOR( m_frame, {} ) );
83
84 m_referenceAttr = new wxGridCellAttr;
85 GRID_CELL_TEXT_EDITOR* referenceEditor = new GRID_CELL_TEXT_EDITOR();
86 referenceEditor->SetValidator( m_referenceValidator );
87 m_referenceAttr->SetEditor( referenceEditor );
88
89 m_valueAttr = new wxGridCellAttr;
91 valueEditor->SetValidator( m_valueValidator );
92 m_valueAttr->SetEditor( valueEditor );
93
94 m_urlAttr = new wxGridCellAttr;
95 GRID_CELL_URL_EDITOR* urlEditor = new GRID_CELL_URL_EDITOR( m_dialog, nullptr, aFiles );
96 urlEditor->SetValidator( m_urlValidator );
97 m_urlAttr->SetEditor( urlEditor );
98
99 m_eval = std::make_unique<NUMERIC_EVALUATOR>( m_frame->GetUserUnits() );
100
101 m_frame->Bind( EDA_EVT_UNITS_CHANGED, &PCB_FIELDS_GRID_TABLE::onUnitsChanged, this );
102}
103
104
106{
107 m_readOnlyAttr->DecRef();
108 m_boolColAttr->DecRef();
109 m_orientationColAttr->DecRef();
110 m_layerColAttr->DecRef();
111 m_referenceAttr->DecRef();
112 m_valueAttr->DecRef();
113 m_urlAttr->DecRef();
114
115 m_frame->Unbind( EDA_EVT_UNITS_CHANGED, &PCB_FIELDS_GRID_TABLE::onUnitsChanged, this );
116}
117
118
120{
121 int mandatoryRows = 0;
122
123 for( const PCB_FIELD& field : *this )
124 {
125 if( field.IsMandatory() )
126 mandatoryRows++;
127 }
128
129 return mandatoryRows;
130}
131
132
133void PCB_FIELDS_GRID_TABLE::onUnitsChanged( wxCommandEvent& aEvent )
134{
135 if( GetView() )
136 GetView()->ForceRefresh();
137
138 aEvent.Skip();
139}
140
141
143{
144 switch( aCol )
145 {
146 case PFC_NAME: return _( "Name" );
147 case PFC_VALUE: return _( "Value" );
148 case PFC_SHOWN: return _( "Show" );
149 case PFC_WIDTH: return _( "Width" );
150 case PFC_HEIGHT: return _( "Height" );
151 case PFC_THICKNESS: return _( "Thickness" );
152 case PFC_ITALIC: return _( "Italic" );
153 case PFC_LAYER: return _( "Layer" );
154 case PFC_ORIENTATION: return _( "Orientation" );
155 case PFC_UPRIGHT: return _( "Keep Upright" );
156 case PFC_XOFFSET: return _( "X Offset" );
157 case PFC_YOFFSET: return _( "Y Offset" );
158 case PFC_KNOCKOUT: return _( "Knockout" );
159 case PFC_MIRRORED: return _( "Mirrored" );
160 default: wxFAIL; return wxEmptyString;
161 }
162}
163
164
165bool PCB_FIELDS_GRID_TABLE::CanGetValueAs( int aRow, int aCol, const wxString& aTypeName )
166{
167 switch( aCol )
168 {
169 case PFC_NAME:
170 case PFC_VALUE:
171 case PFC_WIDTH:
172 case PFC_HEIGHT:
173 case PFC_THICKNESS:
174 case PFC_ORIENTATION:
175 case PFC_XOFFSET:
176 case PFC_YOFFSET:
177 return aTypeName == wxGRID_VALUE_STRING;
178
179 case PFC_SHOWN:
180 case PFC_ITALIC:
181 case PFC_UPRIGHT:
182 case PFC_KNOCKOUT:
183 case PFC_MIRRORED:
184 return aTypeName == wxGRID_VALUE_BOOL;
185
186 case PFC_LAYER:
187 return aTypeName == wxGRID_VALUE_NUMBER;
188
189 default:
190 wxFAIL;
191 return false;
192 }
193}
194
195
196bool PCB_FIELDS_GRID_TABLE::CanSetValueAs( int aRow, int aCol, const wxString& aTypeName )
197{
198 return CanGetValueAs( aRow, aCol, aTypeName );
199}
200
201
202wxGridCellAttr* PCB_FIELDS_GRID_TABLE::GetAttr( int aRow, int aCol,
203 wxGridCellAttr::wxAttrKind aKind )
204{
205 const PCB_FIELD& field = this->at( (size_t) aRow );
206
207 switch( aCol )
208 {
209 case PFC_NAME:
210 if( field.IsMandatory() )
211 {
212 m_readOnlyAttr->IncRef();
213 return enhanceAttr( m_readOnlyAttr, aRow, aCol, aKind );
214 }
215
216 return enhanceAttr( nullptr, aRow, aCol, aKind );
217
218 case PFC_VALUE:
219 if( field.GetId() == FIELD_T::REFERENCE )
220 {
221 m_referenceAttr->IncRef();
222 return enhanceAttr( m_referenceAttr, aRow, aCol, aKind );
223 }
224 else if( field.GetId() == FIELD_T::VALUE )
225 {
226 m_valueAttr->IncRef();
227 return enhanceAttr( m_valueAttr, aRow, aCol, aKind );
228 }
229 else if( field.GetId() == FIELD_T::DATASHEET || field.IsHypertext() )
230 {
231 m_urlAttr->IncRef();
232 return enhanceAttr( m_urlAttr, aRow, aCol, aKind );
233 }
234
235 return enhanceAttr( nullptr, aRow, aCol, aKind );
236
237 case PFC_WIDTH:
238 case PFC_HEIGHT:
239 case PFC_THICKNESS:
240 case PFC_XOFFSET:
241 case PFC_YOFFSET:
242 return enhanceAttr( nullptr, aRow, aCol, aKind );
243
244 case PFC_SHOWN:
245 case PFC_ITALIC:
246 case PFC_UPRIGHT:
247 case PFC_KNOCKOUT:
248 case PFC_MIRRORED:
249 m_boolColAttr->IncRef();
250 return enhanceAttr( m_boolColAttr, aRow, aCol, aKind );
251
252 case PFC_LAYER:
253 m_layerColAttr->IncRef();
254 return enhanceAttr( m_layerColAttr, aRow, aCol, aKind );
255
256 case PFC_ORIENTATION:
257 m_orientationColAttr->IncRef();
258 return enhanceAttr( m_orientationColAttr, aRow, aCol, aKind );
259
260 default:
261 wxFAIL;
262 return enhanceAttr( nullptr, aRow, aCol, aKind );
263 }
264}
265
266
267wxString PCB_FIELDS_GRID_TABLE::GetValue( int aRow, int aCol )
268{
269 wxGrid* grid = GetView();
270 const PCB_FIELD& field = this->at( (size_t) aRow );
271
272 if( grid->GetGridCursorRow() == aRow && grid->GetGridCursorCol() == aCol
273 && grid->IsCellEditControlShown() )
274 {
275 auto it = m_evalOriginal.find( { aRow, aCol } );
276
277 if( it != m_evalOriginal.end() )
278 return it->second;
279 }
280
281 switch( aCol )
282 {
283 case PFC_NAME: return field.GetName();
284 case PFC_VALUE: return field.GetText();
285 case PFC_WIDTH: return m_frame->StringFromValue( field.GetTextWidth(), true );
286 case PFC_HEIGHT: return m_frame->StringFromValue( field.GetTextHeight(), true );
287 case PFC_THICKNESS: return m_frame->StringFromValue( field.GetTextThickness(), true );
288 case PFC_LAYER: return field.GetLayerName();
289
290 case PFC_ORIENTATION:
291 {
292 EDA_ANGLE angle = field.GetTextAngle() - field.GetParentFootprint()->GetOrientation();
293 return m_frame->StringFromValue( angle, true );
294 }
295
296 case PFC_XOFFSET: return m_frame->StringFromValue( field.GetFPRelativePosition().x, true );
297 case PFC_YOFFSET: return m_frame->StringFromValue( field.GetFPRelativePosition().y, true );
298
299 default:
300 // we can't assert here because wxWidgets sometimes calls this without checking
301 // the column type when trying to see if there's an overflow
302 return wxT( "bad wxWidgets!" );
303 }
304}
305
306
308{
309 PCB_FIELD& field = this->at( (size_t) aRow );
310
311 switch( aCol )
312 {
313 case PFC_SHOWN: return field.IsVisible();
314 case PFC_ITALIC: return field.IsItalic();
315 case PFC_UPRIGHT: return field.IsKeepUpright();
316 case PFC_KNOCKOUT: return field.IsKnockout();
317 case PFC_MIRRORED: return field.IsMirrored();
318
319 default:
320 wxFAIL_MSG( wxString::Format( wxT( "column %d doesn't hold a bool value" ), aCol ) );
321 return false;
322 }
323}
324
325
327{
328 PCB_FIELD& field = this->at( (size_t) aRow );
329
330 switch( aCol )
331 {
332 case PFC_LAYER: return field.GetLayer();
333
334 default:
335 wxFAIL_MSG( wxString::Format( wxT( "column %d doesn't hold a long value" ), aCol ) );
336 return 0;
337 }
338}
339
340
341void PCB_FIELDS_GRID_TABLE::SetValue( int aRow, int aCol, const wxString &aValue )
342{
343 PCB_FIELD& field = this->at( (size_t) aRow );
344 VECTOR2I pos;
345 wxString value = aValue;
346
347 switch( aCol )
348 {
349 case PFC_WIDTH:
350 case PFC_HEIGHT:
351 case PFC_THICKNESS:
352 case PFC_XOFFSET:
353 case PFC_YOFFSET:
354 m_eval->SetDefaultUnits( m_frame->GetUserUnits() );
355
356 if( m_eval->Process( value ) )
357 {
358 m_evalOriginal[ { aRow, aCol } ] = value;
359 value = m_eval->Result();
360 }
361
362 break;
363
364 default:
365 break;
366 }
367
368 switch( aCol )
369 {
370 case PFC_NAME: field.SetName( value ); break;
371 case PFC_VALUE: field.SetText( value ); break;
372 case PFC_WIDTH: field.SetTextWidth( m_frame->ValueFromString( value ) ); break;
373 case PFC_HEIGHT: field.SetTextHeight( m_frame->ValueFromString( value ) ); break;
374 case PFC_THICKNESS: field.SetTextThickness( m_frame->ValueFromString( value ) ); break;
375
376 case PFC_ORIENTATION:
378 + field.GetParentFootprint()->GetOrientation() );
379 break;
380
381 case PFC_XOFFSET:
382 case PFC_YOFFSET:
383 pos = field.GetFPRelativePosition();
384
385 if( aCol == PFC_XOFFSET )
386 pos.x = m_frame->ValueFromString( value );
387 else
388 pos.y = m_frame->ValueFromString( value );
389
390 field.SetFPRelativePosition( pos );
391 break;
392
393 default:
394 wxFAIL_MSG( wxString::Format( wxT( "column %d doesn't hold a string value" ), aCol ) );
395 break;
396 }
397
398 GetView()->Refresh();
400}
401
402
403void PCB_FIELDS_GRID_TABLE::SetValueAsBool( int aRow, int aCol, bool aValue )
404{
405 PCB_FIELD& field = this->at( (size_t) aRow );
406
407 switch( aCol )
408 {
409 case PFC_SHOWN: field.SetVisible( aValue ); break;
410 case PFC_ITALIC: field.SetItalic( aValue ); break;
411 case PFC_UPRIGHT: field.SetKeepUpright( aValue ); break;
412 case PFC_KNOCKOUT: field.SetIsKnockout( aValue ); break;
413 case PFC_MIRRORED: field.SetMirrored( aValue ); break;
414
415 default:
416 wxFAIL_MSG( wxString::Format( wxT( "column %d doesn't hold a bool value" ), aCol ) );
417 break;
418 }
419
421}
422
423
424void PCB_FIELDS_GRID_TABLE::SetValueAsLong( int aRow, int aCol, long aValue )
425{
426 PCB_FIELD& field = this->at( (size_t) aRow );
427
428 switch( aCol )
429 {
430 case PFC_LAYER:
431 field.SetLayer( ToLAYER_ID( (int) aValue ) );
432 field.SetMirrored( IsBackLayer( field.GetLayer() ) );
433 break;
434
435 default:
436 wxFAIL_MSG( wxString::Format( wxT( "column %d doesn't hold a long value" ), aCol ) );
437 break;
438 }
439
441}
442
virtual PCB_LAYER_ID GetLayer() const
Return the primary layer this item is on.
Definition: board_item.h:229
virtual bool IsKnockout() const
Definition: board_item.h:316
virtual void SetIsKnockout(bool aKnockout)
Definition: board_item.h:317
virtual void SetLayer(PCB_LAYER_ID aLayer)
Set the layer this item is on.
Definition: board_item.h:280
FOOTPRINT * GetParentFootprint() const
Definition: board_item.cpp:299
VECTOR2I GetFPRelativePosition() const
Definition: board_item.cpp:328
void SetFPRelativePosition(const VECTOR2I &aPos)
Definition: board_item.cpp:342
wxString GetLayerName() const
Return the name of the PCB layer on which the item resides.
Definition: board_item.cpp:140
Dialog helper object to sit in the inheritance tree between wxDialog and any class written by wxFormB...
Definition: dialog_shim.h:52
void OnModify()
int GetTextHeight() const
Definition: eda_text.h:254
bool IsItalic() const
Definition: eda_text.h:156
const EDA_ANGLE & GetTextAngle() const
Definition: eda_text.h:134
virtual const wxString & GetText() const
Return the string associated with the text object.
Definition: eda_text.h:98
bool IsKeepUpright() const
Definition: eda_text.h:193
virtual bool IsVisible() const
Definition: eda_text.h:174
void SetMirrored(bool isMirrored)
Definition: eda_text.cpp:386
int GetTextWidth() const
Definition: eda_text.h:251
void SetTextWidth(int aWidth)
Definition: eda_text.cpp:549
virtual void SetVisible(bool aVisible)
Definition: eda_text.cpp:379
void SetTextThickness(int aWidth)
The TextThickness is that set by the user.
Definition: eda_text.cpp:284
void SetTextHeight(int aHeight)
Definition: eda_text.cpp:560
bool IsMirrored() const
Definition: eda_text.h:177
void SetKeepUpright(bool aKeepUpright)
Definition: eda_text.cpp:418
virtual void SetText(const wxString &aText)
Definition: eda_text.cpp:270
virtual void SetTextAngle(const EDA_ANGLE &aAngle)
Definition: eda_text.cpp:292
int GetTextThickness() const
Definition: eda_text.h:126
void SetItalic(bool aItalic)
Set the text to be italic - this will also update the font if needed.
Definition: eda_text.cpp:300
EDA_ANGLE GetOrientation() const
Definition: footprint.h:232
This class works around a bug in wxGrid where the first keystroke doesn't get sent through the valida...
virtual void SetValidator(const wxValidator &validator) override
Base PCB main window class for Pcbnew, Gerbview, and CvPcb footprint viewer.
wxGridCellAttr * m_referenceAttr
void onUnitsChanged(wxCommandEvent &aEvent)
void SetValue(int aRow, int aCol, const wxString &aValue) override
bool GetValueAsBool(int aRow, int aCol) override
wxString GetColLabelValue(int aCol) override
bool CanGetValueAs(int aRow, int aCol, const wxString &aTypeName) override
std::map< std::pair< int, int >, wxString > m_evalOriginal
long GetValueAsLong(int aRow, int aCol) override
wxGridCellAttr * m_readOnlyAttr
FIELD_VALIDATOR m_urlValidator
FIELD_VALIDATOR m_valueValidator
wxString GetValue(int aRow, int aCol) override
wxGridCellAttr * GetAttr(int aRow, int aCol, wxGridCellAttr::wxAttrKind aKind) override
std::unique_ptr< NUMERIC_EVALUATOR > m_eval
void SetValueAsBool(int aRow, int aCol, bool aValue) override
FIELD_VALIDATOR m_referenceValidator
PCB_FIELDS_GRID_TABLE(PCB_BASE_FRAME *aFrame, DIALOG_SHIM *aDialog, EMBEDDED_FILES *aFiles)
bool CanSetValueAs(int aRow, int aCol, const wxString &aTypeName) override
wxGridCellAttr * m_boolColAttr
wxGridCellAttr * m_layerColAttr
wxGridCellAttr * m_orientationColAttr
void SetValueAsLong(int aRow, int aCol, long aValue) override
bool IsMandatory() const
Definition: pcb_field.cpp:117
FIELD_T GetId() const
Definition: pcb_field.h:108
bool IsHypertext() const
Definition: pcb_field.cpp:126
wxString GetName(bool aUseDefaultName=true) const
Return the field name (not translated).
Definition: pcb_field.cpp:100
void SetName(const wxString &aName)
Definition: pcb_field.h:106
EDA_ANGLE AngleValueFromString(const wxString &aTextValue) const
EDA_UNITS GetUserUnits() const
wxString StringFromValue(double aValue, bool aAddUnitLabel=false, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
Converts aValue in internal units into a united string.
int ValueFromString(const wxString &aTextValue, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE) const
Converts aTextValue in aUnits to internal units used by the frame.
wxGridCellAttr * enhanceAttr(wxGridCellAttr *aInputAttr, int aRow, int aCol, wxGridCellAttr::wxAttrKind aKind)
Definition: wx_grid.cpp:45
#define _(s)
@ USER
The main config directory (e.g. ~/.config/kicad/)
bool IsBackLayer(PCB_LAYER_ID aLayerId)
Layer classification: check if it's a back layer.
Definition: layer_ids.h:788
PCB_LAYER_ID ToLAYER_ID(int aLayer)
Definition: lset.cpp:723
KICOMMON_API wxString GetText(EDA_UNITS aUnits, EDA_DATA_TYPE aType=EDA_DATA_TYPE::DISTANCE)
Get the units string for a given units type.
Definition: eda_units.cpp:127
wxArrayString g_menuOrientations
@ MYID_SHOW_DATASHEET
@ MYID_SELECT_FOOTPRINT
@ PFC_XOFFSET
@ PFC_KNOCKOUT
@ PFC_ITALIC
@ PFC_HEIGHT
@ PFC_THICKNESS
@ PFC_MIRRORED
@ PFC_ORIENTATION
@ PFC_UPRIGHT
@ PFC_YOFFSET
FIELD_T
The set of all field indices assuming an array like sequence that a SCH_COMPONENT or LIB_PART can hol...
@ REFERENCE
Field Reference of part, i.e. "IC21".
@ VALUE
Field Value of part, i.e. "3.3K".