2016-01-21 10:27:43 +03:00
|
|
|
/***************************************************************************
|
|
|
|
XspiceGeneric.cpp
|
|
|
|
---------------
|
|
|
|
begin : Tue Dez 28 2004
|
|
|
|
copyright : (C) 2004 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. *
|
|
|
|
* *
|
|
|
|
***************************************************************************/
|
2022-02-15 22:41:25 +01:00
|
|
|
#include <QString>
|
|
|
|
#include <QStringList>
|
2016-01-21 10:27:43 +03:00
|
|
|
#include <QFontMetrics>
|
2022-02-23 21:51:56 +01:00
|
|
|
#include <QtGlobal>
|
2016-01-21 10:27:43 +03:00
|
|
|
|
|
|
|
#include "xspicegeneric.h"
|
|
|
|
|
|
|
|
#include "main.h"
|
2024-05-20 21:46:20 +03:00
|
|
|
#include "node.h"
|
2016-01-21 10:27:43 +03:00
|
|
|
#include "extsimkernels/spicecompat.h"
|
|
|
|
|
|
|
|
|
|
|
|
XspiceGeneric::XspiceGeneric()
|
|
|
|
{
|
2016-01-24 14:45:10 +03:00
|
|
|
Description = QObject::tr("XSPICE generic device");
|
2023-06-09 13:54:29 +03:00
|
|
|
Simulator = spicecompat::simSpice;
|
2016-01-21 10:27:43 +03:00
|
|
|
// Property descriptions not needed, but must not be empty !
|
2016-01-26 13:24:42 +03:00
|
|
|
Props.append(new Property("PortList", "v,v", true, QObject::tr("PortsList")));
|
2016-01-24 14:45:10 +03:00
|
|
|
Props.append(new Property("Model", "generic_model", false, QObject::tr(".MODEL definition reference")));
|
2016-01-21 10:27:43 +03:00
|
|
|
|
|
|
|
Model = "XSPICE_A";
|
|
|
|
SpiceModel = "A";
|
|
|
|
Name = "A";
|
|
|
|
changed = false;
|
|
|
|
|
|
|
|
// Do NOT call createSymbol() here. But create port to let it rotate.
|
|
|
|
Ports.append(new Port(0, 0));
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------
|
|
|
|
Component* XspiceGeneric::newOne()
|
|
|
|
{
|
|
|
|
XspiceGeneric *p = new XspiceGeneric();
|
|
|
|
p->recreate(0); // createSymbol() is NOT called in constructor !!!
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------
|
|
|
|
Element* XspiceGeneric::info(QString& Name, char* &BitmapFile, bool getNewOne)
|
|
|
|
{
|
|
|
|
Name = QObject::tr("XSPICE generic device");
|
|
|
|
BitmapFile = (char *) "xspicegeneric";
|
|
|
|
|
|
|
|
if(getNewOne) {
|
|
|
|
XspiceGeneric *p = new XspiceGeneric();
|
|
|
|
p->recreate(0); // createSymbol() is NOT called in constructor !!!
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// -------------------------------------------------------
|
|
|
|
void XspiceGeneric::createSymbol()
|
|
|
|
{
|
|
|
|
QFont f = QucsSettings.font; // get the basic font
|
|
|
|
// symbol text is smaller (10 pt default)
|
|
|
|
f.setPointSize(10);
|
|
|
|
// use the screen-compatible metric
|
|
|
|
QFontMetrics smallmetrics(f, 0); // get size of text
|
|
|
|
int fHeight = smallmetrics.lineSpacing();
|
|
|
|
|
2016-01-26 13:24:42 +03:00
|
|
|
QStringList t_ports = Props.at(0)->Value.split(',');
|
|
|
|
QStringList n_ports;
|
|
|
|
int k=0;
|
2023-01-15 01:17:09 +03:00
|
|
|
for (QString t_port : t_ports) {
|
2016-04-14 09:01:59 +03:00
|
|
|
t_port.remove(']').remove('[');
|
2016-09-12 15:07:46 +03:00
|
|
|
if ((t_port.remove(' ').endsWith('d'))&&
|
|
|
|
(t_port!="d")) {
|
2016-01-26 13:24:42 +03:00
|
|
|
n_ports.append(t_port+QString::number(k)+"+");
|
|
|
|
n_ports.append(t_port+QString::number(k)+"-");
|
|
|
|
} else n_ports.append(t_port+QString::number(k));
|
|
|
|
k++;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int No = n_ports.count();
|
2016-01-21 10:27:43 +03:00
|
|
|
QString tmp;
|
2024-12-30 16:48:05 +01:00
|
|
|
|
2016-01-21 10:27:43 +03:00
|
|
|
// draw symbol outline
|
2016-01-22 18:02:20 +03:00
|
|
|
#define HALFWIDTH 27
|
2016-01-21 10:27:43 +03:00
|
|
|
int h = 30*((No-1)/2) + 15;
|
2022-02-20 15:43:44 +01:00
|
|
|
Lines.append(new qucs::Line(-HALFWIDTH, -h, HALFWIDTH, -h,QPen(Qt::darkBlue,2)));
|
|
|
|
Lines.append(new qucs::Line( HALFWIDTH, -h, HALFWIDTH, h,QPen(Qt::darkBlue,2)));
|
|
|
|
Lines.append(new qucs::Line(-HALFWIDTH, h, HALFWIDTH, h,QPen(Qt::darkBlue,2)));
|
|
|
|
Lines.append(new qucs::Line(-HALFWIDTH, -h,-HALFWIDTH, h,QPen(Qt::darkBlue,2)));
|
2016-01-21 10:27:43 +03:00
|
|
|
|
|
|
|
int w, i = fHeight/2;
|
|
|
|
|
2016-01-22 18:02:20 +03:00
|
|
|
tmp = QObject::tr("XSPICE");
|
2016-01-21 10:27:43 +03:00
|
|
|
w = smallmetrics.boundingRect(tmp).width();
|
|
|
|
Texts.append(new Text(w/-2, -i, tmp));
|
|
|
|
|
2016-01-26 13:24:42 +03:00
|
|
|
i = 0;
|
2016-01-21 10:27:43 +03:00
|
|
|
int y = 15-h;
|
2016-01-26 13:24:42 +03:00
|
|
|
while(i<No) { // add ports lines and numbers
|
2022-02-20 15:43:44 +01:00
|
|
|
Lines.append(new qucs::Line(-40, y,-HALFWIDTH, y,QPen(Qt::darkBlue,2)));
|
2016-01-22 18:02:20 +03:00
|
|
|
Ports.append(new Port(-40, y));
|
2016-01-21 10:27:43 +03:00
|
|
|
// tmp = PortNames.section(',', i, i).mid(4);
|
2016-01-26 13:24:42 +03:00
|
|
|
tmp = n_ports.at(i);
|
2022-02-23 21:51:56 +01:00
|
|
|
w = smallmetrics.boundingRect(tmp).width();
|
2016-01-22 18:02:20 +03:00
|
|
|
Texts.append(new Text(-40-w, y-fHeight-2, tmp)); // text right-aligned
|
2016-01-21 10:27:43 +03:00
|
|
|
i++;
|
|
|
|
|
2016-01-26 13:24:42 +03:00
|
|
|
if(i == No) break; // if odd number of ports there will be one port less on the right side
|
2022-02-20 15:43:44 +01:00
|
|
|
Lines.append(new qucs::Line(HALFWIDTH, y, 40, y,QPen(Qt::darkBlue,2)));
|
2016-01-22 18:02:20 +03:00
|
|
|
Ports.append(new Port( 40, y));
|
2016-01-26 13:24:42 +03:00
|
|
|
tmp = n_ports.at(i);
|
2016-01-22 18:02:20 +03:00
|
|
|
Texts.append(new Text( 40, y-fHeight-2, tmp)); // text left-aligned
|
2016-01-21 10:27:43 +03:00
|
|
|
y += 60;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
x1 = -30; y1 = -h-2;
|
|
|
|
x2 = 30; y2 = h+15;
|
|
|
|
|
|
|
|
// compute component name text position - normal size font
|
|
|
|
QFontMetrics metrics(QucsSettings.font, 0); // use the screen-compatible metric
|
|
|
|
fHeight = metrics.lineSpacing();
|
|
|
|
tx = x1+4;
|
|
|
|
ty = y1 - fHeight - 4;
|
|
|
|
if(Props.first()->display) ty -= fHeight;
|
|
|
|
changed = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
QString XspiceGeneric::netlist()
|
|
|
|
{
|
2024-10-19 15:13:10 +08:00
|
|
|
return QString();
|
2016-01-21 10:27:43 +03:00
|
|
|
}
|
|
|
|
|
2024-12-30 16:48:05 +01:00
|
|
|
QString XspiceGeneric::spice_netlist(spicecompat::SpiceDialect dialect /* = spicecompat::SPICEDefault */)
|
2016-01-21 10:27:43 +03:00
|
|
|
{
|
2024-12-30 16:48:05 +01:00
|
|
|
Q_UNUSED(dialect);
|
|
|
|
|
2016-01-24 14:45:10 +03:00
|
|
|
QString s = spicecompat::check_refdes(Name,SpiceModel);
|
2016-01-21 10:27:43 +03:00
|
|
|
|
2016-01-26 13:24:42 +03:00
|
|
|
QStringList t_ports = Props.at(0)->Value.split(',');
|
|
|
|
|
|
|
|
int i=0;
|
2023-01-15 01:17:09 +03:00
|
|
|
for(QString t_port : t_ports) {
|
2016-01-26 13:24:42 +03:00
|
|
|
QString nod = spicecompat::normalize_node_name(Ports.at(i)->Connection->Name);
|
2016-04-14 09:01:59 +03:00
|
|
|
|
|
|
|
if (t_port.contains('[')) { // Process array ports
|
|
|
|
t_port.remove('['); // Defined by brackets
|
|
|
|
s += '[';
|
|
|
|
}
|
|
|
|
bool closing_bracket = t_port.contains(']');
|
|
|
|
if (closing_bracket) t_port.remove(']');
|
|
|
|
|
2016-09-12 15:07:46 +03:00
|
|
|
if ((t_port.remove(' ').endsWith('d'))&&
|
|
|
|
(t_port!="d")) {
|
2016-01-26 13:24:42 +03:00
|
|
|
i++;
|
|
|
|
QString nod1 = spicecompat::normalize_node_name(Ports.at(i)->Connection->Name);
|
2024-11-04 15:53:11 +08:00
|
|
|
s += QStringLiteral(" %%1(%2 %3) ").arg(t_port).arg(nod).arg(nod1);
|
2016-01-26 13:24:42 +03:00
|
|
|
} else {
|
2024-11-04 15:53:11 +08:00
|
|
|
s += QStringLiteral(" %%1(%2) ").arg(t_port).arg(nod);
|
2016-01-26 13:24:42 +03:00
|
|
|
}
|
2016-04-14 09:01:59 +03:00
|
|
|
if (closing_bracket) s += ']';
|
2016-01-26 13:24:42 +03:00
|
|
|
i++;
|
2016-01-21 10:27:43 +03:00
|
|
|
}
|
|
|
|
s += " " + Props.at(1)->Value + "\n";
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|