/*************************************************************************** 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 #endif #include #include #include #include #include #include #include //#include #include #include #include #include #include #include #include #include #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 //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("DefaultSimulator"); QucsSettings.firstRun = _settings::Get().item("firstRun"); /*** Temporarily continue to use QucsSettings to make sure all settings convert okay and remain compatible ***/ QucsSettings.font.fromString(_settings::Get().item("font")); QucsSettings.appFont.fromString(_settings::Get().item("appFont")); QucsSettings.textFont.fromString(_settings::Get().item("textFont")); QucsSettings.largeFontSize = _settings::Get().item("LargeFontSize"); QucsSettings.maxUndo = _settings::Get().item("maxUndo"); QucsSettings.NodeWiring = _settings::Get().item("NodeWiring"); QucsSettings.BGColor = _settings::Get().item("BGColor"); QucsSettings.Editor = _settings::Get().item("Editor"); QucsSettings.FileTypes = _settings::Get().item("FileTypes"); QucsSettings.Language = _settings::Get().item("Language"); // Editor syntax highlighting settings. QucsSettings.Comment = _settings::Get().item("Comment"); QucsSettings.String = _settings::Get().item("String"); QucsSettings.Integer = _settings::Get().item("Integer"); QucsSettings.Real = _settings::Get().item("Real"); QucsSettings.Character = _settings::Get().item("Character"); QucsSettings.Type = _settings::Get().item("Type"); QucsSettings.Attribute = _settings::Get().item("Attribute"); QucsSettings.Directive = _settings::Get().item("Directive"); QucsSettings.Task = _settings::Get().item("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("AdmsXmlBinDir")); QucsSettings.AscoBinDir.setPath(_settings::Get().item("AscoBinDir")); QucsSettings.NgspiceExecutable = _settings::Get().item("NgspiceExecutable"); QucsSettings.XyceExecutable = _settings::Get().item("XyceExecutable"); QucsSettings.XyceParExecutable = _settings::Get().item("XyceParExecutable"); QucsSettings.SpiceOpusExecutable = _settings::Get().item("SpiceOpusExecutable"); QucsSettings.NProcs = _settings::Get().item("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("S4Q_workdir"); if (QucsSettings.S4Qworkdir == "") QucsSettings.S4Qworkdir = QDir::toNativeSeparators(QucsSettings.QucsWorkDir.absolutePath()+"/spice4qucs"); QucsSettings.OctaveExecutable = _settings::Get().item("OctaveExecutable"); QucsSettings.OpenVAFExecutable = _settings::Get().item("OpenVAFExecutable"); QucsSettings.RFLayoutExecutable = _settings::Get().item("RFLayoutExecutable"); QucsSettings.qucsWorkspaceDir.setPath(_settings::Get().item("QucsHomeDir")); QucsSettings.QucsWorkDir = QucsSettings.qucsWorkspaceDir; QucsSettings.tempFilesDir.setPath(QStandardPaths::writableLocation(QStandardPaths::CacheLocation)); QucsSettings.IgnoreFutureVersion = _settings::Get().item("IgnoreVersion"); QucsSettings.GraphAntiAliasing = _settings::Get().item("GraphAntiAliasing"); QucsSettings.TextAntiAliasing = _settings::Get().item("TextAntiAliasing"); QucsSettings.fullTraceName = _settings::Get().item("fullTraceName"); QucsSettings.RecentDocs = _settings::Get().item("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() is identical to _settings::get().item() settingsManager& qs = _settings::Get(); qs.setItem("DefaultSimulator", QucsSettings.DefaultSimulator); qs.setItem("firstRun", false); qs.setItem("font", QucsSettings.font.toString()); qs.setItem("appFont", QucsSettings.appFont.toString()); qs.setItem("textFont", QucsSettings.textFont.toString()); if (QucsMain != nullptr) { qs.setItem("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("LargeFontSize", QString::number(QucsSettings.largeFontSize)); qs.setItem("maxUndo", QucsSettings.maxUndo); qs.setItem("NodeWiring", QucsSettings.NodeWiring); qs.setItem("BGColor", QucsSettings.BGColor.name()); qs.setItem("Editor", QucsSettings.Editor); qs.setItem("FileTypes", QucsSettings.FileTypes); qs.setItem("Language", QucsSettings.Language); qs.setItem("Comment", QucsSettings.Comment.name()); qs.setItem("String", QucsSettings.String.name()); qs.setItem("Integer", QucsSettings.Integer.name()); qs.setItem("Real", QucsSettings.Real.name()); qs.setItem("Character", QucsSettings.Character.name()); qs.setItem("Type", QucsSettings.Type.name()); qs.setItem("Attribute", QucsSettings.Attribute.name()); qs.setItem("Directive", QucsSettings.Directive.name()); qs.setItem("Task", QucsSettings.Task.name()); qs.setItem("AdmsXmlBinDir", QucsSettings.AdmsXmlBinDir.canonicalPath()); qs.setItem("AscoBinDir", QucsSettings.AscoBinDir.canonicalPath()); qs.setItem("NgspiceExecutable",QucsSettings.NgspiceExecutable); qs.setItem("XyceExecutable",QucsSettings.XyceExecutable); qs.setItem("XyceParExecutable",QucsSettings.XyceParExecutable); qs.setItem("SpiceOpusExecutable",QucsSettings.SpiceOpusExecutable); qs.setItem("Qucsator",QucsSettings.Qucsator); qs.setItem("Nprocs",QucsSettings.NProcs); qs.setItem("S4Q_workdir",QucsSettings.S4Qworkdir); qs.setItem("OctaveExecutable",QucsSettings.OctaveExecutable); qs.setItem("OpenVAFExecutable",QucsSettings.OpenVAFExecutable); qs.setItem("QucsHomeDir", QucsSettings.qucsWorkspaceDir.canonicalPath()); qs.setItem("IgnoreVersion", QucsSettings.IgnoreFutureVersion); qs.setItem("GraphAntiAliasing", QucsSettings.GraphAntiAliasing); qs.setItem("TextAntiAliasing", QucsSettings.TextAntiAliasing); qs.setItem("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. * * */ 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(openSchematic(schematicFileName)); if (!schematic) { return 1; } if (!netlist2Console) { qDebug() << "*** try to write netlist :" << netlistFileName; } QScopedPointer netlistFile; QScopedPointer netlistStream; QScopedPointer 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(openSchematic(schematicFileName)); if (!schematic) { return 1; } QScopedPointer 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(openSchematic(schematicFileName)); if (!schematic) { return 1; } QScopedPointer 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(openSchematic(schematicFileName)); if (!schematic) { return 1; } QScopedPointer 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(openSchematic(schematicFileName)); if (!schematic) { return -1; } QScopedPointer netlistStream; QScopedPointer netlistString; QScopedPointer 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(openSchematic(schematicFileName)); if (!schematic) { return 1; } QScopedPointer 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(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 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 Lines = c->Lines; QList Arcs = c-> Arcs; QList Rects = c-> Rects; QList Ellips = c-> Ellipses; QList Ports = c->Ports; QList 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 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 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 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("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(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; }