got rig of throwing exceptions, and friend classes: undoCommands get contexts

master
Denes Matetelki 14 years ago
parent 113c28dba3
commit ebbd74147d

@ -9,63 +9,57 @@
class GraphLogic; class GraphLogic;
// exceptions: struct UndoContext
class NoActiveNodeException : public std::exception
{
public:
const char* what() const throw();
};
class CannotPlaceNewNodeException : public std::exception
{ {
public: GraphLogic *m_graphLogic;
const char* what() const throw(); Node *m_activeNode;
}; Node *m_hintNode;
QList <Node *> *m_nodeList;
class CannotDeleteBaseNodeException : public std::exception
{
public:
const char* what() const throw();
};
class BaseNodeCannotBeEdgeTargetException : public std::exception
{
public:
const char* what() const throw();
};
class EdgeExistsBetweenNodesException : public std::exception
{
public:
const char* what() const throw();
};
class EdgeDoesntExistsBetweenNodesException : public std::exception QPointF m_pos;
{ QColor m_color;
public: Node *m_source;
const char* what() const throw(); Node *m_destination;
bool m_secondary;
UndoContext(GraphLogic *graphLogic = 0,
Node *activeNode = 0,
Node *hintNode = 0,
QList <Node *> *nodeList = 0,
QPointF pos = QPointF(),
QColor color = QColor(),
Node *source = 0,
Node *destination = 0,
bool secondary = false)
: m_graphLogic(graphLogic)
, m_activeNode(activeNode)
, m_hintNode(hintNode)
, m_nodeList(nodeList)
, m_pos(pos)
, m_color(color)
, m_source(source)
, m_destination(destination)
, m_secondary(secondary) {};
}; };
// commands:
class InsertNodeCommand : public QUndoCommand class InsertNodeCommand : public QUndoCommand
{ {
public: public:
InsertNodeCommand(GraphLogic *graphLogic); InsertNodeCommand(UndoContext context);
~InsertNodeCommand();
void undo(); void undo();
void redo(); void redo();
private: private:
GraphLogic *m_graphLogic; bool m_done;
UndoContext m_context;
Node *m_node; Node *m_node;
QPointF m_pos;
Node *m_activeNode; Node *m_activeNode;
Edge *m_edge; Edge *m_edge;
}; };
@ -75,14 +69,15 @@ class RemoveNodeCommand : public QUndoCommand
public: public:
RemoveNodeCommand(GraphLogic *graphLogic); RemoveNodeCommand(UndoContext context);
void undo(); void undo();
void redo(); void redo();
private: private:
GraphLogic *m_graphLogic; bool m_done;
UndoContext m_context;
Node *m_activeNode; Node *m_activeNode;
Node *m_hintNode; Node *m_hintNode;
@ -96,18 +91,18 @@ class AddEdgeCommand : public QUndoCommand
public: public:
AddEdgeCommand(GraphLogic *graphLogic, Node *source, Node *destinaion); AddEdgeCommand(UndoContext context);
~AddEdgeCommand();
void undo(); void undo();
void redo(); void redo();
private: private:
GraphLogic *m_graphLogic; bool m_done;
UndoContext m_context;
Node *m_activeNode; Node *m_activeNode;
Node *m_source;
Node *m_destination;
Edge *m_edge; Edge *m_edge;
}; };
@ -116,18 +111,17 @@ class RemoveEdgeCommand : public QUndoCommand
public: public:
RemoveEdgeCommand(GraphLogic *graphLogic, Node *source, Node *destinaion); RemoveEdgeCommand(UndoContext context);
void undo(); void undo();
void redo(); void redo();
private: private:
GraphLogic *m_graphLogic; bool m_done;
UndoContext m_context;
Node *m_activeNode; Node *m_activeNode;
Node *m_source;
Node *m_destination;
Edge *m_edge; Edge *m_edge;
}; };

@ -11,6 +11,7 @@ class Edge : public QGraphicsItem
public: public:
Edge(Node *sourceNode, Node *destNode); Edge(Node *sourceNode, Node *destNode);
~Edge();
Node *sourceNode() const; Node *sourceNode() const;
Node *destNode() const; Node *destNode() const;

