P
PeknicJune 30, 2016, 4:17 a.m.

Присвоение ID объекту и обращение к нему

id, item, qlist, Qt

Имеются объекты на сцене. Необходимо определять расстояние между объектами. Решил объекты пронумеровать.А как сказать qt что мне нужна координата n-го объекта, а потом другого объекта.
We recommend hosting TIMEWEB
We recommend hosting TIMEWEB
Stable hosting, on which the social network EVILEG is located. For projects on Django we recommend VDS hosting.

Do you like it? Share on social networks!

11
Evgenii Legotckoi
  • June 30, 2016, 5:22 a.m.

Рассмотрим такой пример. Имеются прямоугольники, при создании которых происходит инкрементирование счётчика ID и каждый раз на сцену добавляется объект с уникальным ID. Для этого нужно будет создать отдельный класс, который реализует данную логику. Далее, добавим на графическую сцену 20 таких объектов и проведём между двумя из них прямую линию. Допустим между 5-ым и 10-ым. То есть ID у них будут равны 5 и 10, соответственно.

Заголовочный файл прямоугольника

#ifndef RECTANGLE_H
#define RECTANGLE_H
 
#include <QObject>
#include <QGraphicsItem>
#include <QPainter>
#include <QStyleOptionGraphicsItem>
 
class Rectangle : public QGraphicsItem
{
 
public:
    explicit Rectangle();
    static int countId;         // Статическая переменная, счетчик номеров прямоугольников
    int getId() const;          // Функция для возврата локального номера прямоугольника
 
protected:
    void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget = nullptr) override;
    QRectF boundingRect() const override;
 
private:
    int m_id;   // ID прямоугольника
};
 
#endif // RECTANGLE_H

Исходный код прямоугольника

#include "rectangle.h"
 
Rectangle::Rectangle() : QGraphicsItem()
{
    countId++;
    m_id = countId;
}
 
int Rectangle::getId() const
{
    return m_id;
}
 
void Rectangle::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
{
    painter->setPen(Qt::black);
    painter->setBrush(Qt::green);
    painter->drawRect(-10, -10, 20, 20);
    Q_UNUSED(option);
    Q_UNUSED(widget);
}
 
QRectF Rectangle::boundingRect() const
{
    return QRectF (-10, -10, 20, 20);
}
 
int Rectangle::countId = 0;  // Тот самый статический счётчик

Проработаем всё это дело в главном окне приложения, в которому у нас будет создана графическая сцена.

#ifndef MAINWINDOW_H
#define MAINWINDOW_H
 
#include <QMainWindow>
#include <QGraphicsScene>
 
namespace Ui {
class MainWindow;
}
 
class MainWindow : public QMainWindow
{
    Q_OBJECT
 
public:
    explicit MainWindow(QWidget *parent = 0);
    ~MainWindow();
 
private:
    Ui::MainWindow *ui;
    QGraphicsScene *scene;
};
 
#endif // MAINWINDOW_H

Ну и проработаем всю логику, которая была написана в начале

#include "mainwindow.h"
#include "ui_mainwindow.h"
#include <QDebug>
#include "rectangle.h"
 
/* Функция для получения рандомного числа
 * в диапазоне от минимального до максимального
 * */
static int randomBetween(int low, int high)
{
    return (qrand() % ((high + 1) - low) + low);
}
 
MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);
    this->resize(600, 600);                 // Изменим размер главного окна приложения
    scene = new QGraphicsScene(this);       // Инициализируем графическую сцену
    ui->graphicsView->setScene(scene);      // Установим сцену в Graphics View
    scene->setSceneRect(0, 0, 500, 500);    // Установим размеры сцены
 
    // Создадим 20 прямоугольников со случайными координатами
    for (auto i = 0; i < 20; i++) {
        Rectangle *rect = new Rectangle();
        rect->setX(randomBetween(15, 485));
        rect->setY(randomBetween(15, 485));
        scene->addItem(rect);
    }
 
    // Временные указатели на прямоугольники, координаты которых нас будут интересовать
    Rectangle *item_1;
    Rectangle *item_2;
 
    // Пройдёмся по всем объектам на грфической сцене, чтобы найти нужные
    foreach (QGraphicsItem *item, scene->items()) {
        // Необходимо скастовать QGraphicsItem в объект класса прямоугольника,
        // чтобы получить доступ к его ID
        Rectangle *rect = static_cast<Rectangle *>(item);
        // Если ID равен 5 или 10, то сохраним указатель на него в соответсвующие указатели item_1 или item_2
        switch (rect->getId()) {
        case 5:
            item_1 = rect;
            break;
        case 10:
            item_2 = rect;
            break;
        default:
            break;
        }
    }
 
    if (item_1 != nullptr && item_2 != nullptr) {
        // создадим линию между этими двумя объектами
        QLineF lineBetweenItems;
        lineBetweenItems.setP1(item_1->scenePos()); // используем координаты прямоугольников на сцене,
        lineBetweenItems.setP2(item_2->scenePos()); // чтобы задать положение концов линии
 
        // добавляем линию на сцену
        scene->addLine(lineBetweenItems);
        qDebug() << lineBetweenItems.length(); // Выводим расстояние между объектами
    }
}
 
