Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 3 additions & 1 deletion .github/workflows/clang-format.yml
Original file line number Diff line number Diff line change
@@ -1,8 +1,10 @@
name: "[PR] Clang-format"



on:
pull_request:
types: [opened, synchronize, reopened]

paths:
- 'src/**'
- '.github/workflows/clang-format.yml'
Expand Down
1 change: 0 additions & 1 deletion src/battle_game/core/input_data.cpp
Original file line number Diff line number Diff line change
@@ -1,3 +1,2 @@
#include "battle_game/core/input_data.h"

namespace battle_game {}
3 changes: 1 addition & 2 deletions src/battle_game/core/player.cpp
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
#include "battle_game/core/player.h"

#include "battle_game/core/game_core.h"

namespace battle_game {
Expand All @@ -19,4 +18,4 @@ void Player::Update() {
}
}
}
} // namespace battle_game
} // namespace battle_game
1 change: 1 addition & 0 deletions src/battle_game/core/selectable_units.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@ void GameCore::GeneratePrimaryUnitList() {
ADD_SELECTABLE_UNIT(unit::ReboundingSampleTank);
ADD_SELECTABLE_UNIT(unit::RageTank);
ADD_SELECTABLE_UNIT(unit::SmokeBombTank);
ADD_SELECTABLE_UNIT(unit::Strangetank);
ADD_SELECTABLE_UNIT(unit::Sparky);
ADD_SELECTABLE_UNIT(unit::CritTank);
ADD_SELECTABLE_UNIT(unit::Railgun);
Expand Down
165 changes: 165 additions & 0 deletions src/battle_game/core/units/strange_tank.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,165 @@
#include "battle_game/core/units/strange_tank.h"

#include "battle_game/core/bullets/bullets.h"
#include "battle_game/core/game_core.h"
#include "battle_game/graphics/graphics.h"

namespace battle_game::unit {

namespace {
uint32_t tank_body_model_index = 0xffffffffu;
uint32_t tank_turret_model_index = 0xffffffffu;
} // namespace

Strangetank::Strangetank(GameCore *game_core, uint32_t id, uint32_t player_id)
: Unit(game_core, id, player_id) {
if (!~tank_body_model_index) {
auto mgr = AssetsManager::GetInstance();
{
/* Tank Body */
tank_body_model_index = mgr->RegisterModel(
{
{{-0.8f, 0.8f}, {0.0f, 0.0f}, {1.0f, 1.0f, 1.0f, 1.0f}},
{{-0.8f, -1.0f}, {0.0f, 0.0f}, {1.0f, 1.0f, 1.0f, 1.0f}},
{{0.8f, 0.8f}, {0.0f, 0.0f}, {1.0f, 1.0f, 1.0f, 1.0f}},
{{0.8f, -1.0f}, {0.0f, 0.0f}, {1.0f, 1.0f, 1.0f, 1.0f}},
// distinguish front and back
{{0.6f, 1.0f}, {0.0f, 0.0f}, {1.0f, 1.0f, 1.0f, 1.0f}},
{{-0.6f, 1.0f}, {0.0f, 0.0f}, {1.0f, 1.0f, 1.0f, 1.0f}},
},
{0, 1, 2, 1, 2, 3, 0, 2, 5, 2, 4, 5});
}

{
/* Tank Turret */
std::vector<ObjectVertex> turret_vertices;
std::vector<uint32_t> turret_indices;
const int precision = 60;
const float inv_precision = 1.0f / float(precision);
for (int i = 0; i < precision; i++) {
auto theta = (float(i) + 0.5f) * inv_precision;
theta *= glm::pi<float>() * 2.0f;
auto sin_theta = std::sin(theta);
auto cos_theta = std::cos(theta);
turret_vertices.push_back({{sin_theta * 0.5f, cos_theta * 0.5f},
{0.0f, 0.0f},
{0.7f, 0.7f, 0.7f, 1.0f}});
turret_indices.push_back(i);
turret_indices.push_back((i + 1) % precision);
turret_indices.push_back(precision);
}
turret_vertices.push_back(
{{0.0f, 0.0f}, {0.0f, 0.0f}, {0.7f, 0.7f, 0.7f, 1.0f}});
turret_vertices.push_back(
{{-0.1f, 0.0f}, {0.0f, 0.0f}, {0.7f, 0.7f, 0.7f, 1.0f}});
turret_vertices.push_back(
{{0.1f, 0.0f}, {0.0f, 0.0f}, {0.7f, 0.7f, 0.7f, 1.0f}});
turret_vertices.push_back(
{{-0.1f, 1.2f}, {0.0f, 0.0f}, {0.7f, 0.7f, 0.7f, 1.0f}});
turret_vertices.push_back(
{{0.1f, 1.2f}, {0.0f, 0.0f}, {0.7f, 0.7f, 0.7f, 1.0f}});
turret_indices.push_back(precision + 1 + 0);
turret_indices.push_back(precision + 1 + 1);
turret_indices.push_back(precision + 1 + 2);
turret_indices.push_back(precision + 1 + 1);
turret_indices.push_back(precision + 1 + 2);
turret_indices.push_back(precision + 1 + 3);
tank_turret_model_index =
mgr->RegisterModel(turret_vertices, turret_indices);
}
}
}

void Strangetank::Render() {
battle_game::SetTransformation(position_, rotation_);
battle_game::SetTexture(0);
battle_game::SetColor(game_core_->GetPlayerColor(player_id_));
battle_game::DrawModel(tank_body_model_index);
battle_game::SetRotation(turret_rotation_);
battle_game::DrawModel(tank_turret_model_index);
}

void Strangetank::Update() {
TankMove(3.0f, glm::radians(180.0f));
TurretRotate();
Fire();
}

void Strangetank::TankMove(float move_speed, float rotate_angular_speed) {
auto player = game_core_->GetPlayer(player_id_);
if (player) {
auto &input_data = player->GetInputData();
glm::vec2 offset{0.0f};
if (input_data.key_down[GLFW_KEY_W]) {
offset.y += 1.0f;
}
if (input_data.key_down[GLFW_KEY_S]) {
offset.y -= 1.0f;
}
float speed = move_speed * GetSpeedScale();
offset *= kSecondPerTick * speed;
auto new_position =
position_ + glm::vec2{glm::rotate(glm::mat4{1.0f}, rotation_,
glm::vec3{0.0f, 0.0f, 1.0f}) *
glm::vec4{offset, 0.0f, 0.0f}};
if (!game_core_->IsBlockedByObstacles(new_position)) {
game_core_->PushEventMoveUnit(id_, new_position);
}
float rotation_offset = 0.0f;
if (input_data.key_down[GLFW_KEY_A]) {
rotation_offset += 1.0f;
}
if (input_data.key_down[GLFW_KEY_D]) {
rotation_offset -= 1.0f;
}
rotation_offset *= kSecondPerTick * rotate_angular_speed * GetSpeedScale();
game_core_->PushEventRotateUnit(id_, rotation_ + rotation_offset);
}
}

void Strangetank::TurretRotate() {
auto player = game_core_->GetPlayer(player_id_);
if (player) {
auto &input_data = player->GetInputData();
auto diff = input_data.mouse_cursor_position - position_;
if (glm::length(diff) < 1e-4) {
turret_rotation_ = rotation_;
} else {
turret_rotation_ = std::atan2(diff.y, diff.x) - glm::radians(90.0f);
}
}
}

void Strangetank::Fire() {
if (fire_count_down_) {
fire_count_down_--;
} else {
auto player = game_core_->GetPlayer(player_id_);
if (player) {
auto &input_data = player->GetInputData();
if (input_data.mouse_button_down[GLFW_MOUSE_BUTTON_LEFT]) {
auto velocity = Rotate(glm::vec2{0.0f, 20.0f}, turret_rotation_);
GenerateBullet<bullet::CannonBall>(
position_ + Rotate({0.0f, 1.2f}, turret_rotation_),
turret_rotation_, GetDamageScale(), velocity);
fire_count_down_ = kTickPerSecond; // Fire interval 1 second.
}
}
}
}

bool Strangetank::IsHit(glm::vec2 position) const {
position = WorldToLocal(position);
return position.x > -0.8f && position.x < 0.8f && position.y > -1.0f &&
position.y < 1.0f && position.x + position.y < 1.6f &&
position.y - position.x < 1.6f;
}

const char *Strangetank::UnitName() const {
return "Strangetank";
}

const char *Strangetank::Author() const {
return "wzjwzjwzj";
}
} // namespace battle_game::unit
23 changes: 23 additions & 0 deletions src/battle_game/core/units/strange_tank.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
#pragma once
#include "battle_game/core/unit.h"

namespace battle_game::unit {
class Strangetank : public Unit {
public:
Strangetank(GameCore *game_core, uint32_t id, uint32_t player_id);
void Render() override;
void Update() override;
[[nodiscard]] bool IsHit(glm::vec2 position) const override;

protected:
void TankMove(float move_speed, float rotate_angular_speed);
void TurretRotate();
void Fire();
[[nodiscard]] const char *UnitName() const override;
[[nodiscard]] const char *Author() const override;

float turret_rotation_{0.0f};
uint32_t fire_count_down_{0};
uint32_t mine_count_down_{0};
};
} // namespace battle_game::unit
1 change: 1 addition & 0 deletions src/battle_game/core/units/units.h
Original file line number Diff line number Diff line change
Expand Up @@ -23,3 +23,4 @@
#include "battle_game/core/units/triple_shot_tank.h"
#include "battle_game/core/units/udongein.h"
#include "battle_game/core/units/zibeng_dog.h"
#include "battle_game/core/units/strange_tank.h"