@ -23,6 +23,8 @@ class GraphLogic : public QObject
public: public:
explicit GraphLogic(GraphWidget *parent = 0); explicit GraphLogic(GraphWidget *parent = 0);
GraphWidget *graphWidget() const;
void setUndoStack(QUndoStack *stack); void setUndoStack(QUndoStack *stack);
bool processKeyEvent(QKeyEvent *event); bool processKeyEvent(QKeyEvent *event);
@ -33,6 +35,11 @@ public:
void writeContentToXmlFile(const QString &fileName); void writeContentToXmlFile(const QString &fileName);
void writeContentToPngFile(const QString &fileName); void writeContentToPngFile(const QString &fileName);
Node *nodeFactory();
void setActiveNode(Node *node);
void setHintNode(Node *node);
void reShowNumbers();
public slots: public slots:
// commands from toolbars: // commands from toolbars:
@ -74,9 +81,7 @@ private:
void delNumber(); void delNumber();
void applyNumber(); void applyNumber();
Node *nodeFactory();
void selectNode(Node *node); void selectNode(Node *node);
void setActiveNode(Node *node);
// functions on the edges // functions on the edges
QList<Edge *> allEdges() const; QList<Edge *> allEdges() const;
@ -104,10 +109,10 @@ private:
QUndoStack *m_undoStack; QUndoStack *m_undoStack;
friend class InsertNodeCommand; // friend class InsertNodeCommand;
friend class RemoveNodeCommand; // friend class RemoveNodeCommand;
friend class AddEdgeCommand; // friend class AddEdgeCommand;
friend class RemoveEdgeCommand; // friend class RemoveEdgeCommand;
}; };
#endif // GRAPHLOGIC_H #endif // GRAPHLOGIC_H

