1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-05 18:16:23 +03:00

0029750: Samples - function arrows are not updated by moving a node in FuncDemo qt sample

The visual links between functions are recovered.
The demo is synchronized with a corresponding demo from Qt (qt486-vc10-32\examples\graphicsview\elasticnodes). It may be successfully compiled by any further versions of Qt including 5.10.1
Also, because Open CASCADE (and OCAF in particular) is improved for usage in multi-threading mode, usage of mutexes is added in this sample (for access to the sharing TNaming_UsedShapes attribute, for example).
This commit is contained in:
vro 2020-12-17 13:17:46 +03:00 committed by bugmaster
parent 41046145c4
commit 894133a5ad
21 changed files with 388 additions and 81 deletions

@ -1,3 +1,3 @@
@echo off
rem Define QTDIR variable
set QTDIR=
set QTDIR=%PRODUCTS_PATH%/qt486-vc10-32

@ -36,7 +36,6 @@ void BaseDriver::Arguments(TDF_LabelList& args) const
}
}
// Returns the results of the function
void BaseDriver::Results(TDF_LabelList& res) const
{
@ -50,6 +49,12 @@ void BaseDriver::Results(TDF_LabelList& res) const
}
}
// Sets a mutex for execution of the driver.
void BaseDriver::SetMutex(Standard_Mutex* pmutex)
{
myMutex = pmutex;
}
// Execution.
Standard_Integer BaseDriver::Execute(Handle(TFunction_Logbook)& ) const
{

@ -5,14 +5,11 @@
#if !defined(_BASEDRIVER_H_)
#define _BASEDRIVER_H_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#include <Standard_DefineHandle.hxx>
#include <TFunction_Driver.hxx>
#include <TFunction_Logbook.hxx>
#include <TDF_LabelList.hxx>
#include <Standard_Mutex.hxx>
DEFINE_STANDARD_HANDLE(BaseDriver, TFunction_Driver)
@ -30,10 +27,16 @@ public:
// Returns the results of the function
virtual void Results(TDF_LabelList& res) const;
// Sets a mutex for execution of the driver.
void SetMutex(Standard_Mutex* pmutex);
// Execution.
virtual Standard_Integer Execute(Handle(TFunction_Logbook)& log) const;
DEFINE_STANDARD_RTTIEXT(BaseDriver, TFunction_Driver)
protected:
Standard_Mutex* myMutex;
};
#endif // !defined(_BASEDRIVER_H_)

@ -5,10 +5,6 @@
#if !defined(_CIRCLEDRIVER_H_)
#define _CIRCLEDRIVER_H_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#include "BaseDriver.h"
#include <Standard_DefineHandle.hxx>

@ -5,10 +5,6 @@
#if !defined(_CONEDRIVER_H_)
#define _CONEDRIVER_H_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#include "BaseDriver.h"
#include <Standard_DefineHandle.hxx>

@ -5,10 +5,6 @@
#if !defined(_CYLINDERDRIVER_H_)
#define _CYLINDERDRIVER_H_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#include "BaseDriver.h"
#include <Standard_DefineHandle.hxx>