MainWindow::~MainWindow()
{
    delete ui;
}

    P
    • July 6, 2016, 5:12 a.m.

    Спасибо. Все работает. У меня еще вопросик. А как сделать так чтоб объект одного класса взаимодействовал только с определенным объектом другого класса ( связать объекты между собой, как на картинке) классы унаследованны от одного родителя. И как можно присвоить и отобразить имена объектов? Пробовал через Qlabel,не получилось.

      Evgenii Legotckoi
      • July 6, 2016, 10:31 a.m.
      • The answer was marked as a solution.

      связать объекты между собой, как на картинке

      Ну если я правильно понял, то проблем в этом никаких быть не должно. В том примере, который я привел, будем считать, что класс Rectangle – это у нас базовый класс, от которого будут наследованы остальные классы. Таким образом у нас остаётся глобальная система id и для этого базового класса и для всех его наследников. Поэтому можно сделать допустим два класса наследника, и переопределить только вид их отрисовки. Например, у нас это будут красные и синие прямоугольники.
      Синий прямоугольник

      #ifndef RECTANGLEBLUE_H
      #define RECTANGLEBLUE_H
       
      #include "rectangle.h"
       
      class RectangleBlue : public Rectangle
      {
      public:
          RectangleBlue();
       
      protected:
          void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget = nullptr) override;
          QRectF boundingRect() const override;
      };
       
      #endif // RECTANGLEBLUE_H
      #include "rectangleblue.h"
       
      RectangleBlue::RectangleBlue() : Rectangle()
      {
       
      }
       
      void RectangleBlue::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
      {
          painter->setPen(Qt::black);
          painter->setBrush(Qt::blue);
          painter->drawRect(-10, -10, 20, 20);
          painter->drawText(-10, 20, QString::number(getId()));
          Q_UNUSED(option);
          Q_UNUSED(widget);
      }
       
      QRectF RectangleBlue::boundingRect() const
      {
          return QRectF (-10, -10, 20, 30);
      }

      Красный прямоугольник

      #ifndef RECTANGLERED_H
      #define RECTANGLERED_H
       
      #include "rectangle.h"
       
      class RectangleRed : public Rectangle
      {
      public:
          RectangleRed();
       
      protected:
          void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget = nullptr) override;
          QRectF boundingRect() const override;
      };
       
      #endif // RECTANGLERED_H
      #include "rectanglered.h"
       
      RectangleRed::RectangleRed() : Rectangle()
      {
       
      }
       
      void RectangleRed::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
      {
          painter->setPen(Qt::black);
          painter->setBrush(Qt::red);
          painter->drawRect(-10, -10, 20, 20);
          painter->drawText(-10, 20, QString::number(getId()));
          Q_UNUSED(option);
          Q_UNUSED(widget);
      }
       
      QRectF RectangleRed::boundingRect() const
      {
          return QRectF (-10, -10, 20, 30);
      }

      Ну а далее я приведу немного модифицированный код MainWindow, где вместо зеленных прямоугольников класса Rectangle, будут создавать его наследники, классов RectangleBlue и RectangleRed.
      Обращаю внимание на то, что во время перебора всех элементов, благодаря полиморфизму, их можно кастовать к их общему базовому классу Rectangle, у которого, также как и его наследников доступен метод getId().

      #include "mainwindow.h"
      #include "ui_mainwindow.h"
      #include <QDebug>
      #include "rectangle.h"
      #include "rectanglered.h"
      #include "rectangleblue.h"
       
      /* Функция для получения рандомного числа
       * в диапазоне от минимального до максимального
       * */
      static int randomBetween(int low, int high)
      {
          return (qrand() % ((high + 1) - low) + low);
      }
       
      MainWindow::MainWindow(QWidget *parent) :
          QMainWindow(parent),
          ui(new Ui::MainWindow)
      {
          ui->setupUi(this);
          this->resize(600, 600);                 // Изменим размер главного окна приложения
          scene = new QGraphicsScene(this);       // Инициализируем графическую сцену
          ui->graphicsView->setScene(scene);      // Установим сцену в Graphics View
          scene->setSceneRect(0, 0, 500, 500);    // Установим размеры сцены
       
          // Создадим 10 красных прямоугольников со случайными координатами
          for (auto i = 0; i < 10; i++) {
              RectangleRed *rect = new RectangleRed();
              rect->setX(randomBetween(15, 485));
              rect->setY(randomBetween(15, 485));
              scene->addItem(rect);
          }
       
          // Создадим 20 синих прямоугольников со случайными координатами
          for (auto i = 0; i < 20; i++) {
              RectangleBlue *rect = new RectangleBlue();
              rect->setX(randomBetween(15, 485));
              rect->setY(randomBetween(15, 485));
              scene->addItem(rect);
          }
       
          // Временные указатели на прямоугольники, координаты которых нас будут интересовать
          Rectangle *item_1;
          Rectangle *item_2;
       
          // Пройдёмся по всем объектам на грфической сцене, чтобы найти нужные
          foreach (QGraphicsItem *item, scene->items()) {
              // Необходимо скастовать QGraphicsItem в объект класса прямоугольника,
              // чтобы получить доступ к его ID
              Rectangle *rect = static_cast<Rectangle *>(item);
              // Если ID равен 5 или 10, то сохраним указатель на него в соответсвующие указатели item_1 или item_2
              switch (rect->getId()) {
              case 5:
                  item_1 = rect;
                  break;
              case 15:
                  item_2 = rect;
                  break;
              default:
                  break;
              }
          }
       
          if (item_1 != nullptr && item_2 != nullptr) {
              // создадим линию между этими двумя объектами
              QLineF lineBetweenItems;
              lineBetweenItems.setP1(item_1->scenePos()); // используем координаты прямоугольников на сцене,
              lineBetweenItems.setP2(item_2->scenePos()); // чтобы задать положение концов линии
       
              // добавляем линию на сцену
              scene->addLine(lineBetweenItems);
              qDebug() << lineBetweenItems.length(); // Выводим расстояние между объектами
          }
      }
       
      MainWindow::~MainWindow()
      {
          delete ui;
      }

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

        P
        • July 8, 2016, 8:44 a.m.

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

        Еще возникла такая проблема. При включении слота двойного клика мыши, перестаем отображаться на сцене planeAEW, унаследованный от plane,причем другие объекты классов, унаследованные от plane отображаются.

        plane.h

        planeaew.h

          P
          • July 8, 2016, 8:47 a.m.
          #include "plane.h"
          #include "mainwindow.h"
           
          int q=0;
           
          Plane::Plane(QObject *parent) : QObject(parent), QGraphicsItem()
          {
              ResID++;
              IDplane = ResID;
              n = ResID;
          }
           
          Plane::~Plane()
          {
           
          }
           
          QRectF Plane::boundingRect() const
          {
              return QRectF(-30,-20,30,40);   //ограничиваем область в которой лежит треугольник
          }
           
          void Plane::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *MainWindow)
          {
              //рисуем самолет
              painter->setPen(QPen(Qt::black,2));
              painter->drawLine(0,0,-20,0);    //тело
              painter->drawLine(0,0,-4,3);     //нос
              painter->drawLine(0,0,-4,-3);    //нос
              painter->drawLine(-8,10,-8,-10); //крыло
              painter->drawLine(-11,10,-11,-10);//крыло
           
              //рисуем хвост
              QPolygon polygon;
              polygon << QPoint(-20,0) << QPoint(-24,3) << QPoint(-24,-3);
              painter->setBrush(Qt::black);
              painter->drawPolygon(polygon);
           
              //установим имя
              painter->drawText(-12, 20, QString::number(getID()));
           
              Q_UNUSED(option);
              Q_UNUSED(MainWindow);
           
          }
           
          //движение объекта
          void Plane::slotGameTimer()
          {
              if(q<=20)
              {
              setPos(mapToParent(2,-1));
              }
              else
              {
              setPos(mapToParent(2,0));
              }
              q++;
           
          //проверка выхода за границы поля
          //если объект выходит за заданные границы, то возвращаем его назад
              if(this->x() + 2 < 0)
              {
                  this->setX(0);//слева
              }
              if(this->x() - 2 > 600)
              {
                  emit signalStopTimer();
                  //this->deleteLater();
              }
              if(this->y() + 2 < -600)
              {
                  this->setY(-600);//сверху
              }
              if(this->y() - 2 > 0)
              {
                  this->setY(0);  //снизу
              }
              emit signalNumber();
          }
           
          void Plane::mouseMoveEvent(QGraphicsSceneMouseEvent *event)
          {
              //устанавливаем объект в новую позицию
              this->setPos(mapToScene(event->pos()));
          }
           
          void Plane::mousePressEvent(QGraphicsSceneMouseEvent *event)
          {
              //при нажатии на элемент меняем курсор
              this->setCursor(QCursor(Qt::ClosedHandCursor));
              //emit signalNumber();
              Q_UNUSED(event);
          }
           
          void Plane::mouseReleaseEvent(QGraphicsSceneMouseEvent *event)
          {
              //после отпускания меняем курсор на стрелку
              this->setCursor(QCursor(Qt::ArrowCursor));
              Q_UNUSED(event);
          }
           
          //void Plane::mouseDoubleClickEvent(QGraphicsSceneMouseEvent *event)
          //{
          //    this->N = this->IDplane;
          //    Q_UNUSED(event);
          //}
           
          int Plane::getID() const
          {
              return IDplane;
          }
           
          int Plane::ResID = 0;
          int Plane::n=0;
          //int Plane::N;

           

            Evgenii Legotckoi
            • July 8, 2016, 11:56 a.m.

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

            Я Вам показал всю механику определения по ID способа “связывания” объектов. Всё остальное делается по аналогии. Для определения, к какому типу (то бишь классу) относится тот или иной объект перечитайте урок связанный с полиморфизмом
            Там имеется enum с перечислениями типов объектов. Такое перечисление можно сделать в базовом классе. А во всех наследованных от него объектах можно переопределить виртуальный метод type(), который относится к классу QGraphicsItem, от которого должен быть наследован уже ваш базовый класс, от которого наследуются все остальные графические объекты.

            То есть в заголовочном файле базового класса

            enum Types {
                RedBlock = UserType + 1,
                BlueBlock,
                GreenBlock,
                BlackBlock
            }

            В заголовочном файле классов наследников
            int type() const override

            и в файле исходных кодов классов наследников (Допустим в синем прямоугольнике)

            int type()
            {
             return BlueBlock;
            }

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

              Evgenii Legotckoi
              • July 8, 2016, 12:10 p.m.

              При включении слота двойного клика мыши, перестаем отображаться на сцене planeAEW

              А вот тут я понятия не имею, на первый взгляд, весь код в полном порядке. Ну в том плане, что не заметно чего-то криминального из-за чего бы объект должен был быть нерабочим.

                f
                • May 1, 2018, 7:27 a.m.

                А как нарисовать дугу, между объектами?

                painter.drawArc
                видимо не подходит, тк QPainter, не связан с  QGraphicsObject / QGraphicsItem.
                https://upload.wikimedia.org/wikipedia/commons/thumb/9/9d/DFAexample.svg/250px-DFAexample.svg.png
                хотелось бы уметь рисовать что нибудь похожее на это
                  f
                  • May 1, 2018, 7:46 a.m.
                  QGraphicsEllipseItem *arc = scene->addEllipse(QRectF(_source->scenePos(),
                              QSize(_source->scenePos().rx() - _destination->scenePos().rx(),
                                    _source->scenePos().ry() - _destination->scenePos().ry()) ));
                          arc->setSpanAngle(90 * 16);

                  Нашел setSpanAngle, но он вместо дуги рисует "кусок пирога"
                    f
                    • May 1, 2018, 10:01 a.m.
                    • (edited)

                    И сразу вопрос как нарисовать стрелку с помощью линий?


                    void Widget::printArrow(QLineF lineF)
                    {
                        QGraphicsLineItem *line = new GraphicsLineItem();
                        QLineF ll1; // /\ острая часть стрелы
                        QLineF ll2; // /\ острая часть стрелы
                    
                        qreal x, y, angl, len = 20;
                        x = lineF.p2().x();
                        y = lineF.p2().y();
                        angl = lineF.angle();
                    
                        ll1.setLine(x, y,
                                    x + len * cos((angl - 30) * 3.14159265 / 180),
                                    y + len * cos((angl - 30) * 3.14159265 / 180));
                        ll2.setLine(x, y,
                                    x + len * cos((angl + 30) * 3.14159265 / 180),
                                    y + len * cos((angl + 30) * 3.14159265 / 180));
                    
                        QGraphicsLineItem *l1 = new QGraphicsLineItem(ll1);
                        QGraphicsLineItem *l2 = new QGraphicsLineItem(ll2);
                        line->setLine(lineF);
                        scene->addItem(line);
                        scene->addItem(l1);
                        scene->addItem(l2);
                    }

                    Почему то не работает. У двух линий козырька почему то всегда угол 315 или 135 градусов
                      f
                      • May 1, 2018, 10:16 a.m.

                      косяк со стрелкой у себя нашел. Надо было синус у игрека ставить а не косинус.

                      void Widget::printArrow(QLineF lineF)
                      {
                          GraphicsLineItem *line = new GraphicsLineItem();
                          QLineF ll1; // /\ острая часть стрелы
                          QLineF ll2; // /\ острая часть стрелы
                      
                          qreal x, y, angl, len = 20;
                          x = lineF.p2().x();
                          y = lineF.p2().y();
                          angl = lineF.angle();
                          ll1.setLine(x, y,
                                      x - len * cos((angl - 10) * 3.14159265 / 180),
                                      y + len * sin((angl - 10) * 3.14159265 / 180));
                          ll2.setLine(x, y,
                                      x - len * cos((angl + 10) * 3.14159265 / 180),
                                      y + len * sin((angl + 10) * 3.14159265 / 180));
                          QGraphicsLineItem *l1 = new QGraphicsLineItem(ll1);
                          QGraphicsLineItem *l2 = new QGraphicsLineItem(ll2);
                          line->setLine(lineF);
                          scene->addItem(line);
                          scene->addItem(l1);
                          scene->addItem(l2);
                      }

                        Comments

                        Only authorized users can post comments.
                        Please, Log in or Sign up
                        e
                        • ehot
                        • March 31, 2024, 9:29 p.m.

                        C++ - Тест 003. Условия и циклы

                        • Result:78points,
                        • Rating points2
                        B

                        C++ - Test 002. Constants

                        • Result:16points,
                        • Rating points-10
                        B

                        C++ - Test 001. The first program and data types

                        • Result:46points,
                        • Rating points-6
                        Last comments
                        k
                        kmssrFeb. 9, 2024, 2:43 a.m.
                        Qt Linux - Lesson 001. Autorun Qt application under Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                        Qt WinAPI - Lesson 007. Working with ICMP Ping in Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                        EVA
                        EVADec. 25, 2023, 6:30 p.m.
                        Boost - static linking in CMake project under Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
                        J
                        JonnyJoDec. 25, 2023, 4:38 p.m.
                        Boost - static linking in CMake project under Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
                        G
                        GvozdikDec. 19, 2023, 5:01 a.m.
                        Qt/C++ - Lesson 056. Connecting the Boost library in Qt for MinGW and MSVC compilers Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
                        Now discuss on the forum
                        G
                        GarApril 22, 2024, 12:46 p.m.
                        Clipboard Как скопировать окно целиком в clipb?
                        DA
                        Dr Gangil AcademicsApril 20, 2024, 2:45 p.m.
                        Unlock Your Aesthetic Potential: Explore MSC in Facial Aesthetics and Cosmetology in India Embark on a transformative journey with an msc in facial aesthetics and cosmetology in india . Delve into the intricate world of beauty and rejuvenation, guided by expert faculty and …
                        a
                        a_vlasovApril 14, 2024, 1:41 p.m.
                        Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Евгений, добрый день! Такой вопрос. Верно ли следующее утверждение: Любое Android-приложение, написанное на Java/Kotlin чисто теоретически (пусть и с большими трудностями) можно написать и на C+…
                        Павел Дорофеев
                        Павел ДорофеевApril 14, 2024, 9:35 a.m.
                        QTableWidget с 2 заголовками Вот тут есть кастомный QTableView с многорядностью проект поддерживается, обращайтесь
                        f
                        fastrexApril 4, 2024, 11:47 a.m.
                        Вернуть старое поведение QComboBox, не менять индекс при resetModel Добрый день! У нас много проектов в которых используется QComboBox, в версии 5.5.1, когда модель испускает сигнал resetModel, currentIndex не менялся. В версии 5.15 при resetModel происходит try…

                        Follow us in social networks