KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
dialog_sim_model.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) 2022 Mikolaj Wielgus
5 * Copyright (C) 2022 CERN
6 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
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 3
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 * https://www.gnu.org/licenses/gpl-3.0.html
21 * or you may search the http://www.gnu.org website for the version 3 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
27#include <dialog_sim_model.h>
28#include <sim/sim_property.h>
30#include <sim/sim_model.h>
31#include <sim/sim_model_ibis.h>
35#include <grid_tricks.h>
38#include <kiplatform/ui.h>
39#include <confirm.h>
40#include <string_utils.h>
41#include <locale_io.h>
42#include <wx/filedlg.h>
43#include <fmt/format.h>
44#include <sch_edit_frame.h>
48#include <wx/filedlg.h>
49#include <wx/log.h>
50
52
53#define FORCE_UPDATE_PINS true
54
55
56bool equivalent( SIM_MODEL::DEVICE_T a, SIM_MODEL::DEVICE_T b )
57{
58 // A helper to handle SPICE's use of 'E' and 'H' for voltage sources and 'F' and 'G' for
59 // current sources
60 return a == b
61 || SIM_MODEL::DeviceInfo( a ).description == SIM_MODEL::DeviceInfo( b ).description;
62};
63
64
65template <typename T>
66DIALOG_SIM_MODEL<T>::DIALOG_SIM_MODEL( wxWindow* aParent, EDA_BASE_FRAME* aFrame, T& aSymbol,
67 std::vector<SCH_FIELD>& aFields ) :
68 DIALOG_SIM_MODEL_BASE( aParent ),
69 m_frame( aFrame ),
70 m_symbol( aSymbol ),
71 m_fields( aFields ),
72 m_libraryModelsMgr( &Prj(), nullptr ),
73 m_builtinModelsMgr( &Prj(), nullptr ),
74 m_prevModel( nullptr ),
75 m_curModelType( SIM_MODEL::TYPE::NONE ),
76 m_scintillaTricksCode( nullptr ),
77 m_scintillaTricksSubckt( nullptr ),
78 m_firstCategory( nullptr ),
79 m_prevParamGridSelection( nullptr ),
80 m_lastParamGridWidth( 0 )
81{
82 m_browseButton->SetBitmap( KiBitmapBundle( BITMAPS::small_folder ) );
84
85 if constexpr (std::is_same_v<T, SCH_SYMBOL>)
86 m_files = aSymbol.Schematic();
87 else
88 m_files = &aSymbol;
89
92
93 for( SCH_PIN* pin : aSymbol.GetPins() )
94 {
95 // De Morgan conversions are equivalences, not additional items to simulate
96 if( !pin->GetParentSymbol()->HasAlternateBodyStyle() || pin->GetBodyStyle() < 2 )
97 m_sortedPartPins.push_back( pin );
98 }
99
100 std::sort( m_sortedPartPins.begin(), m_sortedPartPins.end(),
101 []( const SCH_PIN* lhs, const SCH_PIN* rhs )
102 {
103 // We sort by StrNumCmp because SIM_MODEL_BASE sorts with it too.
104 return StrNumCmp( lhs->GetNumber(), rhs->GetNumber(), true ) < 0;
105 } );
106
107 m_waveformChoice->Clear();
108 m_deviceChoice->Clear();
109 m_deviceSubtypeChoice->Clear();
110
111 m_scintillaTricksCode = new SCINTILLA_TRICKS( m_codePreview, wxT( "{}" ), false );
112 m_scintillaTricksSubckt = new SCINTILLA_TRICKS( m_subckt, wxT( "()" ), false );
113
114 m_paramGridMgr->Bind( wxEVT_PG_SELECTED, &DIALOG_SIM_MODEL::onParamGridSelectionChange, this );
115
116 wxPropertyGrid* grid = m_paramGrid->GetGrid();
117 grid->SetCellDisabledTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
118 grid->Bind( wxEVT_SET_FOCUS, &DIALOG_SIM_MODEL::onParamGridSetFocus, this );
119 grid->Bind( wxEVT_UPDATE_UI, &DIALOG_SIM_MODEL::onUpdateUI, this );
120
121 grid->DedicateKey( WXK_RETURN );
122 grid->DedicateKey( WXK_NUMPAD_ENTER );
123 grid->DedicateKey( WXK_UP );
124 grid->DedicateKey( WXK_DOWN );
125
126#if wxCHECK_VERSION( 3, 3, 0 )
127 grid->AddActionTrigger( wxPGKeyboardAction::Edit, WXK_RETURN );
128 grid->AddActionTrigger( wxPGKeyboardAction::NextProperty, WXK_RETURN );
129 grid->AddActionTrigger( wxPGKeyboardAction::Edit, WXK_NUMPAD_ENTER );
130 grid->AddActionTrigger( wxPGKeyboardAction::NextProperty, WXK_NUMPAD_ENTER );
131#else
132 grid->AddActionTrigger( wxPG_ACTION_EDIT, WXK_RETURN );
133 grid->AddActionTrigger( wxPG_ACTION_NEXT_PROPERTY, WXK_RETURN );
134 grid->AddActionTrigger( wxPG_ACTION_EDIT, WXK_NUMPAD_ENTER );
135 grid->AddActionTrigger( wxPG_ACTION_NEXT_PROPERTY, WXK_NUMPAD_ENTER );
136#endif
137
139 m_pinAssignmentsGrid->PushEventHandler( new GRID_TRICKS( m_pinAssignmentsGrid ) );
140
143}
144
145
146template <typename T>
148{
149 // Disable all properties. This is necessary because some of their methods are called after
150 // destruction of DIALOG_SIM_MODEL, oddly. When disabled, they never access their models.
151 for( wxPropertyGridIterator it = m_paramGrid->GetIterator(); !it.AtEnd(); ++it )
152 {
153 SIM_PROPERTY* prop = dynamic_cast<SIM_PROPERTY*>( *it );
154
155 if( !prop )
156 continue;
157
158 prop->Disable();
159 }
160
161 // Delete the GRID_TRICKS.
162 m_pinAssignmentsGrid->PopEventHandler( true );
163
164 delete m_scintillaTricksCode;
165 delete m_scintillaTricksSubckt;
166}
167
168
169template <typename T>
171{
172 wxCommandEvent dummyEvent;
173 wxString deviceType;
174 wxString modelType;
175 wxString modelParams;
176 wxString pinMap;
177 bool storeInValue = false;
178
179 WX_STRING_REPORTER reporter;
180
181 // Infer RLC and VI models if they aren't specified
182 if( SIM_MODEL::InferSimModel( m_symbol, &m_fields, false, SIM_VALUE_GRAMMAR::NOTATION::SI,
183 &deviceType, &modelType, &modelParams, &pinMap ) )
184 {
185 SetFieldValue( m_fields, SIM_DEVICE_FIELD, deviceType.ToStdString() );
186
187 if( !modelType.IsEmpty() )
188 SetFieldValue( m_fields, SIM_DEVICE_SUBTYPE_FIELD, modelType.ToStdString() );
189
190 SetFieldValue( m_fields, SIM_PARAMS_FIELD, modelParams.ToStdString() );
191
192 SetFieldValue( m_fields, SIM_PINS_FIELD, pinMap.ToStdString() );
193
194 storeInValue = true;
195
196 // In case the storeInValue checkbox is turned off (if it's left on then we'll overwrite
197 // this field with the actual value):
198 FindField( m_fields, FIELD_T::VALUE )->SetText( wxT( "${SIM.PARAMS}" ) );
199 }
200
201 wxString libraryFilename = GetFieldValue( &m_fields, SIM_LIBRARY::LIBRARY_FIELD );
202
203 if( libraryFilename != "" )
204 {
205 // The model is sourced from a library, optionally with instance overrides.
206 m_rbLibraryModel->SetValue( true );
207
208 if( !loadLibrary( libraryFilename, reporter ) )
209 {
210 if( reporter.HasMessage() )
211 m_infoBar->ShowMessage( reporter.GetMessages() );
212
213 m_libraryPathText->ChangeValue( libraryFilename );
214 m_curModelType = SIM_MODEL::ReadTypeFromFields( m_fields, reporter );
215
216 m_libraryModelsMgr.CreateModel( nullptr, m_sortedPartPins, m_fields, reporter );
217
218 m_modelListBox->Append( _( "<unknown>" ) );
219 m_modelListBox->SetSelection( 0 );
220 }
221 else
222 {
223 std::string modelName = GetFieldValue( &m_fields, SIM_LIBRARY::NAME_FIELD );
224 int modelIdx = m_modelListBox->FindString( modelName );
225
226 if( modelIdx == wxNOT_FOUND )
227 {
228 m_infoBar->ShowMessage( wxString::Format( _( "No model named '%s' in library." ),
229 modelName ) );
230
231 // Default to first item in library
232 m_modelListBox->SetSelection( 0 );
233 }
234 else
235 {
236 m_infoBar->Hide();
237 m_modelListBox->SetSelection( modelIdx );
238 }
239
240 m_curModelType = curModel().GetType();
241 }
242
243 if( isIbisLoaded() && ( m_modelListBox->GetSelection() >= 0 ) )
244 {
245 int idx = 0;
246 wxString sel = m_modelListBox->GetStringSelection();
247
248 if( m_modelListBoxEntryToLibraryIdx.contains( sel ) )
249 idx = m_modelListBoxEntryToLibraryIdx.at( sel );
250
251 auto ibismodel =
252 dynamic_cast<SIM_MODEL_IBIS*>( &m_libraryModelsMgr.GetModels()[idx].get() );
253
254 if( ibismodel )
255 {
256 onModelNameChoice( dummyEvent ); // refresh list of pins
257
258 int i = 0;
259
260 for( const std::pair<std::string, std::string>& strs : ibismodel->GetIbisPins() )
261 {
262 if( strs.first == GetFieldValue( &m_fields, SIM_LIBRARY_IBIS::PIN_FIELD ) )
263 {
264 auto ibisLibrary = static_cast<const SIM_LIBRARY_IBIS*>( library() );
265
266 ibismodel->ChangePin( *ibisLibrary, strs.first );
267 m_pinCombobox->SetSelection( static_cast<int>( i ) );
268 break;
269 }
270 i++;
271 }
272
273 if( i < static_cast<int>( ibismodel->GetIbisPins().size() ) )
274 {
275 onPinCombobox( dummyEvent ); // refresh list of models
276
277 m_pinModelCombobox->SetStringSelection( GetFieldValue( &m_fields, SIM_LIBRARY_IBIS::MODEL_FIELD ) );
278 }
279
280 if( GetFieldValue( &m_fields, SIM_LIBRARY_IBIS::DIFF_FIELD ) == "1" )
281 {
282 ibismodel->SwitchSingleEndedDiff( true );
283 m_differentialCheckbox->SetValue( true );
284 }
285 else
286 {
287 ibismodel->SwitchSingleEndedDiff( false );
288 m_differentialCheckbox->SetValue( false );
289 }
290 }
291 }
292 }
293 else if( !GetFieldValue( &m_fields, SIM_DEVICE_FIELD ).empty()
294 || !GetFieldValue( &m_fields, SIM_DEVICE_SUBTYPE_FIELD ).empty() )
295 {
296 // The model is sourced from the instance.
297 m_rbBuiltinModel->SetValue( true );
298
299 reporter.Clear();
300 m_curModelType = SIM_MODEL::ReadTypeFromFields( m_fields, reporter );
301
302 if( reporter.HasMessage() )
303 DisplayErrorMessage( this, reporter.GetMessages() );
304 }
305
306 for( SIM_MODEL::TYPE type : SIM_MODEL::TYPE_ITERATOR() )
307 {
308 if( m_rbBuiltinModel->GetValue() && type == m_curModelType )
309 {
310 reporter.Clear();
311 m_builtinModelsMgr.CreateModel( m_fields, m_sortedPartPins, false, reporter );
312
313 if( reporter.HasMessage() )
314 {
315 DisplayErrorMessage( this, _( "Failed to read simulation model from fields." )
316 + wxT( "\n\n" ) + reporter.GetMessages() );
317 }
318 }
319 else
320 {
321 m_builtinModelsMgr.CreateModel( type, m_sortedPartPins, reporter );
322 }
323
324 SIM_MODEL::DEVICE_T deviceTypeT = SIM_MODEL::TypeInfo( type ).deviceType;
325
326 if( !m_curModelTypeOfDeviceType.count( deviceTypeT ) )
327 m_curModelTypeOfDeviceType[deviceTypeT] = type;
328 }
329
330 if( storeInValue )
331 curModel().SetIsStoredInValue( true );
332
333 m_saveInValueCheckbox->SetValue( curModel().IsStoredInValue() );
334
335 onRadioButton( dummyEvent );
336 return DIALOG_SIM_MODEL_BASE::TransferDataToWindow();
337}
338
339
340template <typename T>
342{
343 m_pinAssignmentsGrid->CommitPendingChanges();
344 m_paramGrid->GetGrid()->CommitChangesFromEditor();
345
346 if( !DIALOG_SIM_MODEL_BASE::TransferDataFromWindow() )
347 return false;
348
349 SIM_MODEL& model = curModel();
350 std::string path;
351 std::string name;
352
353 if( m_rbLibraryModel->GetValue() )
354 {
355 path = m_libraryPathText->GetValue();
356 wxFileName fn( path );
357
358 if( !path.starts_with( FILEEXT::KiCadUriPrefix ) && fn.MakeRelativeTo( Prj().GetProjectPath() )
359 && !fn.GetFullPath().StartsWith( ".." ) )
360 {
361 path = fn.GetFullPath();
362 }
363
364 if( m_modelListBox->GetSelection() >= 0 )
365 name = m_modelListBox->GetStringSelection().ToStdString();
366 else if( dynamic_cast<SIM_MODEL_SPICE_FALLBACK*>( &model ) )
367 name = GetFieldValue( &m_fields, SIM_LIBRARY::NAME_FIELD, false );
368 }
369
370 SetFieldValue( m_fields, SIM_LIBRARY::LIBRARY_FIELD, path, false );
371 SetFieldValue( m_fields, SIM_LIBRARY::NAME_FIELD, name, false );
372
373 if( isIbisLoaded() )
374 {
375 int idx = 0;
376 wxString sel = m_modelListBox->GetStringSelection();
377
378 if( m_modelListBoxEntryToLibraryIdx.contains( sel ) )
379 idx = m_modelListBoxEntryToLibraryIdx.at( sel );
380
381 auto* ibismodel =
382 static_cast<SIM_MODEL_IBIS*>( &m_libraryModelsMgr.GetModels().at( idx ).get() );
383
384 if( ibismodel )
385 {
386 std::string pins;
387 std::string modelName = std::string( m_pinModelCombobox->GetValue().c_str() );
388 std::string differential;
389
390 if( m_pinCombobox->GetSelection() >= 0 )
391 pins = ibismodel->GetIbisPins().at( m_pinCombobox->GetSelection() ).first;
392
393 if( ibismodel->CanDifferential() && m_differentialCheckbox->GetValue() )
394 differential = "1";
395
397 SetFieldValue( m_fields, SIM_LIBRARY_IBIS::MODEL_FIELD, modelName );
398 SetFieldValue( m_fields, SIM_LIBRARY_IBIS::DIFF_FIELD, differential );
399 }
400 }
401
402 if( model.GetType() == SIM_MODEL::TYPE::RAWSPICE )
403 {
404 if( m_modelNotebook->GetSelection() == 0 )
405 updateModelCodeTab( &model );
406
407 wxString code = m_codePreview->GetText().Trim( true ).Trim( false );
408 model.SetParamValue( "model", std::string( code.ToUTF8() ) );
409 }
410
411 model.SetIsStoredInValue( m_saveInValueCheckbox->GetValue() );
412
413 for( int row = 0; row < m_pinAssignmentsGrid->GetNumberRows(); ++row )
414 {
415 wxString modelPinName = m_pinAssignmentsGrid->GetCellValue( row, PIN_COLUMN::MODEL );
416 wxString symbolPinName = m_sortedPartPins.at( row )->GetShownNumber();
417
418 model.AssignSymbolPinNumberToModelPin( getModelPinIndex( modelPinName ),
419 std::string( symbolPinName.ToUTF8() ) );
420 }
421
422 removeOrphanedPinAssignments( &model );
423
424 curModel().WriteFields( m_fields );
425
426 return true;
427}
428
429
430template <typename T>
432{
433 // always enable the library browser button -- it makes for fewer clicks if the user has a
434 // whole bunch of inferred passives that they want to specify library models for
435 m_browseButton->Enable();
436
437 // if we're in an undetermined state then enable everything for faster access
438 bool undetermined = !m_rbLibraryModel->GetValue() && !m_rbBuiltinModel->GetValue();
439 bool enableLibCtrls = m_rbLibraryModel->GetValue() || undetermined;
440 bool enableBuiltinCtrls = m_rbBuiltinModel->GetValue() || undetermined;
441
442 m_pathLabel->Enable( enableLibCtrls );
443 m_libraryPathText->Enable( enableLibCtrls );
444 m_modelNameLabel->Enable( enableLibCtrls );
445 m_modelFilter->Enable( enableLibCtrls && !isIbisLoaded() );
446 m_modelListBox->Enable( enableLibCtrls );
447 m_pinLabel->Enable( enableLibCtrls );
448 m_pinCombobox->Enable( enableLibCtrls );
449 m_differentialCheckbox->Enable( enableLibCtrls );
450 m_pinModelLabel->Enable( enableLibCtrls );
451 m_pinModelCombobox->Enable( enableLibCtrls );
452 m_waveformLabel->Enable( enableLibCtrls );
453 m_waveformChoice->Enable( enableLibCtrls );
454
455 m_deviceLabel->Enable( enableBuiltinCtrls );
456 m_deviceChoice->Enable( enableBuiltinCtrls );
457 m_deviceSubtypeLabel->Enable( enableBuiltinCtrls );
458 m_deviceSubtypeChoice->Enable( enableBuiltinCtrls );
459
460 SIM_MODEL* model = &curModel();
461
462 updateIbisWidgets( model );
463 updateBuiltinModelWidgets( model );
464 updateModelParamsTab( model );
465 updateModelCodeTab( model );
466 updatePinAssignments( model, model != m_prevModel );
467
468 std::string ref = GetFieldValue( &m_fields, SIM_REFERENCE_FIELD );
469
470 m_modelPanel->Layout();
471 m_pinAssignmentsPanel->Layout();
472 m_parametersPanel->Layout();
473 m_codePanel->Layout();
474
475 SendSizeEvent( wxSEND_EVENT_POST );
476
477 m_prevModel = &curModel();
478}
479
480
481template <typename T>
483{
484 SIM_MODEL_IBIS* modelibis = isIbisLoaded() ? dynamic_cast<SIM_MODEL_IBIS*>( aModel )
485 : nullptr;
486
487 m_pinLabel->Show( isIbisLoaded() );
488 m_pinCombobox->Show( isIbisLoaded() );
489 m_pinModelLabel->Show( isIbisLoaded() );
490 m_pinModelCombobox->Show( isIbisLoaded() );
491 m_waveformLabel->Show( isIbisLoaded() );
492 m_waveformChoice->Show( isIbisLoaded() );
493
494 if( aModel != m_prevModel )
495 {
496 m_waveformChoice->Clear();
497
498 if( isIbisLoaded() )
499 {
500 for( SIM_MODEL::TYPE type : { SIM_MODEL::TYPE::KIBIS_DEVICE,
501 SIM_MODEL::TYPE::KIBIS_DRIVER_DC,
502 SIM_MODEL::TYPE::KIBIS_DRIVER_RECT,
503 SIM_MODEL::TYPE::KIBIS_DRIVER_PRBS } )
504 {
505 SIM_MODEL::DEVICE_T deviceType = SIM_MODEL::TypeInfo( type ).deviceType;
506 const std::string& deviceTypeDesc = SIM_MODEL::DeviceInfo( deviceType ).description;
507
508 if( deviceType == aModel->GetDeviceType()
509 || deviceTypeDesc == aModel->GetDeviceInfo().description )
510 {
511 m_waveformChoice->Append( SIM_MODEL::TypeInfo( type ).description );
512
513 if( type == aModel->GetType() )
514 m_waveformChoice->SetSelection( m_waveformChoice->GetCount() - 1 );
515 }
516 }
517 }
518 }
519
520 m_differentialCheckbox->Show( isIbisLoaded() && modelibis && modelibis->CanDifferential() );
521 m_modelNameLabel->SetLabel( isIbisLoaded() ? _( "Component:" ) : _( "Model:" ) );
522}
523
524
525template <typename T>
527{
528 // Change the Type choice to match the current device type.
529 if( aModel != m_prevModel )
530 {
531 m_deviceChoice->Clear();
532 m_deviceSubtypeChoice->Clear();
533
534 if( !m_rbLibraryModel->GetValue() )
535 {
536 for( SIM_MODEL::DEVICE_T deviceType : SIM_MODEL::DEVICE_T_ITERATOR() )
537 {
538 if( !SIM_MODEL::DeviceInfo( deviceType ).showInMenu )
539 continue;
540
541 m_deviceChoice->Append( SIM_MODEL::DeviceInfo( deviceType ).description );
542
543 if( equivalent( deviceType, aModel->GetDeviceType() ) )
544 m_deviceChoice->SetSelection( m_deviceChoice->GetCount() - 1 );
545 }
546
547 for( SIM_MODEL::TYPE type : SIM_MODEL::TYPE_ITERATOR() )
548 {
549 if( type == SIM_MODEL::TYPE::KIBIS_DEVICE
550 || type == SIM_MODEL::TYPE::KIBIS_DRIVER_DC
551 || type == SIM_MODEL::TYPE::KIBIS_DRIVER_RECT
552 || type == SIM_MODEL::TYPE::KIBIS_DRIVER_PRBS )
553 {
554 continue;
555 }
556
557 SIM_MODEL::DEVICE_T deviceType = SIM_MODEL::TypeInfo( type ).deviceType;
558 const std::string& deviceTypeDesc = SIM_MODEL::DeviceInfo( deviceType ).description;
559
560 if( deviceType == aModel->GetDeviceType()
561 || deviceTypeDesc == aModel->GetDeviceInfo().description )
562 {
563 m_deviceSubtypeChoice->Append( SIM_MODEL::TypeInfo( type ).description );
564
565 if( type == aModel->GetType() )
566 m_deviceSubtypeChoice->SetSelection( m_deviceSubtypeChoice->GetCount()
567 - 1 );
568 }
569 }
570 }
571
572 m_deviceSubtypeLabel->Show( m_deviceSubtypeChoice->GetCount() > 1 );
573 m_deviceSubtypeChoice->Show( m_deviceSubtypeChoice->GetCount() > 1 );
574 }
575
576 if( dynamic_cast<SIM_MODEL_RAW_SPICE*>( aModel ) )
577 m_modelNotebook->SetSelection( 1 );
578 else
579 m_modelNotebook->SetSelection( 0 );
580
581 if( aModel->HasPrimaryValue() )
582 {
583 const SIM_MODEL::PARAM& primary = aModel->GetParam( 0 );
584
585 m_saveInValueCheckbox->SetLabel( wxString::Format( _( "Save parameter '%s (%s)' in Value "
586 "field" ),
587 primary.info.description,
588 primary.info.name ) );
589 m_saveInValueCheckbox->Enable( true );
590 }
591 else
592 {
593 m_saveInValueCheckbox->SetLabel( _( "Save primary parameter in Value field" ) );
594 m_saveInValueCheckbox->SetValue( false );
595 m_saveInValueCheckbox->Enable( false );
596 }
597}
598
599
600template <typename T>
602{
603 if( aModel != m_prevModel )
604 {
605 // This wxPropertyGridManager column and header stuff has to be here because it segfaults in
606 // the constructor.
607
608 m_paramGridMgr->SetColumnCount( PARAM_COLUMN::END_ );
609
610 m_paramGridMgr->SetColumnTitle( PARAM_COLUMN::DESCRIPTION, _( "Parameter" ) );
611 m_paramGridMgr->SetColumnTitle( PARAM_COLUMN::UNIT, _( "Unit" ) );
612 m_paramGridMgr->SetColumnTitle( PARAM_COLUMN::DEFAULT, _( "Default" ) );
613 m_paramGridMgr->SetColumnTitle( PARAM_COLUMN::TYPE, _( "Type" ) );
614
615 m_paramGridMgr->ShowHeader();
616
617
618 m_paramGrid->Clear();
619
620 m_firstCategory = m_paramGrid->Append( new wxPropertyCategory( "Geometry" ) );
621 m_paramGrid->HideProperty( "Geometry" );
622
623 m_paramGrid->Append( new wxPropertyCategory( "AC" ) );
624 m_paramGrid->HideProperty( "AC" );
625
626 m_paramGrid->Append( new wxPropertyCategory( "DC" ) );
627 m_paramGrid->HideProperty( "DC" );
628
629 m_paramGrid->Append( new wxPropertyCategory( "S-Parameters" ) );
630 m_paramGrid->HideProperty( "S-Parameters" );
631
632 m_paramGrid->Append( new wxPropertyCategory( "Capacitance" ) );
633 m_paramGrid->HideProperty( "Capacitance" );
634
635 m_paramGrid->Append( new wxPropertyCategory( "Temperature" ) );
636 m_paramGrid->HideProperty( "Temperature" );
637
638 m_paramGrid->Append( new wxPropertyCategory( "Noise" ) );
639 m_paramGrid->HideProperty( "Noise" );
640
641 m_paramGrid->Append( new wxPropertyCategory( "Distributed Quantities" ) );
642 m_paramGrid->HideProperty( "Distributed Quantities" );
643
644 m_paramGrid->Append( new wxPropertyCategory( "Waveform" ) );
645 m_paramGrid->HideProperty( "Waveform" );
646
647 m_paramGrid->Append( new wxPropertyCategory( "Limiting Values" ) );
648 m_paramGrid->HideProperty( "Limiting Values" );
649
650 m_paramGrid->Append( new wxPropertyCategory( "Advanced" ) );
651 m_paramGrid->HideProperty( "Advanced" );
652
653 m_paramGrid->Append( new wxPropertyCategory( "Flags" ) );
654 m_paramGrid->HideProperty( "Flags" );
655
656 m_paramGrid->CollapseAll();
657
658 for( int i = 0; i < aModel->GetParamCount(); ++i )
659 addParamPropertyIfRelevant( aModel, i );
660
661 m_paramGrid->CollapseAll();
662 m_paramGrid->Expand( "AC" );
663 m_paramGrid->Expand( "Waveform" );
664 }
665
666 adjustParamGridColumns( m_paramGrid->GetGrid()->GetSize().GetX(), true );
667
668 // Set all properties to default colors.
669 // Update properties in models that have autofill.
670 for( wxPropertyGridIterator it = m_paramGrid->GetIterator(); !it.AtEnd(); ++it )
671 {
672 wxColour bgCol = m_paramGrid->GetGrid()->GetPropertyDefaultCell().GetBgCol();
673 wxColour fgCol = m_paramGrid->GetGrid()->GetPropertyDefaultCell().GetFgCol();
674
675 for( int col = 0; col < m_paramGridMgr->GetColumnCount(); ++col )
676 {
677 ( *it )->GetCell( col ).SetBgCol( bgCol );
678 ( *it )->GetCell( col ).SetFgCol( fgCol );
679 }
680
681 SIM_PROPERTY* prop = dynamic_cast<SIM_PROPERTY*>( *it );
682
683 if( !prop )
684 continue;
685
686 const SIM_MODEL::PARAM& param = prop->GetParam();
687
688 // Model values other than the currently edited value may have changed. Update them.
689 // This feature is called "autofill" and present only in certain models. Don't do it for
690 // models that don't have it for performance reasons.
691 if( aModel->HasAutofill() )
692 ( *it )->SetValueFromString( param.value );
693 }
694}
695
696
697template <typename T>
699{
700 if( dynamic_cast<SIM_MODEL_SPICE_FALLBACK*>( aModel ) )
701 return;
702
703 wxString text;
704 SPICE_ITEM item;
705
706 item.modelName = m_modelListBox->GetStringSelection();
707
708 if( m_rbBuiltinModel->GetValue() || item.modelName == "" )
709 item.modelName = GetFieldValue( &m_fields, FIELD_T::REFERENCE );
710
711 text << aModel->SpiceGenerator().Preview( item );
712
713 m_codePreview->SetText( text );
714 m_codePreview->SelectNone();
715}
716
717
718template <typename T>
719void DIALOG_SIM_MODEL<T>::updatePinAssignments( SIM_MODEL* aModel, bool aForceUpdatePins )
720{
721 if( m_pinAssignmentsGrid->GetNumberRows() == 0 )
722 {
723 m_pinAssignmentsGrid->AppendRows( static_cast<int>( m_sortedPartPins.size() ) );
724
725 for( int ii = 0; ii < m_pinAssignmentsGrid->GetNumberRows(); ++ii )
726 {
727 wxString symbolPinString = getSymbolPinString( ii );
728
729 m_pinAssignmentsGrid->SetReadOnly( ii, PIN_COLUMN::SYMBOL );
730 m_pinAssignmentsGrid->SetCellValue( ii, PIN_COLUMN::SYMBOL, symbolPinString );
731 }
732
733 aForceUpdatePins = true;
734 }
735
736 if( aForceUpdatePins )
737 {
738 // Reset the grid.
739 for( int row = 0; row < m_pinAssignmentsGrid->GetNumberRows(); ++row )
740 m_pinAssignmentsGrid->SetCellValue( row, PIN_COLUMN::MODEL, _( "Not Connected" ) );
741
742 // Now set up the grid values in the Model column.
743 for( int modelPinIndex = 0; modelPinIndex < aModel->GetPinCount(); ++modelPinIndex )
744 {
745 wxString symbolPinNumber = aModel->GetPin( modelPinIndex ).symbolPinNumber;
746
747 if( symbolPinNumber == "" )
748 continue;
749
750 int symbolPinRow = findSymbolPinRow( symbolPinNumber );
751
752 if( symbolPinRow == -1 )
753 continue;
754
755 wxString modelPinString = getModelPinString( aModel, modelPinIndex );
756 m_pinAssignmentsGrid->SetCellValue( symbolPinRow, PIN_COLUMN::MODEL, modelPinString );
757 }
758 }
759
760 for( int ii = 0; ii < m_pinAssignmentsGrid->GetNumberRows(); ++ii )
761 {
762 // Set up the Model column cell editors with dropdown options.
763 std::vector<BITMAPS> modelPinIcons;
764 wxArrayString modelPinChoices;
765
766 for( int jj = 0; jj < aModel->GetPinCount(); ++jj )
767 {
768 if( aModel->GetPin( jj ).symbolPinNumber != "" )
769 modelPinIcons.push_back( PinShapeGetBitmap( GRAPHIC_PINSHAPE::LINE ) );
770 else
771 modelPinIcons.push_back( BITMAPS::INVALID_BITMAP );
772
773 modelPinChoices.Add( getModelPinString( aModel, jj ) );
774 }
775
776 modelPinIcons.push_back( BITMAPS::INVALID_BITMAP );
777 modelPinChoices.Add( _( "Not Connected" ) );
778
779 // Using `new` here shouldn't cause a memory leak because `SetCellEditor()` calls
780 // `DecRef()` on its last editor.
781 m_pinAssignmentsGrid->SetCellEditor( ii, PIN_COLUMN::MODEL,
782 new GRID_CELL_ICON_TEXT_POPUP( modelPinIcons,
783 modelPinChoices ) );
784 }
785
786 // TODO: Show a preview of the symbol with the pin numbers shown.
787
788 if( aModel->GetType() == SIM_MODEL::TYPE::SUBCKT )
789 {
790 SIM_MODEL_SUBCKT* subckt = static_cast<SIM_MODEL_SUBCKT*>( aModel );
791 m_subckt->SetText( subckt->GetSpiceCode() );
792 m_subckt->SetEditable( false );
793 }
794 else
795 {
796 m_subcktLabel->Show( false );
797 m_subckt->Show( false );
798 }
799}
800
801
802template <typename T>
804{
805 for( int i = 0; i < aModel->GetPinCount(); ++i )
806 {
807 if( !m_symbol.GetPin( aModel->GetPin( i ).symbolPinNumber ) )
808 aModel->AssignSymbolPinNumberToModelPin( i, "" );
809 }
810}
811
812
813template <typename T>
814bool DIALOG_SIM_MODEL<T>::loadLibrary( const wxString& aLibraryPath, REPORTER& aReporter,
815 bool aForceReload )
816{
817 if( m_prevLibrary == aLibraryPath && !aForceReload )
818 return true;
819
820 m_libraryModelsMgr.SetForceFullParse();
821 m_libraryModelsMgr.SetLibrary( aLibraryPath, aReporter );
822
824 return false;
825
826 std::string modelName = GetFieldValue( &m_fields, SIM_LIBRARY::NAME_FIELD );
827
828 for( const auto& [baseModelName, baseModel] : library()->GetModels() )
829 {
830 if( baseModelName == modelName )
831 m_libraryModelsMgr.CreateModel( &baseModel, m_sortedPartPins, m_fields, aReporter );
832 else
833 m_libraryModelsMgr.CreateModel( &baseModel, m_sortedPartPins, aReporter );
834 }
835
836 m_rbLibraryModel->SetValue( true );
837 m_libraryPathText->ChangeValue( aLibraryPath );
838
839 m_modelListBoxEntryToLibraryIdx.clear();
840 wxArrayString modelNames;
841
842 bool modelNameExists = false;
843 for( const auto& [name, model] : library()->GetModels() )
844 {
845 modelNames.Add( name );
846 m_modelListBoxEntryToLibraryIdx[name] = m_modelListBoxEntryToLibraryIdx.size();
847 if( name == modelName )
848 modelNameExists = true;
849 }
850
851 modelNames.Sort();
852
853 m_modelListBox->Clear();
854 m_modelListBox->Append( modelNames );
855
856 if( !modelNameExists )
857 {
858 m_infoBar->ShowMessage(
859 wxString::Format( _( "No model named '%s' in '%s'." ), modelName, aLibraryPath ) );
860 return false;
861 }
862
863 if( isIbisLoaded() )
864 {
865 wxArrayString emptyArray;
866 m_pinModelCombobox->Set( emptyArray );
867 m_pinCombobox->Set( emptyArray );
868 m_pinModelCombobox->SetSelection( -1 );
869 m_pinCombobox->SetSelection( -1 );
870 }
871
872 m_modelListBox->SetStringSelection( modelName );
873
874 if( m_modelListBox->GetSelection() < 0 && m_modelListBox->GetCount() > 0 )
875 m_modelListBox->SetSelection( 0 );
876
877 m_curModelType = curModel().GetType();
878
879 m_prevLibrary = aLibraryPath;
880 return true;
881}
882
883
884template <typename T>
886{
887 if( aModel->GetParam( aParamIndex ).info.dir == SIM_MODEL::PARAM::DIR_OUT )
888 return;
889
890 switch( aModel->GetParam( aParamIndex ).info.category )
891 {
892 case CATEGORY::AC:
893 m_paramGrid->HideProperty( "AC", false );
894 m_paramGrid->AppendIn( "AC", newParamProperty( aModel, aParamIndex ) );
895 break;
896
897 case CATEGORY::DC:
898 m_paramGrid->HideProperty( "DC", false );
899 m_paramGrid->AppendIn( "DC", newParamProperty( aModel, aParamIndex ) );
900 break;
901
902 case CATEGORY::S_PARAM:
903 m_paramGrid->HideProperty( "S-Parameters", false );
904 m_paramGrid->AppendIn( "S-Parameters", newParamProperty( aModel, aParamIndex ) );
905 break;
906
907 case CATEGORY::CAPACITANCE:
908 m_paramGrid->HideProperty( "Capacitance", false );
909 m_paramGrid->AppendIn( "Capacitance", newParamProperty( aModel, aParamIndex ) );
910 break;
911
912 case CATEGORY::TEMPERATURE:
913 m_paramGrid->HideProperty( "Temperature", false );
914 m_paramGrid->AppendIn( "Temperature", newParamProperty( aModel, aParamIndex ) );
915 break;
916
917 case CATEGORY::NOISE:
918 m_paramGrid->HideProperty( "Noise", false );
919 m_paramGrid->AppendIn( "Noise", newParamProperty( aModel, aParamIndex ) );
920 break;
921
922 case CATEGORY::DISTRIBUTED_QUANTITIES:
923 m_paramGrid->HideProperty( "Distributed Quantities", false );
924 m_paramGrid->AppendIn( "Distributed Quantities", newParamProperty( aModel, aParamIndex ) );
925 break;
926
927 case CATEGORY::WAVEFORM:
928 m_paramGrid->HideProperty( "Waveform", false );
929 m_paramGrid->AppendIn( "Waveform", newParamProperty( aModel, aParamIndex ) );
930 break;
931
932 case CATEGORY::GEOMETRY:
933 m_paramGrid->HideProperty( "Geometry", false );
934 m_paramGrid->AppendIn( "Geometry", newParamProperty( aModel, aParamIndex ) );
935 break;
936
937 case CATEGORY::LIMITING_VALUES:
938 m_paramGrid->HideProperty( "Limiting Values", false );
939 m_paramGrid->AppendIn( "Limiting Values", newParamProperty( aModel, aParamIndex ) );
940 break;
941
942 case CATEGORY::ADVANCED:
943 m_paramGrid->HideProperty( "Advanced", false );
944 m_paramGrid->AppendIn( "Advanced", newParamProperty( aModel, aParamIndex ) );
945 break;
946
947 case CATEGORY::FLAGS:
948 m_paramGrid->HideProperty( "Flags", false );
949 m_paramGrid->AppendIn( "Flags", newParamProperty( aModel, aParamIndex ) );
950 break;
951
952 default:
953 m_paramGrid->Insert( m_firstCategory, newParamProperty( aModel, aParamIndex ) );
954 break;
955
956 case CATEGORY::INITIAL_CONDITIONS:
957 case CATEGORY::SUPERFLUOUS:
958 return;
959 }
960}
961
962
963template <typename T>
964wxPGProperty* DIALOG_SIM_MODEL<T>::newParamProperty( SIM_MODEL* aModel, int aParamIndex ) const
965{
966 const SIM_MODEL::PARAM& param = aModel->GetParam( aParamIndex );
967 wxString paramDescription;
968
969 if( param.info.description == "" )
970 paramDescription = wxString::Format( "%s", param.info.name );
971 else
972 paramDescription = wxString::Format( "%s (%s)", param.info.description, param.info.name );
973
974 wxPGProperty* prop = nullptr;
975
976 switch( param.info.type )
977 {
979 // TODO.
980 prop = new SIM_BOOL_PROPERTY( paramDescription, param.info.name, *aModel, aParamIndex );
981 prop->SetAttribute( wxPG_BOOL_USE_CHECKBOX, true );
982 break;
983
985 prop = new SIM_STRING_PROPERTY( paramDescription, param.info.name, *aModel, aParamIndex,
987 break;
988
990 prop = new SIM_STRING_PROPERTY( paramDescription, param.info.name, *aModel, aParamIndex,
992 break;
993
994 //case TYPE_COMPLEX:
995 // break;
996
998 // Special case: K-line mutual inductance statement parameters l1 and l2 are references
999 // to other inductors in the circuit.
1000 if( dynamic_cast<SIM_MODEL_L_MUTUAL*>( aModel ) != nullptr
1001 && ( param.info.name == "l1" || param.info.name == "l2" ) )
1002 {
1003 wxArrayString inductors;
1004
1005 if( SCH_EDIT_FRAME* schEditFrame = dynamic_cast<SCH_EDIT_FRAME*>( m_frame ) )
1006 {
1007 SPICE_CIRCUIT_MODEL circuit( &schEditFrame->Schematic() );
1008 NULL_REPORTER devNul;
1009
1011 devNul );
1012
1013 for( const SPICE_ITEM& item : circuit.GetItems() )
1014 {
1015 if( item.model->GetDeviceType() == SIM_MODEL::DEVICE_T::L )
1016 inductors.push_back( item.refName );
1017 }
1018
1019 inductors.Sort(
1020 []( const wxString& a, const wxString& b ) -> int
1021 {
1022 return StrNumCmp( a, b, true );
1023 } );
1024 }
1025
1026 if( inductors.empty() )
1027 {
1028 prop = new SIM_STRING_PROPERTY( paramDescription, param.info.name, *aModel,
1029 aParamIndex, SIM_VALUE::TYPE_STRING );
1030 }
1031 else
1032 {
1033 prop = new SIM_ENUM_PROPERTY( paramDescription, param.info.name, *aModel,
1034 aParamIndex, inductors );
1035 }
1036 }
1037 else if( param.info.enumValues.empty() )
1038 {
1039 prop = new SIM_STRING_PROPERTY( paramDescription, param.info.name, *aModel,
1040 aParamIndex, SIM_VALUE::TYPE_STRING );
1041 }
1042 else
1043 {
1044 wxArrayString values;
1045
1046 for( const std::string& string : aModel->GetParam( aParamIndex ).info.enumValues )
1047 values.Add( string );
1048
1049 prop = new SIM_ENUM_PROPERTY( paramDescription, param.info.name, *aModel, aParamIndex,
1050 values );
1051 }
1052 break;
1053
1054 default:
1055 prop = new wxStringProperty( paramDescription, param.info.name );
1056 break;
1057 }
1058
1059 prop->SetAttribute( wxPG_ATTR_UNITS, wxString::FromUTF8( param.info.unit.c_str() ) );
1060
1061 // Legacy due to the way we extracted the parameters from Ngspice.
1062 prop->SetCell( 3, wxString::FromUTF8( param.info.defaultValue ) );
1063
1064 wxString typeStr;
1065
1066 switch( param.info.type )
1067 {
1068 case SIM_VALUE::TYPE_BOOL: typeStr = wxT( "Bool" ); break;
1069 case SIM_VALUE::TYPE_INT: typeStr = wxT( "Int" ); break;
1070 case SIM_VALUE::TYPE_FLOAT: typeStr = wxT( "Float" ); break;
1071 case SIM_VALUE::TYPE_COMPLEX: typeStr = wxT( "Complex" ); break;
1072 case SIM_VALUE::TYPE_STRING: typeStr = wxT( "String" ); break;
1073 case SIM_VALUE::TYPE_BOOL_VECTOR: typeStr = wxT( "Bool Vector" ); break;
1074 case SIM_VALUE::TYPE_INT_VECTOR: typeStr = wxT( "Int Vector" ); break;
1075 case SIM_VALUE::TYPE_FLOAT_VECTOR: typeStr = wxT( "Float Vector" ); break;
1076 case SIM_VALUE::TYPE_COMPLEX_VECTOR: typeStr = wxT( "Complex Vector" ); break;
1077 }
1078
1079 prop->SetCell( PARAM_COLUMN::TYPE, typeStr );
1080
1081 return prop;
1082}
1083
1084
1085template <typename T>
1086int DIALOG_SIM_MODEL<T>::findSymbolPinRow( const wxString& aSymbolPinNumber ) const
1087{
1088 for( int row = 0; row < static_cast<int>( m_sortedPartPins.size() ); ++row )
1089 {
1090 SCH_PIN* pin = m_sortedPartPins[row];
1091
1092 if( pin->GetNumber() == aSymbolPinNumber )
1093 return row;
1094 }
1095
1096 return -1;
1097}
1098
1099
1100template <typename T>
1102{
1103 if( m_rbLibraryModel->GetValue() )
1104 {
1105 wxString sel = m_modelListBox->GetStringSelection();
1106
1107 if( m_modelListBoxEntryToLibraryIdx.contains( sel ) )
1108 return m_libraryModelsMgr.GetModels()
1109 .at( m_modelListBoxEntryToLibraryIdx.at( sel ) )
1110 .get();
1111 }
1112 else
1113 {
1114 if( static_cast<int>( m_curModelType )
1115 < static_cast<int>( m_builtinModelsMgr.GetModels().size() ) )
1116 return m_builtinModelsMgr.GetModels().at( static_cast<int>( m_curModelType ) );
1117 }
1118
1119 return m_builtinModelsMgr.GetModels().at( static_cast<int>( SIM_MODEL::TYPE::NONE ) );
1120}
1121
1122
1123template <typename T>
1125{
1126 if( m_libraryModelsMgr.GetLibraries().size() == 1 )
1127 return &m_libraryModelsMgr.GetLibraries().begin()->second.get();
1128
1129 return nullptr;
1130}
1131
1132
1133template <typename T>
1134wxString DIALOG_SIM_MODEL<T>::getSymbolPinString( int symbolPinIndex ) const
1135{
1136 SCH_PIN* pin = m_sortedPartPins.at( symbolPinIndex );
1137 wxString pinNumber;
1138 wxString pinName;
1139
1140 if( pin )
1141 {
1142 pinNumber = pin->GetShownNumber();
1143 pinName = pin->GetShownName();
1144 }
1145
1146 if( !pinName.IsEmpty() && pinName != pinNumber )
1147 pinNumber += wxString::Format( wxT( " (\"%s\")" ), pinName );
1148
1149 return pinNumber;
1150}
1151
1152
1153template <typename T>
1154wxString DIALOG_SIM_MODEL<T>::getModelPinString( SIM_MODEL* aModel, int aModelPinIndex ) const
1155{
1156 const wxString& modelPinName = aModel->GetPin( aModelPinIndex ).modelPinName;
1157
1158 LOCALE_IO toggle;
1159
1160 wxString modelPinNumber = wxString::Format( "%d", aModelPinIndex + 1 );
1161
1162 if( !modelPinName.IsEmpty() && modelPinName != modelPinNumber )
1163 modelPinNumber += wxString::Format( wxT( " (\"%s\")" ), modelPinName );
1164
1165 return modelPinNumber;
1166}
1167
1168
1169template <typename T>
1170int DIALOG_SIM_MODEL<T>::getModelPinIndex( const wxString& aModelPinString ) const
1171{
1172 if( aModelPinString == "Not Connected" )
1174
1175 int length = aModelPinString.Find( " " );
1176
1177 if( length == wxNOT_FOUND )
1178 length = static_cast<int>( aModelPinString.Length() );
1179
1180 long result = 0;
1181 aModelPinString.Mid( 0, length ).ToCLong( &result );
1182
1183 return static_cast<int>( result - 1 );
1184}
1185
1186
1187template <typename T>
1188void DIALOG_SIM_MODEL<T>::onRadioButton( wxCommandEvent& aEvent )
1189{
1190 m_prevModel = nullptr; // Ensure the Model panel will be rebuild after updating other params.
1191 updateWidgets();
1192}
1193
1194
1195template <typename T>
1196void DIALOG_SIM_MODEL<T>::onLibraryPathText( wxCommandEvent& aEvent )
1197{
1198 m_rbLibraryModel->SetValue( true );
1199}
1200
1201
1202template <typename T>
1204{
1205 m_rbLibraryModel->SetValue( true );
1206
1207 WX_STRING_REPORTER reporter;
1208 wxString path = m_libraryPathText->GetValue();
1209
1210 if( loadLibrary( path, reporter, true ) || path.IsEmpty() )
1211 m_infoBar->Hide();
1212 else if( reporter.HasMessage() )
1213 m_infoBar->ShowMessage( reporter.GetMessages() );
1214
1215 updateWidgets();
1216}
1217
1218
1219template <typename T>
1221{
1222 CallAfter(
1223 [this]()
1224 {
1225 // Disable logging -- otherwise we'll end up in an endless loop of show-log,
1226 // kill-focus, show-log, kill-focus, etc.
1227 wxLogNull doNotLog;
1228
1229 wxCommandEvent dummy;
1230 onLibraryPathTextEnter( dummy );
1231 } );
1232
1233 aEvent.Skip(); // mandatory in wxFocusEvent events
1234}
1235
1236
1237template <typename T>
1238void DIALOG_SIM_MODEL<T>::onBrowseButtonClick( wxCommandEvent& aEvent )
1239{
1240 static wxString s_mruPath;
1241
1242 wxString path = s_mruPath.IsEmpty() ? Prj().GetProjectPath() : s_mruPath;
1243 wxFileDialog dlg( this, _( "Browse Models" ), path );
1244 FILEDLG_OPEN_EMBED_FILE customize( false );
1245
1246 dlg.SetCustomizeHook( customize );
1247
1248 if( dlg.ShowModal() == wxID_CANCEL )
1249 return;
1250
1251 m_rbLibraryModel->SetValue( true );
1252
1253 path = dlg.GetPath();
1254 wxFileName fn( path );
1255 s_mruPath = fn.GetPath();
1256
1257 if( customize.GetEmbed() )
1258 {
1259 EMBEDDED_FILES::EMBEDDED_FILE* result = m_files->AddFile( fn, false );
1260 path = result->GetLink();
1261 }
1262 else if( fn.MakeRelativeTo( Prj().GetProjectPath() ) && !fn.GetFullPath().StartsWith( wxS( ".." ) ) )
1263 path = fn.GetFullPath();
1264
1265 WX_STRING_REPORTER reporter;
1266
1267 if( loadLibrary( path, reporter, true ) )
1268 m_infoBar->Hide();
1269 else
1270 m_infoBar->ShowMessage( reporter.GetMessages() );
1271
1272 updateWidgets();
1273}
1274
1275
1276template <typename T>
1277void DIALOG_SIM_MODEL<T>::onFilterCharHook( wxKeyEvent& aKeyStroke )
1278{
1279 int sel = m_modelListBox->GetSelection();
1280
1281 switch( aKeyStroke.GetKeyCode() )
1282 {
1283 case WXK_UP:
1284 if( sel == wxNOT_FOUND )
1285 sel = m_modelListBox->GetCount() - 1;
1286 else
1287 sel--;
1288
1289 break;
1290
1291 case WXK_DOWN:
1292 if( sel == wxNOT_FOUND )
1293 sel = 0;
1294 else
1295 sel++;
1296
1297 break;
1298
1299 case WXK_RETURN:
1300 wxPostEvent( this, wxCommandEvent( wxEVT_COMMAND_BUTTON_CLICKED, wxID_OK ) );
1301 return;
1302
1303 default:
1304 aKeyStroke.Skip(); // Any other key: pass on to search box directly.
1305 return;
1306 }
1307
1308 if( sel >= 0 && sel < (int) m_modelListBox->GetCount() )
1309 m_modelListBox->SetSelection( sel );
1310}
1311
1312
1313template <typename T>
1314void DIALOG_SIM_MODEL<T>::onModelFilter( wxCommandEvent& aEvent )
1315{
1316 wxArrayString modelNames;
1317 wxString current = m_modelListBox->GetStringSelection();
1318 wxString filter = wxT( "*" ) + m_modelFilter->GetValue() + wxT( "*" );
1319
1320 for( const auto& [name, model] : library()->GetModels() )
1321 {
1322 wxString wx_name( name );
1323
1324 if( wx_name.Matches( filter ) )
1325 modelNames.Add( wx_name );
1326 }
1327
1328 modelNames.Sort();
1329
1330 m_modelListBox->Clear();
1331 m_modelListBox->Append( modelNames );
1332
1333 if( !m_modelListBox->SetStringSelection( current ) )
1334 m_modelListBox->SetSelection( 0 );
1335}
1336
1337
1338template <typename T>
1339void DIALOG_SIM_MODEL<T>::onModelNameChoice( wxCommandEvent& aEvent )
1340{
1341 if( isIbisLoaded() )
1342 {
1343 wxArrayString pinLabels;
1344 SIM_MODEL_IBIS* modelkibis = dynamic_cast<SIM_MODEL_IBIS*>( &curModel() );
1345
1346 wxCHECK2( modelkibis, return );
1347
1348 for( std::pair<wxString, wxString> strs : modelkibis->GetIbisPins() )
1349 pinLabels.Add( strs.first + wxT( " - " ) + strs.second );
1350
1351 m_pinCombobox->Set( pinLabels );
1352
1353 wxArrayString emptyArray;
1354 m_pinModelCombobox->Set( emptyArray );
1355 }
1356
1357 m_rbLibraryModel->SetValue( true );
1358
1359 if( SIM_MODEL_SPICE_FALLBACK* fallback =
1360 dynamic_cast<SIM_MODEL_SPICE_FALLBACK*>( &curModel() ) )
1361 {
1362 wxArrayString lines = wxSplit( fallback->GetSpiceCode(), '\n' );
1363 wxString code;
1364
1365 for( const wxString& line : lines )
1366 {
1367 if( !line.StartsWith( '*' ) )
1368 {
1369 if( !code.IsEmpty() )
1370 code += "\n";
1371
1372 code += line;
1373 }
1374 }
1375
1376 m_infoBar->ShowMessage( wxString::Format( _( "Failed to parse:\n\n"
1377 "%s\n"
1378 "Using generic SPICE model." ),
1379 code ) );
1380 }
1381 else
1382 {
1383 m_infoBar->Hide();
1384 }
1385
1386 updateWidgets();
1387}
1388
1389
1390template <typename T>
1391void DIALOG_SIM_MODEL<T>::onPinCombobox( wxCommandEvent& aEvent )
1392{
1393 wxArrayString modelLabels;
1394
1395 SIM_MODEL_IBIS& ibisModel = static_cast<SIM_MODEL_IBIS&>( curModel() );
1396
1397 std::vector<std::pair<std::string, std::string>> strs = ibisModel.GetIbisPins();
1398 std::string pinNumber = strs.at( m_pinCombobox->GetSelection() ).first;
1399
1400 const SIM_LIBRARY_IBIS* ibisLibrary = dynamic_cast<const SIM_LIBRARY_IBIS*>( library() );
1401
1402 ibisModel.ChangePin( *ibisLibrary, pinNumber );
1403
1404 ibisModel.m_enableDiff = ibisLibrary->isPinDiff( ibisModel.GetComponentName(), pinNumber );
1405
1406 for( wxString modelName : ibisModel.GetIbisModels() )
1407 modelLabels.Add( modelName );
1408
1409 m_pinModelCombobox->Set( modelLabels );
1410
1411 if( m_pinModelCombobox->GetCount() == 1 )
1412 m_pinModelCombobox->SetSelection( 0 );
1413 else
1414 m_pinModelCombobox->SetSelection( -1 );
1415
1416 updateWidgets();
1417}
1418
1419
1420template <typename T>
1422{
1423 m_pinCombobox->SetSelection( m_pinCombobox->FindString( m_pinCombobox->GetValue() ) );
1424
1425 onPinModelCombobox( aEvent );
1426}
1427
1428
1429template <typename T>
1430void DIALOG_SIM_MODEL<T>::onPinModelCombobox( wxCommandEvent& aEvent )
1431{
1432 updateWidgets();
1433}
1434
1435
1436template <typename T>
1438{
1439 m_pinModelCombobox->SetSelection(
1440 m_pinModelCombobox->FindString( m_pinModelCombobox->GetValue() ) );
1441}
1442
1443
1444template <typename T>
1446{
1447 if( SIM_MODEL_IBIS* modelibis = dynamic_cast<SIM_MODEL_IBIS*>( &curModel() ) )
1448 {
1449 bool diff = m_differentialCheckbox->GetValue() && modelibis->CanDifferential();
1450 modelibis->SwitchSingleEndedDiff( diff );
1451
1452 updateWidgets();
1453 }
1454}
1455
1456
1457template <typename T>
1458void DIALOG_SIM_MODEL<T>::onDeviceTypeChoice( wxCommandEvent& aEvent )
1459{
1460 m_rbBuiltinModel->SetValue( true );
1461
1462 for( SIM_MODEL::DEVICE_T deviceType : SIM_MODEL::DEVICE_T_ITERATOR() )
1463 {
1464 if( SIM_MODEL::DeviceInfo( deviceType ).description
1465 == m_deviceChoice->GetStringSelection() )
1466 {
1467 m_curModelType = m_curModelTypeOfDeviceType.at( deviceType );
1468 break;
1469 }
1470 }
1471
1472 updateWidgets();
1473}
1474
1475
1476template <typename T>
1477void DIALOG_SIM_MODEL<T>::onWaveformChoice( wxCommandEvent& aEvent )
1478{
1479 SIM_MODEL::DEVICE_T deviceType = curModel().GetDeviceType();
1480 wxString typeDescription = m_waveformChoice->GetStringSelection();
1481
1482 for( SIM_MODEL::TYPE type : { SIM_MODEL::TYPE::KIBIS_DEVICE,
1483 SIM_MODEL::TYPE::KIBIS_DRIVER_DC,
1484 SIM_MODEL::TYPE::KIBIS_DRIVER_RECT,
1485 SIM_MODEL::TYPE::KIBIS_DRIVER_PRBS } )
1486 {
1487 if( equivalent( deviceType, SIM_MODEL::TypeInfo( type ).deviceType )
1488 && typeDescription == SIM_MODEL::TypeInfo( type ).description )
1489 {
1490 int idx = 0;
1491 wxString sel = m_modelListBox->GetStringSelection();
1492
1493 if( m_modelListBoxEntryToLibraryIdx.contains( sel ) )
1494 idx = m_modelListBoxEntryToLibraryIdx.at( sel );
1495
1496 auto& baseModel =
1497 static_cast<SIM_MODEL_IBIS&>( m_libraryModelsMgr.GetModels()[idx].get() );
1498
1499 m_libraryModelsMgr.SetModel( idx, std::make_unique<SIM_MODEL_IBIS>( type, baseModel ) );
1500
1501 try
1502 {
1503 m_libraryModelsMgr.GetModels()[idx].get().ReadDataFields( &m_fields,
1504 m_sortedPartPins );
1505 }
1506 catch( IO_ERROR& err )
1507 {
1508 DisplayErrorMessage( this, err.What() );
1509 }
1510
1511 m_curModelType = type;
1512 break;
1513 }
1514 }
1515
1516 m_curModelTypeOfDeviceType.at( deviceType ) = m_curModelType;
1517 updateWidgets();
1518}
1519
1520
1521template <typename T>
1522void DIALOG_SIM_MODEL<T>::onTypeChoice( wxCommandEvent& aEvent )
1523{
1524 SIM_MODEL::DEVICE_T deviceType = curModel().GetDeviceType();
1525 wxString typeDescription = m_deviceSubtypeChoice->GetStringSelection();
1526
1527 for( SIM_MODEL::TYPE type : SIM_MODEL::TYPE_ITERATOR() )
1528 {
1529 if( equivalent( deviceType, SIM_MODEL::TypeInfo( type ).deviceType )
1530 && typeDescription == SIM_MODEL::TypeInfo( type ).description )
1531 {
1532 m_curModelType = type;
1533 break;
1534 }
1535 }
1536
1537 m_curModelTypeOfDeviceType.at( deviceType ) = m_curModelType;
1538 updateWidgets();
1539}
1540
1541
1542template <typename T>
1543void DIALOG_SIM_MODEL<T>::onPageChanging( wxBookCtrlEvent& event )
1544{
1545 updateModelCodeTab( &curModel() );
1546}
1547
1548
1549template <typename T>
1551{
1552 int symbolPinIndex = aEvent.GetRow();
1553 wxString oldModelPinName = aEvent.GetString();
1554 wxString modelPinName = m_pinAssignmentsGrid->GetCellValue( aEvent.GetRow(), aEvent.GetCol() );
1555
1556 int oldModelPinIndex = getModelPinIndex( oldModelPinName );
1557 int modelPinIndex = getModelPinIndex( modelPinName );
1558
1559 if( oldModelPinIndex != SIM_MODEL_PIN::NOT_CONNECTED )
1560 curModel().AssignSymbolPinNumberToModelPin( oldModelPinIndex, "" );
1561
1562 if( modelPinIndex != SIM_MODEL_PIN::NOT_CONNECTED )
1563 {
1564 SCH_PIN* symbolPin = m_sortedPartPins.at( symbolPinIndex );
1565
1566 curModel().AssignSymbolPinNumberToModelPin( modelPinIndex, symbolPin->GetShownNumber() );
1567 }
1568
1569 updatePinAssignments( &curModel(), FORCE_UPDATE_PINS );
1570
1571 aEvent.Skip();
1572}
1573
1574
1575template <typename T>
1577{
1578 wxGridUpdateLocker deferRepaintsTillLeavingScope( m_pinAssignmentsGrid );
1579
1580 int gridWidth = KIPLATFORM::UI::GetUnobscuredSize( m_pinAssignmentsGrid ).x;
1581 m_pinAssignmentsGrid->SetColSize( PIN_COLUMN::MODEL, gridWidth / 2 );
1582 m_pinAssignmentsGrid->SetColSize( PIN_COLUMN::SYMBOL, gridWidth / 2 );
1583
1584 aEvent.Skip();
1585}
1586
1587
1588template <typename T>
1590{
1591 // By default, when a property grid is focused, the textbox is not immediately focused until
1592 // Tab key is pressed. This is inconvenient, so we fix that here.
1593
1594 wxPropertyGrid* grid = m_paramGrid->GetGrid();
1595 wxPGProperty* selected = grid->GetSelection();
1596
1597 if( !selected )
1598 selected = grid->wxPropertyGridInterface::GetFirst();
1599
1600#if wxCHECK_VERSION( 3, 3, 0 )
1601 if( selected )
1602 grid->DoSelectProperty( selected, wxPGSelectPropertyFlags::Focus );
1603#else
1604 if( selected )
1605 grid->DoSelectProperty( selected, wxPG_SEL_FOCUS );
1606#endif
1607
1608 aEvent.Skip();
1609}
1610
1611
1612template <typename T>
1613void DIALOG_SIM_MODEL<T>::onParamGridSelectionChange( wxPropertyGridEvent& aEvent )
1614{
1615 wxPropertyGrid* grid = m_paramGrid->GetGrid();
1616
1617 // Jump over categories.
1618 if( grid->GetSelection() && grid->GetSelection()->IsCategory() )
1619 {
1620 wxPGProperty* selection = grid->GetSelection();
1621
1622 // If the new selection is immediately above the previous selection, we jump up. Otherwise
1623 // we jump down. We do this by simulating up or down arrow keys.
1624
1625 wxPropertyGridIterator it = grid->GetIterator( wxPG_ITERATE_VISIBLE, selection );
1626 it.Next();
1627
1628 wxKeyEvent* keyEvent = new wxKeyEvent( wxEVT_KEY_DOWN );
1629
1630 if( *it == m_prevParamGridSelection )
1631 {
1632 if( !selection->IsExpanded() )
1633 {
1634 grid->Expand( selection );
1635 keyEvent->m_keyCode = WXK_DOWN;
1636 wxQueueEvent( grid, keyEvent );
1637
1638 // Does not work for some reason.
1639 /*m_paramGrid->DoSelectProperty( selection->Item( selection->GetChildCount() - 1 ),
1640 wxPG_SEL_FOCUS );*/
1641 }
1642 else
1643 {
1644 keyEvent->m_keyCode = WXK_UP;
1645 wxQueueEvent( grid, keyEvent );
1646 }
1647 }
1648 else
1649 {
1650 if( !selection->IsExpanded() )
1651 grid->Expand( selection );
1652
1653 keyEvent->m_keyCode = WXK_DOWN;
1654 wxQueueEvent( grid, keyEvent );
1655 }
1656
1657 m_prevParamGridSelection = grid->GetSelection();
1658 return;
1659 }
1660
1661 wxWindow* editorControl = grid->GetEditorControl();
1662
1663 if( !editorControl )
1664 {
1665 m_prevParamGridSelection = grid->GetSelection();
1666 return;
1667 }
1668
1669 // Without this the user had to press tab before they could edit the field.
1670 editorControl->SetFocus();
1671 m_prevParamGridSelection = grid->GetSelection();
1672}
1673
1674
1675template <typename T>
1676void DIALOG_SIM_MODEL<T>::onUpdateUI( wxUpdateUIEvent& aEvent )
1677{
1678 // This is currently patched in wxPropertyGrid::ScrollWindow() in the Mac wxWidgets fork.
1679 // However, we may need this version if it turns out to be an issue on other platforms and
1680 // we can't get it upstreamed.
1681#if 0
1682 // It's a shame to do this on the UpdateUI event, but neither the wxPropertyGridManager,
1683 // wxPropertyGridPage, wxPropertyGrid, nor the wxPropertyGrid's GetCanvas() window appear
1684 // to get scroll events.
1685
1686 wxPropertyGrid* grid = m_paramGrid->GetGrid();
1687 wxTextCtrl* ctrl = grid->GetEditorTextCtrl();
1688
1689 if( ctrl )
1690 {
1691 wxRect ctrlRect = ctrl->GetScreenRect();
1692 wxRect gridRect = grid->GetScreenRect();
1693
1694 if( ctrlRect.GetTop() < gridRect.GetTop() || ctrlRect.GetBottom() > gridRect.GetBottom() )
1695 grid->ClearSelection();
1696 }
1697#endif
1698}
1699
1700
1701template <typename T>
1702void DIALOG_SIM_MODEL<T>::adjustParamGridColumns( int aWidth, bool aForce )
1703{
1704 wxPropertyGrid* grid = m_paramGridMgr->GetGrid();
1705 int margin = 15;
1706 int indent = 20;
1707
1708 if( aWidth != m_lastParamGridWidth || aForce )
1709 {
1710 m_lastParamGridWidth = aWidth;
1711
1712 grid->FitColumns();
1713
1714 std::vector<int> colWidths;
1715
1716 for( size_t ii = 0; ii < grid->GetColumnCount(); ii++ )
1717 {
1718 if( ii == PARAM_COLUMN::DESCRIPTION )
1719 colWidths.push_back( grid->GetState()->GetColumnWidth( ii ) + margin + indent );
1720 else if( ii == PARAM_COLUMN::VALUE )
1721 colWidths.push_back( std::max( 72,
1722 grid->GetState()->GetColumnWidth( ii ) ) + margin );
1723 else
1724 colWidths.push_back( 60 + margin );
1725
1726 aWidth -= colWidths[ ii ];
1727 }
1728
1729 for( size_t ii = 0; ii < grid->GetColumnCount(); ii++ )
1730 grid->SetColumnProportion( ii, colWidths[ ii ] );
1731
1732 grid->ResetColumnSizes();
1733 grid->RefreshEditor();
1734 }
1735}
1736
1737
1738template <typename T>
1740{
1741 adjustParamGridColumns( event.GetSize().GetX(), false );
1742
1743 event.Skip();
1744}
1745
1746
1747template class DIALOG_SIM_MODEL<SCH_SYMBOL>;
1748template class DIALOG_SIM_MODEL<LIB_SYMBOL>;
const char * name
Definition: DXF_plotter.cpp:59
wxBitmapBundle KiBitmapBundle(BITMAPS aBitmap, int aMinHeight)
Definition: bitmap.cpp:110
@ INVALID_BITMAP
void finishDialogSettings()
In all dialogs, we must call the same functions to fix minimal dlg size, the default position and per...
Class DIALOG_SIM_MODEL_BASE.
wxPropertyGridManager * m_paramGridMgr
STD_BITMAP_BUTTON * m_browseButton
wxStyledTextCtrl * m_subckt
wxPropertyGridPage * m_paramGrid
wxStyledTextCtrl * m_codePreview
EMBEDDED_FILES * m_files
void onTypeChoice(wxCommandEvent &aEvent) override
void onLibraryPathTextKillFocus(wxFocusEvent &aEvent) override
wxString getSymbolPinString(int aSymbolPinNumber) const
void updateBuiltinModelWidgets(SIM_MODEL *aModel)
void onPinAssignmentsGridCellChange(wxGridEvent &aEvent) override
void onFilterCharHook(wxKeyEvent &aKeyStroke) override
int findSymbolPinRow(const wxString &aSymbolPinNumber) const
void onModelNameChoice(wxCommandEvent &aEvent) override
void onRadioButton(wxCommandEvent &aEvent) override
SCINTILLA_TRICKS * m_scintillaTricksSubckt
bool loadLibrary(const wxString &aLibraryPath, REPORTER &aReporter, bool aForceReload=false)
SIM_LIB_MGR m_builtinModelsMgr
int getModelPinIndex(const wxString &aModelPinString) const
void onModelFilter(wxCommandEvent &aEvent) override
void onLibraryPathText(wxCommandEvent &aEvent) override
void onDifferentialCheckbox(wxCommandEvent &event) override
void onParamGridSelectionChange(wxPropertyGridEvent &aEvent)
void removeOrphanedPinAssignments(SIM_MODEL *aModel)
void adjustParamGridColumns(int aWidth, bool aForce)
std::vector< SCH_PIN * > m_sortedPartPins
Pins of the current part.
wxPGProperty * newParamProperty(SIM_MODEL *aModel, int aParamIndex) const
void onPinModelCombobox(wxCommandEvent &event) override
const SIM_LIBRARY * library() const
SIM_MODEL & curModel() const
SIM_LIB_MGR m_libraryModelsMgr
void onBrowseButtonClick(wxCommandEvent &aEvent) override
void onPinComboboxTextEnter(wxCommandEvent &event) override
void onPinCombobox(wxCommandEvent &event) override
SCINTILLA_TRICKS * m_scintillaTricksCode
void updatePinAssignments(SIM_MODEL *aModel, bool aForceUpdatePins)
void onUpdateUI(wxUpdateUIEvent &aEvent)
DIALOG_SIM_MODEL(wxWindow *aParent, EDA_BASE_FRAME *aFrame, T &aSymbol, std::vector< SCH_FIELD > &aFields)
wxString getModelPinString(SIM_MODEL *aModel, int aModelPinIndex) const
void onPinAssignmentsGridSize(wxSizeEvent &aEvent) override
void updateIbisWidgets(SIM_MODEL *aModel)
void onParamGridSetFocus(wxFocusEvent &aEvent)
void updateModelParamsTab(SIM_MODEL *aModel)
bool TransferDataFromWindow() override
void updateModelCodeTab(SIM_MODEL *aModel)
void onSizeParamGrid(wxSizeEvent &event) override
void onPageChanging(wxNotebookEvent &event) override
void onWaveformChoice(wxCommandEvent &aEvent) override
void onPinModelComboboxTextEnter(wxCommandEvent &event) override
void onLibraryPathTextEnter(wxCommandEvent &aEvent) override
void onDeviceTypeChoice(wxCommandEvent &aEvent) override
bool TransferDataToWindow() override
void addParamPropertyIfRelevant(SIM_MODEL *aModel, int aParamIndex)
The base frame for deriving all KiCad main window classes.
Add mouse and command handling (such as cut, copy, and paste) to a WX_GRID instance.
Definition: grid_tricks.h:61
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
Definition: ki_exception.h:77
virtual const wxString What() const
A composite of Problem() and Where()
Definition: exceptions.cpp:30
Instantiate the current locale within a scope in which you are expecting exceptions to be thrown.
Definition: locale_io.h:49
virtual bool ReadSchematicAndLibraries(unsigned aNetlistOptions, REPORTER &aReporter)
Process the schematic and Spice libraries to create net mapping and a list of SPICE_ITEMs.
const std::list< SPICE_ITEM > & GetItems() const
Return the list of items representing schematic symbols in the Spice world.
A singleton reporter that reports to nowhere.
Definition: reporter.h:217
virtual const wxString GetProjectPath() const
Return the full path of the project.
Definition: project.cpp:146
A pure virtual class used to derive REPORTER objects from.
Definition: reporter.h:73
virtual bool HasMessageOfSeverity(int aSeverityMask) const
Returns true if the reporter has one or more messages matching the specified severity mask.
Definition: reporter.h:140
Schematic editor (Eeschema) main window.
void SetText(const wxString &aText) override
Definition: sch_field.cpp:1069
wxString GetShownNumber() const
Definition: sch_pin.cpp:539
Add cut/copy/paste, dark theme, autocomplete and brace highlighting to a wxStyleTextCtrl instance.
static constexpr auto MODEL_FIELD
static constexpr auto PIN_FIELD
static constexpr auto DIFF_FIELD
bool isPinDiff(const std::string &aComp, const std::string &aPinNumber) const
static constexpr auto LIBRARY_FIELD
Definition: sim_library.h:35
static constexpr auto NAME_FIELD
Definition: sim_library.h:36
void SetFiles(EMBEDDED_FILES *aFiles)
Definition: sim_lib_mgr.h:49
std::vector< std::pair< std::string, std::string > > GetIbisPins() const
bool CanDifferential() const
std::vector< std::string > GetIbisModels() const
bool ChangePin(const SIM_LIBRARY_IBIS &aLib, const std::string &aPinNumber)
update the list of available models based on the pin number.
std::string GetComponentName() const
std::string GetSpiceCode() const
static TYPE ReadTypeFromFields(const std::vector< SCH_FIELD > &aFields, REPORTER &aReporter)
Definition: sim_model.cpp:390
static INFO TypeInfo(TYPE aType)
Definition: sim_model.cpp:105
int GetPinCount() const
Definition: sim_model.h:464
void ReadDataFields(const std::vector< SCH_FIELD > *aFields, const std::vector< SCH_PIN * > &aPins)
Definition: sim_model.cpp:429
const SPICE_GENERATOR & SpiceGenerator() const
Definition: sim_model.h:428
virtual const PARAM & GetParam(unsigned aParamIndex) const
Definition: sim_model.cpp:726
static bool InferSimModel(T &aSymbol, std::vector< SCH_FIELD > *aFields, bool aResolve, SIM_VALUE_GRAMMAR::NOTATION aNotation, wxString *aDeviceType, wxString *aModelType, wxString *aModelParams, wxString *aPinMap)
Definition: sim_model.cpp:1019
int GetParamCount() const
Definition: sim_model.h:474
void AssignSymbolPinNumberToModelPin(int aPinIndex, const wxString &aSymbolPinNumber)
Definition: sim_model.cpp:695
DEVICE_INFO GetDeviceInfo() const
Definition: sim_model.h:453
DEVICE_T GetDeviceType() const
Definition: sim_model.h:456
static DEVICE_INFO DeviceInfo(DEVICE_T aDeviceType)
Definition: sim_model.cpp:60
virtual bool HasAutofill() const
Definition: sim_model.h:491
void SetParamValue(int aParamIndex, const std::string &aValue, SIM_VALUE::NOTATION aNotation=SIM_VALUE::NOTATION::SI)
Definition: sim_model.cpp:782
const SIM_MODEL_PIN & GetPin(unsigned aIndex) const
Definition: sim_model.h:465
void SetIsStoredInValue(bool aIsStoredInValue)
Definition: sim_model.h:497
virtual bool HasPrimaryValue() const
Definition: sim_model.h:492
TYPE GetType() const
Definition: sim_model.h:457
const SIM_MODEL::PARAM & GetParam() const
Definition: sim_property.h:60
@ TYPE_BOOL
Definition: sim_value.h:67
@ TYPE_FLOAT_VECTOR
Definition: sim_value.h:75
@ TYPE_BOOL_VECTOR
Definition: sim_value.h:73
@ TYPE_INT
Definition: sim_value.h:68
@ TYPE_FLOAT
Definition: sim_value.h:69
@ TYPE_INT_VECTOR
Definition: sim_value.h:74
@ TYPE_COMPLEX_VECTOR
Definition: sim_value.h:76
@ TYPE_STRING
Definition: sim_value.h:71
@ TYPE_COMPLEX
Definition: sim_value.h:70
Special netlist exporter flavor that allows one to override simulation commands.
virtual std::string Preview(const SPICE_ITEM &aItem) const
void SetBitmap(const wxBitmapBundle &aBmp)
void ClearRows()
wxWidgets recently added an ASSERT which fires if the position is greater than or equal to the number...
Definition: wx_grid.h:193
void AddCloseButton(const wxString &aTooltip=_("Hide this message."))
Add the default close button to the infobar on the right side.
Definition: wx_infobar.cpp:343
A wrapper for reporting to a wxString object.
Definition: reporter.h:190
void Clear() override
Definition: reporter.cpp:89
bool HasMessage() const override
Returns true if the reporter client is non-empty.
Definition: reporter.cpp:96
const wxString & GetMessages() const
Definition: reporter.cpp:83
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.
static bool empty(const wxTextEntryBase *aCtrl)
bool equivalent(SIM_MODEL::DEVICE_T a, SIM_MODEL::DEVICE_T b)
#define FORCE_UPDATE_PINS
#define _(s)
@ NONE
Definition: eda_shape.h:69
static const std::string KiCadUriPrefix
PROJECT & Prj()
Definition: kicad.cpp:597
wxSize GetUnobscuredSize(const wxWindow *aWindow)
Tries to determine the size of the viewport of a scrollable widget (wxDataViewCtrl,...
Definition: wxgtk/ui.cpp:252
KICOMMON_API wxFont GetInfoFont(wxWindow *aWindow)
Definition: ui_common.cpp:156
BITMAPS PinShapeGetBitmap(GRAPHIC_PINSHAPE aShape)
Definition: pin_type.cpp:254
@ RPT_SEVERITY_ERROR
@ RPT_SEVERITY_UNDEFINED
wxString GetFieldValue(const std::vector< SCH_FIELD > *aFields, FIELD_T aFieldType)
Definition: sch_field.h:406
const SCH_FIELD * FindField(const std::vector< SCH_FIELD > &aFields, FIELD_T aFieldId)
Definition: sch_field.h:367
void SetFieldValue(std::vector< SCH_FIELD > &aFields, const wxString &aFieldName, const std::string &aValue, bool aIsVisible=true)
Definition: sch_field.h:431
SIM_MODEL::TYPE TYPE
Definition: sim_model.cpp:57
#define SIM_PINS_FIELD
Definition: sim_model.h:54
#define SIM_DEVICE_FIELD
Definition: sim_model.h:52
#define SIM_REFERENCE_FIELD
Definition: sim_model.h:49
#define SIM_PARAMS_FIELD
Definition: sim_model.h:55
#define SIM_DEVICE_SUBTYPE_FIELD
Definition: sim_model.h:53
std::vector< FAB_LAYER_COLOR > dummy
int StrNumCmp(const wxString &aString1, const wxString &aString2, bool aIgnoreCase)
Compare two strings with alphanumerical content.
std::vector< std::string > enumValues
Definition: sim_model.h:388
SIM_VALUE::TYPE type
Definition: sim_model.h:379
std::string defaultValue
Definition: sim_model.h:382
std::string description
Definition: sim_model.h:383
std::string value
Definition: sim_model.h:400
const INFO & info
Definition: sim_model.h:401
static constexpr auto NOT_CONNECTED
Definition: sim_model.h:73
const std::string modelPinName
Definition: sim_model.h:70
wxString symbolPinNumber
Definition: sim_model.h:71
std::string modelName
@ REFERENCE
Field Reference of part, i.e. "IC21".
@ VALUE
Field Value of part, i.e. "3.3K".