source: orange/source/orangeqt/networkcurve.cpp @ 10924:b5b1c0b30806

Revision 10924:b5b1c0b30806, 33.5 KB checked in by mstajdohar, 22 months ago (diff)

Try FR with animation.

Line 
1/*
2    This file is part of the plot module for Orange
3    Copyright (C) 2011  Miha Čančula <miha@noughmad.eu>
4
5    This program is free software: you can redistribute it and/or modify
6    it under the terms of the GNU General Public License as published by
7    the Free Software Foundation, either version 3 of the License, or
8    (at your option) any later version.
9
10    This program is distributed in the hope that it will be useful,
11    but WITHOUT ANY WARRANTY; without even the implied warranty of
12    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13    GNU General Public License for more details.
14
15    You should have received a copy of the GNU General Public License
16    along with this program.  If not, see <http://www.gnu.org/licenses/>.
17*/
18
19#include "networkcurve.h"
20#include "point.h"
21
22#include <QtCore/QMap>
23#include <QtCore/QList>
24#include <QtGui/QTextBlockFormat>
25#include <QtGui/QTextCursor>
26
27#include <QtCore/qmath.h>
28#include <limits>
29#include <QStyleOptionGraphicsItem>
30#include <QCoreApplication>
31
32#include <QtCore/QTime>
33#include <QtCore/QParallelAnimationGroup>
34
35/************/
36/* NodeItem */ 
37/************/
38
39#define PI 3.14159265
40
41
42ModelItem::ModelItem(int index, int symbol, QColor color, int size, QGraphicsItem* parent): NodeItem(index, symbol, color, size, parent)
43{
44    representative = 0;
45}
46
47void ModelItem::paint(QPainter* painter, const QStyleOptionGraphicsItem* option, QWidget* widget)
48{
49    NetworkCurve *curve = (NetworkCurve*)parentItem();
50    //bool on_marked_only = curve->labels_on_marked();
51
52    if (image != NULL)
53    {
54        const int ps = size() + 4;
55        double style = 1;
56        int _size = size() + 5;
57
58        int pen_size = (representative) ? 3 : 1;
59        painter->setPen(QPen(QBrush(color()), pen_size, Qt::SolidLine, Qt::RoundCap));
60
61        QRadialGradient gradient(QPointF(0, 0), _size);
62        gradient.setColorAt(0, color());
63        gradient.setColorAt(1, QColor(255, 255, 255, 0));
64
65        if (is_selected())
66        {
67            QColor brushColor(Qt::yellow);
68            brushColor.setAlpha(150);
69            gradient = QRadialGradient(QPointF(0, 0), _size);
70            gradient.setColorAt(0, brushColor);
71            gradient.setColorAt(1, QColor(255, 255, 255, 0));
72        }
73        else if (is_marked())
74        {
75            QColor brushColor(Qt::cyan);
76            brushColor.setAlpha(80);
77            gradient = QRadialGradient(QPointF(0, 0), _size);
78            gradient.setColorAt(0, brushColor);
79            gradient.setColorAt(1, QColor(255, 255, 255, 0));
80        }
81
82        painter->setBrush(QBrush(gradient));
83        painter->drawRoundedRect(-_size/2, -_size/2, _size, _size, style, style, Qt::RelativeSize);
84
85        _size = image->size().width();
86        painter->drawPixmap(QPointF(-_size/2, -_size/2), *image);
87        /*
88        if (!on_marked_only || is_marked() || is_selected())
89        {
90            QFontMetrics metrics = option->fontMetrics;
91            int th = metrics.height();
92            int tw = metrics.width(label());
93            QRect r(-tw/2, 0, tw, th);
94            //painter->fillRect(r, QBrush(Qt::white));
95            painter->drawText(r, Qt::AlignHCenter, label());
96        }
97        */
98    }
99    /*
100    else
101    {
102        const QString l = label();
103        if (on_marked_only && !(is_marked() || is_selected()))
104        {
105            set_label(QString());
106        }
107
108        Point::paint(painter, option, widget);
109
110        set_label(l);
111    }
112    */
113}
114
115void ModelItem::set_representative(bool value)
116{
117    representative = value;
118}
119
120bool ModelItem::is_representative() const
121{
122    return representative;
123}
124
125NodeItem::NodeItem(int index, int symbol, QColor color, int size, QGraphicsItem* parent): Point(symbol, color, size, parent)
126{
127    set_index(index);
128    set_coordinates(((qreal)(qrand() % 1000)) * 1000, ((qreal)(qrand() % 1000)) * 1000);
129    setZValue(0.5);
130    m_size_value = 1;
131    set_marked(false);
132    set_selected(false);
133    //set_label("");
134    setAcceptHoverEvents(true);
135    set_transparent(false);
136    image = NULL;
137}
138
139NodeItem::~NodeItem()
140{
141}
142
143void NodeItem::paint(QPainter* painter, const QStyleOptionGraphicsItem* option, QWidget* widget)
144{
145    NetworkCurve *curve = (NetworkCurve*)parentItem();
146    /*
147    bool on_marked_only = curve->labels_on_marked_only();
148    const QString l = label();
149
150    if (on_marked_only && !(is_marked() || is_selected()))
151    {
152        set_label(QString());
153    }
154    */
155    if (image != NULL)
156    {
157        const int ps = size() + 4;
158        painter->drawPixmap(QPointF(-0.5*ps, -0.5*ps), *image);
159        /*
160        if (!label().isEmpty())
161        {
162            QFontMetrics metrics = option->fontMetrics;
163            int th = metrics.height();
164            int tw = metrics.width(label());
165            QRect r(-tw/2, 0, tw, th);
166            //painter->fillRect(r, QBrush(Qt::white));
167            painter->drawText(r, Qt::AlignHCenter, label());
168        }
169        */
170    }
171    else
172    {
173        Point::paint(painter, option, widget);
174    }
175
176    //set_label(l);
177}
178
179void NodeItem::set_image(QPixmap* im)
180{
181    image = im;
182}
183
184void NodeItem::set_coordinates(double x, double y)
185{
186    m_x = x;
187    m_y = y;
188   
189    DataPoint p;
190    p.x = m_x;
191    p.y = m_y;
192    Point::set_coordinates(p);
193}
194
195void NodeItem::set_index(int index)
196{
197    m_index = index;
198}
199
200int NodeItem::index() const
201{
202    return m_index;
203}
204
205void NodeItem::set_graph_transform(const QTransform& transform)
206{
207    m_graph_transform = transform;
208    set_coordinates(m_x, m_y);
209}
210
211QTransform NodeItem::graph_transform() const
212{
213    return m_graph_transform;
214}
215
216void NodeItem::set_x(double x)
217{
218    set_coordinates(x, m_y);
219}
220
221double NodeItem::x() const
222{
223    return m_x;
224}
225
226void NodeItem::set_y(double y)
227{
228    set_coordinates(m_x, y);
229}
230
231double NodeItem::y() const
232{
233    return m_y;
234}
235
236void NodeItem::set_tooltip(const QString& tooltip)
237{
238    setToolTip(tooltip);
239}
240
241void NodeItem::set_uuid(int uuid)
242{
243    m_uuid = uuid;
244}
245
246int NodeItem::uuid() const
247{
248    return m_uuid;
249}
250
251void NodeItem::add_connected_edge(EdgeItem* edge)
252{
253    if (!m_connected_edges.contains(edge))
254    {
255        m_connected_edges << edge;
256    }
257}
258
259void NodeItem::remove_connected_edge(EdgeItem* edge)
260{
261    m_connected_edges.removeAll(edge);
262}
263
264QList<EdgeItem*> NodeItem::connected_edges()
265{
266    return m_connected_edges;
267}
268
269QList<NodeItem*> NodeItem::neighbors()
270{
271    QList<NodeItem*> neighbors;
272
273    EdgeItem *e;
274    QList<EdgeItem*> edges = connected_edges();
275    int size = edges.size();
276    foreach(e, edges)
277    {
278        if (e->u()->index() == index())
279        {
280            neighbors.append(e->v());
281        }
282        else
283        {
284            neighbors.append(e->u());
285        }
286    }
287
288    return neighbors;
289}
290
291/************/
292/* EdgeItem */
293/************/
294
295QHash<ArrowData, QPixmap> EdgeItem::arrow_cache;
296
297uint qHash(const ArrowData& data)
298{
299    // uint has 32 bits:
300    uint ret = data.size;
301    // QRgb takes the full uins, so we just XOR by it
302    ret ^= data.color.rgba();
303    return ret;
304}
305
306bool operator==(const ArrowData& one, const ArrowData& other)
307{
308    return one.size == other.size && one.color == other.color;
309}
310
311EdgeItem::EdgeItem(NodeItem* u, NodeItem* v, QGraphicsItem* parent, QGraphicsScene* scene): QAbstractGraphicsShapeItem(parent, scene),
312m_u(0), m_v(0)
313{
314    set_u(u);
315    set_v(v);
316    QPen p = pen();
317    p.setWidthF(1);
318    p.setCosmetic(true);
319    setPen(p);
320    setZValue(0);
321    //setFlag(ItemIgnoresTransformations);
322}
323
324EdgeItem::~EdgeItem()
325{
326    if (m_u)
327    {
328        m_u->remove_connected_edge(this);
329    }
330    if (m_v)
331    {
332        m_v->remove_connected_edge(this);
333    }
334}
335
336
337void EdgeItem::paint(QPainter* painter, const QStyleOptionGraphicsItem* option, QWidget* widget)
338{
339    painter->setRenderHint(QPainter::Antialiasing, false);
340    painter->setPen(pen());
341    QLineF _line;
342    if (m_u && m_v)
343    {
344        _line.setPoints(m_u->pos(), m_v->pos());
345        painter->drawLine(_line);
346
347        if ((m_arrows & (ArrowU | ArrowV)) && m_u->pos() != m_v->pos())
348        {
349            double size = 10;
350            double half_size = 0.5 * size;
351            const ArrowData key(1, pen().color());
352            if (!arrow_cache.contains(key))
353            {
354                QBrush brush(key.color);
355                QPixmap pixmap(size, size);
356                pixmap.fill(Qt::transparent);
357                QPainter p;
358
359                QPen pen(key.color);
360                //pen.setWidth(0);
361                pen.setStyle(Qt::NoPen);
362
363                p.begin(&pixmap);
364                p.setRenderHints(painter->renderHints() | QPainter::Antialiasing);
365
366                QPainterPath path;
367                path.moveTo(half_size, 0);
368                path.lineTo(size, size);
369                path.lineTo(0, size);
370
371                p.setBrush(brush);
372                p.setPen(pen);
373                p.drawPath(path);
374                arrow_cache.insert(key, pixmap);
375            }
376            double x1 = m_u->pos().x();
377            double y1 = m_u->pos().y();
378            double x2 = m_v->pos().x();
379            double y2 = m_v->pos().y();
380            if (m_arrows & ArrowU)
381            {
382                double phi = atan2(y1-y2, x1-x2) * 180 / PI + 90;
383                painter->save();
384                painter->translate(m_u->pos());
385                painter->rotate(phi);
386                //painter->setViewTransformEnabled(false);
387                painter->drawPixmap(-half_size, 0.5 * m_u->size(), arrow_cache.value(key));
388                painter->restore();
389            }
390
391            if (m_arrows & ArrowV)
392            {
393                double phi = atan2(y2-y1, x2-x1) * 180 / PI + 90;
394                painter->save();
395                painter->translate(m_v->pos());
396                painter->rotate(phi);
397                //painter->setViewTransformEnabled(false);
398                painter->drawPixmap(-half_size, 0.5 * m_v->size(), arrow_cache.value(key));
399                painter->restore();
400            }
401        }
402    }
403
404    if (!m_label.isEmpty())
405    {
406        NetworkCurve *curve = (NetworkCurve*)parentItem();
407        bool on_marked_only = curve->labels_on_marked();
408        bool is_marked = (u()->is_marked() || u()->is_selected()) && (v()->is_marked() || v()->is_selected());
409
410        if(!on_marked_only || (on_marked_only && is_marked))
411        {
412            double x = (_line.x1() + _line.x2()) / 2;
413            double y = (_line.y1() + _line.y2()) / 2;
414            QFontMetrics metrics = option->fontMetrics;
415            int th = metrics.height();
416            int tw = metrics.width(m_label);
417            QRect r(x-tw/2, y-th/2, tw, th);
418            //painter->fillRect(r, QBrush(Qt::white));
419            QPen p = painter->pen();
420            p.setColor(Qt::black);
421            painter->setPen(p);
422            painter->drawText(r, Qt::AlignHCenter, m_label);
423        }
424    }
425}
426
427QRectF EdgeItem::boundingRect() const
428{
429    return QRectF(m_u->pos(), m_v->pos());
430}
431
432QPainterPath EdgeItem::shape() const
433{
434    QPainterPath path;
435    path.moveTo(m_u->pos());
436    path.lineTo(m_v->pos());
437    return path;
438}
439
440void EdgeItem::set_u(NodeItem* item)
441{
442    if (m_u)
443    {
444        m_u->remove_connected_edge(this);
445    }
446    if (item)
447    {
448        item->add_connected_edge(this);
449    }
450    m_u = item;
451}
452
453NodeItem* EdgeItem::u()
454{
455    return m_u;
456}
457
458void EdgeItem::set_v(NodeItem* item)
459{
460    if (m_v)
461    {
462        m_v->remove_connected_edge(this);
463    }
464    if (item)
465    {
466        item->add_connected_edge(this);
467    }
468    m_v = item;
469}
470
471NodeItem* EdgeItem::v()
472{
473    return m_v;
474}
475
476void EdgeItem::set_tooltip(const QString& tooltip)
477{
478    setToolTip(tooltip);
479}
480
481void EdgeItem::set_label(const QString& label)
482{
483    m_label = label;
484}
485
486QString EdgeItem::label() const
487{
488    return m_label;
489}
490
491void EdgeItem::set_links_index(int index)
492{
493    m_links_index = index;
494}
495
496int EdgeItem::links_index() const
497{
498    return m_links_index;
499}
500
501void EdgeItem::set_arrow(EdgeItem::Arrow arrow, bool enable)
502{
503    if (enable)
504    {
505        set_arrows(arrows() | arrow);
506    }
507    else
508    {
509        set_arrows(arrows() & ~arrow);
510    }
511}
512
513EdgeItem::Arrows EdgeItem::arrows()
514{
515    return m_arrows;
516}
517
518void EdgeItem::set_arrows(EdgeItem::Arrows arrows)
519{
520    m_arrows = arrows;
521    // TODO: Update the QGraphicsItem element, add arrows
522}
523
524void EdgeItem::set_weight(double weight)
525{
526    m_weight = weight;
527}
528
529double EdgeItem::weight() const
530{
531    return m_weight;
532}
533
534/****************/
535/* NetworkCurve */
536/****************/
537
538NetworkCurve::NetworkCurve(QGraphicsItem* parent): Curve(parent)
539{
540     m_min_node_size = 5;
541     m_max_node_size = 5;
542}
543
544NetworkCurve::~NetworkCurve()
545{
546    cancel_all_updates();
547    qDeleteAll(m_edges);
548    m_edges.clear();
549    qDeleteAll(m_nodes);
550    m_nodes.clear();
551    qDeleteAll(m_labels);
552    m_labels.clear();
553}
554
555
556void NetworkCurve::paint(QPainter* painter, const QStyleOptionGraphicsItem* option, QWidget* widget)
557{
558    NetworkCurve::paint(painter, option, widget);
559
560    if (m_show_component_distances)
561    {
562        // TODO: move code from Python to C++
563    }
564}
565
566
567void NetworkCurve::update_properties()
568{
569    update_point_positions();
570}
571
572QRectF NetworkCurve::data_rect() const
573{
574    QRectF r;
575    bool first = true;
576    foreach (const NodeItem* node, m_nodes)
577    {
578        if (first)
579        {
580            r = QRectF(node->x(), node->y(), 0, 0);
581            first = false;
582        }
583        else
584        {
585            r.setTop( qMin(r.top(), node->y()) );
586            r.setBottom( qMax(r.bottom(), node->y()) );
587            r.setLeft( qMin(r.left(), node->x()) );
588            r.setRight( qMax(r.right(), node->x()) );
589        }
590    }
591    return r;
592}
593
594void NetworkCurve::scale_axes()
595{
596    QRectF r2 = QRectF(plot()->contentsRect());
597    r2.adjust(1, 1, -1, -1);
598    double t = r2.top();
599    r2.setTop(r2.bottom());
600    r2.setBottom(t);
601
602    QPair< int, int > axes_;
603    axes_ = axes();
604    QRectF r1 = plot()->data_rect_for_axes(axes_.first, axes_.second);
605
606    double dx = r1.width() / 20.0;
607    double dy = r1.height() / 20.0;
608    r1.adjust(-dx, -dy, dx, dy);
609
610    QTransform tr1 = QTransform().translate(-r1.left(), -r1.top());
611    QTransform ts = QTransform().scale(r2.width()/r1.width(), r2.height()/r1.height());
612    QTransform tr2 = QTransform().translate(r2.left(), r2.top());
613    set_graph_transform(tr1 * ts * tr2);
614}
615
616int NetworkCurve::random()
617{
618    Nodes::ConstIterator uit = m_nodes.constBegin();
619    Nodes::ConstIterator uend = m_nodes.constEnd();
620
621    for (; uit != uend; ++uit)
622    {
623        uit.value()->set_coordinates(((qreal)(qrand() % 1000)) * 1000, ((qreal)(qrand() % 1000)) * 1000);
624    }
625
626    return 0;
627}
628
629#define PI 3.14159265
630
631int NetworkCurve::circular(CircularLayoutType type)
632{
633    // type
634    // 0 - original
635    // 1 - random
636    // 2 - crossing reduction
637
638    if (type == NetworkCurve::circular_crossing)
639    {
640        qDebug() << "crossing_reduction";
641        return circular_crossing_reduction();
642    }
643
644    if (type == NetworkCurve::circular_original)
645        qDebug() << "original";
646
647    if (type == NetworkCurve::circular_random)
648            qDebug() << "random";
649
650    QRectF rect = data_rect();
651    int xCenter = rect.width() / 2;
652    int yCenter = rect.height() / 2;
653    int r = (rect.width() < rect.height()) ? rect.width() * 0.38 : rect.height() * 0.38;
654
655    int i;
656    double fi = PI;
657    double step = 2 * PI / m_nodes.size();
658
659    qsrand(QTime(0,0,0).secsTo(QTime::currentTime()));
660    std::vector<int> vertices;
661    Nodes::ConstIterator it;
662    for (it = m_nodes.constBegin(); it != m_nodes.constEnd(); ++it)
663    {
664        vertices.push_back(it.key());
665    }
666
667    for (i = 0; i < m_nodes.size(); ++i)
668    {
669        if (type == NetworkCurve::circular_original)
670        {
671            m_nodes[vertices[i]]->set_coordinates(r * cos(fi) + xCenter, r * sin(fi) + yCenter);
672        }
673        else if (type == NetworkCurve::circular_random)
674        {
675            int ndx = rand() % vertices.size();
676                        m_nodes[vertices[ndx]]->set_coordinates(r * cos(fi) + xCenter, r * sin(fi) + yCenter);
677            vertices.erase(vertices.begin() + ndx);
678        }
679        fi = fi - step;
680    }
681
682    return 0;
683}
684
685
686int NetworkCurve::circular_crossing_reduction()
687{
688    QMap<int, QueueVertex*> qvertices;
689    std::vector<QueueVertex*> vertices;
690    std::vector<QueueVertex*> original;
691
692    Nodes::ConstIterator it;
693    for (it = m_nodes.constBegin(); it != m_nodes.constEnd(); ++it)
694    {
695        QueueVertex *vertex = new QueueVertex();
696        vertex->ndx = it.key();
697        qvertices[it.key()] = vertex;
698
699        std::vector<int> neighbours;
700        vertex->unplacedNeighbours = neighbours.size();
701        vertex->neighbours = neighbours;
702        vertices.push_back(vertex);
703    }
704    int i;
705    EdgeItem *edge;
706    for (i = 0; i < m_edges.size(); i++)
707    {
708        edge = m_edges[i];
709        int u = edge->u()->index();
710        int v = edge->v()->index();
711        qvertices[u]->neighbours.push_back(v);
712        qvertices[u]->unplacedNeighbours += 1;
713        qvertices[v]->neighbours.push_back(u);
714        qvertices[v]->unplacedNeighbours += 1;
715    }
716    original.assign(vertices.begin(), vertices.end());
717
718    std::deque<int> positions;
719    while (vertices.size() > 0)
720    {
721        std::sort(vertices.begin(), vertices.end(), QueueVertex());
722        QueueVertex *vertex = vertices.back();
723
724
725        // update neighbours
726        for (i = 0; i < vertex->neighbours.size(); ++i)
727        {
728            int ndx = vertex->neighbours[i];
729
730            original[ndx]->placedNeighbours++;
731            original[ndx]->unplacedNeighbours--;
732        }
733
734        // count left & right crossings
735        if (vertex->placedNeighbours > 0)
736        {
737            int left = 0;
738            std::vector<int> lCrossings;
739            std::vector<int> rCrossings;
740            for (i = 0; i < positions.size(); ++i)
741            {
742                int ndx = positions[i];
743
744                if (vertex->hasNeighbour(ndx))
745                {
746                    lCrossings.push_back(left);
747                    left += original[ndx]->unplacedNeighbours;
748                    rCrossings.push_back(left);
749                }
750                else
751                    left += original[ndx]->unplacedNeighbours;
752            }
753
754            int leftCrossings = 0;
755            int rightCrossings = 0;
756
757            for (i = 0; i < lCrossings.size(); --i)
758                leftCrossings += lCrossings[i];
759
760            rCrossings.push_back(left);
761            for (i = rCrossings.size() - 1; i > 0 ; --i)
762                rightCrossings += rCrossings[i] - rCrossings[i - 1];
763            //cout << "left: " << leftCrossings << " right: " <<rightCrossings << endl;
764            if (leftCrossings < rightCrossings)
765                positions.push_front(vertex->ndx);
766            else
767                positions.push_back(vertex->ndx);
768
769        }
770        else
771            positions.push_back(vertex->ndx);
772
773        vertices.pop_back();
774    }
775
776    // Circular sifting
777    for (i = 0; i < positions.size(); ++i)
778        original[positions[i]]->position = i;
779
780    int step;
781    for (step = 0; step < 5; ++step)
782    {
783        for (i = 0; i < m_nodes.size(); ++i)
784        {
785            bool stop = false;
786            int switchNdx = -1;
787            QueueVertex *u = original[positions[i]];
788            int vNdx = (i + 1) % m_nodes.size();
789
790            while (!stop)
791            {
792                QueueVertex *v = original[positions[vNdx]];
793
794                int midCrossings = u->neighbours.size() * v->neighbours.size() / 2;
795                int crossings = 0;
796                int j,k;
797                for (j = 0; j < u->neighbours.size(); ++j)
798                    for (k = 0; k < v->neighbours.size(); ++k)
799                        if ((original[u->neighbours[j]]->position == v->position) || (original[v->neighbours[k]]->position == u->position))
800                            midCrossings = (u->neighbours.size() - 1) * (v->neighbours.size() - 1) / 2;
801                        else if ((original[u->neighbours[j]]->position + m_nodes.size() - u->position) % m_nodes.size() < (original[v->neighbours[k]]->position + m_nodes.size() - u->position) % m_nodes.size())
802                            ++crossings;
803
804                //cout << "v: " <<  v->ndx << " crossings: " << crossings << " u.n.size: " << u->neighbours.size() << " v.n.size: " << v->neighbours.size() << " mid: " << midCrossings << endl;
805                if (crossings > midCrossings)
806                    switchNdx = vNdx;
807                else
808                    stop = true;
809
810                vNdx = (vNdx + 1) % m_nodes.size();
811            }
812            int j;
813            if (switchNdx > -1)
814            {
815                //cout << "u: " << u->ndx << " switch: " << original[switchNdx]->ndx << endl << endl;
816                positions.erase(positions.begin() + i);
817                positions.insert(positions.begin() + switchNdx, u->ndx);
818
819                for (j = i; j <= switchNdx; ++j)
820                    original[positions[j]]->position = j;
821            }
822            //else
823            //  cout << "u: " << u->ndx << " switch: " << switchNdx << endl;
824        }
825    }
826
827    QRectF rect = data_rect();
828    int xCenter = rect.width() / 2;
829    int yCenter = rect.height() / 2;
830    int r = (rect.width() < rect.height()) ? rect.width() * 0.38 : rect.height() * 0.38;
831    double fi = PI;
832    double fiStep = 2 * PI / m_nodes.size();
833
834    for (i = 0; i < m_nodes.size(); ++i)
835    {
836        m_nodes[positions[i]]->set_x(r * cos(fi) + xCenter);
837        m_nodes[positions[i]]->set_y(r * sin(fi) + yCenter);
838        fi = fi - fiStep;
839    }
840
841        qDeleteAll(original);
842
843    original.clear();
844    vertices.clear();
845    qvertices.clear();
846
847    return 0;
848}
849
850int NetworkCurve::fr(int steps, bool weighted, bool smooth_cooling)
851{
852    int i, j;
853    int count = 0;
854    NodeItem *u, *v;
855    EdgeItem *edge;
856    m_stop_optimization = false;
857
858    double rect[4] = {std::numeric_limits<double>::max(),
859              std::numeric_limits<double>::max(),
860              std::numeric_limits<double>::min(),
861              std::numeric_limits<double>::min()};
862
863    QMap<int, DataPoint> disp;
864    foreach (const NodeItem*   node, m_nodes)
865    {
866        //disp[node->index()] = node->coordinates();
867
868        double x = node->x();
869        double y = node->y();
870        if (rect[0] > x) rect[0] = x;
871        if (rect[1] > y) rect[1] = y;
872        if (rect[2] < x) rect[2] = x;
873        if (rect[3] < y) rect[3] = y;
874    }
875    QRectF data_r(rect[0], rect[1], rect[2]-rect[0], rect[3]-rect[1]);
876    double area = data_r.width() * data_r.height();
877    double k2 = area / m_nodes.size();
878    double k = sqrt(k2);
879    double kk = 2 * k;
880    double kk2 = kk * kk;
881    double jitter = sqrt(area) / 2000;
882    double temperature, cooling, cooling_switch, cooling_1, cooling_2;
883    temperature = sqrt(area) / 5;
884    cooling = exp(log(k / 10 / temperature) / steps);
885
886    if (steps > 20)
887    {
888        cooling_switch = sqrt(area) / 100;
889        cooling_1 = (temperature - cooling_switch) / 20;
890        cooling_2 = (cooling_switch - sqrt(area) / 2000 ) / (steps - 20);
891    }
892    else
893    {
894        cooling_switch = sqrt(area) / 1000;
895        cooling_1 = (temperature - cooling_switch) / steps;
896        cooling_2 = 0;
897    }
898
899    if (smooth_cooling)
900    {
901        if (steps < 20)
902        {
903            steps = 20;
904        }
905        temperature = cooling_switch;
906        cooling_1 = 0;
907        cooling_2 = (cooling_switch - sqrt(area) / 2000 ) / steps;
908    }
909
910    // iterations
911    //clock_t refresh_time = clock() + 0.05 * CLOCKS_PER_SEC;
912    Plot *p = plot();
913    //bool animation_enabled = p->animate_points;
914    //p->animate_points = false;
915
916    QTime refresh_time = QTime::currentTime();
917    for (i = 0; i < steps; ++i)
918    {
919        foreach (const NodeItem* node, m_nodes)
920        {
921            disp[node->index()].x = 0;
922            disp[node->index()].y = 0;
923        }
924
925        // calculate repulsive force
926        Nodes::ConstIterator uit = m_nodes.constBegin();
927        Nodes::ConstIterator uend = m_nodes.constEnd();
928        for (; uit != uend; ++uit)
929        {
930            u = uit.value();
931            Nodes::ConstIterator vit(uit);
932            ++vit;
933            for (; vit != uend; ++vit)
934            {
935                v = vit.value();
936
937                double difx = u->x() - v->x();
938                double dify = u->y() - v->y();
939
940                double dif2 = difx * difx + dify * dify;
941
942                // if nodes are close, apply repulsive force
943                if (dif2 < kk2)
944                {
945                    // if nodes overlap
946                    if (dif2 == 0)
947                    {
948                        dif2 = 1 / k;
949                        u->set_x(u->x() + jitter);
950                        u->set_y(u->y() + jitter);
951                        v->set_x(v->x() - jitter);
952                        v->set_y(v->y() - jitter);
953                    }
954
955                    double dX = difx * k2 / dif2;
956                    double dY = dify * k2 / dif2;
957
958                    disp[u->index()].x += dX;
959                    disp[u->index()].y += dY;
960
961                    disp[v->index()].x -= dX;
962                    disp[v->index()].y -= dY;
963                }
964            }
965        }
966        // calculate attractive forces
967        for (j = 0; j < m_edges.size(); ++j)
968        {
969            edge = m_edges[j];
970            double difx = edge->u()->x() - edge->v()->x();
971            double dify = edge->u()->y() - edge->v()->y();
972
973            double dif = sqrt(difx * difx + dify * dify);
974
975            double dX = difx * dif / k;
976            double dY = dify * dif / k;
977
978            if (weighted) {
979                dX *= edge->weight();
980                dY *= edge->weight();
981            }
982
983            disp[edge->u()->index()].x -= dX;
984            disp[edge->u()->index()].y -= dY;
985
986            disp[edge->v()->index()].x += dX;
987            disp[edge->v()->index()].y += dY;
988        }
989
990        // limit the maximum displacement to the temperature t
991        // and then prevent from being displaced outside frame
992        Nodes::Iterator nit = m_nodes.begin();
993        for (; nit != m_nodes.end(); ++nit)
994        {
995            u = nit.value();
996            double dif = sqrt(disp[u->index()].x * disp[u->index()].x + disp[u->index()].y * disp[u->index()].y);
997
998            if (dif == 0)
999                dif = 1;
1000
1001            u->set_coordinates(u->x() + (disp[u->index()].x * qMin(fabs(disp[u->index()].x), temperature) / dif),
1002                               u->y() + (disp[u->index()].y * qMin(fabs(disp[u->index()].y), temperature) / dif));
1003        }
1004
1005        QTime before_refresh_time = QTime::currentTime();
1006        if (before_refresh_time > refresh_time && i % 2 == 0)
1007        {
1008            scale_axes();
1009            update_properties();
1010            QCoreApplication::processEvents();
1011            int refresh_duration = before_refresh_time.msecsTo(QTime::currentTime());
1012            refresh_time = before_refresh_time.addMSecs(qMax(refresh_duration * 3, 10));
1013        }
1014        if (m_stop_optimization)
1015        {
1016            break;
1017        }
1018        if (floor(temperature) > cooling_switch)
1019        {
1020            temperature -= cooling_1;
1021        }
1022        else
1023        {
1024            temperature -= cooling_2;
1025        }
1026    }
1027
1028    //p->animate_points = animation_enabled;
1029    return 0;
1030}
1031
1032NetworkCurve::Labels NetworkCurve::labels() const
1033{
1034    return m_labels;
1035}
1036
1037void NetworkCurve::set_labels(const NetworkCurve::Labels& labels)
1038{
1039    cancel_all_updates();
1040    qDeleteAll(m_labels);
1041    m_labels = labels;
1042    //Q_ASSERT(m_labels.uniqueKeys() == m_labels.keys());
1043}
1044
1045void NetworkCurve::add_labels(const NetworkCurve::Labels& labels)
1046{
1047    Labels::ConstIterator it = labels.constBegin();
1048    Labels::ConstIterator end = labels.constEnd();
1049    QList<int> indices;
1050    for (it; it != end; ++it)
1051    {
1052        indices.append(it.key());
1053
1054        if (m_labels.contains(it.key()))
1055        {
1056            remove_label(it.key());
1057        }
1058    }
1059
1060    m_labels.unite(labels);
1061    Q_ASSERT(m_labels.uniqueKeys() == m_labels.keys());
1062}
1063
1064void NetworkCurve::remove_label(int index)
1065{
1066    cancel_all_updates();
1067    if (!m_labels.contains(index))
1068    {
1069        qWarning() << "Trying to remove label for node " << index << " which is not in the network";
1070        return;
1071    }
1072    QGraphicsItem* label = m_labels.take(index);
1073    //Q_ASSERT(node->index() == index);
1074    /*
1075    Plot* p = plot();
1076    if (p)
1077    {
1078        p->remove_point(node, this);
1079    }
1080    */
1081    delete label;
1082}
1083
1084void NetworkCurve::remove_labels(const QList<int>& labels)
1085{
1086    cancel_all_updates();
1087    foreach (int i, labels)
1088    {
1089        remove_label(i);
1090    }
1091}
1092
1093void NetworkCurve::set_edges(const NetworkCurve::Edges& edges)
1094{
1095    cancel_all_updates();
1096    qDeleteAll(m_edges);
1097    m_edges = edges;
1098}
1099
1100NetworkCurve::Edges NetworkCurve::edges() const
1101{
1102    return m_edges;
1103}
1104
1105QList<QPair<int, int> > NetworkCurve::edge_indices()
1106{
1107    int i;
1108    EdgeItem *e;
1109    QList<QPair<int, int> > edge_indices;
1110
1111    for (i = 0; i < m_edges.size(); ++i)
1112    {
1113        e = m_edges[i];
1114        edge_indices.append(QPair<int, int>(e->u()->index(), e->v()->index()));
1115    }
1116
1117    return edge_indices;
1118}
1119
1120void NetworkCurve::set_nodes(const NetworkCurve::Nodes& nodes)
1121{
1122    cancel_all_updates();
1123    qDeleteAll(m_edges);
1124    m_edges.clear();
1125    qDeleteAll(m_nodes);
1126    m_nodes = nodes;
1127    Q_ASSERT(m_nodes.uniqueKeys() == m_nodes.keys());
1128    register_points();
1129}
1130
1131NetworkCurve::Nodes NetworkCurve::nodes() const
1132{
1133    return m_nodes;
1134}
1135
1136void NetworkCurve::remove_nodes(const QList<int>& nodes)
1137{
1138    cancel_all_updates();
1139    foreach (int i, nodes)
1140    {
1141        remove_node(i);
1142    }
1143}
1144
1145void NetworkCurve::remove_node(int index)
1146{
1147    cancel_all_updates();
1148    if (!m_nodes.contains(index))
1149    {
1150        qWarning() << "Trying to remove node" << index << "which is not in the network";
1151        return;
1152    }
1153    NodeItem* node = m_nodes.take(index);
1154    Q_ASSERT(node->index() == index);
1155
1156    if (node->label != NULL)
1157    {
1158        remove_label(index);
1159        node->label = NULL;
1160    }
1161
1162    Plot* p = plot();
1163    if (p)
1164    {
1165        p->remove_point(node, this);
1166    }
1167   
1168    foreach (EdgeItem* edge, node->connected_edges())
1169    {
1170        m_edges.removeOne(edge);
1171        delete edge;
1172    }
1173    Q_ASSERT(node->connected_edges().isEmpty());
1174    delete node;
1175}
1176
1177void NetworkCurve::add_edges(const NetworkCurve::Edges& edges)
1178{
1179    m_edges.append(edges);
1180}
1181
1182void NetworkCurve::add_nodes(const NetworkCurve::Nodes& nodes)
1183{
1184    Nodes::ConstIterator it = nodes.constBegin();
1185    Nodes::ConstIterator end = nodes.constEnd();
1186    QList<int> indices;
1187    for (it; it != end; ++it)
1188    {
1189        indices.append(it.key());
1190
1191        if (m_nodes.contains(it.key()))
1192        {
1193            remove_node(it.key());
1194        }
1195    }
1196
1197    m_nodes.unite(nodes);
1198    Q_ASSERT(m_nodes.uniqueKeys() == m_nodes.keys());
1199    register_points();
1200}
1201
1202void NetworkCurve::set_node_colors(const QMap<int, QColor>& colors)
1203{
1204    QMap<int, QColor>::ConstIterator it;
1205    for (it = colors.constBegin(); it != colors.constEnd(); ++it)
1206    {
1207        m_nodes[it.key()]->set_color(it.value());
1208    }
1209}
1210
1211void NetworkCurve::set_node_sizes(const QMap<int, double>& sizes, double min_size, double max_size)
1212{
1213    cancel_all_updates();
1214
1215    NodeItem* node;
1216    Nodes::ConstIterator nit;
1217
1218    double min_size_value = std::numeric_limits<double>::max();
1219    double max_size_value = std::numeric_limits<double>::min();
1220
1221    QMap<int, double>::ConstIterator it;
1222    for (it = sizes.begin(); it != sizes.end(); ++it)
1223    {
1224        m_nodes[it.key()]->m_size_value = it.value();
1225
1226        if (it.value() < min_size_value)
1227        {
1228            min_size_value = it.value();
1229        }
1230
1231        if (it.value() > max_size_value)
1232        {
1233            max_size_value = it.value();
1234        }
1235    }
1236
1237    // find min and max size value in nodes dict
1238    bool min_changed = true;
1239    bool max_changed = true;
1240    for (nit = m_nodes.constBegin(); nit != m_nodes.constEnd(); ++nit)
1241    {
1242        node = nit.value();
1243
1244        if (node->m_size_value < min_size_value)
1245        {
1246            min_size_value = node->m_size_value;
1247            min_changed = false;
1248        }
1249
1250        if (node->m_size_value > max_size_value)
1251        {
1252            max_size_value = node->m_size_value;
1253            max_changed = false;
1254        }
1255    }
1256
1257    double size_span = max_size_value - min_size_value;
1258
1259
1260    if (min_size > 0 || max_size > 0 || min_changed || max_changed)
1261    {
1262        if (min_size > 0)
1263        {
1264            m_min_node_size = min_size;
1265        }
1266
1267        if (max_size > 0)
1268        {
1269            m_max_node_size = max_size;
1270        }
1271
1272        double node_size_span = m_max_node_size - m_min_node_size;
1273        // recalibrate all
1274        if (size_span > 0)
1275        {
1276            for (nit = m_nodes.constBegin(); nit != m_nodes.constEnd(); ++nit)
1277            {
1278                node = nit.value();
1279                node->set_size((node->m_size_value - min_size_value) / size_span * node_size_span + m_min_node_size);
1280            }
1281        }
1282        else
1283        {
1284            for (nit = m_nodes.constBegin(); nit != m_nodes.constEnd(); ++nit)
1285            {
1286                node = nit.value();
1287                node->set_size(m_min_node_size);
1288            }
1289        }
1290    }
1291    else if (sizes.size() > 0)
1292    {
1293        double node_size_span = m_max_node_size - m_min_node_size;
1294        // recalibrate given
1295        if (size_span > 0)
1296        {
1297            for (it = sizes.begin(); it != sizes.end(); ++it)
1298            {
1299                node = m_nodes[it.key()];
1300                node->set_size((node->m_size_value - min_size_value) / size_span * node_size_span + m_min_node_size);
1301            }
1302        }
1303        else
1304        {
1305            for (it = sizes.begin(); it != sizes.end(); ++it)
1306            {
1307                node = m_nodes[it.key()];
1308                node->set_size(m_min_node_size);
1309            }
1310        }
1311    }
1312}
1313
1314void NetworkCurve::set_node_labels(const QMap<int, QString>& labels)
1315{
1316    cancel_all_updates();
1317    foreach (int i, m_labels.keys())
1318    {
1319        m_nodes[i]->label = NULL;
1320    }
1321    qDeleteAll(m_labels);
1322    m_labels.clear();
1323    QMap<int, QString>::ConstIterator it;
1324    for (it = labels.constBegin(); it != labels.constEnd(); ++it)
1325    {
1326        LabelItem* item = new LabelItem(it.value(), this);
1327        item->setZValue(0.6);
1328        item->setFont(plot()->font());
1329        item->setFlag(ItemIgnoresTransformations);
1330        //item->setPos(m_nodes[it.key()]->pos() - QPointF(item->boundingRect().width() / 2, 0));
1331        item->setPos(m_nodes[it.key()]->pos());
1332
1333        /*
1334        QFontMetrics fm(item->font());
1335        QTransform t;
1336        t.translate(-(fm.width(it.value()) / 2), -5);
1337        item->setTransform(t);
1338        item->setTextWidth(item->boundingRect().width());
1339        QTextBlockFormat format;
1340        format.setAlignment(Qt::AlignHCenter);
1341        QTextCursor cursor = item->textCursor();
1342        cursor.select(QTextCursor::Document);
1343        cursor.mergeBlockFormat(format);
1344        cursor.clearSelection();
1345        item->setTextCursor(cursor);
1346        */
1347        if (labels_on_marked() && !(m_nodes[it.key()]->is_marked() || m_nodes[it.key()]->is_selected()))
1348        {
1349            item->hide();
1350        }
1351        m_nodes[it.key()]->label = item;
1352        m_labels.insert(it.key(), item);
1353    }
1354}
1355
1356void NetworkCurve::set_node_tooltips(const QMap<int, QString>& tooltips)
1357{
1358    cancel_all_updates();
1359    QMap<int, QString>::ConstIterator it;
1360    for (it = tooltips.constBegin(); it != tooltips.constEnd(); ++it)
1361    {
1362        m_nodes[it.key()]->setToolTip(it.value());
1363    }
1364}
1365
1366void NetworkCurve::set_node_marks(const QMap<int, bool>& marks)
1367{
1368    cancel_all_updates();
1369    QMap<int, bool>::ConstIterator it;
1370    for (it = marks.constBegin(); it != marks.constEnd(); ++it)
1371    {
1372        m_nodes[it.key()]->set_marked(it.value());
1373    }
1374
1375    plot()->emit_marked_points_changed();
1376}
1377
1378void NetworkCurve::clear_node_marks()
1379{
1380    cancel_all_updates();
1381    Nodes::Iterator it;
1382    for (it = m_nodes.begin(); it != m_nodes.end(); ++it)
1383    {
1384        it.value()->set_marked(false);
1385    }
1386
1387    plot()->emit_marked_points_changed();
1388}
1389
1390void NetworkCurve::set_node_coordinates(const QMap<int, QPair<double, double> >& coordinates)
1391{
1392    NodeItem *node;
1393    QMap<int, QPair<double, double> >::ConstIterator it = coordinates.constBegin();
1394    for (; it != coordinates.constEnd(); ++it)
1395    {
1396        node = m_nodes[it.key()];
1397        node->set_x(it.value().first);
1398        node->set_y(it.value().second);
1399    }
1400    //register_points();
1401}
1402
1403void NetworkCurve::set_edge_colors(const QList<QColor>& colors)
1404{
1405    cancel_all_updates();
1406    int i;
1407    for (i = 0; i < colors.size(); ++i)
1408    {
1409        QPen p = m_edges[i]->pen();
1410        p.setColor(colors[i]);
1411        m_edges[i]->setPen(p);
1412    }
1413}
1414
1415void NetworkCurve::set_edge_sizes(double max_size)
1416{
1417    cancel_all_updates();
1418
1419    double min_size_value = std::numeric_limits<double>::max();
1420    double max_size_value = std::numeric_limits<double>::min();
1421
1422    int i;
1423    for (i = 0; i < m_edges.size(); ++i)
1424    {
1425        double w = m_edges[i]->weight();
1426        if (w < min_size_value)
1427        {
1428            min_size_value = w;
1429        }
1430        if (w > max_size_value)
1431        {
1432            max_size_value = w;
1433        }
1434    }
1435
1436    double size_span = max_size_value - min_size_value;
1437    double edge_size_span = (max_size > 0) ? max_size - 1 : 0;
1438
1439    if (size_span > 0 && edge_size_span > 0)
1440    {
1441        for (i = 0; i < m_edges.size(); ++i)
1442        {
1443            double w = m_edges[i]->weight();
1444            QPen p = m_edges[i]->pen();
1445            p.setWidthF((w - min_size_value) / size_span * edge_size_span + 1);
1446            m_edges[i]->setPen(p);
1447        }
1448    }
1449    else
1450    {
1451        for (i = 0; i < m_edges.size(); ++i)
1452        {
1453            QPen p = m_edges[i]->pen();
1454            p.setWidthF(1);
1455            m_edges[i]->setPen(p);
1456        }
1457    }
1458}
1459
1460void NetworkCurve::set_edge_labels(const QList<QString>& labels)
1461{
1462    cancel_all_updates();
1463    int i;
1464    for (i = 0; i < labels.size(); ++i)
1465    {
1466        m_edges[i]->set_label(labels[i]);
1467    }
1468}
1469
1470void NetworkCurve::set_min_node_size(double size)
1471{
1472    set_node_sizes(QMap<int, double>(), size, 0);
1473}
1474
1475double NetworkCurve::min_node_size() const
1476{
1477    return m_min_node_size;
1478}
1479
1480void NetworkCurve::set_max_node_size(double size)
1481{
1482    set_node_sizes(QMap<int, double>(), 0, size);
1483}
1484
1485double NetworkCurve::max_node_size() const
1486{
1487    return m_max_node_size;
1488}
1489
1490void NetworkCurve::register_points()
1491{
1492    QList<Point*> list;
1493    foreach (NodeItem* node, m_nodes)
1494    {
1495        list << node;
1496    }
1497    set_points(list);
1498    Curve::register_points();
1499}
1500
1501void NetworkCurve::set_use_animations(bool use_animations)
1502{
1503    m_use_animations = use_animations;
1504}
1505
1506bool NetworkCurve::use_animations() const
1507{
1508    return m_use_animations;
1509}
1510
1511void NetworkCurve::set_show_component_distances(bool show_component_distances)
1512{
1513    m_show_component_distances = show_component_distances;
1514}
1515 
1516void NetworkCurve::stop_optimization()
1517{
1518    m_stop_optimization = true;
1519}
Note: See TracBrowser for help on using the repository browser.