mirror of
https://github.com/ra3xdh/qucs_s
synced 2025-03-28 21:13:26 +00:00

-Introduced cli parameter --cdl for netlisting CDL -Implemented netlisting to console for ngspice and xyce -Increased readability and introduced smart-pointer semantics for some qucs-s main.cpp functions -Increased readability and introduced c++ cast's for casting to Schematic* for affected QucsApp::slotSimulateWithSpice and QucsApp::slotSaveNetlist Signed-off-by: ThomasZecha <zecha@ihp-microelectronics.com>
1238 lines
44 KiB
C++
1238 lines
44 KiB
C++
/***************************************************************************
|
|
main.cpp
|
|
----------
|
|
begin : Thu Aug 28 2003
|
|
copyright : (C) 2003 by Michael Margraf
|
|
email : michael.margraf@alumni.tu-berlin.de
|
|
***************************************************************************/
|
|
|
|
/***************************************************************************
|
|
* *
|
|
* 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 main.cpp
|
|
* \brief Implementation of the main application.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include <config.h>
|
|
#endif
|
|
|
|
#include <stdlib.h>
|
|
#include <ctype.h>
|
|
#include <locale.h>
|
|
#include <iostream>
|
|
|
|
#include <QApplication>
|
|
#include <QString>
|
|
#include <QStringList>
|
|
//#include <QTextCodec>
|
|
#include <QTranslator>
|
|
#include <QFile>
|
|
#include <QMessageBox>
|
|
#include <QRegularExpression>
|
|
#include <QtSvg>
|
|
#include <QCommandLineParser>
|
|
#include <QTextStream>
|
|
#include <QScopedPointer>
|
|
|
|
#include "qucs.h"
|
|
#include "main.h"
|
|
#include "node.h"
|
|
#include "printerwriter.h"
|
|
#include "imagewriter.h"
|
|
#include "schematic.h"
|
|
#include "settings.h"
|
|
#include "module.h"
|
|
#include "misc.h"
|
|
|
|
|
|
#include "extsimkernels/ngspice.h"
|
|
#include "extsimkernels/xyce.h"
|
|
#include "extsimkernels/CdlNetlistWriter.h"
|
|
|
|
#if defined(_WIN32) ||defined(__MINGW32__)
|
|
#include <windows.h> //for OutputDebugString
|
|
#endif
|
|
|
|
#if defined(_WIN32) ||defined(__MINGW32__)
|
|
#define executableSuffix ".exe"
|
|
#else
|
|
#define executableSuffix ""
|
|
#endif
|
|
|
|
tQucsSettings QucsSettings;
|
|
|
|
QucsApp *QucsMain = nullptr; // the Qucs application itself
|
|
QString lastDir; // to remember last directory for several dialogs
|
|
QStringList qucsPathList;
|
|
VersionTriplet QucsVersion; // Qucs version string
|
|
|
|
// #########################################################################
|
|
// Loads the settings file and stores the settings.
|
|
bool loadSettings()
|
|
{
|
|
QSettings settings("qucs","qucs_s");
|
|
|
|
QucsSettings.DefaultSimulator = _settings::Get().item<int>("DefaultSimulator");
|
|
QucsSettings.firstRun = _settings::Get().item<bool>("firstRun");
|
|
|
|
/*** Temporarily continue to use QucsSettings to make sure all settings convert okay and remain compatible ***/
|
|
QucsSettings.font.fromString(_settings::Get().item<QString>("font"));
|
|
QucsSettings.appFont.fromString(_settings::Get().item<QString>("appFont"));
|
|
QucsSettings.textFont.fromString(_settings::Get().item<QString>("textFont"));
|
|
QucsSettings.largeFontSize = _settings::Get().item<double>("LargeFontSize");
|
|
QucsSettings.maxUndo = _settings::Get().item<int>("maxUndo");
|
|
QucsSettings.NodeWiring = _settings::Get().item<int>("NodeWiring");
|
|
QucsSettings.BGColor = _settings::Get().item<QString>("BGColor");
|
|
QucsSettings.Editor = _settings::Get().item<QString>("Editor");
|
|
QucsSettings.FileTypes = _settings::Get().item<QStringList>("FileTypes");
|
|
QucsSettings.Language = _settings::Get().item<QString>("Language");
|
|
|
|
// Editor syntax highlighting settings.
|
|
QucsSettings.Comment = _settings::Get().item<QString>("Comment");
|
|
QucsSettings.String = _settings::Get().item<QString>("String");
|
|
QucsSettings.Integer = _settings::Get().item<QString>("Integer");
|
|
QucsSettings.Real = _settings::Get().item<QString>("Real");
|
|
QucsSettings.Character = _settings::Get().item<QString>("Character");
|
|
QucsSettings.Type = _settings::Get().item<QString>("Type");
|
|
QucsSettings.Attribute = _settings::Get().item<QString>("Attribute");
|
|
QucsSettings.Directive = _settings::Get().item<QString>("Directive");
|
|
QucsSettings.Task = _settings::Get().item<QString>("Task");
|
|
|
|
// TODO: Convert this to the new settings model.
|
|
if(settings.contains("Qucsator")) {
|
|
QucsSettings.Qucsator = settings.value("Qucsator").toString();
|
|
QFileInfo inf(QucsSettings.Qucsator);
|
|
QucsSettings.QucsatorDir = inf.canonicalPath() + QDir::separator();
|
|
if (QucsSettings.Qucsconv.isEmpty())
|
|
QucsSettings.Qucsconv = QStandardPaths::findExecutable("qucsconv_rf",{QucsSettings.QucsatorDir});
|
|
} else {
|
|
QucsSettings.Qucsator = QStandardPaths::findExecutable("qucsator_rf",{QucsSettings.BinDir});
|
|
QucsSettings.QucsatorDir = QucsSettings.BinDir;
|
|
if (QucsSettings.Qucsconv.isEmpty())
|
|
QucsSettings.Qucsconv = QStandardPaths::findExecutable("qucsconv_rf",{QucsSettings.BinDir});
|
|
}
|
|
|
|
QucsSettings.AdmsXmlBinDir.setPath(_settings::Get().item<QString>("AdmsXmlBinDir"));
|
|
QucsSettings.AscoBinDir.setPath(_settings::Get().item<QString>("AscoBinDir"));
|
|
QucsSettings.NgspiceExecutable = _settings::Get().item<QString>("NgspiceExecutable");
|
|
QucsSettings.XyceExecutable = _settings::Get().item<QString>("XyceExecutable");
|
|
QucsSettings.XyceParExecutable = _settings::Get().item<QString>("XyceParExecutable");
|
|
QucsSettings.SpiceOpusExecutable = _settings::Get().item<QString>("SpiceOpusExecutable");
|
|
QucsSettings.NProcs = _settings::Get().item<int>("Nprocs");
|
|
|
|
// TODO: Currently the default settings cannot include other settings during initialisation. This is a
|
|
// problem for this setting as it needs to include the QucsWorkDir setting. Therefore, set the default to an
|
|
// empty string and populate it here by brute force.
|
|
QucsSettings.S4Qworkdir = _settings::Get().item<QString>("S4Q_workdir");
|
|
if (QucsSettings.S4Qworkdir == "")
|
|
QucsSettings.S4Qworkdir = QDir::toNativeSeparators(QucsSettings.QucsWorkDir.absolutePath()+"/spice4qucs");
|
|
|
|
QucsSettings.OctaveExecutable = _settings::Get().item<QString>("OctaveExecutable");
|
|
QucsSettings.OpenVAFExecutable = _settings::Get().item<QString>("OpenVAFExecutable");
|
|
|
|
QucsSettings.RFLayoutExecutable = _settings::Get().item<QString>("RFLayoutExecutable");
|
|
|
|
QucsSettings.qucsWorkspaceDir.setPath(_settings::Get().item<QString>("QucsHomeDir"));
|
|
QucsSettings.QucsWorkDir = QucsSettings.qucsWorkspaceDir;
|
|
QucsSettings.tempFilesDir.setPath(QStandardPaths::writableLocation(QStandardPaths::CacheLocation));
|
|
|
|
QucsSettings.IgnoreFutureVersion = _settings::Get().item<bool>("IgnoreVersion");
|
|
QucsSettings.GraphAntiAliasing = _settings::Get().item<bool>("GraphAntiAliasing");
|
|
QucsSettings.TextAntiAliasing = _settings::Get().item<bool>("TextAntiAliasing");
|
|
QucsSettings.fullTraceName = _settings::Get().item<bool>("fullTraceName");
|
|
QucsSettings.RecentDocs = _settings::Get().item<QString>("RecentDocs").split("*", Qt::SkipEmptyParts);
|
|
QucsSettings.numRecentDocs = QucsSettings.RecentDocs.count();
|
|
QucsSettings.spiceExtensions << "*.sp" << "*.cir" << "*.spc" << "*.spi";
|
|
|
|
// If present read in the list of directory paths in which Qucs should
|
|
// search for subcircuit schematics
|
|
int npaths = settings.beginReadArray("Paths");
|
|
for (int i = 0; i < npaths; ++i)
|
|
{
|
|
settings.setArrayIndex(i);
|
|
QString apath = settings.value("path").toString();
|
|
qucsPathList.append(apath);
|
|
}
|
|
settings.endArray();
|
|
|
|
QucsSettings.numRecentDocs = 0;
|
|
|
|
return true;
|
|
}
|
|
|
|
// #########################################################################
|
|
// Saves the settings in the settings file.
|
|
bool saveApplSettings()
|
|
{
|
|
QSettings settings ("qucs","qucs_s");
|
|
|
|
// Note: It is not really necessary to take the following reference, but it
|
|
// arguably makes the code slightly cleaner - thoughts? To be clear:
|
|
// qs.item<int>() is identical to _settings::get().item<int>()
|
|
settingsManager& qs = _settings::Get();
|
|
|
|
qs.setItem<int>("DefaultSimulator", QucsSettings.DefaultSimulator);
|
|
qs.setItem<bool>("firstRun", false);
|
|
qs.setItem<QString>("font", QucsSettings.font.toString());
|
|
qs.setItem<QString>("appFont", QucsSettings.appFont.toString());
|
|
qs.setItem<QString>("textFont", QucsSettings.textFont.toString());
|
|
if (QucsMain != nullptr) {
|
|
qs.setItem<QByteArray>("MainWindowGeometry", QucsMain->saveGeometry());
|
|
}
|
|
|
|
// store LargeFontSize as a string, so it will be also human-readable in the settings file (will be a @Variant() otherwise)
|
|
qs.setItem<QString>("LargeFontSize", QString::number(QucsSettings.largeFontSize));
|
|
qs.setItem<unsigned int>("maxUndo", QucsSettings.maxUndo);
|
|
qs.setItem<unsigned int>("NodeWiring", QucsSettings.NodeWiring);
|
|
qs.setItem<QString>("BGColor", QucsSettings.BGColor.name());
|
|
qs.setItem<QString>("Editor", QucsSettings.Editor);
|
|
qs.setItem<QStringList>("FileTypes", QucsSettings.FileTypes);
|
|
qs.setItem<QString>("Language", QucsSettings.Language);
|
|
qs.setItem<QString>("Comment", QucsSettings.Comment.name());
|
|
qs.setItem<QString>("String", QucsSettings.String.name());
|
|
qs.setItem<QString>("Integer", QucsSettings.Integer.name());
|
|
qs.setItem<QString>("Real", QucsSettings.Real.name());
|
|
qs.setItem<QString>("Character", QucsSettings.Character.name());
|
|
qs.setItem<QString>("Type", QucsSettings.Type.name());
|
|
qs.setItem<QString>("Attribute", QucsSettings.Attribute.name());
|
|
qs.setItem<QString>("Directive", QucsSettings.Directive.name());
|
|
qs.setItem<QString>("Task", QucsSettings.Task.name());
|
|
qs.setItem<QString>("AdmsXmlBinDir", QucsSettings.AdmsXmlBinDir.canonicalPath());
|
|
qs.setItem<QString>("AscoBinDir", QucsSettings.AscoBinDir.canonicalPath());
|
|
qs.setItem<QString>("NgspiceExecutable",QucsSettings.NgspiceExecutable);
|
|
qs.setItem<QString>("XyceExecutable",QucsSettings.XyceExecutable);
|
|
qs.setItem<QString>("XyceParExecutable",QucsSettings.XyceParExecutable);
|
|
qs.setItem<QString>("SpiceOpusExecutable",QucsSettings.SpiceOpusExecutable);
|
|
qs.setItem<QString>("Qucsator",QucsSettings.Qucsator);
|
|
qs.setItem<int>("Nprocs",QucsSettings.NProcs);
|
|
qs.setItem<QString>("S4Q_workdir",QucsSettings.S4Qworkdir);
|
|
qs.setItem<QString>("OctaveExecutable",QucsSettings.OctaveExecutable);
|
|
qs.setItem<QString>("OpenVAFExecutable",QucsSettings.OpenVAFExecutable);
|
|
qs.setItem<QString>("QucsHomeDir", QucsSettings.qucsWorkspaceDir.canonicalPath());
|
|
qs.setItem<bool>("IgnoreVersion", QucsSettings.IgnoreFutureVersion);
|
|
qs.setItem<bool>("GraphAntiAliasing", QucsSettings.GraphAntiAliasing);
|
|
qs.setItem<bool>("TextAntiAliasing", QucsSettings.TextAntiAliasing);
|
|
qs.setItem<bool>("fullTraceName",QucsSettings.fullTraceName);
|
|
|
|
// Copy the list of directory paths in which Qucs should
|
|
// search for subcircuit schematics from qucsPathList
|
|
settings.remove("Paths");
|
|
settings.beginWriteArray("Paths");
|
|
int i = 0;
|
|
for (QString& path: qucsPathList) {
|
|
settings.setArrayIndex(i);
|
|
settings.setValue("path", path);
|
|
i++;
|
|
}
|
|
settings.endArray();
|
|
|
|
return true;
|
|
}
|
|
|
|
/*!
|
|
* \brief qucsMessageOutput handles qDebug, qWarning, qCritical, qFatal.
|
|
* \param type Message type (Qt enum)
|
|
* \param msg Message
|
|
*
|
|
* The message handler is used to get control of the messages.
|
|
* Particularly on Windows, as the messages are sent to the debugger and do not
|
|
* show on the terminal. The handler could also be extended to create a log
|
|
* mechanism.
|
|
* <http://qt-project.org/doc/qt-4.8/debug.html#warning-and-debugging-messages>
|
|
* <http://qt-project.org/doc/qt-4.8/qtglobal.html#qInstallMsgHandler>
|
|
*/
|
|
void qucsMessageOutput(QtMsgType type, const QMessageLogContext &context, const QString &msg)
|
|
{
|
|
QByteArray localMsg = msg.toLocal8Bit();
|
|
const char *file = context.file ? context.file : "";
|
|
const char *function = context.function ? context.function : "";
|
|
switch (type) {
|
|
case QtDebugMsg:
|
|
fprintf(stderr, "Debug: %s (%s:%u, %s)\n", localMsg.constData(), file, context.line, function);
|
|
break;
|
|
case QtInfoMsg:
|
|
fprintf(stderr, "Info: %s (%s:%u, %s)\n", localMsg.constData(), file, context.line, function);
|
|
break;
|
|
case QtWarningMsg:
|
|
fprintf(stderr, "Warning: %s (%s:%u, %s)\n", localMsg.constData(), file, context.line, function);
|
|
break;
|
|
case QtCriticalMsg:
|
|
fprintf(stderr, "Critical: %s (%s:%u, %s)\n", localMsg.constData(), file, context.line, function);
|
|
break;
|
|
case QtFatalMsg:
|
|
fprintf(stderr, "Fatal: %s (%s:%u, %s)\n", localMsg.constData(), file, context.line, function);
|
|
break;
|
|
}
|
|
fflush(stderr);
|
|
}
|
|
|
|
Schematic* openSchematic(const QString& schematicFileName)
|
|
{
|
|
qDebug() << "*** try to load schematic :" << schematicFileName;
|
|
|
|
QFile file(schematicFileName); // save simulator messages
|
|
if (file.open(QIODevice::ReadOnly))
|
|
{
|
|
file.close();
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "Error: Could not load schematic %s\n", schematicFileName.toLatin1().data());
|
|
return nullptr;
|
|
}
|
|
|
|
// populate Modules list
|
|
//Module::registerModules ();
|
|
|
|
// new schematic from file
|
|
Schematic *schematic = new Schematic(nullptr, schematicFileName);
|
|
|
|
// load schematic file if possible
|
|
if (!schematic->loadDocument())
|
|
{
|
|
fprintf(stderr, "Error: Could not load schematic %s\n", schematicFileName.toLatin1().data());
|
|
delete schematic;
|
|
return nullptr;
|
|
}
|
|
|
|
return schematic;
|
|
}
|
|
|
|
int doNetlist(QString schematicFileName, QString netlistFileName, bool netlist2Console)
|
|
{
|
|
QucsSettings.DefaultSimulator = spicecompat::simQucsator;
|
|
Module::registerModules();
|
|
|
|
QScopedPointer<Schematic> schematic(openSchematic(schematicFileName));
|
|
if (!schematic)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
if (!netlist2Console)
|
|
{
|
|
qDebug() << "*** try to write netlist :" << netlistFileName;
|
|
}
|
|
|
|
QScopedPointer<QFile> netlistFile;
|
|
QScopedPointer<QTextStream> netlistStream;
|
|
QScopedPointer<QString> netlistString;
|
|
|
|
if (!netlist2Console)
|
|
{
|
|
netlistFile.reset(new QFile(netlistFileName));
|
|
if (!netlistFile->open(QIODevice::WriteOnly))
|
|
{
|
|
fprintf(stderr, "Error: Could not load netlist %s\n", netlistFileName.toLatin1().data());
|
|
return -1;
|
|
}
|
|
|
|
netlistStream.reset(new QTextStream(netlistFile.get()));
|
|
}
|
|
else
|
|
{
|
|
netlistString.reset(new QString);
|
|
netlistStream.reset(new QTextStream(netlistString.get()));
|
|
}
|
|
|
|
QPlainTextEdit errText; // dummy
|
|
QStringList Collect; // clear list for NodeSets, SPICE components etc.
|
|
int SimPorts = schematic->prepareNetlist(*netlistStream, Collect, &errText);
|
|
|
|
if (SimPorts < -5)
|
|
{
|
|
QByteArray ba = netlistFileName.toLatin1();
|
|
fprintf(stderr, "Error: Could not prepare netlist %s\n", ba.data());
|
|
/// \todo better handling for error/warnings
|
|
qCritical() << errText.toPlainText();
|
|
return 1;
|
|
}
|
|
|
|
// output NodeSets, SPICE simulations etc.
|
|
for (QStringList::Iterator it = Collect.begin(); it != Collect.end(); ++it)
|
|
{
|
|
// don't put library includes into netlist...
|
|
if ((*it).right(4) != ".lst" &&
|
|
(*it).right(5) != ".vhdl" &&
|
|
(*it).right(4) != ".vhd" &&
|
|
(*it).right(2) != ".v")
|
|
{
|
|
*netlistStream << *it << '\n';
|
|
}
|
|
}
|
|
|
|
*netlistStream << '\n';
|
|
|
|
schematic->createNetlist(*netlistStream, SimPorts);
|
|
|
|
if (netlist2Console)
|
|
{
|
|
std::cout << std::endl << netlistString->toLatin1().constData() << std::endl;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int runNgspice(QString schematicFileName, QString dataset)
|
|
{
|
|
QucsSettings.DefaultSimulator = spicecompat::simNgspice;
|
|
Module::registerModules();
|
|
|
|
QScopedPointer<Schematic> schematic(openSchematic(schematicFileName));
|
|
if (!schematic)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
QScopedPointer<Ngspice> ngspice(new Ngspice(schematic.get()));
|
|
ngspice->slotSimulate();
|
|
bool ok = ngspice->waitEndOfSimulation();
|
|
if (!ok)
|
|
{
|
|
fprintf(stderr, "Ngspice timed out or start error!\n");
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
ngspice->convertToQucsData(dataset);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int runXyce(QString schematicFileName, QString dataset)
|
|
{
|
|
QucsSettings.DefaultSimulator = spicecompat::simXyce;
|
|
Module::registerModules();
|
|
|
|
QScopedPointer<Schematic> schematic(openSchematic(schematicFileName));
|
|
if (!schematic)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
QScopedPointer<Xyce> xyce(new Xyce(schematic.get()));
|
|
xyce->slotSimulate();
|
|
bool ok = xyce->waitEndOfSimulation();
|
|
if (!ok)
|
|
{
|
|
fprintf(stderr, "Xyce timed out or start error!\n");
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
xyce->convertToQucsData(dataset);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int doNgspiceNetlist(QString schematicFileName, QString netlistFileName, bool netlist2Console)
|
|
{
|
|
QucsSettings.DefaultSimulator = spicecompat::simNgspice;
|
|
Module::registerModules();
|
|
|
|
QScopedPointer<Schematic> schematic(openSchematic(schematicFileName));
|
|
if (!schematic)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
QScopedPointer<Ngspice> ngspice(new Ngspice(schematic.get()));
|
|
ngspice->SaveNetlist(netlistFileName, netlist2Console);
|
|
|
|
if (!netlist2Console && !QFile::exists(netlistFileName))
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int doCdlNetlist(QString schematicFileName, QString netlistFileName, bool netlist2Console)
|
|
{
|
|
QucsSettings.DefaultSimulator = spicecompat::simNgspice;
|
|
Module::registerModules();
|
|
|
|
QScopedPointer<Schematic> schematic(openSchematic(schematicFileName));
|
|
if (!schematic)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
QScopedPointer<QTextStream> netlistStream;
|
|
QScopedPointer<QString> netlistString;
|
|
QScopedPointer<QFile> cdlFile;
|
|
|
|
if (netlist2Console)
|
|
{
|
|
netlistString.reset(new QString());
|
|
netlistStream.reset(new QTextStream(netlistString.get()));
|
|
}
|
|
else
|
|
{
|
|
cdlFile.reset(new QFile(netlistFileName));
|
|
|
|
if (cdlFile->open(QFile::WriteOnly))
|
|
{
|
|
netlistStream.reset(new QTextStream(cdlFile.get()));
|
|
}
|
|
else
|
|
{
|
|
QString msg = QStringLiteral(
|
|
"Tried to save netlist \nto %1\n(could not open for writing!)")
|
|
.arg(netlistFileName);
|
|
QString finalMsg = QStringLiteral(
|
|
"%1\n This could be an error in the QSettings settings file\n"
|
|
"(usually in ~/.config/qucs/qucs_s.conf)\n"
|
|
"The value for S4Q_workdir (default:/spice4qucs) needs to be writeable!")
|
|
.arg(msg);
|
|
QMessageBox::critical(nullptr, QStringLiteral("Problem with SaveNetlist"), finalMsg, QMessageBox::Ok);
|
|
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
CdlNetlistWriter cdlWriter(*netlistStream, schematic.get());
|
|
if (!cdlWriter.write())
|
|
{
|
|
QMessageBox::critical(
|
|
nullptr,
|
|
QStringLiteral("Save CDL netlist"),
|
|
QStringLiteral("Save CDL netlist failed!"),
|
|
QMessageBox::Ok);
|
|
return -1;
|
|
}
|
|
|
|
if (netlist2Console)
|
|
{
|
|
std::cout << std::endl << netlistString->toUtf8().constData() << std::endl;
|
|
}
|
|
else
|
|
{
|
|
if (!QFile::exists(netlistFileName))
|
|
{
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int doXyceNetlist(QString schematicFileName, QString netlistFileName, bool netlist2Console)
|
|
{
|
|
QucsSettings.DefaultSimulator = spicecompat::simXyce;
|
|
Module::registerModules();
|
|
QScopedPointer<Schematic> schematic(openSchematic(schematicFileName));
|
|
|
|
if (!schematic)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
QScopedPointer<Xyce> xyce(new Xyce(schematic.get()));
|
|
xyce->SaveNetlist(netlistFileName, netlist2Console);
|
|
|
|
if (!netlist2Console && !QFile::exists(netlistFileName))
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int doPrint(QString schematicFileName, QString printFile,
|
|
QString page, int dpi, QString color, QString orientation)
|
|
{
|
|
QucsSettings.DefaultSimulator = spicecompat::simQucsator;
|
|
|
|
QScopedPointer<Schematic> schematic(openSchematic(schematicFileName));
|
|
if (!schematic)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
schematic->a_Nodes = &(schematic->a_DocNodes);
|
|
schematic->a_Wires = &(schematic->a_DocWires);
|
|
schematic->a_Diagrams = &(schematic->a_DocDiags);
|
|
schematic->a_Paintings = &(schematic->a_DocPaints);
|
|
schematic->a_Components = &(schematic->a_DocComps);
|
|
schematic->reloadGraphs();
|
|
|
|
qDebug() << "*** try to print file :" << printFile;
|
|
|
|
// determine filetype
|
|
if (printFile.endsWith(".pdf")) {
|
|
//initial printer
|
|
PrinterWriter *Printer = new PrinterWriter();
|
|
Printer->setFitToPage(true);
|
|
Printer->noGuiPrint(schematic.get(), printFile, page, dpi, color, orientation);
|
|
}
|
|
else
|
|
{
|
|
ImageWriter *Printer = new ImageWriter("");
|
|
Printer->noGuiPrint(schematic.get(), printFile, color);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*!
|
|
* \brief createIcons Create component icons (png) from command line.
|
|
*/
|
|
void createIcons() {
|
|
|
|
int nCats = 0, nComps = 0;
|
|
|
|
if(!QDir("./bitmaps_generated").exists()){
|
|
QDir().mkdir("bitmaps_generated");
|
|
}
|
|
Module::registerModules ();
|
|
QStringList cats = Category::getCategories ();
|
|
|
|
for (const QString& category: cats) {
|
|
|
|
QList<Module *> Comps;
|
|
Comps = Category::getModules(category);
|
|
|
|
// crash with diagrams, skip
|
|
if(category == "diagrams") break;
|
|
|
|
char * File;
|
|
QString Name;
|
|
|
|
for (Module *Mod: Comps) {
|
|
if (Mod->info) {
|
|
|
|
Element *e = (Mod->info) (Name, File, true);
|
|
|
|
Component *c = (Component* ) e;
|
|
|
|
QList<qucs::Line *> Lines = c->Lines;
|
|
QList<struct qucs::Arc *> Arcs = c-> Arcs;
|
|
QList<qucs::Rect *> Rects = c-> Rects;
|
|
QList<qucs::Ellips *> Ellips = c-> Ellipses;
|
|
QList<Port *> Ports = c->Ports;
|
|
QList<Text*> Texts = c->Texts;
|
|
|
|
QGraphicsScene *scene = new QGraphicsScene();
|
|
|
|
for (qucs::Line *l : Lines) {
|
|
scene->addLine(l->x1, l->y1, l->x2, l->y2, l->style);
|
|
}
|
|
|
|
for (struct qucs::Arc *a: Arcs) {
|
|
// we need an open item here; QGraphisEllipseItem draws a filled ellipse and doesn't do the job here...
|
|
QPainterPath *path = new QPainterPath();
|
|
// the components do not contain the angles in degrees but in 1/16th degrees -> conversion needed
|
|
path->arcMoveTo(a->x,a->y,a->w,a->h,a->angle/16);
|
|
path->arcTo(a->x,a->y,a->w,a->h,a->angle/16,a->arclen/16);
|
|
scene->addPath(*path);
|
|
}
|
|
|
|
for(qucs::Rect *a: Rects) {
|
|
scene->addRect(a->x, a->y, a->w, a->h, a->Pen, a->Brush);
|
|
}
|
|
|
|
for(qucs::Ellips *a: Ellips) {
|
|
scene->addEllipse(a->x, a->y, a->w, a->h, a->Pen, a->Brush);
|
|
}
|
|
|
|
for(Port *p: Ports) {
|
|
scene->addEllipse(p->x-4, p->y-4, 8, 8, QPen(Qt::red));
|
|
}
|
|
|
|
for(Text *t: Texts) {
|
|
QFont myFont;
|
|
myFont.setPointSize(10);
|
|
QGraphicsTextItem* item = new QGraphicsTextItem(t->s);
|
|
item->setX(t->x);
|
|
item->setY(t->y);
|
|
item->setFont(myFont);
|
|
|
|
scene->addItem(item);
|
|
}
|
|
|
|
// this uses the size of the component as icon size
|
|
// Qt bug ? The returned sceneRect() is often 1 px short on bottom
|
|
// and right sides without anti-aliasing. 1 px more missing on top
|
|
// and left when anti-aliasing is used
|
|
QRectF rScene = scene->sceneRect().adjusted(-1,-1,1,1);
|
|
// image and scene need to be the same size, since render()
|
|
// will fill the entire image, otherwise the scaling will
|
|
// introduce artifacts
|
|
QSize sImage = rScene.size().toSize(); // rounding seems not to be an issue
|
|
// ARGB32_Premultiplied is faster (Qt docs)
|
|
//QImage image(sImage.toSize(), QImage::Format_ARGB32);
|
|
QImage image(sImage, QImage::Format_ARGB32_Premultiplied);
|
|
// this uses a fixed size for the icon (32 x 32)
|
|
//QImage image(32, 32, QImage::Format_ARGB32);
|
|
image.fill(Qt::transparent);
|
|
|
|
QPainter painter(&image);
|
|
QPainter::RenderHints hints = QPainter::RenderHints();
|
|
// Ask to antialias drawings if requested
|
|
if (QucsSettings.GraphAntiAliasing) hints |= QPainter::Antialiasing;
|
|
// Ask to antialias text if requested
|
|
if (QucsSettings.TextAntiAliasing) hints |= QPainter::TextAntialiasing;
|
|
painter.setRenderHints(hints);
|
|
|
|
// pass target and source size eplicitly, otherwise sceneRect() is used
|
|
// for the source size, which is often wrong (see comment above)
|
|
scene->render(&painter, image.rect(), rScene);
|
|
|
|
image.save("./bitmaps_generated/" + QString(File) + ".png");
|
|
|
|
fprintf(stdout, "[%s] %s\n", category.toLatin1().data(), File);
|
|
}
|
|
nComps++;
|
|
} // module
|
|
nCats++;
|
|
} // category
|
|
fprintf(stdout, "Created %i component icons from %i categories\n", nComps, nCats);
|
|
}
|
|
|
|
/*!
|
|
* \brief createDocData Create data used for documentation.
|
|
*
|
|
* It creates the following:
|
|
* - list of categories: categories.txt
|
|
* - category directory, ex.: ./lumped components/
|
|
* - CSV with component data fields. Ex [component#]_data.csv
|
|
* - CSV with component properties. Ex [component#]_props.csv
|
|
*/
|
|
void createDocData() {
|
|
|
|
QMap<int, QString> typeMap;
|
|
typeMap.insert(0x30000, "Component");
|
|
typeMap.insert(0x30002, "ComponentText");
|
|
typeMap.insert(0x10000, "AnalogComponent");
|
|
typeMap.insert(0x20000, "DigitalComponent") ;
|
|
|
|
Module::registerModules ();
|
|
QStringList cats = Category::getCategories ();
|
|
int nCats = cats.size();
|
|
|
|
QStringList catHeader;
|
|
catHeader << "# Note: auto-generated file (changes will be lost on update)";
|
|
QFile file("categories.txt");
|
|
if (!file.open(QFile::WriteOnly | QFile::Text)) return;
|
|
QTextStream out(&file);
|
|
out << cats.join("\n");
|
|
file.close();
|
|
|
|
int nComps = 0;
|
|
|
|
// table for quick reference, schematic and netlist entry
|
|
for (const QString& category: cats) {
|
|
|
|
QList<Module *> Comps;
|
|
Comps = Category::getModules(category);
|
|
|
|
// \fixme, crash with diagrams, skip
|
|
if(category == "diagrams") break;
|
|
|
|
// one dir per category
|
|
QString curDir = "./"+category+"/";
|
|
qDebug() << "Creating dir:" << curDir;
|
|
if(!QDir(curDir).exists()){
|
|
QDir().mkdir(curDir);
|
|
}
|
|
|
|
char * File;
|
|
QString Name;
|
|
|
|
int num = 0; // component id inside category
|
|
|
|
for (Module *Mod: Comps) {
|
|
num += 1;
|
|
|
|
nComps += 1;
|
|
|
|
Element *e = (Mod->info) (Name, File, true);
|
|
Component *c = (Component* ) e;
|
|
|
|
// object info
|
|
QStringList compData;
|
|
|
|
compData << "# Note: auto-generated file (changes will be lost on update)";
|
|
compData << "Caption; " + Name;
|
|
compData << "Description; " + c->Description;
|
|
compData << "Identifier; ``" + c->Model + "``"; // backticks for reST verbatim
|
|
compData << "Default name; ``" + c->Name + "``";
|
|
compData << "Type; " + typeMap.value(c->Type);
|
|
compData << "Bitmap file; " + QString(File);
|
|
compData << "Properties; " + QString::number(c->Props.count());
|
|
compData << "Category; " + category;
|
|
|
|
// 001_data.csv - CSV file with component data
|
|
QString ID = QStringLiteral("%1").arg(num,3,'d',0,'0');
|
|
QString objDataFile;
|
|
objDataFile = QStringLiteral("%1_data.csv").arg( ID ) ;
|
|
|
|
QFile file(curDir + objDataFile);
|
|
if (!file.open(QFile::WriteOnly | QFile::Text)) return;
|
|
QTextStream out(&file);
|
|
out << compData.join("\n");
|
|
file.close();
|
|
fprintf(stdout, "[%s] %s %s \n", category.toLatin1().data(), c->Model.toLatin1().data(), file.fileName().toLatin1().data());
|
|
|
|
QStringList compProps;
|
|
compProps << "# Note: auto-generated file (changes will be lost on update)";
|
|
compProps << QStringLiteral("# %1; %2; %3; %4").arg( "Name", "Value", "Display", "Description");
|
|
for (Property *prop : c->Props) {
|
|
compProps << QStringLiteral("%1; \"%2\"; %3; \"%4\"").arg(
|
|
prop->Name,
|
|
prop->Value,
|
|
prop->display?"yes":"no",
|
|
prop->Description.replace("\"","\"\"")); // escape quote in quote
|
|
}
|
|
|
|
// 001_props.csv - CSV file with component properties
|
|
QString objPropFile = QStringLiteral("%1_prop.csv").arg( ID ) ;
|
|
|
|
QFile fileProps(curDir + objPropFile );
|
|
if (!fileProps.open(QFile::WriteOnly | QFile::Text)) return;
|
|
QTextStream outProps(&fileProps);
|
|
outProps << compProps.join("\n");
|
|
compProps.clear();
|
|
file.close();
|
|
fprintf(stdout, "[%s] %s %s \n", category.toLatin1().data(), c->Model.toLatin1().data(), fileProps.fileName().toLatin1().data());
|
|
} // module
|
|
} // category
|
|
fprintf(stdout, "Created data for %i components from %i categories\n", nComps, nCats);
|
|
}
|
|
|
|
/*!
|
|
* \brief createListNetEntry prints to stdout the available netlist formats
|
|
*
|
|
* Prints the default component entries format for:
|
|
* - Qucs schematic
|
|
* - Qucsator netlist
|
|
*/
|
|
void createListComponentEntry(){
|
|
|
|
Module::registerModules ();
|
|
QStringList cats = Category::getCategories ();
|
|
// table for quick reference, schematic and netlist entry
|
|
for (QString category: cats) {
|
|
|
|
QList<Module *> Comps;
|
|
Comps = Category::getModules(category);
|
|
|
|
// \fixme, crash with diagrams, skip
|
|
if(category == "diagrams") break;
|
|
|
|
char * File;
|
|
QString Name;
|
|
|
|
for (Module *Mod: Comps) {
|
|
Element *e = (Mod->info) (Name, File, true);
|
|
Component *c = (Component* ) e;
|
|
|
|
QString qucsEntry = c->save();
|
|
fprintf(stdout, "%s; qucs ; %s\n", c->Model.toLatin1().data(), qucsEntry.toLatin1().data());
|
|
|
|
// add dummy ports/wires, avoid segfault
|
|
int port = 0;
|
|
for (Port *p: c->Ports) {
|
|
Node *n = new Node(0,0);
|
|
n->Name="_net"+QString::number(port);
|
|
p->Connection = n;
|
|
port +=1;
|
|
}
|
|
|
|
// skip Subcircuit, segfault, there is nothing to netlist
|
|
if (c->Model == "Sub" or c->Model == ".Opt") {
|
|
fprintf(stdout, "WARNING, qucsator netlist not generated for %s\n\n", c->Model.toLatin1().data());
|
|
continue;
|
|
}
|
|
|
|
QString qucsatorEntry = c->getNetlist();
|
|
fprintf(stdout, "%s; qucsator; %s\n", c->Model.toLatin1().data(), qucsatorEntry.toLatin1().data());
|
|
} // module
|
|
} // category
|
|
}
|
|
|
|
// #########################################################################
|
|
// ########## ##########
|
|
// ########## Program Start ##########
|
|
// ########## ##########
|
|
// #########################################################################
|
|
int main(int argc, char *argv[])
|
|
{
|
|
qInstallMessageHandler(qucsMessageOutput);
|
|
// set the Qucs version string
|
|
QucsVersion = VersionTriplet(PACKAGE_VERSION);
|
|
|
|
// apply default settings
|
|
//QucsSettings.font = QFont("Helvetica", 12);
|
|
QucsSettings.largeFontSize = 16.0;
|
|
QucsSettings.maxUndo = 20;
|
|
QucsSettings.NodeWiring = 0;
|
|
|
|
// initially center the application
|
|
QApplication app(argc, argv);
|
|
//QDesktopWidget *d = app.desktop();
|
|
QucsSettings.font = QApplication::font();
|
|
QucsSettings.appFont = QApplication::font();
|
|
QucsSettings.textFont = QFontDatabase::systemFont(QFontDatabase::FixedFont);
|
|
QucsSettings.font.setPointSize(12);
|
|
|
|
// default
|
|
QString QucsWorkdirPath = QDir::homePath()+QDir::toNativeSeparators ("/QucsWorkspace");
|
|
QucsSettings.qucsWorkspaceDir.setPath(QucsWorkdirPath);
|
|
QucsSettings.QucsWorkDir.setPath(QucsSettings.qucsWorkspaceDir.canonicalPath());
|
|
|
|
// load existing settings (if any)
|
|
loadSettings();
|
|
|
|
/* restore saved style */
|
|
QString savedStyle = _settings::Get().item<QString>("AppStyle");
|
|
QStyle* style = QStyleFactory::create(savedStyle);
|
|
if (style) {
|
|
QApplication::setStyle(style);
|
|
}
|
|
/* restore saved style */
|
|
|
|
QDir().mkpath(QucsSettings.qucsWorkspaceDir.absolutePath());
|
|
QDir().mkpath(QucsSettings.tempFilesDir.absolutePath());
|
|
|
|
// continue to set up overrides or default settings (some are saved on exit)
|
|
|
|
// check for relocation env variable
|
|
QDir QucsDir;
|
|
QString QucsApplicationPath = QCoreApplication::applicationDirPath();
|
|
#ifdef __APPLE__
|
|
QucsDir = QDir(QucsApplicationPath.section("/bin",0,0));
|
|
#else
|
|
QucsDir = QDir(QucsApplicationPath);
|
|
QucsDir.cdUp();
|
|
#endif
|
|
|
|
QucsSettings.BinDir = QucsApplicationPath.contains("bin") ?
|
|
(QucsApplicationPath + QDir::separator()) : QucsDir.absoluteFilePath("bin/");
|
|
QucsSettings.LangDir = QucsDir.canonicalPath() + "/share/" QUCS_NAME "/lang/";
|
|
|
|
QucsSettings.LibDir = QucsDir.canonicalPath() + "/share/" QUCS_NAME "/library/";
|
|
QucsSettings.OctaveDir = QucsDir.canonicalPath() + "/share/" QUCS_NAME "/octave/";
|
|
QucsSettings.ExamplesDir = QucsDir.canonicalPath() + "/share/" QUCS_NAME "/examples/";
|
|
QucsSettings.DocDir = QucsDir.canonicalPath() + "/share/" QUCS_NAME "/docs/";
|
|
QucsSettings.Editor = "qucs";
|
|
|
|
/// \todo Make the setting up of all executables below more consistent
|
|
char *var = NULL; // Don't use QUCSDIR with Qucs-S
|
|
var = getenv("QUCSATOR");
|
|
if (var != NULL) {
|
|
QucsSettings.QucsatorVar = QString(var);
|
|
}
|
|
else {
|
|
QucsSettings.QucsatorVar = "";
|
|
}
|
|
|
|
var = getenv("QUCSCONV");
|
|
if (var != NULL) {
|
|
QucsSettings.Qucsconv = QString(var);
|
|
}
|
|
|
|
var = getenv("ADMSXMLBINDIR");
|
|
if (var != NULL) {
|
|
QucsSettings.AdmsXmlBinDir.setPath(QString(var));
|
|
}
|
|
else {
|
|
// default admsXml bindir same as Qucs
|
|
QString admsExec;
|
|
#if defined(_WIN32) || defined(__MINGW32__)
|
|
admsExec = QDir::toNativeSeparators(QucsSettings.BinDir+"/"+"admsXml.exe");
|
|
#else
|
|
admsExec = QDir::toNativeSeparators(QucsSettings.BinDir+"/"+"admsXml");
|
|
#endif
|
|
QFile adms(admsExec);
|
|
if (adms.exists())
|
|
{
|
|
QucsSettings.AdmsXmlBinDir.setPath(QucsSettings.BinDir);
|
|
}
|
|
}
|
|
|
|
var = getenv("ASCOBINDIR");
|
|
if (var != NULL) {
|
|
QucsSettings.AscoBinDir.setPath(QString(var));
|
|
}
|
|
else {
|
|
// default ASCO bindir same as Qucs
|
|
QString ascoExec;
|
|
#if defined(_WIN32) || defined(__MINGW32__)
|
|
ascoExec = QDir::toNativeSeparators(QucsSettings.BinDir+"/"+"asco.exe");
|
|
#else
|
|
ascoExec = QDir::toNativeSeparators(QucsSettings.BinDir+"/"+"asco");
|
|
#endif
|
|
QFile asco(ascoExec);
|
|
if (asco.exists())
|
|
{
|
|
QucsSettings.AscoBinDir.setPath(QucsSettings.BinDir);
|
|
}
|
|
}
|
|
|
|
var = getenv("QUCS_OCTAVE");
|
|
if (var != NULL) {
|
|
QucsSettings.QucsOctave = QString(var);
|
|
} else {
|
|
QucsSettings.QucsOctave.clear();
|
|
}
|
|
|
|
if(!QucsSettings.BGColor.isValid())
|
|
QucsSettings.BGColor.setRgb(255, 250, 225);
|
|
|
|
// syntax highlighting
|
|
if(!QucsSettings.Comment.isValid())
|
|
QucsSettings.Comment = Qt::gray;
|
|
if(!QucsSettings.String.isValid())
|
|
QucsSettings.String = Qt::red;
|
|
if(!QucsSettings.Integer.isValid())
|
|
QucsSettings.Integer = Qt::blue;
|
|
if(!QucsSettings.Real.isValid())
|
|
QucsSettings.Real = Qt::darkMagenta;
|
|
if(!QucsSettings.Character.isValid())
|
|
QucsSettings.Character = Qt::magenta;
|
|
if(!QucsSettings.Type.isValid())
|
|
QucsSettings.Type = Qt::darkRed;
|
|
if(!QucsSettings.Attribute.isValid())
|
|
QucsSettings.Attribute = Qt::darkCyan;
|
|
if(!QucsSettings.Directive.isValid())
|
|
QucsSettings.Directive = Qt::darkCyan;
|
|
if(!QucsSettings.Task.isValid())
|
|
QucsSettings.Task = Qt::darkRed;
|
|
|
|
QucsSettings.sysDefaultFont = QApplication::font();
|
|
QApplication::setFont(QucsSettings.appFont);
|
|
|
|
QTranslator tor(nullptr);
|
|
QString lang = QucsSettings.Language;
|
|
if (lang.isEmpty()) {
|
|
QLocale loc;
|
|
lang = loc.name();
|
|
// lang = QTextCodec::locale();
|
|
}
|
|
static_cast<void>(tor.load( QStringLiteral("qucs_") + lang, QucsSettings.LangDir));
|
|
QApplication::installTranslator( &tor );
|
|
|
|
// This seems to be necessary on a few system to make strtod()
|
|
// work properly !???!
|
|
setlocale (LC_NUMERIC, "C");
|
|
|
|
#ifdef GIT
|
|
const QString applicationVersion(QString::fromUtf8("qucs s%1 (%2)").arg(PACKAGE_VERSION).arg(GIT));
|
|
#else
|
|
const QString applicationVersion(QString::fromUtf8("Qucs %1").arg(PACKAGE_VERSION));
|
|
#endif
|
|
|
|
QCoreApplication::setApplicationVersion(applicationVersion);
|
|
QStringList cmdArgs;
|
|
for (int i = 0; i < argc; ++i)
|
|
{
|
|
cmdArgs << argv[i];
|
|
}
|
|
|
|
QCommandLineParser parser;
|
|
parser.setSingleDashWordOptionMode(QCommandLineParser::ParseAsLongOptions);
|
|
parser.addVersionOption();
|
|
|
|
parser.addOptions({
|
|
{{"h", "help"}, QCoreApplication::translate("main", "display this help and exit")},
|
|
{{"n", "netlist"}, QCoreApplication::translate("main", "convert Qucs schematic into netlist")},
|
|
{{"p", "print"}, QCoreApplication::translate("main", "print Qucs schematic to file (eps needs inkscape)")},
|
|
{"page", QCoreApplication::translate("main", "set print page size (default A4)"), "A4|A3|B4|B5", "A4"},
|
|
{"dpi", QCoreApplication::translate("main", "set dpi value (default 96)"), "NUMBER", "96"},
|
|
{"color", QCoreApplication::translate("main", "set color mode (default RGB)"), "RGB|BW", "RGB"},
|
|
{"orin", QCoreApplication::translate("main", "set orientation (default portraid)"), "portraid|landscape", "portraid"},
|
|
{"i", QCoreApplication::translate("main", "use file as input schematic"), "FILENAME"},
|
|
{"o", QCoreApplication::translate("main", "use file as output netlist"), "FILENAME"},
|
|
{"ngspice", QCoreApplication::translate("main", "create Ngspice netlist")},
|
|
{"cdl", QCoreApplication::translate("main", "create CDL netlist")},
|
|
{"xyce", QCoreApplication::translate("main", "Xyce netlist")},
|
|
{"run", QCoreApplication::translate("main", "execute Ngspice/Xyce immediately")},
|
|
{"icons", QCoreApplication::translate("main", "create component icons under ./bitmaps_generated")},
|
|
{"doc", QCoreApplication::translate(
|
|
"main",
|
|
"dump data for documentation:\n"
|
|
"* file with of categories: categories.txt\n"
|
|
"* one directory per category (e.g. ./lumped\n"
|
|
" components/)\n"
|
|
" - CSV file with component data\n"
|
|
" ([comp#]_data.csv)\n"
|
|
" - CSV file with component properties.\n"
|
|
" ([comp#]_props.csv)"
|
|
)
|
|
},
|
|
{"list-entries", QCoreApplication::translate("main", "list component entry formats for schematic and netlist")},
|
|
{{"c", "netlist2Console"}, QCoreApplication::translate("main", "write netlist to console")},
|
|
});
|
|
|
|
parser.process(cmdArgs);
|
|
|
|
if (parser.isSet("help"))
|
|
{
|
|
// some modification of the Qt-generated usage text
|
|
|
|
QString helpText = parser.helpText();
|
|
helpText.insert(
|
|
helpText.indexOf('\n', 0)+1,
|
|
QString::fromUtf8(
|
|
" qucs -n -i FILENAME -o FILENAME\n"
|
|
" qucs -p -i FILENAME -o FILENAME.[pdf|png|svg|eps]\n"));
|
|
|
|
QRegularExpression optIndent(QString::fromUtf8("--page|--dpi|--color|--orin|--ngspice|--xyce|--run|--cdl"));
|
|
int idx;
|
|
int from = 0;
|
|
while ((idx = helpText.indexOf(optIndent, from)) != -1)
|
|
{
|
|
helpText.insert(idx, " ");
|
|
from = idx +3;
|
|
}
|
|
|
|
std::cout << helpText.toUtf8().constData();
|
|
exit(0);
|
|
}
|
|
|
|
const bool netlist_flag(parser.isSet("netlist"));
|
|
const bool print_flag(parser.isSet("print"));
|
|
const QString page(parser.value("page"));
|
|
const int dpi(parser.value("dpi").toInt());
|
|
const QString color(parser.value("color"));
|
|
const QString orientation(parser.value("orin"));
|
|
const QString inputfile(parser.value("i"));
|
|
const QString outputfile(parser.value("o"));
|
|
const bool ngspice_flag(parser.isSet("ngspice"));
|
|
const bool cdl_flag(parser.isSet("cdl"));
|
|
const bool xyce_flag(parser.isSet("xyce"));
|
|
const bool run_flag(parser.isSet("run"));
|
|
const bool netlist2Console(parser.isSet("netlist2Console"));
|
|
|
|
#if 0
|
|
std::cout << "Current cli values:" << std::endl;
|
|
std::cout << "netlist_flag: " << netlist_flag << std::endl;
|
|
std::cout << "print_flag: " << print_flag << std::endl;
|
|
std::cout << "page: " << page.toUtf8().constData() << std::endl;
|
|
std::cout << "dpi: " << dpi << std::endl;
|
|
std::cout << "color: " << color.toUtf8().constData() << std::endl;
|
|
std::cout << "orientation: " << orientation.toUtf8().constData() << std::endl;
|
|
std::cout << "inputfile: " << inputfile.toUtf8().constData() << std::endl;
|
|
std::cout << "outputfile: " << outputfile.toUtf8().constData() << std::endl;
|
|
std::cout << "ngspice_flag: " << ngspice_flag << std::endl;
|
|
std::cout << "cdl_flag: " << cdl_flag << std::endl;
|
|
std::cout << "xyce_flag: " << xyce_flag << std::endl;
|
|
std::cout << "run_flag: " << run_flag << std::endl;
|
|
std::cout << "netlist2Console: " << netlist2Console << std::endl;
|
|
std::cout << "icons: " << parser.isSet("icons") << std::endl;
|
|
std::cout << "doc: " << parser.isSet("doc") << std::endl;
|
|
std::cout << "list-entries: " << parser.isSet("list-entries") << std::endl;
|
|
|
|
exit(0);
|
|
#endif
|
|
|
|
if (parser.isSet("icons"))
|
|
{
|
|
createIcons();
|
|
return 0;
|
|
}
|
|
|
|
if (parser.isSet("doc"))
|
|
{
|
|
createDocData();
|
|
return 0;
|
|
}
|
|
|
|
if (parser.isSet("list-entries"))
|
|
{
|
|
createListComponentEntry();
|
|
return 0;
|
|
}
|
|
|
|
// check operation and its required arguments
|
|
if (netlist_flag and print_flag)
|
|
{
|
|
fprintf(stderr, "Error: --print and --netlist cannot be used together\n");
|
|
return -1;
|
|
}
|
|
else if (cdl_flag and run_flag)
|
|
{
|
|
fprintf(stderr, "Error: --cdl and --run cannot be used together\n");
|
|
return -1;
|
|
}
|
|
else if (((ngspice_flag || xyce_flag || cdl_flag) && print_flag) || (run_flag && print_flag))
|
|
{
|
|
fprintf(stderr, "Error: --print and Ngspice/CDL/Xyce cannot be used together\n");
|
|
return -1;
|
|
}
|
|
else if (netlist_flag or print_flag)
|
|
{
|
|
if (inputfile.isEmpty())
|
|
{
|
|
fprintf(stderr, "Error: Expected input file.\n");
|
|
return -1;
|
|
}
|
|
if (!netlist2Console && outputfile.isEmpty())
|
|
{
|
|
fprintf(stderr, "Error: Expected output file.\n");
|
|
return -1;
|
|
}
|
|
// create netlist from schematic
|
|
if (netlist_flag)
|
|
{
|
|
if (!run_flag)
|
|
{
|
|
if (ngspice_flag)
|
|
{
|
|
return doNgspiceNetlist(inputfile, outputfile, netlist2Console);
|
|
}
|
|
else if (cdl_flag)
|
|
{
|
|
return doCdlNetlist(inputfile, outputfile, netlist2Console);
|
|
}
|
|
else if (xyce_flag)
|
|
{
|
|
return doXyceNetlist(inputfile, outputfile, netlist2Console);
|
|
}
|
|
else
|
|
{
|
|
return doNetlist(inputfile, outputfile, netlist2Console);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (ngspice_flag)
|
|
{
|
|
return runNgspice(inputfile, outputfile);
|
|
}
|
|
else if (xyce_flag)
|
|
{
|
|
return runXyce(inputfile, outputfile);
|
|
}
|
|
else
|
|
{
|
|
return 1;
|
|
}
|
|
}
|
|
}
|
|
else if (print_flag)
|
|
{
|
|
return doPrint(inputfile, outputfile, page, dpi, color, orientation);
|
|
}
|
|
}
|
|
|
|
QucsMain = new QucsApp(netlist2Console);
|
|
//1a.setMainWidget(QucsMain);
|
|
|
|
QucsMain->show();
|
|
int result = app.exec();
|
|
//saveApplSettings(QucsMain);
|
|
return result;
|
|
}
|