31#include <fmt/format.h>
32#include <wx/wfstream.h>
33#include <wx/stdstream.h>
35#include <wx/clipbrd.h>
60 int res =
static_cast<int>( aFirst ) |
static_cast<int>( aSecond);
121 void showPopupMenu( wxMenu& menu, wxGridEvent& aEvent )
override;
172 menu.AppendSeparator();
173 menu.Append(
MYID_FOURIER,
_(
"Perform Fourier Analysis..." ) );
176 menu.AppendSeparator();
179 menu.AppendSeparator();
182 m_grid->PopupMenu( &menu );
189 menu.AppendSeparator();
191 wxString msg =
m_grid->GetColLabelValue(
m_grid->GetNumberCols() - 1 );
193 menu.AppendSeparator();
196 m_grid->PopupMenu( &menu );
202 m_grid->PopupMenu( &menu );
209 std::vector<wxString> signals;
211 wxGridCellCoordsArray cells1 =
m_grid->GetSelectionBlockTopLeft();
212 wxGridCellCoordsArray cells2 =
m_grid->GetSelectionBlockBottomRight();
214 for(
size_t i = 0; i < cells1.Count(); i++ )
218 for(
int j = cells1[i].GetRow(); j < cells2[i].GetRow() + 1; j++ )
220 signals.push_back(
m_grid->GetCellValue( j, cells1[i].GetCol() ) );
225 wxGridCellCoordsArray cells3 =
m_grid->GetSelectedCells();
227 for(
size_t i = 0; i < cells3.Count(); i++ )
230 signals.push_back(
m_grid->GetCellValue( cells3[i].GetRow(), cells3[i].GetCol() ) );
233 if( signals.size() < 1 )
236 auto addMeasurement =
237 [
this](
const wxString& cmd, wxString signal )
239 if( signal.EndsWith(
_(
" (phase)" ) ) )
242 if( signal.EndsWith(
_(
" (gain)" ) ) || signal.EndsWith(
_(
" (amplitude)" ) ) )
244 signal = signal.Left( signal.length() - 7 );
246 if( signal.Upper().StartsWith( wxS(
"V(" ) ) )
247 signal = wxS(
"vdb" ) + signal.Mid( 1 );
255 for(
const wxString& signal : signals )
256 addMeasurement( wxS(
"MIN" ), signal );
260 for(
const wxString& signal : signals )
261 addMeasurement( wxS(
"MAX" ), signal );
265 for(
const wxString& signal : signals )
266 addMeasurement( wxS(
"AVG" ), signal );
270 for(
const wxString& signal : signals )
271 addMeasurement( wxS(
"RMS" ), signal );
275 for(
const wxString& signal : signals )
276 addMeasurement( wxS(
"PP" ), signal );
280 for(
const wxString& signal : signals )
281 addMeasurement( wxS(
"MIN_AT" ), signal );
285 for(
const wxString& signal : signals )
286 addMeasurement( wxS(
"MAX_AT" ), signal );
290 for(
const wxString& signal : signals )
291 addMeasurement( wxS(
"INTEG" ), signal );
296 wxString fundamental = wxT(
"1K" );
298 if( signals.size() == 1 )
299 title.Printf(
_(
"Fourier Analysis of %s" ), signals[0] );
301 title =
_(
"Fourier Analyses of Multiple Signals" );
311 for(
const wxString& signal : signals )
319 for(
const wxString& signal : signals )
327 if( wxTheClipboard->Open() )
329 wxTheClipboard->SetData(
new wxTextDataObject( txt ) );
330 wxTheClipboard->Flush();
331 wxTheClipboard->Close();
356 void showPopupMenu( wxMenu& menu, wxGridEvent& aEvent )
override;
376 menu.AppendSeparator();
386 [
this](
int row ) -> wxString
390 if( signal.EndsWith(
"[2 - 1]" ) )
391 signal = signal.Left( signal.length() - 7 );
402 if( formatDialog.
ShowModal() == wxID_OK )
404 for(
int row = 0; row <
m_grid->GetNumberRows(); ++row )
406 if( getSignalName( row ) == getSignalName(
m_menuRow ) )
429 void showPopupMenu( wxMenu& menu, wxGridEvent& aEvent )
override;
455 menu.AppendSeparator();
468 if( formatDialog.
ShowModal() == wxID_OK )
477 std::vector<int> measurements;
479 wxGridCellCoordsArray cells1 =
m_grid->GetSelectionBlockTopLeft();
480 wxGridCellCoordsArray cells2 =
m_grid->GetSelectionBlockBottomRight();
482 for(
size_t i = 0; i < cells1.Count(); i++ )
486 for(
int j = cells1[i].GetRow(); j < cells2[i].GetRow() + 1; j++ )
487 measurements.push_back( j );
491 wxGridCellCoordsArray cells3 =
m_grid->GetSelectedCells();
493 for(
size_t i = 0; i < cells3.Count(); i++ )
496 measurements.push_back( cells3[i].GetRow() );
499 if( measurements.size() < 1 )
504 sort( measurements.begin(), measurements.end(), std::greater<>() );
506 for(
int row : measurements )
539#define ID_SIM_REFRESH 10207
540#define REFRESH_INTERVAL 50
546 m_SuppressGridEvents( 0 ),
547 m_simulatorFrame( aSimulatorFrame ),
548 m_schematicFrame( aSchematicFrame ),
566 wxGridCellAttr* attr =
new wxGridCellAttr;
570 attr =
new wxGridCellAttr;
574 attr =
new wxGridCellAttr;
578 attr =
new wxGridCellAttr;
584 attr =
new wxGridCellAttr;
594 [&]( wxTimerEvent& aEvent )
603#ifndef wxHAS_NATIVE_TABART
623 m_cursorFormat[0] = { 3, wxS(
"~s" ) };
624 m_cursorFormat[1] = { 3, wxS(
"~V" ) };
635 for(
size_t index2 = 0; index2 < std::size(
m_cursorFormats[0] ); index2++ )
651 maxCursor.Replace(
_(
"Cursor " ),
"" );
653 int tmpMax = wxAtoi( maxCursor );
655 if( nameMax < tmpMax )
665 std::vector<SPICE_VALUE_FORMAT> tmp;
677 wxGridCellAttr* attr =
new wxGridCellAttr;
697 for(
int i = 0; i < rows; i++ )
699 if(
m_signalsGrid->GetCellValue( i, col - 1 ) == wxS(
"1" ) )
702 wxGridEvent aDummy( wxID_ANY, wxEVT_GRID_CELL_CHANGED,
m_signalsGrid, i, col - 1 );
722 for(
int ii = 0; ii < static_cast<int>(
m_plotNotebook->GetPageCount() ); ++ii )
726 simTab->OnLanguageChanged();
728 wxString pageTitle(
simulator()->TypeToName( simTab->GetSimType(),
true ) );
729 pageTitle.Prepend( wxString::Format(
_(
"Analysis %u - " ), ii+1 ) );
750 tuner->ShowChangedLanguage();
788 for( std::size_t i = 0; i < pageCount; ++i )
791 auto simTab =
dynamic_cast<SIM_TAB*
>( page );
792 wxASSERT( simTab !=
nullptr );
793 simTab->ApplyPreferences( aPrefs );
800 if( !
simulator()->Settings()->GetWorkbookFilename().IsEmpty() )
802 wxFileName filename =
simulator()->Settings()->GetWorkbookFilename();
806 simulator()->Settings()->SetWorkbookFilename(
"" );
810 wxString schTextSimCommand =
circuitModel()->GetSchTextSimCommand();
812 if( !schTextSimCommand.IsEmpty() )
845 std::sort( signals.begin(), signals.end(),
846 [](
const wxString& lhs,
const wxString& rhs )
849 if( lhs.Upper().StartsWith(
'V' ) && !rhs.Upper().StartsWith(
'V' ) )
851 else if( !lhs.Upper().StartsWith(
'V' ) && rhs.Upper().StartsWith(
'V' ) )
854 return StrNumCmp( lhs, rhs, true ) < 0;
871 std::vector<wxString> signals;
875 wxStringTokenizer tokenizer( plotPanel->
GetSimCommand(), wxT(
" \t\r\n" ), wxTOKEN_STRTOK );
877 while( tokenizer.HasMoreTokens() && tokenizer.GetNextToken().Lower() != wxT(
"fft" ) )
880 while( tokenizer.HasMoreTokens() )
881 signals.emplace_back( tokenizer.GetNextToken() );
888 for(
const wxString& signal :
m_signals )
889 signals.push_back( signal );
893 if( simType ==
ST_AC )
895 signals.push_back( signal +
_(
" (gain)" ) );
896 signals.push_back( signal +
_(
" (phase)" ) );
898 else if( simType ==
ST_SP )
900 signals.push_back( signal +
_(
" (amplitude)" ) );
901 signals.push_back( signal +
_(
" (phase)" ) );
905 signals.push_back( signal );
912 if( aFilter.IsEmpty() )
913 aFilter = wxS(
"*" );
918 for(
const wxString& signal : signals )
920 if( matcher.
Find( signal.Upper() ) )
929 wxGridCellAttr* attr =
new wxGridCellAttr;
930 attr->SetRenderer(
new wxGridCellBoolRenderer() );
932 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
937 attr =
new wxGridCellAttr;
942 attr =
new wxGridCellAttr;
946 attr =
new wxGridCellAttr;
954 attr =
new wxGridCellAttr;
964 attr =
new wxGridCellAttr;
967 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
972 attr =
new wxGridCellAttr;
973 attr->SetRenderer(
new wxGridCellBoolRenderer() );
975 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
979 attr =
new wxGridCellAttr;
980 attr->SetRenderer(
new wxGridCellBoolRenderer() );
982 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
990 attr =
new wxGridCellAttr;
991 attr->SetRenderer(
new wxGridCellBoolRenderer() );
993 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
1012 wxString unconnected = wxString( wxS(
"unconnected-(" ) );
1017 unconnected.Replace(
'(',
'_' );
1020 [&](
const wxString& aSignalName )
1022 if( simType ==
ST_AC )
1024 m_signals.push_back( aSignalName +
_(
" (gain)" ) );
1025 m_signals.push_back( aSignalName +
_(
" (phase)" ) );
1027 else if( simType ==
ST_SP )
1029 m_signals.push_back( aSignalName +
_(
" (amplitude)" ) );
1030 m_signals.push_back( aSignalName +
_(
" (phase)" ) );
1047 if( netname ==
"GND" || netname ==
"0" || netname.StartsWith( unconnected ) )
1051 addSignal( wxString::Format( wxS(
"V(%s)" ), netname ) );
1061 for(
const std::string&
name : item.model->SpiceGenerator().CurrentNames( item ) )
1071 if( item.model->GetPinCount() >= 2 )
1073 wxString
name = item.model->SpiceGenerator().ItemName( item );
1074 addSignal( wxString::Format( wxS(
"P(%s)" ),
name ) );
1081 addSignal( wxS(
"inoise_spectrum" ) );
1082 addSignal( wxS(
"onoise_spectrum" ) );
1085 if( simType ==
ST_SP )
1087 std::vector<std::string> portnums;
1091 wxString
name = item.model->SpiceGenerator().ItemName( item );
1094 if( !
name.StartsWith(
"V" ) )
1097 std::string portnum =
"";
1099 if(
const SIM_MODEL::PARAM* portnum_param = item.model->FindParam(
"portnum" ) )
1103 portnums.push_back( portnum );
1106 for(
const std::string& portnum1 : portnums )
1108 for(
const std::string& portnum2 : portnums )
1110 addSignal( wxString::Format( wxS(
"S_%s_%s" ), portnum1, portnum2 ) );
1116 for(
const wxString& directive :
circuitModel()->GetDirectives() )
1118 wxStringTokenizer directivesTokenizer( directive, wxT(
"\r\n" ), wxTOKEN_STRTOK );
1120 while( directivesTokenizer.HasMoreTokens() )
1122 wxString line = directivesTokenizer.GetNextToken().Upper();
1123 wxString directiveParams;
1125 if( line.StartsWith( wxS(
".SAVE" ), &directiveParams )
1126 || line.StartsWith( wxS(
".PROBE" ), &directiveParams ) )
1128 wxStringTokenizer paramsTokenizer( directiveParams, wxT(
" \t" ), wxTOKEN_STRTOK );
1130 while( paramsTokenizer.HasMoreTokens() )
1131 addSignal( paramsTokenizer.GetNextToken() );
1154 wxString pageTitle(
simulator()->TypeToName( simType,
true ) );
1155 pageTitle.Prepend( wxString::Format(
_(
"Analysis %u - " ),
static_cast<unsigned int>( ++
m_plotNumber ) ) );
1171 wxPoint pos = aEvent.GetPosition();
1172 wxRect ctrlRect =
m_filter->GetScreenRect();
1173 int buttonWidth = ctrlRect.GetHeight();
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 );
1180 SetCursor( wxCURSOR_IBEAM );
1186 return wxString::Format( wxS(
"user%d" ), aUserDefinedSignalId );
1195 const wxString& aSignalName,
1198 std::map<wxString, int> suffixes;
1214 wxUniChar firstChar = aSignalName.Upper()[0];
1216 if( firstChar ==
'V' )
1218 else if( firstChar ==
'I' )
1220 else if( firstChar ==
'P' )
1226 wxString
name = aSignalName;
1228 for(
const auto& [ candidate, type ] : suffixes )
1230 if(
name.EndsWith( candidate ) )
1232 name =
name.Left(
name.Length() - candidate.Length() );
1235 *aTraceType |= type;
1243 if(
name == signal )
1261 int row = aEvent.GetRow();
1262 int col = aEvent.GetCol();
1270 if(
text == wxS(
"1" ) )
1306 TRACE* activeTrace =
nullptr;
1308 if(
text == wxS(
"1" ) )
1312 activeTrace = plotTab->
GetTrace( vectorName, traceType );
1323 if( trace != activeTrace && trace->
HasCursor(
id ) )
1346 int row = aEvent.GetRow();
1347 int col = aEvent.GetCol();
1355 CURSOR* cursor1 =
nullptr;
1356 CURSOR* cursor2 =
nullptr;
1358 std::vector<CURSOR*> cursorsVec;
1380 cursorsVec.emplace_back(
cursor );
1382 if( cursorName == ( wxString(
"" ) << i ) &&
cursor )
1383 cursor->SetCoordX( value );
1390 if( cursorName == wxS(
"1" ) && cursor1 )
1392 else if( cursorName == wxS(
"2" ) && cursor2 )
1394 else if( cursorName ==
_(
"Diff" ) && cursor1 && cursor2 )
1402 wxFAIL_MSG( wxT(
"All other columns are supposed to be read-only!" ) );
1435 int row = aEvent.GetRow();
1436 int col = aEvent.GetCol();
1447 wxFAIL_MSG( wxT(
"All other columns are supposed to be read-only!" ) );
1455 for( row = rowCount - 1; row >= 0; row-- )
1465 int killRows = emptyRows - 1;
1468 else if( emptyRows == 0 )
1479 if( plotTab->GetLegendPosition() != plotTab->m_LastLegendPosition )
1481 plotTab->m_LastLegendPosition = plotTab->GetLegendPosition();
1504 static wxRegEx measureParamsRegEx( wxT(
"^"
1508 "([a-zA-Z]*)\\(([^\\)]+)\\)" ) );
1517 if(
text.IsEmpty() )
1524 wxString resultName = wxString::Format( wxS(
"meas_result_%u" ), aRow );
1525 wxString result = wxS(
"?" );
1527 if( measureParamsRegEx.Matches(
text ) )
1529 wxString func = measureParamsRegEx.GetMatch(
text, 1 ).Upper();
1530 wxString signalType = measureParamsRegEx.GetMatch(
text, 2 ).Upper();
1531 wxString deviceName = measureParamsRegEx.GetMatch(
text, 3 );
1535 if( signalType.EndsWith( wxS(
"DB" ) ) )
1537 units = wxS(
"dB" );
1539 else if( signalType.StartsWith(
'I' ) )
1543 else if( signalType.StartsWith(
'P' ) )
1547 text = func +
" " + deviceName +
":power";
1554 if( func.EndsWith( wxS(
"_AT" ) ) )
1557 units = wxS(
"Hz" );
1561 else if( func.StartsWith( wxS(
"INTEG" ) ) )
1566 if ( signalType.StartsWith(
'P' ) )
1569 units += wxS(
".s" );
1579 units += wxS(
"·Hz" );
1587 units += wxS(
"·?" );
1600 wxString cmd = wxString::Format( wxS(
"meas %s %s %s" ), simType, resultName,
text );
1601 simulator()->Command(
"echo " + cmd.ToStdString() );
1602 simulator()->Command( cmd.ToStdString() );
1604 std::vector<double> resultVec =
simulator()->GetGainVector( resultName.ToStdString() );
1606 if( resultVec.size() > 0 )
1621 "the value of a passive R, L, C model or voltage or "
1622 "current source." ) );
1626 wxString ref = aSymbol->
GetRef( &aSheetPath );
1631 if( tuner->GetSymbolRef() == ref )
1654 const wxString& aRef,
const wxString& aValue )
1662 + wxString::Format(
_(
"%s not found" ), aRef ) );
1675 + wxString::Format(
_(
"%s is not tunable" ), aRef ) );
1741 wxString cmd = wxString::Format( wxS(
"fourier %s %s" ),
1745 simulator()->Command( cmd.ToStdString() );
1759 m_simConsole->AppendText(
_(
"Error: no current simulation.\n" ) );
1768 m_simConsole->AppendText(
_(
"Error: simulation type not defined.\n" ) );
1774 m_simConsole->AppendText(
_(
"Error: simulation type doesn't support plotting.\n" ) );
1781 if( simType ==
ST_AC )
1786 else if( simType ==
ST_SP )
1796 plotTab->GetPlotWin()->UpdateAll();
1818 if( aNewSignals.count(
id ) == 0 )
1823 plotTab->
DeleteTrace( vectorName, traceType | subType );
1828 plotTab->
DeleteTrace( vectorName, traceType | subType );
1841 if(
TRACE* trace = plotTab->
GetTrace( vectorName, traceType | subType ) )
1842 trace->SetName( aNewSignals.at(
id ) );
1849 if(
TRACE* trace = plotTab->
GetTrace( vectorName, traceType | subType ) )
1850 trace->SetName( aNewSignals.at(
id ) );
1855 if(
TRACE* trace = plotTab->
GetTrace( vectorName, traceType ) )
1856 trace->SetName( aNewSignals.at(
id ) );
1890 wxString simVectorName = aVectorName;
1893 simVectorName = simVectorName.AfterFirst(
'(' ).BeforeLast(
')' ) + wxS(
":power" );
1898 simulator()->Command( wxString::Format( wxT(
"print %s" ), aVectorName ).ToStdString() );
1903 std::vector<double> data_x;
1904 std::vector<double> data_y;
1906 if( !aDataX || aClearData )
1910 if( aDataX->empty() && !aClearData )
1912 wxString xAxisName(
simulator()->GetXAxis( simType ) );
1914 if( xAxisName.IsEmpty() )
1917 *aDataX =
simulator()->GetGainVector( (
const char*) xAxisName.c_str() );
1920 unsigned int size = aDataX->size();
1926 data_y =
simulator()->GetGainVector( (
const char*) simVectorName.c_str(), size );
1928 data_y =
simulator()->GetPhaseVector( (
const char*) simVectorName.c_str(), size );
1930 wxFAIL_MSG( wxT(
"Plot type missing AC_PHASE or AC_MAG bit" ) );
1935 data_y =
simulator()->GetGainVector( (
const char*) simVectorName.c_str(), size );
1937 data_y =
simulator()->GetPhaseVector( (
const char*) simVectorName.c_str(), size );
1939 wxFAIL_MSG( wxT(
"Plot type missing AC_PHASE or SPT_SP_AMP bit" ) );
1944 data_y =
simulator()->GetGainVector( (
const char*) simVectorName.c_str(), -1 );
1950 data_y =
simulator()->GetGainVector( (
const char*) simVectorName.c_str(), size );
1954 wxFAIL_MSG( wxT(
"Unhandled plot type" ) );
1959 size_t sweepSize = std::numeric_limits<size_t>::max();
1961 if( simType ==
ST_DC
1968 sweepSize = aDataX->size() / sweepCount;
1973 if( data_y.size() >= size )
1974 aPlotTab->
SetTraceData( trace, *aDataX, data_y, sweepCount, sweepSize );
1981template void SIMULATOR_FRAME_UI::signalsGridCursorUpdate<SIGNALS_GRID_COLUMNS, int, int>(
1984template <
typename T,
typename U,
typename R>
1993 wxGridCellAttrPtr attr =
m_signalsGrid->GetOrCreateCellAttrPtr( r,
static_cast<int>( t ) );
1995 if(
TRACE* trace = plotTab ? plotTab->
GetTrace( vectorName, traceType ) :
nullptr )
1997 attr->SetReadOnly();
1999 if( t >= SIGNALS_GRID_COLUMNS::COL_SIGNAL_SHOW )
2001 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
2004 if constexpr ( std::is_enum<T>::value )
2006 if( t == SIGNALS_GRID_COLUMNS::COL_SIGNAL_SHOW )
2008 m_signalsGrid->SetCellValue( r,
static_cast<int>( t ), wxS(
"1" ) );
2010 else if( t == SIGNALS_GRID_COLUMNS::COL_SIGNAL_COLOR )
2012 if( !attr->HasRenderer() )
2015 if( !attr->HasEditor() )
2018 attr->SetAlignment( wxALIGN_CENTER, wxALIGN_CENTER );
2019 attr->SetReadOnly(
false );
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 )
2028 if( !attr->HasRenderer() )
2029 attr->SetRenderer(
new wxGridCellBoolRenderer() );
2031 if( u > 0 && trace->HasCursor( u ) )
2032 m_signalsGrid->SetCellValue( r,
static_cast<int>( t ), wxS(
"1" ) );
2034 m_signalsGrid->SetCellValue( r,
static_cast<int>( t ), wxEmptyString );
2040 if constexpr ( std::is_enum<T>::value )
2042 if( t == SIGNALS_GRID_COLUMNS::COL_SIGNAL_SHOW )
2044 m_signalsGrid->SetCellValue( r,
static_cast<int>( t ), wxEmptyString );
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 )
2051 attr->SetEditor(
nullptr );
2052 attr->SetRenderer(
nullptr );
2053 attr->SetReadOnly();
2054 m_signalsGrid->SetCellValue( r,
static_cast<int>( t ), wxEmptyString );
2063 for(
int row = 0; row <
m_signalsGrid->GetNumberRows(); ++row )
2088 auto quoteNetNames = [&]( wxString aExpression ) -> wxString
2090 std::vector<bool> mask( aExpression.length(),
false );
2094 size_t pos = aExpression.find( netname );
2096 while( pos != wxString::npos )
2098 for(
size_t i = 0; i < netname.length(); ++i )
2100 mask[pos + i] =
true;
2102 pos = aExpression.find( netname, pos + 1 );
2106 wxString quotedNetnames =
"";
2107 bool startQuote =
true;
2110 for(
size_t i = 0; i < aExpression.length(); i++ )
2112 if( mask[i] && startQuote )
2114 quotedNetnames = quotedNetnames +
"\"";
2117 else if( !mask[i] && !startQuote )
2119 quotedNetnames = quotedNetnames +
"\"";
2122 wxString ch = aExpression[i];
2123 quotedNetnames = quotedNetnames + ch;
2128 quotedNetnames = quotedNetnames +
"\"";
2130 return quotedNetnames;
2135 constexpr const char* cmd =
"let user{} = {}";
2137 simulator()->Command(
"echo " + fmt::format( cmd,
id, signal.ToStdString() ) );
2138 simulator()->Command( fmt::format( cmd,
id, quoteNetNames( signal ).ToStdString() ) );
2150 wxString ref = tuner->GetSymbolRef();
2151 KIID symbolId = tuner->GetSymbol( &sheetPath );
2157 reporter.
Report( wxString::Format(
_(
"%s not found" ), ref ) );
2165 reporter.
Report( wxString::Format(
_(
"%s is not tunable" ), ref ) );
2169 double floatVal = tuner->GetValue().ToDouble();
2181 wxTextFile file( aPath );
2186 wxString firstLine = file.GetFirstLine();
2188 bool legacy = firstLine.StartsWith( wxT(
"version " ) ) || firstLine.ToLong( &
dummy );
2213 wxFileName filename( aPath );
2217 simulator()->Settings()->SetWorkbookFilename( filename.GetFullPath() );
2225 wxFFileInputStream fp( aPath, wxT(
"rt" ) );
2226 wxStdInputStream fstream( fp );
2233 nlohmann::json js = nlohmann::json::parse( fstream,
nullptr,
true,
true );
2235 std::map<SIM_PLOT_TAB*, nlohmann::json> traceInfo;
2237 for(
const nlohmann::json& tab_js : js[
"tabs" ] )
2239 wxString simCommand;
2243 for(
const nlohmann::json& cmd : tab_js[
"commands" ] )
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" )
2256 simCommand += wxString( cmd.get<wxString>() ).Trim();
2266 if( tab_js.contains(
"traces" ) )
2267 traceInfo[plotTab] = tab_js[
"traces" ];
2269 if( tab_js.contains(
"measurements" ) )
2271 for(
const nlohmann::json& m_js : tab_js[
"measurements" ] )
2272 plotTab->
Measurements().emplace_back( m_js[
"expr" ], m_js[
"format" ] );
2276 plotTab->
ShowGrid( tab_js[
"showGrid" ] );
2278 if( tab_js.contains(
"fixedY1scale" ) )
2280 const nlohmann::json& scale_js = tab_js[
"fixedY1scale" ];
2281 plotTab->
SetY1Scale(
true, scale_js[
"min" ], scale_js[
"max" ] );
2285 if( tab_js.contains(
"fixedY2scale" ) )
2287 const nlohmann::json& scale_js = tab_js[
"fixedY2scale" ];
2288 plotTab->
SetY2Scale(
true, scale_js[
"min" ], scale_js[
"max" ] );
2292 if( tab_js.contains(
"fixedY3scale" ) )
2295 const nlohmann::json& scale_js = tab_js[
"fixedY3scale" ];
2296 plotTab->
SetY3Scale(
true, scale_js[
"min" ], scale_js[
"max" ] );
2300 if( tab_js.contains(
"legend" ) )
2302 const nlohmann::json& legend_js = tab_js[
"legend" ];
2307 if( tab_js.contains(
"margins" ) )
2309 const nlohmann::json& margins_js = tab_js[
"margins" ];
2311 margins_js[
"right" ],
2312 margins_js[
"bottom" ],
2313 margins_js[
"left" ] );
2320 if( js.contains(
"user_defined_signals" ) )
2322 for(
const nlohmann::json& signal_js : js[
"user_defined_signals" ] )
2331 firstTab->SetLastSchTextSimCommand( js[
"last_sch_text_sim_command"] );
2334 int tempCustomCursorsCnt = 0;
2336 if( js.contains(
"custom_cursors" ) )
2337 tempCustomCursorsCnt = js[
"custom_cursors"];
2339 tempCustomCursorsCnt = 2;
2349 int aCursorId,
const nlohmann::json& aCursor_js )
2351 if( aCursorId >= 1 )
2355 cursor->SetName( aSignalName );
2356 cursor->SetCoordX( aCursor_js[
"position" ] );
2358 aTrace->SetCursor( aCursorId,
cursor );
2362 if( aCursorId == -1 )
2373 aCursor_js[
"x_format"] );
2375 aCursor_js[
"y_format"] );
2385 for(
const auto& [ plotTab, traces_js ] : traceInfo )
2387 for(
const nlohmann::json& trace_js : traces_js )
2389 wxString signalName = trace_js[
"signal" ];
2391 TRACE* trace = plotTab->GetOrAddTrace( vectorName, trace_js[
"trace_type" ] );
2395 if( trace_js.contains(
"cursorD" ) )
2396 addCursor( plotTab, trace, signalName, -1, trace_js[
"cursorD" ] );
2398 std::vector<const char*> aVec;
2401 for(
int i = 1; i <= tempCustomCursorsCnt; i++ )
2403 wxString str =
"cursor" + std::to_string( i );
2404 aVec.emplace_back( str.c_str() );
2406 if( trace_js.contains( aVec[i - 1] ) )
2407 addCursor( plotTab, trace, signalName, i, trace_js[aVec[i - 1]] );
2410 if( trace_js.contains(
"color" ) )
2413 color.Set( wxString( trace_js[
"color"].get<wxString>() ) );
2415 plotTab->UpdateTraceStyle( trace );
2420 plotTab->UpdatePlotColors();
2423 catch( nlohmann::json::parse_error& error )
2425 wxLogTrace(
traceSettings, wxT(
"Json parse error reading %s: %s" ), aPath, error.what() );
2429 catch( nlohmann::json::type_error& error )
2431 wxLogTrace(
traceSettings, wxT(
"Json type error reading %s: %s" ), aPath, error.what() );
2435 catch( nlohmann::json::invalid_iterator& error )
2437 wxLogTrace(
traceSettings, wxT(
"Json invalid_iterator error reading %s: %s" ), aPath, error.what() );
2441 catch( nlohmann::json::out_of_range& error )
2443 wxLogTrace(
traceSettings, wxT(
"Json out_of_range error reading %s: %s" ), aPath, error.what() );
2449 wxLogTrace(
traceSettings, wxT(
"Error reading %s" ), aPath );
2458 int cursorIdAfterD = aCursorId;
2461 cursorIdAfterD = cursorIdAfterD - 1;
2466 aTraceJs[
"cursor" + wxString(
"" ) << aCursorId] =
2467 nlohmann::json( { {
"position",
cursor->GetCoords().x },
2474 aTraceJs[
"cursorD"] =
2485 wxFileName filename = aPath;
2490 file.Create( filename.GetFullPath(),
true );
2492 if( !file.IsOpened() )
2495 nlohmann::json tabs_js = nlohmann::json::array();
2506 nlohmann::json commands_js = nlohmann::json::array();
2513 commands_js.push_back(
".kicad adjustpaths" );
2516 commands_js.push_back(
".save all" );
2519 commands_js.push_back(
".probe alli" );
2522 commands_js.push_back(
".probe allp" );
2525 commands_js.push_back(
".kicad esavenone" );
2527 nlohmann::json tab_js = nlohmann::json(
2529 {
"commands", commands_js } } );
2533 nlohmann::json traces_js = nlohmann::json::array();
2535 auto findSignalName =
2536 [&](
const wxString& aVectorName ) -> wxString
2538 wxString vectorName;
2541 if( aVectorName.EndsWith(
_(
" (phase)" ) ) )
2542 suffix =
_(
" (phase)" );
2543 else if( aVectorName.EndsWith(
_(
" (gain)" ) ) )
2544 suffix =
_(
" (gain)" );
2546 vectorName = aVectorName.Left( aVectorName.Length() - suffix.Length() );
2551 return signal + suffix;
2557 for(
const auto& [
name, trace] : plotTab->GetTraces() )
2559 nlohmann::json trace_js = nlohmann::json(
2560 { {
"trace_type", (int) trace->GetType() },
2561 {
"signal", findSignalName( trace->GetDisplayName() ) },
2567 if( trace->GetCursor( 1 ) || trace->GetCursor( 2 ) )
2569 trace_js[
"cursorD"] = nlohmann::json(
2574 traces_js.push_back( trace_js );
2577 nlohmann::json measurements_js = nlohmann::json::array();
2579 for(
const auto& [
measurement, format ] : plotTab->Measurements() )
2581 measurements_js.push_back( nlohmann::json( { {
"expr",
measurement },
2582 {
"format", format } } ) );
2585 tab_js[
"traces" ] = traces_js;
2586 tab_js[
"measurements" ] = measurements_js;
2587 tab_js[
"dottedSecondary" ] = plotTab->GetDottedSecondary();
2588 tab_js[
"showGrid" ] = plotTab->IsGridShown();
2592 if( plotTab->GetY1Scale( &min, &max ) )
2593 tab_js[
"fixedY1scale" ] = nlohmann::json( { {
"min", min }, {
"max", max } } );
2595 if( plotTab->GetY2Scale( &min, &max ) )
2596 tab_js[
"fixedY2scale" ] = nlohmann::json( { {
"min", min }, {
"max", max } } );
2598 if( plotTab->GetY3Scale( &min, &max ) )
2599 tab_js[
"fixedY3scale" ] = nlohmann::json( { {
"min", min }, {
"max", max } } );
2601 if( plotTab->IsLegendShown() )
2603 tab_js[
"legend" ] = nlohmann::json( { {
"x", plotTab->GetLegendPosition().x },
2604 {
"y", plotTab->GetLegendPosition().y } } );
2607 mpWindow* plotWin = plotTab->GetPlotWin();
2609 tab_js[
"margins" ] = nlohmann::json( { {
"left", plotWin->
GetMarginLeft() },
2615 tabs_js.push_back( tab_js );
2618 nlohmann::json userDefinedSignals_js = nlohmann::json::array();
2621 userDefinedSignals_js.push_back( signal );
2624 nlohmann::json js = nlohmann::json( { {
"version", 7 },
2625 {
"tabs", tabs_js },
2626 {
"user_defined_signals", userDefinedSignals_js },
2639 std::stringstream buffer;
2640 buffer << std::setw( 2 ) << js << std::endl;
2642 bool res = file.Write( buffer.str() );
2649 simulator()->Settings()->SetWorkbookFilename( filename.GetFullPath() );
2667 default: wxFAIL_MSG( wxS(
"Unhandled simulation type" ) );
return SPT_UNKNOWN;
2686 &source, &
scale, &pts, &fStart, &fStop, &saveAll );
2694 int& aSashPosition )
2696 bool isShown = aPanel->IsShown();
2699 aSashPosition = aSplitterWindow->GetSashPosition();
2701 aPanel->Show( !isShown );
2703 aSplitterWindow->SetSashInvisible( isShown );
2704 aSplitterWindow->SetSashPosition( isShown ? -1 : aSashPosition,
true );
2706 aSplitterWindow->UpdateSize();
2707 m_parent->Refresh();
2744 for(
size_t page = 0; page <
m_plotNotebook->GetPageCount(); page++ )
2789 std::vector<std::pair<wxString, wxString>>& measurements = plotTab->Measurements();
2791 measurements.clear();
2829 simulator()->Command(
"setplot " + simTab->GetSpicePlotName().ToStdString() );
2843 for(
const auto& [
measurement, format ] : plotTab->Measurements() )
2851 if( plotTab->GetSimType() ==
ST_TRAN || plotTab->GetSimType() ==
ST_AC
2852 || plotTab->GetSimType() ==
ST_DC || plotTab->GetSimType() ==
ST_SP )
2889 CURSOR* cursor1 =
nullptr;
2890 wxString cursor1Name;
2891 wxString cursor1Units;
2892 CURSOR* cursor2 =
nullptr;
2893 wxString cursor2Name;
2894 wxString cursor2Units;
2897 [&](
TRACE* aTrace ) -> wxString
2918 [&](
TRACE* aTrace ) -> wxString
2939 [
this](
double aValue,
int aCursorId,
int aCol ) -> wxString
2952 cursor1Name = getNameY( trace );
2953 cursor1Units = getUnitsY( trace );
2955 wxRealPoint coords =
cursor->GetCoords();
2975 cursor2Name = getNameY( trace );
2976 cursor2Units = getUnitsY( trace );
2978 wxRealPoint coords =
cursor->GetCoords();
2993 if( cursor1 && cursor2 && cursor1Units == cursor2Units )
3002 signal = wxString::Format( wxS(
"%s[2 - 1]" ), cursor2->
GetName() );
3004 signal = wxString::Format( wxS(
"%s - %s" ), cursor2->
GetName(), cursor1->
GetName() );
3015 wxString valColName =
_(
"Value" );
3017 if( !cursor1Name.IsEmpty() )
3019 if( cursor2Name.IsEmpty() || cursor1Name == cursor2Name )
3020 valColName = cursor1Name;
3022 else if( !cursor2Name.IsEmpty() )
3024 valColName = cursor2Name;
3038 wxString cursName = getNameY( trace );
3039 wxString cursUnits = getUnitsY( trace );
3041 wxRealPoint coords =
cursor->GetCoords();
3056 valColName =
_(
"Value" );
3058 if( !cursName.IsEmpty()
3061 valColName = cursName;
3082 plotTab->ResetScales(
true );
3102 std::vector<wxString> signals;
3104 for(
const std::string& vec :
simulator()->AllVectors() )
3105 signals.emplace_back( vec );
3113 std::vector<wxString> signals;
3115 for(
const wxString& signal :
m_signals )
3116 signals.emplace_back( signal );
3119 signals.emplace_back( signal );
3151 if( simType ==
ST_NOISE && aFinal )
3153 m_simConsole->AppendText(
_(
"\n\nSimulation results:\n\n" ) );
3160 simulator()->CurrentPlotName().Mid( 5 ).ToLong( &number );
3162 for(
const std::string& vec :
simulator()->AllVectors() )
3164 std::vector<double> val_list =
simulator()->GetRealVector( vec, 1 );
3167 msg.Printf( wxS(
"%s: %sV\n" ), vec, value );
3173 simulator()->Command( fmt::format(
"setplot noise{}", number - 1 ) );
3178 wxCHECK_RET( plotTab, wxT(
"not a SIM_PLOT_TAB" ) );
3187 std::map<TRACE*, TRACE_INFO> traceMap;
3190 traceMap[ trace ] = { wxEmptyString,
SPT_UNKNOWN,
false };
3195 for(
const wxString& signal :
m_signals )
3200 if(
TRACE* trace = plotTab->
GetTrace( vectorName, traceType ) )
3201 traceMap[ trace ] = { vectorName, traceType,
false };
3209 if( simType ==
ST_AC )
3215 if(
TRACE* trace = plotTab->
GetTrace( vectorName, subType ) )
3216 traceMap[ trace ] = { vectorName, subType, !aFinal };
3219 else if( simType ==
ST_SP )
3225 if(
TRACE* trace = plotTab->
GetTrace( vectorName, subType ) )
3226 traceMap[trace] = { vectorName, subType, !aFinal };
3231 if(
TRACE* trace = plotTab->
GetTrace( vectorName, traceType ) )
3232 traceMap[ trace ] = { vectorName, traceType, !aFinal };
3238 for(
const auto& [ trace, traceInfo ] : traceMap )
3240 if( traceInfo.Vector.IsEmpty() )
3244 for(
const auto& [ trace,
info ] : traceMap )
3246 std::vector<double> data_x;
3248 if( !
info.Vector.IsEmpty() )
3266 else if( simType ==
ST_OP && aFinal )
3268 m_simConsole->AppendText(
_(
"\n\nSimulation results:\n\n" ) );
3271 for(
const std::string& vec :
simulator()->AllVectors() )
3273 std::vector<double> val_list =
simulator()->GetRealVector( vec, 1 );
3275 if( val_list.empty() )
3282 const size_t tab = 25;
3283 size_t padding = ( signal.length() < tab ) ? ( tab - signal.length() ) : 1;
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;
3293 msg.Printf( wxT(
"%s%s\n" ),
3294 ( signal + wxT(
":" ) ).Pad( padding, wxUniChar(
' ' ) ),
3301 signal = signal.SubString( 2, signal.Length() - 2 );
3304 signal += wxS(
":power" );
3309 else if( simType ==
ST_PZ && aFinal )
3311 m_simConsole->AppendText(
_(
"\n\nSimulation results:\n\n" ) );
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
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.
const wxRealPoint & GetCoords() const
void SetCoordX(double aValue)
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.
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.
A color representation with 4 components: red, green, blue, alpha.
wxString ToCSSString() const
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...
const wxString What() const
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.
@ OPTION_SAVE_ALL_CURRENTS
@ OPTION_SAVE_ALL_VOLTAGES
@ OPTION_SAVE_ALL_DISSIPATIONS
@ OPTION_ADJUST_INCLUDE_PATHS
@ OPTION_ADJUST_PASSIVE_VALS
const SPICE_ITEM * FindItem(const wxString &aRefName) const
Find and return the item corresponding to aRefName.
A singleton reporter that reports to nowhere.
virtual const wxString GetProjectPath() const
Return the full path of the project.
Holds all the data relating to one schematic.
void SetOperatingPoint(const wxString &aSignal, double aValue)
Set operating points from a .op simulation.
void ClearOperatingPoints()
Clear operating points from a .op simulation.
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,...
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.
void GetFields(std::vector< SCH_FIELD * > &aVector, bool aVisibleOnly) const override
Populate a std::vector with SCH_FIELDs, sorted in ordinal order.
const wxString GetRef(const SCH_SHEET_PATH *aSheet, bool aIncludeUnit=false) const override
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.
wxBoxSizer * m_sizerTuners
WX_GRID * m_measurementsGrid
wxSplitterWindow * m_splitterLeftRight
wxSplitterWindow * m_splitterMeasurements
wxTextCtrl * m_simConsole
wxSplitterWindow * m_splitterCursors
wxSplitterWindow * m_splitterPlotAndConsole
wxSplitterWindow * m_splitterSignals
wxAuiNotebook * m_plotNotebook
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.
int m_splitterTuneValuesSashPosition
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)
int m_splitterCursorsSashPosition
void ToggleSimulationSidePanel()
void UpdateMeasurement(int aRow)
Update a measurement in the measurements grid.
wxString getNoiseSource() const
std::vector< wxString > SimPlotVectors() const
unsigned int m_plotNumber
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.
int m_splitterLeftRightSashPosition
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)
void updateMeasurementsFromGrid()
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.
void ToggleDarkModePlots()
SIM_TRACE_TYPE getXAxisType(SIM_TYPE aType) const
Return X axis for a given simulation type.
void OnPlotSettingsChanged()
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)
int m_splitterSignalsSashPosition
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.
int m_splitterPlotAndConsoleSashPosition
SIMULATOR_FRAME * m_simulatorFrame
void ShowChangedLanguage()
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...
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
const SPICE_GENERATOR & SpiceGenerator() const
void WriteFields(std::vector< SCH_FIELD > &aFields) const
void SetParamValue(int aParamIndex, const std::string &aValue, SIM_VALUE::NOTATION aNotation=SIM_VALUE::NOTATION::SI)
static void FillDefaultColorList(bool aWhiteBg)
Fills m_colorList by a default set of colors.
bool DeleteTrace(const wxString &aVectorName, int aTraceType)
wxString GetLabelY1() const
mpWindow * GetPlotWin() const
void ShowGrid(bool aEnable)
wxString GetUnitsY2() const
void SetY2Scale(bool aLock, double aMin, double aMax)
TRACE * GetTrace(const wxString &aVecName, int aType) const
wxString GetLabelX() const
const std::map< wxString, TRACE * > & GetTraces() const
wxString GetLabelY3() const
void SetY1Scale(bool aLock, double aMin, double aMax)
void SetY3Scale(bool aLock, double aMin, double aMax)
std::vector< std::pair< wxString, wxString > > & Measurements()
void UpdateTraceStyle(TRACE *trace)
Update plot colors.
void SetLegendPosition(const wxPoint &aPosition)
void ResetScales(bool aIncludeX)
Update trace line style.
void ShowLegend(bool aEnable)
wxString GetLabelY2() const
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.
void ApplyPreferences(const SIM_PREFERENCES &aPrefs) override
wxString GetUnitsY1() const
void DisableCursor(TRACE *aTrace, int aCursorId)
Reset scale ranges to fit the current traces.
wxString GetUnitsY3() const
int GetSimOptions() const
SIM_TYPE GetSimType() const
const wxString & GetSimCommand() const
static bool IsPlottable(SIM_TYPE aSimType)
void SetSimOptions(int aOptions)
wxString GetLastSchTextSimCommand() const
void SetSpicePlotName(const wxString &aPlotName)
static std::string ToSpice(const std::string &aString)
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.
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.
~SUPPRESS_GRID_CELL_EVENTS()
SUPPRESS_GRID_CELL_EVENTS(SIMULATOR_FRAME_UI *aFrame)
SIMULATOR_FRAME_UI * m_frame
void SetTraceColour(const wxColour &aColour)
bool HasCursor(int aCursorId)
CURSOR * GetCursor(int aCursorId)
Custom widget to handle quick component values modification and simulation on the fly.
void ClearRows()
wxWidgets recently added an ASSERT which fires if the position is greater than or equal to the number...
A wrapper for reporting to a wxString object.
bool HasMessage() const override
Returns true if the reporter client is non-empty.
REPORTER & Report(const wxString &aText, SEVERITY aSeverity=RPT_SEVERITY_UNDEFINED) override
Report a string with a given severity.
const wxString & GetMessages() const
A KICAD version of wxTextEntryDialog which supports the various improvements/work-arounds from DIALOG...
wxString GetValue() const
const wxString & GetName() const
Get layer name.
const wxPen & GetPen() const
Get pen set for this layer.
Canvas for plotting mpLayer implementations.
int GetMarginLeft() const
void SetMargins(int top, int right, int bottom, int left)
Set window margins, creating a blank area where some kinds of layers cannot draw.
void UpdateAll()
Refresh display.
int GetMarginRight() const
int GetMarginBottom() const
bool AddLayer(mpLayer *layer, bool refreshDisplay=true)
Add a plot layer to the canvas.
void Fit() override
Set view to fit global bounding box of all plot layers and refresh display.
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
This file is part of the common library.
Abstract pattern-matching tool and implementations.
@ GRIDTRICKS_FIRST_CLIENT_ID
static const std::string WorkbookFileExtension
KICOMMON_API wxFont GetStatusFont(wxWindow *aWindow)
SIM_TYPE
< Possible simulation types
void sortSignals(std::vector< wxString > &signals)
wxString vectorNameFromSignalId(int aUserDefinedSignalId)
MEASUREMENTS_GIRD_COLUMNS
SIM_TRACE_TYPE operator|(SIM_TRACE_TYPE aFirst, SIM_TRACE_TYPE aSecond)
@ MYID_DELETE_MEASUREMENT
std::vector< FAB_LAYER_COLOR > dummy
wxString UnescapeString(const wxString &aSource)
int measurements_panel_height
SIM_PREFERENCES preferences
Contains preferences pertaining to the simulator.
Definition of file extensions used in Kicad.