@ -1,5 +1,6 @@
#include "FThread.h"
#include "graphwidget.h"
#include "BaseDriver.h"
#include <TFunction_Function.hxx>
#include <TFunction_IFunction.hxx>
@ -39,6 +40,11 @@ void FThread::setThreadIndex(const int theIndex)
this->thread_index = theIndex;
}
void FThread::setMutex(Standard_Mutex* pmutex)
{
this->pmutex = pmutex;
}
// Returns any free (not executed yet) function
TDF_Label FThread::getFreeFunction()
{
@ -94,12 +100,16 @@ void FThread::run()
const bool must = D->MustExecute(log);
if (must)
{
// Usage of mutex for execution of Open CASCADE code is the most stupid thing!!!
// But it makes the execution more reliable...
// Usage of mutex for execution of Open CASCADE code.
// It makes the execution more reliable...
if (!Handle(BaseDriver)::DownCast(D).IsNull())
Handle(BaseDriver)::DownCast(D)->SetMutex(pmutex);
// Execute the driver.
const int ret = D->Execute(log);
if (ret == 0)
{
// Successfully executed!
// Successfuly executed!
itr.SetStatus(L, TFunction_ES_Succeeded);
TDF_LabelList res;

@ -8,6 +8,7 @@
#include <TFunction_Logbook.hxx>
#include <TFunction_Iterator.hxx>
#include <TFunction_Driver.hxx>
#include <Standard_Mutex.hxx>
class GraphWidget; // shows graphically execution of functions
@ -25,6 +26,7 @@ public:
void setLogbook(const Handle(TFunction_Logbook)& ); // to set logbook with modifications
void setGraph(GraphWidget* ); // to change color of a graph circle
void setThreadIndex(const int ); // to set the index of the thread
void setMutex(Standard_Mutex* );
protected:
@ -35,6 +37,7 @@ private:
TFunction_Iterator itr;
Handle(TFunction_Logbook) log;
Standard_Mutex* pmutex;
int thread_index;
GraphWidget* graph;

@ -47,9 +47,15 @@ Standard_Integer PointDriver::Execute(Handle(TFunction_Logbook)& log) const
// Make the result
TopoDS_Vertex V = BRepBuilderAPI_MakeVertex(gp_Pnt(x, y, z));
if (myMutex)
myMutex->Lock();
// Set the result
TNaming_Builder B(Label());
B.Generated(V);
if (myMutex)
myMutex->Unlock();
return BaseDriver::Execute(log);
}

@ -5,10 +5,6 @@
#if !defined(_PointDRIVER_H_)
#define _PointDRIVER_H_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#include "BaseDriver.h"
#include <Standard_DefineHandle.hxx>

@ -5,10 +5,6 @@
#if !defined(_PRISMDRIVER_H_)
#define _PRISMDRIVER_H_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#include "BaseDriver.h"
#include <Standard_DefineHandle.hxx>

@ -56,10 +56,14 @@ Standard_Integer ShapeSaverDriver::Execute(Handle(TFunction_Logbook)& log) const
}
}
//BRepTools::Write(C, "result.brep");
if (myMutex)
myMutex->Lock();
TNaming_Builder Bui(Label());
Bui.Generated(C);
if (myMutex)
myMutex->Unlock();
return BaseDriver::Execute(log);
}

@ -5,10 +5,6 @@
#if !defined(_SHAPESAVERDRIVER_H_)
#define _SHAPESAVERDRIVER_H_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#include "BaseDriver.h"
#include <Standard_DefineHandle.hxx>

@ -5,10 +5,6 @@
#if !defined(_SIMPLEDRIVER_H_)
#define _SIMPLEDRIVER_H_
#if _MSC_VER > 1000
#pragma once
#endif // _MSC_VER > 1000
#include <Standard_DefineHandle.hxx>
#include <TFunction_Driver.hxx>
#include <TFunction_Logbook.hxx>

@ -50,6 +50,7 @@
static const double Pi = 3.14159265358979323846264338327950288419717;
static double TwoPi = 2.0 * Pi;
//! [0]
Edge::Edge(Node *sourceNode, Node *destNode)
: arrowSize(10)
{
@ -60,7 +61,9 @@ Edge::Edge(Node *sourceNode, Node *destNode)
dest->addEdge(this);
adjust();
}
//! [0]
//! [1]
Node *Edge::sourceNode() const
{
return source;
@ -70,7 +73,9 @@ Node *Edge::destNode() const
{
return dest;
}
//! [1]
//! [2]
void Edge::adjust()
{
if (!source || !dest)
@ -89,7 +94,9 @@ void Edge::adjust()
sourcePoint = destPoint = line.p1();
}
}
//! [2]
//! [3]
QRectF Edge::boundingRect() const
{
if (!source || !dest)
@ -103,7 +110,9 @@ QRectF Edge::boundingRect() const
.normalized()
.adjusted(-extra, -extra, extra, extra);
}
//! [3]
//! [4]
void Edge::paint(QPainter *painter, const QStyleOptionGraphicsItem *, QWidget *)
{
if (!source || !dest)
@ -112,11 +121,15 @@ void Edge::paint(QPainter *painter, const QStyleOptionGraphicsItem *, QWidget *)
QLineF line(sourcePoint, destPoint);
if (qFuzzyCompare(line.length(), qreal(0.)))
return;
//! [4]
//! [5]
// Draw the line itself
painter->setPen(QPen(Qt::black, 1, Qt::SolidLine, Qt::RoundCap, Qt::RoundJoin));
painter->drawLine(line);
//! [5]
//! [6]
// Draw the arrows
double angle = ::acos(line.dx() / line.length());
if (line.dy() >= 0)
@ -135,3 +148,4 @@ void Edge::paint(QPainter *painter, const QStyleOptionGraphicsItem *, QWidget *)
painter->drawPolygon(QPolygonF() << line.p1() << sourceArrowP1 << sourceArrowP2);
painter->drawPolygon(QPolygonF() << line.p2() << destArrowP1 << destArrowP2);
}
//! [6]

