C
Creator02 січня 2017 р. 19:13

Коллизия объектов

Коллизия, qt

Пишу игру bomberman. Проблема вот в чем. У меня есть игрок, он ходит, устанавливает бомбы, но нужно создать механизм обработки этих взрывов, что бы объекты, которые попадают под данный радиус взрыва уничтожались. Как обнаружить коллизию при создании взрыва, происходит ошибка если проверять коллизию в конструкторе класса? Когда это лучше всего сделать проверку на коллизию. У меня есть класс взрыв и класс стена, стена должна уничтожится, если она попадает под взрыв.

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

Вам це подобається? Поділіться в соціальних мережах!

8
C
  • 02 січня 2017 р. 20:40

Поправлю вопрос. Как проверить на коллизию, сразу при добавлении объекта на сцену.

    Evgenii Legotckoi
    • 03 січня 2017 р. 07:02

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

    #ifndef MAINWINDOW_H
    #define MAINWINDOW_H
    
    #include <QMainWindow>
    #include <customscene.h>
    
    namespace Ui {
    class MainWindow;
    }
    
    class MainWindow : public QMainWindow
    {
        Q_OBJECT
    
    public:
        explicit MainWindow(QWidget *parent = 0);
        ~MainWindow();
    
    private:
        Ui::MainWindow *ui;
        CustomScene *scene;
    };
    
    #endif // MAINWINDOW_H
    
    #include "mainwindow.h"
    #include "ui_mainwindow.h"
    #include "player.h"
    
    MainWindow::MainWindow(QWidget *parent) :
        QMainWindow(parent),
        ui(new Ui::MainWindow)
    {
        ui->setupUi(this);
        resize(600, 600);
        scene = new CustomScene(this);
        ui->graphicsView->setSceneRect(0, 0, 500, 500);
        ui->graphicsView->setScene(scene);
    }
    
    MainWindow::~MainWindow()
    {
        delete ui;
    }
    

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

    #ifndef CUSTOMSCENE_H
    #define CUSTOMSCENE_H
    
    #include <QGraphicsScene>
    #include <QTimer>
    #include "player.h"
    
    class CustomScene : public QGraphicsScene
    {
        Q_OBJECT
    public:
        explicit CustomScene(QObject *parent = nullptr);
    
    protected:
        virtual void mousePressEvent(QGraphicsSceneMouseEvent *event) override;
    
    signals:
    
    public slots:
    
    private:
        QTimer m_createWallsTimer;
        Player *player;             // Игрок
    };
    
    #endif // CUSTOMSCENE_H
    
    #include "customscene.h"
    
    #include <QGraphicsSceneMouseEvent>
    #include "bomb.h"
    #include "wall.h"
    #include <QTimer>
    #include <QDebug>
    
    static int randomBetween(int low, int high)
    {
        return (qrand() % ((high + 1) - low) + low);
    }
    
    CustomScene::CustomScene(QObject *parent) : QGraphicsScene(parent)
    {
        // Добавляем на игровую сцену игрока
        player = new Player();
        player->setPos(250, 250);
        player->setZValue(1000);
        addItem(player);
    
        // Периодическое создание стен на игровом поле
        connect(&m_createWallsTimer, &QTimer::timeout, [this]()
        {
            Wall* wall = new Wall();
            wall->setPos(randomBetween(15, 485), randomBetween(15, 485));
            addItem(wall);
        });
        m_createWallsTimer.start(1000);
    }
    
    void CustomScene::mousePressEvent(QGraphicsSceneMouseEvent *event)
    {
        switch (event->button()) {
        case Qt::RightButton:
        {
            // При клике по сцене правой кнопкой мыши, создаём бомбу
            Bomb* bomb = new Bomb();
            bomb->setPos(event->scenePos());
            addItem(bomb);
            break;
        }
        case Qt::LeftButton:
        {
            // При клике левой кнопкой мыши, указываем, куда идти игроку
            player->slotTarget(event->scenePos());
            break;
        }
        default:
            break;
        }
    }
    

    Все игровые объекты должны быть наследованы от одного абстрактного класса, в котором будут объявлены все возможные типы графических объектов. Это упростит определение типа объекта на графической сцене при обработке взрывов и перемещении игрока. Это возможно благодаря тому, что базовый класс абстрактного игрового объекта является QGraphicsItem , а он в свою очередь имеет метод type(), при перегрузке которого мы и будем получать тип объекта

    Реализация подобного подхода возможна благодаря парадигме полиморфизма в C++ .

    #ifndef GAMEOBJECT_H
    #define GAMEOBJECT_H
    
    #include <QGraphicsObject>
    
    class GameObject : public QGraphicsObject
    {
        Q_OBJECT
    public:
        explicit GameObject(QGraphicsItem *parent = nullptr);
    
        // Базовый абстрактный ирговой объект, в котором перечислены все
        // типы возможных игровых объектов
        // От данного класса должны быть наследованы все остальные
        enum Type
        {
            Player = UserType + 1,
            Wall,
            Bomb
        };
    
    signals:
    
    public slots:
    };
    
    #endif // GAMEOBJECT_H
    
    #include "gameobject.h"
    
    GameObject::GameObject(QGraphicsItem *parent) : QGraphicsObject(parent)
    {
    
    }
    

    Я не стал добавлять объект взрыва, но логика его срабатывания будет похожа на логику работы бомбочки. Разве что он просто должен произойти, но не влиять никак на другие объекты. Поскольку именно бомбочка будет уничтожать другие объекты в области поражения. А вот уничтожать объекты она будет в деструкторе класса. Это довольно удобно. Бомбочка уничтожилась и прихватила за собой другие объекты.

    #ifndef BOMB_H
    #define BOMB_H
    
    #include "gameobject.h"
    #include <QTimer>
    
    class Bomb : public GameObject
    {
        Q_OBJECT
    public:
        explicit Bomb(QGraphicsItem *parent = 0);
        virtual ~Bomb();
    
        virtual int type() const override;
    
    protected:
        virtual QRectF boundingRect() const override;
        virtual void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget) override;
    
    signals:
    
    public slots:
    
    private:
        QTimer m_liveTimer;
    };
    
    #endif // BOMB_H
    
    #include "bomb.h"
    #include <QGraphicsScene>
    #include <QPainter>
    
    Bomb::Bomb(QGraphicsItem *parent) : GameObject(parent)
    {
        // Бомба взрывает не сразу, а через 3 секунды после своего появления
        connect(&m_liveTimer, &QTimer::timeout, this, &QGraphicsObject::deleteLater);
        m_liveTimer.start(3000);
    }
    
    Bomb::~Bomb()
    {
        // Взрыв выполняем в деструкторе бомбы
    
        // Формируем зону взрыва. Можно создать хону взрыва любой формы
        // с помощью QPainterPath
        // Но в данном случае формируем зону взрыва в виде круга
        QPainterPath explosionZone;
        explosionZone.addEllipse(this->pos(), 100, 100);
    
        // Находим все объекты, которые попали в зону взрыва
        QList<QGraphicsItem *> foundedItems = scene()->items(explosionZone);
        for (auto item : foundedItems)
        {
            switch (item->type()) {
            case Type::Wall:
                // Но уничтожаем только стены
                scene()->removeItem(item);
                delete item;
                break;
            default:
                break;
            }
        }
    }
    
    int Bomb::type() const
    {
        // Возвращаем тип объекта "Бомба"
        return Type::Bomb;
    }
    
    QRectF Bomb::boundingRect() const
    {
        return QRectF(-10, -10, 20, 20);
    }
    
    void Bomb::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
    {
        Q_UNUSED(option)
        Q_UNUSED(widget)
    
        painter->drawEllipse(-10, -10, 20 ,20);
    }
    

    Стена - это будет просто зелёный квадрат. Она уничтожается бомбочкой в её деструкторе и не позволяет игроку пройти сквозь себя (это уже обрабатывается в коде передвижения игрока)

    #ifndef WALL_H
    #define WALL_H
    
    #include "gameobject.h"
    
    class Wall : public GameObject
    {
        Q_OBJECT
    public:
        explicit Wall(QGraphicsItem *parent = 0);
    
        virtual int type() const override;
    
    protected:
        virtual QRectF boundingRect() const override;
        virtual void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget) override;
    
    signals:
    
    public slots:
    };
    
    #endif // WALL_H
    
    #include "wall.h"
    #include <QPainter>
    
    Wall::Wall(QGraphicsItem *parent) : GameObject(parent)
    {
    
    }
    
    int Wall::type() const
    {
        // Возвращаем тип объекта "Стена"
        return Type::Wall;
    }
    
    QRectF Wall::boundingRect() const
    {
        return QRectF(-10, -10, 20, 20);
    }
    
    void Wall::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
    {
        Q_UNUSED(option)
        Q_UNUSED(widget)
    
        painter->setBrush(Qt::green);
        painter->drawRect(-10, -10, 20, 20);
    }
    
    Наиболее сложный код у игрока, там происходит обработка движения в сторону клика , обработка коллизий со стеной и прохождение сквозь бомбочку.
    #ifndef PLAYER_H
    #define PLAYER_H
    
    #include "gameobject.h"
    #include <QTimer>
    
    #define GO true
    #define STOP false
    
    class Player : public GameObject
    {
        Q_OBJECT
    public:
        explicit Player(QGraphicsItem *parent = 0);
    
        virtual int type() const override;
    
    protected:
        virtual QRectF boundingRect() const override;
        virtual void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget) override;
    
    signals:
    
    public slots:
        void slotTarget(QPointF point);
    
    private slots:
        void slotGameTimer();
    
    private:
        QTimer gameTimer;
        QPointF m_target;
        bool m_state;
    };
    
    #endif // PLAYER_H
    
    #include "player.h"
    #include <QPainter>
    #include <QGraphicsScene>
    
    static const double Pi = 3.14159265358979323846264338327950288419717;
    static double TwoPi = 2.0 * Pi;
    
    static qreal normalizeAngle(qreal angle)
    {
        while (angle < 0)
            angle += TwoPi;
        while (angle > TwoPi)
            angle -= TwoPi;
        return angle;
    }
    
    Player::Player(QGraphicsItem *parent)
        : GameObject(parent),
          m_state(STOP)
    {
        connect(&gameTimer, &QTimer::timeout, this, &Player::slotGameTimer);
        gameTimer.start(25);
    }
    
    int Player::type() const
    {
        // Возвращаем тип объекта "Игрок"
        return Type::Player;
    }
    
    QRectF Player::boundingRect() const
    {
        return QRectF(-12,-15,24,30);
    }
    
    void Player::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
    {
        Q_UNUSED(option)
        Q_UNUSED(widget)
    
        QPolygon polygon;
        polygon << QPoint(0,-15) << QPoint(12,15) << QPoint(-12,15);
        painter->setBrush(Qt::red);
        painter->drawPolygon(polygon);
    }
    
    void Player::slotGameTimer()
    {
        if(m_state){
            QLineF lineToTarget(QPoint(0,0), mapFromScene(m_target));
            // Продвигаемся к цели
            if(lineToTarget.length() > 2)
            {
                 setPos(mapToParent(0, -1));
                 // Находим все объекты, на которые наткнулся игрок
                 QList <QGraphicsItem *> items = scene()->collidingItems(this);
                 if (!items.isEmpty())
                 {
                     for (auto item : items)
                     {
                         // Но не можем пройти только сквозь стены
                         // Стеныы отталкивают игрока назад
                         switch (item->type()) {
                         case Type::Wall:
                             setX(this->x() + 1);
                             break;
                         default:
                             // Тогда как через остальные объекты проходим насквозь
                             break;
                         }
                     }
                 }
            }
            else
            {
                m_state = STOP;
            }
            /* Проверка выхода за границы поля
             * Если объект выходит за заданные границы, то возвращаем его назад
             * */
            if(this->x() - 30 < 0)
            {
                this->setX(30);         /// слева
                m_state = STOP;         // Останавливаемся
            }
            if(this->x() + 30 > 520)
            {
                this->setX(520 - 30);   /// справа
                m_state = STOP;         // Останавливаемся
            }
            if(this->y() - 30 < 0)
            {
                this->setY(30);         /// сверху
                m_state = STOP;         // Останавливаемся
            }
            if(this->y() + 30 > 520)
            {
                this->setY(520 - 30);   /// снизу
                m_state = STOP;         // Останавливаемся
            }
        }
    }
    
    void Player::slotTarget(QPointF point)
    {
        // Определяем расстояние до цели
        m_target = point;
        QLineF lineToTarget(QPointF(0, 0), mapFromScene(m_target));
        // Угол поворота в направлении к цели
        qreal angleToTarget = ::acos(lineToTarget.dx() / lineToTarget.length());
        if (lineToTarget.dy() < 0)
            angleToTarget = TwoPi - angleToTarget;
        angleToTarget = normalizeAngle((Pi - angleToTarget) + Pi / 2);
    
        // Поворачиваем героя к цели
        if (angleToTarget >= 0 && angleToTarget < Pi)
        {
            // Rotate left
            setRotation(rotation() - angleToTarget * 180 /Pi);
        }
        else if (angleToTarget <= TwoPi && angleToTarget > Pi)
        {
            // Rotate right
            setRotation(rotation() + (angleToTarget - TwoPi )* (-180) /Pi);
        }
    
        m_state = GO; // Разрешаем идти
    }
    
    Программный код примера во вложении.
      Evgenii Legotckoi
      • 03 січня 2017 р. 07:15

      Что касается проверки коллизий сразу при добавлении на графическую сцену, то тут несколько не хватает условий. Для каких именно других графических объектов нужно делать проверку.

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

      Так что вопрос спорный, а нужно ли это выполнять сразу при добавлении?

        C
        • 03 січня 2017 р. 07:31

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

        for(int i = 1; i < sizeExplosion; i++)
            {
                ExplosionHorizontal *explosionRH = new ExplosionHorizontal(playerID, QPointF(this->pos().x() - 32 * i, this->pos().y()));
                scene()->addItem(explosionRH);
                rightExplosion = explosionRH->checkCollision();
                if(rightExplosion)
                    break;
            }
        
        И сам метод проверки:
        bool ExplosionHorizontal::checkCollision()
        {
            foreach(QGraphicsItem *item, scene()->collidingItems(this))
            {
                if(item->data(0).toInt() == 1)
                {
                    Triangle *it = qgraphicsitem_cast <Triangle *> (item);
                    if(((int (it->x())) / 32) * 32 == this->x() && ((int (it->y())) / 32) * 32 == this->y())
                        it->kill();
                }
                if(item->data(1).toInt() == 2)
                {
                    qDebug() << "StoneNoDestroy";
                    this->deleteLater();
                    return true;
                }
                if(item->data(1).toInt() == 3)
                {
                    qDebug() << "StoneDestroy";
                    StoneDestroy *it = qgraphicsitem_cast <StoneDestroy *> (item);
                    it->destroy();
                    this->deleteLater();
                    return true;
                }
                if(item->data(1).toInt() == 4)
                {
                    qDebug() << "BoomCollision";
                    Bomb *it = qgraphicsitem_cast <Bomb *> (item);
                    it->bombCollision();
                    this->deleteLater();
                    return true;
                }
            }
            return false;
        }
        
        Так же добавил скрин интерфейса. И как срабатывает взрыв
          C
          • 03 січня 2017 р. 07:33

          К этой же теме, игрок движется с разной скоростью при разных размерах виджета. Я создал новую тему по этому вопросу

            Evgenii Legotckoi
            • 03 січня 2017 р. 07:51
            • Відповідь була позначена як рішення.

            Да нормальное решение.

            Я уничтожаю объекты при уничтожении бомбочки, а вы при создании взрыва

            Мне в вашем коде другое не понравилось, вы используете много кастов объектов, всё-таки, если кастов много, значит в коде что-то не так. Лучше перейдите на типы через метод type() , который я предложил. Это будет правильнее и более красиво.

            И ещё, используйте for вместо foreach . Дело в том, что foreach был сделан для того, чтобы проходить по контейнерам, поскольку в for не было поддержки контейнеров, но в современном стандарте C++ эта поддержка уже внедрена. Лично я сейчас уже отошёл от использования foreach в своих проектах.

            А как вы добавляли скрины? Я что-то не вижу их в сообщении. Возможно логика обработки сообщений на форуме выпилила скрины. (возможно нужно подкорректировать код на сайте, поэтому опишите подробнее этот момент со скринами)

              C
              • 03 січня 2017 р. 07:57

              Понял, учту. Можно пример условий для for . Именно для контейнеров.

              Добавил в Прикрепления Но видимо скрипт на сайте не захотел принять это изображение. Сейчас попробую ещё раз

                Evgenii Legotckoi
                • 03 січня 2017 р. 10:38

                Конечно, вот пример для for :

                for(QGraphicsItem* item : scene()->collidingItems(this))
                {
                    // Сделать что-нибудь с item    
                }
                
                То есть тоже самое, что и вы делали с foreach , только будет for и синтаксис немного другой.
                В Qt foreach был разработан, когда у обычного for не было синтаксиса для перебора всяких vector или QList. А сейчас это уже поддерживается самим C++.

                  Коментарі

                  Only authorized users can post comments.
                  Please, Log in or Sign up
                  AD

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

                  • Результат:50бали,
                  • Рейтинг балів-4
                  m
                  • molni99
                  • 26 жовтня 2024 р. 01:37

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

                  • Результат:80бали,
                  • Рейтинг балів4
                  m
                  • molni99
                  • 26 жовтня 2024 р. 01:29

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

                  • Результат:20бали,
                  • Рейтинг балів-10
                  Останні коментарі
                  ИМ
                  Игорь Максимов22 листопада 2024 р. 11:51
                  Django - Підручник 017. Налаштуйте сторінку входу до Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
                  Evgenii Legotckoi
                  Evgenii Legotckoi31 жовтня 2024 р. 14:37
                  Django - Урок 064. Як написати розширення для Python Markdown Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
                  A
                  ALO1ZE19 жовтня 2024 р. 08:19
                  Читалка файлів fb3 на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                  ИМ
                  Игорь Максимов05 жовтня 2024 р. 07:51
                  Django - Урок 064. Як написати розширення для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                  d
                  dblas505 липня 2024 р. 11:02
                  QML - Урок 016. База даних SQLite та робота з нею в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                  Тепер обговоріть на форумі
                  Evgenii Legotckoi
                  Evgenii Legotckoi24 червня 2024 р. 15:11
                  добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
                  t
                  tonypeachey115 листопада 2024 р. 06:04
                  google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
                  NSProject
                  NSProject04 червня 2022 р. 03:49
                  Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…
                  9
                  9Anonim25 жовтня 2024 р. 09:10
                  Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

                  Слідкуйте за нами в соціальних мережах