source: orange/source/orangeqt/networkcurve.cpp @ 10934:f41e7d5a8565

Revision 10934:f41e7d5a8565, 33.1 KB checked in by mstajdohar, 22 months ago (diff)

Fine tune FR.

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    Curve::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    cancel_all_updates();
570    update_point_positions();
571}
572
573QRectF NetworkCurve::data_rect() const
574{
575    QRectF r;
576    bool first = true;
577    foreach (const NodeItem* node, m_nodes)
578    {
579        if (first)
580        {
581            r = QRectF(node->x(), node->y(), 0, 0);
582            first = false;
583        }
584        else
585        {
586            r.setTop( qMin(r.top(), node->y()) );
587            r.setBottom( qMax(r.bottom(), node->y()) );
588            r.setLeft( qMin(r.left(), node->x()) );
589            r.setRight( qMax(r.right(), node->x()) );
590        }
591    }
592    return r;
593}
594
595void NetworkCurve::scale_axes()
596{
597    cancel_all_updates();
598    QRectF r2 = QRectF(plot()->contentsRect());
599    r2.adjust(1, 1, -1, -1);
600    double t = r2.top();
601    r2.setTop(r2.bottom());
602    r2.setBottom(t);
603
604    QPair< int, int > axes_;
605    axes_ = axes();
606    QRectF r1 = plot()->data_rect_for_axes(axes_.first, axes_.second);
607
608    double dx = r1.width() / 20.0;
609    double dy = r1.height() / 20.0;
610    r1.adjust(-dx, -dy, dx, dy);
611
612    QTransform tr1 = QTransform().translate(-r1.left(), -r1.top());
613    QTransform ts = QTransform().scale(r2.width()/r1.width(), r2.height()/r1.height());
614    QTransform tr2 = QTransform().translate(r2.left(), r2.top());
615    set_graph_transform(tr1 * ts * tr2);
616}
617
618int NetworkCurve::random()
619{
620    Nodes::ConstIterator uit = m_nodes.constBegin();
621    Nodes::ConstIterator uend = m_nodes.constEnd();
622
623    for (; uit != uend; ++uit)
624    {
625        uit.value()->set_coordinates(((qreal)(qrand() % 1000)), ((qreal)(qrand() % 1000)));
626    }
627
628    return 0;
629}
630
631#define PI 3.14159265
632
633int NetworkCurve::circular(CircularLayoutType type)
634{
635    // type
636    // 0 - original
637    // 1 - random
638    // 2 - crossing reduction
639
640    if (type == NetworkCurve::circular_crossing)
641    {
642        qDebug() << "crossing_reduction";
643        return circular_crossing_reduction();
644    }
645
646    if (type == NetworkCurve::circular_original)
647        qDebug() << "original";
648
649    if (type == NetworkCurve::circular_random)
650            qDebug() << "random";
651
652    QRectF rect = data_rect();
653    int xCenter = rect.width() / 2;
654    int yCenter = rect.height() / 2;
655    int r = (rect.width() < rect.height()) ? rect.width() * 0.38 : rect.height() * 0.38;
656
657    int i;
658    double fi = PI;
659    double step = 2 * PI / m_nodes.size();
660
661    qsrand(QTime(0,0,0).secsTo(QTime::currentTime()));
662    std::vector<int> vertices;
663    Nodes::ConstIterator it;
664    for (it = m_nodes.constBegin(); it != m_nodes.constEnd(); ++it)
665    {
666        vertices.push_back(it.key());
667    }
668
669    for (i = 0; i < m_nodes.size(); ++i)
670    {
671        if (type == NetworkCurve::circular_original)
672        {
673            m_nodes[vertices[i]]->set_coordinates(r * cos(fi) + xCenter, r * sin(fi) + yCenter);
674        }
675        else if (type == NetworkCurve::circular_random)
676        {
677            int ndx = rand() % vertices.size();
678                        m_nodes[vertices[ndx]]->set_coordinates(r * cos(fi) + xCenter, r * sin(fi) + yCenter);
679            vertices.erase(vertices.begin() + ndx);
680        }
681        fi = fi - step;
682    }
683
684    return 0;
685}
686
687
688int NetworkCurve::circular_crossing_reduction()
689{
690    QMap<int, QueueVertex*> qvertices;
691    std::vector<QueueVertex*> vertices;
692    std::vector<QueueVertex*> original;
693
694    Nodes::ConstIterator it;
695    for (it = m_nodes.constBegin(); it != m_nodes.constEnd(); ++it)
696    {
697        QueueVertex *vertex = new QueueVertex();
698        vertex->ndx = it.key();
699        qvertices[it.key()] = vertex;
700
701        std::vector<int> neighbours;
702        vertex->unplacedNeighbours = neighbours.size();
703        vertex->neighbours = neighbours;
704        vertices.push_back(vertex);
705    }
706    int i;
707    EdgeItem *edge;
708    for (i = 0; i < m_edges.size(); i++)
709    {
710        edge = m_edges[i];
711        int u = edge->u()->index();
712        int v = edge->v()->index();
713        qvertices[u]->neighbours.push_back(v);
714        qvertices[u]->unplacedNeighbours += 1;
715        qvertices[v]->neighbours.push_back(u);
716        qvertices[v]->unplacedNeighbours += 1;
717    }
718    original.assign(vertices.begin(), vertices.end());
719
720    std::deque<int> positions;
721    while (vertices.size() > 0)
722    {
723        std::sort(vertices.begin(), vertices.end(), QueueVertex());
724        QueueVertex *vertex = vertices.back();
725
726
727        // update neighbours
728        for (i = 0; i < vertex->neighbours.size(); ++i)
729        {
730            int ndx = vertex->neighbours[i];
731
732            original[ndx]->placedNeighbours++;
733            original[ndx]->unplacedNeighbours--;
734        }
735
736        // count left & right crossings
737        if (vertex->placedNeighbours > 0)
738        {
739            int left = 0;
740            std::vector<int> lCrossings;
741            std::vector<int> rCrossings;
742            for (i = 0; i < positions.size(); ++i)
743            {
744                int ndx = positions[i];
745
746                if (vertex->hasNeighbour(ndx))
747                {
748                    lCrossings.push_back(left);
749                    left += original[ndx]->unplacedNeighbours;
750                    rCrossings.push_back(left);
751                }
752                else
753                    left += original[ndx]->unplacedNeighbours;
754            }
755
756            int leftCrossings = 0;
757            int rightCrossings = 0;
758
759            for (i = 0; i < lCrossings.size(); --i)
760                leftCrossings += lCrossings[i];
761
762            rCrossings.push_back(left);
763            for (i = rCrossings.size() - 1; i > 0 ; --i)
764                rightCrossings += rCrossings[i] - rCrossings[i - 1];
765            //cout << "left: " << leftCrossings << " right: " <<rightCrossings << endl;
766            if (leftCrossings < rightCrossings)
767                positions.push_front(vertex->ndx);
768            else
769                positions.push_back(vertex->ndx);
770
771        }
772        else
773            positions.push_back(vertex->ndx);
774
775        vertices.pop_back();
776    }
777
778    // Circular sifting
779    for (i = 0; i < positions.size(); ++i)
780        original[positions[i]]->position = i;
781
782    int step;
783    for (step = 0; step < 5; ++step)
784    {
785        for (i = 0; i < m_nodes.size(); ++i)
786        {
787            bool stop = false;
788            int switchNdx = -1;
789            QueueVertex *u = original[positions[i]];
790            int vNdx = (i + 1) % m_nodes.size();
791
792            while (!stop)
793            {
794                QueueVertex *v = original[positions[vNdx]];
795
796                int midCrossings = u->neighbours.size() * v->neighbours.size() / 2;
797                int crossings = 0;
798                int j,k;
799                for (j = 0; j < u->neighbours.size(); ++j)
800                    for (k = 0; k < v->neighbours.size(); ++k)
801                        if ((original[u->neighbours[j]]->position == v->position) || (original[v->neighbours[k]]->position == u->position))
802                            midCrossings = (u->neighbours.size() - 1) * (v->neighbours.size() - 1) / 2;
803                        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())
804                            ++crossings;
805
806                //cout << "v: " <<  v->ndx << " crossings: " << crossings << " u.n.size: " << u->neighbours.size() << " v.n.size: " << v->neighbours.size() << " mid: " << midCrossings << endl;
807                if (crossings > midCrossings)
808                    switchNdx = vNdx;
809                else
810                    stop = true;
811
812                vNdx = (vNdx + 1) % m_nodes.size();
813            }
814            int j;
815            if (switchNdx > -1)
816            {
817                //cout << "u: " << u->ndx << " switch: " << original[switchNdx]->ndx << endl << endl;
818                positions.erase(positions.begin() + i);
819                positions.insert(positions.begin() + switchNdx, u->ndx);
820
821                for (j = i; j <= switchNdx; ++j)
822                    original[positions[j]]->position = j;
823            }
824            //else
825            //  cout << "u: " << u->ndx << " switch: " << switchNdx << endl;
826        }
827    }
828
829    QRectF rect = data_rect();
830    int xCenter = rect.width() / 2;
831    int yCenter = rect.height() / 2;
832    int r = (rect.width() < rect.height()) ? rect.width() * 0.38 : rect.height() * 0.38;
833    double fi = PI;
834    double fiStep = 2 * PI / m_nodes.size();
835
836    for (i = 0; i < m_nodes.size(); ++i)
837    {
838        m_nodes[positions[i]]->set_x(r * cos(fi) + xCenter);
839        m_nodes[positions[i]]->set_y(r * sin(fi) + yCenter);
840        fi = fi - fiStep;
841    }
842
843        qDeleteAll(original);
844
845    original.clear();
846    vertices.clear();
847    qvertices.clear();
848
849    return 0;
850}
851
852int NetworkCurve::fr(int steps, bool weighted, bool smooth_cooling)
853{
854    int i, j;
855    int count = 0;
856    NodeItem *u, *v;
857    EdgeItem *edge;
858    m_stop_optimization = false;
859
860    double rect[4] = {std::numeric_limits<double>::max(),
861              std::numeric_limits<double>::max(),
862              std::numeric_limits<double>::min(),
863              std::numeric_limits<double>::min()};
864
865    QMap<int, DataPoint> disp;
866    foreach (const NodeItem*   node, m_nodes)
867    {
868        //disp[node->index()] = node->coordinates();
869
870        double x = node->x();
871        double y = node->y();
872        if (rect[0] > x) rect[0] = x;
873        if (rect[1] > y) rect[1] = y;
874        if (rect[2] < x) rect[2] = x;
875        if (rect[3] < y) rect[3] = y;
876    }
877    QRectF data_r(rect[0], rect[1], rect[2]-rect[0], rect[3]-rect[1]);
878    double area = data_r.width() * data_r.height();
879    double k2 = area / m_nodes.size();
880    double k = sqrt(k2);
881    double kk = 2 * k;
882    double kk2 = kk * kk;
883    double jitter = sqrt(area) / 2000;
884    double temperature, cooling, cooling_switch, cooling_1, cooling_2;
885    temperature = sqrt(area) / 5;
886    cooling = exp(log(k / 10 / temperature) / steps);
887
888    if (steps > 20)
889    {
890        cooling_switch = sqrt(area) / 100;
891        cooling_1 = (temperature - cooling_switch) / 20;
892        cooling_2 = (cooling_switch - sqrt(area) / 2000 ) / (steps - 20);
893    }
894    else
895    {
896        cooling_switch = sqrt(area) / 1000;
897        cooling_1 = (temperature - cooling_switch) / steps;
898        cooling_2 = 0;
899    }
900
901    QTime refresh_time = QTime::currentTime();
902
903    if (smooth_cooling)
904    {
905        if (steps < 20)
906        {
907            steps = 20;
908        }
909        temperature = cooling_switch;
910        cooling_1 = 0;
911        cooling_2 = (cooling_switch - sqrt(area) / 2000 ) / steps;
912    }
913    else
914    {
915        // start refreshing after 1/2s
916        refresh_time.addMSecs(500);
917    }
918
919    Plot *p = plot();
920    bool animation_enabled = p->animate_points;
921    p->animate_points = false;
922
923    // iterations
924    for (i = 0; i < steps; ++i)
925    {
926        foreach (const NodeItem* node, m_nodes)
927        {
928            disp[node->index()].x = 0;
929            disp[node->index()].y = 0;
930        }
931
932        // calculate repulsive force
933        Nodes::ConstIterator uit = m_nodes.constBegin();
934        Nodes::ConstIterator uend = m_nodes.constEnd();
935        for (; uit != uend; ++uit)
936        {
937            u = uit.value();
938            Nodes::ConstIterator vit(uit);
939            ++vit;
940            for (; vit != uend; ++vit)
941            {
942                v = vit.value();
943
944                double difx = u->x() - v->x();
945                double dify = u->y() - v->y();
946
947                double dif2 = difx * difx + dify * dify;
948
949                // if nodes are close, apply repulsive force
950                if (dif2 < kk2)
951                {
952                    // if nodes overlap
953                    if (dif2 == 0)
954                    {
955                        dif2 = 1 / k;
956                        u->set_x(u->x() + jitter);
957                        u->set_y(u->y() + jitter);
958                        v->set_x(v->x() - jitter);
959                        v->set_y(v->y() - jitter);
960                    }
961
962                    double dX = difx * k2 / dif2;
963                    double dY = dify * k2 / dif2;
964
965                    disp[u->index()].x += dX;
966                    disp[u->index()].y += dY;
967
968                    disp[v->index()].x -= dX;
969                    disp[v->index()].y -= dY;
970                }
971            }
972        }
973        // calculate attractive forces
974        for (j = 0; j < m_edges.size(); ++j)
975        {
976            edge = m_edges[j];
977            double difx = edge->u()->x() - edge->v()->x();
978            double dify = edge->u()->y() - edge->v()->y();
979
980            double dif = sqrt(difx * difx + dify * dify);
981
982            double dX = difx * dif / k;
983            double dY = dify * dif / k;
984
985            if (weighted) {
986                dX *= edge->weight();
987                dY *= edge->weight();
988            }
989
990            disp[edge->u()->index()].x -= dX;
991            disp[edge->u()->index()].y -= dY;
992
993            disp[edge->v()->index()].x += dX;
994            disp[edge->v()->index()].y += dY;
995        }
996
997        // limit the maximum displacement to the temperature t
998        // and then prevent from being displaced outside frame
999        Nodes::Iterator nit = m_nodes.begin();
1000        for (; nit != m_nodes.end(); ++nit)
1001        {
1002            u = nit.value();
1003            double dif = sqrt(disp[u->index()].x * disp[u->index()].x + disp[u->index()].y * disp[u->index()].y);
1004
1005            if (dif == 0)
1006                dif = 1;
1007
1008            u->set_coordinates(u->x() + (disp[u->index()].x * qMin(fabs(disp[u->index()].x), temperature) / dif),
1009                               u->y() + (disp[u->index()].y * qMin(fabs(disp[u->index()].y), temperature) / dif));
1010        }
1011
1012        QTime before_refresh_time = QTime::currentTime();
1013        if (before_refresh_time > refresh_time && i % 2 == 0)
1014        {
1015            scale_axes();
1016            update_properties();
1017
1018            QCoreApplication::processEvents();
1019            int refresh_duration = before_refresh_time.msecsTo(QTime::currentTime());
1020            refresh_time = before_refresh_time.addMSecs(qMax(refresh_duration * 3, 40));
1021        }
1022        if (m_stop_optimization)
1023        {
1024            break;
1025        }
1026        if (floor(temperature) > cooling_switch)
1027        {
1028            temperature -= cooling_1;
1029        }
1030        else
1031        {
1032            temperature -= cooling_2;
1033        }
1034    }
1035
1036    p->animate_points = animation_enabled;
1037    return 0;
1038}
1039
1040NetworkCurve::Labels NetworkCurve::labels() const
1041{
1042    return m_labels;
1043}
1044
1045void NetworkCurve::set_labels(const NetworkCurve::Labels& labels)
1046{
1047    cancel_all_updates();
1048    qDeleteAll(m_labels);
1049    m_labels = labels;
1050    //Q_ASSERT(m_labels.uniqueKeys() == m_labels.keys());
1051}
1052
1053void NetworkCurve::add_labels(const NetworkCurve::Labels& labels)
1054{
1055    cancel_all_updates();
1056    Labels::ConstIterator it = labels.constBegin();
1057    Labels::ConstIterator end = labels.constEnd();
1058    QList<int> indices;
1059    for (it; it != end; ++it)
1060    {
1061        indices.append(it.key());
1062
1063        if (m_labels.contains(it.key()))
1064        {
1065            remove_label(it.key());
1066        }
1067    }
1068
1069    m_labels.unite(labels);
1070    Q_ASSERT(m_labels.uniqueKeys() == m_labels.keys());
1071}
1072
1073void NetworkCurve::remove_label(int index)
1074{
1075    cancel_all_updates();
1076    if (!m_labels.contains(index))
1077    {
1078        qWarning() << "Trying to remove label for node " << index << " which is not in the network";
1079        return;
1080    }
1081    QGraphicsItem* label = m_labels.take(index);
1082    //Q_ASSERT(node->index() == index);
1083    /*
1084    Plot* p = plot();
1085    if (p)
1086    {
1087        p->remove_point(node, this);
1088    }
1089    */
1090    delete label;
1091}
1092
1093void NetworkCurve::remove_labels(const QList<int>& labels)
1094{
1095    cancel_all_updates();
1096    foreach (int i, labels)
1097    {
1098        remove_label(i);
1099    }
1100}
1101
1102void NetworkCurve::set_edges(const NetworkCurve::Edges& edges)
1103{
1104    cancel_all_updates();
1105    qDeleteAll(m_edges);
1106    m_edges = edges;
1107}
1108
1109NetworkCurve::Edges NetworkCurve::edges() const
1110{
1111    return m_edges;
1112}
1113
1114QList<QPair<int, int> > NetworkCurve::edge_indices()
1115{
1116    int i;
1117    EdgeItem *e;
1118    QList<QPair<int, int> > edge_indices;
1119
1120    for (i = 0; i < m_edges.size(); ++i)
1121    {
1122        e = m_edges[i];
1123        edge_indices.append(QPair<int, int>(e->u()->index(), e->v()->index()));
1124    }
1125
1126    return edge_indices;
1127}
1128
1129void NetworkCurve::set_nodes(const NetworkCurve::Nodes& nodes)
1130{
1131    cancel_all_updates();
1132    qDeleteAll(m_edges);
1133    m_edges.clear();
1134    qDeleteAll(m_nodes);
1135    m_nodes = nodes;
1136    Q_ASSERT(m_nodes.uniqueKeys() == m_nodes.keys());
1137    register_points();
1138}
1139
1140NetworkCurve::Nodes NetworkCurve::nodes() const
1141{
1142    return m_nodes;
1143}
1144
1145void NetworkCurve::remove_nodes(const QList<int>& nodes)
1146{
1147    cancel_all_updates();
1148    foreach (int i, nodes)
1149    {
1150        remove_node(i);
1151    }
1152}
1153
1154void NetworkCurve::remove_node(int index)
1155{
1156    cancel_all_updates();
1157    if (!m_nodes.contains(index))
1158    {
1159        qWarning() << "Trying to remove node" << index << "which is not in the network";
1160        return;
1161    }
1162    NodeItem* node = m_nodes.take(index);
1163    Q_ASSERT(node->index() == index);
1164
1165    if (node->label != NULL)
1166    {
1167        remove_label(index);
1168        node->label = NULL;
1169    }
1170
1171    Plot* p = plot();
1172    if (p)
1173    {
1174        p->remove_point(node, this);
1175    }
1176   
1177    foreach (EdgeItem* edge, node->connected_edges())
1178    {
1179        m_edges.removeOne(edge);
1180        delete edge;
1181    }
1182    Q_ASSERT(node->connected_edges().isEmpty());
1183    delete node;
1184}
1185
1186void NetworkCurve::add_edges(const NetworkCurve::Edges& edges)
1187{
1188    cancel_all_updates();
1189    m_edges.append(edges);
1190}
1191
1192void NetworkCurve::add_nodes(const NetworkCurve::Nodes& nodes)
1193{
1194    cancel_all_updates();
1195
1196    Nodes::ConstIterator it = nodes.constBegin();
1197    Nodes::ConstIterator end = nodes.constEnd();
1198    QList<int> indices;
1199    for (it; it != end; ++it)
1200    {
1201        indices.append(it.key());
1202
1203        if (m_nodes.contains(it.key()))
1204        {
1205            remove_node(it.key());
1206        }
1207    }
1208
1209    m_nodes.unite(nodes);
1210    Q_ASSERT(m_nodes.uniqueKeys() == m_nodes.keys());
1211    register_points();
1212}
1213
1214void NetworkCurve::set_node_colors(const QMap<int, QColor>& colors)
1215{
1216    cancel_all_updates();
1217
1218    QMap<int, QColor>::ConstIterator it;
1219    for (it = colors.constBegin(); it != colors.constEnd(); ++it)
1220    {
1221        m_nodes[it.key()]->set_color(it.value());
1222    }
1223}
1224
1225void NetworkCurve::set_node_sizes(const QMap<int, double>& sizes, double min_size, double max_size)
1226{
1227    cancel_all_updates();
1228
1229    NodeItem* node;
1230    Nodes::ConstIterator nit;
1231
1232    double min_size_value = std::numeric_limits<double>::max();
1233    double max_size_value = std::numeric_limits<double>::min();
1234
1235    QMap<int, double>::ConstIterator it;
1236    for (it = sizes.begin(); it != sizes.end(); ++it)
1237    {
1238        m_nodes[it.key()]->m_size_value = it.value();
1239
1240        if (it.value() < min_size_value)
1241        {
1242            min_size_value = it.value();
1243        }
1244
1245        if (it.value() > max_size_value)
1246        {
1247            max_size_value = it.value();
1248        }
1249    }
1250
1251    // find min and max size value in nodes dict
1252    bool min_changed = true;
1253    bool max_changed = true;
1254    for (nit = m_nodes.constBegin(); nit != m_nodes.constEnd(); ++nit)
1255    {
1256        node = nit.value();
1257
1258        if (node->m_size_value < min_size_value)
1259        {
1260            min_size_value = node->m_size_value;
1261            min_changed = false;
1262        }
1263
1264        if (node->m_size_value > max_size_value)
1265        {
1266            max_size_value = node->m_size_value;
1267            max_changed = false;
1268        }
1269    }
1270
1271    double size_span = max_size_value - min_size_value;
1272
1273
1274    if (min_size > 0 || max_size > 0 || min_changed || max_changed)
1275    {
1276        if (min_size > 0)
1277        {
1278            m_min_node_size = min_size;
1279        }
1280
1281        if (max_size > 0)
1282        {
1283            m_max_node_size = max_size;
1284        }
1285
1286        double node_size_span = m_max_node_size - m_min_node_size;
1287        // recalibrate all
1288        if (size_span > 0)
1289        {
1290            for (nit = m_nodes.constBegin(); nit != m_nodes.constEnd(); ++nit)
1291            {
1292                node = nit.value();
1293                node->set_size((node->m_size_value - min_size_value) / size_span * node_size_span + m_min_node_size);
1294            }
1295        }
1296        else
1297        {
1298            for (nit = m_nodes.constBegin(); nit != m_nodes.constEnd(); ++nit)
1299            {
1300                node = nit.value();
1301                node->set_size(m_min_node_size);
1302            }
1303        }
1304    }
1305    else if (sizes.size() > 0)
1306    {
1307        double node_size_span = m_max_node_size - m_min_node_size;
1308        // recalibrate given
1309        if (size_span > 0)
1310        {
1311            for (it = sizes.begin(); it != sizes.end(); ++it)
1312            {
1313                node = m_nodes[it.key()];
1314                node->set_size((node->m_size_value - min_size_value) / size_span * node_size_span + m_min_node_size);
1315            }
1316        }
1317        else
1318        {
1319            for (it = sizes.begin(); it != sizes.end(); ++it)
1320            {
1321                node = m_nodes[it.key()];
1322                node->set_size(m_min_node_size);
1323            }
1324        }
1325    }
1326}
1327
1328void NetworkCurve::set_node_labels(const QMap<int, QString>& labels)
1329{
1330    cancel_all_updates();
1331    foreach (int i, m_labels.keys())
1332    {
1333        m_nodes[i]->label = NULL;
1334    }
1335    qDeleteAll(m_labels);
1336    m_labels.clear();
1337    QMap<int, QString>::ConstIterator it;
1338    for (it = labels.constBegin(); it != labels.constEnd(); ++it)
1339    {
1340        LabelItem* item = new LabelItem(it.value(), this);
1341        item->setZValue(0.6);
1342        item->setFont(plot()->font());
1343        item->setFlag(ItemIgnoresTransformations);
1344        item->setPos(m_nodes[it.key()]->pos());
1345
1346        if (labels_on_marked() && !(m_nodes[it.key()]->is_marked() || m_nodes[it.key()]->is_selected()))
1347        {
1348            item->hide();
1349        }
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    cancel_all_updates();
1393    NodeItem *node;
1394    QMap<int, QPair<double, double> >::ConstIterator it = coordinates.constBegin();
1395    for (; it != coordinates.constEnd(); ++it)
1396    {
1397        node = m_nodes[it.key()];
1398        node->set_x(it.value().first);
1399        node->set_y(it.value().second);
1400    }
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.