@ -6,136 +6,83 @@
#include <math.h> #include <math.h>
InsertNodeCommand::InsertNodeCommand(UndoContext context)
const char* NoActiveNodeException::what() const throw() : m_done(false)
{ , m_context(context)
return QObject::tr("No active node.").toStdString().c_str(); , m_activeNode(context.m_activeNode)
}
const char* CannotPlaceNewNodeException::what() const throw()
{
return QObject::tr("New node would be placed outside of the scene.").
toStdString().c_str();
}
const char* CannotDeleteBaseNodeException::what() const throw()
{ {
return QObject::tr("Base node cannot be deleted.").toStdString().c_str(); setText(QObject::tr("Node added to \"").append(
} m_activeNode == m_context.m_nodeList->first() ?
const char* BaseNodeCannotBeEdgeTargetException::what() const throw()
{
return QObject::tr("Base node cannot be a target.").toStdString().c_str();
}
const char* EdgeExistsBetweenNodesException::what() const throw()
{
return QObject::tr("There is already an edge between these two nodes.").
toStdString().c_str();
}
const char* EdgeDoesntExistsBetweenNodesException::what() const throw()
{
return QObject::tr("There is no edge between these two nodes.").
toStdString().c_str();
}
InsertNodeCommand::InsertNodeCommand(GraphLogic *graphLogic)
: m_graphLogic(graphLogic)
, m_node(0)
, m_activeNode(m_graphLogic->m_activeNode)
, m_edge(0)
{
if (!m_activeNode)
throw NoActiveNodeException();
setText(QObject::tr("Node added to ").append(
m_activeNode == m_graphLogic->m_nodeList.first() ?
QObject::tr("Base node") : QObject::tr("Base node") :
QString("\"").append(m_activeNode->toPlainText().append("\"")))); m_activeNode->toPlainText()).append("\""));
m_graphLogic->nodeLostFocus();
// get the biggest angle between the edges of the Node. m_context.m_graphLogic->nodeLostFocus();
double angle(m_activeNode->calculateBiggestAngle());
// let the distance between the current and new Node be 100 pixels // create new node which inherits the color and textColor
qreal length(100); m_node = m_context.m_graphLogic->nodeFactory();
m_pos = m_activeNode->sceneBoundingRect().center() +
QPointF(length * cos(angle), length * sin(angle)) -
Node::newNodeCenter;
QRectF rect (m_graphLogic->m_graphWidget->scene()->sceneRect().topLeft(),
m_graphLogic->m_graphWidget->scene()->sceneRect().bottomRight()
- Node::newNodeBottomRigth);
if (!rect.contains(m_pos))
throw CannotPlaceNewNodeException();
// add a new node which inherits the color and textColor
m_node = m_graphLogic->nodeFactory();
m_node->setColor(m_activeNode->color()); m_node->setColor(m_activeNode->color());
m_node->setTextColor(m_activeNode->textColor()); m_node->setTextColor(m_activeNode->textColor());
m_node->setHtml(QString("")); m_node->setHtml(QString(""));
// new edge inherits colors and size from target
m_edge = new Edge(m_activeNode, m_node); m_edge = new Edge(m_activeNode, m_node);
m_edge->setColor(m_node->color()); m_edge->setColor(m_node->color());
m_edge->setWidth(m_node->scale()*2 + 1); m_edge->setWidth(m_node->scale()*2 + 1);
m_edge->setSecondary(false); m_edge->setSecondary(false);
} }
InsertNodeCommand::~InsertNodeCommand()
{
if (!m_done)
{
delete m_edge;
delete m_node;
}
}
void InsertNodeCommand::undo() void InsertNodeCommand::undo()
{ {
m_graphLogic->m_nodeList.removeAll(m_node); // remove node
m_graphLogic->m_graphWidget->scene()->removeItem(m_node); m_context.m_nodeList->removeAll(m_node);
m_context.m_graphLogic->graphWidget()->scene()->removeItem(m_node);
m_context.m_graphLogic->setActiveNode(m_activeNode);
// m_node->deleteEdges(); // remove edge
m_edge->sourceNode()->removeEdge(m_edge); m_edge->sourceNode()->removeEdge(m_edge);
m_edge->destNode()->removeEdge(m_edge); m_edge->destNode()->removeEdge(m_edge);
m_graphLogic->m_graphWidget->scene()->removeItem(m_edge); m_context.m_graphLogic->graphWidget()->scene()->removeItem(m_edge);
m_graphLogic->setActiveNode(m_activeNode);
emit m_graphLogic->contentChanged(false); m_context.m_graphLogic->reShowNumbers();
m_done = false;
} }
void InsertNodeCommand::redo() void InsertNodeCommand::redo()
{ {
m_graphLogic->m_graphWidget->scene()->addItem(m_node); // add node
m_graphLogic->m_nodeList.append(m_node); m_context.m_graphLogic->graphWidget()->scene()->addItem(m_node);
m_context.m_nodeList->append(m_node);
m_node->setPos(m_context.m_pos);
m_context.m_graphLogic->setActiveNode(m_node);
m_node->setPos(m_pos); if (m_context.m_graphLogic->graphWidget()->hasFocus())
m_context.m_graphLogic->nodeEdited();
// add edge
m_edge->sourceNode()->addEdge(m_edge,true); m_edge->sourceNode()->addEdge(m_edge,true);
m_edge->destNode()->addEdge(m_edge,false); m_edge->destNode()->addEdge(m_edge,false);
m_graphLogic->m_graphWidget->scene()->addItem(m_edge); m_context.m_graphLogic->graphWidget()->scene()->addItem(m_edge);
m_graphLogic->setActiveNode(m_node);
if (m_graphLogic->m_graphWidget->hasFocus())
m_graphLogic->nodeEdited();
emit m_graphLogic->contentChanged();
// it we are in hint mode, the numbers shall be re-calculated m_context.m_graphLogic->reShowNumbers();
if (m_graphLogic->m_showingNodeNumbers) m_done = true;
m_graphLogic->showNodeNumbers();
} }
RemoveNodeCommand::RemoveNodeCommand(GraphLogic *graphLogic) RemoveNodeCommand::RemoveNodeCommand(UndoContext context)
: m_graphLogic(graphLogic) : m_context(context)
, m_activeNode(m_graphLogic->m_activeNode) , m_activeNode(context.m_activeNode)
, m_hintNode(m_graphLogic->m_hintNode) , m_hintNode(context.m_hintNode)
{ {
if (!m_activeNode) setText(QObject::tr("Node deleted \"").append(
throw NoActiveNodeException();
if (m_activeNode == m_graphLogic->m_nodeList.first())
throw CannotDeleteBaseNodeException();
setText(QObject::tr("Node removed \"").append(
m_activeNode->toPlainText().append("\""))); m_activeNode->toPlainText().append("\"")));
// remove just the active Node or it's subtree too? // remove just the active Node or it's subtree too?
@ -150,6 +97,7 @@ RemoveNodeCommand::RemoveNodeCommand(GraphLogic *graphLogic)
m_nodeList.push_back(m_activeNode); m_nodeList.push_back(m_activeNode);
} }
// collect affected edges
foreach(Node *node, m_nodeList) foreach(Node *node, m_nodeList)
foreach(Edge *edge, node->edges()) foreach(Edge *edge, node->edges())
if (m_edgeList.indexOf(edge) == -1) if (m_edgeList.indexOf(edge) == -1)
@ -158,149 +106,131 @@ RemoveNodeCommand::RemoveNodeCommand(GraphLogic *graphLogic)
void RemoveNodeCommand::undo() void RemoveNodeCommand::undo()
{ {
// add nodes
foreach (Node *node, m_nodeList) foreach (Node *node, m_nodeList)
{ {
m_graphLogic->m_graphWidget->scene()->addItem(node); m_context.m_graphLogic->graphWidget()->scene()->addItem(node);
m_graphLogic->m_nodeList.append(node); m_context.m_nodeList->append(node);
} }
// add edges
foreach (Edge *edge, m_edgeList) foreach (Edge *edge, m_edgeList)
{ {
edge->sourceNode()->addEdge(edge,true); edge->sourceNode()->addEdge(edge,true);
edge->destNode()->addEdge(edge,false); edge->destNode()->addEdge(edge,false);
m_graphLogic->m_graphWidget->scene()->addItem(edge); m_context.m_graphLogic->graphWidget()->scene()->addItem(edge);
} }
emit m_graphLogic->contentChanged(false); m_context.m_graphLogic->setActiveNode(m_activeNode);
m_graphLogic->m_activeNode = m_activeNode; m_context.m_graphLogic->setHintNode(m_hintNode);
m_graphLogic->m_hintNode = m_hintNode;
m_context.m_graphLogic->reShowNumbers();
// it we are in hint mode, the numbers shall be re-calculated
if (m_graphLogic->m_showingNodeNumbers)
m_graphLogic->showNodeNumbers();
} }
void RemoveNodeCommand::redo() void RemoveNodeCommand::redo()
{ {
foreach(Node *node, m_nodeList) foreach(Node *node, m_nodeList)
{ {
if (m_graphLogic->m_hintNode==node) if (m_context.m_hintNode==node)
m_graphLogic->m_hintNode=0; m_context.m_graphLogic->setHintNode(0);
m_graphLogic->m_nodeList.removeAll(node); m_context.m_nodeList->removeAll(node);
m_graphLogic->m_graphWidget->scene()->removeItem(node); m_context.m_graphLogic->graphWidget()->scene()->removeItem(node);
} }
foreach(Edge *edge, m_edgeList) foreach(Edge *edge, m_edgeList)
{ {
edge->sourceNode()->removeEdge(edge); edge->sourceNode()->removeEdge(edge);
edge->destNode()->removeEdge(edge); edge->destNode()->removeEdge(edge);
m_graphLogic->m_graphWidget->scene()->removeItem(edge); m_context.m_graphLogic->graphWidget()->scene()->removeItem(edge);
} }
m_graphLogic->m_activeNode = 0; m_context.m_graphLogic->setActiveNode(0);
emit m_graphLogic->contentChanged();
// it we are in hint mode, the numbers shall be re-calculated m_context.m_graphLogic->reShowNumbers();
if (m_graphLogic->m_showingNodeNumbers)
m_graphLogic->showNodeNumbers();
} }
AddEdgeCommand::AddEdgeCommand(GraphLogic *graphLogic, Node *source, Node *destinaion) AddEdgeCommand::AddEdgeCommand(UndoContext context)
: m_graphLogic(graphLogic) : m_done(false)
, m_activeNode(m_graphLogic->m_activeNode) , m_context(context)
, m_source(source) , m_activeNode(context.m_activeNode)
, m_destination(destinaion)
{ {
if (m_destination == m_graphLogic->m_nodeList.first())
throw BaseNodeCannotBeEdgeTargetException();
if (m_source->isConnected(m_destination))
throw EdgeExistsBetweenNodesException();
setText(QObject::tr("Edge added between \"").append( setText(QObject::tr("Edge added between \"").append(
m_source->toPlainText()).append( m_context.m_source == m_context.m_nodeList->first() ?
QObject::tr("Base node") :
m_context.m_source->toPlainText()).append(
QObject::tr("\" and \"").append( QObject::tr("\" and \"").append(
m_destination->toPlainText()).append("\""))); m_context.m_destination == m_context.m_nodeList->first() ?
QObject::tr("Base node") :
m_context.m_destination->toPlainText()).append("\"")));
// aviod the graph beeing acyclic. (ok, Nodes having multiple parents)
bool sec(false);
if (!m_destination->edgesToThis().empty())
{
emit m_graphLogic->notification(
QObject::tr("The graph is acyclic, edge added as secondary edge."));
sec = true;
}
m_edge = new Edge(m_source, m_destination); m_edge = new Edge(m_context.m_source, m_context.m_destination);
m_edge->setColor(m_destination->color()); m_edge->setColor(m_context.m_destination->color());
m_edge->setWidth(m_destination->scale()*2 + 1); m_edge->setWidth(m_context.m_destination->scale()*2 + 1);
// The Edge is secondary, because the Node already has a parent // The Edge is secondary, because the Node already has a parent
// (it is already a destination of another Edge) // (it is already a destination of another Edge)
m_edge->setSecondary(sec); m_edge->setSecondary(m_context.m_secondary);
} }
void AddEdgeCommand::undo() void AddEdgeCommand::undo()
{ {
m_source->removeEdge(m_edge); m_context.m_source->removeEdge(m_edge);
m_destination->removeEdge(m_edge); m_context.m_destination->removeEdge(m_edge);
m_graphLogic->m_graphWidget->scene()->removeItem(m_edge); m_context.m_graphLogic->graphWidget()->scene()->removeItem(m_edge);
m_graphLogic->setActiveNode(m_activeNode);
emit m_graphLogic->contentChanged(false); m_context.m_graphLogic->setActiveNode(m_activeNode);
m_done = false;
} }
void AddEdgeCommand::redo() void AddEdgeCommand::redo()
{ {
m_source->addEdge(m_edge, true); m_context.m_source->addEdge(m_edge, true);
m_destination->addEdge(m_edge, false); m_context.m_destination->addEdge(m_edge, false);
m_graphLogic->m_graphWidget->scene()->addItem(m_edge); m_context.m_graphLogic->graphWidget()->scene()->addItem(m_edge);
m_graphLogic->setActiveNode(m_destination); m_context.m_graphLogic->setActiveNode(m_context.m_destination);
emit m_graphLogic->contentChanged(); m_done = true;
} }
RemoveEdgeCommand::RemoveEdgeCommand(GraphLogic *graphLogic, Node *source, Node *destinaion) AddEdgeCommand::~AddEdgeCommand()
: m_graphLogic(graphLogic)
, m_activeNode(m_graphLogic->m_activeNode)
, m_source(source)
, m_destination(destinaion)
, m_edge(source->edgeTo(destinaion))
{ {
if (!m_source->isConnected(m_destination)) if (!m_done)
throw EdgeDoesntExistsBetweenNodesException(); delete m_edge;
}
setText(QObject::tr("Edge releted between \"").append( RemoveEdgeCommand::RemoveEdgeCommand(UndoContext context)
m_source->toPlainText()).append( : m_context(context)
, m_activeNode(context.m_activeNode)
{
setText(QObject::tr("Edge deleted between \"").append(
m_context.m_source == m_context.m_nodeList->first() ?
QObject::tr("Base node") :
m_context.m_source->toPlainText()).append(
QObject::tr("\" and \"").append( QObject::tr("\" and \"").append(
m_destination->toPlainText()).append("\""))); m_context.m_destination == m_context.m_nodeList->first() ?
QObject::tr("Base node") :
m_context.m_destination->toPlainText()).append("\"")));
m_edge = m_context.m_source->edgeTo(m_context.m_destination);
} }
void RemoveEdgeCommand::undo() void RemoveEdgeCommand::undo()
{ {
m_source->addEdge(m_edge, true); m_context.m_source->addEdge(m_edge, true);
m_destination->addEdge(m_edge, false); m_context.m_destination->addEdge(m_edge, false);
m_graphLogic->m_graphWidget->scene()->addItem(m_edge); m_context.m_graphLogic->graphWidget()->scene()->addItem(m_edge);
m_graphLogic->setActiveNode(m_destination); m_context.m_graphLogic->setActiveNode(m_activeNode);
emit m_graphLogic->contentChanged(false);
} }
void RemoveEdgeCommand::redo() void RemoveEdgeCommand::redo()
{ {
m_source->removeEdge(m_edge); m_context.m_source->removeEdge(m_edge);
m_destination->removeEdge(m_edge); m_context.m_destination->removeEdge(m_edge);
m_graphLogic->m_graphWidget->scene()->removeItem(m_edge); m_context.m_graphLogic->graphWidget()->scene()->removeItem(m_edge);
m_graphLogic->setActiveNode(m_activeNode);
emit m_graphLogic->contentChanged(); m_context.m_graphLogic->setActiveNode(m_context.m_activeNode);
} }

