diff --git a/include/Component.hpp b/include/Component.hpp index dd786bc..70b7b6f 100644 --- a/include/Component.hpp +++ b/include/Component.hpp @@ -10,6 +10,9 @@ class Component : public QFrame { public: explicit Component(QWidget* parent); + explicit Component(QWidget* parent, const QString& resource); + + void mouseMoveEvent(QMouseEvent* event) override; private: Ui::Component* ui; diff --git a/include/GhostLabel.hpp b/include/GhostLabel.hpp new file mode 100644 index 0000000..2013bf9 --- /dev/null +++ b/include/GhostLabel.hpp @@ -0,0 +1,19 @@ +#pragma once + +#include + +class GhostLabel : public QLabel +{ +public: + explicit GhostLabel(QWidget* parent) : + QLabel(parent) + { + setMouseTracking(true); + } + + void mouseMoveEvent(QMouseEvent* event) override + { + // Ignore all mouse move events, who cares tbh + event->ignore(); + } +}; \ No newline at end of file diff --git a/include/Window.hpp b/include/Window.hpp index cda3f7b..9026803 100644 --- a/include/Window.hpp +++ b/include/Window.hpp @@ -7,6 +7,8 @@ namespace Ui { } class Component; +class QActionGroup; +class GhostLabel; class Window : public QMainWindow { @@ -21,6 +23,8 @@ protected: private: void ToggleSimulation(); + void ToggleComponentPlacer(); + void LoadGhostLabel(const QString& resource); Ui::Window* ui; bool simulating; @@ -29,4 +33,8 @@ private: Component* component; QPoint relativePos; } dragInfo; + + GhostLabel* ghostImage; + QString resourcePath; + QActionGroup* componentGroup; }; \ No newline at end of file diff --git a/src/Component.cpp b/src/Component.cpp index 56c2353..e70a95a 100644 --- a/src/Component.cpp +++ b/src/Component.cpp @@ -1,9 +1,27 @@ #include "Component.hpp" #include "ui_Component.h" +#include +#include + Component::Component(QWidget* parent) : + Component(parent, ":/components/and.png") +{ + +} + +Component::Component(QWidget* parent, const QString& resource) : QFrame(parent), ui(nullptr) { ui = new Ui::Component(); ui->setupUi(this); -} \ No newline at end of file + + ui->label->setPixmap(QPixmap(":/components/and.png")); + + std::cout << "Created Component --" << ui->label->pos().x() << ", " << ui->label->pos().y() << std::endl; +} + +void Component::mouseMoveEvent(QMouseEvent* event) +{ + event->ignore(); +} diff --git a/src/Window.cpp b/src/Window.cpp index 1d41a62..5fe28eb 100644 --- a/src/Window.cpp +++ b/src/Window.cpp @@ -1,16 +1,35 @@ #include "Window.hpp" #include "ui_Window.h" +#include "Component.hpp" + #include #include +#include "GhostLabel.hpp" +#include +#include +#include Window::Window() : - QMainWindow(), ui(nullptr), simulating(false) + QMainWindow(), ui(nullptr), simulating(false), dragInfo{nullptr, QPoint()}, + ghostImage(nullptr), componentGroup(nullptr) { ui = new Ui::Window(); ui->setupUi(this); + ui->centralwidget->setAttribute(Qt::WA_TransparentForMouseEvents); + setMouseTracking(true); + + LoadGhostLabel(":/components/and.png"); + + componentGroup = new QActionGroup(this); + componentGroup->addAction(ui->actionAddAND); + componentGroup->addAction(ui->actionAddOR); + componentGroup->setExclusionPolicy(QActionGroup::ExclusionPolicy::ExclusiveOptional); + connect(ui->actionStart, &QAction::triggered, this, &Window::ToggleSimulation); + connect(ui->actionAddAND, &QAction::triggered, this, [this]() {resourcePath = ":/components/and.png"; ToggleComponentPlacer(); }); + connect(ui->actionAddOR, &QAction::triggered, this, [this]() {resourcePath = ":/components/or.png"; ToggleComponentPlacer(); }); } Window::~Window() @@ -25,20 +44,47 @@ void Window::mousePressEvent(QMouseEvent* event) ui->centralwidget->findChildren(); QFrame* child = static_cast(ui->centralwidget->childAt(event->pos())); - if (child == nullptr) - return; - if (!child->isAncestorOf(ui->centralwidget)) - child = static_cast(child->parentWidget()); + QAction* toggledAction = componentGroup->checkedAction(); + if (toggledAction == nullptr) + { + if (child == nullptr) + return; - dragInfo.component = static_cast(child); - dragInfo.relativePos = event->pos() - child->pos(); + if (!child->isAncestorOf(ui->centralwidget)) + child = static_cast(child->parentWidget()); + + dragInfo.component = static_cast(child); + dragInfo.relativePos = event->pos() - child->pos(); + } + else + { + if (child != nullptr) + return; + + // So Qt can't add widgets outside of the constructor apparently + // So component cannot be a QFrame, it will have to become just a normal object + // And I need one of these open gl gsjdööööööööööööööl< + // TODO: I'll just let this leak for now, I just wanna eat dinner... + Component* leak = new Component(this, resourcePath); + QPoint pos = event->pos() - QPoint {50, 25}; + leak->setGeometry(QRect(280, 160, 100, 50)); + leak->setFrameShape(QFrame::StyledPanel); + leak->setFrameShadow(QFrame::Raised); + + std::cout << leak->pos().x() << ", " << leak->pos().y() << std::endl; + } } void Window::mouseMoveEvent(QMouseEvent* event) { if (dragInfo.component != nullptr) dragInfo.component->move(event->pos() - dragInfo.relativePos); + + if (ghostImage != nullptr) + ghostImage->move(event->pos() - QPoint{50, 25}); // TODO: Hardcoded + + event->accept(); } void Window::mouseReleaseEvent(QMouseEvent* event) @@ -51,3 +97,36 @@ void Window::ToggleSimulation() simulating = !simulating; ui->actionStart->setIcon(QIcon(simulating ? ":/toolbar/stop.png" : ":/toolbar/start.png")); } + +void Window::ToggleComponentPlacer() +{ + QAction* toggledAction = componentGroup->checkedAction(); + if (toggledAction == nullptr) + { + delete ghostImage; + ghostImage = nullptr; + } + else + { + LoadGhostLabel(resourcePath); + } +} + +void Window::LoadGhostLabel(const QString& resource) +{ + if (ghostImage == nullptr) { + ghostImage = new GhostLabel(this); + ghostImage->setGeometry({ 0, 0, 100, 50 }); + ghostImage->setScaledContents(true); + } + + QPainter p; + QImage image(resource); + + p.begin(&image); + p.setCompositionMode(QPainter::CompositionMode_DestinationIn); + p.fillRect(image.rect(), QColor(0, 0, 0, 50)); + p.end(); + + ghostImage->setPixmap(QPixmap::fromImage(image)); +} diff --git a/ui/Component.ui b/ui/Component.ui index a7c334c..7143d27 100644 --- a/ui/Component.ui +++ b/ui/Component.ui @@ -6,8 +6,8 @@ 0 0 - 100 - 100 + 102 + 50 @@ -21,6 +21,9 @@ 19 + + PointingHandCursor + Form @@ -36,15 +39,12 @@ 0 0 100 - 100 + 50 - - :/components/and.png - true diff --git a/ui/Window.ui b/ui/Window.ui index 5c5a2ef..25b9628 100644 --- a/ui/Window.ui +++ b/ui/Window.ui @@ -6,8 +6,8 @@ 0 0 - 800 - 600 + 924 + 700 @@ -20,23 +20,17 @@ - 170 + 360 180 - 100 - 100 + 120 + 80 - - false - - - - - QFrame::NoFrame + QFrame::StyledPanel - QFrame::Plain + QFrame::Raised @@ -52,6 +46,9 @@ false + + + @@ -68,6 +65,46 @@ F5 + + + true + + + true + + + + :/components/and.png:/components/and.png + + + AddAND + + + Create an AND gate + + + 1 + + + + + true + + + + :/components/or.png + :/components/and.png:/components/or.png + + + AddOR + + + Create an OR gate + + + 2 + + diff --git a/ui/assets/README.md b/ui/assets/README.md index 2bf3eb1..502fa00 100644 --- a/ui/assets/README.md +++ b/ui/assets/README.md @@ -1 +1,6 @@ -Start/Stop icons taken from https://iconsmind.com/ (modified) \ No newline at end of file +Start/Stop icons taken from https://iconsmind.com/ (modified) + +Logic Gates taken from Wikimedia Commons. The authors are +* jjbeard (AND, OR) + +The Wikimedia icons are part of the public domain \ No newline at end of file diff --git a/ui/assets/and.png b/ui/assets/and.png index 3cea701..de2a7e5 100644 Binary files a/ui/assets/and.png and b/ui/assets/and.png differ diff --git a/ui/assets/or.png b/ui/assets/or.png new file mode 100644 index 0000000..e5dcb07 Binary files /dev/null and b/ui/assets/or.png differ diff --git a/ui/assets/resources.qrc b/ui/assets/resources.qrc index f8c581d..5ac7291 100644 --- a/ui/assets/resources.qrc +++ b/ui/assets/resources.qrc @@ -5,5 +5,6 @@ and.png + or.png