1
0
mirror of https://git.dev.opencascade.org/repos/occt.git synced 2025-04-21 10:13:43 +03:00
occt/samples/qt/FuncDemo/src/graphwidget.cpp
vro 894133a5ad 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).
2020-12-25 17:08:25 +03:00

579 lines
16 KiB
C++

/****************************************************************************
**
** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies).
** Contact: http://www.qt-project.org/legal
**
** This file is part of the examples of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:BSD$
** You may use this file under the terms of the BSD license as follows:
**
** "Redistribution and use in source and binary forms, with or without
** modification, are permitted provided that the following conditions are
** met:
** * Redistributions of source code must retain the above copyright
** notice, this list of conditions and the following disclaimer.
** * Redistributions in binary form must reproduce the above copyright
** notice, this list of conditions and the following disclaimer in
** the documentation and/or other materials provided with the
** distribution.
** * Neither the name of Digia Plc and its Subsidiary(-ies) nor the names
** of its contributors may be used to endorse or promote products derived
** from this software without specific prior written permission.
**
**
** THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
** "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
** LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
** A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
** OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
** SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
** LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
** OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE."
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include "graphwidget.h"
#include "edge.h"
#include "node.h"
#include <Standard_WarningsDisable.hxx>
#include <QtGui>
#include <Standard_WarningsRestore.hxx>
#include <math.h>
#include <TFunction_Iterator.hxx>
#include <TFunction_IFunction.hxx>
#include <TFunction_GraphNode.hxx>
#include <TFunction_Scope.hxx>
#include <TFunction_DriverTable.hxx>
#include <TFunction_Driver.hxx>
#include <TFunction_Function.hxx>
#include <TDataStd_Name.hxx>
#include <TDF_ChildIterator.hxx>
#include <TDF_ListIteratorOfLabelList.hxx>
#include <TColStd_MapIteratorOfMapOfInteger.hxx>
#include "SimpleDriver.h"
#include "PointDriver.h"
#include "CircleDriver.h"
#include "PrismDriver.h"
#include "ConeDriver.h"
#include "CylinderDriver.h"
#include "ShapeSaverDriver.h"
#include "SimpleDriver.h"
//! [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(-200, -200, 400, 400);
setScene(scene);
setCacheMode(CacheBackground);
setViewportUpdateMode(BoundingRectViewportUpdate);
setRenderHint(QPainter::Antialiasing);
setTransformationAnchor(AnchorUnderMouse);
scale(qreal(0.8), qreal(0.8));
setMinimumSize(400, 400);
setWindowTitle(tr("Function Mechanism"));
//! [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()
{
if (myThread1)
{
myThread1->wait();
myThread1->deleteLater();
}
if (myThread2)
{
myThread2->wait();
myThread2->deleteLater();
}
if (myThread3)
{
myThread3->wait();
myThread3->deleteLater();
}
if (myThread4)
{
myThread4->wait();
myThread4->deleteLater();
}
}
bool GraphWidget::createModel(const Handle(TDocStd_Document)& doc)
{
myDocument = doc;
TFunction_Iterator fIterator(myDocument->Main());
fIterator.SetUsageOfExecutionStatus(false);
Handle(TFunction_Scope) scope = TFunction_Scope::Set(myDocument->Main());
// Find out the size of the grid: number of functions in X and Y directions
int nbx = 0, nby = 0;
while (!fIterator.Current().IsEmpty())
{
const TDF_LabelList& funcs = fIterator.Current();
if (funcs.Extent() > nbx)
nbx = funcs.Extent();
nby++;
fIterator.Next();
}
if (!nbx || !nby)
return false;
// Grid of functions
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;
fIterator.Init(myDocument->Main());
while (!fIterator.Current().IsEmpty())
{
const TDF_LabelList& funcs = fIterator.Current();
TDF_ListIteratorOfLabelList itrl(funcs);
for (; itrl.More(); itrl.Next())
{
TDF_Label L = itrl.Value();
Node *node = new Node(this);
node->setFunction(L);
scene()->addItem(node);
node->setPos(x, y);
x += dx;
if (x > width())
x = x0;
}
y += dy;
if (y > height())
y = y0;
fIterator.Next();
x = x0 + dx;
}
// Draw dependencies
fIterator.Init(myDocument->Main());
while (!fIterator.Current().IsEmpty())
{
const TDF_LabelList& funcs = fIterator.Current();
TDF_ListIteratorOfLabelList itrl(funcs);
for (; itrl.More(); itrl.Next())
{
TDF_Label L = itrl.Value();
Node* node = findNode(L);
if (!node)
continue;
// Find backward dependencies of the function
TFunction_IFunction iFunc(L);
Handle(TFunction_GraphNode) graphNode = iFunc.GetGraphNode();
const TColStd_MapOfInteger& prev = graphNode->GetPrevious();
TColStd_MapIteratorOfMapOfInteger itrm(prev);
for (; itrm.More(); itrm.Next())
{
const int argID = itrm.Key();
const TDF_Label& argL = scope->GetFunction(argID);
Node* n = findNode(argL);
if (!n)
continue;
scene()->addItem(new Edge(n, node));
}
}
fIterator.Next();
}
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]
//! [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.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 = 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)
{
Node* node = 0;
for (int i = 0; i < scene()->items().size(); i++)
{
Node* n = qgraphicsitem_cast<Node *>(scene()->items().at(i));
if (n && n->getFunction() == L)
{
node = n;
break;
}
}
return node;
}
void GraphWidget::compute()
{
myNbFinishedThreads = 0;
TFunction_Iterator fIterator(myDocument->Main());
fIterator.SetUsageOfExecutionStatus(true);
myThread1 = new FThread();
if (myNbThreads > 1)
myThread2 = new FThread();
if (myNbThreads > 2)
myThread3 = new FThread();
if (myNbThreads > 3)
myThread4 = new FThread();
// Logbook
Handle(TFunction_Logbook) log = TFunction_Scope::Set(myDocument->Main())->GetLogbook();
myThread1->setLogbook(log);
if (myNbThreads > 1)
myThread2->setLogbook(log);
if (myNbThreads > 2)
myThread3->setLogbook(log);
if (myNbThreads > 3)
myThread4->setLogbook(log);
myThread1->setIterator(fIterator);
if (myNbThreads > 1)
myThread2->setIterator(fIterator);
if (myNbThreads > 2)
myThread3->setIterator(fIterator);
if (myNbThreads > 3)
myThread4->setIterator(fIterator);
myThread1->setGraph(this);
if (myNbThreads > 1)
myThread2->setGraph(this);
if (myNbThreads > 2)
myThread3->setGraph(this);
if (myNbThreads > 3)
myThread4->setGraph(this);
myThread1->setThreadIndex(1);
if (myNbThreads > 1)
myThread2->setThreadIndex(2);
if (myNbThreads > 2)
myThread3->setThreadIndex(3);
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);
if (myNbThreads > 1 && !myThread2->isRunning())
myThread2->start(priority);
if (myNbThreads > 2 && !myThread3->isRunning())
myThread3->start(priority);
if (myNbThreads > 3 && !myThread4->isRunning())
myThread4->start(priority);
}
void GraphWidget::setNbThreads(const int nb)
{
myNbThreads = nb;
if (myNbThreads < 4 && myThread4)
{
myThread4->wait();
myThread4->deleteLater();
myThread4 = 0;
}
if (myNbThreads < 3 && myThread3)
{
myThread3->wait();
myThread3->deleteLater();
myThread3 = 0;
}
if (myNbThreads < 2 && myThread2)
{
myThread2->wait();
myThread2->deleteLater();
myThread2 = 0;
}
for (int i = 1; i <= myNbThreads; i++)
{
TFunction_DriverTable::Get()->AddDriver(PointDriver::GetID(), new PointDriver(), i);
TFunction_DriverTable::Get()->AddDriver(CircleDriver::GetID(), new CircleDriver(), i);
TFunction_DriverTable::Get()->AddDriver(PrismDriver::GetID(), new PrismDriver(), i);
TFunction_DriverTable::Get()->AddDriver(ConeDriver::GetID(), new ConeDriver(), i);
TFunction_DriverTable::Get()->AddDriver(CylinderDriver::GetID(), new CylinderDriver(), i);
TFunction_DriverTable::Get()->AddDriver(ShapeSaverDriver::GetID(), new ShapeSaverDriver(), i);
TFunction_DriverTable::Get()->AddDriver(SimpleDriver::GetID(), new SimpleDriver(), i);
}
}
int GraphWidget::getNbThreads()
{
return myNbThreads;
}
void GraphWidget::setFinished()
{
myNbFinishedThreads++;
}
bool GraphWidget::isFinished()
{
return myNbThreads == myNbFinishedThreads ;
}
void GraphWidget::accelerateThread(const int thread_index)
{
bool all_slow = true;
if (myThread1 && myThread1->priority() != QThread::LowPriority)
all_slow = false;
if (all_slow && myThread2 && myThread2->priority() != QThread::LowPriority)
all_slow = false;
if (all_slow && myThread3 && myThread3->priority() != QThread::LowPriority)
all_slow = false;
if (all_slow && myThread4 && myThread4->priority() != QThread::LowPriority)
all_slow = false;
if (!all_slow)
return;
QThread::Priority priority = QThread::NormalPriority;
switch (thread_index)
{
case 1:
myThread1->setPriority(priority);
break;
case 2:
myThread2->setPriority(priority);
break;
case 3:
myThread3->setPriority(priority);
break;
case 4:
myThread4->setPriority(priority);
break;
}
}
void GraphWidget::slowDownThread(const int thread_index)
{
QThread::Priority priority = QThread::LowPriority;
switch (thread_index)
{
case 1:
myThread1->setPriority(priority);
break;
case 2:
myThread2->setPriority(priority);
break;
case 3:
myThread3->setPriority(priority);
break;
case 4:
myThread4->setPriority(priority);
break;
}
}