@ -26,6 +26,12 @@ Edge::Edge(Node *sourceNode, Node *destNode)
adjust(); adjust();
} }
Edge::~Edge()
{
m_sourceNode->removeEdge(this);
m_destNode->removeEdge(this);
}
Node * Edge::sourceNode() const Node * Edge::sourceNode() const
{ {
return m_sourceNode; return m_sourceNode;

@ -52,8 +52,11 @@ GraphLogic::GraphLogic(GraphWidget *parent)
(Qt::Key_Return, &GraphLogic::applyNumber)); (Qt::Key_Return, &GraphLogic::applyNumber));
m_memberMap.insert(std::pair<int, void(GraphLogic::*)()> m_memberMap.insert(std::pair<int, void(GraphLogic::*)()>
(Qt::Key_Enter, &GraphLogic::applyNumber)); (Qt::Key_Enter, &GraphLogic::applyNumber));
m_memberMap.insert(std::pair<int, void(GraphLogic::*)()> }
(Qt::Key_Delete, &GraphLogic::removeNode));
GraphWidget *GraphLogic::graphWidget() const
{
return m_graphWidget;
} }
void GraphLogic::setUndoStack(QUndoStack *stack) void GraphLogic::setUndoStack(QUndoStack *stack)
@ -293,32 +296,79 @@ void GraphLogic::writeContentToPngFile(const QString &fileName)
emit notification(tr("MindMap exported as ") + fileName); emit notification(tr("MindMap exported as ") + fileName);
} }
void GraphLogic::reShowNumbers()
{
if (m_showingNodeNumbers)
showNodeNumbers();
}
void GraphLogic::setHintNode(Node *node)
{
m_hintNode = node;
}
void GraphLogic::insertNode() void GraphLogic::insertNode()
{ {
try // checks
if (!m_activeNode)
{ {
QUndoCommand *insertNodeCommand = new InsertNodeCommand(this); emit notification(tr("No active node."));
m_undoStack->push(insertNodeCommand); return;
} }
catch (std::exception &e)
// get the biggest angle between the edges of the Node.
double angle(m_activeNode->calculateBiggestAngle());
// let the distance between the current and new Node be 100 pixels
qreal length(100);
QPointF pos(m_activeNode->sceneBoundingRect().center() +
QPointF(length * cos(angle), length * sin(angle)) -
Node::newNodeCenter);
QRectF rect (m_graphWidget->scene()->sceneRect().topLeft(),
m_graphWidget->scene()->sceneRect().bottomRight()
- Node::newNodeBottomRigth);
if (!rect.contains(pos))
{ {
emit notification(e.what()); emit notification(tr("New node would be placed outside of the scene."));
return; return;
} }
UndoContext context;
context.m_graphLogic = this;
context.m_nodeList = &m_nodeList;
context.m_activeNode = m_activeNode;
context.m_pos = pos;
QUndoCommand *insertNodeCommand = new InsertNodeCommand(context);
m_undoStack->push(insertNodeCommand);
} }
void GraphLogic::removeNode() void GraphLogic::removeNode()
{ {
try if (!m_activeNode)
{ {
QUndoCommand *removeNodeCommand = new RemoveNodeCommand(this); emit notification(tr("No active node."));
m_undoStack->push(removeNodeCommand); return;
} }
catch (std::exception &e)
if (m_activeNode == m_nodeList.first())
{ {
emit notification(e.what()); emit notification(tr("Base node cannot be deleted."));
return; return;
} }
UndoContext context;
context.m_graphLogic = this;
context.m_nodeList = &m_nodeList;
context.m_activeNode = m_activeNode;
context.m_hintNode = m_hintNode;
QUndoCommand *insertNodeCommand = new RemoveNodeCommand(context);
m_undoStack->push(insertNodeCommand);
} }
void GraphLogic::nodeEdited() void GraphLogic::nodeEdited()
@ -700,30 +750,57 @@ QList<Edge *> GraphLogic::allEdges() const
void GraphLogic::addEdge(Node *source, Node *destination) void GraphLogic::addEdge(Node *source, Node *destination)
{ {
try if (destination == m_nodeList.first())
{ {
QUndoCommand *addEdgeCommand = new AddEdgeCommand(this, source, destination); emit notification(tr("Base node cannot be a target."));
m_undoStack->push(addEdgeCommand); return;
} }
catch (std::exception &e)
if (source->isConnected(destination))
{ {
emit notification(e.what()); emit notification(tr("There is already an edge between these two nodes."));
return; return;
} }
}
void GraphLogic::removeEdge(Node *source, Node *destination) // aviod the graph beeing acyclic. (ok, Nodes having multiple parents)
{ bool sec(false);
try if (!destination->edgesToThis().empty())
{ {
QUndoCommand *addEdgeCommand = new RemoveEdgeCommand(this, source, destination); emit notification(
QObject::tr("The graph is acyclic, edge added as secondary edge."));
sec = true;
}
UndoContext context;
context.m_graphLogic = this;
context.m_nodeList = &m_nodeList;
context.m_activeNode = m_activeNode;
context.m_source = source;
context.m_destination = destination;
context.m_secondary = sec;
QUndoCommand *addEdgeCommand = new AddEdgeCommand(context);
m_undoStack->push(addEdgeCommand); m_undoStack->push(addEdgeCommand);
} }
catch (std::exception &e)
void GraphLogic::removeEdge(Node *source, Node *destination)
{
if (!source->isConnected(destination))
{ {
emit notification(e.what()); emit notification(tr("There is no edge between these two nodes."));
return; return;
} }
UndoContext context;
context.m_graphLogic = this;
context.m_nodeList = &m_nodeList;
context.m_activeNode = m_activeNode;
context.m_source = source;
context.m_destination = destination;
QUndoCommand *removeEdgeCommand = new RemoveEdgeCommand(context);
m_undoStack->push(removeEdgeCommand);
} }
// re-draw numbers // re-draw numbers

