2021-09-19 15:44:31 +03:00
|
|
|
//
|
|
|
|
// Created by Иван Ильин on 19.09.2021.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "PlayerController.h"
|
2021-10-09 13:41:12 +03:00
|
|
|
#include "engine/utils/Log.h"
|
|
|
|
#include "engine/animation/AFunction.h"
|
|
|
|
#include "engine/animation/AWait.h"
|
|
|
|
#include "engine/animation/ATranslate.h"
|
|
|
|
#include "engine/animation/ATranslateToPoint.h"
|
|
|
|
#include "engine/animation/Timeline.h"
|
2021-10-16 20:22:55 +03:00
|
|
|
#include "ShooterConsts.h"
|
2021-09-19 15:44:31 +03:00
|
|
|
|
|
|
|
PlayerController::PlayerController(std::shared_ptr<Player> player,
|
|
|
|
std::shared_ptr<Keyboard> keyboard,
|
2021-10-02 20:36:07 +03:00
|
|
|
std::shared_ptr<Mouse> mouse) : _player(player), _keyboard(keyboard), _mouse(mouse) {
|
2021-10-16 20:22:55 +03:00
|
|
|
_slowMoSound.setBuffer(*ResourceManager::loadSoundBuffer(ShooterConsts::SLOW_MO_SOUND));
|
|
|
|
_unSlowMoSound.setBuffer(*ResourceManager::loadSoundBuffer(ShooterConsts::UN_SLOW_MO_SOUND));
|
2021-09-19 15:44:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
void PlayerController::update() {
|
|
|
|
// friction
|
|
|
|
if(_player->inCollision())
|
|
|
|
_player->setVelocity(_player->velocity()*(1.0 - Time::deltaTime() * 2));
|
|
|
|
|
|
|
|
if(_isInSlowMo) {
|
|
|
|
if(_player->ability() > 0)
|
|
|
|
_player->setAbility(_player->ability() - Time::deltaTime());
|
|
|
|
else {
|
|
|
|
_player->setAbility(0);
|
|
|
|
_isInSlowMo = false;
|
2021-10-16 20:22:55 +03:00
|
|
|
_player->setVelocity(_player->velocity() * ShooterConsts::SLOW_MO_COEFFICIENT);
|
|
|
|
_player->setAcceleration(_player->acceleration() * ShooterConsts::SLOW_MO_COEFFICIENT * ShooterConsts::SLOW_MO_COEFFICIENT);
|
2021-09-19 15:44:31 +03:00
|
|
|
_slowMoSound.stop();
|
|
|
|
_unSlowMoSound.play();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-16 20:22:55 +03:00
|
|
|
double coeff = _isInSlowMo ? 1.0 / ShooterConsts::SLOW_MO_COEFFICIENT : 1.0;
|
2021-09-19 15:44:31 +03:00
|
|
|
|
|
|
|
bool inRunning_old = _inRunning;
|
|
|
|
_inRunning = ( Keyboard::isKeyPressed(sf::Keyboard::A) ||
|
|
|
|
Keyboard::isKeyPressed(sf::Keyboard::D) ||
|
|
|
|
Keyboard::isKeyPressed(sf::Keyboard::W) ||
|
|
|
|
Keyboard::isKeyPressed(sf::Keyboard::S));
|
|
|
|
|
2021-10-17 10:21:10 +03:00
|
|
|
std::shared_ptr<Object> camera = _player->attached(ObjectNameTag("camera"));
|
2021-10-16 16:14:51 +03:00
|
|
|
if(camera != nullptr && _inRunning) {
|
2021-10-17 10:21:10 +03:00
|
|
|
if (!Timeline::isInAnimList(AnimationListTag("camera_hor_oscil"))) {
|
|
|
|
Timeline::animate(AnimationListTag("camera_hor_oscil"), new ATranslate(camera, -camera->left() / 6, 0.3,Animation::LoopOut::None, Animation::InterpolationType::cos));
|
|
|
|
Timeline::animate(AnimationListTag("camera_hor_oscil"), new AWait(0));
|
|
|
|
Timeline::animate(AnimationListTag("camera_hor_oscil"), new ATranslate(camera, camera->left() / 6, 0.3, Animation::LoopOut::None, Animation::InterpolationType::cos));
|
2021-10-02 20:36:07 +03:00
|
|
|
|
2021-10-17 10:21:10 +03:00
|
|
|
Timeline::animate(AnimationListTag("camera_vert_oscil"), new ATranslate(camera, -Vec3D{0, 1, 0} / 12, 0.15, Animation::LoopOut::None, Animation::InterpolationType::cos));
|
|
|
|
Timeline::animate(AnimationListTag("camera_vert_oscil"), new AWait(0));
|
|
|
|
Timeline::animate(AnimationListTag("camera_vert_oscil"), new ATranslate(camera, Vec3D{0, 1, 0} / 12, 0.15, Animation::LoopOut::None, Animation::InterpolationType::cos));
|
|
|
|
Timeline::animate(AnimationListTag("camera_vert_oscil"), new AWait(0));
|
|
|
|
Timeline::animate(AnimationListTag("camera_vert_oscil"), new ATranslate(camera, -Vec3D{0, 1, 0} / 12, 0.15, Animation::LoopOut::None, Animation::InterpolationType::cos));
|
|
|
|
Timeline::animate(AnimationListTag("camera_vert_oscil"), new AWait(0));
|
|
|
|
Timeline::animate(AnimationListTag("camera_vert_oscil"), new ATranslate(camera, Vec3D{0, 1, 0} / 12, 0.15, Animation::LoopOut::None, Animation::InterpolationType::cos));
|
2021-10-02 20:36:07 +03:00
|
|
|
|
2021-10-17 10:21:10 +03:00
|
|
|
Timeline::animate(AnimationListTag("camera_init"), new ATranslateToPoint( camera, _player->position() + Vec3D{0, 1.8, 0}, 0.3, Animation::LoopOut::None, Animation::InterpolationType::cos));
|
2021-10-02 20:36:07 +03:00
|
|
|
}
|
2021-10-16 16:14:51 +03:00
|
|
|
} else if(camera != nullptr && inRunning_old && !_inRunning) {
|
2021-10-17 10:21:10 +03:00
|
|
|
Timeline::deleteAnimationList(AnimationListTag("camera_hor_oscil"));
|
|
|
|
Timeline::deleteAnimationList(AnimationListTag("camera_vert_oscil"));
|
|
|
|
Timeline::deleteAnimationList(AnimationListTag("camera_init"));
|
|
|
|
Timeline::animate(AnimationListTag("camera_init"), new ATranslateToPoint( camera, _player->position() + Vec3D{0, 1.8, 0}, 0.15, Animation::LoopOut::None, Animation::InterpolationType::cos));
|
2021-09-19 15:44:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Left and right
|
2021-10-02 20:36:07 +03:00
|
|
|
|
2021-09-19 15:44:31 +03:00
|
|
|
if (Keyboard::isKeyPressed(sf::Keyboard::A)) {
|
2021-10-16 20:22:55 +03:00
|
|
|
_player->translate(_player->left() * Time::deltaTime() * ShooterConsts::WALK_SPEED * coeff);
|
2021-09-19 15:44:31 +03:00
|
|
|
if(_player->inCollision())
|
2021-10-12 17:12:47 +03:00
|
|
|
_player->setVelocity(Vec3D{0,0,0});
|
2021-09-19 15:44:31 +03:00
|
|
|
}
|
|
|
|
if (Keyboard::isKeyPressed(sf::Keyboard::D)) {
|
2021-10-16 20:22:55 +03:00
|
|
|
_player->translate(-_player->left() * Time::deltaTime() * ShooterConsts::WALK_SPEED * coeff);
|
2021-09-19 15:44:31 +03:00
|
|
|
if(_player->inCollision())
|
2021-10-12 17:12:47 +03:00
|
|
|
_player->setVelocity(Vec3D{0,0,0});
|
2021-09-19 15:44:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Forward and backward
|
|
|
|
if (Keyboard::isKeyPressed(sf::Keyboard::W)) {
|
2021-10-16 20:22:55 +03:00
|
|
|
_player->translate(_player->lookAt() * Time::deltaTime() * ShooterConsts::WALK_SPEED * coeff);
|
2021-09-19 15:44:31 +03:00
|
|
|
if(_player->inCollision())
|
2021-10-12 17:12:47 +03:00
|
|
|
_player->setVelocity(Vec3D{0,0,0});
|
2021-09-19 15:44:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (Keyboard::isKeyPressed(sf::Keyboard::S)) {
|
2021-10-16 20:22:55 +03:00
|
|
|
_player->translate(-_player->lookAt() * Time::deltaTime() * ShooterConsts::WALK_SPEED * coeff);
|
2021-09-19 15:44:31 +03:00
|
|
|
|
|
|
|
if(_player->inCollision())
|
2021-10-12 17:12:47 +03:00
|
|
|
_player->setVelocity(Vec3D{0,0,0});
|
2021-09-19 15:44:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if (_player->ability() > 0 && !_isInSlowMo && Keyboard::isKeyPressed(sf::Keyboard::LShift)) {
|
|
|
|
// slow mo
|
|
|
|
_isInSlowMo = true;
|
2021-10-16 20:22:55 +03:00
|
|
|
_player->setVelocity(_player->velocity() / ShooterConsts::SLOW_MO_COEFFICIENT);
|
2021-10-17 10:25:09 +03:00
|
|
|
_player->setAcceleration(Vec3D(0, -ShooterConsts::GRAVITY / (ShooterConsts::SLOW_MO_COEFFICIENT * ShooterConsts::SLOW_MO_COEFFICIENT), 0));
|
2021-09-19 15:44:31 +03:00
|
|
|
_unSlowMoSound.stop();
|
|
|
|
_slowMoSound.play();
|
|
|
|
} else if (_isInSlowMo && !Keyboard::isKeyPressed(sf::Keyboard::LShift)) {
|
|
|
|
_isInSlowMo = false;
|
2021-10-16 20:22:55 +03:00
|
|
|
_player->setVelocity(_player->velocity() * ShooterConsts::SLOW_MO_COEFFICIENT);
|
2021-10-17 10:25:09 +03:00
|
|
|
_player->setAcceleration(Vec3D(0, -ShooterConsts::GRAVITY, 0));
|
2021-09-19 15:44:31 +03:00
|
|
|
_slowMoSound.stop();
|
|
|
|
_unSlowMoSound.play();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Keyboard::isKeyPressed(sf::Keyboard::Space) && _player->inCollision()) {
|
2021-10-09 13:41:12 +03:00
|
|
|
// if we just want to jump, we have to add particular speed
|
|
|
|
if (!_isSliding)
|
2021-10-16 20:22:55 +03:00
|
|
|
_player->addVelocity(Vec3D{ 0, std::abs(_player->collisionNormal().y()) * sqrt(2 * -_player->acceleration().y() * ShooterConsts::JUMP_HEIGHT) * coeff, 0 });
|
2021-10-09 13:41:12 +03:00
|
|
|
// if we want to slide, we have to add speed * 60/fps to make it independent on frame rate
|
|
|
|
else
|
2021-10-16 20:22:55 +03:00
|
|
|
_player->addVelocity(Vec3D{ 0, std::abs(_player->collisionNormal().y()) * sqrt(2 * -_player->acceleration().y() * ShooterConsts::JUMP_HEIGHT) * coeff * 60.0 / Time::fps(), 0 });
|
2021-10-09 13:41:12 +03:00
|
|
|
|
2021-10-16 20:22:55 +03:00
|
|
|
_player->translate(Vec3D{ 0, Time::deltaTime() * ShooterConsts::WALK_SPEED * 2 * coeff * 60.0 / Time::fps(), 0 });
|
2021-10-09 13:41:12 +03:00
|
|
|
_isSliding = true;
|
|
|
|
} else {
|
|
|
|
_isSliding = false;
|
2021-09-19 15:44:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Mouse movement
|
2021-10-12 17:12:47 +03:00
|
|
|
Vec2D displacement = _mouse->getMouseDisplacement();
|
2021-09-19 15:44:31 +03:00
|
|
|
|
2021-10-16 20:22:55 +03:00
|
|
|
_player->rotate(Vec3D{0, -displacement.x() * ShooterConsts::MOUSE_SENSITIVITY, 0});
|
|
|
|
_player->setVelocity(Matrix4x4::RotationY(-displacement.x() * ShooterConsts::MOUSE_SENSITIVITY) * _player->velocity());
|
2021-09-19 15:44:31 +03:00
|
|
|
|
2021-10-16 20:22:55 +03:00
|
|
|
double rotationLeft = displacement.y() * ShooterConsts::MOUSE_SENSITIVITY;
|
2021-09-19 15:44:31 +03:00
|
|
|
|
|
|
|
// You can only see in range [-90 : 90] grad
|
2021-10-09 13:41:12 +03:00
|
|
|
if (_player->headAngle() + rotationLeft > Consts::PI / 2)
|
|
|
|
rotationLeft = Consts::PI / 2 - _player->headAngle();
|
|
|
|
if (_player->headAngle() + rotationLeft < -Consts::PI / 2)
|
|
|
|
rotationLeft = -Consts::PI / 2 - _player->headAngle();
|
2021-09-19 15:44:31 +03:00
|
|
|
|
2021-10-02 20:36:07 +03:00
|
|
|
_player->setHeadAngle(_player->headAngle() + rotationLeft);
|
2021-10-12 17:12:47 +03:00
|
|
|
_player->rotateWeaponsRelativePoint(_player->position() + Vec3D{0, 1.8, 0}, _player->left(), rotationLeft);
|
2021-09-19 15:44:31 +03:00
|
|
|
|
2021-10-16 16:14:51 +03:00
|
|
|
if(camera != nullptr)
|
|
|
|
camera->rotateLeft(_player->headAngle() - camera->angleLeftUpLookAt().x());
|
|
|
|
|
2021-09-19 15:44:31 +03:00
|
|
|
if (_keyboard->isKeyTapped(sf::Keyboard::Right) || _keyboard->isKeyTapped(sf::Keyboard::E)) {
|
|
|
|
_player->nextWeapon();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_keyboard->isKeyTapped(sf::Keyboard::Left) || _keyboard->isKeyTapped(sf::Keyboard::Q)) {
|
|
|
|
_player->previousWeapon();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Mouse::isButtonPressed(sf::Mouse::Button::Left)) {
|
|
|
|
_player->fire();
|
|
|
|
}
|
|
|
|
|
|
|
|
if(Keyboard::isKeyPressed(sf::Keyboard::R)) {
|
|
|
|
_player->reload();
|
|
|
|
}
|
|
|
|
|
2021-10-02 20:36:07 +03:00
|
|
|
if ((_inRunning || _player->velocity().sqrAbs() > 3) && _player->inCollision() && _walkSound.getStatus() != sf::Sound::Status::Playing) {
|
2021-10-09 13:41:12 +03:00
|
|
|
int soundNum = (int)((double) rand() / RAND_MAX * 5) + 1;
|
|
|
|
_walkSound.setBuffer(*ResourceManager::loadSoundBuffer("sound/stonestep" + std::to_string(soundNum) + ".ogg"));
|
2021-10-02 20:36:07 +03:00
|
|
|
_walkSound.play();
|
2021-09-19 15:44:31 +03:00
|
|
|
}
|
|
|
|
}
|