Skip to content
Snippets Groups Projects
QGVScene.cpp 8.23 KiB
Newer Older
nicolas.bergont's avatar
nicolas.bergont committed
/***************************************************************
QGVCore
Copyright (c) 2014, Bergont Nicolas, All rights reserved.

This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 3.0 of the License, or (at your option) any later version.

This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.

You should have received a copy of the GNU Lesser General Public
License along with this library.
***************************************************************/
Nicolas's avatar
Nicolas committed
#include "QGVScene.h"
// The following include allows the automoc to detect, that it must moc this class
#include "moc_QGVScene.cpp"
Nicolas's avatar
Nicolas committed
#include <QDebug>

#include <QGVNode.h>
#include <QGVEdge.h>
#include <QGVSubGraph.h>

#include <QGVCore.h>
#include <QGVGraphPrivate.h>
#include <QGVGvcPrivate.h>
#include <QGVEdgePrivate.h>
#include <QGVNodePrivate.h>

Nicolas's avatar
Nicolas committed
QGVScene::QGVScene(const QString &name, QObject *parent) : QGraphicsScene(parent)
{
		_context = new QGVGvcPrivate(gvContext());
		_graph = new QGVGraphPrivate(agopen(name.toLocal8Bit().data(), Agdirected, NULL));
Nicolas's avatar
Nicolas committed
    //setGraphAttribute("fontname", QFont().family());
}

QGVScene::~QGVScene()
{
		gvFreeLayout(_context->context(), _graph->graph());
		agclose(_graph->graph());
		gvFreeContext(_context->context());
    delete _graph;
    delete _context;
Nicolas's avatar
Nicolas committed
}

void QGVScene::setGraphAttribute(const QString &name, const QString &value)
{
		agattr(_graph->graph(), AGRAPH, name.toLocal8Bit().data(), value.toLocal8Bit().data());
Nicolas's avatar
Nicolas committed
}

void QGVScene::setNodeAttribute(const QString &name, const QString &value)
{
		agattr(_graph->graph(), AGNODE, name.toLocal8Bit().data(), value.toLocal8Bit().data());
Nicolas's avatar
Nicolas committed
}

void QGVScene::setEdgeAttribute(const QString &name, const QString &value)
{
		agattr(_graph->graph(), AGEDGE, name.toLocal8Bit().data(), value.toLocal8Bit().data());
Nicolas's avatar
Nicolas committed
}

QGVNode *QGVScene::addNode(const QString &label)
{
		Agnode_t *node = agnode(_graph->graph(), NULL, TRUE);
nicolas.bergont's avatar
nicolas.bergont committed
    if(node == NULL)
    {
        qWarning()<<"Invalid node :"<<label;
        return 0;
    }
		QGVNode *item = new QGVNode(new QGVNodePrivate(node), this);
nicolas.bergont's avatar
nicolas.bergont committed
    item->setLabel(label);
    addItem(item);
    _nodes.append(item);
    return item;
Nicolas's avatar
Nicolas committed
}

QGVEdge *QGVScene::addEdge(QGVNode *source, QGVNode *target, const QString &label)
{
		Agedge_t* edge = agedge(_graph->graph(), source->_node->node(), target->_node->node(), NULL, TRUE);
nicolas.bergont's avatar
nicolas.bergont committed
    if(edge == NULL)
    {
        qWarning()<<"Invalid egde :"<<label;
        return 0;
    }

		QGVEdge *item = new QGVEdge(new QGVEdgePrivate(edge), this);
nicolas.bergont's avatar
nicolas.bergont committed
    item->setLabel(label);
    addItem(item);
    _edges.append(item);
    return item;
Nicolas's avatar
Nicolas committed
}

nicolas.bergont's avatar
nicolas.bergont committed
QGVSubGraph *QGVScene::addSubGraph(const QString &name, bool cluster)
Nicolas's avatar
Nicolas committed
{
nicolas.bergont's avatar
nicolas.bergont committed
    Agraph_t* sgraph;
    if(cluster)
				sgraph = agsubg(_graph->graph(), ("cluster_" + name).toLocal8Bit().data(), TRUE);
nicolas.bergont's avatar
nicolas.bergont committed
    else
				sgraph = agsubg(_graph->graph(), name.toLocal8Bit().data(), TRUE);
nicolas.bergont's avatar
nicolas.bergont committed

    if(sgraph == NULL)
    {
        qWarning()<<"Invalid subGraph :"<<name;
        return 0;
    }

		QGVSubGraph *item = new QGVSubGraph(new QGVGraphPrivate(sgraph), this);
nicolas.bergont's avatar
nicolas.bergont committed
    addItem(item);
    _subGraphs.append(item);
    return item;
Nicolas's avatar
Nicolas committed
}

