diff --git a/include/Window.hpp b/include/Window.hpp index 9026803..ab06d71 100644 --- a/include/Window.hpp +++ b/include/Window.hpp @@ -25,6 +25,7 @@ private: void ToggleSimulation(); void ToggleComponentPlacer(); void LoadGhostLabel(const QString& resource); + Component* componentAt(const QPoint& point); Ui::Window* ui; bool simulating; @@ -37,4 +38,6 @@ private: GhostLabel* ghostImage; QString resourcePath; QActionGroup* componentGroup; + + std::vector components; }; \ No newline at end of file diff --git a/src/Window.cpp b/src/Window.cpp index 29f6cba..f33bb90 100644 --- a/src/Window.cpp +++ b/src/Window.cpp @@ -20,30 +20,49 @@ Window::Window() : ui->centralwidget->setAttribute(Qt::WA_TransparentForMouseEvents); setMouseTracking(true); - LoadGhostLabel(":/components/and.png"); - componentGroup = new QActionGroup(this); + componentGroup->addAction(ui->actionAddNOT); componentGroup->addAction(ui->actionAddAND); + componentGroup->addAction(ui->actionAddNAND); componentGroup->addAction(ui->actionAddOR); + componentGroup->addAction(ui->actionAddNOR); + componentGroup->addAction(ui->actionAddXOR); + componentGroup->addAction(ui->actionAddXNOR); 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(); }); + connect(ui->actionStart, &QAction::triggered, this, &Window::ToggleSimulation); + connect(ui->actionAddNOT, &QAction::triggered, this, [this]() {resourcePath = ":/components/not.png"; ToggleComponentPlacer(); }); + connect(ui->actionAddAND, &QAction::triggered, this, [this]() {resourcePath = ":/components/and.png"; ToggleComponentPlacer(); }); + connect(ui->actionAddNAND, &QAction::triggered, this, [this]() {resourcePath = ":/components/nand.png"; ToggleComponentPlacer(); }); + connect(ui->actionAddOR, &QAction::triggered, this, [this]() {resourcePath = ":/components/or.png"; ToggleComponentPlacer(); }); + connect(ui->actionAddNOR, &QAction::triggered, this, [this]() {resourcePath = ":/components/nor.png"; ToggleComponentPlacer(); }); + connect(ui->actionAddXOR, &QAction::triggered, this, [this]() {resourcePath = ":/components/xor.png"; ToggleComponentPlacer(); }); + connect(ui->actionAddXNOR, &QAction::triggered, this, [this]() {resourcePath = ":/components/xnor.png"; ToggleComponentPlacer(); }); } Window::~Window() { delete ui; ui = nullptr; + + delete componentGroup; + componentGroup = nullptr; + + delete ghostImage; + ghostImage = nullptr; + + for (Component* c : components) + { + delete c; + c = nullptr; + } } void Window::mousePressEvent(QMouseEvent* event) { if (simulating) return; - ui->centralwidget->findChildren(); - QFrame* child = static_cast(ui->centralwidget->childAt(event->pos())); + Component* child = componentAt(event->pos()); QAction* toggledAction = componentGroup->checkedAction(); if (toggledAction == nullptr) @@ -51,9 +70,6 @@ void Window::mousePressEvent(QMouseEvent* event) if (child == nullptr) return; - if (!child->isAncestorOf(ui->centralwidget)) - child = static_cast(child->parentWidget()); - dragInfo.component = static_cast(child); dragInfo.relativePos = event->pos() - child->pos(); } @@ -62,13 +78,15 @@ void Window::mousePressEvent(QMouseEvent* event) if (child != nullptr) return; - // 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(pos.x(), pos.y(), 100, 50)); leak->setFrameShape(QFrame::NoFrame); leak->setFrameShadow(QFrame::Plain); leak->setVisible(true); + leak->setAttribute(Qt::WA_TransparentForMouseEvents); + + components.push_back(leak); } } @@ -78,7 +96,7 @@ void Window::mouseMoveEvent(QMouseEvent* event) dragInfo.component->move(event->pos() - dragInfo.relativePos); if (ghostImage != nullptr) - ghostImage->move(event->pos() - QPoint{50, 25}); // TODO: Hardcoded + ghostImage->move(event->pos() - QPoint{50, ui->toolBar->height() + 25}); // TODO: Hardcoded event->accept(); } @@ -101,6 +119,9 @@ void Window::ToggleComponentPlacer() { delete ghostImage; ghostImage = nullptr; + + for (Component* c : components) + c->setAttribute(Qt::WA_TransparentForMouseEvents, false); } else { @@ -111,10 +132,14 @@ void Window::ToggleComponentPlacer() void Window::LoadGhostLabel(const QString& resource) { if (ghostImage == nullptr) { - ghostImage = new GhostLabel(this); - ghostImage->setGeometry({ 0, 0, 100, 50 }); + ghostImage = new GhostLabel(ui->centralwidget); + QPoint mousePos = ui->centralwidget->mapFromGlobal(QCursor::pos()); + ghostImage->setGeometry({ mousePos.x() - 50, mousePos.y() - 25, 100, 50 }); ghostImage->setScaledContents(true); ghostImage->setVisible(true); + + for (Component* c : components) + c->setAttribute(Qt::WA_TransparentForMouseEvents, true); } QPainter p; @@ -127,3 +152,14 @@ void Window::LoadGhostLabel(const QString& resource) ghostImage->setPixmap(QPixmap::fromImage(image)); } + +Component* Window::componentAt(const QPoint& point) +{ + for (Component* c : components) + { + if (c->geometry().contains(point)) + return c; + } + + return nullptr; +} diff --git a/ui/Window.ui b/ui/Window.ui index 46d567b..72a389c 100644 --- a/ui/Window.ui +++ b/ui/Window.ui @@ -30,8 +30,13 @@ + + + + + @@ -53,7 +58,7 @@ true - true + false @@ -66,7 +71,7 @@ Create an AND gate - 1 + 2 @@ -85,7 +90,97 @@ Create an OR gate - 2 + 4 + + + + + true + + + + :/components/not.png:/components/not.png + + + AddNOT + + + Create a NOT gate + + + 1 + + + + + true + + + + :/components/nand.png:/components/nand.png + + + AddNAND + + + Create a NAND gate + + + 3 + + + + + true + + + + :/components/nor.png:/components/nor.png + + + AddNOR + + + Create a NOR gate + + + 5 + + + + + true + + + + :/components/xor.png:/components/xor.png + + + AddXOR + + + Create an XOR gate + + + 6 + + + + + true + + + + :/components/xnor.png:/components/xnor.png + + + AddXNOR + + + Create an XNOR gate + + + 7 diff --git a/ui/assets/README.md b/ui/assets/README.md index 502fa00..4a7922f 100644 --- a/ui/assets/README.md +++ b/ui/assets/README.md @@ -1,6 +1,5 @@ Start/Stop icons taken from https://iconsmind.com/ (modified) -Logic Gates taken from Wikimedia Commons. The authors are -* jjbeard (AND, OR) +Logic Gates taken from Wikimedia Commons. https://commons.wikimedia.org/wiki/Logic_gates_unified_symbols The Wikimedia icons are part of the public domain \ No newline at end of file diff --git a/ui/assets/nand.png b/ui/assets/nand.png new file mode 100644 index 0000000..37cb74d Binary files /dev/null and b/ui/assets/nand.png differ diff --git a/ui/assets/nor.png b/ui/assets/nor.png new file mode 100644 index 0000000..b82bcc0 Binary files /dev/null and b/ui/assets/nor.png differ diff --git a/ui/assets/not.png b/ui/assets/not.png new file mode 100644 index 0000000..b78f2d9 Binary files /dev/null and b/ui/assets/not.png differ diff --git a/ui/assets/resources.qrc b/ui/assets/resources.qrc index 5ac7291..3f9044c 100644 --- a/ui/assets/resources.qrc +++ b/ui/assets/resources.qrc @@ -6,5 +6,10 @@ and.png or.png + nand.png + nor.png + not.png + xnor.png + xor.png diff --git a/ui/assets/xnor.png b/ui/assets/xnor.png new file mode 100644 index 0000000..2705095 Binary files /dev/null and b/ui/assets/xnor.png differ diff --git a/ui/assets/xor.png b/ui/assets/xor.png new file mode 100644 index 0000000..63dddbd Binary files /dev/null and b/ui/assets/xor.png differ