/*************************************************************************** mutualx.cpp ----------- begin : Mon Nov 24 2014 copyright : (C) 2010 by Michael Margraf email : michael.margraf@alumni.tu-berlin.de copyright : (C) 2014 by Vadim Kuznetsov email : ra3xdh@gmail.com ***************************************************************************/ /*************************************************************************** * * * 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 "mutualx.h" #include "node.h" #include "extsimkernels/spicecompat.h" #include MutualX::MutualX() { Description = QObject::tr("several mutual inductors"); Simulator = spicecompat::simAll; Model = "MUTX"; Name = "Tr"; SpiceModel = "K"; const int init_coils=2; // initial number of coils // must be the first property! Props.append(new Property("coils", QString::number(init_coils), false, QObject::tr("number of mutual inductances"))); for (int i=1;i<=init_coils; i++) { Props.append(new Property("L"+QString::number(i), "1 mH", false, QObject::tr("inductance of coil") + " " + QString::number(i))); } for(int i = 1; i < init_coils; i++) for(int j = i+1; j <= init_coils; j++) { QString nam = "k" + QString::number(i) + QString::number(j); QString desc = QObject::tr("coupling factor between coil %1 and coil %2").arg(i).arg(j); Props.append(new Property(nam,"0.9",false,desc)); } createSymbol(); } // -------------------------------------------------------- Element* MutualX::info(QString& Name, char* &BitmapFile, bool getNewOne) { Name = QObject::tr("N Mutual Inductors"); BitmapFile = (char *) "mutualx"; if(getNewOne) { MutualX* p = new MutualX(); p->Props.at(0)->Value = "2"; p->recreate(0); return p; } return 0; } Component* MutualX::newOne() { MutualX *p = new MutualX(); p->Props.at(0)->Value=Props.at(0)->Value; p->recreate(0); return p; } QString MutualX::netlist() { QString s = Model + ":" + Name; // output all node names for (Port *p1 : Ports) { s += " "+p1->Connection->Name; // node names } int coils = Props.at(0)->Value.toInt(); QString L=""; for (int i=0;iValue + ";"; } L.chop(1); QString k=""; QString **k_matrix = new QString*[coils]; for (int i=0;iValue; // for mutual inductances } } for (int j=0;jValue.toInt(); if(Num < 2) Num = 2; else if(Num > 8) Num = 8; Props.first()->Value = QString::number(Num); int NumProps,oldNumProps; oldNumProps = Props.count(); NumProps = Num + Num * (Num - 1) / 2 + 1; if (oldNumProps!=NumProps) { // Coils count was changed int oldCoils = rint(0.5*(sqrt(8*oldNumProps-7)-1.0)); // calculate old number of coils // we need to solve quadratic equation int dCoils = abs(oldCoils - Num); // how many coils were added/removed? if (oldCoils>Num) { // reduce coils number for(int i = 0; i < dCoils; i++){ Props.removeAt(Num+1); // remove excess coils } // remove only the no longer valid coupling coefficients, leave the // ones related to existing coils untouched for(int i = 1,state=1; i < oldCoils; i++) for(int j = i+1; j <= oldCoils; j++,state++) { if ((i>Num)||(j>Num)) { Props.removeAt(Num + state); state--; } } } else { // add new coils for(int i = 0; i < dCoils; i++) { // add new properties for coils Props.insert(oldCoils+1, new Property("L"+QString::number(Num-i), "1 mH", false, QObject::tr("inductance of coil") + " " + QString::number(Num-i))); } for(int i = 1,state=1; i < Num; i++) for(int j = i+1; j <= Num; j++,state++) { if ((i>oldCoils)||(j>oldCoils)) { Props.insert(Num + state, new Property("k", "0.9", false, " ")); } } } } // in any case rewrite properties Name and Description // (when loading a component, added properties have a default name) // adjust coils names auto p1 = Props.begin(); ++p1; for(int i = 1; i <= Num; i++) { (*p1)->Name = "L"+QString::number(i); (*p1)->Description = QObject::tr("inductance of coil") + " " + QString::number(i); p1++; } // adjust coupling coeffs names for(int i = 1,state=1; i < Num; i++) for(int j = i+1; j <= Num; j++,state++) { Props.at(Num+state)->Name = "k" + QString::number(i) + QString::number(j); Props.at(Num+state)->Description = QObject::tr("coupling factor between coil %1 and coil %2").arg(i).arg(j); } // draw symbol int x = -10 * (Num-1); Texts.append(new Text(x-9,-22,"1")); x1 = x-6; y1 = -30; x2 = 10-x; y2 = 30; tx = x2+4; ty = y1+4; x -= 6; for(int i=0; iValue.toInt(); QString s; for (int i = 0; i < coils; i++) { QString li = "L" + Name + "_L" + QString::number(i+1); QString prop_l = "L" + QString::number(i+1); QString ind = spicecompat::normalize_value(getProperty(prop_l)->Value); s += QStringLiteral("%1 %2 %3 %4\n").arg(li) .arg(spicecompat::normalize_node_name(Ports.at(2*i)->Connection->Name)) .arg(spicecompat::normalize_node_name(Ports.at(2*i+1)->Connection->Name)) .arg(ind); } for (int i = 0; i < coils; i++) { for (int j = i; j < coils; j++) { if (i == j) continue; QString kij = "K" + Name + "_K" + QString::number(i+1) + QString::number(j+1); QString li = "L" + Name + "_L" + QString::number(i+1); QString lj = "L" + Name + "_L" + QString::number(j+1); auto pp = getProperty("k" + QString::number(i+1) + QString::number(j+1)); if (pp == nullptr) continue; QString val_k = spicecompat::normalize_value(pp->Value); s += QStringLiteral("%1 %2 %3 %4\n").arg(kij).arg(li).arg(lj).arg(val_k); } } return s; }