void QGVScene::setRootNode(QGVNode *node)
{
    Q_ASSERT(_nodes.contains(node));
Martin Zenzes's avatar
Martin Zenzes committed
    char root[] = "root";
		agset(_graph->graph(), root, node->label().toLocal8Bit().data());
nicolas.bergont's avatar
nicolas.bergont committed
}
Nicolas's avatar
Nicolas committed

nicolas.bergont's avatar
nicolas.bergont committed
void QGVScene::loadLayout(const QString &text)
{
		_graph->setGraph(QGVCore::agmemread2(text.toLocal8Bit().constData()));
Nicolas's avatar
Nicolas committed

		if(gvLayout(_context->context(), _graph->graph(), "dot") != 0)
Nicolas's avatar
Nicolas committed
    {
        qCritical()<<"Layout render error"<<agerrors()<<QString::fromLocal8Bit(aglasterr());
        return;
    }

nicolas.bergont's avatar
nicolas.bergont committed
    //Debug output
		//gvRenderFilename(_context->context(), _graph->graph(), "png", "debug.png");
nicolas.bergont's avatar
nicolas.bergont committed

Nicolas's avatar
Nicolas committed
    //Read nodes and edges
		for (Agnode_t* node = agfstnode(_graph->graph()); node != NULL; node = agnxtnode(_graph->graph(), node))
Nicolas's avatar
Nicolas committed
    {
				QGVNode *inode = new QGVNode(new QGVNodePrivate(node), this);
nicolas.bergont's avatar
nicolas.bergont committed
        inode->updateLayout();
        addItem(inode);
				for (Agedge_t* edge = agfstout(_graph->graph(), node); edge != NULL; edge = agnxtout(_graph->graph(), edge))
nicolas.bergont's avatar
nicolas.bergont committed
        {
						QGVEdge *iedge = new QGVEdge(new QGVEdgePrivate(edge), this);
nicolas.bergont's avatar
nicolas.bergont committed
            iedge->updateLayout();
            addItem(iedge);
        }
Nicolas's avatar
Nicolas committed

nicolas.bergont's avatar
nicolas.bergont committed
    }
    update();
Nicolas's avatar
Nicolas committed
}

void QGVScene::applyLayout()
{
    gvFreeLayout(_context->context(), _graph->graph());
    if(gvLayout(_context->context(), _graph->graph(), "dot") != 0)
Nicolas's avatar
Nicolas committed
    {
nicolas.bergont's avatar
nicolas.bergont committed
        /*
         * Si plantage ici :
         *  - Verifier que les dll sont dans le repertoire d'execution
         *  - Verifie que le fichier "configN" est dans le repertoire d'execution !
         */
Nicolas's avatar
Nicolas committed
        qCritical()<<"Layout render error"<<agerrors()<<QString::fromLocal8Bit(aglasterr());
        return;
    }

    //Debug output
		//gvRenderFilename(_context->context(), _graph->graph(), "canon", "debug.dot");
		//gvRenderFilename(_context->context(), _graph->graph(), "png", "debug.png");
Nicolas's avatar
Nicolas committed

nicolas.bergont's avatar
nicolas.bergont committed
    //Update items layout
Nicolas's avatar
Nicolas committed
    foreach(QGVNode* node, _nodes)
        node->updateLayout();

    foreach(QGVEdge* edge, _edges)
        edge->updateLayout();

    foreach(QGVSubGraph* sgraph, _subGraphs)
        sgraph->updateLayout();

nicolas.bergont's avatar
nicolas.bergont committed
    //Graph label
		textlabel_t *xlabel = GD_label(_graph->graph());
nicolas.bergont's avatar
nicolas.bergont committed
    if(xlabel)
    {
        QGraphicsTextItem *item = addText(xlabel->text);
				item->setPos(QGVCore::centerToOrigin(QGVCore::toPoint(xlabel->pos, QGVCore::graphHeight(_graph->graph())), xlabel->dimen.x, -4));
nicolas.bergont's avatar
nicolas.bergont committed
    }

Nicolas's avatar
Nicolas committed
    update();
}

