2006-03-28 06:10:52 +00:00
|
|
|
|
/***************************************************************************
|
|
|
|
|
schematic.cpp
|
|
|
|
|
---------------
|
|
|
|
|
begin : Sat Mar 3 2006
|
|
|
|
|
copyright : (C) 2006 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. *
|
|
|
|
|
* *
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
2024-01-05 16:15:57 +03:00
|
|
|
|
#include <algorithm>
|
2024-01-05 15:30:51 +03:00
|
|
|
|
#include <cassert>
|
2006-03-28 06:10:52 +00:00
|
|
|
|
#include <limits.h>
|
2023-12-11 15:13:33 +03:00
|
|
|
|
#include <stdlib.h>
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
#include <QAction>
|
|
|
|
|
#include <QApplication>
|
|
|
|
|
#include <QClipboard>
|
|
|
|
|
#include <QDebug>
|
2013-11-26 16:06:53 +01:00
|
|
|
|
#include <QDir>
|
2012-10-31 09:15:06 +01:00
|
|
|
|
#include <QDragEnterEvent>
|
2023-12-11 15:13:33 +03:00
|
|
|
|
#include <QDragLeaveEvent>
|
2012-10-31 09:15:06 +01:00
|
|
|
|
#include <QDragMoveEvent>
|
|
|
|
|
#include <QDropEvent>
|
|
|
|
|
#include <QEvent>
|
2023-12-11 15:13:33 +03:00
|
|
|
|
#include <QFileInfo>
|
2014-11-04 12:22:29 +08:00
|
|
|
|
#include <QLineEdit>
|
|
|
|
|
#include <QListWidget>
|
2023-12-11 15:13:33 +03:00
|
|
|
|
#include <QMouseEvent>
|
|
|
|
|
#include <QPaintDevice>
|
|
|
|
|
#include <QPainter>
|
|
|
|
|
#include <QPixmap>
|
2023-12-29 18:27:44 +03:00
|
|
|
|
#include <QPoint>
|
2023-12-11 15:13:33 +03:00
|
|
|
|
#include <QPrinter>
|
2023-12-29 18:27:44 +03:00
|
|
|
|
#include <QRect>
|
2023-12-11 15:13:33 +03:00
|
|
|
|
#include <QTextStream>
|
|
|
|
|
#include <QUrl>
|
|
|
|
|
#include <QWheelEvent>
|
|
|
|
|
#include <qt3_compat/qt_compat.h>
|
2024-12-11 18:53:56 +03:00
|
|
|
|
#include <QRegularExpression>
|
2014-10-30 03:32:41 +08:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
#include "components/vafile.h"
|
|
|
|
|
#include "components/verilogfile.h"
|
|
|
|
|
#include "components/vhdlfile.h"
|
|
|
|
|
#include "diagrams/diagrams.h"
|
2006-07-03 06:02:08 +00:00
|
|
|
|
#include "main.h"
|
2023-12-11 15:13:33 +03:00
|
|
|
|
#include "mouseactions.h"
|
2006-07-03 06:02:08 +00:00
|
|
|
|
#include "node.h"
|
2023-12-11 15:13:33 +03:00
|
|
|
|
#include "paintings/paintings.h"
|
|
|
|
|
#include "qucs.h"
|
|
|
|
|
#include "schematic.h"
|
2024-11-14 21:15:18 +00:00
|
|
|
|
#include "settings.h"
|
2009-10-19 16:34:25 +00:00
|
|
|
|
#include "textdoc.h"
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2022-02-14 15:37:11 +01:00
|
|
|
|
#include "misc.h"
|
|
|
|
|
|
2006-03-28 06:10:52 +00:00
|
|
|
|
// just dummies for empty lists
|
2023-12-11 15:13:33 +03:00
|
|
|
|
Q3PtrList<Wire> SymbolWires;
|
|
|
|
|
Q3PtrList<Node> SymbolNodes;
|
|
|
|
|
Q3PtrList<Diagram> SymbolDiags;
|
2012-10-31 09:15:06 +01:00
|
|
|
|
Q3PtrList<Component> SymbolComps;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-01-25 18:14:45 +03:00
|
|
|
|
/**
|
|
|
|
|
If \c point does not lie within \c rect then returns a new
|
|
|
|
|
rectangle made by enlarging the source rectangle to include
|
|
|
|
|
the \c point. Otherwise returns a rectangle of the same size.
|
|
|
|
|
*/
|
|
|
|
|
static QRect includePoint(QRect rect, QPoint point) {
|
|
|
|
|
if (rect.contains(point)) {
|
|
|
|
|
return rect;
|
|
|
|
|
} else {
|
|
|
|
|
return rect.united(QRect{point, point});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
Schematic::Schematic(QucsApp *App_, const QString &Name_) :
|
|
|
|
|
QucsDoc(App_, Name_),
|
|
|
|
|
a_Wires(&a_DocWires),
|
|
|
|
|
a_DocWires(),
|
|
|
|
|
a_Nodes(&a_DocNodes),
|
|
|
|
|
a_DocNodes(),
|
|
|
|
|
a_Diagrams(&a_DocDiags),
|
|
|
|
|
a_DocDiags(),
|
|
|
|
|
a_Paintings(&a_DocPaints),
|
|
|
|
|
a_DocPaints(),
|
|
|
|
|
a_Components(&a_DocComps),
|
|
|
|
|
a_DocComps(),
|
|
|
|
|
a_SymbolPaints(),
|
|
|
|
|
a_PostedPaintEvents(),
|
|
|
|
|
a_symbolMode(false),
|
|
|
|
|
a_isSymbolOnly(false),
|
|
|
|
|
a_GridX(10),
|
|
|
|
|
a_GridY(10),
|
|
|
|
|
a_ViewX1(0),
|
|
|
|
|
a_ViewY1(0),
|
|
|
|
|
a_ViewX2(1),
|
|
|
|
|
a_ViewY2(1),
|
|
|
|
|
a_showFrame(0), // don't show
|
|
|
|
|
a_Frame_Text0(tr("Title")),
|
|
|
|
|
a_Frame_Text1(tr("Drawn By:")),
|
|
|
|
|
a_Frame_Text2(tr("Date:")),
|
|
|
|
|
a_Frame_Text3(tr("Revision:")),
|
|
|
|
|
a_tmpScale(1.0),
|
|
|
|
|
a_tmpViewX1(-200),
|
|
|
|
|
a_tmpViewY1(-200),
|
|
|
|
|
a_tmpViewX2(200),
|
|
|
|
|
a_tmpViewY2(200),
|
|
|
|
|
a_tmpUsedX1(-200),
|
|
|
|
|
a_tmpUsedY1(-200),
|
|
|
|
|
a_tmpUsedX2(200),
|
|
|
|
|
a_tmpUsedY2(200),
|
|
|
|
|
a_undoActionIdx(0),
|
|
|
|
|
// The 'i' means state for being unchanged.
|
|
|
|
|
a_undoAction((QVector<QString*>() << new QString(" i\n</>\n</>\n</>\n</>\n"))),
|
|
|
|
|
a_undoSymbolIdx(0),
|
|
|
|
|
// The 'i' means state for being unchanged.
|
|
|
|
|
a_undoSymbol((QVector<QString*>() << new QString(" i\n</>\n</>\n</>\n</>\n"))),
|
|
|
|
|
a_UsedX1(INT_MAX),
|
|
|
|
|
a_UsedY1(INT_MAX),
|
|
|
|
|
a_UsedX2(INT_MAX),
|
|
|
|
|
a_UsedY2(INT_MAX),
|
|
|
|
|
a_previousCursorPosition(),
|
|
|
|
|
a_dragIsOkay(false),
|
|
|
|
|
a_FileInfo(),
|
|
|
|
|
a_Signals(),
|
|
|
|
|
a_PortTypes(),
|
|
|
|
|
a_isAnalog(false),
|
|
|
|
|
a_isVerilog(false),
|
|
|
|
|
a_creatingLib(false)
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
setFont(QucsSettings.font);
|
2024-11-15 20:53:58 +00:00
|
|
|
|
a_GridColor = _settings::Get().item<QString>("GridColor");
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_tmpPosX = a_tmpPosY = -100;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_DocComps.setAutoDelete(true);
|
|
|
|
|
a_DocWires.setAutoDelete(true);
|
|
|
|
|
a_DocNodes.setAutoDelete(true);
|
|
|
|
|
a_DocDiags.setAutoDelete(true);
|
|
|
|
|
a_DocPaints.setAutoDelete(true);
|
|
|
|
|
a_SymbolPaints.setAutoDelete(true);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
setVScrollBarMode(Q3ScrollView::AlwaysOn);
|
|
|
|
|
setHScrollBarMode(Q3ScrollView::AlwaysOn);
|
|
|
|
|
misc::setWidgetBackgroundColor(viewport(), QucsSettings.BGColor);
|
|
|
|
|
viewport()->setMouseTracking(true);
|
|
|
|
|
viewport()->setAcceptDrops(true); // enable drag'n drop
|
|
|
|
|
|
|
|
|
|
// to repair some strange scrolling artefacts
|
|
|
|
|
connect(this, SIGNAL(horizontalSliderReleased()), viewport(), SLOT(update()));
|
|
|
|
|
connect(this, SIGNAL(verticalSliderReleased()), viewport(), SLOT(update()));
|
|
|
|
|
if (App_) {
|
2024-01-24 11:34:49 +03:00
|
|
|
|
connect(this,SIGNAL(signalCursorPosChanged(int, int, QString)),App_,SLOT(printCursorPosition(int, int, QString)));
|
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
connect(this, SIGNAL(horizontalSliderPressed()), App_, SLOT(slotHideEdit()));
|
|
|
|
|
connect(this, SIGNAL(verticalSliderPressed()), App_, SLOT(slotHideEdit()));
|
|
|
|
|
connect(this, SIGNAL(signalUndoState(bool)), App_, SLOT(slotUpdateUndo(bool)));
|
|
|
|
|
connect(this, SIGNAL(signalRedoState(bool)), App_, SLOT(slotUpdateRedo(bool)));
|
|
|
|
|
connect(this, SIGNAL(signalFileChanged(bool)), App_, SLOT(slotFileChanged(bool)));
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
Schematic::~Schematic() {}
|
|
|
|
|
|
2006-06-06 06:14:17 +00:00
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
bool Schematic::createSubcircuitSymbol()
|
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// If the number of ports is not equal, remove or add some.
|
|
|
|
|
unsigned int countPort = adjustPortNumbers();
|
2006-06-06 06:14:17 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// If a symbol does not yet exist, create one.
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_SymbolPaints.count() != countPort)
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return false;
|
2006-06-06 06:14:17 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
int h = 30 * ((countPort - 1) / 2) + 10;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_SymbolPaints.prepend(new ID_Text(-20, h + 4));
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_SymbolPaints.append(new GraphicLine(-20, -h, 40, 0, QPen(Qt::darkBlue, 2)));
|
|
|
|
|
a_SymbolPaints.append(new GraphicLine(20, -h, 0, 2 * h, QPen(Qt::darkBlue, 2)));
|
|
|
|
|
a_SymbolPaints.append(new GraphicLine(-20, h, 40, 0, QPen(Qt::darkBlue, 2)));
|
|
|
|
|
a_SymbolPaints.append(new GraphicLine(-20, -h, 0, 2 * h, QPen(Qt::darkBlue, 2)));
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
unsigned int i = 0, y = 10 - h;
|
|
|
|
|
while (i < countPort) {
|
|
|
|
|
i++;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_SymbolPaints.append(new GraphicLine(-30, y, 10, 0, QPen(Qt::darkBlue, 2)));
|
|
|
|
|
a_SymbolPaints.at(i)->setCenter(-30, y);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
if (i == countPort)
|
|
|
|
|
break;
|
|
|
|
|
i++;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_SymbolPaints.append(new GraphicLine(20, y, 10, 0, QPen(Qt::darkBlue, 2)));
|
|
|
|
|
a_SymbolPaints.at(i)->setCenter(30, y);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
y += 60;
|
|
|
|
|
}
|
|
|
|
|
return true;
|
2006-06-06 06:14:17 +00:00
|
|
|
|
}
|
|
|
|
|
|
2006-03-28 06:10:52 +00:00
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
void Schematic::becomeCurrent(bool update)
|
|
|
|
|
{
|
2024-01-24 11:34:49 +03:00
|
|
|
|
emit signalCursorPosChanged(0, 0, "");
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
// update appropriate menu entry
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_symbolMode) {
|
|
|
|
|
a_App->symEdit->setText(tr("Edit Schematic"));
|
|
|
|
|
a_App->symEdit->setStatusTip(tr("Edits the schematic"));
|
|
|
|
|
a_App->symEdit->setWhatsThis(tr("Edit Schematic\n\nEdits the schematic"));
|
2023-12-11 15:13:33 +03:00
|
|
|
|
} else {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->symEdit->setText(tr("Edit Circuit Symbol"));
|
|
|
|
|
a_App->symEdit->setStatusTip(tr("Edits the symbol for this schematic"));
|
|
|
|
|
a_App->symEdit->setWhatsThis(
|
2023-12-11 15:13:33 +03:00
|
|
|
|
tr("Edit Circuit Symbol\n\nEdits the symbol for this schematic"));
|
2009-10-19 16:34:25 +00:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_symbolMode) {
|
|
|
|
|
a_Nodes = &SymbolNodes;
|
|
|
|
|
a_Wires = &SymbolWires;
|
|
|
|
|
a_Diagrams = &SymbolDiags;
|
|
|
|
|
a_Paintings = &a_SymbolPaints;
|
|
|
|
|
a_Components = &SymbolComps;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-04-13 13:58:56 +03:00
|
|
|
|
// "Schematic" is used to edit usual schematic files (containing
|
|
|
|
|
// a schematic and a subcircuit symbol) and *.sym files (which
|
|
|
|
|
// contain *only* a symbol definition). If we're dealing with
|
|
|
|
|
// symbol file, then there is no need to create a subcircuit
|
|
|
|
|
// symbol, a symbol is already there.
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (!a_DocName.endsWith(".sym") && createSubcircuitSymbol()) {
|
2024-04-05 17:49:51 +03:00
|
|
|
|
updateAllBoundingRect();
|
2023-12-11 15:13:33 +03:00
|
|
|
|
setChanged(true, true);
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
emit signalUndoState(a_undoSymbolIdx != 0);
|
|
|
|
|
emit signalRedoState(a_undoSymbolIdx != a_undoSymbol.size() - 1);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
} else {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Nodes = &a_DocNodes;
|
|
|
|
|
a_Wires = &a_DocWires;
|
|
|
|
|
a_Diagrams = &a_DocDiags;
|
|
|
|
|
a_Paintings = &a_DocPaints;
|
|
|
|
|
a_Components = &a_DocComps;
|
|
|
|
|
|
|
|
|
|
emit signalUndoState(a_undoActionIdx != 0);
|
|
|
|
|
emit signalRedoState(a_undoActionIdx != a_undoAction.size() - 1);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (update)
|
|
|
|
|
reloadGraphs(); // load recent simulation data
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
2023-12-11 15:13:33 +03:00
|
|
|
|
void Schematic::setName(const QString &Name_)
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_DocName = Name_;
|
|
|
|
|
QFileInfo Info(a_DocName);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
QString base = Info.completeBaseName();
|
|
|
|
|
QString ext = Info.suffix();
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_DataSet = base + ".dat";
|
|
|
|
|
a_Script = base + ".m";
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (ext != "dpl")
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_DataDisplay = base + ".dpl";
|
2023-12-11 15:13:33 +03:00
|
|
|
|
else
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_DataDisplay = base + ".sch";
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
// Sets the document to be changed or not to be changed.
|
|
|
|
|
void Schematic::setChanged(bool c, bool fillStack, char Op)
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if ((!a_DocChanged) && c)
|
2023-12-11 15:13:33 +03:00
|
|
|
|
emit signalFileChanged(true);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
else if (a_DocChanged && (!c))
|
2023-12-11 15:13:33 +03:00
|
|
|
|
emit signalFileChanged(false);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_DocChanged = c;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_showBias = -1; // schematic changed => bias points may be invalid
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (!fillStack)
|
|
|
|
|
return;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// ................................................
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_symbolMode) { // for symbol edit mode
|
|
|
|
|
while (a_undoSymbol.size() > a_undoSymbolIdx + 1) {
|
|
|
|
|
delete a_undoSymbol.last();
|
|
|
|
|
a_undoSymbol.pop_back();
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_undoSymbol.append(new QString(createSymbolUndoString(Op)));
|
|
|
|
|
a_undoSymbolIdx++;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
emit signalUndoState(true);
|
|
|
|
|
emit signalRedoState(false);
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
while (static_cast<unsigned int>(a_undoSymbol.size())
|
2023-12-11 15:13:33 +03:00
|
|
|
|
> QucsSettings.maxUndo) { // "while..." because
|
2024-11-08 12:46:57 +01:00
|
|
|
|
delete a_undoSymbol.first();
|
|
|
|
|
a_undoSymbol.pop_front();
|
|
|
|
|
a_undoSymbolIdx--;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
|
|
|
|
return;
|
2014-11-08 23:58:37 +08:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// ................................................
|
|
|
|
|
// for schematic edit mode
|
2024-11-08 12:46:57 +01:00
|
|
|
|
while (a_undoAction.size() > a_undoActionIdx + 1) {
|
|
|
|
|
delete a_undoAction.last();
|
|
|
|
|
a_undoAction.pop_back();
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (Op == 'm') { // only one for move marker
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_undoAction.at(a_undoActionIdx)->at(0) == Op) {
|
|
|
|
|
delete a_undoAction.last();
|
|
|
|
|
a_undoAction.pop_back();
|
|
|
|
|
a_undoActionIdx--;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_undoAction.append(new QString(createUndoString(Op)));
|
|
|
|
|
a_undoActionIdx++;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2014-12-10 16:47:32 +08:00
|
|
|
|
emit signalUndoState(true);
|
|
|
|
|
emit signalRedoState(false);
|
2006-04-10 06:12:35 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
while (static_cast<unsigned int>(a_undoAction.size())
|
2023-12-11 15:13:33 +03:00
|
|
|
|
> QucsSettings.maxUndo) { // "while..." because
|
2024-11-08 12:46:57 +01:00
|
|
|
|
delete a_undoAction.first(); // "maxUndo" could be decreased meanwhile
|
|
|
|
|
a_undoAction.pop_front();
|
|
|
|
|
a_undoActionIdx--;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
2006-04-10 06:12:35 +00:00
|
|
|
|
return;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2006-07-17 06:02:57 +00:00
|
|
|
|
// -----------------------------------------------------------
|
2023-12-11 15:13:33 +03:00
|
|
|
|
bool Schematic::sizeOfFrame(int &xall, int &yall)
|
2006-07-17 06:02:57 +00:00
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// Values exclude border of 1.5cm at each side.
|
2024-11-08 12:46:57 +01:00
|
|
|
|
switch (a_showFrame) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
case 1:
|
|
|
|
|
xall = 1020;
|
|
|
|
|
yall = 765;
|
|
|
|
|
break; // DIN A5 landscape
|
|
|
|
|
case 2:
|
|
|
|
|
xall = 765;
|
|
|
|
|
yall = 1020;
|
|
|
|
|
break; // DIN A5 portrait
|
|
|
|
|
case 3:
|
|
|
|
|
xall = 1530;
|
|
|
|
|
yall = 1020;
|
|
|
|
|
break; // DIN A4 landscape
|
|
|
|
|
case 4:
|
|
|
|
|
xall = 1020;
|
|
|
|
|
yall = 1530;
|
|
|
|
|
break; // DIN A4 portrait
|
|
|
|
|
case 5:
|
|
|
|
|
xall = 2295;
|
|
|
|
|
yall = 1530;
|
|
|
|
|
break; // DIN A3 landscape
|
|
|
|
|
case 6:
|
|
|
|
|
xall = 1530;
|
|
|
|
|
yall = 2295;
|
|
|
|
|
break; // DIN A3 portrait
|
|
|
|
|
case 7:
|
|
|
|
|
xall = 1414;
|
|
|
|
|
yall = 1054;
|
|
|
|
|
break; // letter landscape
|
|
|
|
|
case 8:
|
|
|
|
|
xall = 1054;
|
|
|
|
|
yall = 1414;
|
|
|
|
|
break; // letter portrait
|
|
|
|
|
default:
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
2008-10-25 16:44:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-05-18 16:28:29 +03:00
|
|
|
|
void Schematic::paintFrame(QPainter* painter) {
|
|
|
|
|
// dimensions: X cm / 2.54 * 144
|
|
|
|
|
int frame_width, frame_height;
|
|
|
|
|
if (!sizeOfFrame(frame_width, frame_height))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
painter->save();
|
|
|
|
|
painter->setPen(QPen(Qt::darkGray, 1));
|
|
|
|
|
|
|
|
|
|
// Width of stripe along frame border in column and row labels are placed
|
|
|
|
|
const int frame_margin = painter->fontMetrics().lineSpacing() + 4;
|
|
|
|
|
|
|
|
|
|
// Outer rect
|
|
|
|
|
painter->drawRect(0, 0, frame_width, frame_height);
|
|
|
|
|
// a bit smaller than outer rect
|
|
|
|
|
painter->drawRect(frame_margin, frame_margin, frame_width - 2 * frame_margin, frame_height - 2 * frame_margin);
|
|
|
|
|
|
|
|
|
|
// Column labels
|
|
|
|
|
{
|
|
|
|
|
const int h_step = frame_width / ((frame_width + 127) / 255);
|
|
|
|
|
uint column_number = 1;
|
|
|
|
|
|
|
|
|
|
for (int x = h_step; x <= frame_width; x += h_step) {
|
|
|
|
|
painter->drawLine(x, 0, x, frame_margin);
|
|
|
|
|
painter->drawLine(x, frame_height - frame_margin, x, frame_height);
|
|
|
|
|
|
|
|
|
|
auto cn = QString::number(column_number);
|
|
|
|
|
auto tx = x - h_step / 2 + 5;
|
2024-10-15 21:18:07 -04:00
|
|
|
|
painter->drawText(tx, 3, 1, 1, Qt::TextDontClip, cn);
|
|
|
|
|
painter->drawText(tx, frame_height - frame_margin + 3, 1, 1, Qt::TextDontClip, cn);
|
2024-05-18 16:28:29 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
column_number++;
|
2024-05-18 16:28:29 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Row labels
|
|
|
|
|
{
|
|
|
|
|
const int v_step = frame_height / ((frame_height + 127) / 255);
|
|
|
|
|
char row_letter = 'A';
|
|
|
|
|
|
|
|
|
|
for (int y = v_step; y <= frame_height; y += v_step) {
|
|
|
|
|
painter->drawLine(0, y, frame_margin, y);
|
|
|
|
|
painter->drawLine(frame_width - frame_margin, y, frame_width, y);
|
|
|
|
|
|
|
|
|
|
auto rl = QString::fromLatin1(&row_letter, 1);
|
|
|
|
|
auto ty = y - v_step/2 + 5;
|
|
|
|
|
painter->drawText(5, ty, rl);
|
|
|
|
|
painter->drawText(frame_width - frame_margin + 5, ty, rl);
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
row_letter++;
|
2024-05-18 16:28:29 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// draw text box with text
|
|
|
|
|
int x1_ = frame_width - 340 - frame_margin;
|
|
|
|
|
int y1_ = frame_height - 3 - frame_margin;
|
|
|
|
|
int x2_ = frame_width - frame_margin - 3;
|
|
|
|
|
int y2_ = frame_height - frame_margin - 3;
|
|
|
|
|
|
|
|
|
|
const int d = 6;
|
|
|
|
|
const double z = 200.0;
|
|
|
|
|
y1_ -= painter->fontMetrics().lineSpacing() + d;
|
|
|
|
|
painter->drawLine(x1_, y1_, x2_, y1_);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
painter->drawText(x1_ + d, y1_ + (d >> 1), 1, 1, Qt::TextDontClip, a_Frame_Text2);
|
2024-05-18 16:28:29 +03:00
|
|
|
|
painter->drawLine(x1_ + z, y1_, x1_ + z, y1_ + painter->fontMetrics().lineSpacing() + d);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
painter->drawText(x1_ + d + z, y1_ + (d >> 1), 1, 1, Qt::TextDontClip, a_Frame_Text3);
|
2024-05-18 16:28:29 +03:00
|
|
|
|
y1_ -= painter->fontMetrics().lineSpacing() + d;
|
|
|
|
|
painter->drawLine(x1_, y1_, x2_, y1_);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
painter->drawText(x1_ + d, y1_ + (d >> 1), 1, 1, Qt::TextDontClip, a_Frame_Text1);
|
|
|
|
|
y1_ -= (a_Frame_Text0.count('\n') + 1) * painter->fontMetrics().lineSpacing() + d;
|
2024-05-18 16:28:29 +03:00
|
|
|
|
painter->drawRect(x2_, y2_, x1_ - x2_ - 1, y1_ - y2_ - 1);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
painter->drawText(x1_ + d, y1_ + (d >> 1), 1, 1, Qt::TextDontClip, a_Frame_Text0);
|
2024-05-18 16:28:29 +03:00
|
|
|
|
|
|
|
|
|
painter->restore();
|
|
|
|
|
}
|
|
|
|
|
|
2006-03-28 06:10:52 +00:00
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
// Is called when the content (schematic or data display) has to be drawn.
|
|
|
|
|
void Schematic::drawContents(QPainter *p, int, int, int, int)
|
|
|
|
|
{
|
2024-05-18 16:39:19 +03:00
|
|
|
|
QTransform trf{p->transform()};
|
|
|
|
|
trf
|
2024-11-08 12:46:57 +01:00
|
|
|
|
.scale(a_Scale, a_Scale)
|
|
|
|
|
.translate(-a_ViewX1, -a_ViewY1);
|
2024-05-18 16:39:19 +03:00
|
|
|
|
p->setTransform(trf);
|
|
|
|
|
|
|
|
|
|
auto renderHints = p->renderHints();
|
|
|
|
|
renderHints
|
|
|
|
|
.setFlag(QPainter::Antialiasing)
|
|
|
|
|
.setFlag(QPainter::TextAntialiasing)
|
|
|
|
|
.setFlag(QPainter::SmoothPixmapTransform);
|
|
|
|
|
p->setRenderHints(renderHints);
|
|
|
|
|
|
|
|
|
|
p->setFont(QucsSettings.font);
|
|
|
|
|
drawGrid(p);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (!a_symbolMode)
|
2024-05-18 16:39:19 +03:00
|
|
|
|
paintFrame(p);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-05-18 16:39:19 +03:00
|
|
|
|
drawElements(p);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_showBias > 0) {
|
2024-05-23 11:05:23 +03:00
|
|
|
|
drawDcBiasPoints(p);
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-18 16:39:19 +03:00
|
|
|
|
drawPostPaintEvents(p);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-05-18 16:27:04 +03:00
|
|
|
|
void Schematic::drawElements(QPainter* painter) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* component : *a_Components) {
|
2024-05-18 16:27:04 +03:00
|
|
|
|
component->paint(painter);
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* wire : *a_Wires) {
|
2024-05-18 16:27:04 +03:00
|
|
|
|
wire->paint(painter);
|
|
|
|
|
if (wire->Label) {
|
|
|
|
|
wire->Label->paint(painter); // separate because of paintSelected
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* node : *a_Nodes) {
|
2024-05-18 16:27:04 +03:00
|
|
|
|
node->paint(painter);
|
|
|
|
|
if (node->Label) {
|
|
|
|
|
node->Label->paint(painter); // separate because of paintSelected
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* diagram : *a_Diagrams) {
|
2024-05-18 16:27:04 +03:00
|
|
|
|
diagram->paint(painter);
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* painting : *a_Paintings) {
|
2024-05-18 16:27:04 +03:00
|
|
|
|
painting->paint(painter);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Schematic::drawDcBiasPoints(QPainter* painter) {
|
|
|
|
|
painter->save();
|
|
|
|
|
int x, y, z;
|
2024-12-11 18:53:56 +03:00
|
|
|
|
|
|
|
|
|
const int xOffset = 10;
|
|
|
|
|
const int yOffset = 10;
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* pn : *a_Nodes) {
|
2024-05-18 16:27:04 +03:00
|
|
|
|
if (pn->Name.isEmpty())
|
|
|
|
|
continue;
|
2024-12-11 18:53:56 +03:00
|
|
|
|
|
|
|
|
|
QString value = misc::formatValue(pn->Name, 4);
|
|
|
|
|
|
2024-05-18 16:27:04 +03:00
|
|
|
|
x = pn->cx;
|
|
|
|
|
y = pn->cy + 4;
|
|
|
|
|
z = pn->x1;
|
2024-12-11 18:53:56 +03:00
|
|
|
|
|
|
|
|
|
QRect textRect = painter->fontMetrics().boundingRect(value);
|
|
|
|
|
int rectWidth = textRect.width() + 6;
|
|
|
|
|
int rectHeight = textRect.height() + 4;
|
|
|
|
|
|
|
|
|
|
if (z & 0x10) {
|
|
|
|
|
x += xOffset;
|
|
|
|
|
y -= yOffset;
|
|
|
|
|
} else {
|
|
|
|
|
x -= xOffset;
|
|
|
|
|
y -= yOffset;
|
2024-05-18 16:27:04 +03:00
|
|
|
|
}
|
2024-12-11 18:53:56 +03:00
|
|
|
|
|
|
|
|
|
int rectX = x - rectWidth / 2;
|
|
|
|
|
int rectY = y - rectHeight / 2;
|
|
|
|
|
|
|
|
|
|
painter->setBrush(QBrush(QColor(230,230,230)));
|
|
|
|
|
painter->setPen(Qt::NoPen);
|
|
|
|
|
painter->drawRoundedRect( QRectF(rectX, rectY, rectWidth, rectHeight),15,15,Qt::RelativeSize);
|
|
|
|
|
|
|
|
|
|
painter->setPen(z & 0x10 ? Qt::darkGreen : Qt::blue);
|
|
|
|
|
painter->drawText(x - textRect.width() / 2, y + textRect.height() / 4, value);
|
2024-05-18 16:27:04 +03:00
|
|
|
|
}
|
|
|
|
|
painter->restore();
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-11 18:53:56 +03:00
|
|
|
|
|
|
|
|
|
|
2024-05-18 16:27:04 +03:00
|
|
|
|
void Schematic::drawPostPaintEvents(QPainter* painter) {
|
|
|
|
|
painter->save();
|
|
|
|
|
/*
|
|
|
|
|
* The following events used to be drawn from mouseactions.cpp, but since Qt4
|
|
|
|
|
* Paint actions can only be called from within the paint event, so they
|
|
|
|
|
* are put into a QList (PostedPaintEvents) and processed here
|
|
|
|
|
*/
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto p : a_PostedPaintEvents) {
|
2024-05-18 16:27:04 +03:00
|
|
|
|
// QPainter painter2(viewport()); for if(p.PaintOnViewport)
|
|
|
|
|
QPen pen(Qt::black);
|
|
|
|
|
painter->setPen(Qt::black);
|
|
|
|
|
switch (p.pe) {
|
|
|
|
|
case _NotRop:
|
|
|
|
|
painter->setCompositionMode(QPainter::RasterOp_SourceAndNotDestination);
|
|
|
|
|
break;
|
|
|
|
|
case _Rect:
|
|
|
|
|
painter->drawRect(p.x1, p.y1, p.x2, p.y2);
|
|
|
|
|
break;
|
|
|
|
|
case _SelectionRect:
|
2024-06-08 10:19:43 +02:00
|
|
|
|
pen.setCosmetic(true);
|
2024-05-18 16:27:04 +03:00
|
|
|
|
pen.setStyle(Qt::DashLine);
|
|
|
|
|
pen.setColor(QColor(50, 50, 50, 100));
|
|
|
|
|
painter->setPen(pen);
|
|
|
|
|
painter->fillRect(p.x1, p.y1, p.x2, p.y2, QColor(200, 220, 240, 100));
|
|
|
|
|
painter->drawRect(p.x1, p.y1, p.x2, p.y2);
|
|
|
|
|
break;
|
|
|
|
|
case _Line:
|
|
|
|
|
painter->drawLine(p.x1, p.y1, p.x2, p.y2);
|
|
|
|
|
break;
|
|
|
|
|
case _Ellipse:
|
|
|
|
|
painter->drawEllipse(p.x1, p.y1, p.x2, p.y2);
|
|
|
|
|
break;
|
|
|
|
|
case _Arc:
|
|
|
|
|
painter->drawArc(p.x1, p.y1, p.x2, p.y2, p.a, p.b);
|
|
|
|
|
break;
|
|
|
|
|
case _DotLine:
|
|
|
|
|
painter->setPen(Qt::DotLine);
|
|
|
|
|
painter->drawLine(p.x1, p.y1, p.x2, p.y2);
|
|
|
|
|
break;
|
|
|
|
|
case _DotRect:
|
|
|
|
|
painter->setPen(Qt::DotLine);
|
|
|
|
|
painter->drawRect(p.x1, p.y1, p.x2, p.y2);
|
|
|
|
|
break;
|
|
|
|
|
case _Translate:; //painter2.translate(p.x1, p.y1);
|
|
|
|
|
case _Scale:; //painter2.scale(p.x1,p.y1);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_PostedPaintEvents.clear();
|
2024-05-18 16:27:04 +03:00
|
|
|
|
painter->restore();
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
void Schematic::PostPaintEvent(
|
|
|
|
|
PE pe, int x1, int y1, int x2, int y2, int a, int b, bool PaintOnViewport)
|
2012-12-03 17:17:01 +01:00
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
PostedPaintEvent p = {pe, x1, y1, x2, y2, a, b, PaintOnViewport};
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_PostedPaintEvents.push_back(p);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
viewport()->update();
|
|
|
|
|
update();
|
2012-12-03 17:17:01 +01:00
|
|
|
|
}
|
|
|
|
|
|
2006-03-28 06:10:52 +00:00
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
void Schematic::contentsMouseMoveEvent(QMouseEvent *Event)
|
|
|
|
|
{
|
2024-02-25 22:29:16 +03:00
|
|
|
|
const QPoint modelPos = contentsToModel(Event->pos());
|
|
|
|
|
auto xpos = modelPos.x();
|
|
|
|
|
auto ypos = modelPos.y();
|
2024-01-24 11:34:49 +03:00
|
|
|
|
QString text = "";
|
|
|
|
|
|
2024-01-24 23:06:49 +03:00
|
|
|
|
auto doubleToString = [](bool condition, double number) {
|
|
|
|
|
return condition ? misc::num2str(number) : misc::StringNiceNum(number);
|
|
|
|
|
};
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_Diagrams == nullptr) return; // fix for crash on document closing; appears time to time
|
2024-02-04 14:29:36 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (Diagram* diagram = a_Diagrams->last(); diagram != nullptr; diagram = a_Diagrams->prev()) {
|
2024-01-24 11:34:49 +03:00
|
|
|
|
// BUG: Obtaining the diagram type by name is marked as a bug elsewhere (to be solved separately).
|
|
|
|
|
// TODO: Currently only rectangular diagrams are supported.
|
|
|
|
|
if (diagram->getSelected(xpos, ypos) && diagram->Name == "Rect") {
|
|
|
|
|
bool hasY1, hasY2 = false;
|
|
|
|
|
for (auto graph: diagram->Graphs) {
|
|
|
|
|
hasY1 |= graph->yAxisNo == 0;
|
|
|
|
|
hasY2 |= graph->yAxisNo == 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QPointF mouseClickPoint = QPointF(xpos - diagram->cx, diagram->cy - ypos);
|
|
|
|
|
MappedPoint mp = diagram->pointToValue(mouseClickPoint);
|
2024-01-25 10:32:02 +03:00
|
|
|
|
|
2024-01-24 23:06:49 +03:00
|
|
|
|
auto _x = doubleToString(diagram->engineeringNotation, mp.x);
|
|
|
|
|
text = "X=" + _x;
|
2024-01-24 11:34:49 +03:00
|
|
|
|
if (hasY1) {
|
|
|
|
|
text.append("; Y1=");
|
2024-01-24 23:06:49 +03:00
|
|
|
|
auto _y1 = doubleToString(diagram->engineeringNotation, mp.y1);
|
|
|
|
|
text.append(_y1);
|
2024-01-24 11:34:49 +03:00
|
|
|
|
}
|
|
|
|
|
if (hasY2) {
|
|
|
|
|
text.append("; Y2=");
|
2024-01-24 23:06:49 +03:00
|
|
|
|
auto _y2 = doubleToString(diagram->engineeringNotation, mp.y2);
|
|
|
|
|
text.append(_y2);
|
2024-01-24 11:34:49 +03:00
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-25 09:49:52 +03:00
|
|
|
|
emit signalCursorPosChanged(xpos, ypos, text);
|
2024-01-11 00:41:57 +03:00
|
|
|
|
|
|
|
|
|
// Perform "pan with mouse"
|
|
|
|
|
if (Event->buttons() & Qt::MiddleButton) {
|
|
|
|
|
const QPoint currentCursorPosition = contentsToViewport(Event->pos());
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
const int dx = currentCursorPosition.x() - a_previousCursorPosition.x();
|
2024-01-31 16:31:39 +03:00
|
|
|
|
if (dx < 0) {
|
2024-01-11 00:41:57 +03:00
|
|
|
|
scrollRight(std::abs(dx));
|
|
|
|
|
} else if (dx > 0) {
|
|
|
|
|
scrollLeft(dx);
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
const int dy = currentCursorPosition.y() - a_previousCursorPosition.y();
|
2024-01-31 16:31:39 +03:00
|
|
|
|
if (dy < 0) {
|
2024-01-11 00:41:57 +03:00
|
|
|
|
scrollDown(std::abs(dy));
|
|
|
|
|
} else if (dy > 0) {
|
|
|
|
|
scrollUp(dy);
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_previousCursorPosition = currentCursorPosition;
|
2024-01-11 00:41:57 +03:00
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_App->MouseMoveAction)
|
|
|
|
|
(a_App->view->*(a_App->MouseMoveAction))(this, Event);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
void Schematic::contentsMousePressEvent(QMouseEvent *Event)
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->editText->setHidden(true); // disable text edit of component property
|
2023-12-11 15:13:33 +03:00
|
|
|
|
this->setFocus();
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_App->MouseReleaseAction == &MouseActions::MReleasePaste)
|
2007-02-08 07:04:54 +00:00
|
|
|
|
return;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-02-25 22:29:16 +03:00
|
|
|
|
const QPoint inModel = contentsToModel(Event->pos());
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-01-11 00:41:57 +03:00
|
|
|
|
if (Event->button() == Qt::RightButton)
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_App->MousePressAction != &MouseActions::MPressElement)
|
|
|
|
|
if (a_App->MousePressAction != &MouseActions::MPressWire2) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// show menu on right mouse button
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->view->rightPressMenu(this, Event, inModel.x(), inModel.y());
|
|
|
|
|
if (a_App->MouseReleaseAction)
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// Is not called automatically because menu has focus.
|
2024-11-08 12:46:57 +01:00
|
|
|
|
(a_App->view->*(a_App->MouseReleaseAction))(this, Event);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-11 00:41:57 +03:00
|
|
|
|
// Begin "pan with mouse" action. Panning starts if *only*
|
|
|
|
|
// the middle button is pressed.
|
|
|
|
|
if (Event->button() == Qt::MiddleButton) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_previousCursorPosition = contentsToViewport(Event->pos());
|
2024-01-11 00:41:57 +03:00
|
|
|
|
setCursor(Qt::ClosedHandCursor);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_App->MousePressAction)
|
|
|
|
|
(a_App->view->*(a_App->MousePressAction))(this, Event, inModel.x(), inModel.y());
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
void Schematic::contentsMouseReleaseEvent(QMouseEvent *Event)
|
|
|
|
|
{
|
2024-01-11 00:41:57 +03:00
|
|
|
|
// End "pan with mouse" action.
|
|
|
|
|
if (Event->button() == Qt::MiddleButton) {
|
|
|
|
|
unsetCursor();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_App->MouseReleaseAction)
|
|
|
|
|
(a_App->view->*(a_App->MouseReleaseAction))(this, Event);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
void Schematic::contentsMouseDoubleClickEvent(QMouseEvent *Event)
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_App->MouseDoubleClickAction)
|
|
|
|
|
(a_App->view->*(a_App->MouseDoubleClickAction))(this, Event);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-05-18 16:44:27 +03:00
|
|
|
|
void Schematic::print(QPrinter*, QPainter* painter, bool printAll,
|
|
|
|
|
bool fitToPage, QMargins margins) {
|
|
|
|
|
painter->save();
|
|
|
|
|
|
|
|
|
|
const QRectF pageSize{0, 0, static_cast<double>(painter->device()->width()),
|
|
|
|
|
static_cast<double>(painter->device()->height())};
|
|
|
|
|
|
|
|
|
|
QRect printedArea = printAll ? allBoundingRect() : sizeOfSelection();
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (printAll && a_showFrame) {
|
2024-05-18 16:44:27 +03:00
|
|
|
|
int frame_width, frame_height;
|
|
|
|
|
sizeOfFrame(frame_width, frame_height);
|
|
|
|
|
printedArea |= QRect{0, 0, frame_width, frame_height};
|
|
|
|
|
}
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-05-18 16:44:27 +03:00
|
|
|
|
printedArea = printedArea.marginsAdded(margins);
|
|
|
|
|
|
|
|
|
|
double scale = 1.0;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (fitToPage) {
|
2024-05-18 16:44:27 +03:00
|
|
|
|
scale = std::min(pageSize.width() / printedArea.width(),
|
|
|
|
|
pageSize.height() / printedArea.height());
|
|
|
|
|
} else {
|
|
|
|
|
QFontInfo printerFontInfo{QFont{QucsSettings.font, painter->device()}};
|
|
|
|
|
QFontInfo schematicFontInfo{QucsSettings.font};
|
|
|
|
|
|
|
|
|
|
scale = static_cast<double>(printerFontInfo.pixelSize()) /
|
|
|
|
|
static_cast<double>(schematicFontInfo.pixelSize());
|
|
|
|
|
}
|
|
|
|
|
painter->scale(scale, scale);
|
|
|
|
|
|
|
|
|
|
painter->translate(-printedArea.left(), -printedArea.top());
|
|
|
|
|
|
|
|
|
|
// put picture in center
|
|
|
|
|
{
|
|
|
|
|
auto w = pageSize.width() / scale;
|
|
|
|
|
if (printedArea.width() <= w) {
|
|
|
|
|
auto d = (w - printedArea.width()) / 2;
|
|
|
|
|
painter->translate(d, 0);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
|
|
|
|
|
2024-05-18 16:44:27 +03:00
|
|
|
|
auto h = pageSize.height() / scale;
|
|
|
|
|
if (printedArea.height() <= h) {
|
|
|
|
|
auto d = (h - printedArea.height()) / 2;
|
|
|
|
|
painter->translate(0, d);
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-07-17 06:02:57 +00:00
|
|
|
|
|
2024-05-18 16:44:27 +03:00
|
|
|
|
// User chose a font with size in points while looking at the font
|
|
|
|
|
// on screen. The same font size in *points* equals to different
|
|
|
|
|
// amount of pixels when shown on screen or printed on paper,
|
|
|
|
|
// because underlying painting devices have different resolutions.
|
|
|
|
|
// To preserve the ratio of text sizes and elements' sizes
|
|
|
|
|
// font size has to be set in pixels here. This makes lines, squares,
|
|
|
|
|
// circles, etc. and size of font be measured in same units and scale
|
|
|
|
|
// them equally.
|
|
|
|
|
auto f = QucsSettings.font;
|
|
|
|
|
QFontInfo fi{f};
|
|
|
|
|
f.setPixelSize(fi.pixelSize());
|
|
|
|
|
painter->setFont(f);
|
|
|
|
|
|
|
|
|
|
paintSchToViewpainter(painter, printAll);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-05-18 16:44:27 +03:00
|
|
|
|
painter->restore();
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-05-18 16:44:27 +03:00
|
|
|
|
namespace {
|
|
|
|
|
// helper to be used in Schematic::paintSchToViewpainter
|
|
|
|
|
template <typename T> void draw_preserve_selection(T* elem, QPainter* p) {
|
|
|
|
|
bool selected = elem->isSelected;
|
|
|
|
|
elem->isSelected = false;
|
|
|
|
|
elem->paint(p);
|
|
|
|
|
elem->isSelected = selected;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
2024-05-18 16:44:27 +03:00
|
|
|
|
} // namespace
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-05-18 16:44:27 +03:00
|
|
|
|
void Schematic::paintSchToViewpainter(QPainter* painter, bool printAll) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (printAll && a_showFrame && !a_symbolMode) {
|
2024-05-18 16:44:27 +03:00
|
|
|
|
paintFrame(painter);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const auto should_draw = [=](Element* drawable) {
|
|
|
|
|
return printAll || drawable->isSelected;
|
|
|
|
|
};
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* component : *a_Components) {
|
2024-05-18 16:44:27 +03:00
|
|
|
|
if (should_draw(component)) {
|
|
|
|
|
draw_preserve_selection(component, painter);
|
2013-12-09 20:10:30 +04:00
|
|
|
|
}
|
2024-05-18 16:44:27 +03:00
|
|
|
|
}
|
2013-12-09 20:10:30 +04:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* wire : *a_Wires) {
|
2024-05-18 16:44:27 +03:00
|
|
|
|
if (should_draw(wire)) {
|
|
|
|
|
draw_preserve_selection(wire, painter);
|
2013-12-09 20:10:30 +04:00
|
|
|
|
}
|
2024-05-18 16:44:27 +03:00
|
|
|
|
|
|
|
|
|
if (auto* label = wire->Label) {
|
|
|
|
|
if (should_draw(label)) {
|
|
|
|
|
draw_preserve_selection(label, painter);
|
2023-01-15 01:17:09 +03:00
|
|
|
|
}
|
2024-05-18 16:44:27 +03:00
|
|
|
|
}
|
2013-12-09 20:10:30 +04:00
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* node : *a_Nodes) {
|
2024-07-16 18:56:55 +02:00
|
|
|
|
for (auto* connected : *node) {
|
2024-05-18 16:44:27 +03:00
|
|
|
|
if (should_draw(connected)) {
|
|
|
|
|
draw_preserve_selection(node, painter);
|
2023-01-15 01:17:09 +03:00
|
|
|
|
break;
|
|
|
|
|
}
|
2024-05-18 16:44:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (auto* label = node->Label) {
|
|
|
|
|
if (should_draw(label)) {
|
|
|
|
|
draw_preserve_selection(label, painter);
|
2023-01-15 01:17:09 +03:00
|
|
|
|
}
|
2024-05-18 16:44:27 +03:00
|
|
|
|
}
|
2013-12-09 20:10:30 +04:00
|
|
|
|
}
|
2013-12-11 15:12:44 +04:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* painting : *a_Paintings) {
|
2024-05-18 16:44:27 +03:00
|
|
|
|
if (should_draw(painting)) {
|
|
|
|
|
draw_preserve_selection(painting, painter);
|
2013-12-14 14:44:02 +04:00
|
|
|
|
}
|
2024-05-18 16:44:27 +03:00
|
|
|
|
}
|
2013-12-14 14:44:02 +04:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* diagram : *a_Diagrams) {
|
2024-05-18 16:44:27 +03:00
|
|
|
|
if (!should_draw(diagram)) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// if graph or marker is selected, deselect during printing
|
|
|
|
|
for (Graph* pg : diagram->Graphs) {
|
|
|
|
|
if (pg->isSelected) {
|
|
|
|
|
pg->Type |= 1; // remember selection
|
|
|
|
|
}
|
|
|
|
|
pg->isSelected = false;
|
|
|
|
|
for (Marker* pm : pg->Markers) {
|
|
|
|
|
if (pm->isSelected) {
|
|
|
|
|
pm->Type |= 1; // remember selection
|
2023-01-15 01:17:09 +03:00
|
|
|
|
}
|
2024-05-18 16:44:27 +03:00
|
|
|
|
pm->isSelected = false;
|
2023-01-15 01:17:09 +03:00
|
|
|
|
}
|
2024-05-18 16:44:27 +03:00
|
|
|
|
}
|
|
|
|
|
draw_preserve_selection(diagram, painter);
|
2023-01-15 01:17:09 +03:00
|
|
|
|
|
2024-05-18 16:44:27 +03:00
|
|
|
|
// revert selection of graphs and markers
|
|
|
|
|
for (Graph* pg : diagram->Graphs) {
|
|
|
|
|
if (pg->Type & 1) {
|
|
|
|
|
pg->isSelected = true;
|
|
|
|
|
}
|
|
|
|
|
pg->Type &= -2;
|
|
|
|
|
for (Marker* pm : pg->Markers) {
|
|
|
|
|
if (pm->Type & 1) {
|
|
|
|
|
pm->isSelected = true;
|
2023-01-15 01:17:09 +03:00
|
|
|
|
}
|
2024-05-18 16:44:27 +03:00
|
|
|
|
pm->Type &= -2;
|
2023-01-15 01:17:09 +03:00
|
|
|
|
}
|
2013-12-14 14:44:02 +04:00
|
|
|
|
}
|
2024-05-18 16:44:27 +03:00
|
|
|
|
}
|
2014-11-06 17:24:33 +04:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_showBias > 0) { // show DC bias points in schematic ?
|
2024-05-18 16:44:27 +03:00
|
|
|
|
drawDcBiasPoints(painter);
|
2014-11-06 17:24:33 +04:00
|
|
|
|
}
|
2013-12-09 20:10:30 +04:00
|
|
|
|
}
|
|
|
|
|
|
2024-01-09 21:07:31 +03:00
|
|
|
|
void Schematic::zoomAroundPoint(double offeredScaleChange, QPoint coords, bool viewportRelative=true)
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
const double desiredScale = a_Scale * offeredScaleChange;
|
2024-01-25 18:14:45 +03:00
|
|
|
|
const auto viewportCoords =
|
|
|
|
|
viewportRelative ? coords : coords - QPoint{contentsX(), contentsY()};
|
|
|
|
|
const auto focusPoint = viewportToModel(viewportCoords);
|
|
|
|
|
const auto model = includePoint(modelRect(), focusPoint);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-01-25 18:14:45 +03:00
|
|
|
|
renderModel(desiredScale, model, focusPoint, viewportCoords);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2006-11-06 06:58:05 +00:00
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
float Schematic::zoomBy(float s)
|
|
|
|
|
{
|
2024-01-05 21:14:23 +03:00
|
|
|
|
// Change scale and keep the point displayed in the center
|
|
|
|
|
// of the viewport at same place after scaling.
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
const double newScale = a_Scale * s;
|
2024-01-05 21:14:23 +03:00
|
|
|
|
const auto vpCenter = viewportRect().center();
|
2024-01-25 18:14:45 +03:00
|
|
|
|
const auto centerPoint = viewportToModel(vpCenter);
|
|
|
|
|
const auto model = includePoint(modelRect(), centerPoint);
|
|
|
|
|
|
|
|
|
|
return renderModel(newScale, model, centerPoint, vpCenter);
|
2006-11-06 06:58:05 +00:00
|
|
|
|
}
|
|
|
|
|
|
2006-04-18 06:03:52 +00:00
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
void Schematic::showAll()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
sizeOfAll(a_UsedX1, a_UsedY1, a_UsedX2, a_UsedY2);
|
|
|
|
|
if (a_UsedX1 == 0)
|
|
|
|
|
if (a_UsedX2 == 0)
|
|
|
|
|
if (a_UsedY1 == 0)
|
|
|
|
|
if (a_UsedY2 == 0) {
|
|
|
|
|
a_UsedX1 = a_UsedY1 = INT_MAX;
|
|
|
|
|
a_UsedX2 = a_UsedY2 = INT_MIN;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-05 16:15:57 +03:00
|
|
|
|
// Reshape model plane to cut off unused parts
|
|
|
|
|
constexpr int margin = 40;
|
|
|
|
|
QRect newModelBounds = modelRect();
|
2024-11-08 12:46:57 +01:00
|
|
|
|
newModelBounds.setLeft(a_UsedX1 - margin);
|
|
|
|
|
newModelBounds.setTop(a_UsedY1 - margin);
|
|
|
|
|
newModelBounds.setRight(a_UsedX2 + margin);
|
|
|
|
|
newModelBounds.setBottom(a_UsedY2 + margin);
|
2024-01-05 16:15:57 +03:00
|
|
|
|
|
|
|
|
|
// The shape of the model plane may not fit the shape of the viewport,
|
|
|
|
|
// so we looking for a scale value which enables to fit the whole model
|
|
|
|
|
// into the viewport
|
|
|
|
|
const double xScale = static_cast<double>(viewport()->width()) /
|
|
|
|
|
static_cast<double>(newModelBounds.width());
|
|
|
|
|
const double yScale = static_cast<double>(viewport()->height()) /
|
|
|
|
|
static_cast<double>(newModelBounds.height());
|
|
|
|
|
const double newScale = std::min(xScale, yScale);
|
|
|
|
|
|
|
|
|
|
renderModel(newScale, newModelBounds, newModelBounds.center(), viewportRect().center());
|
2006-04-18 06:03:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-10-17 21:29:39 +03:00
|
|
|
|
// ------------------------------------------------------
|
2024-01-05 16:40:43 +03:00
|
|
|
|
void Schematic::zoomToSelection() {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
sizeOfAll(a_UsedX1, a_UsedY1, a_UsedX2, a_UsedY2);
|
|
|
|
|
if (a_UsedX1 == 0)
|
|
|
|
|
if (a_UsedX2 == 0)
|
|
|
|
|
if (a_UsedY1 == 0)
|
|
|
|
|
if (a_UsedY2 == 0) {
|
|
|
|
|
a_UsedX1 = a_UsedY1 = INT_MAX;
|
|
|
|
|
a_UsedX2 = a_UsedY2 = INT_MIN;
|
2024-01-05 16:40:43 +03:00
|
|
|
|
|
|
|
|
|
// No elements present – nothing can be selected; quit
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-08 13:33:55 +03:00
|
|
|
|
const QRect selectedBoundingRect{ sizeOfSelection() };
|
2024-01-05 16:40:43 +03:00
|
|
|
|
|
|
|
|
|
// Working with raw coordinates is clumsy, abstract them out
|
2024-11-08 12:46:57 +01:00
|
|
|
|
const QRect usedBoundingRect{a_UsedX1, a_UsedY1, a_UsedX2 - a_UsedX1, a_UsedY2 - a_UsedY1};
|
2024-01-05 16:40:43 +03:00
|
|
|
|
|
|
|
|
|
if (selectedBoundingRect.width() == 0 || selectedBoundingRect.height() == 0) {
|
|
|
|
|
// If nothing is selected, then what should be shown? Probably it's best
|
|
|
|
|
// to do nothing.
|
2023-10-20 01:00:46 +03:00
|
|
|
|
return;
|
2023-10-19 15:53:15 +03:00
|
|
|
|
}
|
2023-10-17 21:29:39 +03:00
|
|
|
|
|
2024-01-05 16:40:43 +03:00
|
|
|
|
// While we here, lets reshape model plane to cut off unused parts
|
|
|
|
|
constexpr int margin = 40;
|
|
|
|
|
QRect modelBounds = modelRect();
|
|
|
|
|
modelBounds.setLeft(usedBoundingRect.left() - margin);
|
|
|
|
|
modelBounds.setTop(usedBoundingRect.top() - margin);
|
|
|
|
|
modelBounds.setRight(usedBoundingRect.right() + margin);
|
|
|
|
|
modelBounds.setBottom(usedBoundingRect.bottom() + margin);
|
2023-10-20 01:00:46 +03:00
|
|
|
|
|
2024-01-05 16:40:43 +03:00
|
|
|
|
// Find out the scale at which selected area's longest side would fit
|
|
|
|
|
// into the viewport
|
|
|
|
|
const double xScale = static_cast<double>(viewport()->width()) /
|
|
|
|
|
static_cast<double>(selectedBoundingRect.width());
|
|
|
|
|
const double yScale = static_cast<double>(viewport()->height()) /
|
|
|
|
|
static_cast<double>(selectedBoundingRect.height());
|
|
|
|
|
const double newScale = std::min(xScale, yScale);
|
2023-10-20 01:00:46 +03:00
|
|
|
|
|
2024-01-05 16:40:43 +03:00
|
|
|
|
renderModel(newScale, modelBounds, selectedBoundingRect.center(), viewportRect().center());
|
2023-10-17 21:29:39 +03:00
|
|
|
|
}
|
|
|
|
|
|
2006-04-18 06:03:52 +00:00
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
void Schematic::showNoZoom()
|
|
|
|
|
{
|
2024-01-05 20:40:57 +03:00
|
|
|
|
constexpr double noScale = 1.0;
|
|
|
|
|
const QPoint vpCenter = viewportRect().center();
|
|
|
|
|
const QPoint displayedInCenter = viewportToModel(vpCenter);
|
2006-04-18 06:03:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
sizeOfAll(a_UsedX1, a_UsedY1, a_UsedX2, a_UsedY2);
|
|
|
|
|
if (a_UsedX1 == 0)
|
|
|
|
|
if (a_UsedX2 == 0)
|
|
|
|
|
if (a_UsedY1 == 0)
|
|
|
|
|
if (a_UsedY2 == 0) {
|
|
|
|
|
a_UsedX1 = a_UsedY1 = INT_MAX;
|
|
|
|
|
a_UsedX2 = a_UsedY2 = INT_MIN;
|
2024-01-05 20:40:57 +03:00
|
|
|
|
// If there is no elements in schematic, then just set scale 1.0
|
|
|
|
|
// at the place we currently in.
|
2024-01-25 18:14:45 +03:00
|
|
|
|
renderModel(noScale, includePoint(modelRect(), displayedInCenter), displayedInCenter, vpCenter);
|
2024-01-05 20:40:57 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-01-05 20:40:57 +03:00
|
|
|
|
// Working with raw coordinates is clumsy. Wrap them in useful abstraction.
|
2024-11-08 12:46:57 +01:00
|
|
|
|
const QRect usedBoundingRect{a_UsedX1, a_UsedY1, a_UsedX2 - a_UsedX1, a_UsedY2 - a_UsedY1};
|
2024-01-05 20:40:57 +03:00
|
|
|
|
|
|
|
|
|
// Trim unused model space
|
|
|
|
|
constexpr int margin = 40;
|
|
|
|
|
QRect newModelBounds = modelRect();
|
|
|
|
|
newModelBounds.setLeft(usedBoundingRect.left() - margin);
|
|
|
|
|
newModelBounds.setTop(usedBoundingRect.top() - margin);
|
|
|
|
|
newModelBounds.setRight(usedBoundingRect.right() + margin);
|
|
|
|
|
newModelBounds.setBottom(usedBoundingRect.bottom() + margin);
|
|
|
|
|
|
|
|
|
|
// If a part of "used" area is currently displayed in the center of the
|
|
|
|
|
// viewport, then keep it in the same place after scaling. Otherwise focus
|
|
|
|
|
// on the center of the used area after scale change.
|
|
|
|
|
if (usedBoundingRect.contains(displayedInCenter)) {
|
|
|
|
|
renderModel(noScale, newModelBounds, displayedInCenter, vpCenter);
|
|
|
|
|
} else {
|
|
|
|
|
renderModel(noScale, newModelBounds, usedBoundingRect.center(), vpCenter);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2024-01-05 20:40:57 +03:00
|
|
|
|
}
|
2006-04-18 06:03:52 +00:00
|
|
|
|
|
2024-01-06 23:02:30 +03:00
|
|
|
|
// If the model plane is smaller than rectangle described by points (x1, y1)
|
|
|
|
|
// and (x2, y2) than extend the model plane size.
|
|
|
|
|
void Schematic::enlargeView(int x1, int y1, int x2, int y2) {
|
|
|
|
|
// Set 'Used' area size to the of the given rectangle
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (x1 < a_UsedX1)
|
|
|
|
|
a_UsedX1 = x1;
|
|
|
|
|
if (y1 < a_UsedY1)
|
|
|
|
|
a_UsedY1 = y1;
|
|
|
|
|
if (x2 > a_UsedX2)
|
|
|
|
|
a_UsedX2 = x2;
|
|
|
|
|
if (y2 > a_UsedY2)
|
|
|
|
|
a_UsedY2 = y2;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-01-06 23:02:30 +03:00
|
|
|
|
// Construct the desired model plane
|
|
|
|
|
constexpr int margin = 40;
|
|
|
|
|
QRect newModel = modelRect();
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (x1 < a_ViewX1)
|
2024-01-06 23:02:30 +03:00
|
|
|
|
newModel.setLeft(x1 - margin);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (y1 < a_ViewY1)
|
2024-01-06 23:02:30 +03:00
|
|
|
|
newModel.setTop(y1 - margin);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (x2 > a_ViewX2)
|
2024-01-06 23:02:30 +03:00
|
|
|
|
newModel.setRight(x2 + margin);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (y2 > a_ViewY2)
|
2024-01-06 23:02:30 +03:00
|
|
|
|
newModel.setBottom(y2 + margin);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-01-06 23:02:30 +03:00
|
|
|
|
const auto vpCenter = viewportRect().center();
|
|
|
|
|
const auto displayedInCenter = viewportToModel(vpCenter);
|
2024-01-25 18:14:45 +03:00
|
|
|
|
newModel = includePoint(newModel, displayedInCenter);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
renderModel(a_Scale, newModel, displayedInCenter, vpCenter);
|
2024-01-06 23:02:30 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-02-12 22:23:22 +01:00
|
|
|
|
QPoint Schematic::setOnGrid(const QPoint& p) {
|
|
|
|
|
QPoint snappedToGrid{p.x(), p.y()};
|
|
|
|
|
setOnGrid(snappedToGrid.rx(), snappedToGrid.ry());
|
|
|
|
|
return snappedToGrid;
|
|
|
|
|
}
|
|
|
|
|
|
2006-03-28 06:10:52 +00:00
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
// Sets an arbitrary coordinate onto the next grid coordinate.
|
2023-12-11 15:13:33 +03:00
|
|
|
|
void Schematic::setOnGrid(int &x, int &y)
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (x < 0)
|
2024-11-08 12:46:57 +01:00
|
|
|
|
x -= (a_GridX >> 1) - 1;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
else
|
2024-11-08 12:46:57 +01:00
|
|
|
|
x += a_GridX >> 1;
|
|
|
|
|
x -= x % a_GridX;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (y < 0)
|
2024-11-08 12:46:57 +01:00
|
|
|
|
y -= (a_GridY >> 1) - 1;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
else
|
2024-11-08 12:46:57 +01:00
|
|
|
|
y += a_GridY >> 1;
|
|
|
|
|
y -= y % a_GridY;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-05-18 16:31:17 +03:00
|
|
|
|
void Schematic::drawGrid(QPainter* painter) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (!a_GridOn)
|
2024-05-18 16:31:17 +03:00
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
painter->save();
|
|
|
|
|
// Painter might have been scaled somewhere upstream in call stack,
|
|
|
|
|
// and we don't grid points to change their size or thickness
|
|
|
|
|
// depending on zoom level. Thus we remove any transformations
|
|
|
|
|
// and draw on "raw" painter, controlling all offsets manually
|
|
|
|
|
painter->setTransform(QTransform{});
|
|
|
|
|
|
|
|
|
|
// A grid drawn with pen of 1.0 width reportedly looks good both
|
|
|
|
|
// on standard and HiDPI displays.
|
|
|
|
|
// See here for details https://github.com/ra3xdh/qucs_s/pull/524
|
2024-11-15 21:05:09 +00:00
|
|
|
|
painter->setPen(QPen{ a_GridColor, 1.0 });
|
2024-05-18 16:31:17 +03:00
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
// Draw small cross at origin of coordinates
|
|
|
|
|
const QPoint origin = modelToViewport(QPoint{0, 0});
|
|
|
|
|
painter->drawLine(origin.x() - 3, origin.y(), origin.x() + 4, origin.y()); // horizontal stick
|
|
|
|
|
painter->drawLine(origin.x(), origin.y() - 3, origin.x(), origin.y() + 4); // vertical stick
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Grid is drawn as a set of nodes, each node looks like a point and a node
|
|
|
|
|
// is located at every horizontal and vertical step:
|
|
|
|
|
// . . . . .
|
|
|
|
|
// . . . . .
|
|
|
|
|
// . . . . .
|
|
|
|
|
// . . . . .
|
|
|
|
|
//
|
|
|
|
|
// To find out where to start drawing grid nodes, we find a point
|
|
|
|
|
// which is currently shown at the top left corner of the viewport
|
|
|
|
|
// and then find a grid-node nearest to this point. We then convert these
|
|
|
|
|
// grid-node coordinates back to viewport-coordinates. This gives us
|
|
|
|
|
// coordinates of a point somewhere around the top-left corner of the
|
|
|
|
|
// viewport. This point corresponds to a grid-node. The same is done to a
|
|
|
|
|
// bottom-right corner. Two resulting points decsribe the area where
|
|
|
|
|
// grid-nodes should be drawn — where to start and where to finish drawing
|
|
|
|
|
// these nodes.
|
|
|
|
|
|
|
|
|
|
QPoint topLeft = viewportToModel(viewportRect().topLeft());
|
|
|
|
|
const QPoint gridTopLeft = modelToViewport(setOnGrid(topLeft));
|
|
|
|
|
|
|
|
|
|
QPoint bottomRight = viewportToModel(viewportRect().bottomRight());
|
|
|
|
|
const QPoint gridBottomRight = modelToViewport(setOnGrid(bottomRight));
|
|
|
|
|
|
|
|
|
|
// This is the minimal distance between drawn grid-nodes. No matter how
|
|
|
|
|
// a user scales the view, any two adjacent nodes must have at least this
|
|
|
|
|
// amount of "free space" between them.
|
|
|
|
|
constexpr double minimalVisibleGridStep = 8.0;
|
|
|
|
|
|
|
|
|
|
// In some scales drawing a point for every step may lead to a very dense
|
|
|
|
|
// grid without much space between nodes. But we want to have some minimal
|
|
|
|
|
// distance between them. In such cases nodes shouldn't be drawn for every
|
|
|
|
|
// grid-step, but instead for every two grid-steps, or every three, and so on.
|
|
|
|
|
//
|
|
|
|
|
// To find out how frequently grid nodes should be drawn, we start from single
|
|
|
|
|
// grid-step and grow it until its "size in scale" gets larger than the minimal
|
|
|
|
|
// distance between points.
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
double horizontalStep{ a_GridX * a_Scale };
|
2024-05-18 16:31:17 +03:00
|
|
|
|
for (int n = 2; horizontalStep < minimalVisibleGridStep; n++) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
horizontalStep = n * a_GridX * a_Scale;
|
2024-05-18 16:31:17 +03:00
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
double verticalStep{ a_GridY * a_Scale };
|
2024-05-18 16:31:17 +03:00
|
|
|
|
for (int n = 2; verticalStep < minimalVisibleGridStep; n++) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
verticalStep = n * a_GridY * a_Scale;
|
2024-05-18 16:31:17 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Finally draw the grid-nodes
|
|
|
|
|
for (double x = gridTopLeft.x(); x <= gridBottomRight.x(); x += horizontalStep) {
|
|
|
|
|
for (double y = gridTopLeft.y(); y <= gridBottomRight.y(); y += verticalStep) {
|
|
|
|
|
painter->drawPoint(std::round(x), std::round(y));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
painter->restore();
|
|
|
|
|
}
|
|
|
|
|
|
2024-07-13 14:31:26 +02:00
|
|
|
|
void Schematic::relativeRotation(int &newX, int &newY, int comX, int comY, int oldX, int oldY)
|
|
|
|
|
{
|
|
|
|
|
// Shift coordinate system to center of mass
|
|
|
|
|
// Rotate
|
|
|
|
|
// Shift coordinate system back to origin
|
|
|
|
|
newX = (oldY-comY)+comX;
|
|
|
|
|
newY = -(oldX-comX)+comY;
|
|
|
|
|
}
|
|
|
|
|
|
2024-04-05 11:59:17 +03:00
|
|
|
|
void Schematic::updateAllBoundingRect() {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
sizeOfAll(a_UsedX1, a_UsedY1, a_UsedX2, a_UsedY2);
|
2024-04-05 11:59:17 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QRect Schematic::allBoundingRect() {
|
|
|
|
|
updateAllBoundingRect();
|
2024-11-08 12:46:57 +01:00
|
|
|
|
return QRect{a_UsedX1, a_UsedY1, (a_UsedX2 - a_UsedX1), (a_UsedY2 - a_UsedY1)};
|
2024-04-05 11:59:17 +03:00
|
|
|
|
}
|
|
|
|
|
|
2006-03-28 06:10:52 +00:00
|
|
|
|
// ---------------------------------------------------
|
2023-12-11 15:13:33 +03:00
|
|
|
|
void Schematic::sizeOfAll(int &xmin, int &ymin, int &xmax, int &ymax)
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
xmin = INT_MAX;
|
|
|
|
|
ymin = INT_MAX;
|
|
|
|
|
xmax = INT_MIN;
|
|
|
|
|
ymax = INT_MIN;
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_Components->isEmpty() && a_Wires->isEmpty() && a_Diagrams->isEmpty() && a_Paintings->isEmpty()) {
|
2024-03-24 01:51:52 +03:00
|
|
|
|
xmin = xmax = 0;
|
|
|
|
|
ymin = ymax = 0;
|
|
|
|
|
return;
|
|
|
|
|
}
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
int x1, y1, x2, y2;
|
|
|
|
|
// find boundings of all components
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* pc : *a_Components) {
|
2024-02-27 23:20:40 +03:00
|
|
|
|
pc->entireBounds(x1, y1, x2, y2);
|
2024-03-24 01:50:21 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// find boundings of all wires
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* pw : *a_Wires) {
|
2024-03-24 01:50:21 +03:00
|
|
|
|
xmin = std::min(pw->x1, xmin);
|
|
|
|
|
xmax = std::max(pw->x2, xmax);
|
|
|
|
|
ymin = std::min(pw->y1, ymin);
|
|
|
|
|
ymax = std::max(pw->y2, ymax);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-03-24 01:45:38 +03:00
|
|
|
|
if (auto* pl = pw->Label; pl) { // check position of wire label
|
2023-12-11 15:13:33 +03:00
|
|
|
|
pl->getLabelBounding(x1, y1, x2, y2);
|
2024-03-24 01:50:21 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// find boundings of all node labels
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* pn : *a_Nodes) {
|
2024-03-24 01:45:38 +03:00
|
|
|
|
if (auto* pl = pn->Label; pl) { // check position of node label
|
2023-12-11 15:13:33 +03:00
|
|
|
|
pl->getLabelBounding(x1, y1, x2, y2);
|
2024-03-24 01:50:21 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// find boundings of all diagrams
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* pd : *a_Diagrams) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
pd->Bounding(x1, y1, x2, y2);
|
2024-03-24 01:50:21 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-03-24 01:45:38 +03:00
|
|
|
|
for (auto* pg : pd->Graphs)
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// test all markers of diagram
|
2024-03-24 01:45:38 +03:00
|
|
|
|
for (auto* pm : pg->Markers) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
pm->Bounding(x1, y1, x2, y2);
|
2024-03-24 01:50:21 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// find boundings of all Paintings
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* pp : *a_Paintings) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
pp->Bounding(x1, y1, x2, y2);
|
2024-03-24 01:50:21 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-03-08 14:10:39 +03:00
|
|
|
|
QRect Schematic::sizeOfSelection() const {
|
2024-03-08 13:33:55 +03:00
|
|
|
|
int xmin = INT_MAX;
|
|
|
|
|
int ymin = INT_MAX;
|
|
|
|
|
int xmax = INT_MIN;
|
|
|
|
|
int ymax = INT_MIN;
|
2023-10-17 21:29:39 +03:00
|
|
|
|
|
|
|
|
|
bool isAnySelected = false;
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_Components->isEmpty() && a_Wires->isEmpty() && a_Diagrams->isEmpty() &&
|
|
|
|
|
a_Paintings->isEmpty()) {
|
2024-03-08 14:00:42 +03:00
|
|
|
|
return QRect{};
|
|
|
|
|
}
|
2023-10-17 21:29:39 +03:00
|
|
|
|
|
|
|
|
|
int x1, y1, x2, y2;
|
|
|
|
|
// find boundings of all components
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* pc : *a_Components) {
|
2023-10-17 21:29:39 +03:00
|
|
|
|
if (!pc->isSelected) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
isAnySelected = true;
|
2024-02-27 23:20:40 +03:00
|
|
|
|
pc->entireBounds(x1, y1, x2, y2);
|
2024-03-08 13:57:28 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2023-10-17 21:29:39 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// find boundings of all wires
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* pw : *a_Wires) {
|
2023-10-17 21:29:39 +03:00
|
|
|
|
if (!pw->isSelected) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
isAnySelected = true;
|
2024-03-08 14:10:39 +03:00
|
|
|
|
xmin = std::min(pw->x1, xmin);
|
|
|
|
|
xmax = std::max(pw->x2, xmax);
|
|
|
|
|
ymin = std::min(pw->y1, ymin);
|
|
|
|
|
ymax = std::max(pw->y2, ymax);
|
2023-10-17 21:29:39 +03:00
|
|
|
|
|
2024-03-08 14:10:39 +03:00
|
|
|
|
if (auto* pl = pw->Label; pl) { // check position of wire label
|
2023-12-11 15:13:33 +03:00
|
|
|
|
pl->getLabelBounding(x1, y1, x2, y2);
|
2024-03-08 13:57:28 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2023-10-17 21:29:39 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// find boundings of all node labels
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* pn : *a_Nodes) {
|
2023-10-17 21:29:39 +03:00
|
|
|
|
if (!pn->isSelected) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2024-03-08 13:25:48 +03:00
|
|
|
|
|
2024-03-08 14:10:39 +03:00
|
|
|
|
if (auto* pl = pn->Label; pl) { // check position of node label
|
2024-01-18 18:14:57 +03:00
|
|
|
|
isAnySelected = true;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
pl->getLabelBounding(x1, y1, x2, y2);
|
2024-03-08 13:57:28 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2023-10-17 21:29:39 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// find boundings of all diagrams
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* pd : *a_Diagrams) {
|
2023-10-17 21:29:39 +03:00
|
|
|
|
if (!pd->isSelected) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
isAnySelected = true;
|
|
|
|
|
pd->Bounding(x1, y1, x2, y2);
|
2024-03-08 13:57:28 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2023-10-17 21:29:39 +03:00
|
|
|
|
|
2024-03-08 14:10:39 +03:00
|
|
|
|
for (Graph* pg : pd->Graphs) {
|
2023-10-17 21:29:39 +03:00
|
|
|
|
// test all markers of diagram
|
2024-03-08 14:10:39 +03:00
|
|
|
|
for (Marker* pm : pg->Markers) {
|
2023-10-17 21:29:39 +03:00
|
|
|
|
pm->Bounding(x1, y1, x2, y2);
|
2024-03-08 13:57:28 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2023-10-17 21:29:39 +03:00
|
|
|
|
}
|
2024-03-08 14:10:39 +03:00
|
|
|
|
}
|
2023-10-17 21:29:39 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// find boundings of all Paintings
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (auto* pp : *a_Paintings) {
|
2023-10-17 21:29:39 +03:00
|
|
|
|
if (!pp->isSelected) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
isAnySelected = true;
|
|
|
|
|
pp->Bounding(x1, y1, x2, y2);
|
2024-03-08 13:57:28 +03:00
|
|
|
|
xmin = std::min(x1, xmin);
|
|
|
|
|
xmax = std::max(x2, xmax);
|
|
|
|
|
ymin = std::min(y1, ymin);
|
|
|
|
|
ymax = std::max(y2, ymax);
|
2023-10-17 21:29:39 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!isAnySelected) {
|
2024-03-08 13:33:55 +03:00
|
|
|
|
return QRect{};
|
2023-10-17 21:29:39 +03:00
|
|
|
|
}
|
2024-03-08 13:33:55 +03:00
|
|
|
|
|
|
|
|
|
return QRect{xmin, ymin, xmax - xmin, ymax - ymin};
|
2023-10-17 21:29:39 +03:00
|
|
|
|
}
|
|
|
|
|
|
2006-03-28 06:10:52 +00:00
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
// Rotates all selected components around their midpoint.
|
|
|
|
|
bool Schematic::rotateElements()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Wires->setAutoDelete(false);
|
|
|
|
|
a_Components->setAutoDelete(false);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-07-13 14:31:26 +02:00
|
|
|
|
// To rotate a selected area its necessary to work with half steps
|
2023-12-11 15:13:33 +03:00
|
|
|
|
int x1 = INT_MAX, y1 = INT_MAX;
|
|
|
|
|
int x2 = INT_MIN, y2 = INT_MIN;
|
|
|
|
|
QList<Element *> ElementCache;
|
|
|
|
|
copyLabels(x1, y1, x2, y2, &ElementCache); // must be first of all !
|
|
|
|
|
copyComponents(x1, y1, x2, y2, &ElementCache);
|
|
|
|
|
copyWires(x1, y1, x2, y2, &ElementCache);
|
|
|
|
|
copyPaintings(x1, y1, x2, y2, &ElementCache);
|
|
|
|
|
if (y1 == INT_MAX)
|
2024-07-13 14:31:26 +02:00
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return false; // no element selected
|
2024-07-13 14:31:26 +02:00
|
|
|
|
}
|
|
|
|
|
int comX = (x1 + ((x2-x1) / 2)); // center of mass
|
|
|
|
|
int comY = (y1 + ((y2-y1) / 2));
|
|
|
|
|
int newPosX = INT_MIN;
|
|
|
|
|
int newPosY = INT_MIN;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Wires->setAutoDelete(true);
|
|
|
|
|
a_Components->setAutoDelete(true);
|
2024-07-13 14:31:26 +02:00
|
|
|
|
setOnGrid(comX, comY);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
Wire *pw;
|
|
|
|
|
Painting *pp;
|
|
|
|
|
Component *pc;
|
|
|
|
|
WireLabel *pl;
|
|
|
|
|
// re-insert elements
|
|
|
|
|
for (Element *pe : ElementCache)
|
|
|
|
|
switch (pe->Type) {
|
|
|
|
|
case isComponent:
|
|
|
|
|
case isAnalogComponent:
|
|
|
|
|
case isDigitalComponent:
|
|
|
|
|
pc = (Component *) pe;
|
|
|
|
|
pc->rotate(); //rotate component !before! rotating its center
|
2024-07-13 14:31:26 +02:00
|
|
|
|
relativeRotation(newPosX, newPosY, comX, comY, pc->cx, pc->cy);
|
|
|
|
|
pc->setCenter(newPosX, newPosY);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
insertRawComponent(pc);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case isWire:
|
|
|
|
|
pw = (Wire *) pe;
|
2024-07-13 14:31:26 +02:00
|
|
|
|
relativeRotation(newPosX, newPosY, comX, comY, pw->x1, pw->y1);
|
|
|
|
|
pw->x1 = newPosX;
|
|
|
|
|
pw->y1 = newPosY;
|
|
|
|
|
relativeRotation(newPosX, newPosY, comX, comY, pw->x2, pw->y2);
|
|
|
|
|
pw->x2 = newPosX;
|
|
|
|
|
pw->y2 = newPosY;
|
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
pl = pw->Label;
|
|
|
|
|
if (pl) {
|
|
|
|
|
x2 = pl->cx;
|
2024-07-13 14:31:26 +02:00
|
|
|
|
relativeRotation(newPosX, newPosY, comX, comY, pl->cx, pl->cy);
|
|
|
|
|
pl->cx = newPosX;
|
|
|
|
|
pl->cy = newPosY;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pl->Type == isHWireLabel)
|
|
|
|
|
pl->Type = isVWireLabel;
|
|
|
|
|
else
|
|
|
|
|
pl->Type = isHWireLabel;
|
|
|
|
|
}
|
|
|
|
|
insertWire(pw);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case isHWireLabel:
|
|
|
|
|
case isVWireLabel:
|
|
|
|
|
pl = (WireLabel *) pe;
|
2024-07-13 14:31:26 +02:00
|
|
|
|
relativeRotation(newPosX, newPosY, comX, comY, pl->x1, pl->y1);
|
|
|
|
|
pl->x1 = newPosX;
|
|
|
|
|
pl->y1 = newPosY;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
break;
|
|
|
|
|
case isNodeLabel:
|
|
|
|
|
pl = (WireLabel *) pe;
|
|
|
|
|
if (pl->pOwner == 0) {
|
2024-07-13 14:31:26 +02:00
|
|
|
|
relativeRotation(newPosX, newPosY, comX, comY, pl->x1, pl->y1);
|
|
|
|
|
pl->x1 = newPosX;
|
|
|
|
|
pl->y1 = newPosY;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2024-07-13 14:31:26 +02:00
|
|
|
|
relativeRotation(newPosX, newPosY, comX, comY, pl->cx, pl->cy);
|
|
|
|
|
pl->cx = newPosX;
|
|
|
|
|
pl->cy = newPosY;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
insertNodeLabel(pl);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case isPainting:
|
|
|
|
|
pp = (Painting *) pe;
|
|
|
|
|
pp->rotate(x1, y1); // rotate around the center x1 y1
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Paintings->append(pp);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
break;
|
|
|
|
|
default:;
|
2006-05-22 06:01:55 +00:00
|
|
|
|
}
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
ElementCache.clear();
|
|
|
|
|
|
|
|
|
|
setChanged(true, true);
|
|
|
|
|
return true;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
2006-05-26 06:03:15 +00:00
|
|
|
|
// Mirrors all selected components.
|
|
|
|
|
// First copy them to 'ElementCache', then mirror and insert again.
|
2006-03-28 06:10:52 +00:00
|
|
|
|
bool Schematic::mirrorXComponents()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Wires->setAutoDelete(false);
|
|
|
|
|
a_Components->setAutoDelete(false);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
int x1, y1, x2, y2;
|
|
|
|
|
QList<Element *> ElementCache;
|
|
|
|
|
if (!copyComps2WiresPaints(x1, y1, x2, y2, &ElementCache))
|
|
|
|
|
return false;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Wires->setAutoDelete(true);
|
|
|
|
|
a_Components->setAutoDelete(true);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
y1 = (y1 + y2) >> 1; // axis for mirroring
|
|
|
|
|
setOnGrid(y2, y1);
|
|
|
|
|
y1 <<= 1;
|
|
|
|
|
|
|
|
|
|
Wire *pw;
|
|
|
|
|
Painting *pp;
|
|
|
|
|
Component *pc;
|
|
|
|
|
WireLabel *pl;
|
|
|
|
|
// re-insert elements
|
|
|
|
|
for (Element *pe : ElementCache)
|
|
|
|
|
switch (pe->Type) {
|
|
|
|
|
case isComponent:
|
|
|
|
|
case isAnalogComponent:
|
|
|
|
|
case isDigitalComponent:
|
|
|
|
|
pc = (Component *) pe;
|
|
|
|
|
pc->mirrorX(); // mirror component !before! mirroring its center
|
|
|
|
|
pc->setCenter(pc->cx, y1 - pc->cy);
|
|
|
|
|
insertRawComponent(pc);
|
|
|
|
|
break;
|
|
|
|
|
case isWire:
|
|
|
|
|
pw = (Wire *) pe;
|
|
|
|
|
pw->y1 = y1 - pw->y1;
|
|
|
|
|
pw->y2 = y1 - pw->y2;
|
|
|
|
|
pl = pw->Label;
|
|
|
|
|
if (pl)
|
|
|
|
|
pl->cy = y1 - pl->cy;
|
|
|
|
|
insertWire(pw);
|
|
|
|
|
break;
|
|
|
|
|
case isHWireLabel:
|
|
|
|
|
case isVWireLabel:
|
|
|
|
|
pl = (WireLabel *) pe;
|
|
|
|
|
pl->y1 = y1 - pl->y1;
|
|
|
|
|
break;
|
|
|
|
|
case isNodeLabel:
|
|
|
|
|
pl = (WireLabel *) pe;
|
|
|
|
|
if (pl->pOwner == 0)
|
|
|
|
|
pl->y1 = y1 - pl->y1;
|
|
|
|
|
pl->cy = y1 - pl->cy;
|
|
|
|
|
insertNodeLabel(pl);
|
|
|
|
|
break;
|
|
|
|
|
case isPainting:
|
|
|
|
|
pp = (Painting *) pe;
|
|
|
|
|
pp->getCenter(x2, y2);
|
|
|
|
|
pp->mirrorX(); // mirror painting !before! mirroring its center
|
|
|
|
|
pp->setCenter(x2, y1 - y2);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Paintings->append(pp);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
break;
|
|
|
|
|
default:;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ElementCache.clear();
|
|
|
|
|
setChanged(true, true);
|
|
|
|
|
return true;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
// Mirrors all selected components. First copy them to 'ElementCache', then mirror and insert again.
|
|
|
|
|
bool Schematic::mirrorYComponents()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Wires->setAutoDelete(false);
|
|
|
|
|
a_Components->setAutoDelete(false);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
int x1, y1, x2, y2;
|
|
|
|
|
QList<Element *> ElementCache;
|
|
|
|
|
if (!copyComps2WiresPaints(x1, y1, x2, y2, &ElementCache))
|
|
|
|
|
return false;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Wires->setAutoDelete(true);
|
|
|
|
|
a_Components->setAutoDelete(true);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
x1 = (x1 + x2) >> 1; // axis for mirroring
|
|
|
|
|
setOnGrid(x1, x2);
|
|
|
|
|
x1 <<= 1;
|
|
|
|
|
|
|
|
|
|
Wire *pw;
|
|
|
|
|
Painting *pp;
|
|
|
|
|
Component *pc;
|
|
|
|
|
WireLabel *pl;
|
|
|
|
|
// re-insert elements
|
|
|
|
|
for (Element *pe : ElementCache)
|
|
|
|
|
switch (pe->Type) {
|
|
|
|
|
case isComponent:
|
|
|
|
|
case isAnalogComponent:
|
|
|
|
|
case isDigitalComponent:
|
|
|
|
|
pc = (Component *) pe;
|
|
|
|
|
pc->mirrorY(); // mirror component !before! mirroring its center
|
|
|
|
|
pc->setCenter(x1 - pc->cx, pc->cy);
|
|
|
|
|
insertRawComponent(pc);
|
|
|
|
|
break;
|
|
|
|
|
case isWire:
|
|
|
|
|
pw = (Wire *) pe;
|
|
|
|
|
pw->x1 = x1 - pw->x1;
|
|
|
|
|
pw->x2 = x1 - pw->x2;
|
|
|
|
|
pl = pw->Label;
|
|
|
|
|
if (pl)
|
|
|
|
|
pl->cx = x1 - pl->cx;
|
|
|
|
|
insertWire(pw);
|
|
|
|
|
break;
|
|
|
|
|
case isHWireLabel:
|
|
|
|
|
case isVWireLabel:
|
|
|
|
|
pl = (WireLabel *) pe;
|
|
|
|
|
pl->x1 = x1 - pl->x1;
|
|
|
|
|
break;
|
|
|
|
|
case isNodeLabel:
|
|
|
|
|
pl = (WireLabel *) pe;
|
|
|
|
|
if (pl->pOwner == 0)
|
|
|
|
|
pl->x1 = x1 - pl->x1;
|
|
|
|
|
pl->cx = x1 - pl->cx;
|
|
|
|
|
insertNodeLabel(pl);
|
|
|
|
|
break;
|
|
|
|
|
case isPainting:
|
|
|
|
|
pp = (Painting *) pe;
|
|
|
|
|
pp->getCenter(x2, y2);
|
|
|
|
|
pp->mirrorY(); // mirror painting !before! mirroring its center
|
|
|
|
|
pp->setCenter(x1 - x2, y2);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Paintings->append(pp);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
break;
|
|
|
|
|
default:;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ElementCache.clear();
|
|
|
|
|
setChanged(true, true);
|
|
|
|
|
return true;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
// Updates the graph data of all diagrams (load from data files).
|
|
|
|
|
void Schematic::reloadGraphs()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
QFileInfo Info(a_DocName);
|
|
|
|
|
for (Diagram *pd = a_Diagrams->first(); pd != 0; pd = a_Diagrams->next())
|
|
|
|
|
pd->loadGraphData(Info.path() + QDir::separator() + a_DataSet);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// Copy function,
|
2014-12-11 13:28:36 +08:00
|
|
|
|
void Schematic::copy()
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
QString s = createClipboardFile();
|
|
|
|
|
QClipboard *cb = QApplication::clipboard(); // get system clipboard
|
|
|
|
|
if (!s.isEmpty()) {
|
|
|
|
|
cb->setText(s, QClipboard::Clipboard);
|
|
|
|
|
}
|
2014-12-11 13:28:36 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
// Cut function, copy followed by deletion
|
|
|
|
|
void Schematic::cut()
|
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
copy();
|
|
|
|
|
deleteElements(); //delete selected elements
|
|
|
|
|
viewport()->update();
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
// Performs paste function from clipboard
|
2014-06-06 22:06:38 +02:00
|
|
|
|
bool Schematic::paste(QTextStream *stream, Q3PtrList<Element> *pe)
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return pasteFromClipboard(stream, pe);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
// Loads this Qucs document.
|
|
|
|
|
bool Schematic::load()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_DocComps.clear();
|
|
|
|
|
a_DocWires.clear();
|
|
|
|
|
a_DocNodes.clear();
|
|
|
|
|
a_DocDiags.clear();
|
|
|
|
|
a_DocPaints.clear();
|
|
|
|
|
a_SymbolPaints.clear();
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
if (!loadDocument())
|
|
|
|
|
return false;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_lastSaved = QDateTime::currentDateTime();
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
while (!a_undoAction.isEmpty()) {
|
|
|
|
|
delete a_undoAction.last();
|
|
|
|
|
a_undoAction.pop_back();
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_undoActionIdx = 0;
|
|
|
|
|
while (!a_undoSymbol.isEmpty()) {
|
|
|
|
|
delete a_undoSymbol.last();
|
|
|
|
|
a_undoSymbol.pop_back();
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_symbolMode = true;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
setChanged(false, true); // "not changed" state, but put on undo stack
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_undoSymbolIdx = 0;
|
|
|
|
|
a_undoSymbol.at(a_undoSymbolIdx)->replace(1, 1, 'i');
|
|
|
|
|
a_symbolMode = false;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
setChanged(false, true); // "not changed" state, but put on undo stack
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_undoActionIdx = 0;
|
|
|
|
|
a_undoAction.at(a_undoActionIdx)->replace(1, 1, 'i');
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
// The undo stack of the circuit symbol is initialized when first
|
|
|
|
|
// entering its edit mode.
|
|
|
|
|
|
|
|
|
|
// have to call this to avoid crash at sizeOfAll
|
|
|
|
|
becomeCurrent(false);
|
|
|
|
|
|
2024-01-05 21:06:35 +03:00
|
|
|
|
showAll();
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_tmpViewX1 = a_tmpViewY1 = -200; // was used as temporary cache
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return true;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
// Saves this Qucs document. Returns the number of subcircuit ports.
|
|
|
|
|
int Schematic::save()
|
|
|
|
|
{
|
2024-04-20 11:05:32 +03:00
|
|
|
|
int result = 0;
|
|
|
|
|
// When saving *only* a symbol, there is no corresponding schematic:
|
|
|
|
|
// and thus ports in symbol don't have corresponding ports in schematic.
|
|
|
|
|
// There is just nothing to adjust.
|
|
|
|
|
//
|
|
|
|
|
// In other cases we want to delete any dangling ports from symbol
|
|
|
|
|
// and invoke "adjustPortNumbers" for it.
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (!a_isSymbolOnly) {
|
2024-04-20 11:05:32 +03:00
|
|
|
|
result = adjustPortNumbers(); // same port number for schematic and symbol
|
2024-06-16 15:14:19 +03:00
|
|
|
|
} else {
|
|
|
|
|
orderSymbolPorts();
|
2024-04-20 11:05:32 +03:00
|
|
|
|
}
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (saveDocument() < 0)
|
|
|
|
|
return -1;
|
2008-03-25 14:49:15 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
QFileInfo Info(a_DocName);
|
|
|
|
|
a_lastSaved = Info.lastModified();
|
2008-03-25 14:49:15 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (result >= 0) {
|
|
|
|
|
setChanged(false);
|
2014-11-08 23:58:37 +08:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
QVector<QString *>::iterator it;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (it = a_undoAction.begin(); it != a_undoAction.end(); it++) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
(*it)->replace(1, 1, ' '); //at(1) = ' '; state of being changed
|
|
|
|
|
}
|
|
|
|
|
//(1) = 'i'; // state of being unchanged
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_undoAction.at(a_undoActionIdx)->replace(1, 1, 'i');
|
2014-11-08 23:58:37 +08:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (it = a_undoSymbol.begin(); it != a_undoSymbol.end(); it++) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
(*it)->replace(1, 1, ' '); //at(1) = ' '; state of being changed
|
|
|
|
|
}
|
|
|
|
|
//at(1) = 'i'; // state of being unchanged
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_undoSymbol.at(a_undoSymbolIdx)->replace(1, 1, 'i');
|
2014-11-08 23:58:37 +08:00
|
|
|
|
}
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// update the subcircuit file lookup hashes
|
2024-03-12 21:04:10 -04:00
|
|
|
|
//QucsMain->updateSchNameHash();
|
|
|
|
|
//QucsMain->updateSpiceNameHash();
|
2014-01-28 17:31:35 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return result;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
// If the port number of the schematic and of the symbol are not
|
|
|
|
|
// equal add or remove some in the symbol.
|
|
|
|
|
int Schematic::adjustPortNumbers()
|
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
int x1, x2, y1, y2;
|
|
|
|
|
// get size of whole symbol to know where to place new ports
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_symbolMode)
|
2023-12-11 15:13:33 +03:00
|
|
|
|
sizeOfAll(x1, y1, x2, y2);
|
|
|
|
|
else {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Components = &SymbolComps;
|
|
|
|
|
a_Wires = &SymbolWires;
|
|
|
|
|
a_Nodes = &SymbolNodes;
|
|
|
|
|
a_Diagrams = &SymbolDiags;
|
|
|
|
|
a_Paintings = &a_SymbolPaints;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
sizeOfAll(x1, y1, x2, y2);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Components = &a_DocComps;
|
|
|
|
|
a_Wires = &a_DocWires;
|
|
|
|
|
a_Nodes = &a_DocNodes;
|
|
|
|
|
a_Diagrams = &a_DocDiags;
|
|
|
|
|
a_Paintings = &a_DocPaints;
|
2014-11-08 23:58:37 +08:00
|
|
|
|
}
|
2023-12-11 15:13:33 +03:00
|
|
|
|
x1 += 40;
|
|
|
|
|
y2 += 20;
|
|
|
|
|
setOnGrid(x1, y2);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
Painting *pp;
|
|
|
|
|
// delete all port names in symbol
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (pp = a_SymbolPaints.first(); pp != 0; pp = a_SymbolPaints.next())
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pp->Name == ".PortSym ")
|
|
|
|
|
((PortSymbol *) pp)->nameStr = "";
|
|
|
|
|
|
|
|
|
|
QString Str;
|
|
|
|
|
int countPort = 0;
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
QFileInfo Info(a_DataDisplay);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
QString Suffix = Info.suffix();
|
|
|
|
|
|
|
|
|
|
// handle VHDL file symbol
|
|
|
|
|
if (Suffix == "vhd" || Suffix == "vhdl") {
|
|
|
|
|
QStringList::iterator it;
|
|
|
|
|
QStringList Names, GNames, GTypes, GDefs;
|
|
|
|
|
int Number;
|
|
|
|
|
|
|
|
|
|
// get ports from VHDL file
|
2024-11-08 12:46:57 +01:00
|
|
|
|
QFileInfo Info(a_DocName);
|
|
|
|
|
QString Name = Info.path() + QDir::separator() + a_DataDisplay;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
// obtain VHDL information either from open text document or the
|
|
|
|
|
// file directly
|
|
|
|
|
VHDL_File_Info VInfo;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
TextDoc *d = (TextDoc *) a_App->findDoc(Name);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (d)
|
|
|
|
|
VInfo = VHDL_File_Info(d->document()->toPlainText());
|
|
|
|
|
else
|
|
|
|
|
VInfo = VHDL_File_Info(Name, true);
|
|
|
|
|
|
|
|
|
|
if (!VInfo.PortNames.isEmpty())
|
2025-01-03 21:06:41 +03:00
|
|
|
|
Names = VInfo.PortNames.split(",", Qt::SkipEmptyParts);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (pp = a_SymbolPaints.first(); pp != 0; pp = a_SymbolPaints.next())
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pp->Name == ".ID ") {
|
|
|
|
|
ID_Text *id = (ID_Text *) pp;
|
|
|
|
|
id->Prefix = VInfo.EntityName.toUpper();
|
|
|
|
|
id->Parameter.clear();
|
|
|
|
|
if (!VInfo.GenNames.isEmpty())
|
2025-01-03 21:06:41 +03:00
|
|
|
|
GNames = VInfo.GenNames.split(",", Qt::SkipEmptyParts);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (!VInfo.GenTypes.isEmpty())
|
2025-01-03 21:06:41 +03:00
|
|
|
|
GTypes = VInfo.GenTypes.split(",", Qt::SkipEmptyParts);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (!VInfo.GenDefs.isEmpty())
|
2025-01-03 21:06:41 +03:00
|
|
|
|
GDefs = VInfo.GenDefs.split(",", Qt::SkipEmptyParts);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
;
|
|
|
|
|
for (Number = 1, it = GNames.begin(); it != GNames.end(); ++it) {
|
|
|
|
|
id->Parameter.append(
|
|
|
|
|
new SubParameter(true,
|
|
|
|
|
*it + "=" + GDefs[Number - 1],
|
|
|
|
|
tr("generic") + " " + QString::number(Number),
|
|
|
|
|
GTypes[Number - 1]));
|
|
|
|
|
Number++;
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
for (Number = 1, it = Names.begin(); it != Names.end(); ++it, Number++) {
|
|
|
|
|
countPort++;
|
|
|
|
|
|
|
|
|
|
Str = QString::number(Number);
|
|
|
|
|
// search for matching port symbol
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (pp = a_SymbolPaints.first(); pp != 0; pp = a_SymbolPaints.next())
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pp->Name == ".PortSym ")
|
|
|
|
|
if (((PortSymbol *) pp)->numberStr == Str)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (pp)
|
|
|
|
|
((PortSymbol *) pp)->nameStr = *it;
|
|
|
|
|
else {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_SymbolPaints.append(new PortSymbol(x1, y2, Str, *it));
|
2023-12-11 15:13:33 +03:00
|
|
|
|
y2 += 40;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// handle Verilog-HDL file symbol
|
|
|
|
|
else if (Suffix == "v") {
|
|
|
|
|
QStringList::iterator it;
|
|
|
|
|
QStringList Names;
|
|
|
|
|
int Number;
|
|
|
|
|
|
|
|
|
|
// get ports from Verilog-HDL file
|
2024-11-08 12:46:57 +01:00
|
|
|
|
QFileInfo Info(a_DocName);
|
|
|
|
|
QString Name = Info.path() + QDir::separator() + a_DataDisplay;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
// obtain Verilog-HDL information either from open text document or the
|
|
|
|
|
// file directly
|
|
|
|
|
Verilog_File_Info VInfo;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
TextDoc *d = (TextDoc *) a_App->findDoc(Name);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (d)
|
|
|
|
|
VInfo = Verilog_File_Info(d->document()->toPlainText());
|
|
|
|
|
else
|
|
|
|
|
VInfo = Verilog_File_Info(Name, true);
|
|
|
|
|
if (!VInfo.PortNames.isEmpty())
|
2025-01-03 21:06:41 +03:00
|
|
|
|
Names = VInfo.PortNames.split(",", Qt::SkipEmptyParts);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (pp = a_SymbolPaints.first(); pp != 0; pp = a_SymbolPaints.next())
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pp->Name == ".ID ") {
|
|
|
|
|
ID_Text *id = (ID_Text *) pp;
|
|
|
|
|
id->Prefix = VInfo.ModuleName.toUpper();
|
|
|
|
|
id->Parameter.clear();
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
for (Number = 1, it = Names.begin(); it != Names.end(); ++it, Number++) {
|
|
|
|
|
countPort++;
|
|
|
|
|
|
|
|
|
|
Str = QString::number(Number);
|
|
|
|
|
// search for matching port symbol
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (pp = a_SymbolPaints.first(); pp != 0; pp = a_SymbolPaints.next())
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pp->Name == ".PortSym ")
|
|
|
|
|
if (((PortSymbol *) pp)->numberStr == Str)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (pp)
|
|
|
|
|
((PortSymbol *) pp)->nameStr = *it;
|
|
|
|
|
else {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_SymbolPaints.append(new PortSymbol(x1, y2, Str, *it));
|
2023-12-11 15:13:33 +03:00
|
|
|
|
y2 += 40;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// handle Verilog-A file symbol
|
|
|
|
|
else if (Suffix == "va") {
|
|
|
|
|
QStringList::iterator it;
|
|
|
|
|
QStringList Names;
|
|
|
|
|
int Number;
|
|
|
|
|
|
|
|
|
|
// get ports from Verilog-A file
|
2024-11-08 12:46:57 +01:00
|
|
|
|
QFileInfo Info(a_DocName);
|
|
|
|
|
QString Name = Info.path() + QDir::separator() + a_DataDisplay;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
// obtain Verilog-A information either from open text document or the
|
|
|
|
|
// file directly
|
|
|
|
|
VerilogA_File_Info VInfo;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
TextDoc *d = (TextDoc *) a_App->findDoc(Name);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (d)
|
|
|
|
|
VInfo = VerilogA_File_Info(d->toPlainText());
|
|
|
|
|
else
|
|
|
|
|
VInfo = VerilogA_File_Info(Name, true);
|
|
|
|
|
|
|
|
|
|
if (!VInfo.PortNames.isEmpty())
|
2025-01-03 21:06:41 +03:00
|
|
|
|
Names = VInfo.PortNames.split(",", Qt::SkipEmptyParts);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (pp = a_SymbolPaints.first(); pp != 0; pp = a_SymbolPaints.next())
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pp->Name == ".ID ") {
|
|
|
|
|
ID_Text *id = (ID_Text *) pp;
|
|
|
|
|
id->Prefix = VInfo.ModuleName.toUpper();
|
|
|
|
|
id->Parameter.clear();
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
for (Number = 1, it = Names.begin(); it != Names.end(); ++it, Number++) {
|
|
|
|
|
countPort++;
|
|
|
|
|
|
|
|
|
|
Str = QString::number(Number);
|
|
|
|
|
// search for matching port symbol
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (pp = a_SymbolPaints.first(); pp != 0; pp = a_SymbolPaints.next())
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pp->Name == ".PortSym ")
|
|
|
|
|
if (((PortSymbol *) pp)->numberStr == Str)
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
if (pp)
|
|
|
|
|
((PortSymbol *) pp)->nameStr = *it;
|
|
|
|
|
else {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_SymbolPaints.append(new PortSymbol(x1, y2, Str, *it));
|
2023-12-11 15:13:33 +03:00
|
|
|
|
y2 += 40;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// handle schematic symbol
|
|
|
|
|
else {
|
|
|
|
|
// go through all components in a schematic
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (Component *pc = a_DocComps.first(); pc != 0; pc = a_DocComps.next()) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pc->Model == "Port") {
|
|
|
|
|
countPort++;
|
|
|
|
|
|
2024-07-23 00:37:09 +03:00
|
|
|
|
Str = pc->Props.front()->Value;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// search for matching port symbol
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (pp = a_SymbolPaints.first(); pp != 0; pp = a_SymbolPaints.next()) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pp->Name == ".PortSym ") {
|
|
|
|
|
if (((PortSymbol *) pp)->numberStr == Str)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pp) {
|
|
|
|
|
((PortSymbol *) pp)->nameStr = pc->Name;
|
|
|
|
|
} else {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_SymbolPaints.append(new PortSymbol(x1, y2, Str, pc->Name));
|
2023-12-11 15:13:33 +03:00
|
|
|
|
y2 += 40;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
// delete not accounted port symbols
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (pp = a_SymbolPaints.first(); pp != 0;) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (pp->Name == ".PortSym ")
|
|
|
|
|
if (((PortSymbol *) pp)->nameStr.isEmpty()) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_SymbolPaints.remove();
|
|
|
|
|
pp = a_SymbolPaints.current();
|
2023-12-11 15:13:33 +03:00
|
|
|
|
continue;
|
|
|
|
|
}
|
2024-11-08 12:46:57 +01:00
|
|
|
|
pp = a_SymbolPaints.next();
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return countPort;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-06-16 15:14:19 +03:00
|
|
|
|
int Schematic::orderSymbolPorts()
|
|
|
|
|
{
|
|
|
|
|
Painting *pp;
|
|
|
|
|
int countPorts = 0;
|
|
|
|
|
QSet<int> port_numbers, existing_numbers, free_numbers;
|
|
|
|
|
int max_port_number = 0;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (pp = a_SymbolPaints.first(); pp != 0; pp = a_SymbolPaints.next()) {
|
2024-06-16 15:14:19 +03:00
|
|
|
|
if (pp->Name == ".PortSym ") {
|
|
|
|
|
countPorts++;
|
|
|
|
|
QString numstr = ((PortSymbol *) pp)->numberStr;
|
|
|
|
|
if (numstr != "0") {
|
|
|
|
|
if (numstr.toInt() > max_port_number) {
|
|
|
|
|
max_port_number = numstr.toInt();
|
|
|
|
|
}
|
|
|
|
|
existing_numbers.insert(numstr.toInt());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
max_port_number = std::max(countPorts,max_port_number);
|
|
|
|
|
for (int i = 1; i <= max_port_number; i++) {
|
|
|
|
|
port_numbers.insert(i);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
free_numbers = port_numbers - existing_numbers;
|
|
|
|
|
|
|
|
|
|
// Assign new numbers only if port number is empty; Preserve ports order.
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (pp = a_SymbolPaints.first(); pp != 0; pp = a_SymbolPaints.next()) {
|
2024-06-16 15:14:19 +03:00
|
|
|
|
if (pp->Name == ".PortSym ") {
|
|
|
|
|
QString numstr = ((PortSymbol *) pp)->numberStr;
|
|
|
|
|
if (numstr == "0") {
|
|
|
|
|
int free_num = *free_numbers.constBegin();
|
|
|
|
|
free_numbers.remove(free_num);
|
|
|
|
|
((PortSymbol *) pp)->numberStr = QString::number(free_num);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return countPorts;
|
|
|
|
|
}
|
|
|
|
|
|
2006-03-28 06:10:52 +00:00
|
|
|
|
// ---------------------------------------------------
|
2023-12-11 15:13:33 +03:00
|
|
|
|
bool Schematic::undo()
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_symbolMode) {
|
|
|
|
|
if (a_undoSymbolIdx == 0) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return false;
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
rebuildSymbol(a_undoSymbol.at(--a_undoSymbolIdx));
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
emit signalUndoState(a_undoSymbolIdx != 0);
|
|
|
|
|
emit signalRedoState(a_undoSymbolIdx != a_undoSymbol.size() - 1);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_undoSymbol.at(a_undoSymbolIdx)->at(1) == 'i'
|
|
|
|
|
&& a_undoAction.at(a_undoActionIdx)->at(1) == 'i') {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
setChanged(false, false);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
setChanged(true, false);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ...... for schematic edit mode .......
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_undoActionIdx == 0) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
rebuild(a_undoAction.at(--a_undoActionIdx));
|
2023-12-11 15:13:33 +03:00
|
|
|
|
reloadGraphs(); // load recent simulation data
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
emit signalUndoState(a_undoActionIdx != 0);
|
|
|
|
|
emit signalRedoState(a_undoActionIdx != a_undoAction.size() - 1);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_undoAction.at(a_undoActionIdx)->at(1) == 'i') {
|
|
|
|
|
if (a_undoSymbol.isEmpty()) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
setChanged(false, false);
|
|
|
|
|
return true;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
} else if (a_undoSymbol.at(a_undoSymbolIdx)->at(1) == 'i') {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
setChanged(false, false);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2014-11-08 23:58:37 +08:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
|
|
|
|
setChanged(true, false);
|
|
|
|
|
return true;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
bool Schematic::redo()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_symbolMode) {
|
|
|
|
|
if (a_undoSymbolIdx == a_undoSymbol.size() - 1) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return false;
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
rebuildSymbol(a_undoSymbol.at(++a_undoSymbolIdx));
|
2023-12-11 15:13:33 +03:00
|
|
|
|
adjustPortNumbers(); // set port names
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
emit signalUndoState(a_undoSymbolIdx != 0);
|
|
|
|
|
emit signalRedoState(a_undoSymbolIdx != a_undoSymbol.size() - 1);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_undoSymbol.at(a_undoSymbolIdx)->at(1) == 'i'
|
|
|
|
|
&& a_undoAction.at(a_undoActionIdx)->at(1) == 'i') {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
setChanged(false, false);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
setChanged(true, false);
|
|
|
|
|
return true;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// ...... for schematic edit mode .......
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_undoActionIdx == a_undoAction.size() - 1) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return false;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
rebuild(a_undoAction.at(++a_undoActionIdx));
|
2023-12-11 15:13:33 +03:00
|
|
|
|
reloadGraphs(); // load recent simulation data
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
emit signalUndoState(a_undoActionIdx != 0);
|
|
|
|
|
emit signalRedoState(a_undoActionIdx != a_undoAction.size() - 1);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_undoAction.at(a_undoActionIdx)->at(1) == 'i') {
|
|
|
|
|
if (a_undoSymbol.isEmpty()) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
setChanged(false, false);
|
|
|
|
|
return true;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
} else if (a_undoSymbol.at(a_undoSymbolIdx)->at(1) == 'i') {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
setChanged(false, false);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
setChanged(true, false);
|
|
|
|
|
return true;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
// Sets selected elements on grid.
|
2023-12-11 15:13:33 +03:00
|
|
|
|
bool Schematic::elementsOnGrid()
|
|
|
|
|
{
|
2023-01-15 01:17:09 +03:00
|
|
|
|
int x, y, No;
|
|
|
|
|
bool count = false;
|
|
|
|
|
WireLabel *pl, *pLabel;
|
|
|
|
|
Q3PtrList<WireLabel> LabelCache;
|
|
|
|
|
|
|
|
|
|
// test all components
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Components->setAutoDelete(false);
|
|
|
|
|
for (Component *pc = a_Components->last(); pc != nullptr; pc = a_Components->prev())
|
2023-01-15 01:17:09 +03:00
|
|
|
|
if (pc->isSelected) {
|
|
|
|
|
// rescue non-selected node labels
|
2023-12-11 15:13:33 +03:00
|
|
|
|
for (Port *pp : pc->Ports)
|
2023-01-15 01:17:09 +03:00
|
|
|
|
if (pp->Connection->Label)
|
2024-07-16 18:56:55 +02:00
|
|
|
|
if (pp->Connection->conn_count() < 2) {
|
2023-01-15 01:17:09 +03:00
|
|
|
|
LabelCache.append(pp->Connection->Label);
|
|
|
|
|
pp->Connection->Label->pOwner = 0;
|
|
|
|
|
pp->Connection->Label = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
x = pc->cx;
|
|
|
|
|
y = pc->cy;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
No = a_Components->at();
|
2023-01-15 01:17:09 +03:00
|
|
|
|
deleteComp(pc);
|
|
|
|
|
setOnGrid(pc->cx, pc->cy);
|
|
|
|
|
insertRawComponent(pc);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Components->at(No); // restore current list position
|
2023-01-15 01:17:09 +03:00
|
|
|
|
pc->isSelected = false;
|
|
|
|
|
count = true;
|
|
|
|
|
|
|
|
|
|
x -= pc->cx;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
y -= pc->cy; // re-insert node labels and correct position
|
2023-01-15 01:17:09 +03:00
|
|
|
|
for (pl = LabelCache.first(); pl != 0; pl = LabelCache.next()) {
|
|
|
|
|
pl->cx -= x;
|
|
|
|
|
pl->cy -= y;
|
|
|
|
|
insertNodeLabel(pl);
|
|
|
|
|
}
|
|
|
|
|
LabelCache.clear();
|
2006-05-26 06:03:15 +00:00
|
|
|
|
}
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Components->setAutoDelete(true);
|
2006-05-26 06:03:15 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Wires->setAutoDelete(false);
|
2023-01-15 01:17:09 +03:00
|
|
|
|
// test all wires and wire labels
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (Wire *pw = a_Wires->last(); pw != 0; pw = a_Wires->prev()) {
|
2023-01-15 01:17:09 +03:00
|
|
|
|
pl = pw->Label;
|
|
|
|
|
pw->Label = nullptr;
|
|
|
|
|
|
|
|
|
|
if (pw->isSelected) {
|
|
|
|
|
// rescue non-selected node label
|
|
|
|
|
pLabel = nullptr;
|
|
|
|
|
if (pw->Port1->Label) {
|
2024-07-16 18:56:55 +02:00
|
|
|
|
if (pw->Port1->conn_count() < 2) {
|
2023-01-15 01:17:09 +03:00
|
|
|
|
pLabel = pw->Port1->Label;
|
|
|
|
|
pw->Port1->Label = nullptr;
|
|
|
|
|
}
|
|
|
|
|
} else if (pw->Port2->Label) {
|
2024-07-16 18:56:55 +02:00
|
|
|
|
if (pw->Port2->conn_count() < 2) {
|
2023-01-15 01:17:09 +03:00
|
|
|
|
pLabel = pw->Port2->Label;
|
|
|
|
|
pw->Port2->Label = nullptr;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
No = a_Wires->at();
|
2023-01-15 01:17:09 +03:00
|
|
|
|
deleteWire(pw);
|
|
|
|
|
setOnGrid(pw->x1, pw->y1);
|
|
|
|
|
setOnGrid(pw->x2, pw->y2);
|
|
|
|
|
insertWire(pw);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Wires->at(No); // restore current list position
|
2023-01-15 01:17:09 +03:00
|
|
|
|
pw->isSelected = false;
|
|
|
|
|
count = true;
|
|
|
|
|
if (pl)
|
|
|
|
|
setOnGrid(pl->cx, pl->cy);
|
|
|
|
|
|
|
|
|
|
if (pLabel) {
|
|
|
|
|
setOnGrid(pLabel->cx, pLabel->cy);
|
|
|
|
|
insertNodeLabel(pLabel);
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-01-15 01:17:09 +03:00
|
|
|
|
if (pl) {
|
|
|
|
|
pw->Label = pl;
|
|
|
|
|
if (pl->isSelected) {
|
|
|
|
|
setOnGrid(pl->x1, pl->y1);
|
|
|
|
|
pl->isSelected = false;
|
|
|
|
|
count = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2006-05-26 06:03:15 +00:00
|
|
|
|
}
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_Wires->setAutoDelete(true);
|
2023-01-15 01:17:09 +03:00
|
|
|
|
|
|
|
|
|
// test all node labels
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (Node *pn = a_Nodes->first(); pn != 0; pn = a_Nodes->next())
|
2023-01-15 01:17:09 +03:00
|
|
|
|
if (pn->Label)
|
|
|
|
|
if (pn->Label->isSelected) {
|
|
|
|
|
setOnGrid(pn->Label->x1, pn->Label->y1);
|
|
|
|
|
pn->Label->isSelected = false;
|
|
|
|
|
count = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// test all diagrams
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (Diagram *pd = a_Diagrams->last(); pd != 0; pd = a_Diagrams->prev()) {
|
2023-01-15 01:17:09 +03:00
|
|
|
|
if (pd->isSelected) {
|
|
|
|
|
setOnGrid(pd->cx, pd->cy);
|
|
|
|
|
pd->isSelected = false;
|
|
|
|
|
count = true;
|
|
|
|
|
}
|
2006-05-26 06:03:15 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
for (Graph *pg : pd->Graphs)
|
2023-01-15 01:17:09 +03:00
|
|
|
|
// test markers of diagram
|
2023-12-11 15:13:33 +03:00
|
|
|
|
for (Marker *pm : pg->Markers)
|
2023-01-15 01:17:09 +03:00
|
|
|
|
if (pm->isSelected) {
|
|
|
|
|
x = pm->x1 + pd->cx;
|
|
|
|
|
y = pm->y1 + pd->cy;
|
|
|
|
|
setOnGrid(x, y);
|
|
|
|
|
pm->x1 = x - pd->cx;
|
|
|
|
|
pm->y1 = y - pd->cy;
|
|
|
|
|
pm->isSelected = false;
|
|
|
|
|
count = true;
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-01-15 01:17:09 +03:00
|
|
|
|
// test all paintings
|
2024-11-08 12:46:57 +01:00
|
|
|
|
for (Painting *pa = a_Paintings->last(); pa != 0; pa = a_Paintings->prev())
|
2023-01-15 01:17:09 +03:00
|
|
|
|
if (pa->isSelected) {
|
|
|
|
|
setOnGrid(pa->cx, pa->cy);
|
|
|
|
|
pa->isSelected = false;
|
|
|
|
|
count = true;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (count)
|
|
|
|
|
setChanged(true, true);
|
2023-01-15 01:17:09 +03:00
|
|
|
|
return count;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
void Schematic::switchPaintMode()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_symbolMode = !a_symbolMode; // change mode
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
int tmp, t2;
|
|
|
|
|
float temp;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
temp = a_Scale;
|
|
|
|
|
a_Scale = a_tmpScale;
|
|
|
|
|
a_tmpScale = temp;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
tmp = contentsX();
|
|
|
|
|
t2 = contentsY();
|
2024-11-08 12:46:57 +01:00
|
|
|
|
setContentsPos(a_tmpPosX, a_tmpPosY);
|
|
|
|
|
a_tmpPosX = tmp;
|
|
|
|
|
a_tmpPosY = t2;
|
|
|
|
|
tmp = a_ViewX1;
|
|
|
|
|
a_ViewX1 = a_tmpViewX1;
|
|
|
|
|
a_tmpViewX1 = tmp;
|
|
|
|
|
tmp = a_ViewY1;
|
|
|
|
|
a_ViewY1 = a_tmpViewY1;
|
|
|
|
|
a_tmpViewY1 = tmp;
|
|
|
|
|
tmp = a_ViewX2;
|
|
|
|
|
a_ViewX2 = a_tmpViewX2;
|
|
|
|
|
a_tmpViewX2 = tmp;
|
|
|
|
|
tmp = a_ViewY2;
|
|
|
|
|
a_ViewY2 = a_tmpViewY2;
|
|
|
|
|
a_tmpViewY2 = tmp;
|
|
|
|
|
tmp = a_UsedX1;
|
|
|
|
|
a_UsedX1 = a_tmpUsedX1;
|
|
|
|
|
a_tmpUsedX1 = tmp;
|
|
|
|
|
tmp = a_UsedY1;
|
|
|
|
|
a_UsedY1 = a_tmpUsedY1;
|
|
|
|
|
a_tmpUsedY1 = tmp;
|
|
|
|
|
tmp = a_UsedX2;
|
|
|
|
|
a_UsedX2 = a_tmpUsedX2;
|
|
|
|
|
a_tmpUsedX2 = tmp;
|
|
|
|
|
tmp = a_UsedY2;
|
|
|
|
|
a_UsedY2 = a_tmpUsedY2;
|
|
|
|
|
a_tmpUsedY2 = tmp;
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// *********************************************************************
|
|
|
|
|
// ********** **********
|
|
|
|
|
// ********** Function for serving mouse wheel moving **********
|
|
|
|
|
// ********** **********
|
|
|
|
|
// *********************************************************************
|
|
|
|
|
void Schematic::contentsWheelEvent(QWheelEvent *Event)
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->editText->setHidden(true); // disable edit of component property
|
2024-01-26 11:35:44 +03:00
|
|
|
|
|
|
|
|
|
// A mouse wheel angle delta of a single step is typically 120,
|
|
|
|
|
// but other devices may produce various values. For example,
|
|
|
|
|
// angle values produced by a touchpad depend on how fast user
|
|
|
|
|
// moves their fingers.
|
|
|
|
|
//
|
|
|
|
|
// When used for scrolling the view here angle delta is divided by
|
|
|
|
|
// some number ("2" at the moment). There is nothing special about
|
|
|
|
|
// this number, its sole purpose is to reduce a scroll-step
|
|
|
|
|
// to a reasonable size.
|
|
|
|
|
|
|
|
|
|
// Mouse may have a special wheel for horizontal scrolling
|
|
|
|
|
const int horizontalWheelAngleDelta = Event->angleDelta().x();
|
|
|
|
|
const int verticalWheelAngleDelta = Event->angleDelta().y();
|
|
|
|
|
|
|
|
|
|
// Scroll horizontally
|
|
|
|
|
// Horizontal scroll is performed either by a special wheel
|
|
|
|
|
// or by usual mouse wheel with Shift pressed down.
|
|
|
|
|
if ((Event->modifiers() & Qt::ShiftModifier) || horizontalWheelAngleDelta) {
|
|
|
|
|
int delta = (horizontalWheelAngleDelta ? horizontalWheelAngleDelta : verticalWheelAngleDelta) / 2;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (delta > 0) {
|
2024-01-06 18:11:27 +03:00
|
|
|
|
scrollLeft(delta);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
} else {
|
2024-01-06 18:42:44 +03:00
|
|
|
|
scrollRight(-delta);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-26 11:35:44 +03:00
|
|
|
|
// Zoom in or out
|
|
|
|
|
else if (Event->modifiers() & Qt::ControlModifier) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// zoom factor scaled according to the wheel delta, to accommodate
|
|
|
|
|
// values different from 60 (slower or faster zoom)
|
2024-01-26 11:35:44 +03:00
|
|
|
|
double scaleCoef = pow(1.1, verticalWheelAngleDelta / 60.0);
|
2024-01-09 21:07:31 +03:00
|
|
|
|
const QPoint pointer{
|
2024-01-26 11:35:44 +03:00
|
|
|
|
static_cast<int>(Event->position().x()),
|
|
|
|
|
static_cast<int>(Event->position().y())};
|
|
|
|
|
zoomAroundPoint(scaleCoef, pointer);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2024-01-26 11:35:44 +03:00
|
|
|
|
// Scroll vertically
|
|
|
|
|
else {
|
|
|
|
|
int delta = verticalWheelAngleDelta / 2;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (delta > 0) {
|
2024-01-06 19:05:24 +03:00
|
|
|
|
scrollUp(delta);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
} else {
|
2024-01-06 18:59:07 +03:00
|
|
|
|
scrollDown(-delta);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Event->accept(); // QScrollView must not handle this event
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Scrolls the visible area upwards and enlarges or reduces the view
|
|
|
|
|
// area accordingly.
|
2024-01-06 19:05:24 +03:00
|
|
|
|
void Schematic::scrollUp(int step)
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2024-01-06 19:05:24 +03:00
|
|
|
|
assert(step >= 0);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-01-06 19:05:24 +03:00
|
|
|
|
// Y-axis is directed "from top to bottom": the higher a point is
|
|
|
|
|
// located, the smaller its y-coordinate and vice versa. Keep this in mind
|
|
|
|
|
// while reading the code below.
|
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
const int stepInModel = static_cast<int>(std::round(step/a_Scale));
|
2024-01-06 19:05:24 +03:00
|
|
|
|
const QPoint viewportTopLeft = viewportRect().topLeft();
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-01-06 19:05:24 +03:00
|
|
|
|
// A point currently displayed in top left corner
|
|
|
|
|
QPoint mtl = viewportToModel(viewportTopLeft);
|
|
|
|
|
// A point that should be displayed in top left corner after scrolling
|
|
|
|
|
mtl.setY(mtl.y() - stepInModel);
|
|
|
|
|
|
|
|
|
|
QRect modelBounds = modelRect();
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-01-06 19:05:24 +03:00
|
|
|
|
// If the "should-be-displayed" point is located higher than model upper bound,
|
|
|
|
|
// then extend the model
|
|
|
|
|
modelBounds.setTop(std::min(mtl.y(), modelBounds.top()));
|
|
|
|
|
|
|
|
|
|
// Cut off a bit of unused model space from its bottom side.
|
2024-01-31 16:31:39 +03:00
|
|
|
|
const auto b = modelBounds.bottom() - stepInModel;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (b > a_UsedY2) {
|
2024-01-06 19:05:24 +03:00
|
|
|
|
modelBounds.setBottom(b);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
renderModel(a_Scale, modelBounds, mtl, viewportTopLeft);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Scrolls the visible area downwards and enlarges or reduces the view
|
2024-01-06 18:59:07 +03:00
|
|
|
|
// area accordingly.
|
|
|
|
|
void Schematic::scrollDown(int step)
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2024-01-06 18:59:07 +03:00
|
|
|
|
assert(step >= 0);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-01-06 18:59:07 +03:00
|
|
|
|
// Y-axis is directed "from top to bottom": the lower a point is
|
|
|
|
|
// located, the bigger its y-coordinate and vice versa. Keep this in mind
|
|
|
|
|
// while reading the code below.
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
const int stepInModel = static_cast<int>(std::round(step/a_Scale));
|
2024-01-06 18:59:07 +03:00
|
|
|
|
const QPoint viewportBottomLeft = viewportRect().bottomLeft();
|
|
|
|
|
|
|
|
|
|
// A point currently displayed in bottom left corner
|
|
|
|
|
QPoint mbl = viewportToModel(viewportBottomLeft);
|
|
|
|
|
// A point that should be displayed in bottom left corner after scrolling
|
|
|
|
|
mbl.setY(mbl.y() + stepInModel);
|
|
|
|
|
|
|
|
|
|
QRect modelBounds = modelRect();
|
|
|
|
|
|
|
|
|
|
// If the "should-be-displayed" point is lower than model bottom bound,
|
|
|
|
|
// then extend the model
|
|
|
|
|
modelBounds.setBottom(std::max(mbl.y(), modelBounds.bottom()));
|
|
|
|
|
|
|
|
|
|
// Cut off a bit of unused model space from its top side.
|
2024-01-31 16:31:39 +03:00
|
|
|
|
const auto t = modelBounds.top() + stepInModel;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (t < a_UsedY1) {
|
2024-01-06 18:59:07 +03:00
|
|
|
|
modelBounds.setTop(t);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-01-06 18:59:07 +03:00
|
|
|
|
// Render model in its new size and position point in the top left corner of viewport
|
2024-11-08 12:46:57 +01:00
|
|
|
|
renderModel(a_Scale, modelBounds, mbl, viewportBottomLeft);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Scrolls the visible area to the left and enlarges or reduces the view
|
|
|
|
|
// area accordingly.
|
2024-01-06 18:11:27 +03:00
|
|
|
|
void Schematic::scrollLeft(int step)
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2024-01-06 18:11:27 +03:00
|
|
|
|
assert(step >= 0);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-01-06 18:11:27 +03:00
|
|
|
|
// X-axis is directed "from left to right": the more to the left a point is
|
|
|
|
|
// located, the smaller its x-coordinate and vice versa. Keep this in mind
|
|
|
|
|
// while reading the code below.
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
const int stepInModel = static_cast<int>(std::round(step/a_Scale));
|
2024-01-06 18:11:27 +03:00
|
|
|
|
const QPoint viewportTopLeft = viewportRect().topLeft();
|
|
|
|
|
|
|
|
|
|
// A point currently displayed in top left corner
|
|
|
|
|
QPoint mtl = viewportToModel(viewportTopLeft);
|
|
|
|
|
// A point that should be displayed in top left corner after scrolling
|
|
|
|
|
mtl.setX(mtl.x() - stepInModel);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-01-06 18:11:27 +03:00
|
|
|
|
QRect modelBounds = modelRect();
|
|
|
|
|
|
|
|
|
|
// If the "should-be-displayed" point is to the left of model left bound,
|
|
|
|
|
// then extend the model
|
|
|
|
|
modelBounds.setLeft(std::min(mtl.x(), modelBounds.left()));
|
|
|
|
|
|
|
|
|
|
// Cut off a bit of unused model space from its right side.
|
2024-01-31 16:31:39 +03:00
|
|
|
|
const auto r = modelBounds.right() - stepInModel;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (r > a_UsedX2) {
|
2024-01-06 18:11:27 +03:00
|
|
|
|
modelBounds.setRight(r);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
renderModel(a_Scale, modelBounds, mtl, viewportTopLeft);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Scrolls the visible area to the right and enlarges or reduces the
|
2024-01-06 18:42:44 +03:00
|
|
|
|
// view area accordingly.
|
|
|
|
|
void Schematic::scrollRight(int step)
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2024-01-06 18:42:44 +03:00
|
|
|
|
assert(step >= 0);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-01-06 18:42:44 +03:00
|
|
|
|
// X-axis is directed "from left to right": the more to the right a point is
|
|
|
|
|
// located, the bigger its x-coordinate and vice versa. Keep this in mind
|
|
|
|
|
// while reading the code below.
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
const int stepInModel = static_cast<int>(std::round(step/a_Scale));
|
2024-01-06 18:42:44 +03:00
|
|
|
|
const QPoint viewportTopRight = viewportRect().topRight();
|
|
|
|
|
|
|
|
|
|
// A point currently displayed in top right corner
|
|
|
|
|
QPoint mtr = viewportToModel(viewportTopRight);
|
|
|
|
|
// A point that should be displayed in top right corner after scrolling
|
|
|
|
|
mtr.setX(mtr.x() + stepInModel);
|
|
|
|
|
|
|
|
|
|
QRect modelBounds = modelRect();
|
|
|
|
|
|
|
|
|
|
// If the "should-be-displayed" point is to the right of the model right bound,
|
|
|
|
|
// then extend the model
|
|
|
|
|
modelBounds.setRight(std::max(mtr.x(), modelBounds.right()));
|
|
|
|
|
|
|
|
|
|
// Cut off a bit of unused model space from its left side.
|
2024-01-31 16:31:39 +03:00
|
|
|
|
const auto l = modelBounds.left() + stepInModel;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (l < a_UsedX1) {
|
2024-01-06 18:42:44 +03:00
|
|
|
|
modelBounds.setLeft(l);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
renderModel(a_Scale, modelBounds, mtr, viewportTopRight);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
// Is called if the scroll arrow of the ScrollBar is pressed.
|
|
|
|
|
void Schematic::slotScrollUp()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->editText->setHidden(true); // disable edit of component property
|
2023-12-11 15:13:33 +03:00
|
|
|
|
scrollUp(verticalScrollBar()->singleStep());
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
// Is called if the scroll arrow of the ScrollBar is pressed.
|
|
|
|
|
void Schematic::slotScrollDown()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->editText->setHidden(true); // disable edit of component property
|
2024-01-06 18:59:07 +03:00
|
|
|
|
scrollDown(verticalScrollBar()->singleStep());
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
// Is called if the scroll arrow of the ScrollBar is pressed.
|
|
|
|
|
void Schematic::slotScrollLeft()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->editText->setHidden(true); // disable edit of component property
|
2023-12-11 15:13:33 +03:00
|
|
|
|
scrollLeft(horizontalScrollBar()->singleStep());
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -----------------------------------------------------------
|
|
|
|
|
// Is called if the scroll arrow of the ScrollBar is pressed.
|
|
|
|
|
void Schematic::slotScrollRight()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->editText->setHidden(true); // disable edit of component property
|
2024-01-06 18:42:44 +03:00
|
|
|
|
scrollRight(horizontalScrollBar()->singleStep());
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// *********************************************************************
|
|
|
|
|
// ********** **********
|
|
|
|
|
// ********** Function for serving drag'n drop **********
|
|
|
|
|
// ********** **********
|
|
|
|
|
// *********************************************************************
|
|
|
|
|
|
|
|
|
|
// Is called if an object is dropped (after drag'n drop).
|
|
|
|
|
void Schematic::contentsDropEvent(QDropEvent *Event)
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (a_dragIsOkay) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
QList<QUrl> urls = Event->mimeData()->urls();
|
|
|
|
|
if (urls.isEmpty()) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2006-07-31 06:04:54 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// do not close untitled document to avoid segfault
|
|
|
|
|
QucsDoc *d = QucsMain->getDoc(0);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
bool changed = d->getDocChanged();
|
|
|
|
|
d->setDocChanged(true);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// URI: file:/home/linuxuser/Desktop/example.sch
|
|
|
|
|
for (QUrl url : urls) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->gotoPage(QDir::toNativeSeparators(url.toLocalFile()));
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
d->setDocChanged(changed);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return;
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
auto ev_pos = Event->position();
|
2024-02-25 22:29:16 +03:00
|
|
|
|
QPoint inModel = contentsToModel(ev_pos.toPoint());
|
2024-07-29 11:02:30 +03:00
|
|
|
|
//QMouseEvent e(QEvent::MouseButtonPress, ev_pos, Qt::LeftButton, Qt::NoButton, Qt::NoModifier);
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-07-29 11:02:30 +03:00
|
|
|
|
QMouseEvent e(QEvent::MouseButtonPress, ev_pos, mapToGlobal(ev_pos), Qt::LeftButton, Qt::NoButton, Qt::NoModifier);
|
2024-02-25 22:29:16 +03:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->view->MPressElement(this, &e, inModel.x(), inModel.y());
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-11-08 12:46:57 +01:00
|
|
|
|
delete a_App->view->selElem;
|
|
|
|
|
a_App->view->selElem = nullptr; // no component selected
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-06-15 19:24:02 +03:00
|
|
|
|
if (formerAction) {
|
|
|
|
|
formerAction->setChecked(true);
|
|
|
|
|
} else {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
QucsMain->select->setChecked(true); // restore old action
|
2023-06-15 19:24:02 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
void Schematic::contentsDragEnterEvent(QDragEnterEvent *Event)
|
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
//FIXME: the function of drag library component seems not working?
|
|
|
|
|
formerAction = nullptr;
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_dragIsOkay = false;
|
2006-07-17 06:02:57 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// file dragged in ?
|
|
|
|
|
if (Event->mimeData()->hasUrls()) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_dragIsOkay = true;
|
2014-10-30 16:47:17 +08:00
|
|
|
|
Event->accept();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2006-07-03 06:02:08 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// drag library component
|
|
|
|
|
if (Event->mimeData()->hasText()) {
|
|
|
|
|
QString s = Event->mimeData()->text();
|
|
|
|
|
if (s.left(15) == "QucsComponent:<") {
|
|
|
|
|
s = s.mid(14);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->view->selElem = getComponentFromName(s);
|
|
|
|
|
if (a_App->view->selElem) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
Event->accept();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Event->ignore();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// if(Event->format(1) == 0) { // only one MIME type ?
|
2006-07-03 06:02:08 +00:00
|
|
|
|
|
|
|
|
|
// drag component from listview
|
2022-02-14 15:37:11 +01:00
|
|
|
|
//if(Event->provides("application/x-qabstractitemmodeldatalist")) {
|
|
|
|
|
if (Event->mimeData()->hasFormat("application/x-qabstractitemmodeldatalist")) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
QListWidgetItem *Item = a_App->CompComps->currentItem();
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (Item) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
formerAction = a_App->activeAction;
|
|
|
|
|
a_App->slotSelectComponent(Item); // also sets drawn=false
|
|
|
|
|
a_App->MouseMoveAction = 0;
|
|
|
|
|
a_App->MousePressAction = 0;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
|
|
|
|
|
Event->accept();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
2023-12-11 15:13:33 +03:00
|
|
|
|
// }
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
Event->ignore();
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------
|
2023-12-11 15:13:33 +03:00
|
|
|
|
void Schematic::contentsDragLeaveEvent(QDragLeaveEvent *)
|
2006-03-28 06:10:52 +00:00
|
|
|
|
{
|
2023-12-11 15:13:33 +03:00
|
|
|
|
if (formerAction)
|
|
|
|
|
formerAction->setChecked(true); // restore old action
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
2024-07-08 22:35:23 +03:00
|
|
|
|
void Schematic::contentsNativeGestureZoomEvent( QNativeGestureEvent* Event) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->editText->setHidden(true); // disable edit of component property
|
2024-07-08 22:35:23 +03:00
|
|
|
|
|
|
|
|
|
const auto factor = 1.0 + Event->value();
|
|
|
|
|
const auto pointer = mapFromGlobal(Event->globalPosition().toPoint());
|
|
|
|
|
zoomAroundPoint(factor,pointer);
|
|
|
|
|
}
|
|
|
|
|
|
2006-03-28 06:10:52 +00:00
|
|
|
|
// ---------------------------------------------------
|
|
|
|
|
void Schematic::contentsDragMoveEvent(QDragMoveEvent *Event)
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
if (!a_dragIsOkay) {
|
|
|
|
|
if (a_App->view->selElem == 0) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
Event->ignore();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2024-07-29 11:02:30 +03:00
|
|
|
|
auto ev_pos = Event->position();
|
|
|
|
|
/*QMouseEvent e(QEvent::MouseMove,
|
2023-12-11 15:13:33 +03:00
|
|
|
|
Event->position(),
|
|
|
|
|
Qt::NoButton,
|
|
|
|
|
Qt::NoButton,
|
2024-07-29 11:02:30 +03:00
|
|
|
|
Qt::NoModifier);*/
|
|
|
|
|
QMouseEvent e(QEvent::MouseButtonPress, ev_pos, mapToGlobal(ev_pos), Qt::LeftButton, Qt::NoButton, Qt::NoModifier);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_App->view->MMoveElement(this, &e);
|
2023-12-11 15:13:33 +03:00
|
|
|
|
}
|
2006-03-28 06:10:52 +00:00
|
|
|
|
|
2023-12-11 15:13:33 +03:00
|
|
|
|
Event->accept();
|
2006-03-28 06:10:52 +00:00
|
|
|
|
}
|
2013-12-20 18:58:52 +04:00
|
|
|
|
|
2022-07-16 12:02:50 +03:00
|
|
|
|
bool Schematic::checkDplAndDatNames()
|
|
|
|
|
{
|
2024-11-08 12:46:57 +01:00
|
|
|
|
QFileInfo Info(a_DocName);
|
|
|
|
|
if (!a_DocName.isEmpty() && a_DataSet.size() > 4 && a_DataDisplay.size() > 4) {
|
2023-12-11 15:13:33 +03:00
|
|
|
|
QString base = Info.completeBaseName();
|
2024-11-08 12:46:57 +01:00
|
|
|
|
QString base_dat = a_DataSet;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
base_dat.chop(4);
|
2024-11-08 12:46:57 +01:00
|
|
|
|
QString base_dpl = a_DataDisplay;
|
2023-12-11 15:13:33 +03:00
|
|
|
|
base_dpl.chop(4);
|
|
|
|
|
if (base != base_dat || base != base_dpl) {
|
|
|
|
|
QString msg = QObject::tr(
|
|
|
|
|
"The schematic name and dataset/display file name is not matching! "
|
|
|
|
|
"This may happen if schematic was copied using the file manager "
|
|
|
|
|
"instead of using File->SaveAs. Correct dataset and display names "
|
|
|
|
|
"automatically?\n\n");
|
|
|
|
|
msg += QString(QObject::tr("Schematic file: ")) + base + ".sch\n";
|
2024-11-08 12:46:57 +01:00
|
|
|
|
msg += QString(QObject::tr("Dataset file: ")) + a_DataSet + "\n";
|
|
|
|
|
msg += QString(QObject::tr("Display file: ")) + a_DataDisplay + "\n";
|
2023-12-11 15:13:33 +03:00
|
|
|
|
auto r = QMessageBox::information(this,
|
|
|
|
|
QObject::tr("Open document"),
|
|
|
|
|
msg,
|
|
|
|
|
QMessageBox::Yes,
|
|
|
|
|
QMessageBox::No);
|
|
|
|
|
if (r == QMessageBox::Yes) {
|
2024-11-08 12:46:57 +01:00
|
|
|
|
a_DataSet = base + ".dat";
|
|
|
|
|
a_DataDisplay = base + ".dpl";
|
2023-12-11 15:13:33 +03:00
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-07-16 12:02:50 +03:00
|
|
|
|
} else {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
2024-03-12 21:04:10 -04:00
|
|
|
|
}
|