Added simulation functionality

This commit is contained in:
Robert 2020-12-02 21:18:53 +01:00
parent 79c5d147a3
commit 78da43f51d
23 changed files with 385 additions and 21 deletions

View file

@ -12,8 +12,18 @@ public:
explicit Component(QWidget* parent); explicit Component(QWidget* parent);
explicit Component(QWidget* parent, const QString& resource); explicit Component(QWidget* parent, const QString& resource);
void mouseMoveEvent(QMouseEvent* event) override; virtual void mouseMoveEvent(QMouseEvent* event) override;
private: QPoint CenterPos();
void Connect(Component* component);
const QString& Type();
virtual bool Action() = 0;
protected:
Ui::Component* ui; Ui::Component* ui;
QString type;
std::vector<Component*> connections;
}; };

View file

@ -1,6 +1,7 @@
#pragma once #pragma once
#include <qmainwindow.h> #include <qmainwindow.h>
#include <qpen.h>
namespace Ui { namespace Ui {
class Window; class Window;
@ -20,12 +21,14 @@ protected:
void mousePressEvent(QMouseEvent* event) override; void mousePressEvent(QMouseEvent* event) override;
void mouseMoveEvent(QMouseEvent* event) override; void mouseMoveEvent(QMouseEvent* event) override;
void mouseReleaseEvent(QMouseEvent* event) override; void mouseReleaseEvent(QMouseEvent* event) override;
void paintEvent(QPaintEvent* event) override;
private: private:
void ToggleSimulation(); void ToggleSimulation();
void ToggleComponentPlacer(); void ToggleComponentPlacer();
void LoadGhostLabel(const QString& resource); void LoadGhostLabel(const QString& resource);
Component* componentAt(const QPoint& point); Component* componentAt(const QPoint& point);
Component* createComponent(const QPoint& position);
Ui::Window* ui; Ui::Window* ui;
bool simulating; bool simulating;
@ -34,10 +37,18 @@ private:
Component* component; Component* component;
QPoint relativePos; QPoint relativePos;
} dragInfo; } dragInfo;
bool mouseDown;
GhostLabel* ghostImage; GhostLabel* ghostImage;
QString resourcePath; QString resourcePath;
QPen pen;
QActionGroup* componentGroup; QActionGroup* componentGroup;
enum class eComponent {
NOT, AND, NAND, OR, NOR, XOR, XNOR, SWITCH, LAMP
} componentType;
std::vector<Component*> components; std::vector<Component*> components;
std::vector<QLine> wires;
}; };

View file

@ -0,0 +1,14 @@
#pragma once
#include <QWidget.h>
#include "../Component.hpp"
class ANDGate : public Component
{
public:
ANDGate(QWidget* parent) : Component(parent, ":/components/and.png") { type = "and"; }
bool Action() override {
return (connections[0]->Action() && connections[1]->Action());
}
};

View file

@ -0,0 +1,24 @@
#pragma once
#include <QWidget.h>
#include "../Component.hpp"
class Lamp : public Component
{
public:
Lamp(QWidget* parent) : Component(parent, ":/components/lamp_off.png") { type = "lamp"; }
bool Action() override {
return connections[0]->Action();
}
void Turn(bool state) {
if (state)
ui->label->setPixmap(QPixmap(":/components/lamp_on.png"));
else
ui->label->setPixmap(QPixmap(":/components/lamp_off.png"));
}
private:
bool state = false;
};

View file

@ -0,0 +1,14 @@
#pragma once
#include <QWidget.h>
#include "../Component.hpp"
class NANDGate : public Component
{
public:
NANDGate(QWidget* parent) : Component(parent, ":/components/nand.png") { type = "nand"; }
bool Action() override {
return !(connections[0]->Action() && connections[1]->Action());
}
};

View file

@ -0,0 +1,14 @@
#pragma once
#include <QWidget.h>
#include "../Component.hpp"
class NORGate : public Component
{
public:
NORGate(QWidget* parent) : Component(parent, ":/components/nor.png") { type = "nor"; }
bool Action() override {
return !(connections[0]->Action() || connections[1]->Action());
}
};

View file