@ -47,13 +47,13 @@
class Node;
//! [0]
class Edge : public QGraphicsItem
{
public:
Edge(Node *sourceNode, Node *destNode);
Node *sourceNode() const;
Node *destNode() const;
void adjust();
@ -72,5 +72,6 @@ private:
QPointF destPoint;
qreal arrowSize;
};
//! [0]
#endif

@ -43,10 +43,7 @@
#include "node.h"
#include <Standard_WarningsDisable.hxx>
#include <QDebug>
#include <QGraphicsScene>
#include <QWheelEvent>
#include <QApplication>
#include <QtGui>
#include <Standard_WarningsRestore.hxx>
#include <math.h>
@ -73,24 +70,73 @@
#include "ShapeSaverDriver.h"
#include "SimpleDriver.h"
GraphWidget::GraphWidget(QWidget* parent):QGraphicsView(parent),
myThread1(0),myThread2(0),myThread3(0),myThread4(0)
//! [0]
GraphWidget::GraphWidget(QWidget *parent)
: QGraphicsView(parent), timerId(0),
myThread1(0),myThread2(0),myThread3(0),myThread4(0)
{
QGraphicsScene *scene = new QGraphicsScene(this);
scene->setItemIndexMethod(QGraphicsScene::NoIndex);
scene->setSceneRect(1, 1, parent->width(), parent->height());
scene->setSceneRect(-200, -200, 400, 400);
setScene(scene);
setCacheMode(CacheBackground);
setViewportUpdateMode(BoundingRectViewportUpdate);
setRenderHint(QPainter::Antialiasing);
setTransformationAnchor(AnchorUnderMouse);
setResizeAnchor(AnchorViewCenter);
scale(0.7, 0.81);
scale(qreal(0.8), qreal(0.8));
setMinimumSize(400, 400);
setWindowTitle(tr("Function Mechanism"));
setNbThreads(1);
//! [0]
//! [1]
/*
Node *node1 = new Node(this);
Node *node2 = new Node(this);
Node *node3 = new Node(this);
Node *node4 = new Node(this);
centerNode = new Node(this);
Node *node6 = new Node(this);
Node *node7 = new Node(this);
Node *node8 = new Node(this);
Node *node9 = new Node(this);
scene->addItem(node1);
scene->addItem(node2);
scene->addItem(node3);
scene->addItem(node4);
scene->addItem(centerNode);
scene->addItem(node6);
scene->addItem(node7);
scene->addItem(node8);
scene->addItem(node9);
scene->addItem(new Edge(node1, node2));
scene->addItem(new Edge(node2, node3));
scene->addItem(new Edge(node2, centerNode));
scene->addItem(new Edge(node3, node6));
scene->addItem(new Edge(node4, node1));
scene->addItem(new Edge(node4, centerNode));
scene->addItem(new Edge(centerNode, node6));
scene->addItem(new Edge(centerNode, node8));
scene->addItem(new Edge(node6, node9));
scene->addItem(new Edge(node7, node4));
scene->addItem(new Edge(node8, node7));
scene->addItem(new Edge(node9, node8));
node1->setPos(-50, -50);
node2->setPos(0, -50);
node3->setPos(50, -50);
node4->setPos(-50, 0);
centerNode->setPos(0, 0);
node6->setPos(50, 0);
node7->setPos(-50, 50);
node8->setPos(0, 50);
node9->setPos(50, 50);
*/
setNbThreads(4);
}
//! [1]
GraphWidget::~GraphWidget()
{
@ -138,8 +184,8 @@ bool GraphWidget::createModel(const Handle(TDocStd_Document)& doc)
return false;
// Grid of functions
int dx = width() / nbx, dy = height() / nby;
int x0 = dx / 2, y0 = dy / 2; // start position
int dx = width() / nbx / 2, dy = height() / nby;
int x0 = int(-double(width()) / 1.5) + dx, y0 = int(-double(height()) / 1.5) + dy; // start position
// Draw functions
double x = x0, y = y0;
@ -163,6 +209,7 @@ bool GraphWidget::createModel(const Handle(TDocStd_Document)& doc)
if (y > height())
y = y0;
fIterator.Next();
x = x0 + dx;
}
// Draw dependencies
@ -199,41 +246,151 @@ bool GraphWidget::createModel(const Handle(TDocStd_Document)& doc)
return !myDocument.IsNull();
}
//! [2]
void GraphWidget::itemMoved()
{
if (!timerId)
timerId = startTimer(1000 / 25);
}
//! [2]
//! [3]
void GraphWidget::keyPressEvent(QKeyEvent *event)
{
switch (event->key()) {
case Qt::Key_Up:
centerNode->moveBy(0, -20);
break;
case Qt::Key_Down:
centerNode->moveBy(0, 20);
break;
case Qt::Key_Left:
centerNode->moveBy(-20, 0);
break;
case Qt::Key_Right:
centerNode->moveBy(20, 0);
break;
case Qt::Key_Plus:
zoomIn();
break;
case Qt::Key_Minus:
zoomOut();
break;
case Qt::Key_Space:
case Qt::Key_Enter:
shuffle();
break;
default:
QGraphicsView::keyPressEvent(event);
}
}
//! [3]
//! [4]
void GraphWidget::timerEvent(QTimerEvent *event)
{
Q_UNUSED(event);
QList<Node *> nodes;
foreach (QGraphicsItem *item, scene()->items()) {
if (Node *node = qgraphicsitem_cast<Node *>(item))
nodes << node;
}
foreach (Node *node, nodes)
node->calculateForces();
bool itemsMoved = false;
foreach (Node *node, nodes) {
if (node->advance())
itemsMoved = true;
}
if (!itemsMoved) {
killTimer(timerId);
timerId = 0;
}
}
//! [4]
//! [5]
void GraphWidget::wheelEvent(QWheelEvent *event)
{
scaleView(pow((double)2, -event->delta() / 240.0));
}
//! [5]
void GraphWidget::drawBackground(QPainter *painter, const QRectF &rect)
//! [6]
void GraphWidget::drawBackground(QPainter* painter, const QRectF &rect)
{
Q_UNUSED(rect);
Q_UNUSED(painter);
// Shadow
QRectF sceneRect = this->sceneRect();
QRectF rightShadow(sceneRect.right(), sceneRect.top() + 5, 5, sceneRect.height());
QRectF bottomShadow(sceneRect.left() + 5, sceneRect.bottom(), sceneRect.width(), 5);
if (rightShadow.intersects(rect) || rightShadow.contains(rect))
painter->fillRect(rightShadow, Qt::darkGray);
if (bottomShadow.intersects(rect) || bottomShadow.contains(rect))
painter->fillRect(bottomShadow, Qt::darkGray);
// Fill
QLinearGradient gradient(sceneRect.topLeft(), sceneRect.bottomRight());
gradient.setColorAt(0, Qt::white);
gradient.setColorAt(1, Qt::lightGray);
painter->fillRect(rect.intersected(sceneRect), gradient);
painter->setBrush(Qt::NoBrush);
painter->drawRect(sceneRect);
// // Shadow
// QRectF sceneRect = this->sceneRect();
// QRectF rightShadow(sceneRect.right(), sceneRect.top() + 5, 5, sceneRect.height());
// QRectF bottomShadow(sceneRect.left() + 5, sceneRect.bottom(), sceneRect.width(), 5);
// if (rightShadow.intersects(rect) || rightShadow.contains(rect))
// painter->fillRect(rightShadow, Qt::darkGray);
// if (bottomShadow.intersects(rect) || bottomShadow.contains(rect))
// painter->fillRect(bottomShadow, Qt::darkGray);
//
// // Fill
// QLinearGradient gradient(sceneRect.topLeft(), sceneRect.bottomRight());
// gradient.setColorAt(0, Qt::white);
// gradient.setColorAt(1, Qt::lightGray);
// painter->fillRect(rect.intersect(sceneRect), gradient);
// painter->setBrush(Qt::NoBrush);
// painter->drawRect(sceneRect);
//
//#if !defined(Q_OS_SYMBIAN) && !defined(Q_WS_MAEMO_5)
// // Text
// QRectF textRect(sceneRect.left() + 4, sceneRect.top() + 4,
// sceneRect.width() - 4, sceneRect.height() - 4);
// QString message(tr("Click and drag the nodes around, and zoom with the mouse "
// "wheel or the '+' and '-' keys"));
//
// QFont font = painter->font();
// font.setBold(true);
// font.setPointSize(14);
// painter->setFont(font);
// painter->setPen(Qt::lightGray);
// painter->drawText(textRect.translated(2, 2), message);
// painter->setPen(Qt::black);
// painter->drawText(textRect, message);
//#endif
}
//! [6]
//! [7]
void GraphWidget::scaleView(qreal scaleFactor)
{
qreal factor = matrix().scale(scaleFactor, scaleFactor).mapRect(QRectF(0, 0, 1, 1)).width();
qreal factor = transform().scale(scaleFactor, scaleFactor).mapRect(QRectF(0, 0, 1, 1)).width();
if (factor < 0.07 || factor > 100)
return;
scale(scaleFactor, scaleFactor);
}
//! [7]
void GraphWidget::shuffle()
{
foreach (QGraphicsItem *item, scene()->items()) {
if (qgraphicsitem_cast<Node *>(item))
item->setPos(-150 + qrand() % 300, -150 + qrand() % 300);
}
}
void GraphWidget::zoomIn()
{
scaleView(qreal(1.2));
}
void GraphWidget::zoomOut()
{
scaleView(1 / qreal(1.2));
}
// Find node of the function
Node* GraphWidget::findNode(const TDF_Label& L)
@ -300,6 +457,14 @@ void GraphWidget::compute()
if (myNbThreads > 3)
myThread4->setThreadIndex(4);
myThread1->setMutex(&myMutex);
if (myNbThreads > 1)
myThread2->setMutex(&myMutex);
if (myNbThreads > 2)
myThread3->setMutex(&myMutex);
if (myNbThreads > 3)
myThread4->setMutex(&myMutex);
QThread::Priority priority = QThread::LowestPriority;
if (!myThread1->isRunning())
myThread1->start(priority);
@ -410,4 +575,4 @@ void GraphWidget::slowDownThread(const int thread_index)
myThread4->setPriority(priority);
break;
}
}
}