nicolas.bergont's avatar
nicolas.bergont committed
void QGVScene::clear()
{
		gvFreeLayout(_context->context(), _graph->graph());
nicolas.bergont's avatar
nicolas.bergont committed
    _nodes.clear();
    _edges.clear();
    _subGraphs.clear();
    QGraphicsScene::clear();
}

Nicolas's avatar
Nicolas committed
#include <QGraphicsSceneContextMenuEvent>
void QGVScene::contextMenuEvent(QGraphicsSceneContextMenuEvent *contextMenuEvent)
{
nicolas.bergont's avatar
nicolas.bergont committed
    QGraphicsItem *item = itemAt(contextMenuEvent->scenePos(), QTransform());
Nicolas's avatar
Nicolas committed
    if(item)
    {
        item->setSelected(true);
nicolas.bergont's avatar
nicolas.bergont committed
        if(item->type() == QGVNode::Type)
            emit nodeContextMenu(qgraphicsitem_cast<QGVNode*>(item));
        else if(item->type() == QGVEdge::Type)
            emit edgeContextMenu(qgraphicsitem_cast<QGVEdge*>(item));
        else if(item->type() == QGVSubGraph::Type)
            emit subGraphContextMenu(qgraphicsitem_cast<QGVSubGraph*>(item));
        else
            emit graphContextMenuEvent();
Nicolas's avatar
Nicolas committed
    }
    QGraphicsScene::contextMenuEvent(contextMenuEvent);
}

void QGVScene::mouseDoubleClickEvent(QGraphicsSceneMouseEvent *mouseEvent)
{
nicolas.bergont's avatar
nicolas.bergont committed
    QGraphicsItem *item = itemAt(mouseEvent->scenePos(), QTransform());
Nicolas's avatar
Nicolas committed
    if(item)
    {
nicolas.bergont's avatar
nicolas.bergont committed
        if(item->type() == QGVNode::Type)
            emit nodeDoubleClick(qgraphicsitem_cast<QGVNode*>(item));
        else if(item->type() == QGVEdge::Type)
            emit edgeDoubleClick(qgraphicsitem_cast<QGVEdge*>(item));
        else if(item->type() == QGVSubGraph::Type)
            emit subGraphDoubleClick(qgraphicsitem_cast<QGVSubGraph*>(item));
Nicolas's avatar
Nicolas committed
    }
    QGraphicsScene::mouseDoubleClickEvent(mouseEvent);
}

void QGVScene::mouseReleaseEvent(QGraphicsSceneMouseEvent *mouseEvent)
{
    QGraphicsItem *item = itemAt(mouseEvent->scenePos(), QTransform());
    if(item)
    {
        if(item->type() == QGVNode::Type)
            emit nodeMouseRelease (qgraphicsitem_cast<QGVNode*>(item));
    }
    QGraphicsScene::mouseReleaseEvent(mouseEvent);
}

Nicolas's avatar
Nicolas committed
#include <QVarLengthArray>
#include <QPainter>
void QGVScene::drawBackground(QPainter * painter, const QRectF & rect)
{
    const int gridSize = 25;

    const qreal left = int(rect.left()) - (int(rect.left()) % gridSize);
    const qreal top = int(rect.top()) - (int(rect.top()) % gridSize);

    QVarLengthArray<QLineF, 100> lines;

    for (qreal x = left; x < rect.right(); x += gridSize)
        lines.append(QLineF(x, rect.top(), x, rect.bottom()));
    for (qreal y = top; y < rect.bottom(); y += gridSize)
        lines.append(QLineF(rect.left(), y, rect.right(), y));

    painter->setRenderHint(QPainter::Antialiasing, false);

    painter->setPen(QColor(Qt::lightGray).lighter(110));
    painter->drawLines(lines.data(), lines.size());
    painter->setPen(Qt::black);
    //painter->drawRect(sceneRect());
}