qucs_s/qucs/components/component.cpp
luz paz 10c1ee639c Fix various typos (including documenation)
Found via `codespell -q 3 -S *.ts,./qucs/ChangeLog -L ba,coul,inout,leaded,nd,numer,ro`
2022-07-05 07:08:28 -04:00

1772 lines
48 KiB
C++

/***************************************************************************
component.cpp
---------------
begin : Sat Aug 23 2003
copyright : (C) 2003 by Michael Margraf
email : michael.margraf@alumni.tu-berlin.de
***************************************************************************/
/***************************************************************************
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
***************************************************************************/
#include <stdlib.h>
#include <cmath>
#include "components.h"
#include "node.h"
#include "main.h"
#include "qucs.h"
#include "schematic.h"
#include "viewpainter.h"
#include "module.h"
#include "misc.h"
#include <QPen>
#include <QString>
#include <QMessageBox>
#include <QPainter>
#include <QDebug>
/*!
* \file component.cpp
* \brief Implementation of the Component class.
*/
/*!
* \class Component
* \brief The Component class implements a generic analog component
*/
Component::Component()
{
Type = isAnalogComponent;
SpiceModel = "";
isSimulation = false;
isProbe = false;
isEquation = false;
mirroredX = false;
rotated = 0;
isSelected = false;
isActive = COMP_IS_ACTIVE;
showName = true;
cx = 0;
cy = 0;
tx = 0;
ty = 0;
Props.setAutoDelete(true);
containingSchematic = NULL;
}
// -------------------------------------------------------
Component* Component::newOne()
{
return new Component();
}
// -------------------------------------------------------
void Component::Bounding(int& _x1, int& _y1, int& _x2, int& _y2)
{
_x1 = x1+cx;
_y1 = y1+cy;
_x2 = x2+cx;
_y2 = y2+cy;
}
// -------------------------------------------------------
// Size of component text.
int Component::textSize(int& _dx, int& _dy)
{
// get size of text using the screen-compatible metric
QFontMetrics metrics(QucsSettings.font, 0);
int tmp, count=0;
_dx = _dy = 0;
if(showName) {
_dx = metrics.boundingRect(Name).width();
_dy = metrics.height();
count++;
}
for(Property *pp = Props.first(); pp != 0; pp = Props.next())
if(pp->display) {
// get width of text
tmp = metrics.boundingRect(pp->Name+"="+pp->Value).width();
if(tmp > _dx) _dx = tmp;
_dy += metrics.height();
count++;
}
return count;
}
// -------------------------------------------------------
// Boundings including the component text.
void Component::entireBounds(int& _x1, int& _y1, int& _x2, int& _y2, float Corr)
{
_x1 = x1+cx;
_y1 = y1+cy;
_x2 = x2+cx;
_y2 = y2+cy;
// text boundings
if(tx < x1) _x1 = tx+cx;
if(ty < y1) _y1 = ty+cy;
int dx, dy, ny;
ny = textSize(dx, dy);
dy = int(float(ny) / Corr); // correction for unproportional font scaling
if((tx+dx) > x2) _x2 = tx+dx+cx;
if((ty+dy) > y2) _y2 = ty+dy+cy;
}
// -------------------------------------------------------
void Component::setCenter(int x, int y, bool relative)
{
if(relative) { cx += x; cy += y; }
else { cx = x; cy = y; }
}
// -------------------------------------------------------
void Component::getCenter(int& x, int& y)
{
x = cx;
y = cy;
}
// -------------------------------------------------------
int Component::getTextSelected(int x_, int y_, float Corr)
{
x_ -= cx;
y_ -= cy;
if(x_ < tx) return -1;
if(y_ < ty) return -1;
x_ -= tx;
y_ -= ty;
int w, dy = int(float(y_) * Corr); // correction for font scaling
// use the screen-compatible metric
QFontMetrics metrics(QucsSettings.font, 0);
if(showName) {
w = metrics.boundingRect(Name).width();
if(dy < 1) {
if(x_ < w) return 0;
return -1;
}
dy--;
}
Property *pp;
for(pp = Props.first(); pp != 0; pp = Props.next())
if(pp->display)
if((dy--) < 1) break;
if(!pp) return -1;
// get width of text
w = metrics.boundingRect(pp->Name+"="+pp->Value).width();
if(x_ > w) return -1; // clicked past the property text end - selection invalid
return Props.at()+1; // number the property
}
// -------------------------------------------------------
bool Component::getSelected(int x_, int y_)
{
x_ -= cx;
y_ -= cy;
if(x_ >= x1) if(x_ <= x2) if(y_ >= y1) if(y_ <= y2)
return true;
return false;
}
// -------------------------------------------------------
void Component::paint(ViewPainter *p)
{
int x, y, a, b, xb, yb;
QFont f = p->Painter->font(); // save current font
QFont newFont = f;
if(Model.at(0) == '.') { // is simulation component (dc, ac, ...)
newFont.setPointSizeF(p->Scale * Texts.first()->Size);
newFont.setWeight(QFont::DemiBold);
p->Painter->setFont(newFont);
p->map(cx, cy, x, y);
if ((Model==".CUSTOMSIM")||(Model==".DISTO")
||(Model==".NOISE")||(Model==".PZ")||
(Model==".SENS")||(Model==".SENS_AC")||
(Model==".FFT"))
p->Painter->setPen(QPen(Qt::blue,2));
else if ((Model==".XYCESCR")||(Model==".SENS_XYCE")
||(Model==".SENS_TR_XYCE")) p->Painter->setPen(QPen(Qt::darkGreen,2));
else if (Model==".FOURIER") p->Painter->setPen(QPen(Qt::darkRed,2));
else p->Painter->setPen(QPen(Qt::darkBlue,2));
a = b = 0;
QRect r, t;
foreach(Text *pt, Texts) {
t.setRect(x, y+b, 0, 0);
p->Painter->drawText(t, Qt::AlignLeft|Qt::TextDontClip, pt->s, &r);
b += r.height();
if(a < r.width()) a = r.width();
}
xb = a + int(12.0*p->Scale);
yb = b + int(10.0*p->Scale);
x2 = x1+25 + int(float(a) / p->Scale);
y2 = y1+23 + int(float(b) / p->Scale);
if(ty < y2+1) if(ty > y1-r.height()) ty = y2 + 1;
p->map(cx-1, cy, x, y);
p->map(cx-6, cy-5, a, b);
p->Painter->drawRect(a, b, xb, yb);
p->Painter->drawLine(x, y+yb, a, b+yb);
p->Painter->drawLine(x+xb-1, y+yb, x, y+yb);
p->Painter->drawLine(x+xb-1, y+yb, a+xb, b+yb);
p->Painter->drawLine(x+xb-1, y+yb, x+xb-1, y);
p->Painter->drawLine(x+xb-1, y, a+xb, b);
}
else { // normal components go here
// paint all lines
foreach(qucs::Line *p1, Lines) {
p->Painter->setPen(p1->style);
p->drawLine(cx+p1->x1, cy+p1->y1, cx+p1->x2, cy+p1->y2);
}
// paint all arcs
foreach(qucs::Arc *p3, Arcs) {
p->Painter->setPen(p3->style);
p->drawArc(cx+p3->x, cy+p3->y, p3->w, p3->h, p3->angle, p3->arclen);
}
// paint all rectangles
foreach(qucs::Area *pa, Rects) {
p->Painter->setPen(pa->Pen);
p->Painter->setBrush(pa->Brush);
p->drawRect(cx+pa->x, cy+pa->y, pa->w, pa->h);
}
// paint all ellipses
foreach(qucs::Area *pa, Ellips) {
p->Painter->setPen(pa->Pen);
p->Painter->setBrush(pa->Brush);
p->drawEllipse(cx+pa->x, cy+pa->y, pa->w, pa->h);
}
p->Painter->setBrush(Qt::NoBrush);
newFont.setWeight(QFont::Light);
// keep track of painter state
p->Painter->save();
QTransform wm = p->Painter->worldTransform();
// write all text
foreach(Text *pt, Texts) {
p->Painter->setWorldTransform(
QTransform(pt->mCos, -pt->mSin, pt->mSin, pt->mCos,
p->DX + float(cx+pt->x) * p->Scale,
p->DY + float(cy+pt->y) * p->Scale));
newFont.setPointSizeF(p->Scale * pt->Size);
newFont.setOverline(pt->over);
newFont.setUnderline(pt->under);
p->Painter->setFont(newFont);
p->Painter->setPen(pt->Color);
if (0) {
p->Painter->drawText(0, 0, 0, 0, Qt::AlignLeft|Qt::TextDontClip, pt->s);
} else {
int w, h;
w = p->drawTextMapped (pt->s, 0, 0, &h);
Q_UNUSED(w);
}
}
p->Painter->setWorldTransform(wm);
p->Painter->setWorldMatrixEnabled(false);
// restore painter state
p->Painter->restore();
}
// restore old font
p->Painter->setFont(f);
p->Painter->setPen(QPen(Qt::black,1));
p->map(cx+tx, cy+ty, x, y);
if(showName) {
p->Painter->drawText(x, y, 0, 0, Qt::TextDontClip, Name);
y += p->LineSpacing;
}
// write all properties
for(Property *p4 = Props.first(); p4 != 0; p4 = Props.next())
if(p4->display) {
p->Painter->drawText(x, y, 0, 0, Qt::TextDontClip, p4->Name+"="+p4->Value);
y += p->LineSpacing;
}
if(isActive == COMP_IS_OPEN)
p->Painter->setPen(QPen(Qt::red,0));
else if(isActive & COMP_IS_SHORTEN)
p->Painter->setPen(QPen(Qt::darkGreen,0));
if(isActive != COMP_IS_ACTIVE) {
p->drawRect(cx+x1, cy+y1, x2-x1+1, y2-y1+1);
p->drawLine(cx+x1, cy+y1, cx+x2, cy+y2);
p->drawLine(cx+x1, cy+y2, cx+x2, cy+y1);
}
// draw component bounding box
if(isSelected) {
p->Painter->setPen(QPen(Qt::darkGray,3));
p->drawRoundRect(cx+x1, cy+y1, x2-x1, y2-y1);
}
}
// -------------------------------------------------------
// Paints the component when moved with the mouse.
void Component::paintScheme(Schematic *p)
{
// qDebug() << "paintScheme" << Model;
if(Model.at(0) == '.') { // is simulation component (dc, ac, ...)
int a, b, xb, yb;
QFont newFont = p->font();
float Scale =
((Schematic*)QucsMain->DocumentTab->currentWidget())->Scale;
newFont.setPointSizeF(float(Scale) * QucsSettings.largeFontSize);
newFont.setWeight(QFont::DemiBold);
// here the font metric is already the screen metric, since the font
// is the current font the painter is using
QFontMetrics metrics(newFont);
a = b = 0;
QSize r;
foreach(Text *pt, Texts) {
r = metrics.size(0, pt->s);
b += r.height();
if(a < r.width()) a = r.width();
}
xb = a + int(12.0*Scale);
yb = b + int(10.0*Scale);
x2 = x1+25 + int(float(a) / Scale);
y2 = y1+23 + int(float(b) / Scale);
if(ty < y2+1) if(ty > y1-r.height()) ty = y2 + 1;
p->PostPaintEvent(_Rect,cx-6, cy-5, xb, yb);
p->PostPaintEvent(_Line,cx-1, cy+yb, cx-6, cy+yb-5);
p->PostPaintEvent(_Line,cx+xb-2, cy+yb, cx-1, cy+yb);
p->PostPaintEvent(_Line,cx+xb-2, cy+yb, cx+xb-6, cy+yb-5);
p->PostPaintEvent(_Line,cx+xb-2, cy+yb, cx+xb-2, cy);
p->PostPaintEvent(_Line,cx+xb-2, cy, cx+xb-6, cy-5);
return;
}
// paint all lines
foreach(qucs::Line *p1, Lines)
p->PostPaintEvent(_Line,cx+p1->x1, cy+p1->y1, cx+p1->x2, cy+p1->y2);
// paint all ports
foreach(Port *p2, Ports)
if(p2->avail) p->PostPaintEvent(_Ellipse,cx+p2->x-4, cy+p2->y-4, 8, 8);
foreach(qucs::Arc *p3, Arcs) // paint all arcs
p->PostPaintEvent(_Arc,cx+p3->x, cy+p3->y, p3->w, p3->h, p3->angle, p3->arclen);
foreach(qucs::Area *pa, Rects) // paint all rectangles
p->PostPaintEvent(_Rect,cx+pa->x, cy+pa->y, pa->w, pa->h);
foreach(qucs::Area *pa, Ellips) // paint all ellipses
p->PostPaintEvent(_Ellipse,cx+pa->x, cy+pa->y, pa->w, pa->h);
}
// -------------------------------------------------------
// For output on a printer device.
void Component::print(ViewPainter *p, float FontScale)
{
foreach(Text *pt, Texts)
pt->Size *= FontScale;
paint(p);
foreach(Text *pt, Texts)
pt->Size /= FontScale;
}
// -------------------------------------------------------
// Rotates the component 90 counter-clockwise around its center
void Component::rotate()
{
// Port count only available after recreate, createSymbol
if ((Model != "Sub") && (Model !="VHDL") && (Model != "Verilog")
&& (Model != "SpLib")) // skip port count
if(Ports.count() < 1) return; // do not rotate components without ports
int tmp, dx, dy;
// rotate all lines
foreach(qucs::Line *p1, Lines) {
tmp = -p1->x1;
p1->x1 = p1->y1;
p1->y1 = tmp;
tmp = -p1->x2;
p1->x2 = p1->y2;
p1->y2 = tmp;
}
// rotate all ports
foreach(Port *p2, Ports) {
tmp = -p2->x;
p2->x = p2->y;
p2->y = tmp;
}
// rotate all arcs
foreach(qucs::Arc *p3, Arcs) {
tmp = -p3->x;
p3->x = p3->y;
p3->y = tmp - p3->w;
tmp = p3->w;
p3->w = p3->h;
p3->h = tmp;
p3->angle += 16*90;
if(p3->angle >= 16*360) p3->angle -= 16*360;;
}
// rotate all rectangles
foreach(qucs::Area *pa, Rects) {
tmp = -pa->x;
pa->x = pa->y;
pa->y = tmp - pa->w;
tmp = pa->w;
pa->w = pa->h;
pa->h = tmp;
}
// rotate all ellipses
foreach(qucs::Area *pa, Ellips) {
tmp = -pa->x;
pa->x = pa->y;
pa->y = tmp - pa->w;
tmp = pa->w;
pa->w = pa->h;
pa->h = tmp;
}
// rotate all text
float ftmp;
foreach(Text *pt, Texts) {
tmp = -pt->x;
pt->x = pt->y;
pt->y = tmp;
ftmp = -pt->mSin;
pt->mSin = pt->mCos;
pt->mCos = ftmp;
}
tmp = -x1; // rotate boundings
x1 = y1; y1 = -x2;
x2 = y2; y2 = tmp;
tmp = -tx; // rotate text position
tx = ty;
ty = tmp;
// use the screen-compatible metric
QFontMetrics metrics(QucsSettings.font, 0); // get size of text
dx = dy = 0;
if(showName) {
dx = metrics.boundingRect(Name).width();
dy = metrics.lineSpacing();
}
for(Property *pp = Props.first(); pp != 0; pp = Props.next())
if(pp->display) {
// get width of text
tmp = metrics.boundingRect(pp->Name+"="+pp->Value).width();
if(tmp > dx) dx = tmp;
dy += metrics.lineSpacing();
}
if(tx > x2) ty = y1-ty+y2; // rotate text position
else if(ty < y1) ty -= dy;
else if(tx < x1) { tx += dy-dx; ty = y1-ty+y2; }
else ty -= dx;
rotated++; // keep track of what's done
rotated &= 3;
}
// -------------------------------------------------------
// Mirrors the component about the x-axis.
void Component::mirrorX()
{
// Port count only available after recreate, createSymbol
if ((Model != "Sub") && (Model !="VHDL") && (Model != "Verilog")
&& (Model != "SpLib")) // skip port count
if(Ports.count() < 1) return; // do not rotate components without ports
// mirror all lines
foreach(qucs::Line *p1, Lines) {
p1->y1 = -p1->y1;
p1->y2 = -p1->y2;
}
// mirror all ports
foreach(Port *p2, Ports)
p2->y = -p2->y;
// mirror all arcs
foreach(qucs::Arc *p3, Arcs) {
p3->y = -p3->y - p3->h;
if(p3->angle > 16*180) p3->angle -= 16*360;
p3->angle = -p3->angle; // mirror
p3->angle -= p3->arclen; // go back to end of arc
if(p3->angle < 0) p3->angle += 16*360; // angle has to be > 0
}
// mirror all rectangles
foreach(qucs::Area *pa, Rects)
pa->y = -pa->y - pa->h;
// mirror all ellipses
foreach(qucs::Area *pa, Ellips)
pa->y = -pa->y - pa->h;
QFont f = QucsSettings.font;
// mirror all text
foreach(Text *pt, Texts) {
f.setPointSizeF(pt->Size);
// use the screen-compatible metric
QFontMetrics smallMetrics(f, 0);
QSize s = smallMetrics.size(0, pt->s); // use size for more lines
pt->y = -pt->y - int(pt->mCos)*s.height() + int(pt->mSin)*s.width();
}
int tmp = y1;
y1 = -y2; y2 = -tmp; // mirror boundings
// use the screen-compatible metric
QFontMetrics metrics(QucsSettings.font, 0); // get size of text
int dy = 0;
if(showName)
dy = metrics.lineSpacing(); // for "Name"
for(Property *pp = Props.first(); pp != 0; pp = Props.next())
if(pp->display) dy += metrics.lineSpacing();
if((tx > x1) && (tx < x2)) ty = -ty-dy; // mirror text position
else ty = y1+ty+y2;
mirroredX = !mirroredX; // keep track of what's done
rotated += rotated << 1;
rotated &= 3;
}
// -------------------------------------------------------
// Mirrors the component about the y-axis.
void Component::mirrorY()
{
// Port count only available after recreate, createSymbol
if ((Model != "Sub") && (Model !="VHDL") && (Model != "Verilog")
&& (Model != "SpLib")) // skip port count
if(Ports.count() < 1) return; // do not rotate components without ports
// mirror all lines
foreach(qucs::Line *p1, Lines) {
p1->x1 = -p1->x1;
p1->x2 = -p1->x2;
}
// mirror all ports
foreach(Port *p2, Ports)
p2->x = -p2->x;
// mirror all arcs
foreach(qucs::Arc *p3, Arcs) {
p3->x = -p3->x - p3->w;
p3->angle = 16*180 - p3->angle - p3->arclen; // mirror
if(p3->angle < 0) p3->angle += 16*360; // angle has to be > 0
}
// mirror all rectangles
foreach(qucs::Area *pa, Rects)
pa->x = -pa->x - pa->w;
// mirror all ellipses
foreach(qucs::Area *pa, Ellips)
pa->x = -pa->x - pa->w;
int tmp;
QFont f = QucsSettings.font;
// mirror all text
foreach(Text *pt, Texts) {
f.setPointSizeF(pt->Size);
// use the screen-compatible metric
QFontMetrics smallMetrics(f, 0);
QSize s = smallMetrics.size(0, pt->s); // use size for more lines
pt->x = -pt->x - int(pt->mSin)*s.height() - int(pt->mCos)*s.width();
}
tmp = x1;
x1 = -x2; x2 = -tmp; // mirror boundings
// use the screen-compatible metric
QFontMetrics metrics(QucsSettings.font, 0); // get size of text
int dx = 0;
if(showName)
dx = metrics.boundingRect(Name).width();
for(Property *pp = Props.first(); pp != 0; pp = Props.next())
if(pp->display) {
// get width of text
tmp = metrics.boundingRect(pp->Name+"="+pp->Value).width();
if(tmp > dx) dx = tmp;
}
if((ty > y1) && (ty < y2)) tx = -tx-dx; // mirror text position
else tx = x1+tx+x2;
mirroredX = !mirroredX; // keep track of what's done
rotated += rotated << 1;
rotated += 2;
rotated &= 3;
}
// -------------------------------------------------------
QString Component::netlist()
{
QString s = Model+":"+Name;
// output all node names
foreach(Port *p1, Ports)
s += " "+p1->Connection->Name; // node names
// output all properties
for(Property *p2 = Props.first(); p2 != 0; p2 = Props.next())
if(p2->Name != "Symbol")
s += " "+p2->Name+"=\""+p2->Value+"\"";
return s + '\n';
}
// Forms spice parameter list
// ignore_list --- QStringList with spice_incompatible parameters
// convert_list --- QString list with parameters that needs names conversion
// list format is: ( qucs_parameter_name<i> , spice_parameter_name<i>, ... )
QString Component::form_spice_param_list(QStringList& ignore_list, QStringList& convert_list)
{
QString par_str = "";
for (unsigned int i=0;i<Props.count();i++) {
if (!ignore_list.contains(Props.at(i)->Name)) {
QString unit,nam;
if (convert_list.contains(Props.at(i)->Name)) {
nam = convert_list.at(convert_list.indexOf(Props.at(i)->Name)+1);
} else {
nam = Props.at(i)->Name;
}
double val,fac;
misc::str2num(Props.at(i)->Value,val,unit,fac);
val *= fac;
par_str += QString("%1=%2 ").arg(nam).arg(val);
}
}
return par_str;
}
QString Component::spice_netlist(bool)
{
return QString("\n"); // ignore if not implemented
}
QString Component::va_code()
{
return QString(""); // ignore if not implemented
}
// -------------------------------------------------------
QString Component::getNetlist()
{
switch(isActive) {
case COMP_IS_ACTIVE:
return netlist();
case COMP_IS_OPEN:
return QString("");
}
// Component is shortened.
int z=0;
QListIterator<Port *> iport(Ports);
Port *pp = iport.next();
QString Node1 = pp->Connection->Name;
QString s = "";
while (iport.hasNext())
s += "R:" + Name + "." + QString::number(z++) + " " +
Node1 + " " + iport.next()->Connection->Name + " R=\"0\"\n";
return s;
}
QString Component::getSpiceNetlist(bool isXyce)
{
QString s;
switch(isActive) {
case COMP_IS_ACTIVE:
s = spice_netlist(isXyce);
s.replace(" gnd "," 0 ");
return s;
case COMP_IS_OPEN:
return QString("");
}
// Component is shortened.
int z=0;
QListIterator<Port *> iport(Ports);
Port *pp = iport.next();
QString Node1 = pp->Connection->Name;
s = "";
while (iport.hasNext())
s += "R"+Name + QString::number(z++) + " " +
Node1 + " " + iport.next()->Connection->Name + " 0\n";
s.replace(" gnd "," 0 ");
return s;
}
QString Component::getVerilogACode()
{
QString s;
switch(isActive) {
case COMP_IS_ACTIVE:
s = va_code();
return s;
case COMP_IS_OPEN:
default:
return QString("");
}
}
QString Component::getExpression(bool)
{
return QString("");
}
QString Component::getEquations(QString, QStringList &)
{
return QString("");
}
QStringList Component::getExtraVariables()
{
return QStringList();
}
QString Component::getProbeVariable(bool)
{
return QString("");
}
QString Component::getSpiceModel()
{
return QString("");
}
QString Component::getNgspiceBeforeSim(QString sim, int lvl)
{
Q_UNUSED(sim); // To suppress warning
Q_UNUSED(lvl);
return QString("");
}
QString Component::getNgspiceAfterSim(QString sim, int lvl)
{
Q_UNUSED(sim); // To suppress warning
Q_UNUSED(lvl);
return QString("");
}
// -------------------------------------------------------
QString Component::verilogCode(int)
{
return QString(""); // no digital model
}
// -------------------------------------------------------
QString Component::get_Verilog_Code(int NumPorts)
{
switch(isActive) {
case COMP_IS_OPEN:
return QString("");
case COMP_IS_ACTIVE:
return verilogCode(NumPorts);
}
// Component is shortened.
QListIterator<Port *> iport(Ports);
Port *pp = iport.next();
QString Node1 = pp->Connection->Name;
QString s = "";
while (iport.hasNext())
s += " assign " + iport.next()->Connection->Name + " = " + Node1 + ";\n";
return s;
}
// -------------------------------------------------------
QString Component::vhdlCode(int)
{
return QString(""); // no digital model
}
// -------------------------------------------------------
QString Component::get_VHDL_Code(int NumPorts)
{
switch(isActive) {
case COMP_IS_OPEN:
return QString("");
case COMP_IS_ACTIVE:
return vhdlCode(NumPorts);
}
// Component is shortened.
// This puts the signal of the second port onto the first port.
// This is logically correct for the inverter only, but makes
// some sense for the gates (OR, AND etc.).
// Has anyone a better idea?
QString Node1 = Ports.at(0)->Connection->Name;
return " " + Node1 + " <= " + Ports.at(1)->Connection->Name + ";\n";
}
// -------------------------------------------------------
QString Component::save()
{
#if XML
QDomDocument doc;
QDomElement el = doc.createElement (Model);
doc.appendChild (el);
el.setTagName (Model);
el.setAttribute ("inst", Name.isEmpty() ? "*" : Name);
el.setAttribute ("display", isActive | (showName ? 4 : 0));
el.setAttribute ("cx", cx);
el.setAttribute ("cy", cy);
el.setAttribute ("tx", tx);
el.setAttribute ("ty", ty);
el.setAttribute ("mirror", mirroredX);
el.setAttribute ("rotate", rotated);
for (Property *pr = Props.first(); pr != 0; pr = Props.next()) {
QString val = pr->Value;
val.replace("\n","\\n");
val.replace("\"","''");
el.setAttribute (pr->Name, (pr->display ? "1@" : "0@") + val);
}
qDebug (doc.toString());
#endif
QString s = "<" + Model;
if(Name.isEmpty()) s += " * ";
else s += " "+Name+" ";
int i=0;
if(!showName)
i = 4;
i |= isActive;
s += QString::number(i);
s += " "+QString::number(cx)+" "+QString::number(cy);
s += " "+QString::number(tx)+" "+QString::number(ty);
if(mirroredX) s += " 1";
else s += " 0";
s += " "+QString::number(rotated);
// write all properties
for(Property *p1 = Props.first(); p1 != 0; p1 = Props.next()) {
QString val = p1->Value; // enable newline in properties
val.replace("\n","\\n");
val.replace("\"","''");
if(p1->Description.isEmpty()||(p1->Description=="Expression"))
s += " \""+p1->Name+"="+val+"\""; // e.g. for equations
else s += " \""+val+"\"";
if(p1->display) s += " 1";
else s += " 0";
}
return s+">";
}
// -------------------------------------------------------
bool Component::load(const QString& _s)
{
bool ok;
int ttx, tty, tmp;
QString s = _s;
if(s.at(0) != '<') return false;
if(s.at(s.length()-1) != '>') return false;
s = s.mid(1, s.length()-2); // cut off start and end character
QString n;
Name = s.section(' ',1,1); // Name
if(Name == "*") Name = "";
n = s.section(' ',2,2); // isActive
tmp = n.toInt(&ok);
if(!ok) return false;
isActive = tmp & 3;
if(tmp & 4)
showName = false;
else
showName = true;
n = s.section(' ',3,3); // cx
cx = n.toInt(&ok);
if(!ok) return false;
n = s.section(' ',4,4); // cy
cy = n.toInt(&ok);
if(!ok) return false;
n = s.section(' ',5,5); // tx
ttx = n.toInt(&ok);
if(!ok) return false;
n = s.section(' ',6,6); // ty
tty = n.toInt(&ok);
if(!ok) return false;
if(Model.at(0) != '.') { // is simulation component (dc, ac, ...) ?
n = s.section(' ',7,7); // mirroredX
if(n.toInt(&ok) == 1) mirrorX();
if(!ok) return false;
n = s.section(' ',8,8); // rotated
tmp = n.toInt(&ok);
if(!ok) return false;
if(rotated > tmp) // necessary because of historical flaw in ...
tmp += 4; // ... components like "volt_dc"
for(int z=rotated; z<tmp; z++) rotate();
}
tx = ttx; ty = tty; // restore text position (was changed by rotate/mirror)
unsigned int z=0, counts = s.count('"');
if(Model == "Sub")
tmp = 2; // first property (File) already exists
else if(Model == "Lib")
tmp = 3;
else if(Model == "EDD")
tmp = 5;
else if(Model == "RFEDD")
tmp = 8;
else if(Model == "VHDL")
tmp = 2;
else if(Model == "MUTX")
tmp = 5; // number of properties for the default MUTX (2 inductors)
else tmp = counts + 1; // "+1" because "counts" could be zero
for(; tmp<=(int)counts/2; tmp++)
Props.append(new Property("p", "", true, " "));
// load all properties
Property *p1;
for(p1 = Props.first(); p1 != 0; p1 = Props.next()) {
z++;
n = s.section('"',z,z); // property value
n.replace("\\n","\n");
n.replace("''","\"");
z++;
//qDebug() << "LOAD: " << p1->Description;
// not all properties have to be mentioned (backward compatible)
if(z > counts) {
if(p1->Description.isEmpty())
Props.remove(); // remove if allocated in vain
if(Model == "Diode") {
if(counts < 56) { // backward compatible
counts >>= 1;
p1 = Props.at(counts-1);
for(; p1 != 0; p1 = Props.current()) {
if(counts-- < 19)
break;
n = Props.prev()->Value;
p1->Value = n;
}
p1 = Props.at(17);
p1->Value = Props.at(11)->Value;
Props.current()->Value = "0";
}
}
else if(Model == "AND" || Model == "NAND" || Model == "NOR" ||
Model == "OR" || Model == "XNOR"|| Model == "XOR") {
if(counts < 10) { // backward compatible
counts >>= 1;
p1 = Props.at(counts);
for(; p1 != 0; p1 = Props.current()) {
if(counts-- < 4)
break;
n = Props.prev()->Value;
p1->Value = n;
}
Props.current()->Value = "10";
}
}
else if(Model == "Buf" || Model == "Inv") {
if(counts < 8) { // backward compatible
counts >>= 1;
p1 = Props.at(counts);
for(; p1 != 0; p1 = Props.current()) {
if(counts-- < 3)
break;
n = Props.prev()->Value;
p1->Value = n;
}
Props.current()->Value = "10";
}
}
return true;
}
// for equations
if(Model != "EDD" && Model != "RFEDD" && Model != "RFEDD2P")
if(p1->Description.isEmpty()||p1->Description=="Expression") { // unknown number of properties ?
p1->Name = n.section('=',0,0);
n = n.section('=',1);
// allocate memory for a new property (e.g. for equations)
if(Props.count() < (counts>>1)) {
Props.insert(z >> 1, new Property("y", "1", true));
Props.prev();
}
}
if(z == 6) if(counts == 6) // backward compatible
if(Model == "R") {
Props.getLast()->Value = n;
return true;
}
p1->Value = n;
n = s.section('"',z,z); // display
p1->display = (n.at(1) == '1');
}
return true;
}
// *******************************************************************
// *** The following functions are used to load the schematic symbol
// *** from file. (e.g. subcircuit, library component)
int Component::analyseLine(const QString& Row, int numProps)
{
QPen Pen;
QBrush Brush;
QColor Color;
QString s;
int i1, i2, i3, i4, i5, i6;
s = Row.section(' ',0,0); // component type
if((s == "PortSym") || (s == ".PortSym")) { // backward compatible
if(!getIntegers(Row, &i1, &i2, &i3))
return -1;
for(i6 = Ports.count(); i6<i3; i6++) // if ports not in numerical order
Ports.append(new Port(0, 0, false));
Port *po = Ports.at(i3-1);
po->x = i1;
po->y = i2;
po->avail = true;
if(i1 < x1) x1 = i1; // keep track of component boundings
if(i1 > x2) x2 = i1;
if(i2 < y1) y1 = i2;
if(i2 > y2) y2 = i2;
return 0; // do not count Ports
}
else if(s == "Line") {
if(!getIntegers(Row, &i1, &i2, &i3, &i4)) return -1;
if(!getPen(Row, Pen, 5)) return -1;
i3 += i1;
i4 += i2;
Lines.append(new qucs::Line(i1, i2, i3, i4, Pen));
if(i1 < x1) x1 = i1; // keep track of component boundings
if(i1 > x2) x2 = i1;
if(i2 < y1) y1 = i2;
if(i2 > y2) y2 = i2;
if(i3 < x1) x1 = i3;
if(i3 > x2) x2 = i3;
if(i4 < y1) y1 = i4;
if(i4 > y2) y2 = i4;
return 1;
}
else if(s == "EArc") {
if(!getIntegers(Row, &i1, &i2, &i3, &i4, &i5, &i6))
return -1;
if(!getPen(Row, Pen, 7)) return -1;
Arcs.append(new struct qucs::Arc(i1, i2, i3, i4, i5, i6, Pen));
if(i1 < x1) x1 = i1; // keep track of component boundings
if(i1+i3 > x2) x2 = i1+i3;
if(i2 < y1) y1 = i2;
if(i2+i4 > y2) y2 = i2+i4;
return 1;
}
else if(s == ".ID") {
if(!getIntegers(Row, &i1, &i2)) return -1;
tx = i1;
ty = i2;
Name = Row.section(' ',3,3);
if(Name.isEmpty()) Name = "SUB";
i1 = 1;
Property *pp = Props.at(numProps-1);
for(;;) {
s = Row.section('"', i1,i1);
if(s.isEmpty()) break;
pp = Props.next();
if(pp == 0) {
pp = new Property();
Props.append(pp);
pp->display = (s.at(0) == '1');
pp->Value = s.section('=', 2,2);
}
pp->Name = s.section('=', 1,1);
pp->Description = s.section('=', 3,3);
if(pp->Description.isEmpty())
pp->Description = " ";
i1 += 2;
}
while(pp != Props.last())
Props.remove();
return 0; // do not count IDs
}
else if(s == "Arrow") {
if(!getIntegers(Row, &i1, &i2, &i3, &i4, &i5, &i6)) return -1;
if(!getPen(Row, Pen, 7)) return -1;
double beta = atan2(double(i6), double(i5));
double phi = atan2(double(i4), double(i3));
double Length = sqrt(double(i6*i6 + i5*i5));
i3 += i1;
i4 += i2;
if(i1 < x1) x1 = i1; // keep track of component boundings
if(i1 > x2) x2 = i1;
if(i3 < x1) x1 = i3;
if(i3 > x2) x2 = i3;
if(i2 < y1) y1 = i2;
if(i2 > y2) y2 = i2;
if(i4 < y1) y1 = i4;
if(i4 > y2) y2 = i4;
Lines.append(new qucs::Line(i1, i2, i3, i4, Pen)); // base line
double w = beta+phi;
i5 = i3-int(Length*cos(w));
i6 = i4-int(Length*sin(w));
Lines.append(new qucs::Line(i3, i4, i5, i6, Pen)); // arrow head
if(i5 < x1) x1 = i5; // keep track of component boundings
if(i5 > x2) x2 = i5;
if(i6 < y1) y1 = i6;
if(i6 > y2) y2 = i6;
w = phi-beta;
i5 = i3-int(Length*cos(w));
i6 = i4-int(Length*sin(w));
Lines.append(new qucs::Line(i3, i4, i5, i6, Pen));
if(i5 < x1) x1 = i5; // keep track of component boundings
if(i5 > x2) x2 = i5;
if(i6 < y1) y1 = i6;
if(i6 > y2) y2 = i6;
return 1;
}
else if(s == "Ellipse") {
if(!getIntegers(Row, &i1, &i2, &i3, &i4)) return -1;
if(!getPen(Row, Pen, 5)) return -1;
if(!getBrush(Row, Brush, 8)) return -1;
Ellips.append(new qucs::Area(i1, i2, i3, i4, Pen, Brush));
if(i1 < x1) x1 = i1; // keep track of component boundings
if(i1 > x2) x2 = i1;
if(i2 < y1) y1 = i2;
if(i2 > y2) y2 = i2;
if(i1+i3 < x1) x1 = i1+i3;
if(i1+i3 > x2) x2 = i1+i3;
if(i2+i4 < y1) y1 = i2+i4;
if(i2+i4 > y2) y2 = i2+i4;
return 1;
}
else if(s == "Rectangle") {
if(!getIntegers(Row, &i1, &i2, &i3, &i4)) return -1;
if(!getPen(Row, Pen, 5)) return -1;
if(!getBrush(Row, Brush, 8)) return -1;
Rects.append(new qucs::Area(i1, i2, i3, i4, Pen, Brush));
if(i1 < x1) x1 = i1; // keep track of component boundings
if(i1 > x2) x2 = i1;
if(i2 < y1) y1 = i2;
if(i2 > y2) y2 = i2;
if(i1+i3 < x1) x1 = i1+i3;
if(i1+i3 > x2) x2 = i1+i3;
if(i2+i4 < y1) y1 = i2+i4;
if(i2+i4 > y2) y2 = i2+i4;
return 1;
}
else if(s == "Text") { // must be last in order to reuse "s" *********
if(!getIntegers(Row, &i1, &i2, &i3, 0, &i4)) return -1;
Color.setNamedColor(Row.section(' ',4,4));
if(!Color.isValid()) return -1;
s = Row.mid(Row.indexOf('"')+1); // Text (can contain " !!!)
s = s.left(s.length()-1);
if(s.isEmpty()) return -1;
misc::convert2Unicode(s);
Texts.append(new Text(i1, i2, s, Color, float(i3),
float(cos(float(i4)*pi/180.0)),
float(sin(float(i4)*pi/180.0))));
QFont Font(QucsSettings.font);
Font.setPointSizeF(float(i3));
QFontMetrics metrics(Font, 0); // use the screen-compatible metric
QSize r = metrics.size(0, s); // get size of text
i3 = i1 + int(float(r.width()) * Texts.last()->mCos)
+ int(float(r.height()) * Texts.last()->mSin);
i4 = i2 + int(float(r.width()) * -Texts.last()->mSin)
+ int(float(r.height()) * Texts.last()->mCos);
if(i1 < x1) x1 = i1; // keep track of component boundings
if(i2 < y1) y1 = i2;
if(i1 > x2) x2 = i1;
if(i2 > y2) y2 = i2;
if(i3 < x1) x1 = i3;
if(i4 < y1) y1 = i4;
if(i3 > x2) x2 = i3;
if(i4 > y2) y2 = i4;
return 1;
}
return 0;
}
// ---------------------------------------------------------------------
bool Component::getIntegers(const QString& s, int *i1, int *i2, int *i3,
int *i4, int *i5, int *i6)
{
bool ok;
QString n;
if(!i1) return true;
n = s.section(' ',1,1);
*i1 = n.toInt(&ok);
if(!ok) return false;
if(!i2) return true;
n = s.section(' ',2,2);
*i2 = n.toInt(&ok);
if(!ok) return false;
if(!i3) return true;
n = s.section(' ',3,3);
*i3 = n.toInt(&ok);
if(!ok) return false;
if(i4) {
n = s.section(' ',4,4);
*i4 = n.toInt(&ok);
if(!ok) return false;
}
if(!i5) return true;
n = s.section(' ',5,5);
*i5 = n.toInt(&ok);
if(!ok) return false;
if(!i6) return true;
n = s.section(' ',6,6);
*i6 = n.toInt(&ok);
if(!ok) return false;
return true;
}
// ---------------------------------------------------------------------
bool Component::getPen(const QString& s, QPen& Pen, int i)
{
bool ok;
QString n;
n = s.section(' ',i,i); // color
QColor co;
co.setNamedColor(n);
Pen.setColor(co);
if(!Pen.color().isValid()) return false;
i++;
n = s.section(' ',i,i); // thickness
Pen.setWidth(n.toInt(&ok));
if(!ok) return false;
i++;
n = s.section(' ',i,i); // line style
Pen.setStyle((Qt::PenStyle)n.toInt(&ok));
if(!ok) return false;
return true;
}
// ---------------------------------------------------------------------
bool Component::getBrush(const QString& s, QBrush& Brush, int i)
{
bool ok;
QString n;
n = s.section(' ',i,i); // fill color
QColor co;
co.setNamedColor(n);
Brush.setColor(co);
if(!Brush.color().isValid()) return false;
i++;
n = s.section(' ',i,i); // fill style
Brush.setStyle((Qt::BrushStyle)n.toInt(&ok));
if(!ok) return false;
i++;
n = s.section(' ',i,i); // filled
if(n.toInt(&ok) == 0) Brush.setStyle(Qt::NoBrush);
if(!ok) return false;
return true;
}
// ---------------------------------------------------------------------
Property * Component::getProperty(const QString& name)
{
for(Property *pp = Props.first(); pp != 0; pp = Props.next())
if(pp->Name == name) {
return pp;
}
return NULL;
}
// ---------------------------------------------------------------------
void Component::copyComponent(Component *pc)
{
Type = pc->Type;
x1 = pc->x1;
y1 = pc->y1;
x2 = pc->x2;
y2 = pc->y2;
Model = pc->Model;
Name = pc->Name;
showName = pc->showName;
Description = pc->Description;
isActive = pc->isActive;
rotated = pc->rotated;
mirroredX = pc->mirroredX;
tx = pc->tx;
ty = pc->ty;
Props = pc->Props;
Ports = pc->Ports;
Lines = pc->Lines;
Arcs = pc->Arcs;
Rects = pc->Rects;
Ellips = pc->Ellips;
Texts = pc->Texts;
}
// ***********************************************************************
// ******** ********
// ******** Functions of class MultiViewComponent ********
// ******** ********
// ***********************************************************************
void MultiViewComponent::recreate(Schematic *Doc)
{
if(Doc) {
Doc->Components->setAutoDelete(false);
Doc->deleteComp(this);
}
Ellips.clear();
Texts.clear();
Ports.clear();
Lines.clear();
Rects.clear();
Arcs.clear();
createSymbol();
bool mmir = mirroredX;
int rrot = rotated;
if (mmir && rrot==2) // mirrorX and rotate 180 = mirrorY
mirrorY();
else {
if(mmir)
mirrorX(); // mirror
if (rrot)
for(int z=0; z<rrot; z++) rotate(); // rotate
}
rotated = rrot; // restore properties (were changed by rotate/mirror)
mirroredX = mmir;
if(Doc) {
Doc->insertRawComponent(this);
Doc->Components->setAutoDelete(true);
}
}
// ***********************************************************************
// ******** ********
// ******** Functions of class GateComponent ********
// ******** ********
// ***********************************************************************
GateComponent::GateComponent()
{
Type = isComponent; // both analog and digital
Name = "Y";
// the list order must be preserved !!!
Props.append(new Property("in", "2", false,
QObject::tr("number of input ports")));
Props.append(new Property("V", "1 V", false,
QObject::tr("voltage of high level")));
Props.append(new Property("t", "0", false,
QObject::tr("delay time")));
Props.append(new Property("TR", "10", false,
QObject::tr("transfer function scaling factor")));
// this must be the last property in the list !!!
Props.append(new Property("Symbol", "old", false,
QObject::tr("schematic symbol")+" [old, DIN40900]"));
}
// -------------------------------------------------------
QString GateComponent::netlist()
{
QString s = Model+":"+Name;
// output all node names
foreach(Port *pp, Ports)
s += " "+pp->Connection->Name; // node names
// output all properties
Property *p = Props.at(1);
s += " " + p->Name + "=\"" + p->Value + "\"";
p = Props.next();
s += " " + p->Name + "=\"" + p->Value + "\"";
p = Props.next();
s += " " + p->Name + "=\"" + p->Value + "\"\n";
return s;
}
QString GateComponent::spice_netlist(bool)
{
QString s = Model+Name;
// output all node names
foreach(Port *pp, Ports)
s += " "+pp->Connection->Name; // node names
// output all properties
Property *p = Props.at(1);
s += " " + p->Name + "=\"" + p->Value + "\"";
p = Props.next();
s += " " + p->Name + "=\"" + p->Value + "\"";
p = Props.next();
s += " " + p->Name + "=\"" + p->Value + "\"\n";
return s;
}
// -------------------------------------------------------
QString GateComponent::vhdlCode(int NumPorts)
{
QListIterator<Port *> iport(Ports);
Port *pp = iport.next();
QString s = " " + pp->Connection->Name + " <= "; // output port
// xnor NOT defined for std_logic, so here use not and xor
if (Model == "XNOR") {
QString Op = " xor ";
// first input port
pp = iport.next();
QString rhs = pp->Connection->Name;
// output all input ports with node names
while(iport.hasNext()) {
pp = iport.next();
rhs = "not ((" + rhs + ")" + Op + pp->Connection->Name + ")";
}
s += rhs;
}
else {
QString Op = ' ' + Model.toLower() + ' ';
if(Model.at(0) == 'N') {
s += "not ("; // nor, nand is NOT assoziative !!! but xnor is !!!
Op = Op.remove(1, 1);
}
pp = iport.next();
s += pp->Connection->Name; // first input port
// output all input ports with node names
while(iport.hasNext()) {
pp = iport.next();
s += Op + pp->Connection->Name;
}
if(Model.at(0) == 'N')
s += ')';
}
if(NumPorts <= 0) { // no truth table simulation ?
QString td = Props.at(2)->Value; // delay time
if(!misc::VHDL_Delay(td, Name)) return td;
s += td;
}
s += ";\n";
return s;
}
// -------------------------------------------------------
QString GateComponent::verilogCode(int NumPorts)
{
bool synthesize = true;
QListIterator<Port *> iport(Ports);
Port *pp = iport.next();
QString s("");
if(synthesize) {
QString op = Model.toLower();
if(op == "and" || op == "nand")
op = "&";
else if (op == "or" || op == "nor")
op = "|";
else if (op == "xor")
op = "^";
else if (op == "xnor")
op = "^~";
s = " assign";
if(NumPorts <= 0) { // no truth table simulation ?
QString td = Props.at(2)->Value; // delay time
if(!misc::Verilog_Delay(td, Name)) return td;
s += td;
}
s += " " + pp->Connection->Name + " = "; // output port
if(Model.at(0) == 'N') s += "~(";
pp = iport.next();
s += pp->Connection->Name; // first input port
// output all input ports with node names
while (iport.hasNext()) {
pp = iport.next();
s += " " + op + " " + pp->Connection->Name;
}
if(Model.at(0) == 'N') s += ")";
s += ";\n";
}
else {
s = " " + Model.toLower();
if(NumPorts <= 0) { // no truth table simulation ?
QString td = Props.at(2)->Value; // delay time
if(!misc::Verilog_Delay(td, Name)) return td;
s += td;
}
s += " " + Name + " (" + pp->Connection->Name; // output port
pp = iport.next();
s += ", " + pp->Connection->Name; // first input port
// output all input ports with node names
while (iport.hasNext()) {
pp = iport.next();
s += ", " + pp->Connection->Name;
}
s += ");\n";
}
return s;
}
// -------------------------------------------------------
void GateComponent::createSymbol()
{
int Num = Props.getFirst()->Value.toInt();
if(Num < 2) Num = 2;
else if(Num > 8) Num = 8;
Props.getFirst()->Value = QString::number(Num);
int xl, xr, y = 10*Num, z;
x1 = -30; y1 = -y-3;
x2 = 30; y2 = y+3;
tx = x1+4;
ty = y2+4;
z = 0;
if(Model.at(0) == 'N') z = 1;
if(Props.getLast()->Value.at(0) == 'D') { // DIN symbol
xl = -15;
xr = 15;
Lines.append(new qucs::Line( 15,-y, 15, y,QPen(Qt::darkBlue,2)));
Lines.append(new qucs::Line(-15,-y, 15,-y,QPen(Qt::darkBlue,2)));
Lines.append(new qucs::Line(-15, y, 15, y,QPen(Qt::darkBlue,2)));
Lines.append(new qucs::Line(-15,-y,-15, y,QPen(Qt::darkBlue,2)));
Lines.append(new qucs::Line( 15, 0, 30, 0,QPen(Qt::darkBlue,2)));
if(Model.at(z) == 'O') {
Lines.append(new qucs::Line(-11, 6-y,-6, 9-y,QPen(Qt::darkBlue,0)));
Lines.append(new qucs::Line(-11,12-y,-6, 9-y,QPen(Qt::darkBlue,0)));
Lines.append(new qucs::Line(-11,14-y,-6,14-y,QPen(Qt::darkBlue,0)));
Lines.append(new qucs::Line(-11,16-y,-6,16-y,QPen(Qt::darkBlue,0)));
Texts.append(new Text( -4, 3-y, "1", Qt::darkBlue, 15.0));
}
else if(Model.at(z) == 'A')
Texts.append(new Text( -10, 3-y, "&", Qt::darkBlue, 15.0));
else if(Model.at(0) == 'X') {
if(Model.at(1) == 'N') {
Ellips.append(new qucs::Area(xr,-4, 8, 8,
QPen(Qt::darkBlue,0), QBrush(Qt::darkBlue)));
Texts.append(new Text( -11, 3-y, "=1", Qt::darkBlue, 15.0));
}
else
Texts.append(new Text( -11, 3-y, "=1", Qt::darkBlue, 15.0));
}
}
else { // old symbol
if(Model.at(z) == 'O') xl = 10;
else xl = -10;
xr = 10;
Lines.append(new qucs::Line(-10,-y,-10, y,QPen(Qt::darkBlue,2)));
Lines.append(new qucs::Line( 10, 0, 30, 0,QPen(Qt::darkBlue,2)));
Arcs.append(new qucs::Arc(-30,-y, 40, 30, 0, 16*90,QPen(Qt::darkBlue,2)));
Arcs.append(new qucs::Arc(-30,y-30, 40, 30, 0,-16*90,QPen(Qt::darkBlue,2)));
Lines.append(new qucs::Line( 10,15-y, 10, y-15,QPen(Qt::darkBlue,2)));
if(Model.at(0) == 'X') {
Lines.append(new qucs::Line(-5, 0, 5, 0,QPen(Qt::darkBlue,1)));
if(Model.at(1) == 'N') {
Lines.append(new qucs::Line(-5,-3, 5,-3,QPen(Qt::darkBlue,1)));
Lines.append(new qucs::Line(-5, 3, 5, 3,QPen(Qt::darkBlue,1)));
}
else {
Arcs.append(new qucs::Arc(-5,-5, 10, 10, 0, 16*360,QPen(Qt::darkBlue,1)));
Lines.append(new qucs::Line( 0,-5, 0, 5,QPen(Qt::darkBlue,1)));
}
}
}
if(Model.at(0) == 'N')
Ellips.append(new qucs::Area(xr,-4, 8, 8,
QPen(Qt::darkBlue,0), QBrush(Qt::darkBlue)));
Ports.append(new Port( 30, 0));
y += 10;
for(z=0; z<Num; z++) {
y -= 20;
Ports.append(new Port(-30, y));
if(xl == 10) if((z == 0) || (z == Num-1)) {
Lines.append(new qucs::Line(-30, y, 9, y,QPen(Qt::darkBlue,2)));
continue;
}
Lines.append(new qucs::Line(-30, y, xl, y,QPen(Qt::darkBlue,2)));
}
}
// ***********************************************************************
// ******** ********
// ******** The following function does not belong to any class. ********
// ******** It creates a component by getting the identification ********
// ******** string used in the schematic file and for copy/paste. ********
// ******** ********
// ***********************************************************************
Component* getComponentFromName(QString& Line, Schematic* p)
{
Component *c = 0;
Line = Line.trimmed();
if(Line.at(0) != '<') {
QMessageBox::critical(0, QObject::tr("Error"),
QObject::tr("Format Error:\nWrong line start!"));
return 0;
}
QString cstr = Line.section (' ',0,0); // component type
cstr.remove (0,1); // remove leading "<"
if (cstr == "Lib") c = new LibComp ();
else if (cstr == "Eqn") c = new Equation ();
else if (cstr == "SPICE") c = new SpiceFile();
else if (cstr == "Rus") c = new Resistor (false); // backward compatible
else if (cstr.left (6) == "SPfile" && cstr != "SPfile") {
// backward compatible
c = new SParamFile ();
c->Props.getLast()->Value = cstr.mid (6); }
else
c = Module::getComponent (cstr);
if(!c) {
/// \todo enable user to load partial schematic, skip unknown components
if (QucsMain!=0) {
QMessageBox* msg = new QMessageBox(QMessageBox::Warning,QObject::tr("Warning"),
QObject::tr("Format Error:\nUnknown component!\n"
"%1\n\n"
"Do you want to load schematic anyway?\n"
"Unknown components will be replaced \n"
"by dummy subcircuit placeholders.").arg(cstr),
QMessageBox::Yes|QMessageBox::No);
int r = msg->exec();
delete msg;
if (r == QMessageBox::Yes) {
c = new Subcircuit();
// Hack: insert dummy File property before the first property
int pos1 = Line.indexOf('"');
QString filestr = QString("\"%1.sch\" 1 ").arg(cstr);
Line.insert(pos1,filestr);
} else return 0;
} else {
QString err_msg = QString("Schematic loading error! Unknown device %1").arg(cstr);
qCritical()<<err_msg;
return 0;
}
}
if(!c->load(Line)) {
QMessageBox::critical(0, QObject::tr("Error"),
QObject::tr("Format Error:\nWrong 'component' line format!"));
delete c;
return 0;
}
cstr = c->Name; // is perhaps changed in "recreate" (e.g. subcircuit)
int x = c->tx, y = c->ty;
c->setSchematic (p);
c->recreate(0);
c->Name = cstr;
c->tx = x; c->ty = y;
return c;
}