Михаиллл
МихаилллМаусым 11, 2019, 3:01 Т.Ж.

Как в QGraphicsScene стирать старый рисунок при рисовании нового рисунка поверх старого рисунка

QGraphicsScene

Добрый день.
При рисовании в QGraphicsScene непрозрачным цветом, я рисую поверх старого рисунка. При рисовании прозрачным цветом, прозрачный цвет не виден, он остается под старым рисунком(наверно). Как при рисовании стирать старый рисунок и показывать новый?
Рисую как в статье: Qt/C++ - Урок 021. Рисование мышью в Qt

Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Ол саған ұнайды ма? Әлеуметтік желілерде бөлісіңіз!

17
Evgenii Legotckoi
  • Маусым 11, 2019, 4:22 Т.Ж.

Добрый день.

Давайте для начала определимся, что значит рисовать прозрачным цветом? Если вы рисуете каким-то цветом и он прозрачный, то это одно. А если вы стираете, или думаете что стираете картинку, то это совершенно другое. Это другой инструмент. Там нужно реализовывать логику для удаления части изображения.

    Evgenii Legotckoi
    • Маусым 11, 2019, 4:23 Т.Ж.

    И зачем вы тему добавили в PyQt5 раздел, если вопрос чисто по Qt/C++

      Михаиллл
      • Маусым 11, 2019, 4:43 Т.Ж.

      Извините, с разделом ошибся.
      Сначало я нарисовал сцен этим QPen

          myPen.setWidth(20);
          myPen.setColor(Qt::green);
          myPen.setCapStyle(Qt::RoundCap);
          myPen.setStyle(Qt::SolidLine);
      

      Потом рисую этим QPen

          myPen.setWidth(20);
          myPen.setColor(Qt::green);
          //myPen.setColor(QColor(255, 0, 0, 10));
          myPen.setCapStyle(Qt::RoundCap);
          myPen.setStyle(Qt::SolidLine);
      

      Там , где я не рисовал сплошным цветом, виден цвет QColor(255, 0, 0, 10), но там где рисовал Qt::green, там не виден QColor(255, 0, 0, 10).
      И если много раз провести на одном месте QColor(255, 0, 0, 10), то прозрачность меняется в сторону уменьшения прозрачности, видимо цвета накладываются друг на друга.
      Мне нужно удолять старый рисунок под QPen и рисовать новый. Скажите пожалуйста, как это сделать.

        Evgenii Legotckoi
        • Маусым 11, 2019, 5:13 Т.Ж.
        • (өңделген)
        • Жауап шешім ретінде белгіленді.

        Это значительно сложнее будет. Я могу вам подсказать направление.

        1. Все объекты, которые вы добавляете на графическую сцену, хорошо было бы помещать также в pool объектов. У меня есть пример подобного пула в статье о написании игры . Там пул объектов яблок. Это позволит держать список объектов изображений например.
        2. Вам нужно добавить инструмент ластик (его нужно будет написать самостоятельно), который будет рисовать линию, а после завершения рисования, нарисованная линия должна будет создать полигон пересечения с изображением и удалить часть изображения. Нужно будет линию преобразовать в QPolygon, потом взять изображение, создать из него QPolygon области изображения, и с помощью метода QPolygon QPolygon::intersected(const QPolygon &r) const найти вырезаемую часть, которую нужно будет отнять от изображения.
        3. Отнять от изображения полигон. С этим уже сложнее, нужно поискать информацию по запросу crop image polygon. Вариантов там может быть несколько, либо через shape показывать только определённую часть изображения в QGraphicsPixmapItem, нужно будет от него наследоваться и написать свой класс для работы с изображением. Либо создать новой изображение с отрезанной частью и заменить им существующее изображение.

        По поводу создания инструментов, типо карандаш, ластик, линия или ещё что-то можете посмотреть статью о векторном редакторе . Там есть пара инструментов - линии и квадраты, можете посмотреть идеологию, как это там сделано. В качестве абстрактного примера.

          Михаиллл
          • Маусым 11, 2019, 5:45 Т.Ж.
          • (өңделген)

          Не получится разбить на много объектов, так как при загрузке загружаетсяо дним объектом.
          Но 2 и 3 реализуемы, хоть и с трудом.

          А если рисовать с помощью QPainter , новый цвет пиксля будет заменять старый цвет?

            Михаиллл
            • Маусым 11, 2019, 5:51 Т.Ж.

            Или может быть можно рисовать попиксельно на QGraphicsScene? И потом эти пиксели земенять новыми?

              Evgenii Legotckoi
              • Маусым 11, 2019, 5:55 Т.Ж.

              1,2,3 - это не возможные варианты, а пошаговое решение. То есть я вам предложил всего один вариант, как это можно сделать, который совмещает всю эту работу из трёх основных составляющих частей. Ну а если вы загружаете два рисунка одним объектом... что мне до конца непонятно... То тогда вам придётся разделить на два самостоятельных объекта оба рисунка и работать с ними отдельно. В общем задачка интересная сама по себе. Но вопрос реализации довольно обширный. Думаю, что пару тысяч строк кода потянет.

              Хотя, конечно, можете попробовать и пискели заменять, но тоже вопрос, как вы будете определять откуда и куда заменять эти пиксели. Но это явно нехорошее решение, ведь вы потом захотите ещё и передвигать изображения, вот тогда то вам и выйдет боком это передвижение пикселей.

                Михаиллл
                • Маусым 11, 2019, 6:15 Т.Ж.

                Я ограничусь только рисованием пикселей, поэтому не дойду то момента, когда мне выйдет боком.
                Если рисовать пикселями, то по идее, ластик будет легко реализыем просто прозрачным цветом.
                Скажите пожалуйста как заменить эти векторы на пиксели:

                void paintScene::mousePressEvent(QGraphicsSceneMouseEvent *event)
                {
                    // При нажатии кнопки мыши отрисовываем эллипс
                    addEllipse(event->scenePos().x() -  myPen.width()/2,
                               event->scenePos().y() -  myPen.width()/2,
                               myPen.width(),
                                myPen.width(),
                               QPen(Qt::NoPen),
                               QBrush(myPen.color()));
                    // Сохраняем координаты точки нажатия
                    previousPoint = event->scenePos();
                    buttonIsPressed = true;
                }
                
                void paintScene::mouseMoveEvent(QGraphicsSceneMouseEvent *event)
                {
                    if (buttonIsPressed)
                    {
                        // Отрисовываем линии с использованием предыдущей координаты
                        addLine(previousPoint.x(),
                                previousPoint.y(),
                                event->scenePos().x(),
                                event->scenePos().y(),
                                myPen);
                        // Обновляем данные о предыдущей координате
                        previousPoint = event->scenePos();
                    }
                }
                
                  Михаиллл
                  • Маусым 11, 2019, 6:45 Т.Ж.
                  • (өңделген)

                  Попробовал сделать так , но получаю ошибку : QPainter::drawPoints: Painter not active. Скажите пожалуйста как это исправить

                  void paintScene::mousePressEvent(QGraphicsSceneMouseEvent *event)
                  {
                      myPainter.setPen(myPen);
                      myPainter.drawPoint(event->scenePos().x(),event->scenePos().y());
                  
                  }
                  
                    Михаиллл
                    • Маусым 11, 2019, 9 Т.Ж.
                    • (өңделген)

                    Сделал так, но рисунок не рисуется.

                    void paintScene::mousePressEvent(QGraphicsSceneMouseEvent *event)
                    {
                        x = event->scenePos().x();    
                        y = event->scenePos().x();
                    }
                    
                    void paintScene::paintEvent(QPaintEvent *event)
                    {
                        QPainter myPainter; 
                        myPainter.setPen(myPen);
                        myPainter.drawPoint(x,y);
                    }
                    

                    void paintScene::paintEvent(QPaintEvent *event) не срабатывает.

                      Михаиллл
                      • Маусым 11, 2019, 9:43 Т.Ж.

                      а если прописываю void paintEvent(QPaintEvent *event); к главном классе, то оно автоматом срабатывает 1 раз при загрузке приложения, 1 раз при нажатии левой кнопки мыши и больше не работает.

                        Михаиллл
                        • Маусым 11, 2019, 1:04 Т.Қ.
                        • (өңделген)

                        Попробовал для этого сделать класс наследуемый от QWidget. Но событие не вызывается при нажатии и не понятно как получать координаты
                        .h

                        #ifndef WIDGETFORPAINTER_H
                        #define WIDGETFORPAINTER_H
                        
                        #include "QWidget"
                        #include "QPainter"
                        #include "QPen"
                        #include "QDebug"
                        
                        class widgetForPainter: public QWidget
                        {
                        public:
                            widgetForPainter();
                            ~widgetForPainter() {;};
                            QPen myPen;
                            int x,y;
                        
                        
                            private slots:
                            void paintEvent(QPaintEvent *event);
                        
                        public slots:
                            void paintPoint(int x1,int y1);
                        
                        };
                        
                        #endif // WIDGETFORPAINTER_H
                        

                        .cpp

                        #include "widgetforpainter.h"
                        
                        widgetForPainter::widgetForPainter()
                        {
                            //Pen
                            myPen.setWidth(20);
                            myPen.setColor(Qt::green);
                            //myPen.setColor(QColor(255, 0, 0, 10));
                            myPen.setCapStyle(Qt::RoundCap);
                            myPen.setStyle(Qt::SolidLine);
                        }
                        
                        void widgetForPainter::paintEvent(QPaintEvent *event)
                        {
                            Q_UNUSED(event);
                            QPainter myPainter; // Создаём объект отрисовщика
                            myPainter.setPen(myPen);
                            myPainter.drawPoint(x,y);
                            myPainter.drawEllipse(x, y, x+50, x+50);
                            qDebug()<<x<<y;
                        }
                        
                        void widgetForPainter::paintPoint(int x1, int y1)
                        {
                            QPainter myPainter; // Создаём объект отрисовщика
                            myPainter.setPen(myPen);
                            myPainter.drawPoint(x1,y1);
                            myPainter.drawEllipse(x1, y1, x+50, x+50);
                            qDebug()<<x1<<y1;
                        }
                        

                        В конструкторе главной функции написал так:

                        newGraphicsView =  new graphicsView();
                            newGraphicsView->setScene(scene);    
                        
                            ui->verticalLayout->addWidget(newGraphicsView);
                        
                        
                            widgetForPainter *myWidgetForPainter = new widgetForPainter();
                            ui->verticalLayout->addWidget(myWidgetForPainter);
                        

                        Наверное в void paintScene::mousePressEvent(QGraphicsSceneMouseEvent *event) нужно добавить void widgetForPainter::paintPoint(int x1, int y1), но как это сделать?

                          Михаиллл
                          • Маусым 11, 2019, 2:16 Т.Қ.

                          В этом примере рисует пиксельно на QWidget и так же пиксельно стирает. Не могли бы вы мне помочь выделить нужный код, который можно добавить в свой проект?

                            Михаиллл
                            • Маусым 12, 2019, 2:05 Т.Ж.

                            Нашел такой класс, он нормально рисует. Можно ли его сделать не главным классом?
                            .h

                            #ifndef WIDGET_H
                            #define WIDGET_H
                            
                            #include <QWidget>
                            #include <QPainter>
                            #include <QMouseEvent>
                            #include <QEvent>
                            #include <QFileDialog>
                            #include <QImage>
                            #include <QPixmap>
                            
                            namespace Ui {
                            class Widget;
                            }
                            
                            class Widget : public QWidget
                            {
                                Q_OBJECT
                            public:
                                explicit Widget(QWidget *parent = 0);
                                ~Widget();
                                bool eventFilter(QObject *obj, QEvent *event);
                            
                            private slots:
                                void on_open_clicked();
                            
                                void on_save_clicked();
                            
                                void on_R_valueChanged(int value);
                            
                                void on_G_valueChanged(int value);
                            
                                void on_B_valueChanged(int value);
                            
                                void on_pen_dial_valueChanged(int value);
                            
                                void on_pushButton_8_clicked();
                            
                            private:
                                QPen pen;
                                QPen pen2;
                                QPixmap pixmap;
                                QPixmap pixman;
                                QPoint lastPoint;
                                int r;
                                int g;
                                int b;
                                Ui::Widget *ui;
                            };
                            
                            #endif // WIDGET_H
                            
                            

                            .cpp

                            #include "widget.h"
                            #include "ui_widget.h"
                            
                            
                            Widget::Widget(QWidget *parent) :
                                QWidget(parent),
                                ui(new Ui::Widget)
                            {
                                ui->setupUi(this);
                                pixmap = QPixmap(ui->label->width(),ui->label->height());
                                pixmap.fill();
                                pen = QPen( QBrush( Qt::black), 5.0f );
                                r = 0;
                                g = 0;
                                b = 0;
                            
                                ui->label->setPixmap(pixmap);
                            
                                ui->label->installEventFilter(this);
                                ui->red_color->installEventFilter(this);
                                ui->orange_color->installEventFilter(this);
                                ui->yellow_color->installEventFilter(this);
                                ui->green_color->installEventFilter(this);
                                ui->blue_color->installEventFilter(this);
                                ui->purple_color->installEventFilter(this);
                                ui->pushButton_8->installEventFilter(this);
                                ui->R->installEventFilter(this);
                                ui->G->installEventFilter(this);
                                ui->B->installEventFilter(this);
                                ui->setcolor->installEventFilter(this);
                                ui->pen_dial->installEventFilter(this);
                                ui->clear_pixmap->installEventFilter(this);
                            
                            
                                ui->red_color->setStyleSheet("Background-color: red;");
                                ui->orange_color->setStyleSheet("Background-color: #ffa500");
                                ui->yellow_color->setStyleSheet("Background-color: yellow");
                                ui->green_color->setStyleSheet("Background-color: green");
                                ui->blue_color->setStyleSheet("Background-color: blue");
                                ui->purple_color->setStyleSheet("Background-color: #C400AB");
                            
                                pen2 = QPen(QBrush(Qt::black), 61.0f);
                                QBrush st(QColor(Qt::black),Qt::Dense7Pattern);
                                pixman = QPixmap(ui->label_3->width(),ui->label_3->height());
                                pixman.fill();
                                QPainter p2(&pixman);
                                p2.setPen(Qt::SolidLine);
                                p2.setBrush(st);
                                p2.drawRect(0,0,ui->label_3->width(),ui->label_3->height());
                                p2.end();
                                ui->label_3->setPixmap(pixman);
                            
                                //this->setFixedSize(this->width(), this->height());
                            }
                            
                            Widget::~Widget()
                            {
                                delete ui;
                            }
                            
                            bool Widget::eventFilter(QObject *obj, QEvent *event)
                            {
                                if(obj == ui->label and event->type() == QEvent::MouseButtonPress)
                                {
                                    QMouseEvent *mous = (QMouseEvent*)event;
                                    QPainter p(&pixmap);
                                    p.setPen(pen);
                                    p.drawPoint(mous->pos());
                                    p.end();
                                    lastPoint = mous->pos();
                                    ui->label->setPixmap(pixmap);
                                }
                                if(obj == ui->label and event->type() == QEvent::MouseMove)
                                {
                                    QMouseEvent *mous = (QMouseEvent*)event;
                                    QPainter p(&pixmap);
                                    p.setPen(pen);
                                    p.drawLine(lastPoint,mous->pos());
                                    p.end();
                                    lastPoint = mous->pos();
                                    ui->label->setPixmap(pixmap);
                                }
                                if(obj == ui->red_color and event->type() == QEvent::MouseButtonPress) pen.setColor(Qt::red);
                                if(obj == ui->orange_color and event->type() == QEvent::MouseButtonPress) pen.setColor(QColor(255, 128, 0));
                                if(obj == ui->yellow_color and event->type() == QEvent::MouseButtonPress) pen.setColor(Qt::yellow);
                                if(obj == ui->green_color and event->type() == QEvent::MouseButtonPress) pen.setColor(Qt::green);
                                if(obj == ui->blue_color and event->type() == QEvent::MouseButtonPress) pen.setColor(Qt::blue);
                                if(obj == ui->purple_color and event->type() == QEvent::MouseButtonPress) pen.setColor(QColor(196, 0, 171));
                                if(obj == ui->setcolor and event->type() == QEvent::MouseButtonPress)
                                {
                                    bool ok = true;
                                    r = ui->lineEdit->text().toInt(&ok,10);
                                    g = ui->lineEdit_2->text().toInt(&ok,10);
                                    b = ui->lineEdit_3->text().toInt(&ok,10);
                                    if((r <= 255 or g <= 255 or b <= 255)) pen.setColor(QColor(r,g,b));
                                    else ui->info->text() = "Введены неверные значения";
                                }
                                if(obj == ui->pen_dial and event->type() == QEvent::MouseButtonPress and QEvent::MouseMove)
                                {
                                }
                                if(obj == ui->clear_pixmap and event->type() == QEvent::MouseButtonPress)
                                {
                                    pixmap.fill(QColor(Qt::white));
                                    ui->label->setPixmap(pixmap);
                                }
                                if(obj == ui->label_3 and event->type() == QEvent::MouseButtonPress)
                                {
                                    pen.setBrush(Qt::Dense5Pattern);
                                    ui->label->setPixmap(pixmap);
                                }
                            }
                            
                            void Widget::on_open_clicked()
                            {
                                QString file;
                                file = QFileDialog::getOpenFileName(this,tr("Открыть файл"),tr("Картинка.jpg"),tr("JPG (*.jpg)"));
                                pixmap.load(file);
                            }
                            
                            void Widget::on_save_clicked()
                            {
                                QString file;
                                file = QFileDialog::getSaveFileName(this,tr("Сохранить файл"),tr("Картинка.jpg"),tr("JPG (*.jpg"));
                                pixmap.save(file);
                            }
                            
                            void Widget::on_R_valueChanged(int value)
                            {
                                pen.setColor(QColor(value,ui->B->value(),ui->G->value()));
                                ui->Value_1->setNum(value);
                            }
                            
                            void Widget::on_G_valueChanged(int value)
                            {
                                pen.setColor(QColor(ui->R->value(),value,ui->G->value()));
                                ui->Value_2->setNum(value);
                            }
                            
                            void Widget::on_B_valueChanged(int value)
                            {
                                pen.setColor(QColor(ui->R->value(),ui->B->value(),value));
                                ui->Value_3->setNum(value);
                            }
                            
                            void Widget::on_pen_dial_valueChanged(int value)
                            {
                                pen.setWidthF(value);
                                ui->label_2->setNum(value);
                            }
                            
                            void Widget::on_pushButton_8_clicked()
                            {
                            
                            }
                            
                            
                              Михаиллл
                              • Маусым 12, 2019, 6:10 Т.Ж.
                              • (өңделген)

                              ,

                                Михаиллл
                                • Маусым 12, 2019, 8:35 Т.Ж.

                                Перенес этот класс в свой код. Теперь нужно разбираться с прозрачностью.

                                  Evgenii Legotckoi
                                  • Маусым 16, 2019, 4:26 Т.Қ.

                                  У класса QColor в конструкторе есть установка прозрачности. Это альфа канал.

                                  QColor(int r, int g, int b, int a = ...)

                                  Либо через

                                  setAlpha()

                                    Пікірлер

                                    Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
                                    Кіріңіз немесе Тіркеліңіз
                                    OI
                                    • Ora Iro
                                    • Жел. 24, 2024, 6:38 Т.Ж.

                                    C++ - Тест 001. Первая программа и типы данных

                                    • Нәтиже:40ұпай,
                                    • Бағалау ұпайлары-8
                                    AD

                                    C++ - Тест 004. Указатели, Массивы и Циклы

                                    • Нәтиже:50ұпай,
                                    • Бағалау ұпайлары-4
                                    m
                                    • molni99
                                    • Қаз. 26, 2024, 1:37 Т.Ж.

                                    C++ - Тест 004. Указатели, Массивы и Циклы

                                    • Нәтиже:80ұпай,
                                    • Бағалау ұпайлары4
                                    Соңғы пікірлер
                                    ИМ
                                    Игорь МаксимовҚар. 22, 2024, 11:51 Т.Ж.
                                    Django - Оқулық 017. Теңшелген Django кіру беті Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
                                    Evgenii Legotckoi
                                    Evgenii LegotckoiҚаз. 31, 2024, 2:37 Т.Қ.
                                    Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
                                    A
                                    ALO1ZEҚаз. 19, 2024, 8:19 Т.Ж.
                                    Qt Creator көмегімен fb3 файл оқу құралы Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                                    ИМ
                                    Игорь МаксимовҚаз. 5, 2024, 7:51 Т.Ж.
                                    Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                                    d
                                    dblas5Шілде 5, 2024, 11:02 Т.Ж.
                                    QML - Сабақ 016. SQLite деректер қоры және онымен QML Qt-та жұмыс істеу Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                                    Енді форумда талқылаңыз
                                    n
                                    nklyҚаң. 3, 2025, 2:52 Т.Ж.
                                    Нужно запретить перемещение только некоторых итемов, остальные перемещать можно. Вопрос решен. Узнать QModelIndex элемента на который мы перетаскиваем другой элемент, можно с помощью функции indexAt(event->position().toPoint()) представления QTreeViev вызываемой в переопр…
                                    AW
                                    Ayden WatkinsҚаң. 2, 2025, 12:09 Т.Ж.
                                    Why Paying for a Research Paper Can Be a Smart Choice Writing a research paper can be a daunting task, especially when faced with tight deadlines, complex topics, or a lack of resources. For many students, paying for a research paper is a practical…
                                    p
                                    pimacontrols85Жел. 31, 2024, 9:39 Т.Ж.
                                    Finding the Right Rittal Small Enclosure for Your Needs Rittal is a leading manufacturer of enclosures for industrial and IT applications. Their small enclosures offer a compact and reliable solution for a wide range of needs, from housing electronic…
                                    Donald Randolph
                                    Donald RandolphЖел. 30, 2024, 2:59 Т.Ж.
                                    Personal Injury lawyer Santa Monica As an experienced Santa Monica personal injury lawyer, Donald C. Randolph has recovered over $100 Million in verdicts and settlements for our clients. In severe injury cases, this compensation i…
                                    Nirvana Yoga School
                                    Nirvana Yoga SchoolЖел. 30, 2024, 5:13 Т.Ж.
                                    OAuth2.0 через VK, получение email Nirvana Yoga School is one of the most trusted and reputed traditional Rishikesh yoga courses , India certified by Yoga Alliance, USA. We aim to spread traditional yoga teachings so t…

                                    Бізді әлеуметтік желілерде бақылаңыз