KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
simulator_frame_ui.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) 2016-2023 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 * @author Tomasz Wlostowski <tomasz.wlostowski@cern.ch>
7 * @author Maciej Suminski <maciej.suminski@cern.ch>
8 *
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version 3
12 * of the License, or (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, you may find one here:
21 * https://www.gnu.org/licenses/gpl-3.0.html
22 * or you may search the http://www.gnu.org website for the version 3 license,
23 * or you may write to the Free Software Foundation, Inc.,
24 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
25 */
26
27#include <memory>
28#include <type_traits>
29
30#include <wx/event.h>
31#include <fmt/format.h>
32#include <wx/wfstream.h>
33#include <wx/stdstream.h>
34#include <wx/debug.h>
35#include <wx/clipbrd.h>
36#include <wx/log.h>
37
39#include <sch_edit_frame.h>
40#include <confirm.h>
44#include <widgets/wx_grid.h>
45#include <grid_tricks.h>
46#include <eda_pattern_match.h>
47#include <string_utils.h>
48#include <pgm_base.h>
50#include <sim/simulator_frame.h>
51#include <sim/sim_plot_tab.h>
52#include <sim/spice_simulator.h>
55#include <eeschema_settings.h>
56
57
59{
60 int res = static_cast<int>( aFirst ) | static_cast<int>( aSecond);
61
62 return static_cast<SIM_TRACE_TYPE>( res );
63}
64
65
67{
73};
74
75
77{
82};
83
84
86{
90};
91
92
93enum
94{
104
108
109
111{
112public:
114 GRID_TRICKS( aGrid ),
115 m_parent( aParent ),
116 m_menuRow( 0 ),
117 m_menuCol( 0 )
118 {}
119
120protected:
121 void showPopupMenu( wxMenu& menu, wxGridEvent& aEvent ) override;
122 void doPopupSelection( wxCommandEvent& event ) override;
123
124protected:
128};
129
130
131void SIGNALS_GRID_TRICKS::showPopupMenu( wxMenu& menu, wxGridEvent& aEvent )
132{
134
135 if( !panel )
136 return;
137
138 m_menuRow = aEvent.GetRow();
139 m_menuCol = aEvent.GetCol();
140
142 {
143 if( !( m_grid->IsInSelection( m_menuRow, m_menuCol ) ) )
144 m_grid->ClearSelection();
145
146 m_grid->SetGridCursor( m_menuRow, m_menuCol );
147
148 if( panel->GetSimType() == ST_TRAN || panel->GetSimType() == ST_AC
149 || panel->GetSimType() == ST_DC || panel->GetSimType() == ST_SP )
150 {
151 menu.Append( MYID_MEASURE_MIN, _( "Measure Min" ) );
152 menu.Append( MYID_MEASURE_MAX, _( "Measure Max" ) );
153 menu.Append( MYID_MEASURE_AVG, _( "Measure Average" ) );
154 menu.Append( MYID_MEASURE_RMS, _( "Measure RMS" ) );
155 menu.Append( MYID_MEASURE_PP, _( "Measure Peak-to-peak" ) );
156
157 if( panel->GetSimType() == ST_AC || panel->GetSimType() == ST_SP )
158 {
159 menu.Append( MYID_MEASURE_MIN_AT, _( "Measure Frequency of Min" ) );
160 menu.Append( MYID_MEASURE_MAX_AT, _( "Measure Frequency of Max" ) );
161 }
162 else
163 {
164 menu.Append( MYID_MEASURE_MIN_AT, _( "Measure Time of Min" ) );
165 menu.Append( MYID_MEASURE_MAX_AT, _( "Measure Time of Max" ) );
166 }
167
168 menu.Append( MYID_MEASURE_INTEGRAL, _( "Measure Integral" ) );
169
170 if( panel->GetSimType() == ST_TRAN )
171 {
172 menu.AppendSeparator();
173 menu.Append( MYID_FOURIER, _( "Perform Fourier Analysis..." ) );
174 }
175
176 menu.AppendSeparator();
177 menu.Append( GRIDTRICKS_ID_COPY, _( "Copy Signal Name" ) + "\tCtrl+C" );
178
179 menu.AppendSeparator();
180 menu.Append( GRIDTRICKS_ID_SELECT, _( "Create new cursor..." ) );
181
182 m_grid->PopupMenu( &menu );
183 }
184 }
185 else if( m_menuCol > static_cast<int>( COL_CURSOR_2 ) )
186 {
187 menu.Append( GRIDTRICKS_ID_SELECT, _( "Create new cursor..." ) );
188
189 menu.AppendSeparator();
190
191 wxString msg = m_grid->GetColLabelValue( m_grid->GetNumberCols() - 1 );
192
193 menu.AppendSeparator();
194 menu.Append( GRIDTRICKS_ID_DELETE, wxString::Format( _( "Delete %s..." ), msg ) );
195
196 m_grid->PopupMenu( &menu );
197 }
198 else
199 {
200 menu.Append( GRIDTRICKS_ID_SELECT, _( "Create new cursor..." ) );
201
202 m_grid->PopupMenu( &menu );
203 }
204}
205
206
207void SIGNALS_GRID_TRICKS::doPopupSelection( wxCommandEvent& event )
208{
209 std::vector<wxString> signals;
210
211 wxGridCellCoordsArray cells1 = m_grid->GetSelectionBlockTopLeft();
212 wxGridCellCoordsArray cells2 = m_grid->GetSelectionBlockBottomRight();
213
214 for( size_t i = 0; i < cells1.Count(); i++ )
215 {
216 if( cells1[i].GetCol() == COL_SIGNAL_NAME )
217 {
218 for( int j = cells1[i].GetRow(); j < cells2[i].GetRow() + 1; j++ )
219 {
220 signals.push_back( m_grid->GetCellValue( j, cells1[i].GetCol() ) );
221 }
222 }
223 }
224
225 wxGridCellCoordsArray cells3 = m_grid->GetSelectedCells();
226
227 for( size_t i = 0; i < cells3.Count(); i++ )
228 {
229 if( cells3[i].GetCol() == COL_SIGNAL_NAME )
230 signals.push_back( m_grid->GetCellValue( cells3[i].GetRow(), cells3[i].GetCol() ) );
231 }
232
233 if( signals.size() < 1 )
234 signals.push_back( m_grid->GetCellValue( m_menuRow, m_menuCol ) );
235
236 auto addMeasurement =
237 [this]( const wxString& cmd, wxString signal )
238 {
239 if( signal.EndsWith( _( " (phase)" ) ) )
240 return;
241
242 if( signal.EndsWith( _( " (gain)" ) ) || signal.EndsWith( _( " (amplitude)" ) ) )
243 {
244 signal = signal.Left( signal.length() - 7 );
245
246 if( signal.Upper().StartsWith( wxS( "V(" ) ) )
247 signal = wxS( "vdb" ) + signal.Mid( 1 );
248 }
249
250 m_parent->AddMeasurement( cmd + wxS( " " ) + signal );
251 };
252
253 if( event.GetId() == MYID_MEASURE_MIN )
254 {
255 for( const wxString& signal : signals )
256 addMeasurement( wxS( "MIN" ), signal );
257 }
258 else if( event.GetId() == MYID_MEASURE_MAX )
259 {
260 for( const wxString& signal : signals )
261 addMeasurement( wxS( "MAX" ), signal );
262 }
263 else if( event.GetId() == MYID_MEASURE_AVG )
264 {
265 for( const wxString& signal : signals )
266 addMeasurement( wxS( "AVG" ), signal );
267 }
268 else if( event.GetId() == MYID_MEASURE_RMS )
269 {
270 for( const wxString& signal : signals )
271 addMeasurement( wxS( "RMS" ), signal );
272 }
273 else if( event.GetId() == MYID_MEASURE_PP )
274 {
275 for( const wxString& signal : signals )
276 addMeasurement( wxS( "PP" ), signal );
277 }
278 else if( event.GetId() == MYID_MEASURE_MIN_AT )
279 {
280 for( const wxString& signal : signals )
281 addMeasurement( wxS( "MIN_AT" ), signal );
282 }
283 else if( event.GetId() == MYID_MEASURE_MAX_AT )
284 {
285 for( const wxString& signal : signals )
286 addMeasurement( wxS( "MAX_AT" ), signal );
287 }
288 else if( event.GetId() == MYID_MEASURE_INTEGRAL )
289 {
290 for( const wxString& signal : signals )
291 addMeasurement( wxS( "INTEG" ), signal );
292 }
293 else if( event.GetId() == MYID_FOURIER )
294 {
295 wxString title;
296 wxString fundamental = wxT( "1K" );
297
298 if( signals.size() == 1 )
299 title.Printf( _( "Fourier Analysis of %s" ), signals[0] );
300 else
301 title = _( "Fourier Analyses of Multiple Signals" );
302
303 WX_TEXT_ENTRY_DIALOG dlg( m_parent, _( "Fundamental frequency:" ), title, fundamental );
304
305 if( dlg.ShowModal() != wxID_OK )
306 return;
307
308 if( !dlg.GetValue().IsEmpty() )
309 fundamental = dlg.GetValue();
310
311 for( const wxString& signal : signals )
312 m_parent->DoFourier( signal, fundamental );
313 }
314 else if( event.GetId() == GRIDTRICKS_ID_COPY )
315 {
316 wxLogNull doNotLog; // disable logging of failed clipboard actions
317 wxString txt;
318
319 for( const wxString& signal : signals )
320 {
321 if( !txt.IsEmpty() )
322 txt += '\r';
323
324 txt += signal;
325 }
326
327 if( wxTheClipboard->Open() )
328 {
329 wxTheClipboard->SetData( new wxTextDataObject( txt ) );
330 wxTheClipboard->Flush(); // Allow data to be available after closing KiCad
331 wxTheClipboard->Close();
332 }
333 }
334 else if( event.GetId() == GRIDTRICKS_ID_SELECT )
335 {
337 }
338 else if( event.GetId() == GRIDTRICKS_ID_DELETE )
339 {
341 }
342}
343
344
346{
347public:
349 GRID_TRICKS( aGrid ),
350 m_parent( aParent ),
351 m_menuRow( 0 ),
352 m_menuCol( 0 )
353 {}
354
355protected:
356 void showPopupMenu( wxMenu& menu, wxGridEvent& aEvent ) override;
357 void doPopupSelection( wxCommandEvent& event ) override;
358
359protected:
363};
364
365
366void CURSORS_GRID_TRICKS::showPopupMenu( wxMenu& menu, wxGridEvent& aEvent )
367{
368 m_menuRow = aEvent.GetRow();
369 m_menuCol = aEvent.GetCol();
370
372 {
373 wxString msg = m_grid->GetColLabelValue( m_menuCol );
374
375 menu.Append( MYID_FORMAT_VALUE, wxString::Format( _( "Format %s..." ), msg ) );
376 menu.AppendSeparator();
377 }
378
379 GRID_TRICKS::showPopupMenu( menu, aEvent );
380}
381
382
383void CURSORS_GRID_TRICKS::doPopupSelection( wxCommandEvent& event )
384{
385 auto getSignalName =
386 [this]( int row ) -> wxString
387 {
388 wxString signal = m_grid->GetCellValue( row, COL_CURSOR_SIGNAL );
389
390 if( signal.EndsWith( "[2 - 1]" ) )
391 signal = signal.Left( signal.length() - 7 );
392
393 return signal;
394 };
395
396 if( event.GetId() == MYID_FORMAT_VALUE )
397 {
398 int axis = m_menuCol - COL_CURSOR_X;
400 DIALOG_SIM_FORMAT_VALUE formatDialog( m_parent, &format );
401
402 if( formatDialog.ShowModal() == wxID_OK )
403 {
404 for( int row = 0; row < m_grid->GetNumberRows(); ++row )
405 {
406 if( getSignalName( row ) == getSignalName( m_menuRow ) )
407 m_parent->SetCursorFormat( row, axis, format );
408 }
409 }
410 }
411 else
412 {
414 }
415}
416
417
419{
420public:
422 GRID_TRICKS( aGrid ),
423 m_parent( aParent ),
424 m_menuRow( 0 ),
425 m_menuCol( 0 )
426 {}
427
428protected:
429 void showPopupMenu( wxMenu& menu, wxGridEvent& aEvent ) override;
430 void doPopupSelection( wxCommandEvent& event ) override;
431
432protected:
436};
437
438
439void MEASUREMENTS_GRID_TRICKS::showPopupMenu( wxMenu& menu, wxGridEvent& aEvent )
440{
441 m_menuRow = aEvent.GetRow();
442 m_menuCol = aEvent.GetCol();
443
444 if( !( m_grid->IsInSelection( m_menuRow, m_menuCol ) ) )
445 m_grid->ClearSelection();
446
447 m_grid->SetGridCursor( m_menuRow, m_menuCol );
448
450 menu.Append( MYID_FORMAT_VALUE, _( "Format Value..." ) );
451
452 if( m_menuRow < ( m_grid->GetNumberRows() - 1 ) )
453 menu.Append( MYID_DELETE_MEASUREMENT, _( "Delete Measurement" ) );
454
455 menu.AppendSeparator();
456
457 GRID_TRICKS::showPopupMenu( menu, aEvent );
458}
459
460
462{
463 if( event.GetId() == MYID_FORMAT_VALUE )
464 {
466 DIALOG_SIM_FORMAT_VALUE formatDialog( m_parent, &format );
467
468 if( formatDialog.ShowModal() == wxID_OK )
469 {
473 }
474 }
475 else if( event.GetId() == MYID_DELETE_MEASUREMENT )
476 {
477 std::vector<int> measurements;
478
479 wxGridCellCoordsArray cells1 = m_grid->GetSelectionBlockTopLeft();
480 wxGridCellCoordsArray cells2 = m_grid->GetSelectionBlockBottomRight();
481
482 for( size_t i = 0; i < cells1.Count(); i++ )
483 {
484 if( cells1[i].GetCol() == COL_MEASUREMENT )
485 {
486 for( int j = cells1[i].GetRow(); j < cells2[i].GetRow() + 1; j++ )
487 measurements.push_back( j );
488 }
489 }
490
491 wxGridCellCoordsArray cells3 = m_grid->GetSelectedCells();
492
493 for( size_t i = 0; i < cells3.Count(); i++ )
494 {
495 if( cells3[i].GetCol() == COL_MEASUREMENT )
496 measurements.push_back( cells3[i].GetRow() );
497 }
498
499 if( measurements.size() < 1 )
500 measurements.push_back( m_menuRow );
501
502 // When deleting a row, we'll change the indexes.
503 // To avoid problems, we can start with the highest indexes.
504 sort( measurements.begin(), measurements.end(), std::greater<>() );
505
506 for( int row : measurements )
508
509 m_grid->ClearSelection();
510
512 }
513 else
514 {
516 }
517}
518
519
521{
522public:
524 m_frame( aFrame )
525 {
527 }
528
530 {
532 }
533
534private:
536};
537
538
539#define ID_SIM_REFRESH 10207
540#define REFRESH_INTERVAL 50 // 20 frames/second.
541
542
544 SCH_EDIT_FRAME* aSchematicFrame ) :
545 SIMULATOR_FRAME_UI_BASE( aSimulatorFrame ),
546 m_SuppressGridEvents( 0 ),
547 m_simulatorFrame( aSimulatorFrame ),
548 m_schematicFrame( aSchematicFrame ),
549 m_darkMode( true ),
550 m_plotNumber( 0 ),
551 m_refreshTimer( this, ID_SIM_REFRESH )
552{
553 // Get the previous size and position of windows:
555
556 m_filter->SetHint( _( "Filter" ) );
557
558 m_signalsGrid->wxGrid::SetLabelFont( KIUI::GetStatusFont( this ) );
559 m_cursorsGrid->wxGrid::SetLabelFont( KIUI::GetStatusFont( this ) );
560 m_measurementsGrid->wxGrid::SetLabelFont( KIUI::GetStatusFont( this ) );
561
562 m_signalsGrid->PushEventHandler( new SIGNALS_GRID_TRICKS( this, m_signalsGrid ) );
563 m_cursorsGrid->PushEventHandler( new CURSORS_GRID_TRICKS( this, m_cursorsGrid ) );
564 m_measurementsGrid->PushEventHandler( new MEASUREMENTS_GRID_TRICKS( this, m_measurementsGrid ) );
565
566 wxGridCellAttr* attr = new wxGridCellAttr;
567 attr->SetReadOnly();
568 m_signalsGrid->SetColAttr( COL_SIGNAL_NAME, attr );
569
570 attr = new wxGridCellAttr;
571 attr->SetReadOnly();
572 m_cursorsGrid->SetColAttr( COL_CURSOR_NAME, attr );
573
574 attr = new wxGridCellAttr;
575 attr->SetReadOnly();
576 m_cursorsGrid->SetColAttr( COL_CURSOR_SIGNAL, attr );
577
578 attr = new wxGridCellAttr;
579 attr->SetReadOnly();
580 m_cursorsGrid->SetColAttr( COL_CURSOR_Y, attr );
581
583
584 attr = new wxGridCellAttr;
585 attr->SetReadOnly();
586 m_measurementsGrid->SetColAttr( COL_MEASUREMENT_VALUE, attr );
587
588 // Prepare the color list to plot traces
590
591 Bind( EVT_SIM_CURSOR_UPDATE, &SIMULATOR_FRAME_UI::onPlotCursorUpdate, this );
592
593 Bind( wxEVT_TIMER,
594 [&]( wxTimerEvent& aEvent )
595 {
596 OnSimRefresh( false );
597
598 if( m_simulatorFrame->GetSimulator()->IsRunning() )
599 m_refreshTimer.Start( REFRESH_INTERVAL, wxTIMER_ONE_SHOT );
600 },
601 m_refreshTimer.GetId() );
602
603#ifndef wxHAS_NATIVE_TABART
604 // Default non-native tab art has ugly gradients we don't want
605 m_plotNotebook->SetArtProvider( new wxAuiSimpleTabArt() );
606#endif
607}
608
609
611{
612 // Delete the GRID_TRICKS.
613 m_signalsGrid->PopEventHandler( true );
614 m_cursorsGrid->PopEventHandler( true );
615 m_measurementsGrid->PopEventHandler( true );
616}
617
618
620{
621 for( auto& m_cursorFormat : m_cursorFormats )
622 {
623 m_cursorFormat[0] = { 3, wxS( "~s" ) };
624 m_cursorFormat[1] = { 3, wxS( "~V" ) };
625 }
626
627 // proper init and transfer/copy m_cursorFormats
628 // we work on m_cursorFormatsDyn from now on.
629 // TODO: rework +- LOC when m_cursorFormatsDyn and m_cursorFormats get merged.
630 m_cursorFormatsDyn.clear();
631 m_cursorFormatsDyn.resize( std::size( m_cursorFormats ) );
632
633 for( size_t index = 0; index < std::size( m_cursorFormats ); index++ )
634 {
635 for( size_t index2 = 0; index2 < std::size( m_cursorFormats[0] ); index2++ )
636 {
637 m_cursorFormatsDyn[index].push_back( m_cursorFormats[index][index2] );
638 }
639 }
640
641 // Dump string helper, tries to get the current higher cursor name to form the next one.
642 // Based on the column labeling
643 // TODO: "Cursor n" may translate as "n Cursor" in other languages
644 // TBD how to handle; just forbid for now.
645 int nameMax = 0;
646
647 for( int i = 0; i < m_signalsGrid->GetNumberCols(); i++ )
648 {
649 wxString maxCursor = m_signalsGrid->GetColLabelValue( i );
650
651 maxCursor.Replace( _( "Cursor " ), "" );
652
653 int tmpMax = wxAtoi( maxCursor );
654
655 if( nameMax < tmpMax )
656 nameMax = tmpMax;
657 }
658
659 m_customCursorsCnt = nameMax + 1; // Init with a +1 on top of current cursor 2, defaults to 3
660}
661
662
664{
665 std::vector<SPICE_VALUE_FORMAT> tmp;
666 // m_cursorFormatsDyn should be equal with m_cursorFormats on first entry here.
667 m_cursorFormatsDyn.emplace_back( tmp );
668
669 m_cursorFormatsDyn[m_customCursorsCnt].push_back( { 3, wxS( "~s" ) } );
670 m_cursorFormatsDyn[m_customCursorsCnt].push_back( { 3, wxS( "~V" ) } );
671
672 wxString cursor_name = wxString( _( "Cursor " ) ) << m_customCursorsCnt;
673
674 m_signalsGrid->InsertCols( m_signalsGrid->GetNumberCols() , 1, true );
675 m_signalsGrid->SetColLabelValue( m_signalsGrid->GetNumberCols() - 1, cursor_name );
676
677 wxGridCellAttr* attr = new wxGridCellAttr;
678 m_signalsGrid->SetColAttr( COL_CURSOR_2 + m_customCursorsCnt, attr );
679
681
684 OnModify();
685}
686
687
689{
690 int col = m_signalsGrid->GetNumberCols();
691 int rows = m_signalsGrid->GetNumberRows();
692
693 if( col > COL_CURSOR_2 )
694 {
695 // Now we need to find the active cursor and deactivate before removing the column,
696 // Send the dummy event to update the UI
697 for( int i = 0; i < rows; i++ )
698 {
699 if( m_signalsGrid->GetCellValue( i, col - 1 ) == wxS( "1" ) )
700 {
701 m_signalsGrid->SetCellValue( i, col - 1, wxEmptyString );
702 wxGridEvent aDummy( wxID_ANY, wxEVT_GRID_CELL_CHANGED, m_signalsGrid, i, col - 1 );
703 onSignalsGridCellChanged( aDummy );
704 break;
705 }
706
707 }
708
709 m_signalsGrid->DeleteCols( col - 1, 1, false );
710 m_cursorFormatsDyn.pop_back();
712 m_plotNotebook->Refresh();
715 OnModify();
716 }
717}
718
719
721{
722 for( int ii = 0; ii < static_cast<int>( m_plotNotebook->GetPageCount() ); ++ii )
723 {
724 if( SIM_TAB* simTab = dynamic_cast<SIM_TAB*>( m_plotNotebook->GetPage( ii ) ) )
725 {
726 simTab->OnLanguageChanged();
727
728 wxString pageTitle( simulator()->TypeToName( simTab->GetSimType(), true ) );
729 pageTitle.Prepend( wxString::Format( _( "Analysis %u - " ), ii+1 /* 1-based */ ) );
730
731 m_plotNotebook->SetPageText( ii, pageTitle );
732 }
733 }
734
735 m_filter->SetHint( _( "Filter" ) );
736
737 m_signalsGrid->SetColLabelValue( COL_SIGNAL_NAME, _( "Signal" ) );
738 m_signalsGrid->SetColLabelValue( COL_SIGNAL_SHOW, _( "Plot" ) );
739 m_signalsGrid->SetColLabelValue( COL_SIGNAL_COLOR, _( "Color" ) );
740 m_signalsGrid->SetColLabelValue( COL_CURSOR_1, _( "Cursor 1" ) );
741 m_signalsGrid->SetColLabelValue( COL_CURSOR_2, _( "Cursor 2" ) );
742
743 m_cursorsGrid->SetColLabelValue( COL_CURSOR_NAME, _( "Cursor" ) );
744 m_cursorsGrid->SetColLabelValue( COL_CURSOR_SIGNAL, _( "Signal" ) );
745 m_cursorsGrid->SetColLabelValue( COL_CURSOR_X, _( "Time" ) );
746 m_cursorsGrid->SetColLabelValue( COL_CURSOR_Y, _( "Value" ) );
748
749 for( TUNER_SLIDER* tuner : m_tuners )
750 tuner->ShowChangedLanguage();
751}
752
753
755{
756 const EESCHEMA_SETTINGS::SIMULATOR& settings = aCfg->m_Simulator;
757
758 // Read subwindows sizes (should be > 0 )
764 m_darkMode = !settings.view.white_background;
765
766 m_preferences = settings.preferences;
767}
768
769
771{
773
774 settings.view.plot_panel_width = m_splitterLeftRight->GetSashPosition();
775 settings.view.plot_panel_height = m_splitterPlotAndConsole->GetSashPosition();
776 settings.view.signal_panel_height = m_splitterSignals->GetSashPosition();
777 settings.view.cursors_panel_height = m_splitterCursors->GetSashPosition();
778 settings.view.measurements_panel_height = m_splitterMeasurements->GetSashPosition();
779 settings.view.white_background = !m_darkMode;
780}
781
782
784{
785 m_preferences = aPrefs;
786
787 const std::size_t pageCount = m_plotNotebook->GetPageCount();
788 for( std::size_t i = 0; i < pageCount; ++i )
789 {
790 wxWindow* page = m_plotNotebook->GetPage( i );
791 auto simTab = dynamic_cast<SIM_TAB*>( page );
792 wxASSERT( simTab != nullptr );
793 simTab->ApplyPreferences( aPrefs );
794 }
795}
796
797
799{
800 if( !simulator()->Settings()->GetWorkbookFilename().IsEmpty() )
801 {
802 wxFileName filename = simulator()->Settings()->GetWorkbookFilename();
803 filename.SetPath( m_schematicFrame->Prj().GetProjectPath() );
804
805 if( !LoadWorkbook( filename.GetFullPath() ) )
806 simulator()->Settings()->SetWorkbookFilename( "" );
807 }
808 else if( m_simulatorFrame->LoadSimulator( wxEmptyString, 0 ) )
809 {
810 wxString schTextSimCommand = circuitModel()->GetSchTextSimCommand();
811
812 if( !schTextSimCommand.IsEmpty() )
813 {
814 SIM_TAB* simTab = NewSimTab( schTextSimCommand );
816 }
817
819 rebuildSignalsGrid( m_filter->GetValue() );
820 }
821}
822
823
825{
828
831
834
837
840}
841
842
843void sortSignals( std::vector<wxString>& signals )
844{
845 std::sort( signals.begin(), signals.end(),
846 []( const wxString& lhs, const wxString& rhs )
847 {
848 // Sort voltages first
849 if( lhs.Upper().StartsWith( 'V' ) && !rhs.Upper().StartsWith( 'V' ) )
850 return true;
851 else if( !lhs.Upper().StartsWith( 'V' ) && rhs.Upper().StartsWith( 'V' ) )
852 return false;
853
854 return StrNumCmp( lhs, rhs, true /* ignore case */ ) < 0;
855 } );
856}
857
858
860{
861 SUPPRESS_GRID_CELL_EVENTS raii( this );
862
864
865 SIM_PLOT_TAB* plotPanel = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() );
866
867 if( !plotPanel )
868 return;
869
870 SIM_TYPE simType = plotPanel->GetSimType();
871 std::vector<wxString> signals;
872
873 if( plotPanel->GetSimType() == ST_FFT )
874 {
875 wxStringTokenizer tokenizer( plotPanel->GetSimCommand(), wxT( " \t\r\n" ), wxTOKEN_STRTOK );
876
877 while( tokenizer.HasMoreTokens() && tokenizer.GetNextToken().Lower() != wxT( "fft" ) )
878 {};
879
880 while( tokenizer.HasMoreTokens() )
881 signals.emplace_back( tokenizer.GetNextToken() );
882 }
883 else
884 {
885 // NB: m_signals are already broken out into gain/phase, but m_userDefinedSignals are
886 // as the user typed them
887
888 for( const wxString& signal : m_signals )
889 signals.push_back( signal );
890
891 for( const auto& [ id, signal ] : m_userDefinedSignals )
892 {
893 if( simType == ST_AC )
894 {
895 signals.push_back( signal + _( " (gain)" ) );
896 signals.push_back( signal + _( " (phase)" ) );
897 }
898 else if( simType == ST_SP )
899 {
900 signals.push_back( signal + _( " (amplitude)" ) );
901 signals.push_back( signal + _( " (phase)" ) );
902 }
903 else
904 {
905 signals.push_back( signal );
906 }
907 }
908
909 sortSignals( signals );
910 }
911
912 if( aFilter.IsEmpty() )
913 aFilter = wxS( "*" );
914
915 EDA_COMBINED_MATCHER matcher( aFilter.Upper(), CTX_SIGNAL );
916 int row = 0;
917
918 for( const wxString& signal : signals )
919 {
920 if( matcher.Find( signal.Upper() ) )
921 {
922 int traceType = SPT_UNKNOWN;
923 wxString vectorName = vectorNameFromSignalName( plotPanel, signal, &traceType );
924 TRACE* trace = plotPanel->GetTrace( vectorName, traceType );
925
926 m_signalsGrid->AppendRows( 1 );
927 m_signalsGrid->SetCellValue( row, COL_SIGNAL_NAME, signal );
928
929 wxGridCellAttr* attr = new wxGridCellAttr;
930 attr->SetRenderer( new wxGridCellBoolRenderer() );
931 attr->SetReadOnly(); // not really; we delegate interactivity to GRID_TRICKS
932 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
933 m_signalsGrid->SetAttr( row, COL_SIGNAL_SHOW, attr );
934
935 if( !trace )
936 {
937 attr = new wxGridCellAttr;
938 attr->SetReadOnly();
939 m_signalsGrid->SetAttr( row, COL_SIGNAL_COLOR, attr );
940 m_signalsGrid->SetCellValue( row, COL_SIGNAL_COLOR, wxEmptyString );
941
942 attr = new wxGridCellAttr;
943 attr->SetReadOnly();
944 m_signalsGrid->SetAttr( row, COL_CURSOR_1, attr );
945
946 attr = new wxGridCellAttr;
947 attr->SetReadOnly();
948 m_signalsGrid->SetAttr( row, COL_CURSOR_2, attr );
949
950 if( m_customCursorsCnt > 3 )
951 {
952 for( int i = 1; i <= m_customCursorsCnt - 3; i++ )
953 {
954 attr = new wxGridCellAttr;
955 attr->SetReadOnly();
956 m_signalsGrid->SetAttr( row, COL_CURSOR_2 + i, attr );
957 }
958 }
959 }
960 else
961 {
962 m_signalsGrid->SetCellValue( row, COL_SIGNAL_SHOW, wxS( "1" ) );
963
964 attr = new wxGridCellAttr;
965 attr->SetRenderer( new GRID_CELL_COLOR_RENDERER( this ) );
966 attr->SetEditor( new GRID_CELL_COLOR_SELECTOR( this, m_signalsGrid ) );
967 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
968 m_signalsGrid->SetAttr( row, COL_SIGNAL_COLOR, attr );
969 KIGFX::COLOR4D color( trace->GetPen().GetColour() );
970 m_signalsGrid->SetCellValue( row, COL_SIGNAL_COLOR, color.ToCSSString() );
971
972 attr = new wxGridCellAttr;
973 attr->SetRenderer( new wxGridCellBoolRenderer() );
974 attr->SetReadOnly(); // not really; we delegate interactivity to GRID_TRICKS
975 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
976 m_signalsGrid->SetAttr( row, COL_CURSOR_1, attr );
977 m_signalsGrid->SetCellValue( row, COL_CURSOR_1, trace->GetCursor( 1 ) ? "1" : "0" );
978
979 attr = new wxGridCellAttr;
980 attr->SetRenderer( new wxGridCellBoolRenderer() );
981 attr->SetReadOnly(); // not really; we delegate interactivity to GRID_TRICKS
982 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
983 m_signalsGrid->SetAttr( row, COL_CURSOR_2, attr );
984 m_signalsGrid->SetCellValue( row, COL_CURSOR_2, trace->GetCursor( 2 ) ? "1" : "0" );
985
986 if( m_customCursorsCnt > 3 )
987 {
988 for( int i = 1; i <= m_customCursorsCnt - 3; i++ )
989 {
990 attr = new wxGridCellAttr;
991 attr->SetRenderer( new wxGridCellBoolRenderer() );
992 attr->SetReadOnly(); // not really; we delegate interactivity to GRID_TRICKS
993 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
994 m_signalsGrid->SetAttr( row, COL_CURSOR_2 + i, attr );
995 m_signalsGrid->SetCellValue( row, COL_CURSOR_2 + i,
996 trace->GetCursor( i ) ? "1" : "0" );
997 }
998 }
999 }
1000 row++;
1001 }
1002 }
1003}
1004
1005
1007{
1008 m_signals.clear();
1009
1010 int options = m_simulatorFrame->GetCurrentOptions();
1012 wxString unconnected = wxString( wxS( "unconnected-(" ) );
1013
1014 if( simType == ST_UNKNOWN )
1015 simType = ST_TRAN;
1016
1017 unconnected.Replace( '(', '_' ); // Convert to SPICE markup
1018
1019 auto addSignal =
1020 [&]( const wxString& aSignalName )
1021 {
1022 if( simType == ST_AC )
1023 {
1024 m_signals.push_back( aSignalName + _( " (gain)" ) );
1025 m_signals.push_back( aSignalName + _( " (phase)" ) );
1026 }
1027 else if( simType == ST_SP )
1028 {
1029 m_signals.push_back( aSignalName + _( " (amplitude)" ) );
1030 m_signals.push_back( aSignalName + _( " (phase)" ) );
1031 }
1032 else
1033 {
1034 m_signals.push_back( aSignalName );
1035 }
1036 };
1037
1039 && ( simType == ST_TRAN || simType == ST_DC || simType == ST_AC || simType == ST_FFT) )
1040 {
1041 for( const wxString& net : circuitModel()->GetNets() )
1042 {
1043 // netnames are escaped (can contain "{slash}" for '/') Unscape them:
1044 wxString netname = UnescapeString( net );
1046
1047 if( netname == "GND" || netname == "0" || netname.StartsWith( unconnected ) )
1048 continue;
1049
1050 m_netnames.emplace_back( netname );
1051 addSignal( wxString::Format( wxS( "V(%s)" ), netname ) );
1052 }
1053 }
1054
1056 && ( simType == ST_TRAN || simType == ST_DC || simType == ST_AC ) )
1057 {
1058 for( const SPICE_ITEM& item : circuitModel()->GetItems() )
1059 {
1060 // Add all possible currents for the device.
1061 for( const std::string& name : item.model->SpiceGenerator().CurrentNames( item ) )
1062 addSignal( name );
1063 }
1064 }
1065
1067 && ( simType == ST_TRAN || simType == ST_DC ) )
1068 {
1069 for( const SPICE_ITEM& item : circuitModel()->GetItems() )
1070 {
1071 if( item.model->GetPinCount() >= 2 )
1072 {
1073 wxString name = item.model->SpiceGenerator().ItemName( item );
1074 addSignal( wxString::Format( wxS( "P(%s)" ), name ) );
1075 }
1076 }
1077 }
1078
1079 if( simType == ST_NOISE )
1080 {
1081 addSignal( wxS( "inoise_spectrum" ) );
1082 addSignal( wxS( "onoise_spectrum" ) );
1083 }
1084
1085 if( simType == ST_SP )
1086 {
1087 std::vector<std::string> portnums;
1088
1089 for( const SPICE_ITEM& item : circuitModel()->GetItems() )
1090 {
1091 wxString name = item.model->SpiceGenerator().ItemName( item );
1092
1093 // We are only looking for voltage sources in .SP mode
1094 if( !name.StartsWith( "V" ) )
1095 continue;
1096
1097 std::string portnum = "";
1098
1099 if( const SIM_MODEL::PARAM* portnum_param = item.model->FindParam( "portnum" ) )
1100 portnum = SIM_VALUE::ToSpice( portnum_param->value );
1101
1102 if( portnum != "" )
1103 portnums.push_back( portnum );
1104 }
1105
1106 for( const std::string& portnum1 : portnums )
1107 {
1108 for( const std::string& portnum2 : portnums )
1109 {
1110 addSignal( wxString::Format( wxS( "S_%s_%s" ), portnum1, portnum2 ) );
1111 }
1112 }
1113 }
1114
1115 // Add .SAVE and .PROBE directives
1116 for( const wxString& directive : circuitModel()->GetDirectives() )
1117 {
1118 wxStringTokenizer directivesTokenizer( directive, wxT( "\r\n" ), wxTOKEN_STRTOK );
1119
1120 while( directivesTokenizer.HasMoreTokens() )
1121 {
1122 wxString line = directivesTokenizer.GetNextToken().Upper();
1123 wxString directiveParams;
1124
1125 if( line.StartsWith( wxS( ".SAVE" ), &directiveParams )
1126 || line.StartsWith( wxS( ".PROBE" ), &directiveParams ) )
1127 {
1128 wxStringTokenizer paramsTokenizer( directiveParams, wxT( " \t" ), wxTOKEN_STRTOK );
1129
1130 while( paramsTokenizer.HasMoreTokens() )
1131 addSignal( paramsTokenizer.GetNextToken() );
1132 }
1133 }
1134 }
1135}
1136
1137
1138SIM_TAB* SIMULATOR_FRAME_UI::NewSimTab( const wxString& aSimCommand )
1139{
1140 SIM_TAB* simTab = nullptr;
1141 SIM_TYPE simType = SPICE_CIRCUIT_MODEL::CommandToSimType( aSimCommand );
1142
1143 if( SIM_TAB::IsPlottable( simType ) )
1144 {
1145 SIM_PLOT_TAB* panel = new SIM_PLOT_TAB( aSimCommand, m_plotNotebook );
1146 simTab = panel;
1148 }
1149 else
1150 {
1151 simTab = new SIM_NOPLOT_TAB( aSimCommand, m_plotNotebook );
1152 }
1153
1154 wxString pageTitle( simulator()->TypeToName( simType, true ) );
1155 pageTitle.Prepend( wxString::Format( _( "Analysis %u - " ), static_cast<unsigned int>( ++m_plotNumber ) ) );
1156
1157 m_plotNotebook->AddPage( simTab, pageTitle, true );
1158
1159 return simTab;
1160}
1161
1162
1163void SIMULATOR_FRAME_UI::OnFilterText( wxCommandEvent& aEvent )
1164{
1165 rebuildSignalsGrid( m_filter->GetValue() );
1166}
1167
1168
1169void SIMULATOR_FRAME_UI::OnFilterMouseMoved( wxMouseEvent& aEvent )
1170{
1171 wxPoint pos = aEvent.GetPosition();
1172 wxRect ctrlRect = m_filter->GetScreenRect();
1173 int buttonWidth = ctrlRect.GetHeight(); // Presume buttons are square
1174
1175 if( m_filter->IsSearchButtonVisible() && pos.x < buttonWidth )
1176 SetCursor( wxCURSOR_ARROW );
1177 else if( m_filter->IsCancelButtonVisible() && pos.x > ctrlRect.GetWidth() - buttonWidth )
1178 SetCursor( wxCURSOR_ARROW );
1179 else
1180 SetCursor( wxCURSOR_IBEAM );
1181}
1182
1183
1184wxString vectorNameFromSignalId( int aUserDefinedSignalId )
1185{
1186 return wxString::Format( wxS( "user%d" ), aUserDefinedSignalId );
1187}
1188
1189
1195 const wxString& aSignalName,
1196 int* aTraceType )
1197{
1198 std::map<wxString, int> suffixes;
1199 suffixes[ _( " (amplitude)" ) ] = SPT_SP_AMP;
1200 suffixes[ _( " (gain)" ) ] = SPT_AC_GAIN;
1201 suffixes[ _( " (phase)" ) ] = SPT_AC_PHASE;
1202
1203 if( aTraceType )
1204 {
1205 if( aPlotTab && aPlotTab->GetSimType() == ST_NOISE )
1206 {
1207 if( getNoiseSource().Upper().StartsWith( 'I' ) )
1208 *aTraceType = SPT_CURRENT;
1209 else
1210 *aTraceType = SPT_VOLTAGE;
1211 }
1212 else
1213 {
1214 wxUniChar firstChar = aSignalName.Upper()[0];
1215
1216 if( firstChar == 'V' )
1217 *aTraceType = SPT_VOLTAGE;
1218 else if( firstChar == 'I' )
1219 *aTraceType = SPT_CURRENT;
1220 else if( firstChar == 'P' )
1221 *aTraceType = SPT_POWER;
1222 }
1223 }
1224
1225 wxString suffix;
1226 wxString name = aSignalName;
1227
1228 for( const auto& [ candidate, type ] : suffixes )
1229 {
1230 if( name.EndsWith( candidate ) )
1231 {
1232 name = name.Left( name.Length() - candidate.Length() );
1233
1234 if( aTraceType )
1235 *aTraceType |= type;
1236
1237 break;
1238 }
1239 }
1240
1241 for( const auto& [ id, signal ] : m_userDefinedSignals )
1242 {
1243 if( name == signal )
1244 return vectorNameFromSignalId( id );
1245 }
1246
1247 return name;
1248};
1249
1250
1252{
1253 if( m_SuppressGridEvents > 0 )
1254 return;
1255
1256 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() );
1257
1258 if( !plotTab )
1259 return;
1260
1261 int row = aEvent.GetRow();
1262 int col = aEvent.GetCol();
1263 wxString text = m_signalsGrid->GetCellValue( row, col );
1264 wxString signalName = m_signalsGrid->GetCellValue( row, COL_SIGNAL_NAME );
1265 int traceType = SPT_UNKNOWN;
1266 wxString vectorName = vectorNameFromSignalName( plotTab, signalName, &traceType );
1267
1268 if( col == COL_SIGNAL_SHOW )
1269 {
1270 if( text == wxS( "1" ) )
1271 updateTrace( vectorName, traceType, plotTab );
1272 else
1273 plotTab->DeleteTrace( vectorName, traceType );
1274
1275 plotTab->GetPlotWin()->UpdateAll();
1276
1277 // Update enabled/visible states of other controls
1280 OnModify();
1281 }
1282 else if( col == COL_SIGNAL_COLOR )
1283 {
1284 KIGFX::COLOR4D color( m_signalsGrid->GetCellValue( row, COL_SIGNAL_COLOR ) );
1285 TRACE* trace = plotTab->GetTrace( vectorName, traceType );
1286
1287 if( trace )
1288 {
1289 trace->SetTraceColour( color.ToColour() );
1290 plotTab->UpdateTraceStyle( trace );
1291 plotTab->UpdatePlotColors();
1292 OnModify();
1293 }
1294 }
1295 else if( col == COL_CURSOR_1 || col == COL_CURSOR_2
1296 || ( ( std::size( m_cursorFormatsDyn ) > std::size( m_cursorFormats ) )
1297 && col > COL_CURSOR_2 ) )
1298 {
1299 int id = col == COL_CURSOR_1 ? 1 : 2;
1300
1301 if( col > COL_CURSOR_2 ) // TODO: clean up logic
1302 {
1303 id = col - 2; // enum SIGNALS_GRID_COLUMNS offset for Cursor n
1304 }
1305
1306 TRACE* activeTrace = nullptr;
1307
1308 if( text == wxS( "1" ) )
1309 {
1310 signalName = m_signalsGrid->GetCellValue( row, COL_SIGNAL_NAME );
1311 vectorName = vectorNameFromSignalName( plotTab, signalName, &traceType );
1312 activeTrace = plotTab->GetTrace( vectorName, traceType );
1313
1314 if( activeTrace )
1315 plotTab->EnableCursor( activeTrace, id, signalName );
1316
1317 OnModify();
1318 }
1319
1320 // Turn off cursor on other signals.
1321 for( const auto& [name, trace] : plotTab->GetTraces() )
1322 {
1323 if( trace != activeTrace && trace->HasCursor( id ) )
1324 {
1325 plotTab->DisableCursor( trace, id );
1326 OnModify();
1327 }
1328 }
1329
1330 // Update cursor checkboxes (which are really radio buttons)
1332 }
1333}
1334
1335
1337{
1338 if( m_SuppressGridEvents > 0 )
1339 return;
1340
1341 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() );
1342
1343 if( !plotTab )
1344 return;
1345
1346 int row = aEvent.GetRow();
1347 int col = aEvent.GetCol();
1348 wxString text = m_cursorsGrid->GetCellValue( row, col );
1349 wxString cursorName = m_cursorsGrid->GetCellValue( row, COL_CURSOR_NAME );
1350
1351 double value = SPICE_VALUE( text ).ToDouble();
1352
1353 if( col == COL_CURSOR_X )
1354 {
1355 CURSOR* cursor1 = nullptr;
1356 CURSOR* cursor2 = nullptr;
1357
1358 std::vector<CURSOR*> cursorsVec;
1359 cursorsVec.clear();
1360
1361 for( const auto& [name, trace] : plotTab->GetTraces() )
1362 {
1363 if( CURSOR* cursor = trace->GetCursor( 1 ) )
1364 cursor1 = cursor;
1365
1366 if( CURSOR* cursor = trace->GetCursor( 2 ) )
1367 cursor2 = cursor;
1368
1369 int tmp = 3;
1370
1371 if( !cursor1 )
1372 tmp--;
1373 if( !cursor2 )
1374 tmp--;
1375
1376 for( int i = tmp; i < m_customCursorsCnt; i++ )
1377 {
1378 if( CURSOR* cursor = trace->GetCursor( i ) )
1379 {
1380 cursorsVec.emplace_back( cursor );
1381
1382 if( cursorName == ( wxString( "" ) << i ) && cursor )
1383 cursor->SetCoordX( value );
1384 }
1385 }
1386 }
1387
1388 //double value = SPICE_VALUE( text ).ToDouble();
1389
1390 if( cursorName == wxS( "1" ) && cursor1 )
1391 cursor1->SetCoordX( value );
1392 else if( cursorName == wxS( "2" ) && cursor2 )
1393 cursor2->SetCoordX( value );
1394 else if( cursorName == _( "Diff" ) && cursor1 && cursor2 )
1395 cursor2->SetCoordX( cursor1->GetCoords().x + value );
1396
1398 OnModify();
1399 }
1400 else
1401 {
1402 wxFAIL_MSG( wxT( "All other columns are supposed to be read-only!" ) );
1403 }
1404}
1405
1406
1408{
1409 SPICE_VALUE_FORMAT result;
1410 result.FromString( m_measurementsGrid->GetCellValue( aRow, COL_MEASUREMENT_FORMAT ) );
1411 return result;
1412}
1413
1414
1416{
1417 m_measurementsGrid->SetCellValue( aRow, COL_MEASUREMENT_FORMAT, aFormat.ToString() );
1418}
1419
1420
1422{
1423 if( aRow < ( m_measurementsGrid->GetNumberRows() - 1 ) )
1424 m_measurementsGrid->DeleteRows( aRow, 1 );
1425}
1426
1427
1429{
1430 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() );
1431
1432 if( !plotTab )
1433 return;
1434
1435 int row = aEvent.GetRow();
1436 int col = aEvent.GetCol();
1437 wxString text = m_measurementsGrid->GetCellValue( row, col );
1438
1439 if( col == COL_MEASUREMENT )
1440 {
1441 UpdateMeasurement( row );
1443 OnModify();
1444 }
1445 else
1446 {
1447 wxFAIL_MSG( wxT( "All other columns are supposed to be read-only!" ) );
1448 }
1449
1450 // Always leave a single empty row for type-in
1451
1452 int rowCount = static_cast<int>( m_measurementsGrid->GetNumberRows() );
1453 int emptyRows = 0;
1454
1455 for( row = rowCount - 1; row >= 0; row-- )
1456 {
1457 if( m_measurementsGrid->GetCellValue( row, COL_MEASUREMENT ).IsEmpty() )
1458 emptyRows++;
1459 else
1460 break;
1461 }
1462
1463 if( emptyRows > 1 )
1464 {
1465 int killRows = emptyRows - 1;
1466 m_measurementsGrid->DeleteRows( rowCount - killRows, killRows );
1467 }
1468 else if( emptyRows == 0 )
1469 {
1470 m_measurementsGrid->AppendRows( 1 );
1471 }
1472}
1473
1474
1475void SIMULATOR_FRAME_UI::OnUpdateUI( wxUpdateUIEvent& event )
1476{
1477 if( SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() ) )
1478 {
1479 if( plotTab->GetLegendPosition() != plotTab->m_LastLegendPosition )
1480 {
1481 plotTab->m_LastLegendPosition = plotTab->GetLegendPosition();
1482 OnModify();
1483 }
1484 }
1485}
1486
1487
1503{
1504 static wxRegEx measureParamsRegEx( wxT( "^"
1505 " *"
1506 "([a-zA-Z_]+)"
1507 " +"
1508 "([a-zA-Z]*)\\(([^\\)]+)\\)" ) );
1509
1510 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() );
1511
1512 if( !plotTab )
1513 return;
1514
1515 wxString text = m_measurementsGrid->GetCellValue( aRow, COL_MEASUREMENT );
1516
1517 if( text.IsEmpty() )
1518 {
1519 m_measurementsGrid->SetCellValue( aRow, COL_MEASUREMENT_VALUE, wxEmptyString );
1520 return;
1521 }
1522
1523 wxString simType = simulator()->TypeToName( plotTab->GetSimType(), true );
1524 wxString resultName = wxString::Format( wxS( "meas_result_%u" ), aRow );
1525 wxString result = wxS( "?" );
1526
1527 if( measureParamsRegEx.Matches( text ) )
1528 {
1529 wxString func = measureParamsRegEx.GetMatch( text, 1 ).Upper();
1530 wxString signalType = measureParamsRegEx.GetMatch( text, 2 ).Upper();
1531 wxString deviceName = measureParamsRegEx.GetMatch( text, 3 );
1532 wxString units;
1534
1535 if( signalType.EndsWith( wxS( "DB" ) ) )
1536 {
1537 units = wxS( "dB" );
1538 }
1539 else if( signalType.StartsWith( 'I' ) )
1540 {
1541 units = wxS( "A" );
1542 }
1543 else if( signalType.StartsWith( 'P' ) )
1544 {
1545 units = wxS( "W" );
1546 // Our syntax is different from ngspice for power signals
1547 text = func + " " + deviceName + ":power";
1548 }
1549 else
1550 {
1551 units = wxS( "V" );
1552 }
1553
1554 if( func.EndsWith( wxS( "_AT" ) ) )
1555 {
1556 if( plotTab->GetSimType() == ST_AC || plotTab->GetSimType() == ST_SP )
1557 units = wxS( "Hz" );
1558 else
1559 units = wxS( "s" );
1560 }
1561 else if( func.StartsWith( wxS( "INTEG" ) ) )
1562 {
1563 switch( plotTab->GetSimType() )
1564 {
1565 case ST_TRAN:
1566 if ( signalType.StartsWith( 'P' ) )
1567 units = wxS( "J" );
1568 else
1569 units += wxS( ".s" );
1570
1571 break;
1572
1573 case ST_AC:
1574 case ST_SP:
1575 case ST_DISTO:
1576 case ST_NOISE:
1577 case ST_FFT:
1578 case ST_SENS: // If there is a vector, it is frequency
1579 units += wxS( "·Hz" );
1580 break;
1581
1582 case ST_DC: // Could be a lot of things : V, A, deg C, ohm, ...
1583 case ST_OP: // There is no vector for integration
1584 case ST_PZ: // There is no vector for integration
1585 case ST_TF: // There is no vector for integration
1586 default:
1587 units += wxS( "·?" );
1588 break;
1589 }
1590 }
1591
1592 fmt.UpdateUnits( units );
1593 SetMeasureFormat( aRow, fmt );
1594
1596 }
1597
1599 {
1600 wxString cmd = wxString::Format( wxS( "meas %s %s %s" ), simType, resultName, text );
1601 simulator()->Command( "echo " + cmd.ToStdString() );
1602 simulator()->Command( cmd.ToStdString() );
1603
1604 std::vector<double> resultVec = simulator()->GetGainVector( resultName.ToStdString() );
1605
1606 if( resultVec.size() > 0 )
1607 result = SPICE_VALUE( resultVec[0] ).ToString( GetMeasureFormat( aRow ) );
1608 }
1609
1610 m_measurementsGrid->SetCellValue( aRow, COL_MEASUREMENT_VALUE, result );
1611}
1612
1613
1614void SIMULATOR_FRAME_UI::AddTuner( const SCH_SHEET_PATH& aSheetPath, SCH_SYMBOL* aSymbol )
1615{
1616 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() );
1617
1618 if( !plotTab )
1619 {
1620 DisplayErrorMessage( nullptr, _( "The current analysis must have a plot in order to tune "
1621 "the value of a passive R, L, C model or voltage or "
1622 "current source." ) );
1623 return;
1624 }
1625
1626 wxString ref = aSymbol->GetRef( &aSheetPath );
1627
1628 // Do not add multiple instances for the same component.
1629 for( TUNER_SLIDER* tuner : m_tuners )
1630 {
1631 if( tuner->GetSymbolRef() == ref )
1632 return;
1633 }
1634
1635 if( [[maybe_unused]] const SPICE_ITEM* item = GetExporter()->FindItem( ref ) )
1636 {
1637 try
1638 {
1639 TUNER_SLIDER* tuner = new TUNER_SLIDER( this, m_panelTuners, aSheetPath, aSymbol );
1640 m_sizerTuners->Add( tuner );
1641 m_tuners.push_back( tuner );
1642 m_panelTuners->Layout();
1643 OnModify();
1644 }
1645 catch( const KI_PARAM_ERROR& e )
1646 {
1647 DisplayErrorMessage( nullptr, e.What() );
1648 }
1649 }
1650}
1651
1652
1653void SIMULATOR_FRAME_UI::UpdateTunerValue( const SCH_SHEET_PATH& aSheetPath, const KIID& aSymbol,
1654 const wxString& aRef, const wxString& aValue )
1655{
1656 SCH_ITEM* item = aSheetPath.GetItem( aSymbol );
1657 SCH_SYMBOL* symbol = dynamic_cast<SCH_SYMBOL*>( item );
1658
1659 if( !symbol )
1660 {
1661 DisplayErrorMessage( this, _( "Could not apply tuned value(s):" ) + wxS( " " )
1662 + wxString::Format( _( "%s not found" ), aRef ) );
1663 return;
1664 }
1665
1666 NULL_REPORTER devnull;
1668 SIM_MODEL& model = mgr.CreateModel( &aSheetPath, *symbol, devnull ).model;
1669
1670 const SIM_MODEL::PARAM* tunerParam = model.GetTunerParam();
1671
1672 if( !tunerParam )
1673 {
1674 DisplayErrorMessage( this, _( "Could not apply tuned value(s):" ) + wxS( " " )
1675 + wxString::Format( _( "%s is not tunable" ), aRef ) );
1676 return;
1677 }
1678
1679 model.SetParamValue( tunerParam->info.name, std::string( aValue.ToUTF8() ) );
1680 model.WriteFields( symbol->GetFields() );
1681
1682 m_schematicFrame->UpdateItem( symbol, false, true );
1684}
1685
1686
1688{
1689 m_tuners.remove( aTuner );
1690 aTuner->Destroy();
1691 m_panelTuners->Layout();
1692 OnModify();
1693}
1694
1695
1696void SIMULATOR_FRAME_UI::AddMeasurement( const wxString& aCmd )
1697{
1698 // -1 because the last one is for user input
1699 for( int i = 0; i < m_measurementsGrid->GetNumberRows(); i++ )
1700 {
1701 if ( m_measurementsGrid->GetCellValue( i, COL_MEASUREMENT ) == aCmd )
1702 return; // Don't create duplicates
1703 }
1704
1705 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() );
1706
1707 if( !plotTab )
1708 return;
1709
1710 wxString simType = simulator()->TypeToName( plotTab->GetSimType(), true );
1711 int row;
1712
1713 for( row = 0; row < m_measurementsGrid->GetNumberRows(); ++row )
1714 {
1715 if( m_measurementsGrid->GetCellValue( row, COL_MEASUREMENT ).IsEmpty() )
1716 break;
1717 }
1718
1719 if( !m_measurementsGrid->GetCellValue( row, COL_MEASUREMENT ).IsEmpty() )
1720 {
1721 m_measurementsGrid->AppendRows( 1 );
1722 row = m_measurementsGrid->GetNumberRows() - 1;
1723 }
1724
1725 m_measurementsGrid->SetCellValue( row, COL_MEASUREMENT, aCmd );
1726
1727 UpdateMeasurement( row );
1729 OnModify();
1730
1731 // Always leave at least one empty row for type-in:
1732 row = m_measurementsGrid->GetNumberRows() - 1;
1733
1734 if( !m_measurementsGrid->GetCellValue( row, COL_MEASUREMENT ).IsEmpty() )
1735 m_measurementsGrid->AppendRows( 1 );
1736}
1737
1738
1739void SIMULATOR_FRAME_UI::DoFourier( const wxString& aSignal, const wxString& aFundamental )
1740{
1741 wxString cmd = wxString::Format( wxS( "fourier %s %s" ),
1742 SPICE_VALUE( aFundamental ).ToSpiceString(),
1743 aSignal );
1744
1745 simulator()->Command( cmd.ToStdString() );
1746}
1747
1748
1750{
1751 return circuitModel().get();
1752}
1753
1754
1755void SIMULATOR_FRAME_UI::AddTrace( const wxString& aName, SIM_TRACE_TYPE aType )
1756{
1757 if( !GetCurrentSimTab() )
1758 {
1759 m_simConsole->AppendText( _( "Error: no current simulation.\n" ) );
1760 m_simConsole->SetInsertionPointEnd();
1761 return;
1762 }
1763
1764 SIM_TYPE simType = SPICE_CIRCUIT_MODEL::CommandToSimType( GetCurrentSimTab()->GetSimCommand() );
1765
1766 if( simType == ST_UNKNOWN )
1767 {
1768 m_simConsole->AppendText( _( "Error: simulation type not defined.\n" ) );
1769 m_simConsole->SetInsertionPointEnd();
1770 return;
1771 }
1772 else if( !SIM_TAB::IsPlottable( simType ) )
1773 {
1774 m_simConsole->AppendText( _( "Error: simulation type doesn't support plotting.\n" ) );
1775 m_simConsole->SetInsertionPointEnd();
1776 return;
1777 }
1778
1779 if( SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() ) )
1780 {
1781 if( simType == ST_AC )
1782 {
1783 updateTrace( aName, aType | SPT_AC_GAIN, plotTab );
1784 updateTrace( aName, aType | SPT_AC_PHASE, plotTab );
1785 }
1786 else if( simType == ST_SP )
1787 {
1788 updateTrace( aName, aType | SPT_AC_GAIN, plotTab );
1789 updateTrace( aName, aType | SPT_AC_PHASE, plotTab );
1790 }
1791 else
1792 {
1793 updateTrace( aName, aType, plotTab );
1794 }
1795
1796 plotTab->GetPlotWin()->UpdateAll();
1797 }
1798
1800 OnModify();
1801}
1802
1803
1804void SIMULATOR_FRAME_UI::SetUserDefinedSignals( const std::map<int, wxString>& aNewSignals )
1805{
1806 for( size_t ii = 0; ii < m_plotNotebook->GetPageCount(); ++ii )
1807 {
1808 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( m_plotNotebook->GetPage( ii ) );
1809
1810 if( !plotTab )
1811 continue;
1812
1813 for( const auto& [ id, existingSignal ] : m_userDefinedSignals )
1814 {
1815 int traceType = SPT_UNKNOWN;
1816 wxString vectorName = vectorNameFromSignalName( plotTab, existingSignal, &traceType );
1817
1818 if( aNewSignals.count( id ) == 0 )
1819 {
1820 if( plotTab->GetSimType() == ST_AC )
1821 {
1822 for( int subType : { SPT_AC_GAIN, SPT_AC_PHASE } )
1823 plotTab->DeleteTrace( vectorName, traceType | subType );
1824 }
1825 else if( plotTab->GetSimType() == ST_SP )
1826 {
1827 for( int subType : { SPT_SP_AMP, SPT_AC_PHASE } )
1828 plotTab->DeleteTrace( vectorName, traceType | subType );
1829 }
1830 else
1831 {
1832 plotTab->DeleteTrace( vectorName, traceType );
1833 }
1834 }
1835 else
1836 {
1837 if( plotTab->GetSimType() == ST_AC )
1838 {
1839 for( int subType : { SPT_AC_GAIN, SPT_AC_PHASE } )
1840 {
1841 if( TRACE* trace = plotTab->GetTrace( vectorName, traceType | subType ) )
1842 trace->SetName( aNewSignals.at( id ) );
1843 }
1844 }
1845 else if( plotTab->GetSimType() == ST_SP )
1846 {
1847 for( int subType : { SPT_SP_AMP, SPT_AC_PHASE } )
1848 {
1849 if( TRACE* trace = plotTab->GetTrace( vectorName, traceType | subType ) )
1850 trace->SetName( aNewSignals.at( id ) );
1851 }
1852 }
1853 else
1854 {
1855 if( TRACE* trace = plotTab->GetTrace( vectorName, traceType ) )
1856 trace->SetName( aNewSignals.at( id ) );
1857 }
1858 }
1859 }
1860 }
1861
1862 m_userDefinedSignals = aNewSignals;
1863
1866
1868 rebuildSignalsGrid( m_filter->GetValue() );
1871 OnModify();
1872}
1873
1874
1875void SIMULATOR_FRAME_UI::updateTrace( const wxString& aVectorName, int aTraceType,
1876 SIM_PLOT_TAB* aPlotTab, std::vector<double>* aDataX,
1877 bool aClearData )
1878{
1879 if( !m_simulatorFrame->SimFinished() && !simulator()->IsRunning())
1880 {
1881 aPlotTab->GetOrAddTrace( aVectorName, aTraceType );
1882 return;
1883 }
1884
1886
1887 aTraceType &= aTraceType & SPT_Y_AXIS_MASK;
1888 aTraceType |= getXAxisType( simType );
1889
1890 wxString simVectorName = aVectorName;
1891
1892 if( aTraceType & SPT_POWER )
1893 simVectorName = simVectorName.AfterFirst( '(' ).BeforeLast( ')' ) + wxS( ":power" );
1894
1895 if( !SIM_TAB::IsPlottable( simType ) )
1896 {
1897 // There is no plot to be shown
1898 simulator()->Command( wxString::Format( wxT( "print %s" ), aVectorName ).ToStdString() );
1899
1900 return;
1901 }
1902
1903 std::vector<double> data_x;
1904 std::vector<double> data_y;
1905
1906 if( !aDataX || aClearData )
1907 aDataX = &data_x;
1908
1909 // First, handle the x axis
1910 if( aDataX->empty() && !aClearData )
1911 {
1912 wxString xAxisName( simulator()->GetXAxis( simType ) );
1913
1914 if( xAxisName.IsEmpty() )
1915 return;
1916
1917 *aDataX = simulator()->GetGainVector( (const char*) xAxisName.c_str() );
1918 }
1919
1920 unsigned int size = aDataX->size();
1921
1922 switch( simType )
1923 {
1924 case ST_AC:
1925 if( aTraceType & SPT_AC_GAIN )
1926 data_y = simulator()->GetGainVector( (const char*) simVectorName.c_str(), size );
1927 else if( aTraceType & SPT_AC_PHASE )
1928 data_y = simulator()->GetPhaseVector( (const char*) simVectorName.c_str(), size );
1929 else
1930 wxFAIL_MSG( wxT( "Plot type missing AC_PHASE or AC_MAG bit" ) );
1931
1932 break;
1933 case ST_SP:
1934 if( aTraceType & SPT_SP_AMP )
1935 data_y = simulator()->GetGainVector( (const char*) simVectorName.c_str(), size );
1936 else if( aTraceType & SPT_AC_PHASE )
1937 data_y = simulator()->GetPhaseVector( (const char*) simVectorName.c_str(), size );
1938 else
1939 wxFAIL_MSG( wxT( "Plot type missing AC_PHASE or SPT_SP_AMP bit" ) );
1940
1941 break;
1942
1943 case ST_DC:
1944 data_y = simulator()->GetGainVector( (const char*) simVectorName.c_str(), -1 );
1945 break;
1946
1947 case ST_NOISE:
1948 case ST_TRAN:
1949 case ST_FFT:
1950 data_y = simulator()->GetGainVector( (const char*) simVectorName.c_str(), size );
1951 break;
1952
1953 default:
1954 wxFAIL_MSG( wxT( "Unhandled plot type" ) );
1955 }
1956
1957 SPICE_DC_PARAMS source1, source2;
1958 int sweepCount = 1;
1959 size_t sweepSize = std::numeric_limits<size_t>::max();
1960
1961 if( simType == ST_DC
1962 && circuitModel()->ParseDCCommand( aPlotTab->GetSimCommand(), &source1, &source2 )
1963 && !source2.m_source.IsEmpty() )
1964 {
1965 SPICE_VALUE v = ( source2.m_vend - source2.m_vstart ) / source2.m_vincrement;
1966
1967 sweepCount = KiROUND( v.ToDouble() ) + 1;
1968 sweepSize = aDataX->size() / sweepCount;
1969 }
1970
1971 if( TRACE* trace = aPlotTab->GetOrAddTrace( aVectorName, aTraceType ) )
1972 {
1973 if( data_y.size() >= size )
1974 aPlotTab->SetTraceData( trace, *aDataX, data_y, sweepCount, sweepSize );
1975 }
1976}
1977
1978
1979// TODO make sure where to instantiate and how to style correct
1980// Better ask someone..
1981template void SIMULATOR_FRAME_UI::signalsGridCursorUpdate<SIGNALS_GRID_COLUMNS, int, int>(
1982 SIGNALS_GRID_COLUMNS, int, int );
1983
1984template <typename T, typename U, typename R>
1985void SIMULATOR_FRAME_UI::signalsGridCursorUpdate( T t, U u, R r ) // t=cursor type/signals' grid col, u=cursor number/cursor "id", r=table's row
1986{
1987 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() );
1988
1989 wxString signalName = m_signalsGrid->GetCellValue( r, COL_SIGNAL_NAME );
1990 int traceType = SPT_UNKNOWN;
1991 wxString vectorName = vectorNameFromSignalName( plotTab, signalName, &traceType );
1992
1993 wxGridCellAttrPtr attr = m_signalsGrid->GetOrCreateCellAttrPtr( r, static_cast<int>( t ) );
1994
1995 if( TRACE* trace = plotTab ? plotTab->GetTrace( vectorName, traceType ) : nullptr )
1996 {
1997 attr->SetReadOnly(); // not really; we delegate interactivity to GRID_TRICKS
1998
1999 if( t >= SIGNALS_GRID_COLUMNS::COL_SIGNAL_SHOW )
2000 {
2001 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
2002 }
2003
2004 if constexpr ( std::is_enum<T>::value )
2005 {
2006 if( t == SIGNALS_GRID_COLUMNS::COL_SIGNAL_SHOW )
2007 {
2008 m_signalsGrid->SetCellValue( r, static_cast<int>( t ), wxS( "1" ) );
2009 }
2010 else if( t == SIGNALS_GRID_COLUMNS::COL_SIGNAL_COLOR )
2011 {
2012 if( !attr->HasRenderer() )
2013 attr->SetRenderer( new GRID_CELL_COLOR_RENDERER( this ) );
2014
2015 if( !attr->HasEditor() )
2016 attr->SetEditor( new GRID_CELL_COLOR_SELECTOR( this, m_signalsGrid ) );
2017
2018 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
2019 attr->SetReadOnly( false );
2020
2021 KIGFX::COLOR4D color( trace->GetPen().GetColour() );
2022 m_signalsGrid->SetCellValue( r, COL_SIGNAL_COLOR, color.ToCSSString() );
2023 }
2024 else if( t == SIGNALS_GRID_COLUMNS::COL_CURSOR_1
2025 || t == SIGNALS_GRID_COLUMNS::COL_CURSOR_2
2026 || t > SIGNALS_GRID_COLUMNS::COL_CURSOR_2 )
2027 {
2028 if( !attr->HasRenderer() )
2029 attr->SetRenderer( new wxGridCellBoolRenderer() );
2030
2031 if( u > 0 && trace->HasCursor( u ) )
2032 m_signalsGrid->SetCellValue( r, static_cast<int>( t ), wxS( "1" ) );
2033 else
2034 m_signalsGrid->SetCellValue( r, static_cast<int>( t ), wxEmptyString );
2035 }
2036 }
2037 }
2038 else
2039 {
2040 if constexpr ( std::is_enum<T>::value )
2041 {
2042 if( t == SIGNALS_GRID_COLUMNS::COL_SIGNAL_SHOW )
2043 {
2044 m_signalsGrid->SetCellValue( r, static_cast<int>( t ), wxEmptyString );
2045 }
2046 else if( t == SIGNALS_GRID_COLUMNS::COL_SIGNAL_COLOR
2047 || t == SIGNALS_GRID_COLUMNS::COL_CURSOR_1
2048 || t == SIGNALS_GRID_COLUMNS::COL_CURSOR_2
2049 || t > SIGNALS_GRID_COLUMNS::COL_CURSOR_2 )
2050 {
2051 attr->SetEditor( nullptr );
2052 attr->SetRenderer( nullptr );
2053 attr->SetReadOnly();
2054 m_signalsGrid->SetCellValue( r, static_cast<int>( t ), wxEmptyString );
2055 }
2056 }
2057 }
2058}
2059
2060
2062{
2063 for( int row = 0; row < m_signalsGrid->GetNumberRows(); ++row )
2064 {
2069
2070 if( ( m_signalsGrid->GetNumberCols() - 1 ) > COL_CURSOR_2 )
2071 {
2072 for( int i = 3; i < m_customCursorsCnt; i++ )
2073 {
2074 int tm = i + 2;
2076 static_cast<SIGNALS_GRID_COLUMNS>( tm ),
2077 i,
2078 row );
2079 }
2080 }
2081 }
2082 m_signalsGrid->Refresh();
2083}
2084
2085
2087{
2088 auto quoteNetNames = [&]( wxString aExpression ) -> wxString
2089 {
2090 std::vector<bool> mask( aExpression.length(), false );
2091
2092 for( const auto& netname : m_netnames )
2093 {
2094 size_t pos = aExpression.find( netname );
2095
2096 while( pos != wxString::npos )
2097 {
2098 for( size_t i = 0; i < netname.length(); ++i )
2099 {
2100 mask[pos + i] = true; // Mark the positions of the netname
2101 }
2102 pos = aExpression.find( netname, pos + 1 ); // Find the next occurrence
2103 }
2104 }
2105
2106 wxString quotedNetnames = "";
2107 bool startQuote = true;
2108
2109 // put quotes around all the positions that were found above
2110 for( size_t i = 0; i < aExpression.length(); i++ )
2111 {
2112 if( mask[i] && startQuote )
2113 {
2114 quotedNetnames = quotedNetnames + "\"";
2115 startQuote = false;
2116 }
2117 else if( !mask[i] && !startQuote )
2118 {
2119 quotedNetnames = quotedNetnames + "\"";
2120 startQuote = true;
2121 }
2122 wxString ch = aExpression[i];
2123 quotedNetnames = quotedNetnames + ch;
2124 }
2125
2126 if( !startQuote )
2127 {
2128 quotedNetnames = quotedNetnames + "\"";
2129 }
2130 return quotedNetnames;
2131 };
2132
2133 for( const auto& [ id, signal ] : m_userDefinedSignals )
2134 {
2135 constexpr const char* cmd = "let user{} = {}";
2136
2137 simulator()->Command( "echo " + fmt::format( cmd, id, signal.ToStdString() ) );
2138 simulator()->Command( fmt::format( cmd, id, quoteNetNames( signal ).ToStdString() ) );
2139 }
2140}
2141
2142
2144{
2145 WX_STRING_REPORTER reporter;
2146
2147 for( const TUNER_SLIDER* tuner : m_tuners )
2148 {
2149 SCH_SHEET_PATH sheetPath;
2150 wxString ref = tuner->GetSymbolRef();
2151 KIID symbolId = tuner->GetSymbol( &sheetPath );
2152 SCH_ITEM* schItem = sheetPath.GetItem( symbolId );
2153 SCH_SYMBOL* symbol = dynamic_cast<SCH_SYMBOL*>( schItem );
2154
2155 if( !symbol )
2156 {
2157 reporter.Report( wxString::Format( _( "%s not found" ), ref ) );
2158 continue;
2159 }
2160
2161 const SPICE_ITEM* item = GetExporter()->FindItem( tuner->GetSymbolRef() );
2162
2163 if( !item || !item->model->GetTunerParam() )
2164 {
2165 reporter.Report( wxString::Format( _( "%s is not tunable" ), ref ) );
2166 continue;
2167 }
2168
2169 double floatVal = tuner->GetValue().ToDouble();
2170
2171 simulator()->Command( item->model->SpiceGenerator().TunerCommand( *item, floatVal ) );
2172 }
2173
2174 if( reporter.HasMessage() )
2175 DisplayErrorMessage( this, _( "Could not apply tuned value(s):" ) + wxS( "\n" )
2176 + reporter.GetMessages() );
2177}
2178
2179bool SIMULATOR_FRAME_UI::LoadWorkbook( const wxString& aPath )
2180{
2181 wxTextFile file( aPath );
2182
2183 if( !file.Open() )
2184 return false;
2185
2186 wxString firstLine = file.GetFirstLine();
2187 long dummy;
2188 bool legacy = firstLine.StartsWith( wxT( "version " ) ) || firstLine.ToLong( &dummy );
2189
2190 file.Close();
2191
2192 m_plotNotebook->DeleteAllPages();
2193 m_userDefinedSignals.clear();
2194
2195 if( legacy )
2196 {
2197 if( !loadLegacyWorkbook( aPath ) )
2198 return false;
2199 }
2200 else
2201 {
2202 if( !loadJsonWorkbook( aPath ) )
2203 return false;
2204 }
2205
2207
2208 rebuildSignalsGrid( m_filter->GetValue() );
2212
2213 wxFileName filename( aPath );
2214 filename.MakeRelativeTo( m_schematicFrame->Prj().GetProjectPath() );
2215
2216 // Remember the loaded workbook filename.
2217 simulator()->Settings()->SetWorkbookFilename( filename.GetFullPath() );
2218
2219 return true;
2220}
2221
2222
2223bool SIMULATOR_FRAME_UI::loadJsonWorkbook( const wxString& aPath )
2224{
2225 wxFFileInputStream fp( aPath, wxT( "rt" ) );
2226 wxStdInputStream fstream( fp );
2227
2228 if( !fp.IsOk() )
2229 return false;
2230
2231 try
2232 {
2233 nlohmann::json js = nlohmann::json::parse( fstream, nullptr, true, true );
2234
2235 std::map<SIM_PLOT_TAB*, nlohmann::json> traceInfo;
2236
2237 for( const nlohmann::json& tab_js : js[ "tabs" ] )
2238 {
2239 wxString simCommand;
2242
2243 for( const nlohmann::json& cmd : tab_js[ "commands" ] )
2244 {
2245 if( cmd == ".kicad adjustpaths" )
2247 else if( cmd == ".save all" )
2249 else if( cmd == ".probe alli" )
2251 else if( cmd == ".probe allp" )
2253 else if( cmd == ".kicad esavenone" )
2254 simOptions &= ~NETLIST_EXPORTER_SPICE::OPTION_SAVE_ALL_EVENTS;
2255 else
2256 simCommand += wxString( cmd.get<wxString>() ).Trim();
2257 }
2258
2259 SIM_TAB* simTab = NewSimTab( simCommand );
2260 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( simTab );
2261
2262 simTab->SetSimOptions( simOptions );
2263
2264 if( plotTab )
2265 {
2266 if( tab_js.contains( "traces" ) )
2267 traceInfo[plotTab] = tab_js[ "traces" ];
2268
2269 if( tab_js.contains( "measurements" ) )
2270 {
2271 for( const nlohmann::json& m_js : tab_js[ "measurements" ] )
2272 plotTab->Measurements().emplace_back( m_js[ "expr" ], m_js[ "format" ] );
2273 }
2274
2275 plotTab->SetDottedSecondary( tab_js[ "dottedSecondary" ] );
2276 plotTab->ShowGrid( tab_js[ "showGrid" ] );
2277
2278 if( tab_js.contains( "fixedY1scale" ) )
2279 {
2280 const nlohmann::json& scale_js = tab_js[ "fixedY1scale" ];
2281 plotTab->SetY1Scale( true, scale_js[ "min" ], scale_js[ "max" ] );
2282 plotTab->GetPlotWin()->LockY( true );
2283 }
2284
2285 if( tab_js.contains( "fixedY2scale" ) )
2286 {
2287 const nlohmann::json& scale_js = tab_js[ "fixedY2scale" ];
2288 plotTab->SetY2Scale( true, scale_js[ "min" ], scale_js[ "max" ] );
2289 plotTab->GetPlotWin()->LockY( true );
2290 }
2291
2292 if( tab_js.contains( "fixedY3scale" ) )
2293 {
2294 plotTab->EnsureThirdYAxisExists();
2295 const nlohmann::json& scale_js = tab_js[ "fixedY3scale" ];
2296 plotTab->SetY3Scale( true, scale_js[ "min" ], scale_js[ "max" ] );
2297 plotTab->GetPlotWin()->LockY( true );
2298 }
2299
2300 if( tab_js.contains( "legend" ) )
2301 {
2302 const nlohmann::json& legend_js = tab_js[ "legend" ];
2303 plotTab->SetLegendPosition( wxPoint( legend_js[ "x" ], legend_js[ "y" ] ) );
2304 plotTab->ShowLegend( true );
2305 }
2306
2307 if( tab_js.contains( "margins" ) )
2308 {
2309 const nlohmann::json& margins_js = tab_js[ "margins" ];
2310 plotTab->GetPlotWin()->SetMargins( margins_js[ "top" ],
2311 margins_js[ "right" ],
2312 margins_js[ "bottom" ],
2313 margins_js[ "left" ] );
2314 }
2315 }
2316 }
2317
2318 int ii = 0;
2319
2320 if( js.contains( "user_defined_signals" ) )
2321 {
2322 for( const nlohmann::json& signal_js : js[ "user_defined_signals" ] )
2323 m_userDefinedSignals[ii++] = wxString( signal_js.get<wxString>() );
2324 }
2325
2326 if( SIM_TAB* simTab = GetCurrentSimTab() )
2327 {
2328 m_simulatorFrame->LoadSimulator( simTab->GetSimCommand(), simTab->GetSimOptions() );
2329
2330 if( SIM_TAB* firstTab = dynamic_cast<SIM_TAB*>( m_plotNotebook->GetPage( 0 ) ) )
2331 firstTab->SetLastSchTextSimCommand( js["last_sch_text_sim_command"] );
2332 }
2333
2334 int tempCustomCursorsCnt = 0;
2335
2336 if( js.contains( "custom_cursors" ) )
2337 tempCustomCursorsCnt = js["custom_cursors"];
2338 else
2339 tempCustomCursorsCnt = 2; // Kind of virtual, for the initial loading of the new setting
2340
2341 if( ( tempCustomCursorsCnt > m_customCursorsCnt ) && m_customCursorsCnt > 2 )
2342 tempCustomCursorsCnt = 2 * tempCustomCursorsCnt - m_customCursorsCnt;
2343
2344 for( int yy = 0; yy <= ( tempCustomCursorsCnt - m_customCursorsCnt ); yy++ )
2346
2347 auto addCursor =
2348 [=,this]( SIM_PLOT_TAB* aPlotTab, TRACE* aTrace, const wxString& aSignalName,
2349 int aCursorId, const nlohmann::json& aCursor_js )
2350 {
2351 if( aCursorId >= 1 )
2352 {
2353 CURSOR* cursor = new CURSOR( aTrace, aPlotTab );
2354
2355 cursor->SetName( aSignalName );
2356 cursor->SetCoordX( aCursor_js[ "position" ] );
2357
2358 aTrace->SetCursor( aCursorId, cursor );
2359 aPlotTab->GetPlotWin()->AddLayer( cursor );
2360 }
2361
2362 if( aCursorId == -1 )
2363 {
2364 // We are a "cursorD"
2365 m_cursorFormatsDyn[2][0].FromString( aCursor_js["x_format"] );
2366 m_cursorFormatsDyn[2][1].FromString( aCursor_js["y_format"] );
2367 }
2368 else
2369 {
2370 if( aCursorId < 3 )
2371 {
2372 m_cursorFormatsDyn[aCursorId - 1][0].FromString(
2373 aCursor_js["x_format"] );
2374 m_cursorFormatsDyn[aCursorId - 1][1].FromString(
2375 aCursor_js["y_format"] );
2376 }
2377 else
2378 {
2379 m_cursorFormatsDyn[aCursorId][0].FromString( aCursor_js["x_format"] );
2380 m_cursorFormatsDyn[aCursorId][1].FromString( aCursor_js["y_format"] );
2381 }
2382 }
2383 };
2384
2385 for( const auto& [ plotTab, traces_js ] : traceInfo )
2386 {
2387 for( const nlohmann::json& trace_js : traces_js )
2388 {
2389 wxString signalName = trace_js[ "signal" ];
2390 wxString vectorName = vectorNameFromSignalName( plotTab, signalName, nullptr );
2391 TRACE* trace = plotTab->GetOrAddTrace( vectorName, trace_js[ "trace_type" ] );
2392
2393 if( trace )
2394 {
2395 if( trace_js.contains( "cursorD" ) )
2396 addCursor( plotTab, trace, signalName, -1, trace_js[ "cursorD" ] );
2397
2398 std::vector<const char*> aVec;
2399 aVec.clear();
2400
2401 for( int i = 1; i <= tempCustomCursorsCnt; i++ )
2402 {
2403 wxString str = "cursor" + std::to_string( i );
2404 aVec.emplace_back( str.c_str() );
2405
2406 if( trace_js.contains( aVec[i - 1] ) )
2407 addCursor( plotTab, trace, signalName, i, trace_js[aVec[i - 1]] );
2408 }
2409
2410 if( trace_js.contains( "color" ) )
2411 {
2412 wxColour color;
2413 color.Set( wxString( trace_js["color"].get<wxString>() ) );
2414 trace->SetTraceColour( color );
2415 plotTab->UpdateTraceStyle( trace );
2416 }
2417 }
2418 }
2419
2420 plotTab->UpdatePlotColors();
2421 }
2422 }
2423 catch( nlohmann::json::parse_error& error )
2424 {
2425 wxLogTrace( traceSettings, wxT( "Json parse error reading %s: %s" ), aPath, error.what() );
2426
2427 return false;
2428 }
2429 catch( nlohmann::json::type_error& error )
2430 {
2431 wxLogTrace( traceSettings, wxT( "Json type error reading %s: %s" ), aPath, error.what() );
2432
2433 return false;
2434 }
2435 catch( nlohmann::json::invalid_iterator& error )
2436 {
2437 wxLogTrace( traceSettings, wxT( "Json invalid_iterator error reading %s: %s" ), aPath, error.what() );
2438
2439 return false;
2440 }
2441 catch( nlohmann::json::out_of_range& error )
2442 {
2443 wxLogTrace( traceSettings, wxT( "Json out_of_range error reading %s: %s" ), aPath, error.what() );
2444
2445 return false;
2446 }
2447 catch( ... )
2448 {
2449 wxLogTrace( traceSettings, wxT( "Error reading %s" ), aPath );
2450 return false;
2451 }
2452
2453 return true;
2454}
2455
2456void SIMULATOR_FRAME_UI::SaveCursorToWorkbook( nlohmann::json& aTraceJs, TRACE* aTrace, int aCursorId )
2457{
2458 int cursorIdAfterD = aCursorId;
2459
2460 if( aCursorId > 3 )
2461 cursorIdAfterD = cursorIdAfterD - 1;
2462
2463
2464 if( CURSOR* cursor = aTrace->GetCursor( aCursorId ) )
2465 {
2466 aTraceJs["cursor" + wxString( "" ) << aCursorId] =
2467 nlohmann::json( { { "position", cursor->GetCoords().x },
2468 { "x_format", m_cursorFormatsDyn[cursorIdAfterD][0].ToString() },
2469 { "y_format", m_cursorFormatsDyn[cursorIdAfterD][1].ToString() } } );
2470 }
2471
2472 if( cursorIdAfterD < 3 && ( aTrace->GetCursor( 1 ) || aTrace->GetCursor( 2 ) ) )
2473 {
2474 aTraceJs["cursorD"] =
2475 nlohmann::json( { { "x_format", m_cursorFormatsDyn[2][0].ToString() },
2476 { "y_format", m_cursorFormatsDyn[2][1].ToString() } } );
2477 }
2478}
2479
2480
2481bool SIMULATOR_FRAME_UI::SaveWorkbook( const wxString& aPath )
2482{
2484
2485 wxFileName filename = aPath;
2486 filename.SetExt( FILEEXT::WorkbookFileExtension );
2487
2488 wxFile file;
2489
2490 file.Create( filename.GetFullPath(), true /* overwrite */ );
2491
2492 if( !file.IsOpened() )
2493 return false;
2494
2495 nlohmann::json tabs_js = nlohmann::json::array();
2496
2497 for( size_t i = 0; i < m_plotNotebook->GetPageCount(); i++ )
2498 {
2499 SIM_TAB* simTab = dynamic_cast<SIM_TAB*>( m_plotNotebook->GetPage( i ) );
2500
2501 if( !simTab )
2502 continue;
2503
2504 SIM_TYPE simType = simTab->GetSimType();
2505
2506 nlohmann::json commands_js = nlohmann::json::array();
2507
2508 commands_js.push_back( simTab->GetSimCommand() );
2509
2510 int options = simTab->GetSimOptions();
2511
2513 commands_js.push_back( ".kicad adjustpaths" );
2514
2516 commands_js.push_back( ".save all" );
2517
2519 commands_js.push_back( ".probe alli" );
2520
2522 commands_js.push_back( ".probe allp" );
2523
2525 commands_js.push_back( ".kicad esavenone" );
2526
2527 nlohmann::json tab_js = nlohmann::json(
2528 { { "analysis", SPICE_SIMULATOR::TypeToName( simType, true ) },
2529 { "commands", commands_js } } );
2530
2531 if( SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( simTab ) )
2532 {
2533 nlohmann::json traces_js = nlohmann::json::array();
2534
2535 auto findSignalName =
2536 [&]( const wxString& aVectorName ) -> wxString
2537 {
2538 wxString vectorName;
2539 wxString suffix;
2540
2541 if( aVectorName.EndsWith( _( " (phase)" ) ) )
2542 suffix = _( " (phase)" );
2543 else if( aVectorName.EndsWith( _( " (gain)" ) ) )
2544 suffix = _( " (gain)" );
2545
2546 vectorName = aVectorName.Left( aVectorName.Length() - suffix.Length() );
2547
2548 for( const auto& [ id, signal ] : m_userDefinedSignals )
2549 {
2550 if( vectorName == vectorNameFromSignalId( id ) )
2551 return signal + suffix;
2552 }
2553
2554 return aVectorName;
2555 };
2556
2557 for( const auto& [name, trace] : plotTab->GetTraces() )
2558 {
2559 nlohmann::json trace_js = nlohmann::json(
2560 { { "trace_type", (int) trace->GetType() },
2561 { "signal", findSignalName( trace->GetDisplayName() ) },
2562 { "color", COLOR4D( trace->GetTraceColour() ).ToCSSString() } } );
2563
2564 for( int ii = 1; ii <= m_customCursorsCnt; ii++ )
2565 SaveCursorToWorkbook( trace_js, trace, ii );
2566
2567 if( trace->GetCursor( 1 ) || trace->GetCursor( 2 ) )
2568 {
2569 trace_js["cursorD"] = nlohmann::json(
2570 { { "x_format", m_cursorFormatsDyn[2][0].ToString() },
2571 { "y_format", m_cursorFormatsDyn[2][1].ToString() } } );
2572 }
2573
2574 traces_js.push_back( trace_js );
2575 }
2576
2577 nlohmann::json measurements_js = nlohmann::json::array();
2578
2579 for( const auto& [ measurement, format ] : plotTab->Measurements() )
2580 {
2581 measurements_js.push_back( nlohmann::json( { { "expr", measurement },
2582 { "format", format } } ) );
2583 }
2584
2585 tab_js[ "traces" ] = traces_js;
2586 tab_js[ "measurements" ] = measurements_js;
2587 tab_js[ "dottedSecondary" ] = plotTab->GetDottedSecondary();
2588 tab_js[ "showGrid" ] = plotTab->IsGridShown();
2589
2590 double min, max;
2591
2592 if( plotTab->GetY1Scale( &min, &max ) )
2593 tab_js[ "fixedY1scale" ] = nlohmann::json( { { "min", min }, { "max", max } } );
2594
2595 if( plotTab->GetY2Scale( &min, &max ) )
2596 tab_js[ "fixedY2scale" ] = nlohmann::json( { { "min", min }, { "max", max } } );
2597
2598 if( plotTab->GetY3Scale( &min, &max ) )
2599 tab_js[ "fixedY3scale" ] = nlohmann::json( { { "min", min }, { "max", max } } );
2600
2601 if( plotTab->IsLegendShown() )
2602 {
2603 tab_js[ "legend" ] = nlohmann::json( { { "x", plotTab->GetLegendPosition().x },
2604 { "y", plotTab->GetLegendPosition().y } } );
2605 }
2606
2607 mpWindow* plotWin = plotTab->GetPlotWin();
2608
2609 tab_js[ "margins" ] = nlohmann::json( { { "left", plotWin->GetMarginLeft() },
2610 { "right", plotWin->GetMarginRight() },
2611 { "top", plotWin->GetMarginTop() },
2612 { "bottom", plotWin->GetMarginBottom() } } );
2613 }
2614
2615 tabs_js.push_back( tab_js );
2616 }
2617
2618 nlohmann::json userDefinedSignals_js = nlohmann::json::array();
2619
2620 for( const auto& [ id, signal ] : m_userDefinedSignals )
2621 userDefinedSignals_js.push_back( signal );
2622
2623 // clang-format off
2624 nlohmann::json js = nlohmann::json( { { "version", 7 },
2625 { "tabs", tabs_js },
2626 { "user_defined_signals", userDefinedSignals_js },
2627 { "custom_cursors", m_customCursorsCnt - 1 } } ); // Since we start +1 on init
2628 // clang-format on
2629
2630 // Store the value of any simulation command found on the schematic sheet in a SCH_TEXT
2631 // object. If this changes we want to warn the user and ask them if they want to update
2632 // the corresponding panel's sim command.
2633 if( m_plotNotebook->GetPageCount() > 0 )
2634 {
2635 SIM_TAB* simTab = dynamic_cast<SIM_TAB*>( m_plotNotebook->GetPage( 0 ) );
2636 js[ "last_sch_text_sim_command" ] = simTab->GetLastSchTextSimCommand();
2637 }
2638
2639 std::stringstream buffer;
2640 buffer << std::setw( 2 ) << js << std::endl;
2641
2642 bool res = file.Write( buffer.str() );
2643 file.Close();
2644
2645 // Store the filename of the last saved workbook.
2646 if( res )
2647 {
2648 filename.MakeRelativeTo( m_schematicFrame->Prj().GetProjectPath() );
2649 simulator()->Settings()->SetWorkbookFilename( filename.GetFullPath() );
2650 }
2651
2652 return res;
2653}
2654
2655
2657{
2658 switch( aType )
2659 {
2661 case ST_AC: return SPT_LIN_FREQUENCY;
2662 case ST_SP: return SPT_LIN_FREQUENCY;
2663 case ST_FFT: return SPT_LIN_FREQUENCY;
2664 case ST_DC: return SPT_SWEEP;
2665 case ST_TRAN: return SPT_TIME;
2666 case ST_NOISE: return SPT_LIN_FREQUENCY;
2667 default: wxFAIL_MSG( wxS( "Unhandled simulation type" ) ); return SPT_UNKNOWN;
2668 }
2669}
2670
2671
2673{
2674 wxString output;
2675 wxString ref;
2676 wxString source;
2677 wxString scale;
2678 SPICE_VALUE pts;
2679 SPICE_VALUE fStart;
2680 SPICE_VALUE fStop;
2681 bool saveAll;
2682
2683 if( GetCurrentSimTab() )
2684 {
2685 circuitModel()->ParseNoiseCommand( GetCurrentSimTab()->GetSimCommand(), &output, &ref,
2686 &source, &scale, &pts, &fStart, &fStop, &saveAll );
2687 }
2688
2689 return source;
2690}
2691
2692
2693void SIMULATOR_FRAME_UI::TogglePanel( wxPanel* aPanel, wxSplitterWindow* aSplitterWindow,
2694 int& aSashPosition )
2695{
2696 bool isShown = aPanel->IsShown();
2697
2698 if( isShown )
2699 aSashPosition = aSplitterWindow->GetSashPosition();
2700
2701 aPanel->Show( !isShown );
2702
2703 aSplitterWindow->SetSashInvisible( isShown );
2704 aSplitterWindow->SetSashPosition( isShown ? -1 : aSashPosition, true );
2705
2706 aSplitterWindow->UpdateSize();
2707 m_parent->Refresh();
2708 m_parent->Layout();
2709}
2710
2711
2713{
2714 return m_panelConsole->IsShown();
2715}
2716
2717
2719{
2721}
2722
2723
2725{
2726 return m_sidePanel->IsShown();
2727}
2728
2729
2731{
2733}
2734
2735
2737{
2739
2740 // Rebuild the color list to plot traces
2742
2743 // Now send changes to all SIM_PLOT_TAB
2744 for( size_t page = 0; page < m_plotNotebook->GetPageCount(); page++ )
2745 {
2746 wxWindow* curPage = m_plotNotebook->GetPage( page );
2747
2748 // ensure it is truly a plot plotTab and not the (zero plots) placeholder
2749 // which is only SIM_TAB
2750 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( curPage );
2751
2752 if( plotTab )
2753 plotTab->UpdatePlotColors();
2754 }
2755}
2756
2757
2758void SIMULATOR_FRAME_UI::onPlotClose( wxAuiNotebookEvent& event )
2759{
2760 OnModify();
2761}
2762
2763
2764void SIMULATOR_FRAME_UI::onPlotClosed( wxAuiNotebookEvent& event )
2765{
2766 CallAfter( [this]()
2767 {
2769 rebuildSignalsGrid( m_filter->GetValue() );
2771
2772 SIM_TAB* panel = GetCurrentSimTab();
2773
2774 if( !panel || panel->GetSimType() != ST_OP )
2775 {
2776 SCHEMATIC& schematic = m_schematicFrame->Schematic();
2777 schematic.ClearOperatingPoints();
2780 }
2781 } );
2782}
2783
2784
2786{
2787 if( SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() ) )
2788 {
2789 std::vector<std::pair<wxString, wxString>>& measurements = plotTab->Measurements();
2790
2791 measurements.clear();
2792
2793 for( int row = 0; row < m_measurementsGrid->GetNumberRows(); ++row )
2794 {
2795 if( !m_measurementsGrid->GetCellValue( row, COL_MEASUREMENT ).IsEmpty() )
2796 {
2797 measurements.emplace_back( m_measurementsGrid->GetCellValue( row, COL_MEASUREMENT ),
2798 m_measurementsGrid->GetCellValue( row, COL_MEASUREMENT_FORMAT ) );
2799 }
2800 }
2801 }
2802}
2803
2804
2805void SIMULATOR_FRAME_UI::onPlotChanging( wxAuiNotebookEvent& event )
2806{
2808
2809 event.Skip();
2810}
2811
2812
2814{
2816 rebuildSignalsGrid( m_filter->GetValue() );
2818
2820
2821 for( int row = 0; row < m_measurementsGrid->GetNumberRows(); ++row )
2822 UpdateMeasurement( row );
2823}
2824
2825
2826void SIMULATOR_FRAME_UI::onPlotChanged( wxAuiNotebookEvent& event )
2827{
2828 if( SIM_TAB* simTab = GetCurrentSimTab() )
2829 simulator()->Command( "setplot " + simTab->GetSpicePlotName().ToStdString() );
2830
2832
2833 event.Skip();
2834}
2835
2836
2838{
2840
2841 if( SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() ) )
2842 {
2843 for( const auto& [ measurement, format ] : plotTab->Measurements() )
2844 {
2845 int row = m_measurementsGrid->GetNumberRows();
2846 m_measurementsGrid->AppendRows();
2847 m_measurementsGrid->SetCellValue( row, COL_MEASUREMENT, measurement );
2848 m_measurementsGrid->SetCellValue( row, COL_MEASUREMENT_FORMAT, format );
2849 }
2850
2851 if( plotTab->GetSimType() == ST_TRAN || plotTab->GetSimType() == ST_AC
2852 || plotTab->GetSimType() == ST_DC || plotTab->GetSimType() == ST_SP )
2853 {
2854 m_measurementsGrid->AppendRows(); // Empty row at end
2855 }
2856 }
2857}
2858
2859
2860void SIMULATOR_FRAME_UI::onPlotDragged( wxAuiNotebookEvent& event )
2861{
2862}
2863
2864
2865std::shared_ptr<SPICE_SIMULATOR> SIMULATOR_FRAME_UI::simulator() const
2866{
2868}
2869
2870
2871std::shared_ptr<SPICE_CIRCUIT_MODEL> SIMULATOR_FRAME_UI::circuitModel() const
2872{
2874}
2875
2876
2878{
2879 SUPPRESS_GRID_CELL_EVENTS raii( this );
2880
2882
2883 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() );
2884
2885 if( !plotTab )
2886 return;
2887
2888 // Update cursor values
2889 CURSOR* cursor1 = nullptr;
2890 wxString cursor1Name;
2891 wxString cursor1Units;
2892 CURSOR* cursor2 = nullptr;
2893 wxString cursor2Name;
2894 wxString cursor2Units;
2895
2896 auto getUnitsY =
2897 [&]( TRACE* aTrace ) -> wxString
2898 {
2899 if( plotTab->GetSimType() == ST_AC )
2900 {
2901 if( aTrace->GetType() & SPT_AC_PHASE )
2902 return plotTab->GetUnitsY2();
2903 else
2904 return plotTab->GetUnitsY1();
2905 }
2906 else
2907 {
2908 if( aTrace->GetType() & SPT_POWER )
2909 return plotTab->GetUnitsY3();
2910 else if( aTrace->GetType() & SPT_CURRENT )
2911 return plotTab->GetUnitsY2();
2912 else
2913 return plotTab->GetUnitsY1();
2914 }
2915 };
2916
2917 auto getNameY =
2918 [&]( TRACE* aTrace ) -> wxString
2919 {
2920 if( plotTab->GetSimType() == ST_AC )
2921 {
2922 if( aTrace->GetType() & SPT_AC_PHASE )
2923 return plotTab->GetLabelY2();
2924 else
2925 return plotTab->GetLabelY1();
2926 }
2927 else
2928 {
2929 if( aTrace->GetType() & SPT_POWER )
2930 return plotTab->GetLabelY3();
2931 else if( aTrace->GetType() & SPT_CURRENT )
2932 return plotTab->GetLabelY2();
2933 else
2934 return plotTab->GetLabelY1();
2935 }
2936 };
2937
2938 auto formatValue =
2939 [this]( double aValue, int aCursorId, int aCol ) -> wxString
2940 {
2941 if( ( !m_simulatorFrame->SimFinished() && aCol == 1 ) || std::isnan( aValue ) )
2942 return wxS( "--" );
2943 else
2944 return SPICE_VALUE( aValue ).ToString( m_cursorFormatsDyn[ aCursorId ][ aCol ] );
2945 };
2946
2947 for( const auto& [name, trace] : plotTab->GetTraces() )
2948 {
2949 if( CURSOR* cursor = trace->GetCursor( 1 ) )
2950 {
2951 cursor1 = cursor;
2952 cursor1Name = getNameY( trace );
2953 cursor1Units = getUnitsY( trace );
2954
2955 wxRealPoint coords = cursor->GetCoords();
2956 int row = m_cursorsGrid->GetNumberRows();
2957
2958 m_cursorFormatsDyn[0][0].UpdateUnits( plotTab->GetUnitsX() );
2959 m_cursorFormatsDyn[0][1].UpdateUnits( cursor1Units );
2960
2961 m_cursorsGrid->AppendRows( 1 );
2962 m_cursorsGrid->SetCellValue( row, COL_CURSOR_NAME, wxS( "1" ) );
2963 m_cursorsGrid->SetCellValue( row, COL_CURSOR_SIGNAL, cursor->GetName() );
2964 m_cursorsGrid->SetCellValue( row, COL_CURSOR_X, formatValue( coords.x, 0, 0 ) );
2965 m_cursorsGrid->SetCellValue( row, COL_CURSOR_Y, formatValue( coords.y, 0, 1 ) );
2966 break;
2967 }
2968 }
2969
2970 for( const auto& [name, trace] : plotTab->GetTraces() )
2971 {
2972 if( CURSOR* cursor = trace->GetCursor( 2 ) )
2973 {
2974 cursor2 = cursor;
2975 cursor2Name = getNameY( trace );
2976 cursor2Units = getUnitsY( trace );
2977
2978 wxRealPoint coords = cursor->GetCoords();
2979 int row = m_cursorsGrid->GetNumberRows();
2980
2981 m_cursorFormatsDyn[1][0].UpdateUnits( plotTab->GetUnitsX() );
2982 m_cursorFormatsDyn[1][1].UpdateUnits( cursor2Units );
2983
2984 m_cursorsGrid->AppendRows( 1 );
2985 m_cursorsGrid->SetCellValue( row, COL_CURSOR_NAME, wxS( "2" ) );
2986 m_cursorsGrid->SetCellValue( row, COL_CURSOR_SIGNAL, cursor->GetName() );
2987 m_cursorsGrid->SetCellValue( row, COL_CURSOR_X, formatValue( coords.x, 1, 0 ) );
2988 m_cursorsGrid->SetCellValue( row, COL_CURSOR_Y, formatValue( coords.y, 1, 1 ) );
2989 break;
2990 }
2991 }
2992
2993 if( cursor1 && cursor2 && cursor1Units == cursor2Units )
2994 {
2995 wxRealPoint coords = cursor2->GetCoords() - cursor1->GetCoords();
2996 wxString signal;
2997
2998 m_cursorFormatsDyn[2][0].UpdateUnits( plotTab->GetUnitsX() );
2999 m_cursorFormatsDyn[2][1].UpdateUnits( cursor1Units );
3000
3001 if( cursor1->GetName() == cursor2->GetName() )
3002 signal = wxString::Format( wxS( "%s[2 - 1]" ), cursor2->GetName() );
3003 else
3004 signal = wxString::Format( wxS( "%s - %s" ), cursor2->GetName(), cursor1->GetName() );
3005
3006 m_cursorsGrid->AppendRows( 1 );
3007 m_cursorsGrid->SetCellValue( 2, COL_CURSOR_NAME, _( "Diff" ) );
3008 m_cursorsGrid->SetCellValue( 2, COL_CURSOR_SIGNAL, signal );
3009 m_cursorsGrid->SetCellValue( 2, COL_CURSOR_X, formatValue( coords.x, 2, 0 ) );
3010 m_cursorsGrid->SetCellValue( 2, COL_CURSOR_Y, formatValue( coords.y, 2, 1 ) );
3011 }
3012 // Set up the labels
3013 m_cursorsGrid->SetColLabelValue( COL_CURSOR_X, plotTab->GetLabelX() );
3014
3015 wxString valColName = _( "Value" );
3016
3017 if( !cursor1Name.IsEmpty() )
3018 {
3019 if( cursor2Name.IsEmpty() || cursor1Name == cursor2Name )
3020 valColName = cursor1Name;
3021 }
3022 else if( !cursor2Name.IsEmpty() )
3023 {
3024 valColName = cursor2Name;
3025 }
3026
3027 m_cursorsGrid->SetColLabelValue( COL_CURSOR_Y, valColName );
3028
3029 if( m_customCursorsCnt > 3 ) // 2 for the default hardocded cursors plus the initial + 1
3030 {
3031 for( int i = 3; i < m_customCursorsCnt; i++ )
3032 {
3033 for( const auto& [name, trace] : plotTab->GetTraces() )
3034 {
3035 if( CURSOR* cursor = trace->GetCursor( i ) )
3036 {
3037 CURSOR* curs = cursor;
3038 wxString cursName = getNameY( trace );
3039 wxString cursUnits = getUnitsY( trace );
3040
3041 wxRealPoint coords = cursor->GetCoords();
3042 int row = m_cursorsGrid->GetNumberRows();
3043
3044 m_cursorFormatsDyn[i][0].UpdateUnits( plotTab->GetUnitsX() );
3045 m_cursorFormatsDyn[i][1].UpdateUnits( cursUnits );
3046
3047 m_cursorsGrid->AppendRows( 1 );
3048 m_cursorsGrid->SetCellValue( row, COL_CURSOR_NAME, wxS( "" ) + wxString( "" ) << i );
3049 m_cursorsGrid->SetCellValue( row, COL_CURSOR_SIGNAL, curs->GetName() );
3050 m_cursorsGrid->SetCellValue( row, COL_CURSOR_X, formatValue( coords.x, i, 0 ) );
3051 m_cursorsGrid->SetCellValue( row, COL_CURSOR_Y, formatValue( coords.y, i, 1 ) );
3052
3053 // Set up the labels
3054 m_cursorsGrid->SetColLabelValue( COL_CURSOR_X, plotTab->GetLabelX() );
3055
3056 valColName = _( "Value" );
3057
3058 if( !cursName.IsEmpty()
3059 && ( m_cursorsGrid->GetColLabelValue( COL_CURSOR_Y ) == cursName ) )
3060 {
3061 valColName = cursName;
3062 }
3063 m_cursorsGrid->SetColLabelValue( COL_CURSOR_Y, valColName );
3064 break;
3065 }
3066 }
3067 }
3068 }
3069}
3070
3071
3072void SIMULATOR_FRAME_UI::onPlotCursorUpdate( wxCommandEvent& aEvent )
3073{
3075 OnModify();
3076}
3077
3078
3080{
3081 if( SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( GetCurrentSimTab() ) )
3082 plotTab->ResetScales( true );
3083
3084 m_simConsole->Clear();
3085
3086 // Do not export netlist, it is already stored in the simulator
3087 applyTuners();
3088
3089 m_refreshTimer.Start( REFRESH_INTERVAL, wxTIMER_ONE_SHOT );
3090}
3091
3092
3093void SIMULATOR_FRAME_UI::OnSimReport( const wxString& aMsg )
3094{
3095 m_simConsole->AppendText( aMsg + "\n" );
3096 m_simConsole->SetInsertionPointEnd();
3097}
3098
3099
3100std::vector<wxString> SIMULATOR_FRAME_UI::SimPlotVectors() const
3101{
3102 std::vector<wxString> signals;
3103
3104 for( const std::string& vec : simulator()->AllVectors() )
3105 signals.emplace_back( vec );
3106
3107 return signals;
3108}
3109
3110
3111std::vector<wxString> SIMULATOR_FRAME_UI::Signals() const
3112{
3113 std::vector<wxString> signals;
3114
3115 for( const wxString& signal : m_signals )
3116 signals.emplace_back( signal );
3117
3118 for( const auto& [ id, signal ] : m_userDefinedSignals )
3119 signals.emplace_back( signal );
3120
3121 sortSignals( signals );
3122
3123 return signals;
3124}
3125
3126
3128{
3129 if( aFinal )
3130 m_refreshTimer.Stop();
3131
3132 SIM_TAB* simTab = GetCurrentSimTab();
3133
3134 if( !simTab )
3135 return;
3136
3137 SIM_TYPE simType = simTab->GetSimType();
3138 wxString msg;
3139
3140 if( aFinal )
3141 {
3144 }
3145
3146 // If there are any signals plotted, update them
3147 if( SIM_TAB::IsPlottable( simType ) )
3148 {
3149 simTab->SetSpicePlotName( simulator()->CurrentPlotName() );
3150
3151 if( simType == ST_NOISE && aFinal )
3152 {
3153 m_simConsole->AppendText( _( "\n\nSimulation results:\n\n" ) );
3154 m_simConsole->SetInsertionPointEnd();
3155
3156 // The simulator will create noise1 & noise2 on the first run, noise3 and noise4
3157 // on the second, etc. The first plot for each run contains the spectral density
3158 // noise vectors and second contains the integrated noise.
3159 long number;
3160 simulator()->CurrentPlotName().Mid( 5 ).ToLong( &number );
3161
3162 for( const std::string& vec : simulator()->AllVectors() )
3163 {
3164 std::vector<double> val_list = simulator()->GetRealVector( vec, 1 );
3165 wxString value = SPICE_VALUE( val_list[ 0 ] ).ToSpiceString();
3166
3167 msg.Printf( wxS( "%s: %sV\n" ), vec, value );
3168
3169 m_simConsole->AppendText( msg );
3170 m_simConsole->SetInsertionPointEnd();
3171 }
3172
3173 simulator()->Command( fmt::format( "setplot noise{}", number - 1 ) );
3174 simTab->SetSpicePlotName( simulator()->CurrentPlotName() );
3175 }
3176
3177 SIM_PLOT_TAB* plotTab = dynamic_cast<SIM_PLOT_TAB*>( simTab );
3178 wxCHECK_RET( plotTab, wxT( "not a SIM_PLOT_TAB" ) );
3179
3180 struct TRACE_INFO
3181 {
3182 wxString Vector;
3183 int TraceType;
3184 bool ClearData;
3185 };
3186
3187 std::map<TRACE*, TRACE_INFO> traceMap;
3188
3189 for( const auto& [ name, trace ] : plotTab->GetTraces() )
3190 traceMap[ trace ] = { wxEmptyString, SPT_UNKNOWN, false };
3191
3192 // NB: m_signals are already broken out into gain/phase, but m_userDefinedSignals are
3193 // as the user typed them
3194
3195 for( const wxString& signal : m_signals )
3196 {
3197 int traceType = SPT_UNKNOWN;
3198 wxString vectorName = vectorNameFromSignalName( plotTab, signal, &traceType );
3199
3200 if( TRACE* trace = plotTab->GetTrace( vectorName, traceType ) )
3201 traceMap[ trace ] = { vectorName, traceType, false };
3202 }
3203
3204 for( const auto& [ id, signal ] : m_userDefinedSignals )
3205 {
3206 int traceType = SPT_UNKNOWN;
3207 wxString vectorName = vectorNameFromSignalName( plotTab, signal, &traceType );
3208
3209 if( simType == ST_AC )
3210 {
3211 int baseType = traceType &= ~( SPT_AC_GAIN | SPT_AC_PHASE );
3212
3213 for( int subType : { baseType | SPT_AC_GAIN, baseType | SPT_AC_PHASE } )
3214 {
3215 if( TRACE* trace = plotTab->GetTrace( vectorName, subType ) )
3216 traceMap[ trace ] = { vectorName, subType, !aFinal };
3217 }
3218 }
3219 else if( simType == ST_SP )
3220 {
3221 int baseType = traceType &= ~( SPT_SP_AMP | SPT_AC_PHASE );
3222
3223 for( int subType : { baseType | SPT_SP_AMP, baseType | SPT_AC_PHASE } )
3224 {
3225 if( TRACE* trace = plotTab->GetTrace( vectorName, subType ) )
3226 traceMap[trace] = { vectorName, subType, !aFinal };
3227 }
3228 }
3229 else
3230 {
3231 if( TRACE* trace = plotTab->GetTrace( vectorName, traceType ) )
3232 traceMap[ trace ] = { vectorName, traceType, !aFinal };
3233 }
3234 }
3235
3236 // Two passes so that DC-sweep sub-traces get deleted and re-created:
3237
3238 for( const auto& [ trace, traceInfo ] : traceMap )
3239 {
3240 if( traceInfo.Vector.IsEmpty() )
3241 plotTab->DeleteTrace( trace );
3242 }
3243
3244 for( const auto& [ trace, info ] : traceMap )
3245 {
3246 std::vector<double> data_x;
3247
3248 if( !info.Vector.IsEmpty() )
3249 updateTrace( info.Vector, info.TraceType, plotTab, &data_x, info.ClearData );
3250 }
3251
3252 plotTab->GetPlotWin()->UpdateAll();
3253
3254 if( aFinal )
3255 {
3256 for( int row = 0; row < m_measurementsGrid->GetNumberRows(); ++row )
3257 UpdateMeasurement( row );
3258
3259 plotTab->ResetScales( true );
3260 }
3261
3262 plotTab->GetPlotWin()->Fit();
3263
3265 }
3266 else if( simType == ST_OP && aFinal )
3267 {
3268 m_simConsole->AppendText( _( "\n\nSimulation results:\n\n" ) );
3269 m_simConsole->SetInsertionPointEnd();
3270
3271 for( const std::string& vec : simulator()->AllVectors() )
3272 {
3273 std::vector<double> val_list = simulator()->GetRealVector( vec, 1 );
3274
3275 if( val_list.empty() )
3276 continue;
3277
3278 wxString value = SPICE_VALUE( val_list[ 0 ] ).ToSpiceString();
3279 wxString signal;
3280 SIM_TRACE_TYPE type = circuitModel()->VectorToSignal( vec, signal );
3281
3282 const size_t tab = 25; //characters
3283 size_t padding = ( signal.length() < tab ) ? ( tab - signal.length() ) : 1;
3284
3285 switch( type )
3286 {
3287 case SPT_VOLTAGE: value.Append( wxS( "V" ) ); break;
3288 case SPT_CURRENT: value.Append( wxS( "A" ) ); break;
3289 case SPT_POWER: value.Append( wxS( "W" ) ); break;
3290 default: value.Append( wxS( "?" ) ); break;
3291 }
3292
3293 msg.Printf( wxT( "%s%s\n" ),
3294 ( signal + wxT( ":" ) ).Pad( padding, wxUniChar( ' ' ) ),
3295 value );
3296
3297 m_simConsole->AppendText( msg );
3298 m_simConsole->SetInsertionPointEnd();
3299
3300 if( type == SPT_VOLTAGE || type == SPT_CURRENT || type == SPT_POWER )
3301 signal = signal.SubString( 2, signal.Length() - 2 );
3302
3303 if( type == SPT_POWER )
3304 signal += wxS( ":power" );
3305
3306 m_schematicFrame->Schematic().SetOperatingPoint( signal, val_list.at( 0 ) );
3307 }
3308 }
3309 else if( simType == ST_PZ && aFinal )
3310 {
3311 m_simConsole->AppendText( _( "\n\nSimulation results:\n\n" ) );
3312 m_simConsole->SetInsertionPointEnd();
3313 simulator()->Command( "print all" );
3314 }
3315}
3316
3317
3319{
3321}
int color
Definition: DXF_plotter.cpp:60
const char * name
Definition: DXF_plotter.cpp:59
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
void showPopupMenu(wxMenu &menu, wxGridEvent &aEvent) override
void doPopupSelection(wxCommandEvent &event) override
SIMULATOR_FRAME_UI * m_parent
CURSORS_GRID_TRICKS(SIMULATOR_FRAME_UI *aParent, WX_GRID *aGrid)
The SIMULATOR_FRAME holds the main user-interface for running simulations.
Definition: sim_plot_tab.h:64
const wxRealPoint & GetCoords() const
Definition: sim_plot_tab.h:100
void SetCoordX(double aValue)
int ShowModal() override
bool Find(const wxString &aTerm, int &aMatchersTriggered, int &aPosition)
Look in all existing matchers, return the earliest match of any of the existing.
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
Add mouse and command handling (such as cut, copy, and paste) to a WX_GRID instance.
Definition: grid_tricks.h:61
virtual void doPopupSelection(wxCommandEvent &event)
virtual void showPopupMenu(wxMenu &menu, wxGridEvent &aEvent)
WX_GRID * m_grid
I don't own the grid, but he owns me.
Definition: grid_tricks.h:125
A color representation with 4 components: red, green, blue, alpha.
Definition: color4d.h:104
wxString ToCSSString() const
Definition: color4d.cpp:147
Definition: kiid.h:49
PROJECT & Prj() const
Return a reference to the PROJECT associated with this KIWAY.
Hold a translatable error message and may be used when throwing exceptions containing a translated er...
Definition: ki_exception.h:46
const wxString What() const
Definition: ki_exception.h:58
void doPopupSelection(wxCommandEvent &event) override
void showPopupMenu(wxMenu &menu, wxGridEvent &aEvent) override
SIMULATOR_FRAME_UI * m_parent
MEASUREMENTS_GRID_TRICKS(SIMULATOR_FRAME_UI *aParent, WX_GRID *aGrid)
static void ConvertToSpiceMarkup(wxString *aNetName)
Remove formatting wrappers and replace illegal spice net name characters with underscores.
const SPICE_ITEM * FindItem(const wxString &aRefName) const
Find and return the item corresponding to aRefName.
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
Holds all the data relating to one schematic.
Definition: schematic.h:69
void SetOperatingPoint(const wxString &aSignal, double aValue)
Set operating points from a .op simulation.
Definition: schematic.h:250
void ClearOperatingPoints()
Clear operating points from a .op simulation.
Definition: schematic.h:240
SCH_DRAW_PANEL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
EESCHEMA_SETTINGS * eeconfig() const
Schematic editor (Eeschema) main window.
void RefreshOperatingPointDisplay()
Refresh the display of any operating points.
void OnModify() override
Must be called after a schematic change in order to set the "modify" flag and update other data struc...
SCHEMATIC & Schematic() const
void UpdateItem(EDA_ITEM *aItem, bool isAddOrDelete=false, bool aUpdateRtree=false) override
Mark an item for refresh.
Base class for any item which can be embedded within the SCHEMATIC container class,...
Definition: sch_item.h:167
Handle access to a stack of flattened SCH_SHEET objects by way of a path for creating a flattened sch...
SCH_ITEM * GetItem(const KIID &aID) const
Fetch a SCH_ITEM by ID.
Schematic symbol object.
Definition: sch_symbol.h:75
void GetFields(std::vector< SCH_FIELD * > &aVector, bool aVisibleOnly) const override
Populate a std::vector with SCH_FIELDs, sorted in ordinal order.
Definition: sch_symbol.cpp:841
const wxString GetRef(const SCH_SHEET_PATH *aSheet, bool aIncludeUnit=false) const override
Definition: sch_symbol.cpp:611
void doPopupSelection(wxCommandEvent &event) override
void showPopupMenu(wxMenu &menu, wxGridEvent &aEvent) override
SIMULATOR_FRAME_UI * m_parent
SIGNALS_GRID_TRICKS(SIMULATOR_FRAME_UI *aParent, WX_GRID *aGrid)
Class SIMULATOR_FRAME_UI_BASE.
wxSplitterWindow * m_splitterLeftRight
wxSplitterWindow * m_splitterMeasurements
wxSplitterWindow * m_splitterCursors
wxSplitterWindow * m_splitterPlotAndConsole
wxSplitterWindow * m_splitterSignals
The SIMULATOR_FRAME_UI holds the main user-interface for running simulations.
SIM_TAB * NewSimTab(const wxString &aSimCommand)
Create a new simulation tab for a given simulation type.
void SetUserDefinedSignals(const std::map< int, wxString > &aSignals)
void updatePlotCursors()
Update the cursor values (in the grid) and graphics (in the plot window).
void OnSimRefresh(bool aFinal)
void onPlotClose(wxAuiNotebookEvent &event) override
void CustomCursorsInit()
Init handler for custom cursors.
void TogglePanel(wxPanel *aPanel, wxSplitterWindow *aSplitterWindow, int &aSashPosition)
A common toggler for the two main wxSplitterWindow s.
void onPlotChanged(wxAuiNotebookEvent &event) override
void rebuildSignalsGrid(wxString aFilter)
Rebuild the filtered list of signals in the signals grid.
std::vector< std::vector< SPICE_VALUE_FORMAT > > m_cursorFormatsDyn
void DoFourier(const wxString &aSignal, const wxString &aFundamental)
void rebuildMeasurementsGrid()
Rebuild the measurements grid for the current plot.
std::list< TUNER_SLIDER * > m_tuners
SPICE expressions need quoted versions of the netnames since KiCad allows '-' and '/' in netnames.
void rebuildSignalsList()
Rebuild the list of signals available from the netlist.
bool loadLegacyWorkbook(const wxString &aPath)
void UpdateMeasurement(int aRow)
Update a measurement in the measurements grid.
wxString getNoiseSource() const
std::vector< wxString > SimPlotVectors() const
void SetSubWindowsSashSize()
Adjust the sash dimension of splitter windows after reading the config settings must be called after ...
void applyUserDefinedSignals()
Apply user-defined signals to the SPICE session.
SIM_PREFERENCES m_preferences
void DeleteMeasurement(int aRow)
Delete a row from the measurements grid.
SCH_EDIT_FRAME * m_schematicFrame
wxString vectorNameFromSignalName(SIM_PLOT_TAB *aPlotTab, const wxString &aSignalName, int *aTraceType)
Get the simulator output vector name for a given signal name and type.
void updateSignalsGrid()
Update the values in the signals grid.
void onCursorsGridCellChanged(wxGridEvent &aEvent) override
void onPlotDragged(wxAuiNotebookEvent &event) override
std::vector< wxString > Signals() const
bool SaveWorkbook(const wxString &aPath)
Save plot, signal, cursor, measurement, etc.
std::vector< wxString > m_signals
SIM_TAB * GetCurrentSimTab() const
Return the currently opened plot panel (or NULL if there is none).
void SaveCursorToWorkbook(nlohmann::json &aTraceJs, TRACE *aTrace, int aCursorId)
bool LoadWorkbook(const wxString &aPath)
Load plot, signal, cursor, measurement, etc.
SPICE_VALUE_FORMAT GetMeasureFormat(int aRow) const
Get/Set the format of a value in the measurements grid.
std::map< int, wxString > m_userDefinedSignals
void UpdateTunerValue(const SCH_SHEET_PATH &aSheetPath, const KIID &aSymbol, const wxString &aRef, const wxString &aValue)
Safely update a field of the associated symbol without dereferencing the symbol.
std::vector< wxString > m_netnames
void AddTrace(const wxString &aName, SIM_TRACE_TYPE aType)
Add a new trace to the current plot.
void onPlotClosed(wxAuiNotebookEvent &event) override
void RemoveTuner(TUNER_SLIDER *aTuner)
Remove an existing tuner.
void SaveSettings(EESCHEMA_SETTINGS *aCfg)
std::shared_ptr< SPICE_CIRCUIT_MODEL > circuitModel() const
void OnFilterText(wxCommandEvent &aEvent) override
void onMeasurementsGridCellChanged(wxGridEvent &aEvent) override
void DeleteCursor()
Deletes last m_signalsGrid "Cursor n" column, removes vector's m_cursorFormatsDyn last entry,...
void CreateNewCursor()
Creates a column at the end of m_signalsGrid named "Cursor n" ( n = m_customCursorsCnt ),...
void applyTuners()
Apply component values specified using tuner sliders to the current netlist.
bool loadJsonWorkbook(const wxString &aPath)
void OnFilterMouseMoved(wxMouseEvent &aEvent) override
void AddMeasurement(const wxString &aCmd)
Add a measurement to the measurements grid.
void onPlotChanging(wxAuiNotebookEvent &event) override
std::shared_ptr< SPICE_SIMULATOR > simulator() const
void onPlotCursorUpdate(wxCommandEvent &aEvent)
void onSignalsGridCellChanged(wxGridEvent &aEvent) override
void SetCursorFormat(int aCursorId, int aValueCol, const SPICE_VALUE_FORMAT &aFormat)
void InitWorkbook()
Load the currently active workbook stored in the project settings.
SIM_TRACE_TYPE getXAxisType(SIM_TYPE aType) const
Return X axis for a given simulation type.
void signalsGridCursorUpdate(T t, U u, R r)
Updates m_signalsGrid cursor widget, column rendering and attributes.
void SetMeasureFormat(int aRow, const SPICE_VALUE_FORMAT &aFormat)
void OnSimReport(const wxString &aMsg)
void ApplyPreferences(const SIM_PREFERENCES &aPrefs)
Called when settings are changed via the common Preferences dialog.
const SPICE_CIRCUIT_MODEL * GetExporter() const
Return the netlist exporter object used for simulations.
SIMULATOR_FRAME * m_simulatorFrame
void AddTuner(const SCH_SHEET_PATH &aSheetPath, SCH_SYMBOL *aSymbol)
Add a tuner for a symbol.
SPICE_VALUE_FORMAT GetCursorFormat(int aCursorId, int aValueCol) const
Get/Set the number of significant digits and the range for formatting a cursor value.
void OnUpdateUI(wxUpdateUIEvent &event) override
void updateTrace(const wxString &aVectorName, int aTraceType, SIM_PLOT_TAB *aPlotTab, std::vector< double > *aDataX=nullptr, bool aClearData=false)
Update a trace in a particular SIM_PLOT_TAB.
SIMULATOR_FRAME_UI(SIMULATOR_FRAME *aSimulatorFrame, SCH_EDIT_FRAME *aSchematicFrame)
SPICE_VALUE_FORMAT m_cursorFormats[3][2]
void LoadSettings(EESCHEMA_SETTINGS *aCfg)
The SIMULATOR_FRAME holds the main user-interface for running simulations.
bool LoadSimulator(const wxString &aSimCommand, unsigned aSimOptions)
Check and load the current netlist into the simulator.
std::shared_ptr< SPICE_CIRCUIT_MODEL > GetCircuitModel() const
SIM_TYPE GetCurrentSimType() const
void OnModify() override
Must be called after a model change in order to set the "modify" flag and do other frame-specific pro...
bool SimFinished() const
int GetCurrentOptions() const
std::shared_ptr< SPICE_SIMULATOR > GetSimulator() const
SIM_MODEL & CreateModel(SIM_MODEL::TYPE aType, const std::vector< SCH_PIN * > &aPins, REPORTER &aReporter)
virtual const PARAM * GetTunerParam() const
Definition: sim_model.h:477
const SPICE_GENERATOR & SpiceGenerator() const
Definition: sim_model.h:428
void WriteFields(std::vector< SCH_FIELD > &aFields) const
Definition: sim_model.cpp:447
void SetParamValue(int aParamIndex, const std::string &aValue, SIM_VALUE::NOTATION aNotation=SIM_VALUE::NOTATION::SI)
Definition: sim_model.cpp:782
static void FillDefaultColorList(bool aWhiteBg)
Fills m_colorList by a default set of colors.
bool DeleteTrace(const wxString &aVectorName, int aTraceType)
wxString GetLabelY1() const
Definition: sim_plot_tab.h:208
mpWindow * GetPlotWin() const
Definition: sim_plot_tab.h:350
void ShowGrid(bool aEnable)
Definition: sim_plot_tab.h:268
wxString GetUnitsY2() const
void SetY2Scale(bool aLock, double aMin, double aMax)
TRACE * GetTrace(const wxString &aVecName, int aType) const
Definition: sim_plot_tab.h:261
wxString GetLabelX() const
Definition: sim_plot_tab.h:203
const std::map< wxString, TRACE * > & GetTraces() const
Definition: sim_plot_tab.h:256
wxString GetLabelY3() const
Definition: sim_plot_tab.h:218
void SetY1Scale(bool aLock, double aMin, double aMax)
void SetY3Scale(bool aLock, double aMin, double aMax)
std::vector< std::pair< wxString, wxString > > & Measurements()
Definition: sim_plot_tab.h:360
void UpdateTraceStyle(TRACE *trace)
Update plot colors.
void SetLegendPosition(const wxPoint &aPosition)
Definition: sim_plot_tab.h:309
void ResetScales(bool aIncludeX)
Update trace line style.
void UpdatePlotColors()
void ShowLegend(bool aEnable)
Definition: sim_plot_tab.h:293
wxString GetLabelY2() const
Definition: sim_plot_tab.h:213
void SetTraceData(TRACE *aTrace, std::vector< double > &aX, std::vector< double > &aY, int aSweepCount, size_t aSweepSize)
void EnableCursor(TRACE *aTrace, int aCursorId, const wxString &aSignalName)
wxString GetUnitsX() const
void EnsureThirdYAxisExists()
TRACE * GetOrAddTrace(const wxString &aVectorName, int aType)
void SetDottedSecondary(bool aEnable)
Draw secondary signal traces (current or phase) with dotted lines.
Definition: sim_plot_tab.h:319
void ApplyPreferences(const SIM_PREFERENCES &aPrefs) override
Definition: sim_plot_tab.h:198
wxString GetUnitsY1() const
void DisableCursor(TRACE *aTrace, int aCursorId)
Reset scale ranges to fit the current traces.
wxString GetUnitsY3() const
int GetSimOptions() const
Definition: sim_tab.h:55
SIM_TYPE GetSimType() const
Definition: sim_tab.cpp:75
const wxString & GetSimCommand() const
Definition: sim_tab.h:52
static bool IsPlottable(SIM_TYPE aSimType)
Definition: sim_tab.cpp:53
void SetSimOptions(int aOptions)
Definition: sim_tab.h:56
wxString GetLastSchTextSimCommand() const
Definition: sim_tab.h:58
void SetSpicePlotName(const wxString &aPlotName)
Definition: sim_tab.h:62
static std::string ToSpice(const std::string &aString)
Definition: sim_value.cpp:419
Special netlist exporter flavor that allows one to override simulation commands.
static SIM_TYPE CommandToSimType(const wxString &aCmd)
Return simulation type basing on a simulation command directive.
virtual std::string TunerCommand(const SPICE_ITEM &aItem, double aValue) const
static wxString TypeToName(SIM_TYPE aType, bool aShortName)
Return a string with simulation name based on enum.
Helper class to recognize Spice formatted values.
Definition: spice_value.h:56
wxString ToString() const
Return string value as when converting double to string (e.g.
wxString ToSpiceString() const
Return string value in Spice format (e.g.
double ToDouble() const
SUPPRESS_GRID_CELL_EVENTS(SIMULATOR_FRAME_UI *aFrame)
SIMULATOR_FRAME_UI * m_frame
void SetTraceColour(const wxColour &aColour)
Definition: sim_plot_tab.h:181
bool HasCursor(int aCursorId)
Definition: sim_plot_tab.h:173
CURSOR * GetCursor(int aCursorId)
Definition: sim_plot_tab.h:176
Custom widget to handle quick component values modification and simulation on the fly.
Definition: tuner_slider.h:44
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
A wrapper for reporting to a wxString object.
Definition: reporter.h:190
bool HasMessage() const override
Returns true if the reporter client is non-empty.
Definition: reporter.cpp:96
REPORTER & Report(const wxString &aText, SEVERITY aSeverity=RPT_SEVERITY_UNDEFINED) override
Report a string with a given severity.
Definition: reporter.cpp:74
const wxString & GetMessages() const
Definition: reporter.cpp:83
A KICAD version of wxTextEntryDialog which supports the various improvements/work-arounds from DIALOG...
wxString GetValue() const
const wxString & GetName() const
Get layer name.
Definition: mathplot.h:239
const wxPen & GetPen() const
Get pen set for this layer.
Definition: mathplot.h:254
Canvas for plotting mpLayer implementations.
Definition: mathplot.h:908
int GetMarginLeft() const
Definition: mathplot.h:1219
void SetMargins(int top, int right, int bottom, int left)
Set window margins, creating a blank area where some kinds of layers cannot draw.
Definition: mathplot.cpp:2412
int GetMarginTop() const
Definition: mathplot.h:1213
void UpdateAll()
Refresh display.
Definition: mathplot.cpp:2313
int GetMarginRight() const
Definition: mathplot.h:1215
int GetMarginBottom() const
Definition: mathplot.h:1217
void LockY(bool aLock)
Definition: mathplot.h:1263
bool AddLayer(mpLayer *layer, bool refreshDisplay=true)
Add a plot layer to the canvas.
Definition: mathplot.cpp:1990
void Fit() override
Set view to fit global bounding box of all plot layers and refresh display.
Definition: mathplot.cpp:1632
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.
#define _(s)
Abstract pattern-matching tool and implementations.
@ CTX_SIGNAL
@ GRIDTRICKS_ID_SELECT
Definition: grid_tricks.h:46
@ GRIDTRICKS_ID_COPY
Definition: grid_tricks.h:43
@ GRIDTRICKS_ID_DELETE
Definition: grid_tricks.h:44
@ GRIDTRICKS_FIRST_CLIENT_ID
Definition: grid_tricks.h:48
static const std::string WorkbookFileExtension
#define traceSettings
Definition: json_settings.h:52
KICOMMON_API wxFont GetStatusFont(wxWindow *aWindow)
Definition: ui_common.cpp:132
see class PGM_BASE
SIM_TRACE_TYPE
Definition: sim_types.h:50
@ SPT_TIME
Definition: sim_types.h:61
@ SPT_AC_PHASE
Definition: sim_types.h:54
@ SPT_SWEEP
Definition: sim_types.h:64
@ SPT_UNKNOWN
Definition: sim_types.h:67
@ SPT_AC_GAIN
Definition: sim_types.h:55
@ SPT_Y_AXIS_MASK
Definition: sim_types.h:58
@ SPT_SP_AMP
Definition: sim_types.h:57
@ SPT_VOLTAGE
Definition: sim_types.h:52
@ SPT_POWER
Definition: sim_types.h:56
@ SPT_CURRENT
Definition: sim_types.h:53
@ SPT_LIN_FREQUENCY
Definition: sim_types.h:62
SIM_TYPE
< Possible simulation types
Definition: sim_types.h:32
@ ST_SP
Definition: sim_types.h:43
@ ST_TRAN
Definition: sim_types.h:42
@ ST_UNKNOWN
Definition: sim_types.h:33
@ ST_NOISE
Definition: sim_types.h:37
@ ST_AC
Definition: sim_types.h:34
@ ST_DISTO
Definition: sim_types.h:36
@ ST_TF
Definition: sim_types.h:41
@ ST_SENS
Definition: sim_types.h:40
@ ST_DC
Definition: sim_types.h:35
@ ST_OP
Definition: sim_types.h:38
@ ST_FFT
Definition: sim_types.h:44
@ ST_PZ
Definition: sim_types.h:39
void sortSignals(std::vector< wxString > &signals)
wxString vectorNameFromSignalId(int aUserDefinedSignalId)
MEASUREMENTS_GIRD_COLUMNS
@ COL_MEASUREMENT_FORMAT
@ COL_MEASUREMENT_VALUE
@ COL_MEASUREMENT
CURSORS_GRID_COLUMNS
@ COL_CURSOR_NAME
@ COL_CURSOR_SIGNAL
@ COL_CURSOR_X
@ COL_CURSOR_Y
SIGNALS_GRID_COLUMNS
@ COL_SIGNAL_SHOW
@ COL_SIGNAL_NAME
@ COL_CURSOR_1
@ COL_SIGNAL_COLOR
@ COL_CURSOR_2
#define ID_SIM_REFRESH
SIM_TRACE_TYPE operator|(SIM_TRACE_TYPE aFirst, SIM_TRACE_TYPE aSecond)
#define REFRESH_INTERVAL
@ MYID_MEASURE_INTEGRAL
@ MYID_MEASURE_MAX_AT
@ MYID_MEASURE_AVG
@ MYID_MEASURE_MAX
@ MYID_FOURIER
@ MYID_FORMAT_VALUE
@ MYID_MEASURE_RMS
@ MYID_DELETE_MEASUREMENT
@ MYID_MEASURE_MIN
@ MYID_MEASURE_MIN_AT
@ MYID_MEASURE_PP
const int scale
std::vector< FAB_LAYER_COLOR > dummy
wxString UnescapeString(const wxString &aSource)
const INFO & info
Definition: sim_model.h:401
Contains preferences pertaining to the simulator.
SPICE_VALUE m_vincrement
const SIM_MODEL * model
A SPICE_VALUE_FORMAT holds precision and range info for formatting values.Helper class to handle Spic...
Definition: spice_value.h:43
wxString ToString() const
Definition: spice_value.cpp:58
void UpdateUnits(const wxString &aUnits)
Definition: spice_value.cpp:64
void FromString(const wxString &aString)
Definition: spice_value.cpp:41
VECTOR3I res
Definition of file extensions used in Kicad.