@ -52,13 +52,16 @@
class Node;
//! [0]
class GraphWidget : public QGraphicsView
{
Q_OBJECT
public:
GraphWidget(QWidget* parent);
~GraphWidget();
GraphWidget(QWidget *parent = 0);
~GraphWidget();
void itemMoved();
bool createModel(const Handle(TDocStd_Document)& doc);
Handle(TDocStd_Document) getDocument() { return myDocument; }
@ -73,12 +76,25 @@ public:
void setFinished();
bool isFinished();
public slots:
void shuffle();
void zoomIn();
void zoomOut();
protected:
void keyPressEvent(QKeyEvent *event);
void timerEvent(QTimerEvent *event);
void wheelEvent(QWheelEvent *event);
void drawBackground(QPainter *painter, const QRectF &rect);
void scaleView(qreal scaleFactor);
private:
int timerId;
Node *centerNode;
private:
Standard_Mutex myMutex;
Handle(TDocStd_Document) myDocument;
int myNbThreads;
FThread* myThread1;
@ -87,5 +103,6 @@ private:
FThread* myThread4;
int myNbFinishedThreads;
};
//! [0]
#endif

@ -101,6 +101,7 @@ MainWindow::MainWindow()
// Create a new document
createDefaultModel1();
graph->setNbThreads(4);
}
Handle(AppStd_Application) MainWindow::getApplication()
@ -224,7 +225,7 @@ void MainWindow::nbThreads()
{
bool ok;
int nb = QInputDialog::getInt(this, tr("Number of threads"), tr("(1 - 4): "),
graph->getNbThreads(), 1, 4, 1, &ok);
graph->getNbThreads(), 1, 4, 1, &ok);
if (ok)
graph->setNbThreads(nb);
}
@ -335,6 +336,7 @@ void MainWindow::createDefaultModel1()
Handle(TDocStd_Document) doc;
app->NewDocument("XmlOcaf", doc);
TDF_Label mainLabel = doc->Main();
mainLabel.ForgetAllAttributes(true);
// Initialize function drivers
TFunction_DriverTable::Get()->AddDriver(SimpleDriver::GetID(), new SimpleDriver());
@ -423,8 +425,8 @@ void MainWindow::createDefaultModel2()
Handle(AppStd_Application) app = MainWindow::getApplication();
Handle(TDocStd_Document) doc;
app->NewDocument("BinOcaf", doc);
//app->Open("W:\\TestFM\\model2.cbf", doc);
TDF_Label mainLabel = doc->Main();
mainLabel.ForgetAllAttributes(true);
// Initialize function drivers
TFunction_DriverTable::Get()->AddDriver(PointDriver::GetID(), new PointDriver());

