2004-03-28 19:51:04 +00:00
|
|
|
/***************************************************************************
|
2005-05-17 06:35:55 +00:00
|
|
|
qucsfile.cpp
|
|
|
|
--------------
|
2004-03-28 19:51:04 +00:00
|
|
|
begin : Sat Mar 27 2004
|
|
|
|
copyright : (C) 2003 by Michael Margraf
|
2004-06-12 12:35:04 +00:00
|
|
|
email : michael.margraf@alumni.tu-berlin.de
|
2004-03-28 19:51:04 +00:00
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
/***************************************************************************
|
|
|
|
* *
|
|
|
|
* 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. *
|
|
|
|
* *
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
# include <config.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "qucsfile.h"
|
|
|
|
|
2005-05-09 06:32:17 +00:00
|
|
|
#include "node.h"
|
2004-03-28 19:51:04 +00:00
|
|
|
#include "diagrams/diagrams.h"
|
|
|
|
#include "paintings/paintings.h"
|
2005-05-09 06:32:17 +00:00
|
|
|
#include "components/spicefile.h"
|
2005-06-23 06:06:40 +00:00
|
|
|
#include "components/libcomp.h"
|
2004-03-28 19:51:04 +00:00
|
|
|
#include "qucsdoc.h"
|
2005-05-09 06:32:17 +00:00
|
|
|
#include "main.h"
|
2004-03-28 19:51:04 +00:00
|
|
|
|
|
|
|
#include <qmessagebox.h>
|
2004-07-18 18:48:51 +00:00
|
|
|
#include <qdir.h>
|
|
|
|
#include <qstringlist.h>
|
2004-08-07 10:48:45 +00:00
|
|
|
#include <qregexp.h>
|
2005-05-09 06:32:17 +00:00
|
|
|
#include <qprocess.h>
|
2005-05-17 06:35:55 +00:00
|
|
|
#include <qtextedit.h>
|
2004-07-18 18:48:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
extern QDir QucsWorkDir;
|
|
|
|
QStringList StringList;
|
2004-03-28 19:51:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
QucsFile::QucsFile(QucsDoc *p)
|
|
|
|
{
|
|
|
|
Doc = p;
|
|
|
|
|
2004-09-11 16:55:12 +00:00
|
|
|
Wires = &(p->DocWires);
|
|
|
|
Nodes = &(p->DocNodes);
|
|
|
|
Comps = &(p->DocComps);
|
|
|
|
Diags = &(p->DocDiags);
|
|
|
|
Paints = &(p->DocPaints);
|
|
|
|
SymbolPaints = &(p->SymbolPaints);
|
2004-03-28 19:51:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QucsFile::~QucsFile()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
// Creates a Qucs file format (without document properties) in the returning
|
|
|
|
// string. This is used to copy the selected elements into the clipboard.
|
|
|
|
QString QucsFile::createClipboardFile()
|
|
|
|
{
|
|
|
|
int z=0; // counts selected elements
|
|
|
|
Wire *pw;
|
|
|
|
Diagram *pd;
|
|
|
|
Painting *pp;
|
|
|
|
Component *pc;
|
|
|
|
|
2004-04-30 22:32:03 +00:00
|
|
|
QString s("<Qucs Schematic " PACKAGE_VERSION ">\n");
|
2004-03-28 19:51:04 +00:00
|
|
|
|
|
|
|
// Build element document.
|
|
|
|
s += "<Components>\n";
|
2004-09-19 16:38:59 +00:00
|
|
|
for(pc = Doc->Comps->first(); pc != 0; pc = Doc->Comps->next())
|
2004-03-28 19:51:04 +00:00
|
|
|
if(pc->isSelected) {
|
|
|
|
s += pc->save()+"\n"; z++; }
|
|
|
|
s += "</Components>\n";
|
|
|
|
|
|
|
|
s += "<Wires>\n";
|
2004-09-19 16:38:59 +00:00
|
|
|
for(pw = Doc->Wires->first(); pw != 0; pw = Doc->Wires->next())
|
2004-03-28 19:51:04 +00:00
|
|
|
if(pw->isSelected) {
|
2004-10-02 16:21:06 +00:00
|
|
|
z++;
|
|
|
|
if(pw->Label) if(!pw->Label->isSelected) {
|
|
|
|
s += pw->save().section('"', 0, 0)+"\"\" 0 0 0>\n";
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
s += pw->save()+"\n";
|
|
|
|
}
|
2004-09-25 12:10:08 +00:00
|
|
|
for(Node *pn = Nodes->first(); pn != 0; pn = Nodes->next())
|
|
|
|
if(pn->Label) if(pn->Label->isSelected) {
|
|
|
|
s += pn->Label->save()+"\n"; z++; }
|
2004-03-28 19:51:04 +00:00
|
|
|
s += "</Wires>\n";
|
|
|
|
|
|
|
|
s += "<Diagrams>\n";
|
2004-09-19 16:38:59 +00:00
|
|
|
for(pd = Doc->Diags->first(); pd != 0; pd = Doc->Diags->next())
|
2004-03-28 19:51:04 +00:00
|
|
|
if(pd->isSelected) {
|
|
|
|
s += pd->save()+"\n"; z++; }
|
|
|
|
s += "</Diagrams>\n";
|
|
|
|
|
|
|
|
s += "<Paintings>\n";
|
2004-09-19 16:38:59 +00:00
|
|
|
for(pp = Doc->Paints->first(); pp != 0; pp = Doc->Paints->next())
|
|
|
|
if(pp->isSelected)
|
2005-05-02 06:28:45 +00:00
|
|
|
if(pp->Name.at(0) != '.') { // subcircuit specific -> do not copy
|
2004-09-19 16:38:59 +00:00
|
|
|
s += "<"+pp->save()+">\n"; z++; }
|
2004-03-28 19:51:04 +00:00
|
|
|
s += "</Paintings>\n";
|
|
|
|
|
|
|
|
if(z == 0) return ""; // return empty if no selection
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2004-09-18 09:46:19 +00:00
|
|
|
// -------------------------------------------------------------
|
|
|
|
// Only read fields without loading them.
|
|
|
|
bool QucsFile::loadIntoNothing(QTextStream *stream)
|
|
|
|
{
|
|
|
|
QString Line, cstr;
|
|
|
|
while(!stream->atEnd()) {
|
|
|
|
Line = stream->readLine();
|
|
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
|
|
QObject::tr("Format Error:\n'Painting' field is not closed!"));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2004-03-28 19:51:04 +00:00
|
|
|
// -------------------------------------------------------------
|
|
|
|
// Paste from clipboard.
|
|
|
|
bool QucsFile::pasteFromClipboard(QTextStream *stream, QPtrList<Element> *pe)
|
|
|
|
{
|
|
|
|
QString Line;
|
|
|
|
|
|
|
|
Line = stream->readLine();
|
2004-06-12 12:35:04 +00:00
|
|
|
if(Line.left(16) != "<Qucs Schematic ") // wrong file type ?
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
|
2004-04-30 22:32:03 +00:00
|
|
|
QString s = PACKAGE_VERSION;
|
2004-03-28 19:51:04 +00:00
|
|
|
Line = Line.mid(16, Line.length()-17);
|
|
|
|
if(Line != s) { // wrong version number ?
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
|
|
QObject::tr("Wrong document version: ")+Line);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2004-09-18 09:46:19 +00:00
|
|
|
// read content in symbol edit mode *************************
|
|
|
|
if(Doc->symbolMode) {
|
|
|
|
while(!stream->atEnd()) {
|
|
|
|
Line = stream->readLine();
|
|
|
|
if(Line == "<Components>") {
|
|
|
|
if(!loadIntoNothing(stream)) return false; }
|
|
|
|
else
|
|
|
|
if(Line == "<Wires>") {
|
|
|
|
if(!loadIntoNothing(stream)) return false; }
|
|
|
|
else
|
|
|
|
if(Line == "<Diagrams>") {
|
|
|
|
if(!loadIntoNothing(stream)) return false; }
|
|
|
|
else
|
|
|
|
if(Line == "<Paintings>") {
|
|
|
|
if(!loadPaintings(stream, (QPtrList<Painting>*)pe)) return false; }
|
|
|
|
else {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
|
|
QObject::tr("Clipboard Format Error:\nUnknown field!"));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// read content in schematic edit mode *************************
|
2004-03-28 19:51:04 +00:00
|
|
|
while(!stream->atEnd()) {
|
|
|
|
Line = stream->readLine();
|
2004-05-20 17:27:41 +00:00
|
|
|
if(Line == "<Components>") {
|
|
|
|
if(!loadComponents(stream, (QPtrList<Component>*)pe)) return false; }
|
2004-03-28 19:51:04 +00:00
|
|
|
else
|
2004-05-20 17:27:41 +00:00
|
|
|
if(Line == "<Wires>") {
|
2004-09-25 12:10:08 +00:00
|
|
|
if(!loadWires(stream, pe)) return false; }
|
2004-03-28 19:51:04 +00:00
|
|
|
else
|
2004-05-20 17:27:41 +00:00
|
|
|
if(Line == "<Diagrams>") {
|
|
|
|
if(!loadDiagrams(stream, (QPtrList<Diagram>*)pe)) return false; }
|
2004-03-28 19:51:04 +00:00
|
|
|
else
|
2004-05-20 17:27:41 +00:00
|
|
|
if(Line == "<Paintings>") {
|
|
|
|
if(!loadPaintings(stream, (QPtrList<Painting>*)pe)) return false; }
|
2004-03-28 19:51:04 +00:00
|
|
|
else {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-06-22 16:49:55 +00:00
|
|
|
QObject::tr("Clipboard Format Error:\nUnknown field!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
// Returns the number of subcircuit ports.
|
|
|
|
int QucsFile::save()
|
|
|
|
{
|
|
|
|
QFile file(Doc->DocName);
|
|
|
|
if(!file.open(IO_WriteOnly)) {
|
2004-06-12 12:35:04 +00:00
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
|
|
QObject::tr("Cannot save document!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
QTextStream stream(&file);
|
|
|
|
|
2004-04-30 22:32:03 +00:00
|
|
|
stream << "<Qucs Schematic " << PACKAGE_VERSION << ">\n";
|
2004-03-28 19:51:04 +00:00
|
|
|
|
|
|
|
stream << "<Properties>\n";
|
2004-09-18 09:46:19 +00:00
|
|
|
if(Doc->symbolMode) {
|
|
|
|
stream << " <View=" << Doc->tmpViewX1<<","<<Doc->tmpViewY1<<","
|
|
|
|
<< Doc->tmpViewX2<<","<<Doc->tmpViewY2<< ",";
|
|
|
|
stream <<Doc->tmpScale<<","<<Doc->tmpPosX<<","<<Doc->tmpPosY << ">\n";
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
stream << " <View=" << Doc->ViewX1<<","<<Doc->ViewY1<<","
|
|
|
|
<< Doc->ViewX2<<","<<Doc->ViewY2<< ",";
|
2004-12-19 16:06:24 +00:00
|
|
|
stream << Doc->Scale <<","<<Doc->PosX<<","<<Doc->PosY << ">\n";
|
2004-09-18 09:46:19 +00:00
|
|
|
}
|
2004-06-22 16:49:55 +00:00
|
|
|
stream << " <Grid=" << Doc->GridX<<","<<Doc->GridY<<","
|
2004-06-12 12:35:04 +00:00
|
|
|
<< Doc->GridOn << ">\n";
|
2004-06-22 16:49:55 +00:00
|
|
|
stream << " <DataSet=" << Doc->DataSet << ">\n";
|
|
|
|
stream << " <DataDisplay=" << Doc->DataDisplay << ">\n";
|
|
|
|
stream << " <OpenDisplay=" << Doc->SimOpenDpl << ">\n";
|
2004-03-28 19:51:04 +00:00
|
|
|
stream << "</Properties>\n";
|
|
|
|
|
2004-09-11 16:55:12 +00:00
|
|
|
Painting *pp;
|
|
|
|
stream << "<Symbol>\n"; // save all paintings for symbol
|
|
|
|
for(pp = SymbolPaints->first(); pp != 0; pp = SymbolPaints->next())
|
|
|
|
stream << " <" << pp->save() << ">\n";
|
|
|
|
stream << "</Symbol>\n";
|
|
|
|
|
2004-06-12 12:35:04 +00:00
|
|
|
stream << "<Components>\n"; // save all components
|
2004-12-30 12:03:08 +00:00
|
|
|
for(Component *pc = Comps->first(); pc != 0; pc = Comps->next())
|
2004-12-19 16:06:24 +00:00
|
|
|
stream << " " << pc->save() << "\n";
|
2004-03-28 19:51:04 +00:00
|
|
|
stream << "</Components>\n";
|
|
|
|
|
2004-06-12 12:35:04 +00:00
|
|
|
stream << "<Wires>\n"; // save all wires
|
|
|
|
for(Wire *pw = Wires->first(); pw != 0; pw = Wires->next())
|
2004-12-19 16:06:24 +00:00
|
|
|
stream << " " << pw->save() << "\n";
|
2004-06-12 12:35:04 +00:00
|
|
|
|
|
|
|
// save all labeled nodes as wires
|
|
|
|
for(Node *pn = Nodes->first(); pn != 0; pn = Nodes->next())
|
2004-06-22 16:49:55 +00:00
|
|
|
if(pn->Label) stream << " " << pn->Label->save() << "\n";
|
2004-03-28 19:51:04 +00:00
|
|
|
stream << "</Wires>\n";
|
|
|
|
|
2004-06-12 12:35:04 +00:00
|
|
|
stream << "<Diagrams>\n"; // save all diagrams
|
|
|
|
for(Diagram *pd = Diags->first(); pd != 0; pd = Diags->next())
|
2004-06-22 16:49:55 +00:00
|
|
|
stream << " " << pd->save() << "\n";
|
2004-03-28 19:51:04 +00:00
|
|
|
stream << "</Diagrams>\n";
|
|
|
|
|
2004-06-12 12:35:04 +00:00
|
|
|
stream << "<Paintings>\n"; // save all paintings
|
2004-09-11 16:55:12 +00:00
|
|
|
for(pp = Paints->first(); pp != 0; pp = Paints->next())
|
|
|
|
stream << " <" << pp->save() << ">\n";
|
2004-03-28 19:51:04 +00:00
|
|
|
stream << "</Paintings>\n";
|
|
|
|
|
|
|
|
file.close();
|
2004-12-30 12:03:08 +00:00
|
|
|
return 0;
|
2004-03-28 19:51:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
bool QucsFile::loadProperties(QTextStream *stream)
|
|
|
|
{
|
|
|
|
bool ok = true;
|
|
|
|
QString Line, cstr, nstr;
|
|
|
|
while(!stream->atEnd()) {
|
|
|
|
Line = stream->readLine();
|
2004-06-22 16:49:55 +00:00
|
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true; // field end ?
|
2004-03-28 19:51:04 +00:00
|
|
|
Line = Line.stripWhiteSpace();
|
2004-05-29 13:05:04 +00:00
|
|
|
if(Line.isEmpty()) continue;
|
2004-03-28 19:51:04 +00:00
|
|
|
|
|
|
|
if(Line.at(0) != '<') {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-05-29 13:05:04 +00:00
|
|
|
QObject::tr("Format Error:\nWrong property field limiter!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if(Line.at(Line.length()-1) != '>') {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-05-29 13:05:04 +00:00
|
|
|
QObject::tr("Format Error:\nWrong property field limiter!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Line = Line.mid(1, Line.length()-2); // cut off start and end character
|
|
|
|
|
|
|
|
cstr = Line.section('=',0,0); // property type
|
|
|
|
nstr = Line.section('=',1,1); // property value
|
2004-05-29 13:05:04 +00:00
|
|
|
if(cstr == "View") {
|
|
|
|
Doc->ViewX1 = nstr.section(',',0,0).toInt(&ok); if(ok) {
|
|
|
|
Doc->ViewY1 = nstr.section(',',1,1).toInt(&ok); if(ok) {
|
|
|
|
Doc->ViewX2 = nstr.section(',',2,2).toInt(&ok); if(ok) {
|
|
|
|
Doc->ViewY2 = nstr.section(',',3,3).toInt(&ok); if(ok) {
|
|
|
|
Doc->Scale = nstr.section(',',4,4).toDouble(&ok); if(ok) {
|
|
|
|
Doc->PosX = nstr.section(',',5,5).toInt(&ok); if(ok) {
|
|
|
|
Doc->PosY = nstr.section(',',6,6).toInt(&ok); }}}}}} }
|
|
|
|
else if(cstr == "Grid") {
|
|
|
|
Doc->GridX = nstr.section(',',0,0).toInt(&ok); if(ok) {
|
|
|
|
Doc->GridY = nstr.section(',',1,1).toInt(&ok); if(ok) {
|
|
|
|
if(nstr.section(',',2,2).toInt(&ok) == 0) Doc->GridOn = false;
|
|
|
|
else Doc->GridOn = true; }} }
|
2004-03-28 19:51:04 +00:00
|
|
|
else if(cstr == "DataSet") Doc->DataSet = nstr;
|
|
|
|
else if(cstr == "DataDisplay") Doc->DataDisplay = nstr;
|
2004-05-29 13:05:04 +00:00
|
|
|
else if(cstr == "OpenDisplay")
|
|
|
|
if(nstr.toInt(&ok) == 0) Doc->SimOpenDpl = false;
|
|
|
|
else Doc->SimOpenDpl = true;
|
2004-03-28 19:51:04 +00:00
|
|
|
else {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-05-29 13:05:04 +00:00
|
|
|
QObject::tr("Format Error:\nUnknown property: ")+cstr);
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if(!ok) {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-05-29 13:05:04 +00:00
|
|
|
QObject::tr("Format Error:\nNumber expected in property field!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
|
|
QObject::tr("Format Error:\n'Property' field is not closed!"));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
// Inserts a component without performing logic for wire optimization.
|
|
|
|
void QucsFile::simpleInsertComponent(Component *c)
|
|
|
|
{
|
|
|
|
Node *pn;
|
|
|
|
int x, y;
|
|
|
|
// connect every node of component
|
|
|
|
for(Port *pp = c->Ports.first(); pp != 0; pp = c->Ports.next()) {
|
|
|
|
x = pp->x+c->cx;
|
|
|
|
y = pp->y+c->cy;
|
|
|
|
|
|
|
|
// check if new node lies upon existing node
|
|
|
|
for(pn = Nodes->first(); pn != 0; pn = Nodes->next()) // check every node
|
|
|
|
if(pn->cx == x) if(pn->cy == y) break;
|
|
|
|
|
2004-05-29 13:05:04 +00:00
|
|
|
if(pn == 0) { // create new node, if no existing one lies at this position
|
2004-03-28 19:51:04 +00:00
|
|
|
pn = new Node(x, y);
|
|
|
|
Nodes->append(pn);
|
|
|
|
}
|
|
|
|
pn->Connections.append(c); // connect schematic node to component node
|
|
|
|
|
|
|
|
pp->Connection = pn; // connect component node to schematic node
|
|
|
|
}
|
|
|
|
|
|
|
|
Comps->append(c);
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
bool QucsFile::loadComponents(QTextStream *stream, QPtrList<Component> *List)
|
|
|
|
{
|
|
|
|
QString Line, cstr;
|
|
|
|
Component *c;
|
|
|
|
while(!stream->atEnd()) {
|
|
|
|
Line = stream->readLine();
|
2004-06-22 16:49:55 +00:00
|
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true;
|
2004-05-29 13:05:04 +00:00
|
|
|
Line = Line.stripWhiteSpace();
|
|
|
|
if(Line.isEmpty()) continue;
|
2004-03-28 19:51:04 +00:00
|
|
|
|
|
|
|
c = getComponentFromName(Line);
|
|
|
|
if(!c) return false;
|
|
|
|
|
2004-06-22 16:49:55 +00:00
|
|
|
if(List) { // "paste" ?
|
2004-03-28 19:51:04 +00:00
|
|
|
int z;
|
2004-05-29 13:05:04 +00:00
|
|
|
for(z=c->Name.length()-1; z>=0; z--) // cut off number of component name
|
2004-03-28 19:51:04 +00:00
|
|
|
if(!c->Name.at(z).isDigit()) break;
|
|
|
|
c->Name = c->Name.left(z+1);
|
|
|
|
List->append(c);
|
|
|
|
}
|
|
|
|
else simpleInsertComponent(c);
|
|
|
|
}
|
|
|
|
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-05-29 13:05:04 +00:00
|
|
|
QObject::tr("Format Error:\n'Component' field is not closed!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
// Inserts a wire without performing logic for optimizing.
|
|
|
|
void QucsFile::simpleInsertWire(Wire *pw)
|
|
|
|
{
|
|
|
|
Node *pn;
|
|
|
|
// check if first wire node lies upon existing node
|
|
|
|
for(pn = Nodes->first(); pn != 0; pn = Nodes->next()) // check every node
|
|
|
|
if(pn->cx == pw->x1) if(pn->cy == pw->y1) break;
|
|
|
|
|
|
|
|
if(!pn) { // create new node, if no existing one lies at this position
|
|
|
|
pn = new Node(pw->x1, pw->y1);
|
|
|
|
Nodes->append(pn);
|
|
|
|
}
|
|
|
|
|
2004-05-29 13:05:04 +00:00
|
|
|
if(pw->x1 == pw->x2) if(pw->y1 == pw->y2) {
|
|
|
|
pn->Label = pw->Label; // wire with length zero are just node labels
|
2004-10-10 16:06:55 +00:00
|
|
|
if (pn->Label) {
|
2005-08-15 06:04:52 +00:00
|
|
|
pn->Label->Type = isNodeLabel;
|
|
|
|
pn->Label->pOwner = pn;
|
2004-10-10 16:06:55 +00:00
|
|
|
}
|
2004-03-28 19:51:04 +00:00
|
|
|
delete pw; // delete wire because this is not a wire
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
pn->Connections.append(pw); // connect schematic node to component node
|
|
|
|
pw->Port1 = pn;
|
|
|
|
|
|
|
|
// check if second wire node lies upon existing node
|
|
|
|
for(pn = Nodes->first(); pn != 0; pn = Nodes->next()) // check every node
|
|
|
|
if(pn->cx == pw->x2) if(pn->cy == pw->y2) break;
|
|
|
|
|
|
|
|
if(!pn) { // create new node, if no existing one lies at this position
|
|
|
|
pn = new Node(pw->x2, pw->y2);
|
|
|
|
Nodes->append(pn);
|
|
|
|
}
|
|
|
|
pn->Connections.append(pw); // connect schematic node to component node
|
|
|
|
pw->Port2 = pn;
|
|
|
|
|
|
|
|
Wires->append(pw);
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
2004-09-25 12:10:08 +00:00
|
|
|
bool QucsFile::loadWires(QTextStream *stream, QPtrList<Element> *List)
|
2004-03-28 19:51:04 +00:00
|
|
|
{
|
|
|
|
Wire *w;
|
|
|
|
QString Line;
|
|
|
|
while(!stream->atEnd()) {
|
|
|
|
Line = stream->readLine();
|
2004-06-22 16:49:55 +00:00
|
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true;
|
2004-03-28 19:51:04 +00:00
|
|
|
Line = Line.stripWhiteSpace();
|
2004-05-29 13:05:04 +00:00
|
|
|
if(Line.isEmpty()) continue;
|
|
|
|
|
2004-06-22 16:49:55 +00:00
|
|
|
// (Node*)4 = move all ports (later on)
|
|
|
|
w = new Wire(0,0,0,0, (Node*)4,(Node*)4);
|
2004-03-28 19:51:04 +00:00
|
|
|
if(!w->load(Line)) {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-06-22 16:49:55 +00:00
|
|
|
QObject::tr("Format Error:\nWrong 'wire' line format!"));
|
2004-09-25 12:10:08 +00:00
|
|
|
delete w;
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
2004-09-25 12:10:08 +00:00
|
|
|
if(List) {
|
|
|
|
if(w->x1 == w->x2) if(w->y1 == w->y2) if(w->Label) {
|
|
|
|
w->Label->Type = isMovingLabel;
|
|
|
|
List->append(w->Label);
|
|
|
|
delete w;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
List->append(w);
|
2004-10-02 16:21:06 +00:00
|
|
|
if(w->Label) List->append(w->Label);
|
2004-09-25 12:10:08 +00:00
|
|
|
}
|
2004-03-28 19:51:04 +00:00
|
|
|
else simpleInsertWire(w);
|
|
|
|
}
|
|
|
|
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-06-22 16:49:55 +00:00
|
|
|
QObject::tr("Format Error:\n'Wire' field is not closed!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
bool QucsFile::loadDiagrams(QTextStream *stream, QPtrList<Diagram> *List)
|
|
|
|
{
|
|
|
|
Diagram *d;
|
|
|
|
QString Line, cstr;
|
|
|
|
while(!stream->atEnd()) {
|
|
|
|
Line = stream->readLine();
|
2004-06-22 16:49:55 +00:00
|
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true;
|
2004-03-28 19:51:04 +00:00
|
|
|
Line = Line.stripWhiteSpace();
|
2004-05-29 13:05:04 +00:00
|
|
|
if(Line.isEmpty()) continue;
|
2004-03-28 19:51:04 +00:00
|
|
|
|
|
|
|
cstr = Line.section(' ',0,0); // diagram type
|
|
|
|
if(cstr == "<Rect") d = new RectDiagram();
|
|
|
|
else if(cstr == "<Polar") d = new PolarDiagram();
|
|
|
|
else if(cstr == "<Tab") d = new TabDiagram();
|
|
|
|
else if(cstr == "<Smith") d = new SmithDiagram();
|
2004-11-11 18:32:34 +00:00
|
|
|
else if(cstr == "<ySmith") d = new SmithDiagram(0,0,false);
|
|
|
|
else if(cstr == "<PS") d = new PSDiagram();
|
|
|
|
else if(cstr == "<SP") d = new PSDiagram(0,0,false);
|
2005-03-22 18:39:40 +00:00
|
|
|
else if(cstr == "<Rect3D") d = new Rect3DDiagram();
|
2005-04-12 13:22:48 +00:00
|
|
|
else if(cstr == "<Curve") d = new CurveDiagram();
|
2005-11-07 07:04:50 +00:00
|
|
|
else if(cstr == "<Time") d = new TimingDiagram();
|
2004-03-28 19:51:04 +00:00
|
|
|
else {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-06-16 17:41:33 +00:00
|
|
|
QObject::tr("Format Error:\nUnknown diagram!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!d->load(Line, stream)) {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-06-16 17:41:33 +00:00
|
|
|
QObject::tr("Format Error:\nWrong 'diagram' line format!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
delete d;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
List->append(d);
|
|
|
|
}
|
|
|
|
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-06-16 17:41:33 +00:00
|
|
|
QObject::tr("Format Error:\n'Diagram' field is not closed!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
bool QucsFile::loadPaintings(QTextStream *stream, QPtrList<Painting> *List)
|
|
|
|
{
|
2004-09-18 09:46:19 +00:00
|
|
|
Painting *p=0;
|
2004-03-28 19:51:04 +00:00
|
|
|
QString Line, cstr;
|
|
|
|
while(!stream->atEnd()) {
|
|
|
|
Line = stream->readLine();
|
2004-06-22 16:49:55 +00:00
|
|
|
if(Line.at(0) == '<') if(Line.at(1) == '/') return true;
|
2004-09-11 16:55:12 +00:00
|
|
|
|
2004-03-28 19:51:04 +00:00
|
|
|
Line = Line.stripWhiteSpace();
|
2004-05-29 13:05:04 +00:00
|
|
|
if(Line.isEmpty()) continue;
|
2004-09-11 16:55:12 +00:00
|
|
|
if( (Line.at(0) != '<') || (Line.at(Line.length()-1) != '>')) {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
|
|
QObject::tr("Format Error:\nWrong 'painting' line delimiter!"));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
Line = Line.mid(1, Line.length()-2); // cut off start and end character
|
|
|
|
|
2004-03-28 19:51:04 +00:00
|
|
|
cstr = Line.section(' ',0,0); // painting type
|
2004-09-18 09:46:19 +00:00
|
|
|
if(cstr == "Line") p = new GraphicLine();
|
|
|
|
else if(cstr == "EArc") p = new EllipseArc();
|
2004-10-15 07:07:57 +00:00
|
|
|
else if(cstr == ".PortSym") p = new PortSymbol();
|
|
|
|
else if(cstr == ".ID") p = new ID_Text();
|
2004-09-11 16:55:12 +00:00
|
|
|
else if(cstr == "Text") p = new GraphicText();
|
2004-09-18 09:46:19 +00:00
|
|
|
else if(cstr == "Rectangle") p = new Rectangle();
|
2004-09-11 16:55:12 +00:00
|
|
|
else if(cstr == "Arrow") p = new Arrow();
|
|
|
|
else if(cstr == "Ellipse") p = new Ellipse();
|
2004-03-28 19:51:04 +00:00
|
|
|
else {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-05-20 17:27:41 +00:00
|
|
|
QObject::tr("Format Error:\nUnknown painting!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!p->load(Line)) {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-05-20 17:27:41 +00:00
|
|
|
QObject::tr("Format Error:\nWrong 'painting' line format!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
delete p;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
List->append(p);
|
|
|
|
}
|
|
|
|
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-05-20 17:27:41 +00:00
|
|
|
QObject::tr("Format Error:\n'Painting' field is not closed!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
bool QucsFile::load()
|
|
|
|
{
|
|
|
|
QFile file(Doc->DocName);
|
|
|
|
if(!file.open(IO_ReadOnly)) {
|
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
|
|
QObject::tr("Cannot load document: ")+Doc->DocName);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Line;
|
|
|
|
QTextStream stream(&file);
|
|
|
|
|
|
|
|
// read header **************************
|
2004-05-29 13:05:04 +00:00
|
|
|
do {
|
|
|
|
if(stream.atEnd()) {
|
|
|
|
file.close();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Line = stream.readLine();
|
|
|
|
} while(Line.isEmpty());
|
|
|
|
|
2004-03-28 19:51:04 +00:00
|
|
|
if(Line.left(16) != "<Qucs Schematic ") { // wrong file type ?
|
|
|
|
file.close();
|
2004-05-29 13:05:04 +00:00
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
2004-11-06 16:29:51 +00:00
|
|
|
QObject::tr("Wrong document type: ")+Doc->DocName);
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2004-04-30 22:32:03 +00:00
|
|
|
QString s = PACKAGE_VERSION;
|
2004-03-28 19:51:04 +00:00
|
|
|
s.remove('.');
|
|
|
|
Line = Line.mid(16, Line.length()-17);
|
|
|
|
Line.remove('.');
|
|
|
|
if(Line > s) { // wrong version number ? (only backward compatible)
|
|
|
|
file.close();
|
2004-05-29 13:05:04 +00:00
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
|
|
QObject::tr("Wrong document version: ")+Line);
|
2004-03-28 19:51:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// read content *************************
|
|
|
|
while(!stream.atEnd()) {
|
|
|
|
Line = stream.readLine();
|
2004-05-29 13:05:04 +00:00
|
|
|
Line = Line.stripWhiteSpace();
|
|
|
|
if(Line.isEmpty()) continue;
|
|
|
|
|
2004-09-11 16:55:12 +00:00
|
|
|
if(Line == "<Symbol>") {
|
|
|
|
if(!loadPaintings(&stream, SymbolPaints)) {
|
|
|
|
file.close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2004-05-29 13:05:04 +00:00
|
|
|
if(Line == "<Properties>") {
|
|
|
|
if(!loadProperties(&stream)) { file.close(); return false; } }
|
2004-03-28 19:51:04 +00:00
|
|
|
else
|
2004-05-29 13:05:04 +00:00
|
|
|
if(Line == "<Components>") {
|
|
|
|
if(!loadComponents(&stream)) { file.close(); return false; } }
|
2004-03-28 19:51:04 +00:00
|
|
|
else
|
2004-05-29 13:05:04 +00:00
|
|
|
if(Line == "<Wires>") {
|
|
|
|
if(!loadWires(&stream)) { file.close(); return false; } }
|
2004-03-28 19:51:04 +00:00
|
|
|
else
|
2004-05-29 13:05:04 +00:00
|
|
|
if(Line == "<Diagrams>") {
|
|
|
|
if(!loadDiagrams(&stream, Diags)) { file.close(); return false; } }
|
2004-03-28 19:51:04 +00:00
|
|
|
else
|
2004-05-29 13:05:04 +00:00
|
|
|
if(Line == "<Paintings>") {
|
|
|
|
if(!loadPaintings(&stream, Paints)) { file.close(); return false; } }
|
2004-03-28 19:51:04 +00:00
|
|
|
else {
|
2004-05-29 13:05:04 +00:00
|
|
|
QMessageBox::critical(0, QObject::tr("Error"),
|
|
|
|
QObject::tr("File Format Error:\nUnknown field!"));
|
2004-03-28 19:51:04 +00:00
|
|
|
file.close();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
file.close();
|
|
|
|
return true;
|
|
|
|
}
|
2004-06-12 12:35:04 +00:00
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
// Creates a Qucs file format (without document properties) in the returning
|
|
|
|
// string. This is used to save state for undo operation.
|
2004-06-22 16:49:55 +00:00
|
|
|
QString QucsFile::createUndoString(char Op)
|
2004-06-12 12:35:04 +00:00
|
|
|
{
|
|
|
|
Wire *pw;
|
|
|
|
Diagram *pd;
|
|
|
|
Painting *pp;
|
|
|
|
Component *pc;
|
|
|
|
|
|
|
|
// Build element document.
|
2004-08-14 06:40:55 +00:00
|
|
|
QString s = " \n";
|
2004-06-22 16:49:55 +00:00
|
|
|
s.at(0) = Op;
|
2004-06-12 12:35:04 +00:00
|
|
|
for(pc = Comps->first(); pc != 0; pc = Comps->next())
|
|
|
|
s += pc->save()+"\n";
|
2004-06-22 16:49:55 +00:00
|
|
|
s += "</>\n"; // short end flag
|
2004-06-12 12:35:04 +00:00
|
|
|
|
|
|
|
for(pw = Wires->first(); pw != 0; pw = Wires->next())
|
|
|
|
s += pw->save()+"\n";
|
|
|
|
// save all labeled nodes as wires
|
|
|
|
for(Node *pn = Nodes->first(); pn != 0; pn = Nodes->next())
|
|
|
|
if(pn->Label) s += pn->Label->save()+"\n";
|
2004-06-22 16:49:55 +00:00
|
|
|
s += "</>\n";
|
2004-06-12 12:35:04 +00:00
|
|
|
|
|
|
|
for(pd = Diags->first(); pd != 0; pd = Diags->next())
|
|
|
|
s += pd->save()+"\n";
|
2004-06-22 16:49:55 +00:00
|
|
|
s += "</>\n";
|
2004-06-12 12:35:04 +00:00
|
|
|
|
|
|
|
for(pp = Paints->first(); pp != 0; pp = Paints->next())
|
2004-09-19 16:38:59 +00:00
|
|
|
s += "<"+pp->save()+">\n";
|
|
|
|
s += "</>\n";
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
// Same as "createUndoString(char Op)" but for symbol edit mode.
|
|
|
|
QString QucsFile::createSymbolUndoString(char Op)
|
|
|
|
{
|
|
|
|
Painting *pp;
|
|
|
|
|
|
|
|
// Build element document.
|
|
|
|
QString s = " \n";
|
|
|
|
s.at(0) = Op;
|
|
|
|
s += "</>\n"; // short end flag for components
|
|
|
|
s += "</>\n"; // short end flag for wires
|
|
|
|
s += "</>\n"; // short end flag for diagrams
|
|
|
|
|
|
|
|
for(pp = SymbolPaints->first(); pp != 0; pp = SymbolPaints->next())
|
|
|
|
s += "<"+pp->save()+">\n";
|
2004-06-22 16:49:55 +00:00
|
|
|
s += "</>\n";
|
2004-06-12 12:35:04 +00:00
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------------
|
|
|
|
// Is quite similiar to "load()" but with less error checking.
|
|
|
|
// Used for "undo" function.
|
|
|
|
bool QucsFile::rebuild(QString *s)
|
|
|
|
{
|
|
|
|
Wires->clear(); // delete whole document
|
|
|
|
Nodes->clear();
|
|
|
|
Comps->clear();
|
|
|
|
Diags->clear();
|
|
|
|
Paints->clear();
|
|
|
|
|
|
|
|
QString Line;
|
|
|
|
QTextStream stream(s, IO_ReadOnly);
|
2004-06-22 16:49:55 +00:00
|
|
|
Line = stream.readLine(); // skip identity byte
|
2004-06-12 12:35:04 +00:00
|
|
|
|
|
|
|
// read content *************************
|
2004-06-22 16:49:55 +00:00
|
|
|
if(!loadComponents(&stream)) return false;
|
|
|
|
if(!loadWires(&stream)) return false;
|
|
|
|
if(!loadDiagrams(&stream, Diags)) return false;
|
|
|
|
if(!loadPaintings(&stream, Paints)) return false;
|
2004-06-12 12:35:04 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2004-07-18 18:48:51 +00:00
|
|
|
|
2004-09-19 16:38:59 +00:00
|
|
|
// -------------------------------------------------------------
|
|
|
|
// Same as "rebuild(QString *s)" but for symbol edit mode.
|
|
|
|
bool QucsFile::rebuildSymbol(QString *s)
|
|
|
|
{
|
|
|
|
SymbolPaints->clear(); // delete whole document
|
|
|
|
|
|
|
|
QString Line;
|
|
|
|
QTextStream stream(s, IO_ReadOnly);
|
|
|
|
Line = stream.readLine(); // skip identity byte
|
|
|
|
|
|
|
|
// read content *************************
|
|
|
|
Line = stream.readLine(); // skip components
|
|
|
|
Line = stream.readLine(); // skip wires
|
|
|
|
Line = stream.readLine(); // skip diagrams
|
|
|
|
if(!loadPaintings(&stream, SymbolPaints)) return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2004-07-18 18:48:51 +00:00
|
|
|
|
|
|
|
// ***************************************************************
|
|
|
|
// ***** *****
|
|
|
|
// ***** Functions to create netlist *****
|
|
|
|
// ***** *****
|
|
|
|
// ***************************************************************
|
|
|
|
|
2005-10-24 06:10:35 +00:00
|
|
|
void QucsFile::createNodeSet(QStringList& Collect, int& countInit,
|
|
|
|
Conductor *pw, Node *p1)
|
|
|
|
{
|
|
|
|
if(pw->Label)
|
|
|
|
if(!pw->Label->initValue.isEmpty())
|
|
|
|
Collect.append("NodeSet:NS" + QString::number(countInit++) + " " +
|
|
|
|
p1->Name + " U=\"" + pw->Label->initValue + "\"");
|
|
|
|
}
|
2004-07-18 18:48:51 +00:00
|
|
|
|
|
|
|
// ---------------------------------------------------
|
2005-10-24 06:10:35 +00:00
|
|
|
void QucsFile::throughAllNodes(bool User, QStringList& Collect,
|
|
|
|
int& countInit, bool Analog)
|
2004-07-18 18:48:51 +00:00
|
|
|
{
|
2005-10-24 06:10:35 +00:00
|
|
|
int z=0;
|
|
|
|
Node *pn, *p2;
|
2004-07-18 18:48:51 +00:00
|
|
|
Wire *pw;
|
|
|
|
Element *pe;
|
2005-10-24 06:10:35 +00:00
|
|
|
bool setName=false;
|
|
|
|
QPtrList<Node> Cons;
|
|
|
|
|
|
|
|
for(pn = Nodes->first(); pn != 0; pn = Nodes->next()) {
|
|
|
|
if(pn->Name.isEmpty() == User) continue; // already named ?
|
|
|
|
if(!User) {
|
|
|
|
if(Analog) pn->Name = "_net";
|
|
|
|
else pn->Name = "nnnet"; // VHDL names must not begin with '_'
|
|
|
|
pn->Name += QString::number(z++); // create node name
|
|
|
|
}
|
|
|
|
else
|
|
|
|
if(pn->State) continue; // already worked on
|
|
|
|
|
|
|
|
if(!Analog) // collect all node names for VHDL signal declaration
|
|
|
|
Signals.append(pn->Name);
|
|
|
|
|
|
|
|
createNodeSet(Collect, countInit, pn, pn);
|
|
|
|
|
|
|
|
pn->State = 1;
|
|
|
|
Cons.append(pn);
|
|
|
|
for(p2 = Cons.first(); p2 != 0; p2 = Cons.next())
|
|
|
|
for(pe = p2->Connections.first(); pe != 0; pe = p2->Connections.next())
|
|
|
|
if(pe->Type == isWire) {
|
|
|
|
pw = (Wire*)pe;
|
|
|
|
if(p2 != pw->Port1) {
|
|
|
|
if(pw->Port1->Name.isEmpty()) {
|
|
|
|
pw->Port1->Name = pn->Name;
|
|
|
|
pw->Port1->State = 1;
|
|
|
|
Cons.append(pw->Port1);
|
|
|
|
setName = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if(pw->Port2->Name.isEmpty()) {
|
|
|
|
pw->Port2->Name = pn->Name;
|
|
|
|
pw->Port2->State = 1;
|
|
|
|
Cons.append(pw->Port2);
|
|
|
|
setName = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(setName) {
|
|
|
|
Cons.findRef(p2); // back to current Connection
|
|
|
|
createNodeSet(Collect, countInit, pw, pn);
|
|
|
|
setName = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Cons.clear();
|
|
|
|
}
|
|
|
|
}
|
2004-07-18 18:48:51 +00:00
|
|
|
|
2005-10-24 06:10:35 +00:00
|
|
|
// ---------------------------------------------------
|
|
|
|
// Follows the wire lines in order to determine the node names for
|
|
|
|
// each component. Output into "stream", NodeSets are collected in
|
|
|
|
// "Collect" and counted with "countInit".
|
|
|
|
bool QucsFile::giveNodeNames(QTextStream *stream, int& countInit,
|
2005-11-28 07:17:35 +00:00
|
|
|
QStringList& Collect, QTextEdit *ErrText, int NumPorts)
|
2005-10-24 06:10:35 +00:00
|
|
|
{
|
2004-07-18 18:48:51 +00:00
|
|
|
// delete the node names
|
2005-10-24 06:10:35 +00:00
|
|
|
for(Node *pn = Nodes->first(); pn != 0; pn = Nodes->next()) {
|
|
|
|
pn->State = 0;
|
|
|
|
if(pn->Label) pn->Name = pn->Label->Name;
|
|
|
|
else pn->Name = "";
|
|
|
|
}
|
2004-07-18 18:48:51 +00:00
|
|
|
|
|
|
|
// set the wire names to the connected node
|
2005-10-24 06:10:35 +00:00
|
|
|
for(Wire *pw = Wires->first(); pw != 0; pw = Wires->next())
|
2004-07-18 18:48:51 +00:00
|
|
|
if(pw->Label != 0) pw->Port1->Name = pw->Label->Name;
|
|
|
|
|
2005-06-23 06:06:40 +00:00
|
|
|
bool r;
|
2004-07-18 18:48:51 +00:00
|
|
|
QString s;
|
2005-10-24 06:10:35 +00:00
|
|
|
// give the ground nodes the name "gnd", and insert subcircuits etc.
|
2004-07-18 18:48:51 +00:00
|
|
|
for(Component *pc = Comps->first(); pc != 0; pc = Comps->next())
|
2005-10-24 06:10:35 +00:00
|
|
|
if(pc->isActive) {
|
2005-11-28 07:17:35 +00:00
|
|
|
if(NumPorts < 0) {
|
2005-10-24 06:10:35 +00:00
|
|
|
if((pc->Type & isAnalogComponent) == 0) {
|
|
|
|
ErrText->insert(QObject::tr("ERROR: Component \"%1\" has no analog model.").arg(pc->Name));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if((pc->Type & isDigitalComponent) == 0) {
|
2005-11-28 07:17:35 +00:00
|
|
|
if(pc->Name.isEmpty())
|
|
|
|
ErrText->insert(QObject::tr("ERROR: No ground symbol allowed in digital simulation."));
|
|
|
|
else
|
|
|
|
ErrText->insert(QObject::tr("ERROR: Component \"%1\" has no digital model.").arg(pc->Name));
|
2005-10-24 06:10:35 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-07-18 18:48:51 +00:00
|
|
|
if(pc->Model == "GND") pc->Ports.getFirst()->Connection->Name = "gnd";
|
|
|
|
else if(pc->Model == "Sub") {
|
|
|
|
s = pc->Props.getFirst()->Value;
|
|
|
|
if(StringList.findIndex(s) >= 0)
|
|
|
|
continue; // insert each subcircuit just one time
|
|
|
|
|
|
|
|
StringList.append(s);
|
2004-08-15 16:24:35 +00:00
|
|
|
QucsDoc *d = new QucsDoc(0, QucsWorkDir.filePath(s));
|
2004-07-18 18:48:51 +00:00
|
|
|
if(!d->File.load()) { // load document if possible
|
|
|
|
delete d;
|
2005-05-18 06:43:54 +00:00
|
|
|
ErrText->insert(QObject::tr("ERROR: Cannot load subcircuit \"%1\".").arg(s));
|
2004-07-18 18:48:51 +00:00
|
|
|
return false;
|
|
|
|
}
|
2004-08-15 16:24:35 +00:00
|
|
|
d->DocName = s;
|
2005-06-23 06:06:40 +00:00
|
|
|
r = d->File.createSubNetlist(
|
2005-11-28 07:17:35 +00:00
|
|
|
stream, countInit, Collect, ErrText, NumPorts);
|
2004-07-18 18:48:51 +00:00
|
|
|
delete d;
|
2005-05-17 06:35:55 +00:00
|
|
|
if(!r) return false;
|
2004-07-18 18:48:51 +00:00
|
|
|
}
|
2005-06-23 06:06:40 +00:00
|
|
|
else if(pc->Model == "Lib") {
|
|
|
|
s = pc->Props.first()->Value + "_";
|
|
|
|
s += pc->Props.next()->Value;
|
|
|
|
if(StringList.findIndex(s) >= 0)
|
|
|
|
continue; // insert each subcircuit just one time
|
|
|
|
|
|
|
|
StringList.append(s);
|
|
|
|
r = ((LibComp*)pc)->outputSubNetlist(stream);
|
|
|
|
if(!r) {
|
|
|
|
ErrText->insert(
|
|
|
|
QObject::tr("ERROR: Cannot load library component \"%1\".").arg(pc->Name));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2005-05-09 06:32:17 +00:00
|
|
|
else if(pc->Model == "SPICE") {
|
2005-05-17 06:35:55 +00:00
|
|
|
s = pc->Props.first()->Value;
|
|
|
|
if(s.isEmpty()) {
|
2005-05-18 06:43:54 +00:00
|
|
|
ErrText->insert(QObject::tr("ERROR: No file name in SPICE component \"%1\".").
|
|
|
|
arg(pc->Name));
|
2005-05-17 06:35:55 +00:00
|
|
|
return false;
|
|
|
|
}
|
2005-05-09 06:32:17 +00:00
|
|
|
if(StringList.findIndex(s) >= 0)
|
|
|
|
continue; // insert each spice component just one time
|
|
|
|
|
|
|
|
StringList.append(s);
|
2005-05-20 06:19:18 +00:00
|
|
|
s += '"'+pc->Props.next()->Value;
|
|
|
|
if(pc->Props.next()->Value == "yes") s = "SPICE \""+s;
|
|
|
|
else s = "SPICEo\""+s;
|
|
|
|
Collect.append(s);
|
2005-05-09 06:32:17 +00:00
|
|
|
}
|
2005-10-24 06:10:35 +00:00
|
|
|
} // of "if(active)"
|
2004-07-18 18:48:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
// work on named nodes first in order to preserve the user given names
|
2005-11-28 07:17:35 +00:00
|
|
|
throughAllNodes(true, Collect, countInit, NumPorts<0);
|
2004-07-18 18:48:51 +00:00
|
|
|
|
|
|
|
// give names to the remaining (unnamed) nodes
|
2005-11-28 07:17:35 +00:00
|
|
|
throughAllNodes(false, Collect, countInit, NumPorts<0);
|
2004-07-18 18:48:51 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
// Write the netlist as subcircuit to the text stream 'NetlistFile'.
|
2005-05-17 06:35:55 +00:00
|
|
|
bool QucsFile::createSubNetlist(QTextStream *stream, int& countInit,
|
2005-11-28 07:17:35 +00:00
|
|
|
QStringList& Collect, QTextEdit *ErrText, int NumPorts)
|
2004-07-18 18:48:51 +00:00
|
|
|
{
|
2005-05-17 06:35:55 +00:00
|
|
|
int i, z;
|
2004-12-19 16:06:24 +00:00
|
|
|
QString s;
|
2005-10-24 06:10:35 +00:00
|
|
|
// TODO: NodeSets have to be put into the subcircuit block.
|
2005-11-28 07:17:35 +00:00
|
|
|
if(!giveNodeNames(stream, countInit, Collect, ErrText, NumPorts))
|
2005-10-24 06:10:35 +00:00
|
|
|
return false;
|
2004-07-18 18:48:51 +00:00
|
|
|
|
|
|
|
QStringList sl;
|
2004-08-22 14:41:35 +00:00
|
|
|
QStringList::Iterator it;
|
2004-07-18 18:48:51 +00:00
|
|
|
Component *pc;
|
|
|
|
// collect all subcircuit ports and sort their node names into "sl"
|
|
|
|
for(pc = Comps->first(); pc != 0; pc = Comps->next())
|
|
|
|
if(pc->Model == "Port") {
|
|
|
|
i = pc->Props.first()->Value.toInt();
|
2004-08-22 14:41:35 +00:00
|
|
|
for(z=sl.size(); z<i; z++)
|
2005-10-24 06:10:35 +00:00
|
|
|
sl.append(" ");
|
2004-08-22 14:41:35 +00:00
|
|
|
it = sl.at(i-1);
|
|
|
|
(*it) = pc->Ports.getFirst()->Connection->Name;
|
2005-11-28 07:17:35 +00:00
|
|
|
if(NumPorts >= 0) {
|
2005-10-24 06:10:35 +00:00
|
|
|
(*it) += ": ";
|
|
|
|
if(pc->Props.at(1)->Value.at(0) == 'o') // output port ?
|
|
|
|
(*it) += "out";
|
|
|
|
else
|
|
|
|
(*it) += "in";
|
|
|
|
(*it) += " bit";
|
|
|
|
}
|
2004-07-18 18:48:51 +00:00
|
|
|
}
|
|
|
|
|
2005-10-24 06:10:35 +00:00
|
|
|
QString Type = properName(Doc->DocName);
|
|
|
|
|
2005-11-28 07:17:35 +00:00
|
|
|
if(NumPorts < 0)
|
2005-10-24 06:10:35 +00:00
|
|
|
(*stream) << "\n.Def:" << Type << " " << sl.join(" ") << '\n';
|
|
|
|
else {
|
|
|
|
|
|
|
|
// remove all node names connected to ports
|
|
|
|
for(it = sl.begin(); it != sl.end(); it++)
|
|
|
|
Signals.remove(Signals.find((*it).section(':',0,0)));
|
|
|
|
|
|
|
|
for(it = Collect.begin(); it != Collect.end(); )
|
|
|
|
if((*it).left(4) == "use ") { // output all subcircuit uses
|
|
|
|
(*stream) << (*it);
|
|
|
|
it = Collect.remove(it);
|
|
|
|
}
|
|
|
|
else it++;
|
|
|
|
|
|
|
|
(*stream) << "\nentity " << Type << " is\n"
|
|
|
|
<< " port (" << sl.join(";\n ") << ");\n"
|
|
|
|
<< "end entity;\n"
|
|
|
|
<< "architecture ARCH" << Type << " of " << Type << " is\n";
|
|
|
|
if(!Signals.isEmpty())
|
|
|
|
(*stream) << " signal " << Signals.join(",\n ") << " : bit;\n";
|
|
|
|
|
|
|
|
// store own subcircuit declaration
|
|
|
|
Collect.append("use work." + Type + ";\n");
|
|
|
|
(*stream) << "begin\n";
|
|
|
|
}
|
|
|
|
Signals.clear(); // was filled in "giveNodeNames()"
|
2004-07-18 18:48:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
// write all components with node names into the netlist file
|
|
|
|
for(pc = Comps->first(); pc != 0; pc = Comps->next()) {
|
2005-10-24 06:10:35 +00:00
|
|
|
if(pc->Model.at(0) == '.') { // no simulations in subcircuits
|
|
|
|
ErrText->insert(
|
|
|
|
QObject::tr("WARNING: Ignore simulation component in subcircuit \"%1\".").arg(Doc->DocName));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if(pc->Model == "Eqn") { // no equations in subcircuits
|
|
|
|
ErrText->insert(
|
|
|
|
QObject::tr("WARNING: Ignore equation in subcircuit \"%1\".").arg(Doc->DocName));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2005-11-28 07:17:35 +00:00
|
|
|
if(NumPorts < 0)
|
2005-10-24 06:10:35 +00:00
|
|
|
s = pc->NetList();
|
|
|
|
else
|
2005-11-28 07:17:35 +00:00
|
|
|
s = pc->VHDL_Code(NumPorts);
|
2005-10-24 06:10:35 +00:00
|
|
|
|
2004-12-19 16:06:24 +00:00
|
|
|
if(!s.isEmpty()) // not inserted: subcircuit ports, disabled components
|
|
|
|
(*stream) << " " << s << "\n";
|
2004-07-18 18:48:51 +00:00
|
|
|
}
|
|
|
|
|
2005-11-28 07:17:35 +00:00
|
|
|
if(NumPorts < 0)
|
2005-10-24 06:10:35 +00:00
|
|
|
(*stream) << ".Def:End\n\n";
|
|
|
|
else
|
|
|
|
(*stream) << "end architecture;\n\n";
|
|
|
|
|
2004-07-18 18:48:51 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
2005-05-17 06:35:55 +00:00
|
|
|
// Determines the node names and writes subcircuits into netlist file.
|
2005-10-24 06:10:35 +00:00
|
|
|
int QucsFile::prepareNetlist(QTextStream& stream, QStringList& Collect,
|
2005-11-28 07:17:35 +00:00
|
|
|
QTextEdit *ErrText)
|
2004-07-18 18:48:51 +00:00
|
|
|
{
|
2005-08-15 06:04:52 +00:00
|
|
|
if(Doc->showBias > 0) Doc->showBias = -1; // do not show DC bias anymore
|
|
|
|
|
2005-11-28 07:17:35 +00:00
|
|
|
bool isTruthTable = false;
|
|
|
|
int allTypes = 0, NumPorts = 0;
|
2005-10-24 06:10:35 +00:00
|
|
|
// Detect simulation domain (analog/digital) by looking at component types.
|
2005-11-28 07:17:35 +00:00
|
|
|
for(Component *pc = Comps->first(); pc != 0; pc = Comps->next()) {
|
|
|
|
if(!pc->isActive) continue;
|
|
|
|
if(pc->Model.at(0) == '.') {
|
|
|
|
if(pc->Model == ".Digi") {
|
|
|
|
if(allTypes & isDigitalComponent) {
|
|
|
|
ErrText->insert(
|
|
|
|
QObject::tr("ERROR: Only one digital simulation allowed."));
|
|
|
|
return -10;
|
|
|
|
}
|
|
|
|
if(pc->Props.getFirst()->Value != "TimeList")
|
|
|
|
isTruthTable = true;
|
|
|
|
allTypes |= isDigitalComponent;
|
|
|
|
}
|
|
|
|
else allTypes |= isAnalogComponent;
|
|
|
|
|
|
|
|
if((allTypes & isComponent) == isComponent) {
|
|
|
|
ErrText->insert(
|
|
|
|
QObject::tr("ERROR: Analog and digital simulations cannot be mixed."));
|
|
|
|
return -10;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(pc->Model == "DigiSource") NumPorts++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if((allTypes & isAnalogComponent) == 0) {
|
|
|
|
if(NumPorts < 1) {
|
|
|
|
ErrText->insert(
|
|
|
|
QObject::tr("ERROR: Digital simulation needs at least one digital source."));
|
|
|
|
return -10;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!isTruthTable) NumPorts = 0;
|
|
|
|
}
|
|
|
|
else NumPorts = -1;
|
2005-10-24 06:10:35 +00:00
|
|
|
|
|
|
|
|
2004-07-18 18:48:51 +00:00
|
|
|
// first line is documentation
|
2005-11-28 07:17:35 +00:00
|
|
|
if(allTypes & isAnalogComponent)
|
2005-10-24 06:10:35 +00:00
|
|
|
stream << "#";
|
|
|
|
else
|
|
|
|
stream << "--";
|
|
|
|
stream << " Qucs " << PACKAGE_VERSION << " " << Doc->DocName << "\n";
|
2005-11-28 07:17:35 +00:00
|
|
|
// if((allTypes & isAnalogComponent) == 0)
|
2005-10-24 06:10:35 +00:00
|
|
|
// stream << "library ieee;\nuse ieee.std_logic_1164.all;\n\n";
|
|
|
|
|
2004-07-18 18:48:51 +00:00
|
|
|
|
2004-12-19 16:06:24 +00:00
|
|
|
int countInit = 0;
|
2005-10-24 06:10:35 +00:00
|
|
|
StringList.clear(); // no subcircuits yet
|
2005-11-28 07:17:35 +00:00
|
|
|
if(!giveNodeNames(&stream, countInit, Collect, ErrText, NumPorts))
|
|
|
|
return -10;
|
2005-10-24 06:10:35 +00:00
|
|
|
|
2005-11-28 07:17:35 +00:00
|
|
|
if(allTypes & isAnalogComponent)
|
|
|
|
return NumPorts;
|
2005-10-24 06:10:35 +00:00
|
|
|
|
|
|
|
QString Type = properName(Doc->DocName);
|
|
|
|
stream << "entity " << Type << " is\n"
|
|
|
|
<< "end entity;\n\n";
|
2005-11-28 07:17:35 +00:00
|
|
|
return NumPorts;
|
2005-05-17 06:35:55 +00:00
|
|
|
}
|
2004-07-18 18:48:51 +00:00
|
|
|
|
2005-05-17 06:35:55 +00:00
|
|
|
// .................................................
|
|
|
|
// write all components with node names into the netlist file
|
2005-11-28 07:17:35 +00:00
|
|
|
QString QucsFile::createNetlist(QTextStream& stream, int NumPorts)
|
2005-05-17 06:35:55 +00:00
|
|
|
{
|
2005-10-24 06:10:35 +00:00
|
|
|
QString Type = properName(Doc->DocName);
|
2005-11-28 07:17:35 +00:00
|
|
|
if(NumPorts >= 0)
|
2005-10-24 06:10:35 +00:00
|
|
|
stream << "architecture ARCH" << Type << " of " << Type << " is\n"
|
|
|
|
<< " signal " << Signals.join(",\n ")
|
|
|
|
<< " : bit;\n"
|
|
|
|
<< "begin\n";
|
|
|
|
Signals.clear(); // was filled in "giveNodeNames()"
|
|
|
|
|
|
|
|
QString s, Time;
|
2004-07-18 18:48:51 +00:00
|
|
|
for(Component *pc = Comps->first(); pc != 0; pc = Comps->next()) {
|
2005-11-28 07:17:35 +00:00
|
|
|
if(NumPorts < 0)
|
2005-10-24 06:10:35 +00:00
|
|
|
s = pc->NetList();
|
|
|
|
else {
|
|
|
|
if(pc->Model.at(0) == '.')
|
2005-11-28 07:17:35 +00:00
|
|
|
Time = pc->Props.at(1)->Value;
|
|
|
|
s = pc->VHDL_Code(NumPorts);
|
2005-10-24 06:10:35 +00:00
|
|
|
}
|
|
|
|
|
2004-07-18 18:48:51 +00:00
|
|
|
if(!s.isEmpty()) // not inserted: subcircuit ports, disabled components
|
|
|
|
stream << s << "\n";
|
|
|
|
}
|
2005-10-24 06:10:35 +00:00
|
|
|
|
2005-11-28 07:17:35 +00:00
|
|
|
if(NumPorts >= 0)
|
2005-10-24 06:10:35 +00:00
|
|
|
stream << "end architecture;\n";
|
|
|
|
|
2005-11-28 07:17:35 +00:00
|
|
|
if(NumPorts > 0)
|
|
|
|
return QString::number(1 << NumPorts) + "ns";
|
2005-10-24 06:10:35 +00:00
|
|
|
return Time;
|
2004-07-18 18:48:51 +00:00
|
|
|
}
|