移动QWidget时QPainter黑色跟踪

时间:2015-04-15 11:58:53

标签: c++ qt qwidget qpainter qmouseevent

我创建了一个带有2个小部件的小型测试应用程序,一个在另一个内部。 我重新实现了鼠标移动,按下并释放内部窗口小部件的事件,以便能够通过拖放操作将其移动到更大的父窗口内。

然而,当我移动它时,从顶部和左侧出现黑色迹线。这是它的外观:

enter image description here

这是我的代码:

的main.cpp

#include <QApplication>

#include "widget.h"

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);

    Widget w;
    w.show();

    return a.exec();
}

widget.h

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QPaintEvent>

namespace Ui {
class Widget;
}

class Widget : public QWidget
{
    Q_OBJECT

public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();

protected:
    void paintEvent(QPaintEvent *e);
};

#endif // WIDGET_H

widget.cpp

#include "widget.h"
#include "innerwidget.h"

#include <QPainter>

Widget::Widget(QWidget *parent) :
    QWidget(parent)
{
    new InnerWidget(this);

    resize(400, 200);
}

Widget::~Widget()
{
}

void Widget::paintEvent(QPaintEvent* e)
{
    QPainter p(this);
    p.setBrush(Qt::lightGray);
    p.drawRect(e->rect());
}

innerwidget.h

#ifndef INNERWIDGET_H
#define INNERWIDGET_H

#include <QWidget>
#include <QPaintEvent>

class InnerWidget : public QWidget
{
    Q_OBJECT
public:
    explicit InnerWidget(QWidget *parent = 0);
    ~InnerWidget();

protected:
    void mousePressEvent(QMouseEvent *e);
    void mouseReleaseEvent(QMouseEvent *e);
    void mouseMoveEvent(QMouseEvent *e);
    void paintEvent(QPaintEvent *e);

private:
    bool m_leftButtonPressed;
    QPoint m_mousePosOnBar;
};

#endif // INNERWIDGET_H

innerwidget.cpp

#include "innerwidget.h"

#include <QPainter>
#include <QPaintEvent>
#include <QStyleOption>

InnerWidget::InnerWidget(QWidget *parent) : QWidget(parent)
{
    setGeometry(10, 10, 100, 100);
    setStyleSheet("background-color: red");
}

InnerWidget::~InnerWidget()
{

}

void InnerWidget::mousePressEvent(QMouseEvent* e)
{
    if(e->button() == Qt::LeftButton)
    {
        m_mousePosOnBar = e->pos();
        m_leftButtonPressed = true;
    }

    e->accept();
}

void InnerWidget::mouseReleaseEvent(QMouseEvent* e)
{
    if(e->button() == Qt::LeftButton)
    {
        m_leftButtonPressed = false;
    }

    e->accept();
}

void InnerWidget::mouseMoveEvent(QMouseEvent* e)
{
    if(m_leftButtonPressed)
    {
        move(e->pos().x() - m_mousePosOnBar.x() + geometry().x(),
             e->pos().y() - m_mousePosOnBar.y() + geometry().y());
    }


    e->accept();
}

void InnerWidget::paintEvent(QPaintEvent* e)
{
    Q_UNUSED(e)

    QPainter p(this);

    QStyleOption opt;
    opt.init(this);
    style()->drawPrimitive(QStyle::PE_Widget, &opt, &p, this);
}

修改: 当我调用Widget::repaint时,跟踪消失但后来我必须在InnerWidget上安装一个事件过滤器并在每次移动时重新绘制。我想要一个更清洁的解决方案,而不必使用事件过滤器...

谁能告诉我实际发生了什么?

1 个答案:

答案 0 :(得分:0)

QWidget::update()中呼叫Widget::paintEvent解决了问题:

void Widget::paintEvent(QPaintEvent* e)
{
    QPainter p(this);
    p.setBrush(Qt::lightGray);
    p.drawRect(e->rect());

    update();
}