@ -0,0 +1,14 @@
#pragma once
#include <QWidget.h>
#include "../Component.hpp"
class NOTGate : public Component
{
public:
NOTGate(QWidget* parent) : Component(parent, ":/components/not.png") { type = "not"; }
bool Action() override {
return !connections[0]->Action();
}
};

View file

@ -0,0 +1,14 @@
#pragma once
#include <QWidget.h>
#include "../Component.hpp"
class ORGate : public Component
{
public:
ORGate(QWidget* parent) : Component(parent, ":/components/or.png") { type = "or"; }
bool Action() override {
return (connections[0]->Action() || connections[1]->Action());
}
};

View file

@ -0,0 +1,26 @@
#pragma once
#include <QWidget.h>
#include "ui_Component.h"
#include "../Component.hpp"
class Switch : public Component
{
public:
Switch(QWidget* parent) : Component(parent, ":/components/switch_off.png") { type = "switch"; }
bool Action() override {
return state;
}
void Toggle() {
state = !state;
if (state)
ui->label->setPixmap(QPixmap(":/components/switch_on.png"));
else
ui->label->setPixmap(QPixmap(":/components/switch_off.png"));
}
private:
bool state = false;
};

View file

@ -0,0 +1,14 @@
#pragma once
#include <QWidget.h>
#include "../Component.hpp"
class XNORGate : public Component
{
public:
XNORGate(QWidget* parent) : Component(parent, ":/components/xnor.png") { type = "xnor"; }
bool Action() override {
return !((connections[0]->Action() || connections[1]->Action()) && !(connections[0]->Action() && connections[1]->Action()));
}
};

View file

@ -0,0 +1,14 @@
#pragma once
#include <QWidget.h>
#include "../Component.hpp"
class XORGate : public Component
{
public:
XORGate(QWidget* parent) : Component(parent, ":/components/xor.png") { type = "xor"; }
bool Action() override {
return ((connections[0]->Action() || connections[1]->Action()) && !(connections[0]->Action() && connections[1]->Action()));
}
};

View file

@ -23,3 +23,18 @@ void Component::mouseMoveEvent(QMouseEvent* event)
{ {
event->ignore(); event->ignore();
} }
QPoint Component::CenterPos()
{
return pos() + QPoint(size().width() / 2, size().height() / 2);
}
void Component::Connect(Component* component)
{
connections.push_back(component);
}
const QString& Component::Type()
{
return type;
}

View file

