From d4d487298c25b16cebf873374428ecbc0d56cb75 Mon Sep 17 00:00:00 2001 From: Andrey Kalmykov Date: Thu, 6 Feb 2025 18:10:00 +0100 Subject: [PATCH 1/4] Add wire planner --- qucs/CMakeLists.txt | 3 +- qucs/wire_planner.cpp | 107 ++++++++++++++++++++++++++++++++++++++++++ qucs/wire_planner.h | 27 +++++++++++ 3 files changed, 136 insertions(+), 1 deletion(-) create mode 100644 qucs/wire_planner.cpp create mode 100644 qucs/wire_planner.h diff --git a/qucs/CMakeLists.txt b/qucs/CMakeLists.txt index 3a5ac163..a19bfbb3 100755 --- a/qucs/CMakeLists.txt +++ b/qucs/CMakeLists.txt @@ -147,7 +147,7 @@ SET(QUCS_SRCS syntax.cpp misc.cpp messagedock.cpp settings.cpp imagewriter.cpp printerwriter.cpp projectView.cpp - symbolwidget.cpp + symbolwidget.cpp wire_planner.cpp ) SET(QUCS_HDRS @@ -170,6 +170,7 @@ symbolwidget.h textdoc.h wire.h wirelabel.h +wire_planner.h ) # diff --git a/qucs/wire_planner.cpp b/qucs/wire_planner.cpp new file mode 100644 index 00000000..53d9a7b2 --- /dev/null +++ b/qucs/wire_planner.cpp @@ -0,0 +1,107 @@ +#include "wire_planner.h" + +namespace qucs_s { +namespace wire { + +inline bool is_horizontal_or_vertical(const QPoint from, const QPoint to) { + return from.x() == to.x() || from.y() == to.y(); +} + +// TODO: migrate to C++20 and yonger and use std::midpoint instead +int midpoint(int a, int b) { + return a + (b - a) / 2; +} + + +std::vector straight(const QPoint from, const QPoint to) { + return {from, to}; +} + +std::vector two_step_xy(const QPoint from, const QPoint to) { + if (is_horizontal_or_vertical(from, to)) { + return {from, to}; + }; + + /* + From o---+ + | + +---o To + */ + return {from, {to.x(), from.y()}, to}; +} + +std::vector two_step_yx(const QPoint from, const QPoint to) { + if (is_horizontal_or_vertical(from, to)) { + return {from, to}; + }; + + /* + From o + | + +---o To + */ + return {from, {from.x(), to.y()}, to}; +} + +std::vector three_step_xy(const QPoint from, const QPoint to) { + if (is_horizontal_or_vertical(from, to)) { + return {from, to}; + }; + + /* + From o---+ + | + +---o To + */ + int mid_x = midpoint(from.x(), to.x()); + return {from, {mid_x, from.y()}, {mid_x, to.y()}, to}; +} + +std::vector three_step_yx(const QPoint from, const QPoint to) { + if (is_horizontal_or_vertical(from, to)) { + return {from, to}; + }; + + /* + o From + | + +---+ + | + o To + */ + int mid_y = midpoint(from.y(), to.y()); + return {from, {from.x(), mid_y}, {to.x(), mid_y}, to}; +} + + +static const std::map routers = { + {Planner::PlanType::TwoStepXY, two_step_xy}, + {Planner::PlanType::TwoStepYX, two_step_yx}, + {Planner::PlanType::ThreeStepXY,three_step_xy}, + {Planner::PlanType::ThreeStepYX, three_step_yx} +}; + + +Planner::Planner() : current{routers.begin()} {} + +std::vector Planner::plan(PlanType type, const QPoint from, const QPoint to) { + return routers.at(type)(from, to); +} + +std::vector Planner::plan(const QPoint from, const QPoint to) const { + return current->second(from, to); +} + +void Planner::next() { + ++current; + + if (current != routers.cend()) { + return; + } + + current = routers.cbegin(); +} + + +} // namespace wire +} // namespace qucs_s diff --git a/qucs/wire_planner.h b/qucs/wire_planner.h new file mode 100644 index 00000000..d616f96a --- /dev/null +++ b/qucs/wire_planner.h @@ -0,0 +1,27 @@ +#ifndef WIRE_PLANNER_H +#define WIRE_PLANNER_H + +#include +#include +#include + +namespace qucs_s { +namespace wire { + +class Planner { + public: + using RouterFunc = std::function(QPoint, QPoint)>; + enum class PlanType { TwoStepXY, TwoStepYX, ThreeStepXY, ThreeStepYX }; + + Planner(); + static std::vector plan(PlanType type, const QPoint from, const QPoint to); + std::vector plan(const QPoint from, const QPoint to) const; + void next(); + private: + std::map::const_iterator current; +}; + +} // namespace wire +} // namespace qucs_s + +#endif From 43aa0f518bfb68d0d446d826734cf27416f03f0e Mon Sep 17 00:00:00 2001 From: Andrey Kalmykov Date: Thu, 6 Feb 2025 18:11:00 +0100 Subject: [PATCH 2/4] Add Schematic::connectWithWire() and Schematic::showEphemeralWire --- qucs/schematic.h | 5 +++++ qucs/schematic_element.cpp | 33 +++++++++++++++++++++++++++++++++ 2 files changed, 38 insertions(+) diff --git a/qucs/schematic.h b/qucs/schematic.h index 95daa041..bfe4c95e 100644 --- a/qucs/schematic.h +++ b/qucs/schematic.h @@ -33,6 +33,7 @@ #include "diagrams/diagram.h" #include "paintings/painting.h" #include "components/component.h" +#include "wire_planner.h" #include "qt3_compat/qt_compat.h" #include "qt3_compat/q3scrollview.h" @@ -431,6 +432,10 @@ public: Node* insertNode(int, int, Element*); Node* selectedNode(int, int); + qucs_s::wire::Planner a_wirePlanner; + std::pair connectWithWire(const QPoint& a, const QPoint& b) noexcept; + void showEphemeralWire(const QPoint& a, const QPoint& b) noexcept; + int insertWireNode1(Wire*); bool connectHWires1(Wire*); bool connectVWires1(Wire*); diff --git a/qucs/schematic_element.cpp b/qucs/schematic_element.cpp index e1992555..07813d55 100644 --- a/qucs/schematic_element.cpp +++ b/qucs/schematic_element.cpp @@ -3253,4 +3253,37 @@ void Schematic::copyPaintings(int& x1, int& y1, int& x2, int& y2, else pp = a_Paintings->next(); } +std::pair Schematic::connectWithWire(const QPoint& a, const QPoint& b) noexcept { + auto points = a_wirePlanner.plan(a, b); + + int resultFlags = 0; + // Take points by pairs + for (std::size_t i = 1; i < points.size(); i++) { + auto m = points[i-1]; + auto n = points[i]; + resultFlags = resultFlags | insertWire(new Wire(m.x(), m.y(), n.x(), n.y())); + } + + const auto lastPoint = points.back(); + for (auto* node : *a_Nodes) { + if (node->cx == lastPoint.x() && node->cy == lastPoint.y()) { + return {resultFlags, node}; + } + } + + assert(false); // Must never get there + return {resultFlags, nullptr}; +} + +void Schematic::showEphemeralWire(const QPoint& a, const QPoint& b) noexcept { + auto points = a_wirePlanner.plan(a, b); + // Take points by pairs + for (std::size_t i = 1; i < points.size(); i++) { + auto m = points[i-1]; + auto n = points[i]; + PostPaintEvent(_Line, m.x(), m.y(), n.x(), n.y()); + } + +} + // vim:ts=8:sw=2:noet From 6fb5f15c57ed2d61f4f900900bd987d2c4bdb1b4 Mon Sep 17 00:00:00 2001 From: Andrey Kalmykov Date: Thu, 6 Feb 2025 18:12:00 +0100 Subject: [PATCH 3/4] Add new wire routes when laying a new wire --- qucs/mouseactions.cpp | 49 +++++++------------------------------------ 1 file changed, 8 insertions(+), 41 deletions(-) diff --git a/qucs/mouseactions.cpp b/qucs/mouseactions.cpp index f0ecea6b..f8809746 100644 --- a/qucs/mouseactions.cpp +++ b/qucs/mouseactions.cpp @@ -414,16 +414,7 @@ void MouseActions::MMoveWire2(Schematic *Doc, QMouseEvent *Event) Doc->setOnGrid(MAx2, MAy2); paintAim(Doc, MAx2, MAy2); //let we paint aim cross - //because cross slightly masks a wire, let we make wire thicker - //better to make it by increasing of pen, but here we cannot access - //pen - if (MAx1 == 0) { - paintGhostLineV(Doc, MAx3, MAy3, MAy2); - paintGhostLineH(Doc, MAx3, MAy2, MAx2); - } else { - paintGhostLineH(Doc, MAx3, MAy3, MAx2); - paintGhostLineV(Doc, MAx2, MAy3, MAy2); - } + Doc->showEphemeralWire({MAx3, MAy3}, {MAx2, MAy2}); QucsMain->MouseDoubleClickAction = &MouseActions::MDoubleClickWire2; Doc->viewport()->update(); @@ -1498,26 +1489,11 @@ void MouseActions::MPressWire1(Schematic *Doc, QMouseEvent *, float fX, float fY */ void MouseActions::MPressWire2(Schematic *Doc, QMouseEvent *Event, float fX, float fY) { - int set1 = 0, set2 = 0; switch (Event->button()) { - case Qt::LeftButton: - if (MAx1 == 0) { // which wire direction first ? - if (MAy2 != MAy3) - set1 = Doc->insertWire(new Wire(MAx3, MAy3, MAx3, MAy2)); - if (MAx2 != MAx3) { - set2 = set1; - set1 = Doc->insertWire(new Wire(MAx3, MAy2, MAx2, MAy2)); - } - } else { - if (MAx2 != MAx3) - set1 = Doc->insertWire(new Wire(MAx3, MAy3, MAx2, MAy3)); - if (MAy2 != MAy3) { - set2 = set1; - set1 = Doc->insertWire(new Wire(MAx2, MAy3, MAx2, MAy2)); - } - } + case Qt::LeftButton: { + auto [hasChanges, lastNode] = Doc->connectWithWire({MAx3, MAy3}, {MAx2, MAy2}); - if (set1 & 2) { + if (lastNode->conn_count() > 1) { // if last port is connected, then... if (formerAction) { // ...restore old action @@ -1530,18 +1506,15 @@ void MouseActions::MPressWire2(Schematic *Doc, QMouseEvent *Event, float fX, flo } } - //ALYS: excessive update. end of function does it. - //Doc->viewport()->update(); - - if (set1 | set2) + if (hasChanges) Doc->setChanged(true, true); MAx3 = MAx2; MAy3 = MAy2; break; - + } /// \todo document right mouse button changes the wire corner case Qt::RightButton: - + Doc->a_wirePlanner.next(); #if 0 //ALYS - old code preserved because isn't clear - what it was??? //looks like deletion via painting. @@ -1561,13 +1534,7 @@ void MouseActions::MPressWire2(Schematic *Doc, QMouseEvent *Event, float fX, flo Doc->setOnGrid(MAx2, MAy2); MAx1 ^= 1; // change the painting direction of wire corner - if (MAx1 == 0) { - paintGhostLineV(Doc, MAx3, MAy3, MAy2); - paintGhostLineH(Doc, MAx3, MAy2, MAx2); - } else { - paintGhostLineH(Doc, MAx3, MAy3, MAx2); - paintGhostLineV(Doc, MAx2, MAy3, MAy2); - } + Doc->showEphemeralWire({MAx3, MAy3}, {MAx2, MAy2}); break; default:; // avoids compiler warnings From 8f0fc384870f5be26a6bd125c2e7e7bf7df6d3af Mon Sep 17 00:00:00 2001 From: Andrey Kalmykov Date: Fri, 7 Feb 2025 15:53:34 +0300 Subject: [PATCH 4/4] Fix crash when new wire ends at the other wire's end --- qucs/mouseactions.cpp | 2 +- qucs/schematic_element.cpp | 16 +++++++++++++++- 2 files changed, 16 insertions(+), 2 deletions(-) diff --git a/qucs/mouseactions.cpp b/qucs/mouseactions.cpp index f8809746..84b4e25f 100644 --- a/qucs/mouseactions.cpp +++ b/qucs/mouseactions.cpp @@ -1493,7 +1493,7 @@ void MouseActions::MPressWire2(Schematic *Doc, QMouseEvent *Event, float fX, flo case Qt::LeftButton: { auto [hasChanges, lastNode] = Doc->connectWithWire({MAx3, MAy3}, {MAx2, MAy2}); - if (lastNode->conn_count() > 1) { + if (lastNode == nullptr || lastNode->conn_count() > 1) { // if last port is connected, then... if (formerAction) { // ...restore old action diff --git a/qucs/schematic_element.cpp b/qucs/schematic_element.cpp index 07813d55..d8420279 100644 --- a/qucs/schematic_element.cpp +++ b/qucs/schematic_element.cpp @@ -3271,7 +3271,21 @@ std::pair Schematic::connectWithWire(const QPoint& a, const QPoint& } } - assert(false); // Must never get there + // If we've got there, then the point at which the wire ended + // was optimized away. For example if there was a wire AB… + // + // A B + // o--------o + // + // And a new wire was drawn from point M to point A: + // + // M A B + // o------o--------o + // + // then two wires wouild be merged and point A would be removed: + // + // M B + // o---------------o return {resultFlags, nullptr}; }