我目前能够将我的图像加载到图形场景中,然后再次加载到 QGraphicsViewer 中。
我可以通过检测 QEvent::Wheel 然后调用 graphicsViews 的 scale() 函数来实现缩放功能。
但是,我似乎无法弄清楚如何使平移功能正常工作。我基本上想检测鼠标何时单击图像,然后随着鼠标向左、向右、向上或向下移动图像。
截至目前,我基本上有一个 MouseFilter 类来检测事件,并根据事件类型做不同的事情。我将该侦听器附加到 QGraphicsView 对象
我目前能够将我的图像加载到图形场景中,然后再次加载到 QGraphicsViewer 中。
我可以通过检测 QEvent::Wheel 然后调用 graphicsViews 的 scale() 函数来实现缩放功能。
但是,我似乎无法弄清楚如何使平移功能正常工作。我基本上想检测鼠标何时单击图像,然后随着鼠标向左、向右、向上或向下移动图像。
截至目前,我基本上有一个 MouseFilter 类来检测事件,并根据事件类型做不同的事情。我将该侦听器附加到 QGraphicsView 对象
如果有人想知道如何自己做,这实际上很简单。这是我的应用程序中的代码:
class ImageView : public QGraphicsView
{
public:
ImageView(QWidget *parent);
~ImageView();
private:
virtual void mouseMoveEvent(QMouseEvent *event);
virtual void mousePressEvent(QMouseEvent *event);
virtual void mouseReleaseEvent(QMouseEvent *event);
bool _pan;
int _panStartX, _panStartY;
};
您需要存储拖动的开始位置,例如像这样(我使用右键):
void ImageView::mousePressEvent(QMouseEvent *event)
{
if (event->button() == Qt::RightButton)
{
_pan = true;
_panStartX = event->x();
_panStartY = event->y();
setCursor(Qt::ClosedHandCursor);
event->accept();
return;
}
event->ignore();
}
此外,您需要在释放按钮后清除标志并恢复光标:
void ImageView::mouseReleaseEvent(QMouseEvent *event)
{
if (event->button() == Qt::RightButton)
{
_pan = false;
setCursor(Qt::ArrowCursor);
event->accept();
return;
}
event->ignore();
}
要实际管理拖动,您需要覆盖鼠标移动事件。QGraphicsView 继承了一个 QAbstractScrollArea,它的滚动条很容易访问。您还需要更新平移位置:
void ImageView::mouseMoveEvent(QMouseEvent *event)
{
if (_pan)
{
horizontalScrollBar()->setValue(horizontalScrollBar()->value() - (event->x() - _panStartX));
verticalScrollBar()->setValue(verticalScrollBar()->value() - (event->y() - _panStartY));
_panStartX = event->x();
_panStartY = event->y();
event->accept();
return;
}
event->ignore();
}
QGraphicsView 具有内置的鼠标平移支持。设置正确的DragMode,它将处理其余部分。您确实需要启用滚动条才能使其工作。
neuviemeporte 解决方案需要子类化 QGraphicsView。
无需使用 eventFilter 子类化视图即可获得另一个有效的拖动实现。如果您不需要自定义 QGraphicsView 的其他行为,此技术将为您节省一些工作。
假设您的 GUI 逻辑由 QMainWindow 子类维护,并且 QGraphicsView 和 QGraphicsScene 被声明为该子类的私有成员。您必须按如下方式实现 eventFilter 函数:
bool MyMainWindow::eventFilter(QObject *obj, QEvent *event)
{
if (obj == scene && event->type() == Event::GraphicsSceneMouseMove)
{
QGraphicsSceneMouseEvent *m = static_cast<QGraphicsSceneMouseEvent*>(event);
if (m->buttons() & Qt::MiddleButton)
{
QPointF delta = m->lastScreenPos() - m->screenPos();
int newX = view->horizontalScrollBar()->value() + delta.x();
int newY = view->verticalScrollBar()->value() + delta.y();
view->horizontalScrollBar()->setValue(newX);
view->verticalScrollBar()->setValue(newY);
return true;
}
}
return QMainWindow::eventFilter(obj, event);
}
要从 QGraphicsScene 过滤事件,您必须安装 MyMainWindow 作为场景的 eventFilter。也许您可以在设置 GUI 的相同功能中执行此操作。
void MyMainWindow::setupGUI()
{
// along with other GUI stuff...
scene->installEventFilter(this);
}
您可以扩展这个想法,用拖动“手”替换光标,如前所示。