qucs_s/qucs/qucsdoc.cpp
ela e84aa2293a * added some translatable strings
* NEWS file added
* autogen.sh script added
2003-11-23 14:49:01 +00:00

2106 lines
65 KiB
C++

/***************************************************************************
qucsdoc.cpp - description
-------------------
begin : Wed Sep 3 2003
copyright : (C) 2003 by Michael Margraf
email : margraf@mwt.ee.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. *
* *
***************************************************************************/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include "qucsdoc.h"
#include "component.h"
#include "rectdiagram.h"
#include "polardiagram.h"
#include "smithdiagram.h"
#include "tabdiagram.h"
#include <qmessagebox.h>
#include <qfileinfo.h>
#include <qimage.h>
#include <qiconset.h>
#include <limits.h>
// icon for unsaved files (diskette)
static const char *smallsave_xpm[] = {
"16 17 66 1", " c None",
". c #595963","+ c #E6E6F1","@ c #465460","# c #FEFEFF","$ c #DEDEEE","% c #43535F","& c #D1D1E6",
"* c #5E5E66","= c #FFFFFF","- c #C5C5DF","; c #FCF8F9","> c #BDBDDA",", c #BFBFDC","' c #C4C4DF",
") c #FBF7F7","! c #D6D6E9","~ c #CBCBE3","{ c #B5B5D6","] c #BCBCDA","^ c #C6C6E0","/ c #CFCFE5",
"( c #CEC9DC","_ c #D8D8EA",": c #DADAEB","< c #313134","[ c #807FB3","} c #AEAED1","| c #B7B7D7",
"1 c #E2E2EF","2 c #9393C0","3 c #E3E3F0","4 c #DDD5E1","5 c #E8E8F3","6 c #2F2F31","7 c #7B7BAF",
"8 c #8383B5","9 c #151518","0 c #000000","a c #C0C0DC","b c #8E8FBD","c c #8989BA","d c #E7EEF6",
"e c #282829","f c #6867A1","g c #7373A9","h c #A7A7CD","i c #8080B3","j c #7B7CB0","k c #7070A8",
"l c #6D6DA5","m c #6E6EA6","n c #6969A2","o c #7A79AF","p c #DCDCEC","q c #60609A","r c #7777AC",
"s c #5D5D98","t c #7676AB","u c #484785","v c #575793","w c #50506A","x c #8787B8","y c #53536E",
"z c #07070E","A c #666688",
" . ",
" .+. ",
" .+@#. ",
" .$%###. ",
" .&*####=. ",
" .-.#;#####. ",
" .>,'.#)!!!!~. ",
" .{].'^./(!_:<[.",
".}|.1./2.3456789",
"0a.$11.bc.defg9 ",
" 011h11.ij9kl9 ",
" 0_1h1h.mno9 ",
" 0p12h9qr9 ",
" 0hh9st9 ",
" 09uv9w ",
" 0x9y ",
" zA "};
QucsDoc::QucsDoc(QTabBar *b, const QString& _Name)
{
GridX = 10;
GridY = 10;
GridOn = true;
Scale=1.0;
ViewX1=ViewY1=0;
ViewX2=ViewY2=800;
PosX=PosY=0;
DocName = _Name;
if(_Name.isEmpty()) Tab = new QTab(QObject::tr("untitled"));
else {
QFileInfo Info(DocName);
Tab = new QTab(Info.fileName());
DataSet = Info.baseName()+".dat"; // name of the default dataset
if(Info.extension(false) == "sch")
DataDisplay = Info.baseName()+".dpl"; // name of the default data display
else {
DataDisplay = Info.baseName()+".sch"; // name of the default data display
GridOn = false; // data display without grid (per default)
}
}
SimOpenDpl = true;
Bar = b;
Bar->addTab(Tab); // create tab in TabBar
// Bar->setCurrentTab(Tab); // make it the current ===> Qt 3.1 CRASHES !!!???!!!
Bar->repaint();
DocChanged = false;
Comps.setAutoDelete(true);
Wires.setAutoDelete(true);
Nodes.setAutoDelete(true);
Diags.setAutoDelete(true);
}
QucsDoc::~QucsDoc()
{
Bar->removeTab(Tab); // delete tab in TabBar
}
// ---------------------------------------------------
void QucsDoc::setName(const QString& _Name)
{
DocName = _Name;
QFileInfo Info(DocName);
Tab->setText(Info.fileName()); // remove path from file name and show it in TabBar
DataSet = Info.baseName()+".dat"; // name of the default dataset
if(Info.extension(false) == "sch")
DataDisplay = Info.baseName()+".dpl"; // name of the default data display
else
DataDisplay = Info.baseName()+".sch"; // name of the default data display
}
// ---------------------------------------------------
// Sets the document to be changed or not to be changed.
void QucsDoc::setChanged(bool c)
{
if((!DocChanged) & c) {
Tab->setIconSet(QPixmap(smallsave_xpm));
Bar->layoutTabs();
Bar->repaint();
}
else if(DocChanged & (!c)) {
Tab->setIconSet(QIconSet(0)); // no icon in TabBar
Bar->layoutTabs();
Bar->repaint();
}
DocChanged = c;
}
// ---------------------------------------------------
void QucsDoc::paint(QPainter *p)
{
for(Component *pc = Comps.first(); pc != 0; pc = Comps.next())
pc->paint(p); // paint all components
for(Wire *ptr2 = Wires.first(); ptr2 != 0; ptr2 = Wires.next())
ptr2->paint(p); // paint all wires
for(Node *ptr3 = Nodes.first(); ptr3 != 0; ptr3 = Nodes.next())
ptr3->paint(p); // paint all nodes
for(Diagram *ptr4 = Diags.first(); ptr4 != 0; ptr4 = Diags.next())
ptr4->paint(p); // paint all diagrams
}
// ---------------------------------------------------
// Sets an arbitrary coordinate onto the next grid coordinate.
void QucsDoc::setOnGrid(int& x, int& y)
{
if(x<0) x -= (GridX >> 1) - 1;
else x += GridX >> 1;
x -= x % GridX;
if(y<0) y -= (GridY >> 1) - 1;
else y += GridY >> 1;
y -= y % GridY;
}
// ---------------------------------------------------
void QucsDoc::paintGrid(QPainter *p, int cX, int cY, int Width, int Height)
{
if(!GridOn) return;
int x1 = int(cX/Scale)+ViewX1;
if(x1<0) x1 -= GridX - 1;
else x1 += GridX;
x1 -= x1 % (GridX << 1);
x1 -= ViewX1; x1 = int(x1*Scale);
int y1 = int(cY/Scale)+ViewY1;
if(y1<0) y1 -= GridY - 1;
else y1 += GridY;
y1 -= y1 % (GridY << 1);
y1 -= ViewY1; y1 = int(y1*Scale);
int x2 = x1+Width, y2 = y1+Height;
int dx = int(double(2*GridX)*Scale);
int dy = int(double(2*GridY)*Scale);
p->setPen(QPen(QPen::black,1));
for(int x=x1; x<x2; x+=dx)
for(int y=y1; y<y2; y+=dy)
p->drawPoint(x,y); // paint grid
}
// ---------------------------------------------------
// Inserts a port into the schematic and connects it to another node if the
// coordinates are identical. The node is returned.
Node* QucsDoc::insertNode(int x, int y, Element *e)
{
Node *ptr1;
// check if new node lies upon existing node
for(ptr1 = Nodes.first(); ptr1 != 0; ptr1 = Nodes.next()) { // check every node
if((ptr1->x == x) && (ptr1->y == y)) {
ptr1->Connections.append(e);
break;
}
}
if(ptr1 == 0) { // create new node, if no existing one lies at this position
ptr1 = new Node(x, y);
Nodes.append(ptr1);
ptr1->Connections.append(e); // connect schematic node to component node
}
else return ptr1; // return, if node is not new
// check if the new node lies upon an existing wire
Wire *nw;
for(Wire *ptr2 = Wires.first(); ptr2 != 0; ptr2 = Wires.next()) {
if(ptr2->x1 == x) {
if(ptr2->y1 > y) continue;
if(ptr2->y2 < y) continue;
}
else if(ptr2->y1 == y) {
if(ptr2->x1 > x) continue;
if(ptr2->x2 < x) continue;
}
else continue;
// split the wire into two wires
nw = new Wire(x, y, ptr2->x2, ptr2->y2, ptr1, ptr2->Port2);
ptr2->x2 = x;
ptr2->y2 = y;
ptr2->Port2 = ptr1;
nw->Port2->Connections.append(nw);
ptr1->Connections.append(ptr2);
ptr1->Connections.append(nw);
nw->Port2->Connections.removeRef(ptr2);
Wires.append(nw);
return ptr1;
}
return ptr1;
}
// ---------------------------------------------------
void QucsDoc::insertRawComponent(Component *c)
{
Node *p;
// connect every node of component
for(Port *ptr = c->Ports.first(); ptr != 0; ptr = c->Ports.next()) {
p = insertNode(ptr->x+c->cx, ptr->y+c->cy, c);
ptr->Connection = p; // connect component node to schematic node
}
Comps.append(c);
setChanged(true);
}
// ---------------------------------------------------
void QucsDoc::insertComponent(Component *c)
{
Component *ptr2;
Node *p;
// connect every node of the component
for(Port *ptr = c->Ports.first(); ptr != 0; ptr = c->Ports.next()) {
p = insertNode(ptr->x+c->cx, ptr->y+c->cy, c);
ptr->Connection = p; // connect component node to schematic node
}
bool ok;
QString s;
int max=1, len = c->Name.length(), z;
// determines the name by looking for names with the same prefix and increment the number
if(!c->Name.isEmpty()) {
for(ptr2 = Comps.first(); ptr2 != 0; ptr2 = Comps.next())
if(ptr2->Name.left(len) == c->Name) {
s = ptr2->Name.right(ptr2->Name.length()-len);
z = s.toInt(&ok);
if(ok) if(z >= max) max = z + 1;
}
c->Name += QString::number(max); // create component name with new number
}
if(c->Sign == "Pac") { // power sources have to be numbered
max=1;
// look for existing ports and count the numbers
for(ptr2 = Comps.first(); ptr2 != 0; ptr2 = Comps.next())
if(ptr2->Sign == "Pac") max++;
c->Props.first()->Value = QString::number(max); // create port number
}
else
if(c->Sign == "Port") { // subcircuit ports have to be numbered
max=1;
// look for existing ports and count the numbers
for(ptr2 = Comps.first(); ptr2 != 0; ptr2 = Comps.next())
if(ptr2->Sign == "Port") max++;
c->Props.first()->Value = QString::number(max); // create port number
}
Comps.append(c);
setChanged(true);
}
// ---------------------------------------------------
// Inserts a port into the schematic and connects it to another node if the
// coordinates are identical. If 0 is returned, no new wire is inserted.
// If 2 is returned, the wire line ended.
int QucsDoc::insertWireNode1(Wire *w)
{
Node *pn;
// check if new node lies upon an existing node
for(pn = Nodes.first(); pn != 0; pn = Nodes.next()) // check every node
if(pn->x == w->x1) if(pn->y == w->y1) break;
if(pn != 0) {
pn->Connections.append(w);
w->Port1 = pn;
return 2; // node is not new
}
Wire *pw;
// check if the new node lies upon an existing wire
for(Wire *ptr2 = Wires.first(); ptr2 != 0; ptr2 = Wires.next()) {
if(ptr2->x1 == w->x1) {
if(ptr2->y1 > w->y1) continue;
if(ptr2->y2 < w->y1) continue;
if(ptr2->isHorizontal() == w->isHorizontal()) // (ptr2-wire is vertical)
if(ptr2->y2 >= w->y2) { delete w; return 0; } // new wire lies within an existing wire
else { // one part of the wire lies within an existing wire the other part not
if(ptr2->Port2->Connections.count() == 1) {
w->y1 = ptr2->y1;
w->Port1 = ptr2->Port1;
ptr2->Port1->Connections.removeRef(ptr2); // two wires -> one wire
ptr2->Port1->Connections.append(w);
Nodes.removeRef(ptr2->Port2);
Wires.removeRef(ptr2);
return 2;
}
else {
w->y1 = ptr2->y2;
w->Port1 = ptr2->Port2;
ptr2->Port2->Connections.append(w); // shorten new wire
return 2;
}
}
}
else if(ptr2->y1 == w->y1) {
if(ptr2->x1 > w->x1) continue;
if(ptr2->x2 < w->x1) continue;
if(ptr2->isHorizontal() == w->isHorizontal()) // (ptr2-wire is horizontal)
if(ptr2->x2 >= w->x2) { delete w; return 0; } // new wire lies within an existing wire
else { // one part of the wire lies within an existing wire the other part not
if(ptr2->Port2->Connections.count() == 1) {
w->x1 = ptr2->x1;
w->Port1 = ptr2->Port1;
ptr2->Port1->Connections.removeRef(ptr2); // two wires -> one wire
ptr2->Port1->Connections.append(w);
Nodes.removeRef(ptr2->Port2);
Wires.removeRef(ptr2);
return 2;
}
else {
w->x1 = ptr2->x2;
w->Port1 = ptr2->Port2;
ptr2->Port2->Connections.append(w); // shorten new wire
return 2;
}
}
}
else continue;
pn = new Node(w->x1, w->y1); // create new node
Nodes.append(pn);
pn->Connections.append(w); // connect schematic node to the new wire
w->Port1 = pn;
// split the wire into two wires
pw = new Wire(w->x1, w->y1, ptr2->x2, ptr2->y2, pn, ptr2->Port2);
ptr2->x2 = w->x1;
ptr2->y2 = w->y1;
ptr2->Port2 = pn;
pw->Port2->Connections.append(pw);
pn->Connections.append(ptr2);
pn->Connections.append(pw);
pw->Port2->Connections.removeRef(ptr2);
Wires.append(pw);
return 2;
}
pn = new Node(w->x1, w->y1); // create new node
Nodes.append(pn);
pn->Connections.append(w); // connect schematic node to the new wire
w->Port1 = pn;
return 1;
}
// ---------------------------------------------------
// if possible, connect two horizontal wires to one
bool QucsDoc::connectHWires1(Wire *w)
{
Wire *pw;
Node *n = w->Port1;
if(n->Connections.count() != 2) return true;
pw = (Wire*)n->Connections.first();
if(pw->Type == isWire)
if(pw->isHorizontal())
if(pw->x1 < w->x1) {
w->Name = pw->Name; w->nx = pw->nx; w->ny = pw->ny;
w->delta = pw->delta; // do not move node name label
w->x1 = pw->x1;
w->Port1 = pw->Port1; // new wire lengthens an existing one
Nodes.removeRef(n);
w->Port1->Connections.removeRef(pw);
w->Port1->Connections.append(w);
Wires.removeRef(pw);
}
else {
if(pw->x2 >= w->x2) { // new wire lies complete within an existing one ?
w->Port1->Connections.removeRef(w); // second node not yet made
delete w;
return false;
}
if(pw->Port2->Connections.count() < 2) {
pw->Port1->Connections.removeRef(pw); // existing wire lies within the new one
Nodes.removeRef(pw->Port2);
Wires.removeRef(pw);
}
else {
w->x1 = pw->x2; // shorten new wire according to an existing one
w->Port1->Connections.removeRef(w);
w->Port1 = pw->Port2;
w->Port1->Connections.append(w);
}
}
return true;
}
// ---------------------------------------------------
// if possible, connect two vertical wires to one
bool QucsDoc::connectVWires1(Wire *w)
{
Wire *pw;
Node *n = w->Port1;
if(n->Connections.count() != 2) return true;
pw = (Wire*)n->Connections.first();
if(pw->Type == isWire)
if(!pw->isHorizontal())
if(pw->y1 < w->y1) {
w->Name = pw->Name; w->nx = pw->nx; w->ny = pw->ny;
w->delta = pw->delta; // do not move node name label
w->y1 = pw->y1;
w->Port1 = pw->Port1; // new wire lengthens an existing one
Nodes.removeRef(n);
w->Port1->Connections.removeRef(pw);
w->Port1->Connections.append(w);
Wires.removeRef(pw);
}
else {
if(pw->y2 >= w->y2) { // new wire lies complete within an existing one ?
w->Port1->Connections.removeRef(w); // second node not yet made
delete w;
return false;
}
if(pw->Port2->Connections.count() < 2) {
pw->Port1->Connections.removeRef(pw); // existing wire lies within the new one
Nodes.removeRef(pw->Port2);
Wires.removeRef(pw);
}
else {
w->y1 = pw->y2; // shorten new wire according to an existing one
w->Port1->Connections.removeRef(w);
w->Port1 = pw->Port2;
w->Port1->Connections.append(w);
}
}
return true;
}
// ---------------------------------------------------
// Inserts a port into the schematic and connects it to another node if the
// coordinates are identical. If 0 is returned, no new wire is inserted.
// If 2 is returned, the wire line ended.
int QucsDoc::insertWireNode2(Wire *w)
{
Node *pn;
// check if new node lies upon an existing node
for(pn = Nodes.first(); pn != 0; pn = Nodes.next()) // check every node
if(pn->x == w->x2) if(pn->y == w->y2) break;
if(pn != 0) {
pn->Connections.append(w);
w->Port2 = pn;
return 2; // node is not new
}
Wire *pw;
// check if the new node lies upon an existing wire
for(Wire *ptr2 = Wires.first(); ptr2 != 0; ptr2 = Wires.next()) {
if(ptr2->x1 == w->x2) {
if(ptr2->y1 > w->y2) continue;
if(ptr2->y2 < w->y2) continue;
if(ptr2->isHorizontal() == w->isHorizontal()) // (ptr2-wire is vertical)
// one part of the wire lies within an existing wire the other part not
if(ptr2->Port1->Connections.count() == 1) {
w->y2 = ptr2->y2;
w->Port2 = ptr2->Port2;
ptr2->Port2->Connections.removeRef(ptr2); // two wires -> one wire
ptr2->Port2->Connections.append(w);
Nodes.removeRef(ptr2->Port1);
Wires.removeRef(ptr2);
return 2;
}
else {
w->y2 = ptr2->y1;
w->Port2 = ptr2->Port1;
ptr2->Port1->Connections.append(w); // shorten new wire
return 2;
}
}
else if(ptr2->y1 == w->y2) {
if(ptr2->x1 > w->x2) continue;
if(ptr2->x2 < w->x2) continue;
if(ptr2->isHorizontal() == w->isHorizontal()) // (ptr2-wire is horizontal)
// one part of the wire lies within an existing wire the other part not
if(ptr2->Port1->Connections.count() == 1) {
w->x2 = ptr2->x2;
w->Port2 = ptr2->Port2;
ptr2->Port2->Connections.removeRef(ptr2); // two wires -> one wire
ptr2->Port2->Connections.append(w);
Nodes.removeRef(ptr2->Port1);
Wires.removeRef(ptr2);
return 2;
}
else {
w->x2 = ptr2->x1;
w->Port2 = ptr2->Port1;
ptr2->Port1->Connections.append(w); // shorten new wire
return 2;
}
}
else continue;
pn = new Node(w->x2, w->y2); // create new node
Nodes.append(pn);
pn->Connections.append(w); // connect schematic node to the new wire
w->Port2 = pn;
// split the wire into two wires
pw = new Wire(w->x2, w->y2, ptr2->x2, ptr2->y2, pn, ptr2->Port2);
ptr2->x2 = w->x2;
ptr2->y2 = w->y2;
ptr2->Port2 = pn;
pw->Port2->Connections.append(pw);
pn->Connections.append(ptr2);
pn->Connections.append(pw);
pw->Port2->Connections.removeRef(ptr2);
Wires.append(pw);
return 2;
}
pn = new Node(w->x2, w->y2); // create new node
Nodes.append(pn);
pn->Connections.append(w); // connect schematic node to the new wire
w->Port2 = pn;
return 1;
}
// ---------------------------------------------------
// if possible, connect two horizontal wires to one
bool QucsDoc::connectHWires2(Wire *w)
{
Wire *pw;
Node *n = w->Port2;
if(n->Connections.count() != 2) return true;
pw = (Wire*)n->Connections.first();
if(pw->Type == isWire)
if(pw->isHorizontal())
if(pw->x2 > w->x2) {
w->Name = pw->Name; w->nx = pw->nx; w->ny = pw->ny;
w->delta = pw->delta; // do not move node name label
w->x2 = pw->x2;
w->Port2 = pw->Port2; // new wire lengthens an existing one
Nodes.removeRef(n);
w->Port2->Connections.removeRef(pw);
w->Port2->Connections.append(w);
Wires.removeRef(pw);
}
else {
// if(pw->x2 >= w->x2) { // new wire lies complete within an existing one ?
// w->Port1->Connections.removeRef(w); // second node not yet made
// delete w;
// return false;
// }
if(pw->Port1->Connections.count() < 2) {
pw->Port2->Connections.removeRef(pw); // existing wire lies within the new one
Nodes.removeRef(pw->Port1);
Wires.removeRef(pw);
}
else {
w->x2 = pw->x1; // shorten new wire according to an existing one
w->Port2->Connections.removeRef(w);
w->Port2 = pw->Port1;
w->Port2->Connections.append(w);
}
}
return true;
}
// ---------------------------------------------------
// if possible, connect two vertical wires to one
bool QucsDoc::connectVWires2(Wire *w)
{
Wire *pw;
Node *n = w->Port2;
if(n->Connections.count() != 2) return true;
pw = (Wire*)n->Connections.first();
if(pw->Type == isWire)
if(!pw->isHorizontal())
if(pw->y2 > w->y2) {
w->Name = pw->Name; w->nx = pw->nx; w->ny = pw->ny;
w->delta = pw->delta; // do not move node name label
w->y2 = pw->y2;
w->Port2 = pw->Port2; // new wire lengthens an existing one
Nodes.removeRef(n);
w->Port2->Connections.removeRef(pw);
w->Port2->Connections.append(w);
Wires.removeRef(pw);
}
else {
// if(pw->y2 >= w->y2) { // new wire lies complete within an existing one ?
// w->Port1->Connections.removeRef(w); // second node not yet made
// delete w;
// return false;
// }
if(pw->Port1->Connections.count() < 2) {
pw->Port2->Connections.removeRef(pw); // existing wire lies within the new one
Nodes.removeRef(pw->Port1);
Wires.removeRef(pw);
}
else {
w->y2 = pw->y1; // shorten new wire according to an existing one
w->Port2->Connections.removeRef(w);
w->Port2 = pw->Port1;
w->Port2->Connections.append(w);
}
}
return true;
}
// ---------------------------------------------------
// Inserts a vertical or horizontal wire into the schematic and connects the ports that hit together.
// Returns whether the beginning and ending (the ports of the wire) are connected or not.
int QucsDoc::insertWire(Wire *w)
{
int tmp, con = 0;
bool ok;
// change coordinates if necessary (port 1 coordinates must be less port 2 coordinates)
if(w->x1 > w->x2) { tmp = w->x1; w->x1 = w->x2; w->x2 = tmp; }
else
if(w->y1 > w->y2) { tmp = w->y1; w->y1 = w->y2; w->y2 = tmp; }
else con = 0x100;
tmp = insertWireNode1(w);
if(tmp == 0) return 3; // no new wire and no open connection
if(tmp > 1) con |= 2; // insert node and remember if it remains open
if(w->isHorizontal()) ok = connectHWires1(w);
else ok = connectVWires1(w);
if(!ok) return 3;
tmp = insertWireNode2(w);
if(tmp == 0) return 3; // no new wire and no open connection
if(tmp > 1) con |= 1; // insert node and remember if it remains open
if(w->isHorizontal()) ok = connectHWires2(w);
else ok = connectVWires2(w);
if(!ok) return 3;
if(con > 255) con = ((con >> 1) & 1) | ((con << 1) & 2); // change node 1 and 2
Wires.append(w); // add wire to the schematic
int n1, n2;
Wire *pw, *nw;
Node *pn, *pn2;
// ................................................................
// check if the new line covers existing nodes
for(pw = Wires.current(); pw != 0; pw = Wires.next()) // in order to also check new appearing wires
for(pn = Nodes.first(); pn != 0; ) { // check every node
if(pn->x == pw->x1) {
if(pn->y <= pw->y1) { pn = Nodes.next(); continue; }
if(pn->y >= pw->y2) { pn = Nodes.next(); continue; }
}
else if(pn->y == pw->y1) {
if(pn->x <= pw->x1) { pn = Nodes.next(); continue; }
if(pn->x >= pw->x2) { pn = Nodes.next(); continue; }
}
else { pn = Nodes.next(); continue; }
n1 = 2; n2 = 3;
pn2 = pn;
// check all connections of the current node
for(Element *pe = pn->Connections.first(); pe != 0; pe = pn->Connections.next()) {
if(pe->Type != isWire) continue;
nw = (Wire*)pe;
if(pw->isHorizontal() == nw->isHorizontal()) { // wire lies within the new
pn = nw->Port1;
pn2 = nw->Port2;
n1 = pn->Connections.count();
n2 = pn2->Connections.count();
if(n1 == 1) {
Nodes.removeRef(pn); // delete node 1 if open
pn = pn2;
pn->Connections.removeRef(nw); // remove connection
}
if(n2 == 1) {
pn->Connections.removeRef(nw); // remove connection
Nodes.removeRef(pn2); // delete node 2 if open
pn2 = pn;
}
if(pn == pn2) {
Wires.removeRef(nw); // delete wire
Wires.findRef(pw); // set back to current wire
}
break;
}
}
if(n1 == 1) if(n2 == 1) continue;
// split wire into two wires
nw = new Wire(pw->x1, pw->y1, pn->x, pn->y, pw->Port1, pn);
pn->Connections.append(nw);
Wires.append(nw);
Wires.findRef(pw);
pw->Port1->Connections.removeRef(pw);
pw->Port1->Connections.append(nw);
pw->x1 = pn2->x;
pw->y1 = pn2->y;
pw->Port1 = pn2;
pn2->Connections.append(pw);
pn = Nodes.next();
}
oneLabel(w->Port1);
setChanged(true);
return con;
}
// ---------------------------------------------------
Component* QucsDoc::searchSelSubcircuit()
{
Component *sub=0;
for(Component *pc = Comps.first(); pc != 0; pc = Comps.next()) { // test all components
if(!pc->isSelected) continue;
if(pc->Sign.left(3) != "Sub") continue;
if(sub != 0) return 0; // more than one subcircuit selected
sub = pc;
}
return sub;
}
// ---------------------------------------------------
Component* QucsDoc::selectedComponent(int x, int y)
{
int x1, y1, x2, y2;
for(Component *pc = Comps.first(); pc != 0; pc = Comps.next()) { // test all components
pc->Bounding(x1, y1, x2, y2);
if((x >= x1) & (x <= x2) & (y >= y1) & (y <= y2)) return pc;
}
return 0;
}
// ---------------------------------------------------
Diagram* QucsDoc::selectedDiagram(int x, int y)
{
int x1, y1, x2, y2;
for(Diagram *ptr1 = Diags.first(); ptr1 != 0; ptr1 = Diags.next()) { // test all diagrams
ptr1->Bounding(x1, y1, x2, y2);
if((x >= x1) & (x <= x2) & (y >= y1) & (y <= y2)) return ptr1;
}
return 0;
}
// ---------------------------------------------------
Wire* QucsDoc::selectedWire(int x, int y)
{
setOnGrid(x, y);
for(Wire *ptr = Wires.first(); ptr != 0; ptr = Wires.next()) // test all wires
if(ptr->x1 <= x) if(ptr->x2 >= x) if(ptr->y1 <= y) if(ptr->y2 >= y)
return ptr;
return 0;
}
// ---------------------------------------------------
// Follows a wire line and selects it.
void QucsDoc::selectWireLine(Element *pe, Node *pn, bool ctrl)
{
while(pn->Connections.count() == 2) {
if(pn->Connections.first() == pe) pe = pn->Connections.last();
else pe = pn->Connections.first();
if(pe->Type != isWire) break;
if(ctrl) pe->isSelected ^= ctrl;
else pe->isSelected = true;
if(((Wire*)pe)->Port1 == pn) pn = ((Wire*)pe)->Port2;
else pn = ((Wire*)pe)->Port1;
}
}
// ---------------------------------------------------
// Selects the element that contains the coordinates x/y.
// Returns the pointer to the element.
// If 'flag' is true, the element can be deselected.
Element* QucsDoc::selectElement(int x, int y, bool flag)
{
int x1, y1, x2, y2;
Element *pe_1st=0, *pe_sel=0;
for(Component *pc = Comps.last(); pc != 0; pc = Comps.prev()) { // test all components
pc->Bounding(x1, y1, x2, y2);
if(x >= x1) if(x <= x2) if(y >= y1) if(y <= y2) {
if(flag) { pc->isSelected ^= flag; return pc; }
if(pe_sel != 0) { pe_sel->isSelected = false; pc->isSelected = true; return pc; }
if(pe_1st == 0) pe_1st = pc; // give access to elements that lie beneath
if(pc->isSelected) pe_sel = pc;
}
}
setOnGrid(x, y);
for(Wire *pw = Wires.last(); pw != 0; pw = Wires.prev()) { // test all wires
if(pw->x1 <= x) if(pw->x2 >= x) if(pw->y1 <= y) if(pw->y2 >= y) {
if(flag) { pw->isSelected ^= flag; return pw; }
if(pe_sel != 0) { pe_sel->isSelected = false; pw->isSelected = true; return pw; }
if(pe_1st == 0) pe_1st = pw; // give access to elements that lie beneath
if(pw->isSelected) pe_sel = pw;
}
}
for(Diagram *pd = Diags.last(); pd != 0; pd = Diags.prev()) { // test all diagrams
pd->Bounding(x1, y1, x2, y2);
if(x >= x1) if(x <= x2) if(y >= y1) if(y <= y2) {
if(flag) { pd->isSelected ^= flag; return pd; }
if(pe_sel != 0) { pe_sel->isSelected = false; pd->isSelected = true; return pd; }
if(pe_1st == 0) pe_1st = pd; // give access to elements that lie beneath
if(pd->isSelected) pe_sel = pd;
}
}
if(pe_1st != 0) pe_1st->isSelected = true;
return pe_1st;
}
// ---------------------------------------------------
// Deselects all elements except 'e'.
void QucsDoc::deselectElements(Element *e)
{
for(Component *pc = Comps.first(); pc != 0; pc = Comps.next()) // test all components
if(e != pc) pc->isSelected = false;
for(Wire *pw = Wires.first(); pw != 0; pw = Wires.next()) // test all wires
if(e != pw) pw->isSelected = false;
for(Diagram *pd = Diags.first(); pd != 0; pd = Diags.next()) // test all diagrams
if(e != pd) pd->isSelected = false;
}
// ---------------------------------------------------
// For moving elements: If the moving element is connected to a not moving element,
// insert two wires. If the connected element is already a wire, use this wire.
// Otherwise create new wire.
void QucsDoc::NewMovingWires(QPtrList<Element> *p, Node *pn)
{
if(pn->Connections.count() < 1) return; // return, if connected node is also moving
Element *pe = pn->Connections.getFirst();
if(pe == (Element*)1) return; // return, if it was already treated this way
pn->Connections.prepend((Element*)1); // to avoid doubling
if(pn->Connections.count() == 2) // 2, because of prepend (Element*)1
if(pe->Type == isWire) { // is it connected to exactly one wire ?
// .................................................
Wire *pw2=0, *pw = (Wire*)pe;
Node *pn2 = pw->Port1;
if(pn2 == pn) pn2 = pw->Port2;
if(pn2->Connections.count() == 2) { // two existing wires connected ?
Element *pe2 = pn2->Connections.getFirst();
if(pe2 == pe) pe2 = pn2->Connections.getLast();
if(pe2 != (Element*)1)
if(pe2->Type == isWire) // the connected wire is again connected to exactly one wire ?
pw2 = (Wire*)pe2;
}
// .................................................
// reuse one wire
p->append(pw);
pw->Port1->Connections.removeRef(pw); // remove connection 1
pw->Port2->Connections.removeRef(pw); // remove connection 2
Wires.removeRef(pw);
int mask = 1;
if(pw->isHorizontal()) mask = 2;
if(pw->Port1 != pn) {
pw->Port1 = (Node*)mask;
pw->Port2 = (Node*)3;
}
else {
pw->Port1 = (Node*)3;
pw->Port2 = (Node*)mask;
}
// .................................................
// create new wire ?
if(pw2 == 0) {
if(pw->Port1 == (Node*)3)
p->append(new Wire(pw->x2, pw->y2, pw->x2, pw->y2, (Node*)mask, (Node*)0));
else
p->append(new Wire(pw->x1, pw->y1, pw->x1, pw->y1, (Node*)mask, (Node*)0));
return;
}
// .................................................
// reuse a second wire
p->append(pw2);
pw2->Port1->Connections.removeRef(pw2); // remove connection 1
pw2->Port2->Connections.removeRef(pw2); // remove connection 2
Wires.removeRef(pw2);
if(pw2->Port1 != pn2) {
pw2->Port1 = (Node*)0;
pw2->Port2 = (Node*)mask;
}
else {
pw2->Port1 = (Node*)mask;
pw2->Port2 = (Node*)0;
}
return;
}
p->append(new Wire(pn->x, pn->y, pn->x, pn->y, (Node*)0, (Node*)1)); // only x2 moving
p->append(new Wire(pn->x, pn->y, pn->x, pn->y, (Node*)1, (Node*)3)); // x1, x2, y2 moving
}
// ---------------------------------------------------
// For moving of elements: Copies all selected elements into the
// list 'p' and deletes them from the document.
void QucsDoc::copySelectedElements(QPtrList<Element> *p)
{
Port *pp;
Component *pc;
Wire *pw;
Diagram *pd;
Element *pe;//, *pe1;
Node *pn;
Comps.setAutoDelete(false);
Wires.setAutoDelete(false);
Diags.setAutoDelete(false);
for(pc = Comps.first(); pc != 0; ) // test all components
if(pc->isSelected) {
p->append(pc);
for(pp = pc->Ports.first(); pp!=0; pp = pc->Ports.next())
pp->Connection->Connections.removeRef((Element*)pc); // delete all port connections
Comps.removeRef(pc); // delete component
pc = Comps.current();
}
else pc = Comps.next();
for(pw = Wires.first(); pw != 0; ) // test all wires
if(pw->isSelected) {
p->append(pw);
pw->Port1->Connections.removeRef(pw); // remove connection 1
pw->Port2->Connections.removeRef(pw); // remove connection 2
Wires.removeRef(pw);
pw = Wires.current();
}
else pw = Wires.next();
for(pd = Diags.first(); pd != 0; ) // test all diagrams
if(pd->isSelected) {
p->append(pd);
Diags.remove();
pd = Diags.current();
}
else pd = Diags.next();
// ..............................................
// Inserts wires, if a connection to a not moving element is found.
for(pe = p->last(); pe!=0; pe = p->prev()) // go backwards in order not to test the new insertions
switch(pe->Type) {
case isComponent:
pc = (Component*)pe;
for(pp = pc->Ports.first(); pp!=0; pp = pc->Ports.next())
NewMovingWires(p, pp->Connection);
p->findRef(pe); // back to the real current pointer
break;
case isWire:
pw = (Wire*)pe;
NewMovingWires(p, pw->Port1);
NewMovingWires(p, pw->Port2);
p->findRef(pe); // back to the real current pointer
break;
default: ; // avoid compiler warning
}
Comps.setAutoDelete(true);
Wires.setAutoDelete(true);
Diags.setAutoDelete(true);
// delete the unused nodes
for(pn = Nodes.first(); pn!=0; ) {
if(pn->Connections.getFirst() == (Element*)1) {
pn->Connections.removeFirst(); // delete tag
if(pn->Connections.count() == 2)
if(oneTwoWires(pn)) { // if possible, connect two wires to one
pn = Nodes.current();
continue;
}
}
if(pn->Connections.count() == 0) {
Nodes.remove();
pn = Nodes.current();
continue;
}
pn = Nodes.next();
}
}
// ---------------------------------------------------
// Selects components that lie within the rectangle x1/y1, x2/y2.
int QucsDoc::selectComponents(int x1, int y1, int x2, int y2, bool flag)
{
int z=0; // counts selected elements
int cx1, cy1, cx2, cy2;
if(x1 > x2) { cx1 = x1; x1 = x2; x2 = cx1; }
if(y1 > y2) { cy1 = y1; y1 = y2; y2 = cy1; }
for(Component *pc = Comps.first(); pc != 0; pc = Comps.next()) { // test all components
pc->Bounding(cx1, cy1, cx2, cy2);
if(cx1 >= x1) if(cx2 <= x2) if(cy1 >= y1) if(cy2 <= y2) {
pc->isSelected = true; z++;
continue;
}
if(pc->isSelected &= flag) z++;
}
for(Wire *pw = Wires.first(); pw != 0; pw = Wires.next()) { // test all wires
if(pw->x1 >= x1) if(pw->x2 <= x2) if(pw->y1 >= y1) if(pw->y2 <= y2) {
pw->isSelected = true; z++;
continue;
}
if(pw->isSelected &= flag) z++;
}
for(Diagram *pd = Diags.first(); pd != 0; pd = Diags.next()) { // test all diagrams
pd->Bounding(cx1, cy1, cx2, cy2);
if(cx1 >= x1) if(cx2 <= x2) if(cy1 >= y1) if(cy2 <= y2) {
pd->isSelected = true; z++;
continue;
}
if(pd->isSelected &= flag) z++;
}
return z;
}
// ---------------------------------------------------
// Activates/deactivates components that lie within the rectangle x1/y1, x2/y2.
void QucsDoc::activateComps(int x1, int y1, int x2, int y2)
{
int cx1, cy1, cx2, cy2;
if(x1 > x2) { cx1 = x1; x1 = x2; x2 = cx1; }
if(y1 > y2) { cy1 = y1; y1 = y2; y2 = cy1; }
for(Component *pc = Comps.first(); pc != 0; pc = Comps.next()) { // test all components
pc->Bounding(cx1, cy1, cx2, cy2);
if(cx1 >= x1) if(cx2 <= x2) if(cy1 >= y1) if(cy2 <= y2)
pc->isActive ^= true; // change "active status"
}
}
// ---------------------------------------------------
// Activate/deactivate component, if x/y lies within its boundings.
bool QucsDoc::activateComponent(int x, int y)
{
int x1, y1, x2, y2;
for(Component *pc = Comps.first(); pc != 0; pc = Comps.next()) { // test all components
pc->Bounding(x1, y1, x2, y2);
if(x >= x1) if(x <= x2) if(y >= y1) if(y <= y2) {
pc->isActive ^= true; // change "active status"
setChanged(true);
return true;
}
}
return false;
}
// ---------------------------------------------------
// Activates/deactivates all selected elements.
bool QucsDoc::activateComponents()
{
bool sel = false;
for(Component *pc = Comps.first(); pc != 0; pc = Comps.next()) // all selected component
if(pc->isSelected) {
pc->isActive ^= true; // change "active status"
sel = true;
}
setChanged(sel);
return sel;
}
// ---------------------------------------------------
// Test, if wire connects wire line with more than one label and delete all further labels.
void QucsDoc::oneLabel(Node *n1)
{
Wire *pw;
Node *pn;
bool named=false; // wire line already named ?
QPtrList<Node> Cons;
for(pn = Nodes.first(); pn!=0; pn = Nodes.next())
if(!pn->isNamed) pn->Name = ""; // erase all node names
n1->Name = "x"; // mark Node as already checked
Cons.append(n1);
for(pn = Cons.first(); pn!=0; pn = Cons.next())
for(Element *pe = pn->Connections.first(); pe!=0; pe = pn->Connections.next()) {
if(pe->Type != isWire) continue;
pw = (Wire*)pe;
if(pn != pw->Port1) {
if(!pw->Port1->Name.isEmpty()) continue;
pw->Port1->Name = "x"; // mark Node as already checked
Cons.append(pw->Port1);
Cons.findRef(pn);
}
else {
if(!pw->Port2->Name.isEmpty()) continue;
pw->Port2->Name = "x"; // mark Node as already checked
Cons.append(pw->Port2);
Cons.findRef(pn);
}
if(!pw->Name.isEmpty()) {
if(named) pw->Name = ""; // erase double names
else named = true;
}
}
}
// ---------------------------------------------------
// Test, if wire line is already labeled and returns a pointer to the labeled wire.
Wire* QucsDoc::getWireLabel(Wire *w)
{
Wire *pw;
Node *pn;
QPtrList<Node> Cons;
if(!w->Name.isEmpty()) return w;
for(pn = Nodes.first(); pn!=0; pn = Nodes.next())
if(!pn->isNamed) pn->Name = ""; // erase all node names
Cons.append(w->Port1);
w->Port1->Name = "x"; // mark Node as already checked
for(pn = Cons.first(); pn!=0; pn = Cons.next())
for(Element *pe = pn->Connections.first(); pe!=0; pe = pn->Connections.next()) {
if(pe->Type != isWire) continue;
pw = (Wire*)pe;
if(!pw->Name.isEmpty()) return pw;
if(pn != pw->Port1) {
if(!pw->Port1->Name.isEmpty()) continue;
pw->Port1->Name = "x"; // mark Node as already checked
Cons.append(pw->Port1);
Cons.findRef(pn);
}
else {
if(!pw->Port2->Name.isEmpty()) continue;
pw->Port2->Name = "x"; // mark Node as already checked
Cons.append(pw->Port2);
Cons.findRef(pn);
}
}
return 0;
}
// ---------------------------------------------------
void QucsDoc::sizeOfAll(int& xmin, int& ymin, int& xmax, int& ymax)
{
xmin=INT_MAX;
ymin=INT_MAX;
xmax=INT_MIN;
ymax=INT_MIN;
Component *pc;
Diagram *pd;
Wire *pw;
if(Comps.isEmpty())
if(Wires.isEmpty())
if(Diags.isEmpty()) {
xmin = xmax = 0;
ymin = ymax = 0;
return;
}
int x1, y1, x2, y2;
for(pc = Comps.first(); pc != 0; pc = Comps.next()) { // find bounds of all components
pc->entireBounds(x1, y1, x2, y2);
if(x1 < xmin) xmin = x1;
if(x2 > xmax) xmax = x2;
if(y1 < ymin) ymin = y1;
if(y2 > ymax) ymax = y2;
}
for(pw = Wires.first(); pw != 0; pw = Wires.next()) { // find bounds of all wires
if(pw->x1 < xmin) xmin = pw->x1;
if(pw->x2 > xmax) xmax = pw->x2;
if(pw->y1 < ymin) ymin = pw->y1;
if(pw->y2 > ymax) ymax = pw->y2;
}
for(pd = Diags.first(); pd != 0; pd = Diags.next()) { // find bounds of all diagrams
pd->Bounding(x1, y1, x2, y2);
if(x1 < xmin) xmin = x1;
if(x2 > xmax) xmax = x2;
if(y1 < ymin) ymin = y1;
if(y2 > ymax) ymax = y2;
}
}
// ---------------------------------------------------
// Sets the component ports.
void QucsDoc::setCompPorts(Component *pc)
{
for(Port *pp = pc->Ports.first(); pp!=0; pp = pc->Ports.next()) {
pp->Connection->Connections.removeRef((Element*)pc); // delete all port connections
switch(pp->Connection->Connections.count()) {
case 0: Nodes.removeRef(pp->Connection);
break;
case 2: oneTwoWires(pp->Connection); // if possible, connect two wires to one
default: ;
}
// connect component node to schematic node
pp->Connection = insertNode(pp->x+pc->cx, pp->y+pc->cy, pc);
}
}
// ---------------------------------------------------
bool QucsDoc::copyCompsWires(int& x1, int& y1, int& x2, int& y2)
{
x1=INT_MAX;
y1=INT_MAX;
x2=INT_MIN;
y2=INT_MIN;
Component *pc;
Wire *pw;
for(pc = Comps.first(); pc != 0; ) { // find bounds of all selected components
if(pc->isSelected) if(pc->Ports.count() > 0) { // do not rotate components without ports
if(pc->cx < x1) x1 = pc->cx;
if(pc->cx > x2) x2 = pc->cx;
if(pc->cy < y1) y1 = pc->cy;
if(pc->cy > y2) y2 = pc->cy;
ElementCache.append(pc);
deleteComp(pc);
pc = Comps.current();
continue;
}
pc = Comps.next();
}
for(pw = Wires.first(); pw != 0; ) // find bounds of all selected wires
if(pw->isSelected) {
if(pw->x1 < x1) x1 = pw->x1;
if(pw->x2 > x2) x2 = pw->x2;
if(pw->y1 < y1) y1 = pw->y1;
if(pw->y2 > y2) y2 = pw->y2;
ElementCache.append(pw);
deleteWire(pw);
pw = Wires.current();
}
else pw = Wires.next();
if(y1 == INT_MAX) return false; // no element selected
return true;
}
// ---------------------------------------------------
// Rotates all selected components around their midpoint.
bool QucsDoc::rotateComponents()
{
Wires.setAutoDelete(false);
Comps.setAutoDelete(false);
int x1, y1, x2, y2;
if(!copyCompsWires(x1, y1, x2, y2)) return false;
x1 = (x1+x2) >> 1; // center for rotation
y1 = (y1+y2) >> 1;
setOnGrid(x1, y1);
Component *pc;
Wire *pw;
// re-insert elements
for(Element *pe = ElementCache.first(); pe != 0; pe = ElementCache.next())
switch(pe->Type) {
case isComponent: pc = (Component*)pe;
pc->rotate(); // mirror component !before! mirroring its center
x2 = x1 - pc->cx;
pc->setCenter(pc->cy - y1 + x1, x2 + y1);
insertRawComponent(pc);
break;
case isWire: pw = (Wire*)pe;
x2 = pw->x1;
pw->x1 = pw->y1 - y1 + x1;
pw->y1 = x1 - x2 + y1;
x2 = pw->x2;
pw->x2 = pw->y2 - y1 + x1;
pw->y2 = x1 - x2 + y1;
insertWire(pw);
break;
default: ;
}
Wires.setAutoDelete(true);
Comps.setAutoDelete(true);
ElementCache.clear();
setChanged(true);
return true;
}
// ---------------------------------------------------
// Mirrors all selected components. First copy them to 'ElementCache', then rotate and insert again.
bool QucsDoc::mirrorXComponents()
{
Wires.setAutoDelete(false);
Comps.setAutoDelete(false);
int x1, y1, x2, y2;
if(!copyCompsWires(x1, y1, x2, y2)) return false;
y1 = (y1+y2) >> 1; // axis for mirroring
setOnGrid(y2, y1);
Wire *pw;
Component *pc;
// re-insert elements
for(Element *pe = ElementCache.first(); pe != 0; pe = ElementCache.next())
switch(pe->Type) {
case isComponent: pc = (Component*)pe;
pc->mirrorX(); // mirror component !before! mirroring its center
pc->setCenter(pc->cx, (y1<<1) - pc->cy);
insertRawComponent(pc);
break;
case isWire: pw = (Wire*)pe;
pw->y1 = (y1<<1) - pw->y1;
pw->y2 = (y1<<1) - pw->y2;
insertWire(pw);
break;
default: ;
}
Wires.setAutoDelete(true);
Comps.setAutoDelete(true);
ElementCache.clear();
setChanged(true);
return true;
}
// ---------------------------------------------------
// Mirrors all selected components. First copy them to 'ElementCache', then rotate and insert again.
bool QucsDoc::mirrorYComponents()
{
Wires.setAutoDelete(false);
Comps.setAutoDelete(false);
int x1, y1, x2, y2;
if(!copyCompsWires(x1, y1, x2, y2)) return false;
x1 = (x1+x2) >> 1; // axis for mirroring
setOnGrid(x1, x2);
Wire *pw;
Component *pc;
// re-insert elements
for(Element *pe = ElementCache.first(); pe != 0; pe = ElementCache.next())
switch(pe->Type) {
case isComponent: pc = (Component*)pe;
pc->mirrorY(); // mirror component !before! mirroring its center
pc->setCenter((x1<<1) - pc->cx, pc->cy);
insertRawComponent(pc);
break;
case isWire: pw = (Wire*)pe;
pw->x1 = (x1<<1) - pw->x1;
pw->x2 = (x1<<1) - pw->x2;
insertWire(pw);
break;
default: ;
}
Wires.setAutoDelete(true);
Comps.setAutoDelete(true);
ElementCache.clear();
setChanged(true);
return true;
}
// ---------------------------------------------------
// If possible, make one wire out of two wires.
bool QucsDoc::oneTwoWires(Node *n)
{
Wire *e3;
Wire *e1 = (Wire*)n->Connections.first(); // two wires -> one wire
Wire *e2 = (Wire*)n->Connections.last();
if(e1->Type == isWire) if(e2->Type == isWire)
if(e1->isHorizontal() == e2->isHorizontal()) {
if(e1->x1 == e2->x2) if(e1->y1 == e2->y2) {
e3 = e1; e1 = e2; e2 = e3; // e1 must have lesser coordinates
}
if(!e2->Name.isEmpty()) { // take over the node name label ?
e1->Name = e2->Name;
e1->nx = e2->nx;
e1->ny = e2->ny;
e1->delta = e2->delta + e1->x2-e1->x1 + e1->y2-e1->y1;
}
e1->x2 = e2->x2;
e1->y2 = e2->y2;
e1->Port2 = e2->Port2;
Nodes.removeRef(n); // delete node (is auto delete)
e1->Port2->Connections.removeRef(e2);
e1->Port2->Connections.append(e1);
Wires.removeRef(e2);
return true;
}
return false;
}
// ---------------------------------------------------
// Deletes the component 'c'.
void QucsDoc::deleteComp(Component *c)
{
Port *pn;
for(pn = c->Ports.first(); pn!=0; pn = c->Ports.next()) // delete all port connections
switch(pn->Connection->Connections.count()) {
case 1 : Nodes.removeRef(pn->Connection); // delete all open nodes
pn->Connection = 0;
break;
case 3 : pn->Connection->Connections.removeRef(c); // delete connection to component
oneTwoWires(pn->Connection); // two wires -> one wire
break;
default : pn->Connection->Connections.removeRef(c); // remove connection
break;
}
Comps.removeRef(c); // delete component
}
// ---------------------------------------------------
// Deletes the wire 'w'.
void QucsDoc::deleteWire(Wire *w)
{
if(w->Port1->Connections.count() == 1)
Nodes.removeRef(w->Port1); // delete node 1 if open
else {
w->Port1->Connections.removeRef(w); // remove connection
if(w->Port1->Connections.count() == 2) oneTwoWires(w->Port1); // two wires -> one wire
}
if(w->Port2->Connections.count() == 1)
Nodes.removeRef(w->Port2); // delete node 2 if open
else {
w->Port2->Connections.removeRef(w); // remove connection
if(w->Port2->Connections.count() == 2) oneTwoWires(w->Port2); // two wires -> one wire
}
Wires.removeRef(w);
}
// ---------------------------------------------------
// Deletes all selected elements.
bool QucsDoc::deleteElements()
{
bool sel = false;
Component *pc = Comps.first();
while(pc != 0) { // all selected component
if(pc->isSelected) {
deleteComp(pc);
pc = Comps.current();
sel = true;
}
else pc = Comps.next();
}
Wire *ptr2 = Wires.first();
while(ptr2 != 0) { // all selected wires
if(ptr2->isSelected) {
deleteWire(ptr2);
ptr2 = Wires.current();
sel = true;
}
else ptr2 = Wires.next();
}
for(Diagram *ptr3 = Diags.first(); ptr3 != 0; ptr3 = Diags.next()) // test all diagrams
if(ptr3->isSelected) {
Diags.remove();
ptr3 = Diags.current();
sel = true;
}
setChanged(sel);
return sel;
}
// ---------------------------------------------------
void QucsDoc::reloadGraphs()
{
for(Diagram *pd = Diags.first(); pd != 0; pd = Diags.next())
pd->loadGraphData(DataSet); // load graphs from data files
}
// ---------------------------------------------------
QString QucsDoc::copySelected(bool cut)
{
int z=0; // counts selected elements
Component *pc;
Wire *pw;
Diagram *pd;
QString s("<Qucs Schematic " VERSION ">\n");
// Build element document.
s += "<Components>\n";
for(pc = Comps.first(); pc != 0; pc = Comps.next())
if(pc->isSelected) {
s += " "+pc->save()+"\n"; z++; }
s += "</Components>\n";
s += "<Wires>\n";
for(pw = Wires.first(); pw != 0; pw = Wires.next())
if(pw->isSelected) {
s += " "+pw->save()+"\n"; z++; }
s += "</Wires>\n";
s += "<Diagrams>\n";
for(pd = Diags.first(); pd != 0; pd = Diags.next())
if(pd->isSelected) {
s += pd->save()+"\n"; z++; }
s += "</Diagrams>\n";
s += "<Paintings>\n";
// for(Diagram *ptr3 = Diags.first(); ptr3 != 0; ptr3 = Diags.next()) // save all diagrams
// stream << ptr3->save() << "\n";
s += "</Paintings>\n";
if(z == 0) return ""; // return empty if no selection
if(cut) deleteElements(); // delete selected elements if wanted
return s;
}
// ---------------------------------------------------
// Paste from clipboard.
bool QucsDoc::paste(QTextStream *stream, QPtrList<Element> *pe)
{
QString Line;
Cache = pe;
Line = stream->readLine();
if(Line.left(16) != "<Qucs Schematic ") { // wrong file type ?
// QMessageBox::critical(0, "Error", "Clipboard content is not a Qucs schematic!");
return false;
}
QString s = VERSION;
Line = Line.mid(16, Line.length()-17);
if(Line != s) { // wrong version number ?
QMessageBox::critical(0, "Error", "Wrong document version: "+Line);
return false;
}
// read content *************************
while(!stream->atEnd()) {
Line = stream->readLine();
if(Line == "<Components>") { if(!loadComponents(stream, false)) return false; }
else
if(Line == "<Wires>") { if(!loadWires(stream, false)) return false; }
else
if(Line == "<Diagrams>") { if(!loadDiagrams(stream, false)) return false; }
else
if(Line == "<Paintings>") { if(!loadPaintings(stream, false)) return false; }
else {
QMessageBox::critical(0, "Error", "Clipboard Format Error:\nUnknown field!");
return false;
}
}
return true;
}
// ---------------------------------------------------
int QucsDoc::save()
{
QFile file(DocName);
if(!file.open(IO_WriteOnly)) {
QMessageBox::critical(0, "Error", "Cannot save document!");
return -1;
}
QTextStream stream(&file);
stream << "<Qucs Schematic " << VERSION << ">\n";
stream << "<Properties>\n";
stream << " <View=" << ViewX1<<","<<ViewY1<<","<<ViewX2<<","<<ViewY2<<","<<Scale<<",";
stream << PosX<<","<<PosY << ">\n";
stream << " <Grid=" << GridX<<","<<GridY<<","<<GridOn << ">\n";
stream << " <DataSet=" << DataSet << ">\n";
stream << " <DataDisplay=" << DataDisplay << ">\n";
stream << " <OpenDisplay=" << SimOpenDpl << ">\n";
stream << "</Properties>\n";
int z=0;
stream << "<Components>\n";
for(Component *pc = Comps.first(); pc != 0; pc = Comps.next()) { // save all components
stream << " " << pc->save() << "\n";
if(pc->Sign == "Port") z++;
}
stream << "</Components>\n";
stream << "<Wires>\n";
for(Wire *ptr2 = Wires.first(); ptr2 != 0; ptr2 = Wires.next()) // save all wires
stream << " " << ptr2->save() << "\n";
stream << "</Wires>\n";
stream << "<Diagrams>\n";
for(Diagram *ptr3 = Diags.first(); ptr3 != 0; ptr3 = Diags.next()) // save all diagrams
stream << ptr3->save() << "\n";
stream << "</Diagrams>\n";
stream << "<Paintings>\n";
// for(Diagram *ptr3 = Diags.first(); ptr3 != 0; ptr3 = Diags.next()) // save all diagrams
// stream << ptr3->save() << "\n";
stream << "</Paintings>\n";
file.close();
setChanged(false);
return z;
}
// ---------------------------------------------------
bool QucsDoc::loadProperties(QTextStream *stream)
{
bool ok = true;
QString Line, cstr, nstr;
while(!stream->atEnd()) {
Line = stream->readLine();
if(Line == "</Properties>") return true;
Line = Line.stripWhiteSpace();
if(Line.at(0) != '<') {
QMessageBox::critical(0, "Error", "Format Error:\nWrong property field limiter!");
return false;
}
if(Line.at(Line.length()-1) != '>') {
QMessageBox::critical(0, "Error", "Format Error:\nWrong property field limiter!");
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
if(cstr == "View") { ViewX1 = nstr.section(',',0,0).toInt(&ok); if(ok) {
ViewY1 = nstr.section(',',1,1).toInt(&ok); if(ok) {
ViewX2 = nstr.section(',',2,2).toInt(&ok); if(ok) {
ViewY2 = nstr.section(',',3,3).toInt(&ok); if(ok) {
Scale = nstr.section(',',4,4).toDouble(&ok); if(ok) {
PosX = nstr.section(',',5,5).toInt(&ok); if(ok) {
PosY = nstr.section(',',6,6).toInt(&ok); }}}}}} }
else if(cstr == "Grid") { GridX = nstr.section(',',0,0).toInt(&ok); if(ok) {
GridY = nstr.section(',',1,1).toInt(&ok); if(ok) {
if(nstr.section(',',2,2).toInt(&ok) == 0) GridOn = false;
else GridOn = true; }} }
else if(cstr == "DataSet") DataSet = nstr;
else if(cstr == "DataDisplay") DataDisplay = nstr;
else if(cstr == "OpenDisplay") if(nstr.toInt(&ok) == 0) SimOpenDpl = false;
else SimOpenDpl = true;
else {
QMessageBox::critical(0, "Error", "Format Error:\nUnknown property: "+cstr);
return false;
}
if(!ok) {
QMessageBox::critical(0, "Error", "Format Error:\nNumber expected in property field!");
return false;
}
}
QMessageBox::critical(0, "Error", "Format Error:\n'Property' field is not closed!");
return false;
}
bool QucsDoc::loadComponents(QTextStream *stream, bool insert)
{
QString Line, cstr;
Component *c;
while(!stream->atEnd()) {
Line = stream->readLine();
if(Line == "</Components>") return true;
Line = Line.stripWhiteSpace();
cstr = Line.section(' ',0,0); // component type
if(cstr == "<R") c = new Resistor();
else if(cstr == "<Rus") c = new ResistorUS();
else if(cstr == "<C") c = new Capacitor();
else if(cstr == "<L") c = new Inductor();
else if(cstr == "<GND") c = new Ground();
else if(cstr == "<Tr") c = new Transformer();
else if(cstr == "<sTr") c = new symTrafo();
else if(cstr == "<V") c = new Volt_dc();
else if(cstr == "<Vac") c = new Volt_ac();
else if(cstr == "<Pac") c = new Source_ac();
else if(cstr == "<I") c = new Ampere_dc();
else if(cstr == "<VCCS") c = new VCCS();
else if(cstr == "<CCCS") c = new CCCS();
else if(cstr == "<VCVS") c = new VCVS();
else if(cstr == "<CCVS") c = new CCVS();
else if(cstr == "<Port") c = new SubCirPort();
else if(cstr.left(7) == "<SPfile") { c = new SParamFile(cstr.mid(7).toInt()); }
else if(cstr.left(4) == "<Sub") { c = new Subcircuit(cstr.mid(4).toInt()); }
else if(cstr == "<DCblock") c = new dcBlock();
else if(cstr == "<DCfeed") c = new dcFeed();
else if(cstr == "<BiasT") c = new BiasT();
else if(cstr == "<Attenuator") c = new Attenuator();
else if(cstr == "<Isolator") c = new Isolator();
else if(cstr == "<Circulator") c = new Circulator();
else if(cstr == "<TLIN") c = new TLine();
else if(cstr == "<SUBST") c = new Substrate();
else if(cstr == "<MLIN") c = new MSline();
else if(cstr == "<MSTEP") c = new MSstep();
else if(cstr == "<MCORN") c = new MScorner();
else if(cstr == "<MTEE") c = new MStee();
else if(cstr == "<MCROSS") c = new MScross();
else if(cstr == "<Diode") c = new Diode();
else if(cstr == "<.DC") c = new DC_Sim();
else if(cstr == "<.AC") c = new AC_Sim();
else if(cstr == "<.TR") c = new TR_Sim();
else if(cstr == "<.SP") c = new SP_Sim();
else if(cstr == "<.HB") c = new HB_Sim();
else if(cstr == "<.SW") c = new Param_Sweep();
else {
QMessageBox::critical(0, "Error", "Format Error:\nUnknown component!");
return false;
}
if(!c->load(Line)) {
QMessageBox::critical(0, "Error", "Format Error:\nWrong 'component' line format!");
delete c;
return false;
}
if(insert) insertRawComponent(c);
else {
int z;
for(z=c->Name.length()-1; z>=0; z--) // cut off number of component name
if(!c->Name.at(z).isDigit()) break;
c->Name = c->Name.left(z+1);
Cache->append((Element*)c);
}
}
QMessageBox::critical(0, "Error", "Format Error:\n'Component' field is not closed!");
return false;
}
bool QucsDoc::loadWires(QTextStream *stream, bool insert)
{
Wire *w;
QString Line;
while(!stream->atEnd()) {
Line = stream->readLine();
if(Line == "</Wires>") return true;
Line = Line.stripWhiteSpace();
w = new Wire();
if(!w->load(Line)) {
QMessageBox::critical(0, "Error", "Format Error:\nWrong 'wire' line format!");
return false;
}
if(insert) insertWire(w);
else {
w->Port1 = (Node*)4; // move all ports (later on)
w->Port2 = (Node*)4;
Cache->append((Element*)w);
}
}
QMessageBox::critical(0, "Error", "Format Error:\n'Wire' field is not closed!");
return false;
}
bool QucsDoc::loadDiagrams(QTextStream *stream, bool insert)
{
Diagram *d;
QString Line, cstr;
while(!stream->atEnd()) {
Line = stream->readLine();
if(Line == "</Diagrams>") return true;
Line = Line.stripWhiteSpace();
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();
else {
QMessageBox::critical(0, "Error", "Format Error:\nUnknown diagram!");
return false;
}
if(!d->load(Line, stream)) {
QMessageBox::critical(0, "Error", "Format Error:\nWrong 'diagram' line format!");
delete d;
return false;
}
if(insert) {
Diags.append(d);
d->loadGraphData(DataSet); // load graphs from data files
}
else Cache->append((Element*)d);
}
QMessageBox::critical(0, "Error", "Format Error:\n'Diagram' field is not closed!");
return false;
}
bool QucsDoc::loadPaintings(QTextStream *stream, bool insert)
{
// Painting *p;
QString Line;//, cstr;
while(!stream->atEnd()) {
Line = stream->readLine();
if(Line == "</Paintings>") return true;
Line = Line.stripWhiteSpace();
// 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();
else {
QMessageBox::critical(0, "Error", "File Format Error:\nUnknown painting!");
return false;
}*/
// if(!p->load(Line, stream)) {
QMessageBox::critical(0, "Error", "Format Error:\nWrong 'painting' line format!");
// delete p;
return false;
// }
if(insert) ;
// Diags.append(d);
}
QMessageBox::critical(0, "Error", "Format Error:\n'Painting' field is not closed!");
return false;
}
// ......................................................................
bool QucsDoc::load()
{
QFile file(DocName);
if(!file.open(IO_ReadOnly)) {
QMessageBox::critical(0, "Error", "Cannot load document: "+DocName);
return false;
}
QString Line;
QTextStream stream(&file);
// read header **************************
if(stream.atEnd()) {
file.close();
// QMessageBox::critical(0, "Error", "Document is empty!");
// return false;
return true;
}
Line = stream.readLine();
if(Line.left(16) != "<Qucs Schematic ") { // wrong file type ?
file.close();
QMessageBox::critical(0, "Error", "Wrong document type!");
return false;
}
QString s = VERSION;
Line = Line.mid(16, Line.length()-17);
if(Line != s) { // wrong version number ?
file.close();
QMessageBox::critical(0, "Error", "Wrong document version: "+Line);
return false;
}
// read content *************************
while(!stream.atEnd()) {
Line = stream.readLine();
if(Line == "<Properties>") { if(!loadProperties(&stream)) { file.close(); return false; } }
else
if(Line == "<Components>") { if(!loadComponents(&stream)) { file.close(); return false; } }
else
if(Line == "<Wires>") { if(!loadWires(&stream)) { file.close(); return false; } }
else
if(Line == "<Diagrams>") { if(!loadDiagrams(&stream)) { file.close(); return false; } }
else
if(Line == "<Paintings>") { if(!loadPaintings(&stream)) { file.close(); return false; } }
else {
QMessageBox::critical(0, "Error", "File Format Error:\nUnknown field!");
file.close();
return false;
}
}
file.close();
setChanged(false);
return true;
}
// ---------------------------------------------------
// Creates the file "netlist.net" in the project directory. Returns "true"
// if successful.
bool QucsDoc::createNetlist(QFile *NetlistFile)
{
if(!NetlistFile->open(IO_WriteOnly)) return false;
QTextStream stream(NetlistFile);
stream << "# Qucs " << VERSION << " " << DocName << "\n"; // first line is docu
// ................................................
Node *p1, *p2;
Wire *pw;
Element *pe;
// delete the node names not given by the user
for(p1 = Nodes.first(); p1 != 0; p1 = Nodes.next())
if(!p1->isNamed) p1->Name = "";
// set the wire names to the connected node
for(pw = Wires.first(); pw != 0; pw = Wires.next())
if(!pw->Name.isEmpty()) pw->Port1->Name = pw->Name;
// give the ground nodes the name "0"
for(Component *pc = Comps.first(); pc != 0; pc = Comps.next())
if(pc->Sign == "GND") pc->Ports.first()->Connection->Name = "gnd";
QPtrList<Node> Cons;
// work on named nodes first in order to preserve the user given names
for(p1 = Nodes.first(); p1 != 0; p1 = Nodes.next()) {
if(p1->Name.isEmpty()) continue;
Cons.append(p1);
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 = p1->Name;
Cons.append(pw->Port1);
Cons.findRef(p2);
}
}
else {
if(pw->Port2->Name.isEmpty()) {
pw->Port2->Name = p1->Name;
Cons.append(pw->Port2);
Cons.findRef(p2);
}
}
}
Cons.clear();
}
int z=0;
// give names to the remaining (unnamed) nodes
for(p1 = Nodes.first(); p1 != 0; p1 = Nodes.next()) { // work on all nodes
if(!p1->Name.isEmpty()) continue; // already named ?
p1->Name = "_net" + QString::number(z++); // create node name
Cons.append(p1);
for(p2 = Cons.first(); p2 != 0; p2 = Cons.next()) // create list with connections to the node
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 = p1->Name;
Cons.append(pw->Port1);
Cons.findRef(p2); // back to current Connection
}
}
else {
if(pw->Port2->Name.isEmpty()) {
pw->Port2->Name = p1->Name;
Cons.append(pw->Port2);
Cons.findRef(p2);
}
}
}
Cons.clear();
}
// .................................................
QString s;
// write all components with node names into the netlist file
for(Component *pc = Comps.first(); pc != 0; pc = Comps.next()) {
s = pc->NetList();
if(!s.isEmpty()) stream << s << "\n";
}
NetlistFile->close();
return true;
}