@ -2,6 +2,15 @@
#include "ui_Window.h" #include "ui_Window.h"
#include "Component.hpp" #include "Component.hpp"
#include "components/NOTgate.hpp"
#include "components/ANDgate.hpp"
#include "components/NANDgate.hpp"
#include "components/ORgate.hpp"
#include "components/NORgate.hpp"
#include "components/XORgate.hpp"
#include "components/XNORgate.hpp"
#include "components/Switch.hpp"
#include "components/Lamp.hpp"
#include <iostream> #include <iostream>
#include <qevent.h> #include <qevent.h>
@ -21,6 +30,8 @@ Window::Window() :
setMouseTracking(true); setMouseTracking(true);
componentGroup = new QActionGroup(this); componentGroup = new QActionGroup(this);
componentGroup->addAction(ui->actionCursor);
componentGroup->addAction(ui->actionWiring);
componentGroup->addAction(ui->actionAddNOT); componentGroup->addAction(ui->actionAddNOT);
componentGroup->addAction(ui->actionAddAND); componentGroup->addAction(ui->actionAddAND);
componentGroup->addAction(ui->actionAddNAND); componentGroup->addAction(ui->actionAddNAND);
@ -28,16 +39,25 @@ Window::Window() :
componentGroup->addAction(ui->actionAddNOR); componentGroup->addAction(ui->actionAddNOR);
componentGroup->addAction(ui->actionAddXOR); componentGroup->addAction(ui->actionAddXOR);
componentGroup->addAction(ui->actionAddXNOR); componentGroup->addAction(ui->actionAddXNOR);
componentGroup->setExclusionPolicy(QActionGroup::ExclusionPolicy::ExclusiveOptional); componentGroup->addAction(ui->actionAddSwitch);
componentGroup->addAction(ui->actionAddLamp);
componentGroup->setExclusionPolicy(QActionGroup::ExclusionPolicy::Exclusive);
connect(ui->actionStart, &QAction::triggered, this, &Window::ToggleSimulation); connect(ui->actionStart, &QAction::triggered, this, &Window::ToggleSimulation);
connect(ui->actionAddNOT, &QAction::triggered, this, [this]() {resourcePath = ":/components/not.png"; ToggleComponentPlacer(); }); connect(ui->actionCursor, &QAction::triggered, this, &Window::ToggleComponentPlacer);
connect(ui->actionAddAND, &QAction::triggered, this, [this]() {resourcePath = ":/components/and.png"; ToggleComponentPlacer(); }); connect(ui->actionWiring, &QAction::triggered, this, &Window::ToggleComponentPlacer);
connect(ui->actionAddNAND, &QAction::triggered, this, [this]() {resourcePath = ":/components/nand.png"; ToggleComponentPlacer(); }); connect(ui->actionAddNOT, &QAction::triggered, this, [this]() {componentType = eComponent::NOT , resourcePath = ":/components/not.png"; ToggleComponentPlacer(); });
connect(ui->actionAddOR, &QAction::triggered, this, [this]() {resourcePath = ":/components/or.png"; ToggleComponentPlacer(); }); connect(ui->actionAddAND, &QAction::triggered, this, [this]() {componentType = eComponent::AND, resourcePath = ":/components/and.png"; ToggleComponentPlacer(); });
connect(ui->actionAddNOR, &QAction::triggered, this, [this]() {resourcePath = ":/components/nor.png"; ToggleComponentPlacer(); }); connect(ui->actionAddNAND, &QAction::triggered, this, [this]() {componentType = eComponent::NAND , resourcePath = ":/components/nand.png"; ToggleComponentPlacer(); });
connect(ui->actionAddXOR, &QAction::triggered, this, [this]() {resourcePath = ":/components/xor.png"; ToggleComponentPlacer(); }); connect(ui->actionAddOR, &QAction::triggered, this, [this]() {componentType = eComponent::OR , resourcePath = ":/components/or.png"; ToggleComponentPlacer(); });
connect(ui->actionAddXNOR, &QAction::triggered, this, [this]() {resourcePath = ":/components/xnor.png"; ToggleComponentPlacer(); }); connect(ui->actionAddNOR, &QAction::triggered, this, [this]() {componentType = eComponent::NOR , resourcePath = ":/components/nor.png"; ToggleComponentPlacer(); });
connect(ui->actionAddXOR, &QAction::triggered, this, [this]() {componentType = eComponent::XOR, resourcePath = ":/components/xor.png"; ToggleComponentPlacer(); });
connect(ui->actionAddXNOR, &QAction::triggered, this, [this]() {componentType = eComponent::XNOR , resourcePath = ":/components/xnor.png"; ToggleComponentPlacer(); });
connect(ui->actionAddSwitch,&QAction::triggered, this, [this]() {componentType = eComponent::SWITCH , resourcePath = ":/components/switch_off.png"; ToggleComponentPlacer(); });
connect(ui->actionAddLamp, &QAction::triggered, this, [this]() {componentType = eComponent::LAMP , resourcePath = ":/components/lamp_off.png"; ToggleComponentPlacer(); });
pen.setColor(Qt::black);
pen.setWidth(3);
} }
Window::~Window() Window::~Window()
@ -60,12 +80,13 @@ Window::~Window()
void Window::mousePressEvent(QMouseEvent* event) void Window::mousePressEvent(QMouseEvent* event)
{ {
mouseDown = true;
if (simulating) return; if (simulating) return;
Component* child = componentAt(event->pos()); Component* child = componentAt(event->pos());
QAction* toggledAction = componentGroup->checkedAction(); QAction* toggledAction = componentGroup->checkedAction();
if (toggledAction == nullptr) if (toggledAction == ui->actionCursor)
{ {
if (child == nullptr) if (child == nullptr)
return; return;
@ -73,49 +94,106 @@ void Window::mousePressEvent(QMouseEvent* event)
dragInfo.component = static_cast<Component*>(child); dragInfo.component = static_cast<Component*>(child);
dragInfo.relativePos = event->pos() - child->pos(); dragInfo.relativePos = event->pos() - child->pos();
} }
else if (toggledAction == ui->actionWiring)
{
if (child == nullptr)
return;
QPoint mousePos = ui->centralwidget->mapFromParent(event->pos());
mousePos.setY(mousePos.y() + ui->toolBar->height());
wires.push_back(QLine(child->CenterPos(), mousePos));
// std::cout << "(" << child->x() << ", " << child->y() << ") --> (" << mousePos.x() << ", " << mousePos.y() << ")" << std::endl;
}
else else
{ {
if (child != nullptr) if (child != nullptr)
return; return;
Component* leak = new Component(this, resourcePath); createComponent(event->pos());
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);
} }
} }
void Window::mouseMoveEvent(QMouseEvent* event) void Window::mouseMoveEvent(QMouseEvent* event)
{ {
if (simulating) return;
if (dragInfo.component != nullptr) if (dragInfo.component != nullptr)
{
dragInfo.component->move(event->pos() - dragInfo.relativePos); dragInfo.component->move(event->pos() - dragInfo.relativePos);
// Move wires too lol
}
if (ghostImage != nullptr) if (ghostImage != nullptr)
ghostImage->move(event->pos() - QPoint{50, ui->toolBar->height() + 25}); // TODO: Hardcoded ghostImage->move(event->pos() - QPoint{50, ui->toolBar->height() + 25}); // TODO: Hardcoded
if ((componentGroup->checkedAction() == ui->actionWiring) && mouseDown)
{
QPoint mousePos = ui->centralwidget->mapFromParent(event->pos());
mousePos.setY(mousePos.y() + ui->toolBar->height());
wires.back().setP2(mousePos);
}
event->accept(); event->accept();
} }
void Window::mouseReleaseEvent(QMouseEvent* event) void Window::mouseReleaseEvent(QMouseEvent* event)
{ {
if (simulating) return;
mouseDown = false;
if (componentGroup->checkedAction() == ui->actionWiring)
{
Component* connectFrom = componentAt(wires.back().p1());
Component* connectTo = componentAt(event->pos());
if (connectTo == nullptr || connectTo == connectFrom)
{
wires.pop_back();
}
else
{
wires.back().setP2(connectTo->CenterPos());
connectTo->Connect(connectFrom);
}
}
else if(componentGroup->checkedAction() == ui->actionCursor)
{
Component* clicked = componentAt(event->pos());
if (clicked != nullptr)
if (clicked->Type() == "switch")
static_cast<Switch*>(clicked)->Toggle();
}
dragInfo.component = nullptr; dragInfo.component = nullptr;
} }
void Window::paintEvent(QPaintEvent* event)
{
QPainter painter(this);
painter.setPen(pen);
for(QLine line : wires)
painter.drawLine(line);
update();
}
void Window::ToggleSimulation() void Window::ToggleSimulation()
{ {
simulating = !simulating; simulating = !simulating;
ui->actionStart->setIcon(QIcon(simulating ? ":/toolbar/stop.png" : ":/toolbar/start.png")); ui->actionStart->setIcon(QIcon(simulating ? ":/toolbar/stop.png" : ":/toolbar/start.png"));
for (Component* c : components)
{
if (c->Type() == "lamp")
static_cast<Lamp*>(c)->Turn(c->Action());
}
} }
void Window::ToggleComponentPlacer() void Window::ToggleComponentPlacer()
{ {
QAction* toggledAction = componentGroup->checkedAction(); QAction* toggledAction = componentGroup->checkedAction();
if (toggledAction == nullptr) if (toggledAction == ui->actionCursor || toggledAction == ui->actionWiring)
{ {
delete ghostImage; delete ghostImage;
ghostImage = nullptr; ghostImage = nullptr;
@ -163,3 +241,35 @@ Component* Window::componentAt(const QPoint& point)
return nullptr; return nullptr;
} }
Component* Window::createComponent(const QPoint& position)
{
Component* leak = nullptr;
switch (componentType)
{
case eComponent::NOT: leak = new NOTGate(this); break;
case eComponent::AND: leak = new ANDGate(this); break;
case eComponent::NAND: leak = new NANDGate(this); break;
case eComponent::OR: leak = new ORGate(this); break;
case eComponent::NOR: leak = new NORGate(this); break;
case eComponent::XOR: leak = new XORGate(this); break;
case eComponent::XNOR: leak = new XNORGate(this); break;
case eComponent::SWITCH: leak = new Switch(this); break;
case eComponent::LAMP: leak = new Lamp(this); break;
}
if (leak != nullptr)
{
QPoint pos = position - 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);
}
return leak;
}

View file

@ -30,6 +30,9 @@
</attribute> </attribute>
<addaction name="actionStart"/> <addaction name="actionStart"/>
<addaction name="separator"/> <addaction name="separator"/>
<addaction name="actionCursor"/>
<addaction name="actionWiring"/>
<addaction name="separator"/>
<addaction name="actionAddNOT"/> <addaction name="actionAddNOT"/>
<addaction name="actionAddAND"/> <addaction name="actionAddAND"/>
<addaction name="actionAddNAND"/> <addaction name="actionAddNAND"/>
@ -37,6 +40,8 @@
<addaction name="actionAddNOR"/> <addaction name="actionAddNOR"/>
<addaction name="actionAddXOR"/> <addaction name="actionAddXOR"/>
<addaction name="actionAddXNOR"/> <addaction name="actionAddXNOR"/>
<addaction name="actionAddSwitch"/>
<addaction name="actionAddLamp"/>
</widget> </widget>
<action name="actionStart"> <action name="actionStart">
<property name="icon"> <property name="icon">
@ -183,6 +188,63 @@
<string>7</string> <string>7</string>
</property> </property>
</action> </action>
<action name="actionAddSwitch">
<property name="checkable">
<bool>true</bool>
</property>
<property name="icon">
<iconset resource="assets/resources.qrc">
<normaloff>:/components/switch.png</normaloff>:/components/switch.png</iconset>
</property>
<property name="text">
<string>AddSwitch</string>
</property>
</action>
<action name="actionAddLamp">
<property name="checkable">
<bool>true</bool>
</property>
<property name="icon">
<iconset resource="assets/resources.qrc">
<normaloff>:/components/lamp.png</normaloff>:/components/lamp.png</iconset>
</property>
<property name="text">
<string>AddLamp</string>
</property>
</action>
<action name="actionCursor">
<property name="checkable">
<bool>true</bool>
</property>
<property name="checked">
<bool>true</bool>
</property>
<property name="icon">
<iconset resource="assets/resources.qrc">
<normaloff>:/toolbar/cursor.png</normaloff>:/toolbar/cursor.png</iconset>
</property>
<property name="text">
<string>Cursor</string>
</property>
<property name="toolTip">
<string>Default Mode</string>
</property>
</action>
<action name="actionWiring">
<property name="checkable">
<bool>true</bool>
</property>
<property name="icon">
<iconset resource="assets/resources.qrc">
<normaloff>:/toolbar/wire.png</normaloff>:/toolbar/wire.png</iconset>
</property>
<property name="text">
<string>Wiring</string>
</property>
<property name="toolTip">
<string>Wiring mode</string>
</property>
</action>
</widget> </widget>
<resources> <resources>
<include location="assets/resources.qrc"/> <include location="assets/resources.qrc"/>

BIN
ui/assets/cursor.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 5 KiB

BIN
ui/assets/lamp.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 711 B

BIN
ui/assets/lamp_off.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.7 KiB

BIN
ui/assets/lamp_on.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.9 KiB

View file

@ -2,6 +2,8 @@
<qresource prefix="/toolbar"> <qresource prefix="/toolbar">
<file>start.png</file> <file>start.png</file>
<file>stop.png</file> <file>stop.png</file>
<file>cursor.png</file>
<file>wire.png</file>
</qresource> </qresource>
<qresource prefix="/components"> <qresource prefix="/components">
<file>and.png</file> <file>and.png</file>
@ -11,5 +13,11 @@
<file>not.png</file> <file>not.png</file>
<file>xnor.png</file> <file>xnor.png</file>
<file>xor.png</file> <file>xor.png</file>
<file>lamp.png</file>
<file>lamp_off.png</file>
<file>lamp_on.png</file>
<file>switch.png</file>
<file>switch_off.png</file>
<file>switch_on.png</file>
</qresource> </qresource>
</RCC> </RCC>

BIN
ui/assets/switch.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 KiB

BIN
ui/assets/switch_off.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.7 KiB

BIN
ui/assets/switch_on.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.9 KiB

BIN
ui/assets/wire.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 789 B