用C++和SFML写游戏-移动咱们的Player(3)

在上一节中,咱们搭建了游戏的基本框架,用循环处理咱们的游戏世界,还掌握了一些方法去解决因机器性能不一样而引发的问题。在这一节中,咱们将会学习:c++

  1. 建立 Player
  2. 移动咱们的 Player

1、建立 Player 类

这里,将会用**矩形(sf::RectangleShape)**代替咱们以前的圆形,Player 应该有这几个属性,分别是它的形状、大小和颜色,它的方向,它的速度。框架

所以咱们建立了这么一个 Playeride

class Player : public sf::Drawable {
public:
    Player(const Player&) = delete;
    Player& operator=(const Player&) = delete;
    Player();
    
    template<typename ... Args>
    void setPosition(Args&& ... args) {
        _shape.setPosition(std::forward<Args>(args)...);
    }
    
    void update(sf::Time deltaTime);
    bool isMoving;
    int rotation;
    
    
private:
    virtual void draw(sf::RenderTarget& target, sf::RenderStates states) const override;
    sf::RectangleShape _shape;
    sf::Vector2f _velocity;
    
};
复制代码

Player 是一个可绘制的对象,所以它继承自 sf::Drawable性能

这里 delete 关键字的做用跟 Game 类中的是同样的。学习

经过查询 sf::RectangleShape 类中的 **setPosition()**方法,它有两个重载方法,分别是:ui

  1. void setPosition (float x, float y)
  2. void setPosition (const Vector2f &position)

一个有一个参数,另外一个有两个参数,所以 Player 类中的 setPosition() 使用了变长参数模版。spa

isMoving 表明 Player是否在移动,rotation 能够改变方向。code

接下来就是具体实现了。cdn

首先是 Player 的初始化,代码比较简单,这里就不作解释了。对象

Player::Player() : _shape(sf::Vector2f(100,100))
{
    _shape.setFillColor(sf::Color::Blue);
    _shape.setOrigin(50,50);
    _shape.setPosition(400, 300);
}
复制代码

而后是 update 的实现

void Player::update(sf::Time deltaTime) {
    float seconds = deltaTime.asSeconds();
    
    if (rotation != 0) {
        float angle = (rotation > 0 ? 1 : -1)*180*seconds;
        _shape.rotate(angle);
    }

    if (isMoving) {
        float angle = _shape.getRotation() / 180 * M_PI - M_PI/2;
        _velocity += sf::Vector2f(std::cos(angle), std::sin(angle))*60.f*seconds;
    }
    _shape.move(seconds * _velocity);
}
复制代码

经过 rotation 参数能够改变方向

isMoving 断定 Player 是否在移动,若是是则给它继续加速,加速度大小这里为 60.0

float angle = _shape.getRotation() / 180 * M_PI - M_PI/2;
复制代码

这段代码计算出的是当前 Player 移动的角度,减去 M_PI/2 是为了是初始方向是向上的。


2、一些事件处理

有了 Player 类,固然还须要让用户去改变它的状态,这里能够经过简单的事件处理解决。事件处理在 Game 类的 processEvents 方法中,直接看代码:

while (_window.pollEvent(event)) {
    if (event.type == sf::Event::Closed) {
        _window.close();
    } else if (event.type == sf::Event::KeyPressed) {
        if (event.key.code == sf::Keyboard::Escape) {
            _window.close();
        } else if (event.key.code == sf::Keyboard::Up) {
            _player.isMoving = true;
        } else if (event.key.code == sf::Keyboard::Left) {
            _player.rotation = -1;
        } else if (event.key.code == sf::Keyboard::Right) {
            _player.rotation = 1;
        }
    } else if (event.type == sf::Event::KeyReleased) {
        if(event.key.code == sf::Keyboard::Up)
            _player.isMoving = false;
        else if (event.key.code == sf::Keyboard::Left)
            _player.rotation = 0;
        else if (event.key.code == sf::Keyboard::Right)
            _player.rotation = 0;
    }
}
复制代码

代码简单粗暴,方向键的左右控制 Player 向左向右旋转,UP键Player 加速。


3、最终效果

虽然很简陋,可是好歹运行了起来。这里没有作边界处理,能够看到最后咱们的 Player 跑到画面外去了。

这一节中的事件处理方式只作为演示,所以代码很粗糙,下一节咱们将会学习如何将事件处理管理起来。😃

相关文章
相关标签/搜索