@ -54,12 +54,16 @@
#include <TDataStd_Name.hxx>
#include <TDataStd_Tick.hxx>
//! [0]
Node::Node(GraphWidget *graphWidget)
: graph(graphWidget)
{
setFlag(ItemIsMovable);
setZValue(1);
setFlag(ItemSendsGeometryChanges);
setCacheMode(DeviceCoordinateCache);
setZValue(-1);
}
//! [0]
void Node::setFunction(const TDF_Label& func)
{
@ -71,6 +75,7 @@ const TDF_Label& Node::getFunction() const
return myFunction;
}
//! [1]
void Node::addEdge(Edge *edge)
{
edgeList << edge;
@ -81,26 +86,111 @@ QList<Edge *> Node::edges() const
{
return edgeList;
}
//! [1]
//! [2]
void Node::calculateForces()
{
if (!scene() || scene()->mouseGrabberItem() == this) {
newPos = pos();
return;
}
//! [2]
//! [3]
// Sum up all forces pushing this item away
qreal xvel = 0;
qreal yvel = 0;
foreach (QGraphicsItem *item, scene()->items()) {
Node *node = qgraphicsitem_cast<Node *>(item);
if (!node)
continue;
QPointF vec = mapToItem(node, 0, 0);
qreal dx = vec.x();
qreal dy = vec.y();
double l = 2.0 * (dx * dx + dy * dy);
if (l > 0) {
xvel += (dx * 150.0) / l;
yvel += (dy * 150.0) / l;
}
}
//! [3]
//! [4]
// Now subtract all forces pulling items together
double weight = (edgeList.size() + 1) * 10;
foreach (Edge *edge, edgeList) {
QPointF vec;
if (edge->sourceNode() == this)
vec = mapToItem(edge->destNode(), 0, 0);
else
vec = mapToItem(edge->sourceNode(), 0, 0);
xvel -= vec.x() / weight;
yvel -= vec.y() / weight;
}
//! [4]
//! [5]
if (qAbs(xvel) < 0.1 && qAbs(yvel) < 0.1)
xvel = yvel = 0;
//! [5]
//! [6]
QRectF sceneRect = scene()->sceneRect();
newPos = pos() + QPointF(xvel, yvel);
newPos.setX(qMin(qMax(newPos.x(), sceneRect.left() + 10), sceneRect.right() - 10));
newPos.setY(qMin(qMax(newPos.y(), sceneRect.top() + 10), sceneRect.bottom() - 10));
}
//! [6]
//! [7]
bool Node::advance()
{
if (newPos == pos())
return false;
setPos(newPos);
return true;
}
//! [7]
//! [8]
QRectF Node::boundingRect() const
{
#if defined(Q_OS_SYMBIAN) || defined(Q_WS_MAEMO_5)
// Add some extra space around the circle for easier touching with finger
qreal adjust = 30;
return QRectF( -10 - adjust, -10 - adjust,
20 + adjust * 2, 20 + adjust * 2);
#else
qreal adjust = 2;
return QRectF(-15 - adjust, -15 - adjust,
33 + adjust, 33 + adjust);
return QRectF( -10 - adjust, -10 - adjust,
23 + adjust, 23 + adjust);
#endif
}
//! [8]
//! [9]
QPainterPath Node::shape() const
{
QPainterPath path;
path.addEllipse(-15, -15, 30, 30);
#if defined(Q_OS_SYMBIAN) || defined(Q_WS_MAEMO_5)
// Add some extra space around the circle for easier touching with finger
path.addEllipse( -40, -40, 80, 80);
#else
path.addEllipse(-10, -10, 20, 20);
#endif
return path;
}
//! [9]
//! [10]
void Node::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *)
{
painter->setPen(Qt::NoPen);
painter->setBrush(Qt::darkGray);
painter->drawEllipse(-12, -12, 30, 30);
painter->drawEllipse(-7, -7, 20, 20);
QColor light_color(Qt::yellow);
TFunction_IFunction iFunc(myFunction);
@ -137,8 +227,9 @@ void Node::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWid
gradient.setColorAt(1, dark_color);
}
painter->setBrush(gradient);
painter->setPen(QPen(Qt::black, 0));
painter->drawEllipse(-15, -15, 30, 30);
painter->drawEllipse(-10, -10, 20, 20);
QString s;
Handle(TDataStd_Name) n;
@ -146,13 +237,16 @@ void Node::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWid
s = TCollection_AsciiString(n->Get()).ToCString();
painter->drawText(-7, 3, s);
}
//! [10]
//! [11]
QVariant Node::itemChange(GraphicsItemChange change, const QVariant &value)
{
switch (change) {
case ItemPositionChange:
case ItemPositionHasChanged:
foreach (Edge *edge, edgeList)
edge->adjust();
//graph->itemMoved();
break;
default:
break;
@ -160,7 +254,9 @@ QVariant Node::itemChange(GraphicsItemChange change, const QVariant &value)
return QGraphicsItem::itemChange(change, value);
}
//! [11]
//! [12]
void Node::mousePressEvent(QGraphicsSceneMouseEvent *event)
{
update();
@ -172,3 +268,4 @@ void Node::mouseReleaseEvent(QGraphicsSceneMouseEvent *event)
update();
QGraphicsItem::mouseReleaseEvent(event);
}
//! [12]

@ -50,8 +50,11 @@
class Edge;
class GraphWidget;
QT_BEGIN_NAMESPACE
class QGraphicsSceneMouseEvent;
QT_END_NAMESPACE
//! [0]
class Node : public QGraphicsItem
{
public:
@ -66,12 +69,16 @@ public:
enum { Type = UserType + 1 };
int type() const { return Type; }
void calculateForces();
bool advance();
QRectF boundingRect() const;
QPainterPath shape() const;
void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget);
protected:
QVariant itemChange(GraphicsItemChange change, const QVariant &value);
void mousePressEvent(QGraphicsSceneMouseEvent *event);
void mouseReleaseEvent(QGraphicsSceneMouseEvent *event);
@ -81,5 +88,6 @@ private:
GraphWidget *graph;
TDF_Label myFunction;
};
//! [0]
#endif