2017-12-13 22:38:46 +00:00
|
|
|
#include "GraphView.h"
|
|
|
|
|
2019-04-04 05:54:42 +00:00
|
|
|
#include "GraphGridLayout.h"
|
2019-08-03 13:10:44 +00:00
|
|
|
#ifdef CUTTER_ENABLE_GRAPHVIZ
|
|
|
|
#include "GraphvizLayout.h"
|
|
|
|
#endif
|
2020-06-05 23:06:38 +00:00
|
|
|
#include "GraphHorizontalAdapter.h"
|
2019-10-06 17:35:44 +00:00
|
|
|
#include "Helpers.h"
|
2019-04-04 05:54:42 +00:00
|
|
|
|
2017-12-13 22:38:46 +00:00
|
|
|
#include <vector>
|
|
|
|
#include <QPainter>
|
|
|
|
#include <QMouseEvent>
|
2019-04-10 19:07:53 +00:00
|
|
|
#include <QKeyEvent>
|
2017-12-13 22:38:46 +00:00
|
|
|
#include <QPropertyAnimation>
|
2019-09-19 05:19:50 +00:00
|
|
|
#include <QSvgGenerator>
|
2017-12-13 22:38:46 +00:00
|
|
|
|
2019-10-06 17:35:44 +00:00
|
|
|
#ifndef CUTTER_NO_OPENGL_GRAPH
|
2019-04-07 10:53:42 +00:00
|
|
|
#include <QOpenGLContext>
|
|
|
|
#include <QOpenGLWidget>
|
|
|
|
#include <QOpenGLPaintDevice>
|
|
|
|
#include <QOpenGLExtraFunctions>
|
|
|
|
#endif
|
|
|
|
|
2017-12-13 22:38:46 +00:00
|
|
|
GraphView::GraphView(QWidget *parent)
|
|
|
|
: QAbstractScrollArea(parent)
|
2019-04-07 10:53:42 +00:00
|
|
|
, useGL(false)
|
2019-10-06 17:35:44 +00:00
|
|
|
#ifndef CUTTER_NO_OPENGL_GRAPH
|
2019-04-07 10:53:42 +00:00
|
|
|
, cacheTexture(0)
|
|
|
|
, cacheFBO(0)
|
|
|
|
#endif
|
2017-12-13 22:38:46 +00:00
|
|
|
{
|
2019-10-06 17:35:44 +00:00
|
|
|
#ifndef CUTTER_NO_OPENGL_GRAPH
|
2019-04-07 10:53:42 +00:00
|
|
|
if (useGL) {
|
|
|
|
glWidget = new QOpenGLWidget(this);
|
|
|
|
setViewport(glWidget);
|
|
|
|
} else {
|
|
|
|
glWidget = nullptr;
|
|
|
|
}
|
|
|
|
#endif
|
2020-06-05 23:06:38 +00:00
|
|
|
setGraphLayout(makeGraphLayout(Layout::GridMedium));
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
GraphView::~GraphView()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
// Callbacks
|
|
|
|
|
|
|
|
void GraphView::blockClicked(GraphView::GraphBlock &block, QMouseEvent *event, QPoint pos)
|
|
|
|
{
|
|
|
|
Q_UNUSED(block);
|
|
|
|
Q_UNUSED(event);
|
|
|
|
Q_UNUSED(pos);
|
|
|
|
}
|
|
|
|
|
2017-12-14 21:07:48 +00:00
|
|
|
void GraphView::blockDoubleClicked(GraphView::GraphBlock &block, QMouseEvent *event, QPoint pos)
|
|
|
|
{
|
|
|
|
Q_UNUSED(block);
|
|
|
|
Q_UNUSED(event);
|
|
|
|
Q_UNUSED(pos);
|
|
|
|
}
|
|
|
|
|
2017-12-19 16:59:39 +00:00
|
|
|
void GraphView::blockHelpEvent(GraphView::GraphBlock &block, QHelpEvent *event, QPoint pos)
|
|
|
|
{
|
|
|
|
Q_UNUSED(block);
|
|
|
|
Q_UNUSED(event);
|
|
|
|
Q_UNUSED(pos);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GraphView::helpEvent(QHelpEvent *event)
|
|
|
|
{
|
2019-10-06 17:35:44 +00:00
|
|
|
auto p = viewToLogicalCoordinates(event->pos());
|
|
|
|
if (auto block = getBlockContaining(p)) {
|
|
|
|
blockHelpEvent(*block, event, p - QPoint(block->x, block->y));
|
|
|
|
return true;
|
2017-12-19 16:59:39 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-12-13 22:38:46 +00:00
|
|
|
void GraphView::blockTransitionedTo(GraphView::GraphBlock *to)
|
|
|
|
{
|
|
|
|
Q_UNUSED(to);
|
|
|
|
}
|
|
|
|
|
2018-03-21 20:32:32 +00:00
|
|
|
GraphView::EdgeConfiguration GraphView::edgeConfiguration(GraphView::GraphBlock &from,
|
2019-09-19 05:19:50 +00:00
|
|
|
GraphView::GraphBlock *to,
|
|
|
|
bool interactive)
|
2017-12-13 22:38:46 +00:00
|
|
|
{
|
2019-09-19 05:19:50 +00:00
|
|
|
Q_UNUSED(from)
|
|
|
|
Q_UNUSED(to)
|
|
|
|
Q_UNUSED(interactive)
|
2017-12-13 22:38:46 +00:00
|
|
|
qWarning() << "Edge configuration not overridden!";
|
|
|
|
EdgeConfiguration ec;
|
|
|
|
return ec;
|
|
|
|
}
|
|
|
|
|
2019-10-06 17:35:44 +00:00
|
|
|
void GraphView::blockContextMenuRequested(GraphView::GraphBlock &, QContextMenuEvent *, QPoint)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-12-19 16:59:39 +00:00
|
|
|
bool GraphView::event(QEvent *event)
|
|
|
|
{
|
2018-03-21 20:32:32 +00:00
|
|
|
if (event->type() == QEvent::ToolTip) {
|
|
|
|
if (helpEvent(static_cast<QHelpEvent *>(event))) {
|
2017-12-19 16:59:39 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return QAbstractScrollArea::event(event);
|
|
|
|
}
|
|
|
|
|
2019-10-06 17:35:44 +00:00
|
|
|
void GraphView::contextMenuEvent(QContextMenuEvent *event)
|
|
|
|
{
|
|
|
|
event->ignore();
|
|
|
|
if (event->reason() == QContextMenuEvent::Mouse) {
|
|
|
|
QPoint p = viewToLogicalCoordinates(event->pos());
|
|
|
|
if (auto block = getBlockContaining(p)) {
|
|
|
|
blockContextMenuRequested(*block, event, p);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-16 10:43:45 +00:00
|
|
|
void GraphView::computeGraphPlacement()
|
2017-12-13 22:38:46 +00:00
|
|
|
{
|
2019-04-04 05:54:42 +00:00
|
|
|
graphLayoutSystem->CalculateLayout(blocks, entry, width, height);
|
2020-06-16 10:43:45 +00:00
|
|
|
setCacheDirty();
|
2020-07-16 08:05:10 +00:00
|
|
|
clampViewOffset();
|
2017-12-14 21:07:48 +00:00
|
|
|
viewport()->update();
|
2019-02-16 17:17:11 +00:00
|
|
|
}
|
|
|
|
|
2020-07-16 08:05:10 +00:00
|
|
|
void GraphView::cleanupEdges(GraphLayout::Graph &graph)
|
|
|
|
{
|
|
|
|
for (auto &blockIt : graph) {
|
|
|
|
auto &block = blockIt.second;
|
|
|
|
auto outIt = block.edges.begin();
|
|
|
|
std::unordered_set<ut64> seenEdges;
|
|
|
|
for (auto it = block.edges.begin(), end = block.edges.end(); it != end; ++it) {
|
|
|
|
// remove edges going to different functions
|
|
|
|
// and remove duplicate edges, common in switch statements
|
|
|
|
if (graph.find(it->target) != graph.end() &&
|
|
|
|
seenEdges.find(it->target) == seenEdges.end()) {
|
|
|
|
*outIt++ = *it;
|
|
|
|
seenEdges.insert(it->target);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
block.edges.erase(outIt, block.edges.end());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-07 11:02:35 +00:00
|
|
|
void GraphView::beginMouseDrag(QMouseEvent *event)
|
|
|
|
{
|
|
|
|
scroll_base_x = event->x();
|
|
|
|
scroll_base_y = event->y();
|
|
|
|
scroll_mode = true;
|
|
|
|
setCursor(Qt::ClosedHandCursor);
|
|
|
|
viewport()->grabMouse();
|
|
|
|
}
|
|
|
|
|
2019-04-14 12:18:24 +00:00
|
|
|
void GraphView::setViewOffset(QPoint offset)
|
|
|
|
{
|
2019-04-21 16:30:57 +00:00
|
|
|
setViewOffsetInternal(offset);
|
2019-04-14 12:18:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GraphView::setViewScale(qreal scale)
|
|
|
|
{
|
|
|
|
this->current_scale = scale;
|
|
|
|
emit viewScaleChanged(scale);
|
|
|
|
}
|
|
|
|
|
2019-04-08 06:59:16 +00:00
|
|
|
QSize GraphView::getCacheSize()
|
|
|
|
{
|
|
|
|
return
|
2019-10-06 17:35:44 +00:00
|
|
|
#ifndef CUTTER_NO_OPENGL_GRAPH
|
2019-04-08 06:59:16 +00:00
|
|
|
useGL ? cacheSize :
|
|
|
|
#endif
|
|
|
|
pixmap.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
qreal GraphView::getCacheDevicePixelRatioF()
|
|
|
|
{
|
|
|
|
return
|
2019-10-06 17:35:44 +00:00
|
|
|
#ifndef CUTTER_NO_OPENGL_GRAPH
|
2019-04-08 06:59:16 +00:00
|
|
|
useGL ? 1.0 :
|
|
|
|
#endif
|
2019-10-06 17:35:44 +00:00
|
|
|
qhelpers::devicePixelRatio(&pixmap);
|
2019-04-08 06:59:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QSize GraphView::getRequiredCacheSize()
|
|
|
|
{
|
2019-12-22 18:56:33 +00:00
|
|
|
return viewport()->size() * qhelpers::devicePixelRatio(this);
|
2019-04-08 06:59:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
qreal GraphView::getRequiredCacheDevicePixelRatioF()
|
|
|
|
{
|
|
|
|
return
|
2019-10-06 17:35:44 +00:00
|
|
|
#ifndef CUTTER_NO_OPENGL_GRAPH
|
2019-04-08 06:59:16 +00:00
|
|
|
useGL ? 1.0f :
|
|
|
|
#endif
|
2019-10-06 17:35:44 +00:00
|
|
|
qhelpers::devicePixelRatio(this);
|
2019-04-08 06:59:16 +00:00
|
|
|
}
|
|
|
|
|
2019-04-07 10:53:42 +00:00
|
|
|
void GraphView::paintEvent(QPaintEvent *)
|
2017-12-13 22:38:46 +00:00
|
|
|
{
|
2019-10-06 17:35:44 +00:00
|
|
|
#ifndef CUTTER_NO_OPENGL_GRAPH
|
2019-04-07 10:53:42 +00:00
|
|
|
if (useGL) {
|
|
|
|
glWidget->makeCurrent();
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-04-08 06:59:16 +00:00
|
|
|
if (!qFuzzyCompare(getCacheDevicePixelRatioF(), getRequiredCacheDevicePixelRatioF())
|
2019-04-21 16:30:57 +00:00
|
|
|
|| getCacheSize() != getRequiredCacheSize()) {
|
2019-04-08 06:59:16 +00:00
|
|
|
setCacheDirty();
|
|
|
|
}
|
|
|
|
|
2019-04-21 16:30:57 +00:00
|
|
|
if (cacheDirty) {
|
2019-04-07 10:53:42 +00:00
|
|
|
paintGraphCache();
|
2019-04-08 06:59:16 +00:00
|
|
|
cacheDirty = false;
|
2019-04-07 10:53:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (useGL) {
|
2019-10-06 17:35:44 +00:00
|
|
|
#ifndef CUTTER_NO_OPENGL_GRAPH
|
2019-04-07 10:53:42 +00:00
|
|
|
auto gl = glWidget->context()->extraFunctions();
|
|
|
|
gl->glBindFramebuffer(GL_READ_FRAMEBUFFER, cacheFBO);
|
|
|
|
gl->glBindFramebuffer(GL_DRAW_FRAMEBUFFER, glWidget->defaultFramebufferObject());
|
2019-12-22 18:56:33 +00:00
|
|
|
auto dpr = qhelpers::devicePixelRatio(this);
|
2019-04-08 06:59:16 +00:00
|
|
|
gl->glBlitFramebuffer(0, 0, cacheSize.width(), cacheSize.height(),
|
2019-12-22 18:56:33 +00:00
|
|
|
0, 0, viewport()->width() * dpr, viewport()->height() * dpr,
|
2019-04-07 10:53:42 +00:00
|
|
|
GL_COLOR_BUFFER_BIT, GL_NEAREST);
|
|
|
|
glWidget->doneCurrent();
|
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
QPainter p(viewport());
|
2019-12-22 18:56:33 +00:00
|
|
|
p.drawPixmap(QPoint(0, 0), pixmap);
|
2019-04-07 10:53:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-21 16:30:57 +00:00
|
|
|
void GraphView::clampViewOffset()
|
|
|
|
{
|
|
|
|
const qreal edgeFraction = 0.25;
|
|
|
|
qreal edgeX = edgeFraction * (viewport()->width() / current_scale);
|
|
|
|
qreal edgeY = edgeFraction * (viewport()->height() / current_scale);
|
|
|
|
offset.rx() = std::max(std::min(qreal(offset.x()), width - edgeX),
|
|
|
|
- viewport()->width() / current_scale + edgeX);
|
|
|
|
offset.ry() = std::max(std::min(qreal(offset.y()), height - edgeY),
|
|
|
|
- viewport()->height() / current_scale + edgeY);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphView::setViewOffsetInternal(QPoint pos, bool emitSignal)
|
|
|
|
{
|
|
|
|
offset = pos;
|
|
|
|
clampViewOffset();
|
|
|
|
if (emitSignal)
|
|
|
|
emit viewOffsetChanged(offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphView::addViewOffset(QPoint move, bool emitSignal)
|
|
|
|
{
|
|
|
|
setViewOffsetInternal(offset + move, emitSignal);
|
|
|
|
}
|
|
|
|
|
2019-04-07 10:53:42 +00:00
|
|
|
void GraphView::paintGraphCache()
|
|
|
|
{
|
2019-10-06 17:35:44 +00:00
|
|
|
#ifndef CUTTER_NO_OPENGL_GRAPH
|
2019-12-22 18:56:33 +00:00
|
|
|
std::unique_ptr<QOpenGLPaintDevice> paintDevice;
|
2019-04-07 10:53:42 +00:00
|
|
|
#endif
|
|
|
|
QPainter p;
|
|
|
|
if (useGL) {
|
2019-10-06 17:35:44 +00:00
|
|
|
#ifndef CUTTER_NO_OPENGL_GRAPH
|
2019-04-07 10:53:42 +00:00
|
|
|
auto gl = QOpenGLContext::currentContext()->functions();
|
|
|
|
|
|
|
|
bool resizeTex = false;
|
2019-12-22 18:56:33 +00:00
|
|
|
QSize sizeNeed = getRequiredCacheSize();
|
2019-04-07 10:53:42 +00:00
|
|
|
if (!cacheTexture) {
|
|
|
|
gl->glGenTextures(1, &cacheTexture);
|
|
|
|
gl->glBindTexture(GL_TEXTURE_2D, cacheTexture);
|
|
|
|
gl->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
|
|
gl->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
|
|
gl->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
|
|
gl->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
|
|
|
|
gl->glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
|
|
|
|
resizeTex = true;
|
2019-12-22 18:56:33 +00:00
|
|
|
} else if (cacheSize != sizeNeed) {
|
2019-04-07 10:53:42 +00:00
|
|
|
gl->glBindTexture(GL_TEXTURE_2D, cacheTexture);
|
|
|
|
resizeTex = true;
|
|
|
|
}
|
|
|
|
if (resizeTex) {
|
2019-12-22 18:56:33 +00:00
|
|
|
cacheSize = sizeNeed;
|
|
|
|
gl->glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, cacheSize.width(), cacheSize.height(), 0, GL_RGBA,
|
2019-04-21 16:30:57 +00:00
|
|
|
GL_UNSIGNED_BYTE, nullptr);
|
2019-04-07 10:53:42 +00:00
|
|
|
gl->glGenFramebuffers(1, &cacheFBO);
|
|
|
|
gl->glBindFramebuffer(GL_FRAMEBUFFER, cacheFBO);
|
|
|
|
gl->glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, cacheTexture, 0);
|
|
|
|
} else {
|
|
|
|
gl->glBindFramebuffer(GL_FRAMEBUFFER, cacheFBO);
|
|
|
|
}
|
|
|
|
gl->glViewport(0, 0, viewport()->width(), viewport()->height());
|
2019-12-22 18:56:33 +00:00
|
|
|
gl->glClearColor(backgroundColor.redF(), backgroundColor.greenF(), backgroundColor.blueF(), 1.0f);
|
2019-04-07 10:53:42 +00:00
|
|
|
gl->glClear(GL_COLOR_BUFFER_BIT);
|
2018-07-18 10:15:10 +00:00
|
|
|
|
2019-12-22 18:56:33 +00:00
|
|
|
paintDevice.reset(new QOpenGLPaintDevice(cacheSize));
|
|
|
|
p.begin(paintDevice.get());
|
2019-04-07 10:53:42 +00:00
|
|
|
#endif
|
|
|
|
} else {
|
2019-10-06 17:35:44 +00:00
|
|
|
auto dpr = qhelpers::devicePixelRatio(this);
|
2019-12-22 18:56:33 +00:00
|
|
|
pixmap = QPixmap(getRequiredCacheSize() * dpr);
|
2019-04-07 10:53:42 +00:00
|
|
|
pixmap.setDevicePixelRatio(dpr);
|
2019-12-22 18:56:33 +00:00
|
|
|
pixmap.fill(backgroundColor);
|
2019-04-07 10:53:42 +00:00
|
|
|
p.begin(&pixmap);
|
|
|
|
p.setRenderHint(QPainter::Antialiasing);
|
2019-12-22 18:56:33 +00:00
|
|
|
p.setViewport(this->viewport()->rect());
|
2019-04-07 10:53:42 +00:00
|
|
|
}
|
2019-09-19 05:19:50 +00:00
|
|
|
paint(p, offset, this->viewport()->rect(), current_scale);
|
2017-12-14 21:07:48 +00:00
|
|
|
|
2019-09-19 05:19:50 +00:00
|
|
|
p.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphView::paint(QPainter &p, QPoint offset, QRect viewport, qreal scale, bool interactive)
|
|
|
|
{
|
|
|
|
QPointF offsetF(offset.x(), offset.y());
|
2017-12-13 22:38:46 +00:00
|
|
|
p.setBrush(Qt::black);
|
|
|
|
|
2019-09-19 05:19:50 +00:00
|
|
|
int render_width = viewport.width();
|
|
|
|
int render_height = viewport.height();
|
|
|
|
|
|
|
|
// window - rectangle in logical coordinates
|
|
|
|
QRect window = QRect(offset, QSize(qRound(render_width / scale), qRound(render_height / scale)));
|
|
|
|
p.setWindow(window);
|
2019-12-22 18:56:33 +00:00
|
|
|
QRectF windowF(window.x(), window.y(), window.width(), window.height());
|
2017-12-13 22:38:46 +00:00
|
|
|
|
2018-03-21 20:32:32 +00:00
|
|
|
for (auto &blockIt : blocks) {
|
2017-12-13 22:38:46 +00:00
|
|
|
GraphBlock &block = blockIt.second;
|
|
|
|
|
2019-09-19 05:19:50 +00:00
|
|
|
QRectF blockRect(block.x, block.y, block.width, block.height);
|
2019-02-05 15:21:02 +00:00
|
|
|
|
|
|
|
// Check if block is visible by checking if block intersects with view area
|
2019-09-19 05:19:50 +00:00
|
|
|
if (blockRect.intersects(windowF)) {
|
|
|
|
drawBlock(p, block, interactive);
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
p.setBrush(Qt::gray);
|
|
|
|
|
|
|
|
// Always draw edges
|
|
|
|
// TODO: Only draw edges if they are actually visible ...
|
|
|
|
// Draw edges
|
2018-03-21 20:32:32 +00:00
|
|
|
for (GraphEdge &edge : block.edges) {
|
2019-04-05 06:28:11 +00:00
|
|
|
if (edge.polyline.empty()) {
|
|
|
|
continue;
|
|
|
|
}
|
2019-09-19 05:19:50 +00:00
|
|
|
QPolygonF polyline = edge.polyline;
|
2020-05-21 13:31:22 +00:00
|
|
|
EdgeConfiguration ec = edgeConfiguration(block, &blocks[edge.target], interactive);
|
2019-04-04 05:54:42 +00:00
|
|
|
QPen pen(ec.color);
|
2019-07-30 18:35:00 +00:00
|
|
|
pen.setStyle(ec.lineStyle);
|
|
|
|
pen.setWidthF(pen.width() * ec.width_scale);
|
2019-12-22 18:56:33 +00:00
|
|
|
if (scale_thickness_multiplier && ec.width_scale > 1.01 && pen.widthF() * scale < 2) {
|
2019-09-19 05:19:50 +00:00
|
|
|
pen.setWidthF(ec.width_scale / scale);
|
2019-07-30 18:35:00 +00:00
|
|
|
}
|
2019-09-19 05:19:50 +00:00
|
|
|
if (pen.widthF() * scale < 2) {
|
2019-04-21 16:30:57 +00:00
|
|
|
pen.setWidth(0);
|
|
|
|
}
|
2017-12-13 22:38:46 +00:00
|
|
|
p.setPen(pen);
|
2019-04-04 05:54:42 +00:00
|
|
|
p.setBrush(ec.color);
|
2019-02-16 17:17:11 +00:00
|
|
|
p.drawPolyline(polyline);
|
2017-12-13 22:38:46 +00:00
|
|
|
pen.setStyle(Qt::SolidLine);
|
|
|
|
p.setPen(pen);
|
2019-08-03 13:10:44 +00:00
|
|
|
|
|
|
|
auto drawArrow = [&](QPointF tip, QPointF dir) {
|
2019-12-22 18:56:33 +00:00
|
|
|
pen.setWidth(0);
|
|
|
|
p.setPen(pen);
|
2019-08-03 13:10:44 +00:00
|
|
|
QPolygonF arrow;
|
|
|
|
arrow << tip;
|
|
|
|
QPointF dy(-dir.y(), dir.x());
|
|
|
|
QPointF base = tip - dir * 6;
|
|
|
|
arrow << base + 3 * dy;
|
|
|
|
arrow << base - 3 * dy;
|
2019-09-19 05:19:50 +00:00
|
|
|
p.drawConvexPolygon(arrow);
|
2019-08-03 13:10:44 +00:00
|
|
|
};
|
|
|
|
|
2019-04-04 05:54:42 +00:00
|
|
|
if (!polyline.empty()) {
|
|
|
|
if (ec.start_arrow) {
|
|
|
|
auto firstPt = edge.polyline.first();
|
2019-08-03 13:10:44 +00:00
|
|
|
drawArrow(firstPt, QPointF(0, 1));
|
2019-04-04 05:54:42 +00:00
|
|
|
}
|
|
|
|
if (ec.end_arrow) {
|
|
|
|
auto lastPt = edge.polyline.last();
|
2019-08-03 13:10:44 +00:00
|
|
|
QPointF dir(0, -1);
|
2019-09-19 05:19:50 +00:00
|
|
|
switch (edge.arrow) {
|
|
|
|
case GraphLayout::GraphEdge::Down:
|
|
|
|
dir = QPointF(0, 1);
|
|
|
|
break;
|
|
|
|
case GraphLayout::GraphEdge::Up:
|
|
|
|
dir = QPointF(0, -1);
|
|
|
|
break;
|
|
|
|
case GraphLayout::GraphEdge::Left:
|
|
|
|
dir = QPointF(-1, 0);
|
|
|
|
break;
|
|
|
|
case GraphLayout::GraphEdge::Right:
|
|
|
|
dir = QPointF(1, 0);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2019-08-03 13:10:44 +00:00
|
|
|
}
|
|
|
|
drawArrow(lastPt, dir);
|
2019-04-04 05:54:42 +00:00
|
|
|
}
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-09-19 05:19:50 +00:00
|
|
|
}
|
|
|
|
|
2020-03-09 20:59:03 +00:00
|
|
|
void GraphView::saveAsBitmap(QString path, const char *format, double scaler, bool transparent)
|
2019-09-19 05:19:50 +00:00
|
|
|
{
|
2020-07-16 08:05:10 +00:00
|
|
|
QImage image(width * scaler, height * scaler, QImage::Format_ARGB32);
|
|
|
|
if (transparent) {
|
2020-03-09 20:59:03 +00:00
|
|
|
image.fill(qRgba(0, 0, 0, 0));
|
2020-07-16 08:05:10 +00:00
|
|
|
} else {
|
2020-03-09 20:59:03 +00:00
|
|
|
image.fill(backgroundColor);
|
|
|
|
}
|
2019-09-19 05:19:50 +00:00
|
|
|
QPainter p;
|
|
|
|
p.begin(&image);
|
2020-03-09 20:59:03 +00:00
|
|
|
paint(p, QPoint(0, 0), image.rect(), scaler, false);
|
2019-09-19 05:19:50 +00:00
|
|
|
p.end();
|
|
|
|
if (!image.save(path, format)) {
|
|
|
|
qWarning() << "Could not save image";
|
|
|
|
}
|
|
|
|
}
|
2017-12-13 22:38:46 +00:00
|
|
|
|
2019-09-19 05:19:50 +00:00
|
|
|
void GraphView::saveAsSvg(QString path)
|
|
|
|
{
|
|
|
|
QSvgGenerator generator;
|
|
|
|
generator.setFileName(path);
|
|
|
|
generator.setSize(QSize(width, height));
|
|
|
|
generator.setViewBox(QRect(0, 0, width, height));
|
|
|
|
generator.setTitle("Cutter graph export");
|
|
|
|
QPainter p;
|
|
|
|
p.begin(&generator);
|
|
|
|
paint(p, QPoint(0, 0), QRect(0, 0, width, height), 1.0, false);
|
2019-04-07 10:53:42 +00:00
|
|
|
p.end();
|
2019-03-12 07:37:10 +00:00
|
|
|
}
|
|
|
|
|
2019-02-16 17:17:11 +00:00
|
|
|
void GraphView::center()
|
2017-12-13 22:38:46 +00:00
|
|
|
{
|
2019-04-14 12:18:24 +00:00
|
|
|
centerX(false);
|
|
|
|
centerY(false);
|
|
|
|
emit viewOffsetChanged(offset);
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
|
2019-04-14 12:18:24 +00:00
|
|
|
void GraphView::centerX(bool emitSignal)
|
2017-12-13 22:38:46 +00:00
|
|
|
{
|
2019-03-23 08:21:06 +00:00
|
|
|
offset.rx() = -((viewport()->width() - width * current_scale) / 2);
|
|
|
|
offset.rx() /= current_scale;
|
2019-04-21 16:30:57 +00:00
|
|
|
clampViewOffset();
|
2019-04-14 12:18:24 +00:00
|
|
|
if (emitSignal) {
|
|
|
|
emit viewOffsetChanged(offset);
|
|
|
|
}
|
2019-02-16 17:17:11 +00:00
|
|
|
}
|
2017-12-13 22:38:46 +00:00
|
|
|
|
2019-04-14 12:18:24 +00:00
|
|
|
void GraphView::centerY(bool emitSignal)
|
2019-02-16 17:17:11 +00:00
|
|
|
{
|
2019-03-23 08:21:06 +00:00
|
|
|
offset.ry() = -((viewport()->height() - height * current_scale) / 2);
|
|
|
|
offset.ry() /= current_scale;
|
2019-04-21 16:30:57 +00:00
|
|
|
clampViewOffset();
|
2019-04-14 12:18:24 +00:00
|
|
|
if (emitSignal) {
|
|
|
|
emit viewOffsetChanged(offset);
|
|
|
|
}
|
2019-02-16 17:17:11 +00:00
|
|
|
}
|
2017-12-13 22:38:46 +00:00
|
|
|
|
2019-05-19 10:27:15 +00:00
|
|
|
void GraphView::showBlock(GraphBlock &block, bool anywhere)
|
2019-02-16 17:17:11 +00:00
|
|
|
{
|
2020-07-16 08:05:10 +00:00
|
|
|
showRectangle(QRect(block.x, block.y, block.width, block.height), anywhere);
|
|
|
|
blockTransitionedTo(&block);
|
2019-05-19 10:27:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GraphView::showRectangle(const QRect &block, bool anywhere)
|
|
|
|
{
|
|
|
|
QSizeF renderSize = QSizeF(viewport()->size()) / current_scale;
|
2019-02-16 17:17:11 +00:00
|
|
|
if (width * current_scale <= viewport()->width()) {
|
2019-04-14 12:18:24 +00:00
|
|
|
centerX(false);
|
2017-12-13 22:38:46 +00:00
|
|
|
} else {
|
2019-05-19 10:27:15 +00:00
|
|
|
if (!anywhere || block.x() < offset.x() || block.right() > offset.x() + renderSize.width()) {
|
|
|
|
offset.rx() = block.x() - ((renderSize.width() - block.width()) / 2);
|
|
|
|
}
|
2019-02-16 17:17:11 +00:00
|
|
|
}
|
|
|
|
if (height * current_scale <= viewport()->height()) {
|
2019-04-14 12:18:24 +00:00
|
|
|
centerY(false);
|
2019-02-16 17:17:11 +00:00
|
|
|
} else {
|
2019-06-29 06:28:35 +00:00
|
|
|
if (!anywhere || block.y() < offset.y()
|
2019-05-19 10:27:15 +00:00
|
|
|
|| block.bottom() > offset.y() + renderSize.height()) {
|
2019-06-29 06:28:35 +00:00
|
|
|
offset.ry() = block.y();
|
|
|
|
// Leave some space at top if possible
|
|
|
|
const qreal topPadding = 10 / current_scale;
|
|
|
|
if (block.height() + topPadding < renderSize.height()) {
|
|
|
|
offset.ry() -= topPadding;
|
|
|
|
}
|
2019-05-19 10:27:15 +00:00
|
|
|
}
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
2019-04-21 16:30:57 +00:00
|
|
|
clampViewOffset();
|
2019-04-14 12:18:24 +00:00
|
|
|
emit viewOffsetChanged(offset);
|
2017-12-14 21:07:48 +00:00
|
|
|
viewport()->update();
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
|
2019-10-06 17:35:44 +00:00
|
|
|
GraphView::GraphBlock *GraphView::getBlockContaining(QPoint p)
|
|
|
|
{
|
|
|
|
// Check if a block was clicked
|
|
|
|
for (auto &blockIt : blocks) {
|
|
|
|
GraphBlock &block = blockIt.second;
|
|
|
|
|
|
|
|
QRect rec(block.x, block.y, block.width, block.height);
|
|
|
|
if (rec.contains(p)) {
|
|
|
|
return █
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
QPoint GraphView::viewToLogicalCoordinates(QPoint p)
|
|
|
|
{
|
|
|
|
return p / current_scale + offset;
|
|
|
|
}
|
|
|
|
|
2020-07-16 08:05:10 +00:00
|
|
|
QPoint GraphView::logicalToViewCoordinates(QPoint p)
|
|
|
|
{
|
|
|
|
return (p - offset) * current_scale;
|
|
|
|
}
|
|
|
|
|
2020-06-05 23:06:38 +00:00
|
|
|
void GraphView::setGraphLayout(std::unique_ptr<GraphLayout> layout)
|
2019-08-03 13:10:44 +00:00
|
|
|
{
|
2020-06-05 23:06:38 +00:00
|
|
|
graphLayoutSystem = std::move(layout);
|
|
|
|
if (!graphLayoutSystem) {
|
|
|
|
graphLayoutSystem = makeGraphLayout(Layout::GridMedium);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-16 10:43:45 +00:00
|
|
|
void GraphView::setLayoutConfig(const GraphLayout::LayoutConfig &config)
|
|
|
|
{
|
|
|
|
graphLayoutSystem->setLayoutConfig(config);
|
|
|
|
}
|
|
|
|
|
2020-06-05 23:06:38 +00:00
|
|
|
std::unique_ptr<GraphLayout> GraphView::makeGraphLayout(GraphView::Layout layout, bool horizontal)
|
|
|
|
{
|
|
|
|
std::unique_ptr<GraphLayout> result;
|
|
|
|
bool needAdapter = true;
|
2020-07-16 08:05:10 +00:00
|
|
|
|
|
|
|
#ifdef CUTTER_ENABLE_GRAPHVIZ
|
|
|
|
auto makeGraphvizLayout = [&](GraphvizLayout::LayoutType type) {
|
|
|
|
result.reset(new GraphvizLayout(type,
|
|
|
|
horizontal ? GraphvizLayout::Direction::LR : GraphvizLayout::Direction::TB));
|
|
|
|
needAdapter = false;
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2019-08-03 13:10:44 +00:00
|
|
|
switch (layout) {
|
2019-09-19 05:19:50 +00:00
|
|
|
case Layout::GridNarrow:
|
2020-06-05 23:06:38 +00:00
|
|
|
result.reset(new GraphGridLayout(GraphGridLayout::LayoutType::Narrow));
|
2019-09-19 05:19:50 +00:00
|
|
|
break;
|
|
|
|
case Layout::GridMedium:
|
2020-06-05 23:06:38 +00:00
|
|
|
result.reset(new GraphGridLayout(GraphGridLayout::LayoutType::Medium));
|
2019-09-19 05:19:50 +00:00
|
|
|
break;
|
|
|
|
case Layout::GridWide:
|
2020-06-05 23:06:38 +00:00
|
|
|
result.reset(new GraphGridLayout(GraphGridLayout::LayoutType::Wide));
|
2019-09-19 05:19:50 +00:00
|
|
|
break;
|
2020-07-03 17:09:37 +00:00
|
|
|
case Layout::GridAAA:
|
|
|
|
case Layout::GridAAB:
|
|
|
|
case Layout::GridABA:
|
|
|
|
case Layout::GridABB:
|
|
|
|
case Layout::GridBAA:
|
|
|
|
case Layout::GridBAB:
|
|
|
|
case Layout::GridBBA:
|
2020-07-16 08:05:10 +00:00
|
|
|
case Layout::GridBBB: {
|
2020-07-03 17:09:37 +00:00
|
|
|
int options = static_cast<int>(layout) - static_cast<int>(Layout::GridAAA);
|
|
|
|
std::unique_ptr<GraphGridLayout> gridLayout(new GraphGridLayout());
|
|
|
|
gridLayout->setTightSubtreePlacement((options & 1) == 0);
|
|
|
|
gridLayout->setParentBetweenDirectChild((options & 2));
|
|
|
|
gridLayout->setLayoutOptimization((options & 4));
|
|
|
|
result = std::move(gridLayout);
|
|
|
|
break;
|
|
|
|
}
|
2019-08-03 13:10:44 +00:00
|
|
|
#ifdef CUTTER_ENABLE_GRAPHVIZ
|
2019-09-19 05:19:50 +00:00
|
|
|
case Layout::GraphvizOrtho:
|
2020-07-16 08:05:10 +00:00
|
|
|
makeGraphvizLayout(GraphvizLayout::LayoutType::DotOrtho);
|
2019-09-19 05:19:50 +00:00
|
|
|
break;
|
|
|
|
case Layout::GraphvizPolyline:
|
2020-07-16 08:05:10 +00:00
|
|
|
makeGraphvizLayout(GraphvizLayout::LayoutType::DotPolyline);
|
|
|
|
break;
|
|
|
|
case Layout::GraphvizSfdp:
|
|
|
|
makeGraphvizLayout(GraphvizLayout::LayoutType::Sfdp);
|
|
|
|
break;
|
|
|
|
case Layout::GraphvizNeato:
|
|
|
|
makeGraphvizLayout(GraphvizLayout::LayoutType::Neato);
|
|
|
|
break;
|
|
|
|
case Layout::GraphvizTwoPi:
|
|
|
|
makeGraphvizLayout(GraphvizLayout::LayoutType::TwoPi);
|
|
|
|
break;
|
|
|
|
case Layout::GraphvizCirco:
|
|
|
|
makeGraphvizLayout(GraphvizLayout::LayoutType::Circo);
|
2019-09-19 05:19:50 +00:00
|
|
|
break;
|
2019-08-03 13:10:44 +00:00
|
|
|
#endif
|
|
|
|
}
|
2020-06-05 23:06:38 +00:00
|
|
|
if (needAdapter && horizontal) {
|
|
|
|
result.reset(new GraphHorizontalAdapter(std::move(result)));
|
|
|
|
}
|
|
|
|
return result;
|
2019-08-03 13:10:44 +00:00
|
|
|
}
|
|
|
|
|
2017-12-13 22:38:46 +00:00
|
|
|
void GraphView::addBlock(GraphView::GraphBlock block)
|
|
|
|
{
|
2017-12-14 21:07:48 +00:00
|
|
|
blocks[block.entry] = block;
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GraphView::setEntry(ut64 e)
|
|
|
|
{
|
2017-12-14 21:07:48 +00:00
|
|
|
entry = e;
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool GraphView::checkPointClicked(QPointF &point, int x, int y, bool above_y)
|
|
|
|
{
|
|
|
|
int half_target_size = 5;
|
2018-03-21 20:32:32 +00:00
|
|
|
if ((point.x() - half_target_size < x) &&
|
2017-12-13 22:38:46 +00:00
|
|
|
(point.y() - (above_y ? (2 * half_target_size) : 0) < y) &&
|
|
|
|
(x < point.x() + half_target_size) &&
|
2018-03-21 20:32:32 +00:00
|
|
|
(y < point.y() + (above_y ? 0 : (3 * half_target_size)))) {
|
2017-12-13 22:38:46 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mouse events
|
|
|
|
void GraphView::mousePressEvent(QMouseEvent *event)
|
|
|
|
{
|
2019-04-07 11:02:35 +00:00
|
|
|
if (event->button() == Qt::MiddleButton) {
|
|
|
|
beginMouseDrag(event);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-10-06 17:35:44 +00:00
|
|
|
QPoint pos = viewToLogicalCoordinates(event->pos());
|
2017-12-13 22:38:46 +00:00
|
|
|
|
|
|
|
// Check if a block was clicked
|
2019-10-06 17:35:44 +00:00
|
|
|
if (auto block = getBlockContaining(pos)) {
|
|
|
|
blockClicked(*block, event, pos - QPoint(block->x, block->y));
|
|
|
|
// Don't do anything else here! blockClicked might seek and
|
|
|
|
// all our data is invalid then.
|
|
|
|
return;
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check if a line beginning/end was clicked
|
2019-04-18 10:10:18 +00:00
|
|
|
if (event->button() == Qt::LeftButton) {
|
|
|
|
for (auto &blockIt : blocks) {
|
|
|
|
GraphBlock &block = blockIt.second;
|
|
|
|
for (GraphEdge &edge : block.edges) {
|
|
|
|
if (edge.polyline.length() < 2) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
QPointF start = edge.polyline.first();
|
|
|
|
QPointF end = edge.polyline.last();
|
2019-10-06 17:35:44 +00:00
|
|
|
if (checkPointClicked(start, pos.x(), pos.y())) {
|
2019-04-18 10:10:18 +00:00
|
|
|
showBlock(blocks[edge.target]);
|
|
|
|
// TODO: Callback to child
|
|
|
|
return;
|
|
|
|
break;
|
|
|
|
}
|
2019-10-06 17:35:44 +00:00
|
|
|
if (checkPointClicked(end, pos.x(), pos.y(), true)) {
|
2019-04-18 10:10:18 +00:00
|
|
|
showBlock(block);
|
|
|
|
// TODO: Callback to child
|
|
|
|
return;
|
|
|
|
break;
|
|
|
|
}
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// No block was clicked
|
2018-03-21 20:32:32 +00:00
|
|
|
if (event->button() == Qt::LeftButton) {
|
2017-12-13 22:38:46 +00:00
|
|
|
//Left click outside any block, enter scrolling mode
|
2019-04-07 11:02:35 +00:00
|
|
|
beginMouseDrag(event);
|
2019-04-18 10:10:18 +00:00
|
|
|
return;
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
|
2019-04-18 10:10:18 +00:00
|
|
|
QAbstractScrollArea::mousePressEvent(event);
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
|
2018-03-21 20:32:32 +00:00
|
|
|
void GraphView::mouseMoveEvent(QMouseEvent *event)
|
2017-12-13 22:38:46 +00:00
|
|
|
{
|
2018-03-21 20:32:32 +00:00
|
|
|
if (scroll_mode) {
|
2019-04-21 16:30:57 +00:00
|
|
|
addViewOffset(QPoint(scroll_base_x - event->x(), scroll_base_y - event->y()) / current_scale);
|
2017-12-14 21:07:48 +00:00
|
|
|
scroll_base_x = event->x();
|
|
|
|
scroll_base_y = event->y();
|
2019-02-16 17:17:11 +00:00
|
|
|
viewport()->update();
|
2017-12-14 21:07:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GraphView::mouseDoubleClickEvent(QMouseEvent *event)
|
|
|
|
{
|
2019-10-06 17:35:44 +00:00
|
|
|
auto p = viewToLogicalCoordinates(event->pos());
|
|
|
|
if (auto block = getBlockContaining(p)) {
|
|
|
|
blockDoubleClicked(*block, event, p - QPoint(block->x, block->y));
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-21 16:30:57 +00:00
|
|
|
void GraphView::keyPressEvent(QKeyEvent *event)
|
2019-04-10 19:07:53 +00:00
|
|
|
{
|
|
|
|
const int delta = static_cast<int>(30.0 / current_scale);
|
|
|
|
int dx = 0, dy = 0;
|
|
|
|
switch (event->key()) {
|
|
|
|
case Qt::Key_Up:
|
|
|
|
dy = -delta;
|
|
|
|
break;
|
|
|
|
case Qt::Key_Down:
|
|
|
|
dy = delta;
|
|
|
|
break;
|
|
|
|
case Qt::Key_Left:
|
|
|
|
dx = -delta;
|
|
|
|
break;
|
|
|
|
case Qt::Key_Right:
|
|
|
|
dx = delta;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
QAbstractScrollArea::keyPressEvent(event);
|
|
|
|
return;
|
|
|
|
}
|
2019-04-21 16:30:57 +00:00
|
|
|
addViewOffset(QPoint(dx, dy));
|
2019-04-10 19:07:53 +00:00
|
|
|
viewport()->update();
|
|
|
|
event->accept();
|
|
|
|
}
|
|
|
|
|
2018-03-21 20:32:32 +00:00
|
|
|
void GraphView::mouseReleaseEvent(QMouseEvent *event)
|
2017-12-13 22:38:46 +00:00
|
|
|
{
|
2019-04-07 11:02:35 +00:00
|
|
|
if (scroll_mode && (event->buttons() & (Qt::LeftButton | Qt::MiddleButton)) == 0) {
|
2017-12-14 21:07:48 +00:00
|
|
|
scroll_mode = false;
|
|
|
|
setCursor(Qt::ArrowCursor);
|
|
|
|
viewport()->releaseMouse();
|
2017-12-13 22:38:46 +00:00
|
|
|
}
|
|
|
|
}
|
2018-05-21 17:33:46 +00:00
|
|
|
|
|
|
|
void GraphView::wheelEvent(QWheelEvent *event)
|
|
|
|
{
|
2019-04-07 11:02:35 +00:00
|
|
|
if (scroll_mode) {
|
|
|
|
// With some mice it's easy to hit sideway scroll button while holding middle mouse.
|
|
|
|
// That would result in unwanted scrolling while panning.
|
|
|
|
return;
|
|
|
|
}
|
2019-03-23 08:21:06 +00:00
|
|
|
QPoint delta = -event->angleDelta();
|
|
|
|
delta /= current_scale;
|
2019-04-21 16:30:57 +00:00
|
|
|
addViewOffset(delta);
|
2019-02-16 17:17:11 +00:00
|
|
|
viewport()->update();
|
2018-05-21 17:33:46 +00:00
|
|
|
event->accept();
|
|
|
|
}
|