АД
Aug. 30, 2016, 5:23 p.m.

Знакомство с QPainter

paintEngine, QPainter, Qt

Доброго времени суток!
Заранее прошу простить мою некомпетентность в программировании.
Изучая урок, столкнулся с проблемой:при попытке нарисовать внутри QTabWidget линию с помощью QPainter, сталкиваюсь с тем, что ничего не отрисовывается, а в лог выскакивает сообщение
QWidget::paintEngine: Should no longer be called
QPainter::begin: Paint device returned engine == 0, type: 1
QPainter::setPen: Painter not active

Прошу помочь с решением данной задачи.
Самым великолепным будет проект с решением, т.к. мне намного проще разобраться с готовым решением, чем пытаться что-то синтезировать без наставника =)
.h

#ifndef WIDGET_H
#define WIDGET_H
 
#include <QMainWindow>
#include <QWidget>
#include <QPainter>
 
namespace Ui {
class Widget;
}
 
class Widget : public QMainWindow
{
    Q_OBJECT
 
public:
    explicit Widget(QWidget *parent = 0);
    ~Widget();
 
protected:
    /* Определяем виртуальный метод родительского класса
     * для отрисовки содержимого виджета
     * */
    void paintEvent(QPaintEvent *event);
 
private slots:
    void on_pushButton_clicked();
 
private:
    Ui::Widget *ui;
};
 
#endif // WIDGET_H

.cpp

#include "widget.h"
#include "ui_widget.h"
 
Widget::Widget(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);
}
 
Widget::~Widget()
{
    delete ui;
}
 
void Widget::paintEvent(QPaintEvent *event)
{
    Q_UNUSED(event);
    QPainter painter(ui->tab); // Создаём объект отрисовщика
 
    painter.setPen(QPen(Qt::black, 1, Qt::SolidLine, Qt::FlatCap));
    painter.drawEllipse(100, 50, 150, 150);
}
 
void Widget::on_pushButton_clicked()
{
    repaint();
}

1
The question is asked by the articleQt/C++ - Lesson 020. QPainter – Introduction to drawing

Do you like it? Share on social networks!

5
Evgenii Legotckoi
  • Aug. 30, 2016, 6:09 p.m.

Добрый день!

Здесь ошибка в том, что метод paintEvent должен вызываться в классе объекта,
который будет выступать в качестве ui->tab. В уроке передаётся указатель this, который указывается на сам виджет окна. То есть необходимо сделать класс, наследованный от QWidget. В нём уже переопределить метод paintEvent(QPaintEvent * event), и установить экземпляр данного класса в качестве вкладки в QTabWidget. В коде это будет выглядеть следующим образом:

mainwindow.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H
 
#include <QMainWindow>
#include "formtab.h"
 
namespace Ui {
class MainWindow;
}
 
class MainWindow : public QMainWindow
{
    Q_OBJECT
 
public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();
 
private slots:
    void on_pushButton_clicked();
 
private:
    Ui::MainWindow *ui;
    FormTab *m_formTab;
};
 
#endif // MAINWINDOW_H

mainwindow.cpp

#include "mainwindow.h"
#include "ui_mainwindow.h"
 
MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    m_formTab = new FormTab(this);
    ui->tabWidget->addTab(m_formTab, "Paint Tab");
}
 
MainWindow::~MainWindow()
{
    delete ui;
}
 
 
void MainWindow::on_pushButton_clicked()
{
    m_formTab->repaint();
}

formtab.h

#ifndef FORMTAB_H
#define FORMTAB_H
 
#include <QWidget>
#include <QPaintEvent>
 
namespace Ui {
class FormTab;
}
 
class FormTab : public QWidget
{
    Q_OBJECT
 
public:
    explicit FormTab(QWidget *parent = 0);
    ~FormTab();
 
protected:
    void paintEvent(QPaintEvent *event);
 
private:
    Ui::FormTab *ui;
};
 
#endif // FORMTAB_H

formtab.cpp

#include "formtab.h"
#include "ui_formtab.h"
 
#include <QPainter>
 
FormTab::FormTab(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::FormTab)
{
    ui->setupUi(this);
}
 
FormTab::~FormTab()
{
    delete ui;
}
 
void FormTab::paintEvent(QPaintEvent *event)
{
    Q_UNUSED(event);
    QPainter painter(this);
 
    painter.setPen(QPen(Qt::black, 1, Qt::SolidLine, Qt::FlatCap));
    painter.drawEllipse(100, 50, 150, 150);
}

А вообще класс QPainter имеет некоторые ограничения при использовании. Обычно экземпляр QPainter без проблем работает с тем объектом, который является для него родителем, то есть если в экземпляр QPainter передан указатель на тот объект, в котором объявлен QPainter, то всё работает, как правило. В противном случае вылазит сообщение об ошибке, как у Вас.

paintercheck.zip

    Благодарю за развёрнутый ответ.
    А как возможно передать массив в QPainter для отрисовки для данной задачи?
    Предполагается формировать массив в MainWindow.

      В FormTab написать метод, который принимает массив. Передать этот массив в экземпляр FormTab, допустим в слоте обработчике нажатия QPushButton, который у Вас уже написан. Ну и присвоить этот массив той переменной, которая будет содержать данный массив в FormTab. А потом уже вызвать repaint(). Логику отрисовки с этим массивом организовать в переопределённом методе paintEvent(QPaintEvent *event).

      Но я бы порекомендовал использовать сразу контейнер, например, QVector, если требуется передать ряд значений для отрисовки ломанной линии, к примеру. Ну и использовать QPainterPath для отрисовки этой линии. Вот в этом уроке в методе paint() сделана отрисовка и через линии и через QPainterPath.

        АД
        • Aug. 30, 2016, 8:03 p.m.
        • The answer was marked as a solution.

        Евгений, спасибо большое за консультацию.
        Получилось сделать как хотелось.
        Скидываю проект (вдруг кому-то пригодиться).

        painter.rar

          СГ
          • Sept. 24, 2020, 12:49 p.m.

          Здравствуйте! В ваших проектах (paintercheck.zip, painter.rar) отсутствует "ui_formtab.h". Без него не компилируется проект. Я компилирую в Qt Creator 4.3.1 по ubuntu 16.04. Как бы получить этот файл?

            Comments

            Only authorized users can post comments.
            Please, Log in or Sign up
            • Last comments
            • AK
              April 1, 2025, 11:41 a.m.
              Добрый день. В данный момент работаю над проектом, где необходимо выводить звук из программы в определенное аудиоустройство (колонки, наушники, виртуальный кабель и т.д). Пишу на Qt5.12.12 поско…
            • Evgenii Legotckoi
              March 9, 2025, 9:02 p.m.
              К сожалению, я этого подсказать не могу, поскольку у меня нет необходимости в обходе блокировок и т.д. Поэтому я и не задавался решением этой проблемы. Ну выглядит так, что вам действитель…
            • VP
              March 9, 2025, 4:14 p.m.
              Здравствуйте! Я устанавливал Qt6 из исходников а также Qt Creator по отдельности. Все компоненты, связанные с разработкой для Android, установлены. Кроме одного... Когда пытаюсь скомпилиров…
            • ИМ
              Nov. 22, 2024, 9:51 p.m.
              Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
            • Evgenii Legotckoi
              Oct. 31, 2024, 11:37 p.m.
              Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup