qucs_s/qucs/qucs_actions.cpp
andresmmera f4d1c4a225 S-parameter viewer - Initial development (squashed)
This commit contains the first draft of the user interface
Read Touchstone files

It was implemented a basic function to read Touchstone files.

It can only read Touchstone files up to 4 ports and only S-parameter data.


Please see "Touchstone Specification, Version 2.1, ratified January 26 2024 by the IBIS Open Forum": https://ibis.org/touchstone_ver2.1/ 
Add QScrollArea widgets to the files and traces lists

Large number of files and traces are expected, so there is a need for scrollable areas in the files and traces lists.
Basic plotting structure


Add default behaviour when loading one single s2p

A default behavior is added. When a single s2p file is selected, the program automatically displays S21, S11 and S22
Replace the "Delete" message by a trash image

The delete image was taken from here

https://commons.wikimedia.org/wiki/File:Delete-button.svg

This file is licensed under the Creative Commons Attribution-Share Alike 4.0 International license.

    You are free:

        to share – to copy, distribute and transmit the work
        to remix – to adapt the work

    Under the following conditions:

        attribution – You must give appropriate credit, provide a link to the license, and indicate if changes were made. You may do so in any reasonable manner, but not in any way that suggests the licensor endorses you or your use.
        share alike – If you remix, transform, or build upon the material, you must distribute your contributions under the same or compatible license as the original.
Read Touchstone files with more than two ports


Update traces combobox depending on the selected dataset

If the user has loaded data with different number of ports, the traces combobox must be refreshed each time the user changes the dataset selection. Otherwise, this may cause that the user selects a non existing trace
Fix style in buttons for removing files

The QPushButtons were replaced by QToolButtons. With the QPushButtons the widget was too wide
Fix style in buttons for removing traces

QPushButton was converted into QToolButton
Delete dataset and its traces when the user decides to remove a file


Remove files and traces

Rework on the logic on how to remove datasets and traces
After removing a file, remove the associated widgets


Remove trace from Chart


Update file widget position in the grid after removing


Function handler for changing the color of a trace


Change linestyle depending on the combo selection


Set initial color of the color pickers


Added a spinbox control to control the trace width

It was added a spinbox that controls the width of the traces displayed. This is very convenient when a bunch of traces are being displayed and the user wants to highlight one of them easily
Added function  handler for controlling the x-axis

A handler function was added to control the x-axis settings as the user changes the minimum, maximum or the tick interval
Update traces when changing the axis settings


Fix trace plotting refresh


Fix frequency limits when loading a GHz range file


Dockable widgets


Autoadjust y-axis settings


Automatically add K, delta, mu_s, m_p, MAG and MSG traces in S2P files

When a Touchtone file has two ports, the stability metrics are automatically computed and added to the dataset
Add marker table feature

It was added a new dock consisting on a marker table and some widgets for its management
Add dot marker and vertical lines in the QChart


Make case insensitive the frequency scale

Files were found were the frequency scale is all in capital letters. This creates a problem when reading the spar data. This commit fixes this by putting the frequency scale in lower case
Auto adjust x-axis when changing the units


Put x_div values as a ComboBox

It makes no much sense in having a decimal spinbox for defining the tick interval. It leads to decimal ticks. It's better to have a closed list of possible values
y axis tick in combobox


Fix vertical line markers


Fix bad "About Qt" connection 

The "About Qt" message was not properly connected. As a consequence, when the user went to "Help-> About Qt..." nothing showed up.

This commit is intended to fix this by connecting the menu with the handler as it's done in the filter design tool
Link S-parameter viewer to Qucs-S


Add Re{Zin}, Im{Zin} traces to s1p and s2p files


Hide y-axis units

It makes no much sense for now to have it since it may happen that the y-axis represent dB, Ohm or simply its unitless (e.g. K, mu, ...)
Fix segfault when removing one single file

In previous commits, it was observed a segfault when removing one single s-par file. This happened because the program was freeing widgets already freed. This situation is avoided by ordering the list of widgets to remove
Autoadjust y-axis


Remove widgets for marker placement

They are actually not needed. The SpinBox and the combo with units just add clutter. The user can set the marker freq once added
Update x-axis limits after removing file


Increase maximum x-ticks


Get suffix using Qt method

This is more robust than the previous approach
Fix frequency scale in markers


Enable drag and drop to open files


Fix segfault when removing file


Readjust frequency limits when dataset has no traces


Fix read touchstone

Files were found whose header contains no !
Fix initial marker step


Fix autoscale y-axis


Prevent docks from closing

It makes no sense the user can close the docks
Solve infinite loop when fmax=3000 [unit]


Implemented button for removing all files on a row


Implement button for removing all markers on a row
2024-09-08 08:35:42 +02:00

1740 lines
53 KiB
C++