@ -28,8 +28,6 @@ GraphWidget::GraphWidget(MainWindow *parent)
m_graphlogic = new GraphLogic(this); m_graphlogic = new GraphLogic(this);
} }
void GraphWidget::newScene() void GraphWidget::newScene()
{ {
m_graphlogic->removeAllNodes(); m_graphlogic->removeAllNodes();

@ -156,6 +156,8 @@ void MainWindow::saveFile(const bool &checkIfReadonly)
m_graphicsView->graphLogic()->writeContentToXmlFile(m_fileName); m_graphicsView->graphLogic()->writeContentToXmlFile(m_fileName);
contentChanged(false); contentChanged(false);
m_undoStack->clear();
} }
bool MainWindow::saveFileAs() bool MainWindow::saveFileAs()

@ -54,10 +54,7 @@ void Node::deleteEdge(Node *otherEnd)
|| (it->edge->sourceNode() == this && || (it->edge->sourceNode() == this &&
it->edge->destNode() == otherEnd)) it->edge->destNode() == otherEnd))
{ {
Edge *tmp = it->edge; delete it->edge;
tmp->sourceNode()->removeEdge(tmp);
tmp->destNode()->removeEdge(tmp);
delete tmp;
return; return;
} }
} }
@ -66,14 +63,7 @@ void Node::deleteEdge(Node *otherEnd)
void Node::deleteEdges() void Node::deleteEdges()
{ {
foreach (EdgeElement element, m_edgeList) foreach (EdgeElement element, m_edgeList)
{ delete element.edge;
Edge *tmp = element.edge;
tmp->sourceNode()->removeEdge(tmp);
tmp->destNode()->removeEdge(tmp);
/// @bug crashes sometimes
delete tmp;
}
} }
void Node::removeEdge(Edge *edge) void Node::removeEdge(Edge *edge)

Loading…
Cancel
Save