程序化QGraphicsView滚动无法正确更新

时间:2012-07-17 18:14:39

标签: c++ qt user-interface graphics scroll

我有一个派生自QGraphicsView的自定义类,它实现了一个槽调用scrollHorizo​​ntal(int dx),代码内部只是

void CustomView::scrollHorizontal(int dx){
    scrollContentsBy(dx, 0);
}

我的问题是,滚动像这样有效但不能正确更新场景,而是重复在视图边缘找到的任何像素,而不是重新调用项目的paint()方法。

我之后尝试过调用update(),但没有任何反应。我尝试通过拖动和更新工作正常启用滚动!但是我需要以编程方式完成它,因为我有隐藏的滚动条,如horizontalScrollBar()->setValue()之类的东西不会滚动视图。

我也尝试过:

scrollContentsBy(dx, 0);
this->scene()->invalidate(sceneRect());
this->update();

更新

QPointF center = mapToScene(viewport()->rect().center()); 
centerOn(center.x() - dx, center.y()); 
update(); 

正在运行,但现在我的顶视图滚动速度比我的底部视图慢,这是一个新问题。它们与signalslot s相关联,在底部视图中我scrollContentsBy(int dx, int dy)覆盖了emit horizontalScroll(dx);这是由顶视图中的上述slot捕获的。

为什么卷轴以不同的速率发生的任何想法?它可能与滚动条有关,它是底视图的一部分,有效地使它成为一个“更小”的窗口?

更新2:

不同的滚动速率似乎源于一些舍入发生,使用mapToScene(viewport()->rect().center());给我一个基于整数的“中心”,滚动时滚动越慢,此错误加起来越多,滚动速度越快减少总误差。

我有办法解决这个问题吗?我没有看到任何方法来获得浮点中心点。

更新3:

所以我已经解决了这个问题,原来需要mapToScene(我在网上其他地方找到的代码)。

我通过存储视口的FP计算中心的QPointF来修复此问题,现在滚动两个视图时的错误量是不明显的。

最后一个问题是,当您向右滚动任意数量时,视图不再排列,然后调整窗口大小,然后再次滚动。我认为这与计算中心点和定心发生时的逻辑顺序有关。

现在我在QGraphicsScene::ResizeEvent()和其他地方使用以下代码段,根据需要更新中心

QRectF viewPort(viewport()->rect());
QPointF rectCenter((viewPort.x() + viewPort.x() + viewPort.width())/2.0, (viewPort.y() + viewPort.y() + viewPort.height())/2.0);

viewCenter = rectCenter;

和我的horizontalScroll(int dx) slot

void CustomView::horizontalScroll(int dx)
{
    viewCenter.setX(viewCenter.x() - dx);
    centerOn(viewCenter.x(), viewCenter.y());
    update();
}

如何在重新调整窗口大小来解决两个视图的对齐问题时如何解决问题?如果需要进一步澄清,请问我,如果需要,我可以尝试给骷髅提供我所指的内容。

更新4:

粗糙的代码骨架

Class HeaderView:

class HeaderView View : public QGraphicsView
{
    Q_OBJECT
public:
    HeaderView(QWidget * parent = 0);
    HeaderView(QGraphicsScene * scene, QWidget * parent = 0);

private:
    QPointF viewCenter;

protected:
    void resizeEvent ( QResizeEvent * event );

public slots:
    void horizontalScroll(int);
    void addModel(qreal, qreal, const QString&);

};

HeaderView.cpp

void HeaderView::resizeEvent(QResizeEvent *event)
{
    QGraphicsView::resizeEvent(event);
    QRectF viewPort(viewport()->rect());
    QPointF rectCenter((viewPort.x() + viewPort.x() + viewPort.width())/2.0, (viewPort.y() + viewPort.y() + viewPort.height())/2.0);

    viewCenter = rectCenter;
}

void HeaderView::horizontalScroll(int dx)
{
    viewCenter.setX(viewCenter.x() - dx);
    centerOn(viewCenter.x(), viewCenter.y());
    update();
}

类EventView:

class EventView : public QGraphicsView
{
Q_OBJECT
public:
    EventView(QWidget * parent = 0);
    EventView(QGraphicsScene * scene, QWidget * parent = 0);
    QRectF visibleRect();

protected:
     void scrollContentsBy ( int dx, int dy );

signals:
    void horizontalScroll(int);

};

EventView.cpp

void EventView::scrollContentsBy(int dx, int dy)
{
    QGraphicsView::scrollContentsBy(dx, dy);

    if(dx != 0){
        emit horizontalScroll(dx);
    }
}

MainWindow类中的Somwhere:

connect(eventView, SIGNAL(horizontalScroll(int)), headerView, SLOT(horizontalScroll(int));

3 个答案:

答案 0 :(得分:10)

我在Qt 4.6.3 - 4.7.2中使用QGraphicsView并且必须证明您可以以下列方式使用相应的QScrollBar:< / p>

//graphics view initialization
QGraphicsView *graphicsView = new QGraphicsView(parent);
QGraphicsScene *scene = new QGraphicsScene(0,0,widthOfScene,heightOfScene,parent);
graphicsView->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
graphicsView->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
graphicsView->setScene(scene);

//in another method
QScrollBar* yPos=graphicsView->verticalScrollBar();
yPos->setValue((int) newValue);

它们是否隐藏无关紧要。只要您的图形场景大于图形视图,它们仍会响应setValue(int)

QGraphicsView也会响应ensureVisible,这会将滚动条移动到适当的位置。

答案 1 :(得分:3)

您不应按照此处的说明致电scrollContentsByhttp://qt-project.org/doc/qt-4.8/qabstractscrollarea.html#scrollContentsBy

我不知道你是否仍然可以调用隐藏的滚动条来滚动它。如果没有,translate是一个选项。

答案 2 :(得分:2)

您是否尝试使用滚动条?隐藏它们并不会使它们不存在,the documentation says你应该使用QScrollBar::setValue滚动到给定的位置。

另一个选择是将QGraphicsView::centerOn(QPointF)与当前中心点结合使用 - 正如您所尝试的那样 - 但直接计算方法中的中心点(不预先计算和存储中心点),使用QGraphicsView::mapToScene(int,int)

void CustomView::horizontalScroll(int dx)
{
    QPointF viewCenter = mapToScene(width() / 2, height() / 2);
    viewCenter += QPointF(dx, 0); // Why did you subtract instead of add dx?
    centerOn(viewCenter); // BTW, you don't need to do .x(), .y()
    // You can remove update(); as this is already called in centerOn().
}

请注意,如果你有,正如你所说,&#34; scrollContentsBy(int dx, int dy)覆盖了emit horizontalScroll(dx)&#34;,你还必须调用超类方法,以便视图可以滚动本身:

void CustomView::scrollContentsBy(int dx, int dy)
{
    emit horizontalScrolled(dx); // (You should call it different than the slot!)
    QGraphicsView::scrollContentsBy(dx, dy); // <-- This is what I mean!
}