/***************************************************************************
qucs_actions.cpp
-----------------
begin : Sat May 1 2004
copyright : (C) 2004 by Michael Margraf
email : michael.margraf@alumni.tu-berlin.de
***************************************************************************/
/* Copyright (C) 2014 Guilherme Brondani Torri <guitorri@gmail.com> */
/***************************************************************************
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
***************************************************************************/
/*! \file qucs_actions.cpp
* \brief Actions implementation for the GUI menu items
*/
#include <QtCore>
#include <stdlib.h>
#include <limits.h>
#include <QProcess>
#include <qt3_compat/q3ptrlist.h>
#include <QRegularExpressionValidator>
#include <QLineEdit>
#include <QAction>
#include <QStatusBar>
#include <QMessageBox>
#include <QFileDialog>
#include <QMenu>
#include <QComboBox>
#include <QDockWidget>
#include <QTreeWidgetItem>
#include <QMutableHashIterator>
#include <QListWidget>
#include <QDesktopServices>
#include "portsymbol.h"
#include "projectView.h"
#include "main.h"
#include "qucs.h"
#include "schematic.h"
#include "textdoc.h"
#include "mouseactions.h"
#include "messagedock.h"
#include "components/ground.h"
#include "components/subcirport.h"
#include "components/equation.h"
#include "spicecomponents/sp_nutmeg.h"
#include "dialogs/matchdialog.h"
#include "dialogs/changedialog.h"
#include "dialogs/searchdialog.h"
#include "dialogs/librarydialog.h"
#include "dialogs/loaddialog.h"
#include "dialogs/importdialog.h"
#include "dialogs/aboutdialog.h"
#include "module.h"
#include "extsimkernels/xyce.h"
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
// for editing component name on schematic
QRegularExpression Expr_CompProp;
QRegularExpressionValidator Val_CompProp(Expr_CompProp, 0);
// -----------------------------------------------------------------------
// This function is called from all toggle actions.
bool QucsApp::performToggleAction(bool on, QAction *Action,
pToggleFunc Function, pMouseFunc MouseMove, pMouseFunc2 MousePress)
{
slotHideEdit(); // disable text edit of component property
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
// Perform toggle release clean up.
if(!on) {
MouseMoveAction = 0;
MousePressAction = 0;
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
activeAction = 0; // no action active
// Return to select mode.
slotEscape();
return false;
}
do {
if(Function) if((Doc->*Function)()) {
Action->blockSignals(true);
Action->setChecked(false); // release toolbar button
Action->blockSignals(false);
Doc->viewport()->update();
break;
}
if(activeAction) {
activeAction->blockSignals(true); // do not call toggle slot
activeAction->setChecked(false); // set last toolbar button off
activeAction->blockSignals(false);
}
activeAction = Action;
MouseMoveAction = MouseMove;
MousePressAction = MousePress;
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
} while(false); // to perform "break"
Doc->viewport()->update();
return true;
}
// -----------------------------------------------------------------------
// Is called, when "set on grid" action is triggered.
void QucsApp::slotOnGrid(bool on)
{
performToggleAction(on, onGrid, &Schematic::elementsOnGrid,
&MouseActions::MMoveOnGrid, &MouseActions::MPressOnGrid);
}
// -----------------------------------------------------------------------
// Is called when the rotate toolbar button is pressed.
void QucsApp::slotEditRotate(bool on)
{
performToggleAction(on, editRotate, &Schematic::rotateElements,
&MouseActions::MMoveRotate, &MouseActions::MPressRotate);
}
// -----------------------------------------------------------------------
// Is called when the mirror toolbar button is pressed.
void QucsApp::slotEditMirrorX(bool on)
{
performToggleAction(on, editMirror, &Schematic::mirrorXComponents,
&MouseActions::MMoveMirrorX, &MouseActions::MPressMirrorX);
}
// -----------------------------------------------------------------------
// Is called when the mirror toolbar button is pressed.
void QucsApp::slotEditMirrorY(bool on)
{
performToggleAction(on, editMirrorY, &Schematic::mirrorYComponents,
&MouseActions::MMoveMirrorY, &MouseActions::MPressMirrorY);
}
// -----------------------------------------------------------------------
// Is called when the activate/deactivate toolbar button is pressed.
// It also comments out the selected text on a text document
// \todo update the status or tooltip message
void QucsApp::slotEditActivate (bool on)
{
TextDoc * Doc = (TextDoc *) DocumentTab->currentWidget();
if (isTextDocument (Doc)) {
//TODO Doc->clearParagraphBackground (Doc->tmpPosX);
Doc->commentSelected ();
editActivate->blockSignals (true);
editActivate->setChecked(false); // release toolbar button
editActivate->blockSignals (false);
}
else
performToggleAction (on, editActivate,
&Schematic::activateSelectedComponents,
&MouseActions::MMoveActivate, &MouseActions::MPressActivate);
}
// ------------------------------------------------------------------------
// Is called if "Delete"-Button is pressed.
void QucsApp::slotEditDelete(bool on)
{
TextDoc *Doc = (TextDoc*)DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
Doc->viewport()->setFocus();
//Doc->del();
Doc->textCursor().deleteChar();
editDelete->blockSignals(true);
editDelete->setChecked(false); // release toolbar button
editDelete->blockSignals(false);
}
else
performToggleAction(on, editDelete, &Schematic::deleteElements,
&MouseActions::MMoveDelete, &MouseActions::MPressDelete);
}
// -----------------------------------------------------------------------
// Is called if "Wire"-Button is pressed.
void QucsApp::slotSetWire(bool on)
{
performToggleAction(on, insWire, 0,
&MouseActions::MMoveWire1, &MouseActions::MPressWire1);
}
// -----------------------------------------------------------------------
void QucsApp::slotInsertLabel(bool on)
{
performToggleAction(on, insLabel, 0,
&MouseActions::MMoveLabel, &MouseActions::MPressLabel);
}
// -----------------------------------------------------------------------
void QucsApp::slotSetMarker(bool on)
{
performToggleAction(on, setMarker, 0,
&MouseActions::MMoveMarker, &MouseActions::MPressMarker);
}
// -----------------------------------------------------------------------
// Toolbar button to update the diagram limits using the mouse - aka zooming.
void QucsApp::slotSetDiagramLimits(bool on)
{
performToggleAction(on, setDiagramLimits, 0,
&MouseActions::MMoveSetLimits, &MouseActions::MPressSetLimits);
}
// -----------------------------------------------------------------------
// Context menu option to reset the diagram limits to defaults.
void QucsApp::slotResetDiagramLimits()
{
if (view->focusElement && view->focusElement->Type == isDiagram)
{
Diagram* diagram = (Diagram*)(view->focusElement);
diagram->xAxis.autoScale = true;
diagram->yAxis.autoScale = true;
diagram->zAxis.autoScale = true;
// Now read in the data.
auto* Doc = (Schematic*)DocumentTab->currentWidget();
QFileInfo Info(Doc->DocName);
QString defaultDataSet = Info.absolutePath() + QDir::separator() + Doc->DataSet;
diagram->loadGraphData(defaultDataSet);
Doc->setChanged(true, true);
Doc->viewport()->update();
}
// Return to select mode (in case SetDiagramLimits is still selected).
slotEscape();
}
// -----------------------------------------------------------------------
// Is called, when "move component text" action is triggered.
void QucsApp::slotMoveText(bool on)
{
performToggleAction(on, moveText, 0,
&MouseActions::MMoveMoveTextB, &MouseActions::MPressMoveText);
}
// -----------------------------------------------------------------------
// Is called, when "Zoom in" action is triggered.
void QucsApp::slotZoomIn(bool on)
{
auto *Doc = (TextDoc*)DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
Doc->zoomBy(1.5f);
magPlus->blockSignals(true);
magPlus->setChecked(false);
magPlus->blockSignals(false);
}
else
performToggleAction(on, magPlus, 0,
&MouseActions::MMoveZoomIn, &MouseActions::MPressZoomIn);
}
void QucsApp::slotEscape()
{
select->setChecked(true);
slotSearchClear();
}
// -----------------------------------------------------------------------
// Is called when the select toolbar button is pressed.
void QucsApp::slotSelect(bool on)
{
QWidget *w = DocumentTab->currentWidget();
if(isTextDocument(w)) {
((TextDoc*)w)->viewport()->setFocus();
select->blockSignals(true);
select->setChecked(true);
select->blockSignals(false);
return;
}
// goto to insertWire mode if ESC pressed during wiring
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(MouseMoveAction == &MouseActions::MMoveWire2) {
MouseMoveAction = &MouseActions::MMoveWire1;
MousePressAction = &MouseActions::MPressWire1;
Doc->viewport()->update();
select->blockSignals(true);
select->setChecked(false);
select->blockSignals(false);
return;
}
if(performToggleAction(on, select, 0, 0, &MouseActions::MPressSelect)) {
MouseReleaseAction = &MouseActions::MReleaseSelect;
MouseDoubleClickAction = &MouseActions::MDoubleClickSelect;
}
}
// --------------------------------------------------------------------
void QucsApp::slotEditCut()
{
statusBar()->showMessage(tr("Cutting selection..."));
slotHideEdit(); // disable text edit of component property
QWidget *Doc = DocumentTab->currentWidget();
if(isTextDocument (Doc)) {
((TextDoc *)Doc)->cut();
} else {
((Schematic *)Doc)->cut();
}
statusBar()->showMessage(tr("Ready."));
}
// --------------------------------------------------------------------
void QucsApp::slotEditCopy()
{
statusBar()->showMessage(tr("Copying selection to clipboard..."));
QWidget *Doc = DocumentTab->currentWidget();
if(isTextDocument (Doc)) {
((TextDoc *)Doc)->copy();
} else {
((Schematic *)Doc)->copy();
}
statusBar()->showMessage(tr("Ready."));
}
// -----------------------------------------------------------------------
void QucsApp::slotEditPaste(bool on)
{
// get the current document
QWidget *Doc = DocumentTab->currentWidget();
// if the current document is a text document paste in
// the contents of the clipboard as text
if(isTextDocument (Doc))
{
((TextDoc*)Doc)->paste();
editPaste->blockSignals(true);
editPaste->setChecked(false); // release toolbar button
editPaste->blockSignals(false);
return;
}
else {
// if it's not a text doc, prevent the user from editing
// while we perform the paste operation
slotHideEdit();
if(!on)
{
MouseMoveAction = 0;
MousePressAction = 0;
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
activeAction = 0; // no action active
return;
}
if(!view->pasteElements((Schematic *)Doc))
{
editPaste->blockSignals(true); // do not call toggle slot
editPaste->setChecked(false); // set toolbar button off
editPaste->blockSignals(false);
return; // if clipboard empty
}
if(activeAction)
{
activeAction->blockSignals(true); // do not call toggle slot
activeAction->setChecked(false); // set last toolbar button off
activeAction->blockSignals(false);
}
activeAction = editPaste;
MouseMoveAction = &MouseActions::MMovePaste;
view->movingRotated = 0;
MousePressAction = 0;
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
}
}
// -----------------------------------------------------------------------
void QucsApp::slotInsertEntity ()
{
TextDoc * Doc = (TextDoc *) DocumentTab->currentWidget ();
Doc->viewport()->setFocus ();
//TODO Doc->clearParagraphBackground (Doc->tmpPosX);
Doc->insertSkeleton ();
//int x, y;
//Doc->getCursorPosition (&x, &y);
//x = Doc->textCursor().blockNumber();
//y = Doc->textCursor().columnNumber();
Doc->slotCursorPosChanged();
}
// -----------------------------------------------------------------------
// Is called when the mouse is clicked upon the equation toolbar button.
void QucsApp::slotInsertEquation(bool on)
{
slotHideEdit(); // disable text edit of component property
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
if(!on) {
MouseMoveAction = 0;
MousePressAction = 0;
activeAction = 0; // no action active
return;
}
if(activeAction) {
activeAction->blockSignals(true); // do not call toggle slot
activeAction->setChecked(false); // set last toolbar button off
activeAction->blockSignals(false);
}
activeAction = insEquation;
if(view->selElem)
delete view->selElem; // delete previously selected component
if (QucsSettings.DefaultSimulator == spicecompat::simNgspice) {
view->selElem = new NutmegEquation();
} else {
view->selElem = new Equation();
}
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
MouseMoveAction = &MouseActions::MMoveElement;
MousePressAction = &MouseActions::MPressElement;
}
// -----------------------------------------------------------------------
// Is called when the mouse is clicked upon the ground toolbar button.
void QucsApp::slotInsertGround(bool on)
{
slotHideEdit(); // disable text edit of component property
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
if(!on) {
MouseMoveAction = 0;
MousePressAction = 0;
activeAction = 0; // no action active
return;
}
if(activeAction) {
activeAction->blockSignals(true); // do not call toggle slot
activeAction->setChecked(false); // set last toolbar button off
activeAction->blockSignals(false);
}
activeAction = insGround;
if(view->selElem)
delete view->selElem; // delete previously selected component
view->selElem = new Ground();
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
MouseMoveAction = &MouseActions::MMoveElement;
MousePressAction = &MouseActions::MPressElement;
}
// -----------------------------------------------------------------------
// Is called when the mouse is clicked upon the port toolbar button.
void QucsApp::slotInsertPort(bool on)
{
slotHideEdit(); // disable text edit of component property
MouseReleaseAction = 0;
MouseDoubleClickAction = 0;
if(!on) {
MouseMoveAction = 0;
MousePressAction = 0;
activeAction = 0; // no action active
return;
}
if(activeAction) {
activeAction->blockSignals(true); // do not call toggle slot
activeAction->setChecked(false); // set last toolbar button off
activeAction->blockSignals(false);
}
activeAction = insPort;
if(view->selElem)
delete view->selElem; // delete previously selected component
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if (Doc->symbolMode) {
view->selElem = new PortSymbol();
} else {
view->selElem = new SubCirPort();
}
MouseMoveAction = &MouseActions::MMoveElement;
MousePressAction = &MouseActions::MPressElement;
}
// --------------------------------------------------------------
// Is called, when "Undo"-Button is pressed.
void QucsApp::slotEditUndo()
{
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
((TextDoc*)Doc)->viewport()->setFocus();
((TextDoc*)Doc)->undo();
return;
}
slotHideEdit(); // disable text edit of component property
Doc->undo();
Doc->viewport()->update();
}
// --------------------------------------------------------------
// Is called, when "Undo"-Button is pressed.
void QucsApp::slotEditRedo()
{
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
((TextDoc*)Doc)->viewport()->setFocus();
((TextDoc*)Doc)->redo();
return;
}
slotHideEdit(); // disable text edit of component property
Doc->redo();
Doc->viewport()->update();
}
// --------------------------------------------------------------
// Is called, when "Align top" action is triggered.
void QucsApp::slotAlignTop()
{
slotHideEdit(); // disable text edit of component property
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(!Doc->aligning(0))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
// --------------------------------------------------------------
// Is called, when "Align bottom" action is triggered.
void QucsApp::slotAlignBottom()
{
slotHideEdit(); // disable text edit of component property
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(!Doc->aligning(1))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
// --------------------------------------------------------------
// Is called, when "Align left" action is triggered.
void QucsApp::slotAlignLeft()
{
slotHideEdit(); // disable text edit of component property
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(!Doc->aligning(2))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
// --------------------------------------------------------------
// Is called, when "Align right" action is triggered.
void QucsApp::slotAlignRight()
{
slotHideEdit(); // disable text edit of component property
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(!Doc->aligning(3))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
// --------------------------------------------------------------
// Is called, when "Distribute horizontally" action is triggered.
void QucsApp::slotDistribHoriz()
{
slotHideEdit(); // disable text edit of component property
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
Doc->distributeHorizontal();
Doc->viewport()->update();
}
// --------------------------------------------------------------
// Is called, when "Distribute vertically" action is triggered.
void QucsApp::slotDistribVert()
{
slotHideEdit(); // disable text edit of component property
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
Doc->distributeVertical();
Doc->viewport()->update();
}
// --------------------------------------------------------------
// Is called, when "Center horizontally" action is triggered.
void QucsApp::slotCenterHorizontal()
{
slotHideEdit(); // disable text edit of component property
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(!Doc->aligning(4))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
// --------------------------------------------------------------
// Is called, when "Center vertically" action is triggered.
void QucsApp::slotCenterVertical()
{
slotHideEdit(); // disable text edit of component property
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
if(!Doc->aligning(5))
QMessageBox::information(this, tr("Info"),
tr("At least two elements must be selected !"));
Doc->viewport()->update();
}
// ---------------------------------------------------------------------
// Is called when the "select all" action is triggered.
void QucsApp::slotSelectAll()
{
slotHideEdit(); // disable text edit of component property
QWidget *Doc = DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
((TextDoc*)Doc)->viewport()->setFocus();
//((TextDoc*)Doc)->selectAll(true);
((TextDoc*)Doc)->selectAll();
}
else {
auto selectionRect = ((Schematic*)Doc)->allBoundingRect().marginsAdded(QMargins{1, 1, 1, 1});
((Schematic*)Doc)->selectElements(selectionRect, true, false);
((Schematic*)Doc)->viewport()->update();
}
}
// ---------------------------------------------------------------------
// Is called when the "select markers" action is triggered.
void QucsApp::slotSelectMarker()
{
slotHideEdit(); // disable text edit of component property
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
Doc->selectMarkers();
Doc->viewport()->update();
}
extern QString lastDirOpenSave; // to remember last directory and file
// ------------------------------------------------------------------------
// Is called by slotShowLastMsg(), by slotShowLastNetlist() and from the
// component edit dialog.
void QucsApp::editFile(const QString& File)
{
if ((QucsSettings.Editor.toLower() == "qucs") || QucsSettings.Editor.isEmpty())
{
// The Editor is 'qucs' or empty, open a net document tab
if (File.isEmpty()) {
QucsApp::slotTextNew();
}
else
{
slotHideEdit(); // disable text edit of component property
statusBar()->showMessage(tr("Opening file..."));
QFileInfo finfo(File);
if(!finfo.exists())
statusBar()->showMessage(tr("Opening aborted, file not found."), 2000);
else {
gotoPage(File);
lastDirOpenSave = File; // remember last directory and file
statusBar()->showMessage(tr("Ready."));
}
}
}
else
{
// use an external editor
QString prog;
QStringList args;
if (QucsSettings.Editor.toLower().contains("qucsedit")) {
#if defined(_WIN32) || defined(__MINGW32__)
prog = QUCS_NAME"edit.exe";
#elif __APPLE__
prog = "qucsedit.app/Contents/MacOS/qucsedit";
#else
prog = "qucsedit";
#endif
QFileInfo editor(QucsSettings.QucsatorDir + prog);
prog = QDir::toNativeSeparators(editor.canonicalFilePath());
}
else { // user defined editor
QFileInfo editor(QucsSettings.Editor);
prog = QDir::toNativeSeparators(editor.canonicalFilePath());
}
if (!File.isEmpty()) {
args << File;
}
QProcess *QucsEditor = new QProcess();
QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
env.insert("PATH", env.value("PATH") );
QucsEditor->setProcessEnvironment(env);
qDebug() << "Command: " << prog << args.join(" ");
QucsEditor->start(prog, args);
if( !QucsEditor->waitForStarted(1000) ) {
QMessageBox::critical(this, tr("Error"), tr("Cannot start text editor! \n\n%1").arg(prog));
delete QucsEditor;
return;
}
qDebug() << QucsEditor->readAllStandardError();
// to kill it before qucs ends
connect(this, SIGNAL(signalKillEmAll()), QucsEditor, SLOT(kill()));
}
}
// ------------------------------------------------------------------------
// Is called to show the output messages of the last simulation.
void QucsApp::slotShowLastMsg()
{
editFile(QucsSettings.tempFilesDir.filePath("log.txt"));
}
// ------------------------------------------------------------------------
// Is called to show the netlist of the last simulation.
void QucsApp::slotShowLastNetlist()
{
QStringList netlists;
QStringList sim_lst;
QWidget *w = DocumentTab->currentWidget();
if (QucsSettings.DefaultSimulator == spicecompat::simXyce) {
if (isTextDocument(w)) {
QMessageBox::information(this, tr("Show netlist"),
tr("Not a schematic tab!"));
return;
} else {
Schematic *sch = (Schematic *) w;
Xyce *xyce = new Xyce(sch,this);
xyce->determineUsedSimulations(&sim_lst);
delete xyce;
}
}
switch (QucsSettings.DefaultSimulator) {
case spicecompat::simQucsator :
netlists.append(QucsSettings.tempFilesDir.filePath("netlist.txt"));
break;
case spicecompat::simNgspice :
case spicecompat::simSpiceOpus :
netlists.append(QDir::toNativeSeparators(QucsSettings.S4Qworkdir
+ "/spice4qucs.cir"));
break;
case spicecompat::simXyce: // Xyce generates one netlist for every simulation
for(const auto &sim : sim_lst) {
netlists.append(QDir::toNativeSeparators(QucsSettings.S4Qworkdir
+ "/spice4qucs."
+ sim + ".cir"));
}
break;
default: break;
}
if (!isTextDocument(w)) {
Schematic *sch = (Schematic *) w;
if (sch->isDigitalCircuit()) {
netlists.clear();
netlists.append(QucsSettings.tempFilesDir.filePath("netlist.txt"));
}
}
for(const auto &netlist: netlists) {
editFile(netlist);
}
}
// ------------------------------------------------------------------------
// Is called to start the text editor.
void QucsApp::slotCallEditor()
{
editFile(QString(""));
}
// ------------------------------------------------------------------------
// Is called to start the filter synthesis program.
void QucsApp::slotCallFilter()
{
launchTool(QUCS_NAME "filter", "filter synthesis");
}
void QucsApp::slotCallActiveFilter()
{
launchTool(QUCS_NAME "activefilter", "active filter synthesis");
}
// ------------------------------------------------------------------------
// Is called to start the transmission line calculation program.
void QucsApp::slotCallLine()
{
launchTool(QUCS_NAME "trans", "line calculation",QStringList());
}
// --------------------------------------------------------------
// Is called to show a dialog for creating matching circuits.
void QucsApp::slotCallMatch()
{
MatchDialog *d = new MatchDialog(this);
d->exec();
}
// ------------------------------------------------------------------------
// Is called to start the attenuator calculation program.
void QucsApp::slotCallAtt()
{
launchTool(QUCS_NAME "attenuator", "attenuator calculation",QStringList());
}
void QucsApp::slotCallPwrComb()
{
launchTool(QUCS_NAME "powercombining", "power combining calculation",QStringList());
}
void QucsApp::slotCallSPAR_Viewer()
{
launchTool(QUCS_NAME "spar-viewer", "s-parameter viewer",QStringList());
}
/*!
* \brief launch an external application passing arguments
*
* \param prog executable program name (will be transformed depending on OS)
* \param progDesc program description string (used for error messages)
* \param args arguments to pass to the executable
*/
void QucsApp::launchTool(const QString& prog, const QString& progDesc, const QStringList &args,
bool qucs_tool)
{
QProcess *tool = new QProcess();
QString tooldir;
if (qucs_tool) tooldir = QucsSettings.QucsatorDir;
else tooldir = QucsSettings.BinDir;
#if defined(_WIN32) || defined(__MINGW32__)
QString cmd = QDir::toNativeSeparators("\""+tooldir + prog + ".exe\"");
#elif __APPLE__
QString cmd = QDir::toNativeSeparators(tooldir + prog + ".app/Contents/MacOS/" + prog);
#else
QString cmd = QDir::toNativeSeparators(tooldir + prog);
#endif
tool->setWorkingDirectory(tooldir);
qDebug() << "Command :" << cmd;
tool->start(cmd,args);
if(!tool->waitForStarted(1000) ) {
QMessageBox::critical(this, tr("Error"),
tr("Cannot start %1 program! \n\n(%2)").arg(progDesc, cmd));
delete tool;
return;
}
// to kill the application first before qucs finishes exiting
connect(this, SIGNAL(signalKillEmAll()), tool, SLOT(kill()));
}
void QucsApp::slotCallRFLayout()
{
QString input_file, netlist_file, odir;
if (!isTextDocument(DocumentTab->currentWidget())) {
Schematic *sch = (Schematic*)DocumentTab->currentWidget();
if(sch->fileSuffix() == "dpl") {
QMessageBox::critical(this,tr("Error"),
tr("Layouting of display pages is not supported!"));
return;
}
input_file = sch->DocName;
QFileInfo inf(sch->DocName);
odir = inf.absolutePath();
netlist_file = inf.absolutePath() + QDir::separator()
+ inf.baseName() + ".net";
QFile f(netlist_file);
if (!f.open(QIODevice::WriteOnly)) {
QMessageBox::critical(this, tr("Error"), tr("Cannot write netlist!"));
return;
}
QTextStream stream(&f);
QStringList Collect;
QPlainTextEdit *ErrText = new QPlainTextEdit(); //dummy
int pNum = sch->prepareNetlist(stream, Collect, ErrText);
if (!sch->isAnalog) {
QMessageBox::critical(this, tr("Error"), tr("Digital schematic not supported!"));
return;
}
stream << '\n';
sch->createNetlist(stream, pNum);
f.close();
} else {
QMessageBox::critical(this,tr("Error"),
tr("Layouting of text documents is not supported!"));
return;
}
QProcess *tool = new QProcess();
QStringList args;
args.append("-G");
args.append("-i");
args.append(input_file);
args.append("-n");
args.append(netlist_file);
args.append("-o");
args.append(odir);
tool->start(QucsSettings.RFLayoutExecutable,args);
if(!tool->waitForStarted(1000) ) {
QMessageBox::critical(this, tr("Error"),
tr("Cannot start Qucs-RFLayout: \n%1")
.arg(QucsSettings.RFLayoutExecutable));
delete tool;
return;
}
connect(this, SIGNAL(signalKillEmAll()), tool, SLOT(kill()));
}
// --------------------------------------------------------------
void QucsApp::slotHelpIndex()
{
QDesktopServices::openUrl(QUrl("https://qucs-s-help.readthedocs.io/"));
}
void QucsApp::slotHelpQucsIndex()
{
QDesktopServices::openUrl(QUrl("https://qucs-help.readthedocs.io/"));
}
// --------------------------------------------------------------
void QucsApp::slotGettingStarted()
{
QDesktopServices::openUrl(QUrl("https://ra3xdh.github.io/pdf/qucs_s_tutorial.pdf"));
}
// ---------------------------------------------------------------------
// Is called when the find action is triggered.
void QucsApp::slotEditFind()
{
SearchDia->initSearch(DocumentTab->currentWidget(),
((TextDoc *)DocumentTab->currentWidget())->textCursor().selectedText(), false);
}
// --------------------------------------------------------------
void QucsApp::slotChangeProps()
{
QWidget *Doc = DocumentTab->currentWidget();
if(isTextDocument(Doc)) {
((TextDoc*)Doc)->viewport()->setFocus();
SearchDia->initSearch(Doc,
((TextDoc *)Doc)->textCursor().selectedText(), true);
}
else {
ChangeDialog *d = new ChangeDialog((Schematic*)Doc);
if(d->exec() == QDialog::Accepted) {
((Schematic*)Doc)->setChanged(true, true);
((Schematic*)Doc)->viewport()->update();
}
}
}
// --------------------------------------------------------------
void QucsApp::slotAddToProject()
{
slotHideEdit(); // disable text edit of component property
if(ProjName.isEmpty()) {
QMessageBox::critical(this, tr("Error"), tr("No project open!"));
return;
}
QStringList List = QFileDialog::getOpenFileNames(this, tr("Select files to copy"),
lastDir.isEmpty() ? QString(".") : lastDir, QucsFileFilter);
if(List.isEmpty()) {
statusBar()->showMessage(tr("No files copied."), 2000);
return;
}
char *Buffer = (char*)malloc(0x10000);
if(!Buffer) return; // should never happen
QStringList FileList = List; // make a copy as recommended by Qt
QStringList::Iterator it = FileList.begin();
QFileInfo Info(*it);
lastDir = Info.absolutePath(); // remember last directory
// copy all files to project directory
int Num;
QFile origFile, destFile;
while(it != FileList.end()) {
Info.setFile(*it);
origFile.setFileName(*it);
destFile.setFileName(QucsSettings.QucsWorkDir.absolutePath() +
QDir::separator() + Info.fileName());
if(!origFile.open(QIODevice::ReadOnly)) {
QMessageBox::critical(this, tr("Error"), tr("Cannot open \"%1\" !").arg(*it));
it++;
continue;
}
if(destFile.exists())
if(QMessageBox::information(this, tr("Overwrite"),
tr("File \"%1\" already exists.\nOverwrite ?").arg(*it),
QMessageBox::Yes|QMessageBox::No)
!= QMessageBox::Yes) {
origFile.close();
it++;
continue;
}
if(!destFile.open(QIODevice::WriteOnly)) {
QMessageBox::critical(this, tr("Error"), tr("Cannot create \"%1\" !").arg(*it));
origFile.close();
it++;
continue;
}
// copy data
do {
Num = origFile.read(Buffer, 0x10000);
if(Num < 0) {
QMessageBox::critical(this, tr("Error"), tr("Cannot read \"%1\" !").arg(*it));
break;
}
Num = destFile.write(Buffer, Num);
if(Num < 0) {
QMessageBox::critical(this, tr("Error"), tr("Cannot write \"%1\" !").arg(*it));
break;
}
} while(Num == 0x10000);
origFile.close();
destFile.close();
it++;
}
free(Buffer);
slotUpdateTreeview();
statusBar()->showMessage(tr("Ready."));
}
// -----------------------------------------------------------
void QucsApp::slotCursorLeft(bool left)
{
int sign = 1;
if(left){
sign = -1;
}
if(!editText->isHidden()) return; // for edit of component property ?
Q3PtrList<Element> movingElements;
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
int markerCount = Doc->copySelectedElements(&movingElements);
if((movingElements.count() - markerCount) < 1) {
if(markerCount > 0) { // only move marker if nothing else selected
Doc->markerLeftRight(left, &movingElements);
} else if(left) {
Doc->scrollLeft(Doc->horizontalScrollBar()->singleStep());
}else{ // right
Doc->scrollRight(Doc->horizontalScrollBar()->singleStep());
}
Doc->viewport()->update();
return;
} else { // random selection. move all of them
view->moveElements(&movingElements, sign*Doc->GridX, 0);
view->MAx3 = 1; // sign for moved elements
view->endElementMoving(Doc, &movingElements);
}
}
// -----------------------------------------------------------
void QucsApp::slotCursorUp(bool up)
{
if(editText->isHidden()) { // for edit of component property ?
}else if(up){
if(view->MAx3 == 0) return; // edit component namen ?
Component *pc = (Component*)view->focusElement;
Property *pp = pc->Props.at(view->MAx3-1); // current property
int Begin = pp->Description.indexOf('[');
if(Begin < 0) return; // no selection list ?
int End = pp->Description.indexOf(editText->text(), Begin); // current
if(End < 0) return; // should never happen
End = pp->Description.lastIndexOf(',', End);
if(End < Begin) return; // was first item ?
End--;
int Pos = pp->Description.lastIndexOf(',', End);
if(Pos < Begin) Pos = Begin; // is first item ?
Pos++;
if(pp->Description.at(Pos) == ' ') Pos++; // remove leading space
editText->setText(pp->Description.mid(Pos, End-Pos+1));
editText->selectAll();
return;
}else{ // down
if(view->MAx3 == 0) return; // edit component namen ?
Component *pc = (Component*)view->focusElement;
Property *pp = pc->Props.at(view->MAx3-1); // current property
int Pos = pp->Description.indexOf('[');
if(Pos < 0) return; // no selection list ?
Pos = pp->Description.indexOf(editText->text(), Pos); // current list item
if(Pos < 0) return; // should never happen
Pos = pp->Description.indexOf(',', Pos);
if(Pos < 0) return; // was last item ?
Pos++;
if(pp->Description.at(Pos) == ' ') Pos++; // remove leading space
int End = pp->Description.indexOf(',', Pos);
if(End < 0) { // is last item ?
End = pp->Description.indexOf(']', Pos);
if(End < 0) return; // should never happen
}
editText->setText(pp->Description.mid(Pos, End-Pos));
editText->selectAll();
return;
}
Q3PtrList<Element> movingElements;
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
int markerCount = Doc->copySelectedElements(&movingElements);
if((movingElements.count() - markerCount) < 1) { // all selections are markers
if(markerCount > 0) { // only move marker if nothing else selected
Doc->markerUpDown(up, &movingElements);
} else if(up) { // nothing selected at all
Doc->scrollUp(Doc->verticalScrollBar()->singleStep());
} else { // down
Doc->scrollDown(Doc->verticalScrollBar()->singleStep());
}
Doc->viewport()->update();
return;
}else{ // some random selection, put it back
view->moveElements(&movingElements, 0, ((up)?-1:1) * Doc->GridY);
view->MAx3 = 1; // sign for moved elements
view->endElementMoving(Doc, &movingElements);
}
}
// -----------------------------------------------------------
// Is called if user clicked on component text of if return is
// pressed in the component text QLineEdit.
// In "view->MAx3" is the number of the current property.
void QucsApp::slotApplyCompText()
{
QFont f = QucsSettings.font;
Schematic *Doc = (Schematic*)DocumentTab->currentWidget();
f.setPointSizeF( Doc->Scale * float(f.pointSize()) );
editText->setFont(f);
Component *const component = dynamic_cast<Component*>(view->focusElement);
if(!component) return; // should never happen
view->MAx1 = component->cx + component->tx;
view->MAy1 = component->cy + component->ty;
// Here is a bit of *magic* and implicit coupling: the value of view->MAx3
// comes from Component::getTextSelected, and it's equal 0 when component
// name is clicked, N + 1 when Nth component property is clicked.
const int component_text_index = view->MAx3;
const bool is_name = component_text_index == 0;
Property *const component_property = !is_name
? component->Props.at(component_text_index - 1)
: nullptr;
if (editText->isVisible()) { // is called the first time ?
if (is_name) {
const auto new_name{editText->text()};
if (!new_name.isEmpty() && component->Name != new_name) {
// TODO: rewrite with std::none_of after replacing Q3PtrList
// with modern container
bool is_unique = true;
for (auto* other : *Doc->Components) {
if (other->Name == new_name) {
is_unique = false;
break;
}
}
if (is_unique) {
component->Name = new_name;
Doc->setChanged(true, true); // only one undo state
}
}
}
else if (component_property) { // property was applied
if (component_property->Value != editText->text()) {
component_property->Value = editText->text();
Doc->recreateComponent(component); // because of "Num" and schematic symbol
Doc->setChanged(true, true); // only one undo state
}
}
}
const QString s = is_name
? component->Name
: component_property->Value;
editText->setReadOnly(false);
QPoint editTextTopLeft;
if (component_property) { // is it a property ?
editTextTopLeft = Doc->modelToViewport(QPoint{component->cx, component->cy} + component_property->boundingRect().topLeft());
editTextTopLeft.rx() += editText->fontMetrics().boundingRect(component_property->Name + "=" + '\u0020').width();
if(component_property->Description.indexOf('[') >= 0) // is selection list ?
editText->setReadOnly(true);
Expr_CompProp.setPattern("[^\"]*");
}
else { // it is the component name
Expr_CompProp.setPattern("[\\w_]+");
editTextTopLeft = Doc->modelToViewport(QPoint{component->cx + component->tx, component->cy + component->ty});
}
{
auto size = editText->fontMetrics().boundingRect(s ).size();
size.rwidth() += editText->fontMetrics().averageCharWidth();
editText->setFixedSize(size);
}
view->MAx2 = editTextTopLeft.x();
view->MAy2 = editTextTopLeft.y();
Val_CompProp.setRegularExpression(Expr_CompProp);
editText->setValidator(&Val_CompProp);
editText->setText(s);
editText->setStyleSheet("color: black; background-color: " + QucsSettings.BGColor.name());
editText->setFocus();
editText->selectAll();
editText->setParent(Doc->viewport());
editText->move(view->MAx2, view->MAy2);
editText->show();
//editText->reparent(Doc->viewport(), 0, QPoint(view->MAx2, view->MAy2), true);
}
// -----------------------------------------------------------
// Is called if the text of the property edit changed, to match
// the width of the edit field.
void QucsApp::slotResizePropEdit(const QString& t)
{
editText->resize(editText->fontMetrics().boundingRect(t).width()+4,
editText->fontMetrics().lineSpacing());
}
// -----------------------------------------------------------
void QucsApp::slotCreateLib()
{
slotHideEdit(); // disable text edit of component property
if(ProjName.isEmpty()) {
QMessageBox::critical(this, tr("Error"), tr("Please open project with subcircuits!"));
return;
}
LibraryDialog *d = new LibraryDialog(this);
d->fillSchematicList(Content->exportSchematic());
d->exec();
}
// -----------------------------------------------------------
void QucsApp::slotImportData()
{
slotHideEdit(); // disable text edit of component property
QString import_dir = ".";
if (!ProjName.isEmpty()) {
import_dir = QucsSettings.QucsWorkDir.absolutePath();
} else {
QString dname;
if (isTextDocument(DocumentTab->currentWidget())) {
TextDoc *doc = (TextDoc *)DocumentTab->currentWidget();
dname = doc->DocName;
} else {
Schematic *doc = (Schematic *)DocumentTab->currentWidget();
dname = doc->DocName;
}
QFileInfo inf(dname);
if (inf.exists()) {
import_dir = inf.absolutePath();
}
}
ImportDialog *d = new ImportDialog(this);
d->setImportDir(import_dir);
if(d->exec() == QDialog::Accepted)
slotUpdateTreeview();
}
// -----------------------------------------------------------
void QucsApp::slotExportGraphAsCsv()
{
slotHideEdit(); // disable text edit of component property
for(;;) {
if(view->focusElement)
if(view->focusElement->Type == isGraph)
break;
QMessageBox::critical(this, tr("Error"), tr("Please select a diagram graph!"));
return;
}
/*QString s = Q3FileDialog::getSaveFileName(
lastDir.isEmpty() ? QString(".") : lastDir,
tr("CSV file")+" (*.csv);;" + tr("Any File")+" (*)",
this, 0, tr("Enter an Output File Name"));
*/
QString s = QFileDialog::getSaveFileName(this, tr("Enter an Output File Name"),
lastDir.isEmpty() ? QString(".") : lastDir, tr("CSV file")+" (*.csv);;" + tr("Any File")+" (*)");
if(s.isEmpty())
return;
QFileInfo Info(s);
lastDir = Info.absolutePath(); // remember last directory
if(Info.suffix().isEmpty())
s += ".csv";
QFile File(s);
if(File.exists())
if(QMessageBox::information(this, tr("Info"),
tr("Output file already exists!")+"\n"+tr("Overwrite it?"),
QMessageBox::Yes|QMessageBox::No) == QMessageBox::No)
return;
if(!File.open(QIODevice::WriteOnly)) {
QMessageBox::critical(this, QObject::tr("Error"),
QObject::tr("Cannot create output file!"));
return;
}
QTextStream Stream(&File);
DataX const *pD;
Graph *g = (Graph*)view->focusElement;
// First output the names of independent and dependent variables.
for(unsigned ii=0; (pD=g->axis(ii)); ++ii)
Stream << '\"' << pD->Var << "\";";
Stream << "\"r " << g->Var << "\";\"i " << g->Var << "\"\n";
int n, m;
double *py = g->cPointsY;
int Count = g->countY * g->axis(0)->count;
for(n = 0; n < Count; n++) {
m = n;
for(unsigned ii=0; (pD=g->axis(ii)); ++ii) {
Stream << *(pD->Points + m%pD->count) << ';';
m /= pD->count;
}
Stream << *(py) << ';' << *(py+1) << '\n';
py += 2;
}
File.close();
}
void QucsApp::slotOpenRecent()
{
QAction *action = qobject_cast<QAction *>(sender());
if (action) {
gotoPage(action->data().toString());
updateRecentFilesList(action->data().toString());
}
}
void QucsApp::slotUpdateRecentFiles()
{
QMutableStringListIterator it(QucsSettings.RecentDocs);
while(it.hasNext()) {
if (!QFile::exists(it.next())) {
it.remove();
}
}
for (int i = 0; i < MaxRecentFiles; ++i) {
if (i < QucsSettings.RecentDocs.size()) {
fileRecentAction[i]->setText(QucsSettings.RecentDocs[i]);
fileRecentAction[i]->setData(QucsSettings.RecentDocs[i]);
fileRecentAction[i]->setVisible(true);
} else {
fileRecentAction[i]->setVisible(false);
}
}
}
void QucsApp::slotClearRecentFiles()
{
QucsSettings.RecentDocs.clear();
slotUpdateRecentFiles();
}
/*!
* \brief QucsApp::slotLoadModule launches the dialog to select dynamic modueles
*/
void QucsApp::slotLoadModule()
{
qDebug() << "slotLoadModule";
LoadDialog *ld = new LoadDialog(this);
ld->setApp(this);
// fech list of _symbol.json
// \todo fetch timestamp of VA, JSON, if VA newer, need to reload.
QDir projDir = QucsSettings.QucsWorkDir.absolutePath();
QStringList files;
QString fileSuffix = "*_symbol.json";
files = projDir.entryList(QStringList(fileSuffix),
QDir::Files | QDir::NoSymLinks);
// no JSON files or no a project?
if (!files.size()){
QMessageBox::critical(this, tr("Error"),
tr("Symbol files not found in: %1\n\n"
"Is the project open?\n"
"Have you saved the Verilog-A symbols?")
.arg(QString(projDir.absolutePath())));
return;
}
// initialize dialog
// pass list of potential symbol files
ld->symbolFiles << files;
ld->projDir = projDir;
ld->initDialog();
// \todo check what is already loaded, offer skip, reload
//pass stuff to ld dialog
// run, let user do the selections
if (ld->exec() == QDialog::Accepted) {
Module::vaComponents = ld->selectedComponents;
// dialog write new bitmap into JSON
// load, unload, reload
// inform if symbol changed
// populate Module::vaComponents
// vaComponents are selected with the dialog
// dialog should populate according to checkboxes
// build vaComponents QMap
// remove all previously registered modules
QMutableHashIterator<QString, Module *> it( Module::Modules );
while(it.hasNext()) {
it.next();
if (it.value()->category == QObject::tr("verilog-a user devices")) {
it.remove();
}
}
if (! Module::vaComponents.isEmpty()) {
// Register whatever is in Module::vaComponents
//Module::registerDynamicComponents();
// update the combobox, set new category in view
// pick up new category 'verilog-a user components' from `Module::category`
//set new category into view
QucsApp::fillComboBox(true);
CompChoose->setCurrentIndex(CompChoose->count()-1);
slotSetCompView(CompChoose->count()-1);
// icons of dynamically registered components ready to be dragged
}
else {
// remove any previously registered icons from the listview
int foundCat = CompChoose->findText(QObject::tr("verilog-a user devices"));
if (foundCat != -1) {
CompChoose->setCurrentIndex(foundCat);
CompComps->clear();
}
}
}
delete ld;
}
/*!
* \brief QucsApp::slotBuildModule runs admsXml, C++ compiler to build library
*
* Run the va2cpp
* Run the cpp2lib
*
* TODO
* - split into two actions, elaborate and compile?
* - collect, parse and display output of make
*
*/
void QucsApp::slotBuildModule()
{
qDebug() << "slotBuildModule";
// reset message dock on entry
messageDock->reset();
if (QucsSettings.DefaultSimulator == spicecompat::simNgspice) {
buildWithOpenVAF();
return;
}
messageDock->builderTabs->setTabIcon(0,QPixmap());
messageDock->builderTabs->setTabText(0,tr("admsXml"));
messageDock->builderTabs->setTabIcon(1,QPixmap());
messageDock->builderTabs->setTabText(1,tr("Compiler"));
messageDock->msgDock->setWindowTitle(tr("admsXml Dock"));
QString make;
#if defined(_WIN32) || defined(__MINGW32__)
make = "mingw32-make.exe"; // must be on the path!
#else
make = "make"; // must be on the path!
#endif
QFileInfo inf(QucsSettings.Qucsator);
QString QucsatorPath = inf.path()+QDir::separator();
QDir prefix = QDir(QucsatorPath+"../");
QDir include = QDir(QucsatorPath+"../include/qucs-core");
QString workDir = QucsSettings.QucsWorkDir.absolutePath();
// need to cd into project to make sure output is dropped there?
// need to cd - into previous location?
QDir::setCurrent(workDir);
QProcess *builder = new QProcess();
builder->setProcessChannelMode(QProcess::MergedChannels);
// get current va document
QucsDoc *Doc = getDoc();
QString vaModule = Doc->fileBase(Doc->DocName);
QString admsXml = QucsSettings.AdmsXmlBinDir.canonicalPath();
#if defined(_WIN32) || defined(__MINGW32__)
admsXml = QDir::toNativeSeparators(admsXml+"/"+"admsXml.exe");
#else
admsXml = QDir::toNativeSeparators(admsXml+"/"+"admsXml");
#endif
// admsXml emits C++
QStringList Arguments;
Arguments << "-f" << QDir::toNativeSeparators(include.absoluteFilePath("va2cpp.makefile"))
<< QString("ADMSXML=%1").arg(admsXml)
<< QString("PREFIX=%1").arg(QDir::toNativeSeparators(prefix.absolutePath()))
<< QString("MODEL=%1").arg(vaModule);
QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
env.insert("PATH", env.value("PATH") );
builder->setProcessEnvironment(env);
// prepend command to log
QString cmdString = QString("%1 %2\n").arg(make, Arguments.join(" "));
messageDock->admsOutput->appendPlainText(cmdString);
qDebug() << "Command :" << make << Arguments.join(" ");
builder->start(make, Arguments);
// admsXml seems to communicate all via stdout, or is it because of make?
QString vaStatus;
if (!builder->waitForFinished()) {
vaStatus = builder->errorString();
qDebug() << "Make failed:" << vaStatus;
}
else {
vaStatus = builder->readAll();
qDebug() << "Make stdout" << vaStatus;
}
//build libs
qDebug() << "\nbuild libs\n";
Arguments.clear();
Arguments << "-f" << QDir::toNativeSeparators(include.absoluteFilePath("cpp2lib.makefile"))
<< QString("PREFIX=\"%1\"").arg(QDir::toNativeSeparators(prefix.absolutePath()))
<< QString("PROJDIR=\"%1\"").arg(QDir::toNativeSeparators(workDir))
<< QString("MODEL=%1").arg(vaModule);
// prepend command to log
cmdString = QString("%1 %2\n").arg(make, Arguments.join(" "));
messageDock->cppOutput->appendPlainText(cmdString);
builder->start(make, Arguments);
QString cppStatus;
if (!builder->waitForFinished()) {
cppStatus = builder->errorString();
qDebug() << "Make failed:" << cppStatus;
}
else {
cppStatus = builder->readAll();
qDebug() << "Make output:" << cppStatus;
}
delete builder;
// push make output to message dock
messageDock->admsOutput->appendPlainText(vaStatus);
messageDock->cppOutput->appendPlainText(cppStatus);
// shot the message docks
messageDock->msgDock->show();
}
void QucsApp::buildWithOpenVAF()
{
messageDock->builderTabs->setTabIcon(0,QPixmap());
messageDock->builderTabs->setTabText(0,tr("OpenVAF"));
messageDock->msgDock->setWindowTitle(tr("OpenVAF Dock"));
QString workDir = QucsSettings.QucsWorkDir.absolutePath();
QDir::setCurrent(workDir);
QProcess *builder = new QProcess();
builder->setProcessChannelMode(QProcess::MergedChannels);
// get current va document
QucsDoc *Doc = getDoc();
QString vaModule = Doc->DocName;
QString openVAF = QucsSettings.OpenVAFExecutable;
// admsXml emits C++
QStringList Arguments;
Arguments<<vaModule;
QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
env.insert("PATH", env.value("PATH") );
builder->setProcessEnvironment(env);
// prepend command to log
QString cmdString = QString("%1 %2\n").arg(openVAF, Arguments.join(" "));
messageDock->admsOutput->appendPlainText(cmdString);
qDebug() << "Command :" << openVAF << Arguments.join(" ");
builder->start(openVAF, Arguments);
// admsXml seems to communicate all via stdout, or is it because of make?
QString vaStatus;
if (!builder->waitForFinished()) {
vaStatus = builder->errorString();
qDebug() << "OpenVAF failed:" << vaStatus;
}
else {
vaStatus = builder->readAll();
qDebug() << "OpenVAF stdout" << vaStatus;
}
delete builder;
// push make output to message dock
messageDock->admsOutput->appendPlainText(vaStatus);
// shot the message docks
messageDock->msgDock->show();
}
// ----------------------------------------------------------
void QucsApp::slotHelpAbout()
{
AboutDialog *ad = new AboutDialog(this);
ad->exec();
}
// vim:ts=8:sw=2:noet