Skip to content
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 @@ -22,6 +22,7 @@ void GameCore::GeneratePrimaryUnitList() {
* TODO: Add Your Unit Here!
* */
ADD_SELECTABLE_UNIT(unit::Tank);
ADD_SELECTABLE_UNIT(unit::wcj_Tank);

unit.reset();
}
Expand Down
163 changes: 163 additions & 0 deletions src/battle_game/core/units/wcj_tank.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,163 @@
#include "wcj_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;
}

wcj_Tank::wcj_Tank(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_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}},
{{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});
}

{
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 wcj_Tank::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 wcj_Tank::Update() {
TankMove(3.0f, glm::radians(180.0f));
TurretRotate();
Fire();
}

void wcj_Tank::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 wcj_Tank::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 wcj_Tank::Fire() {
if (fire_count_down_ == 0) {
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;
}
}
}
if (fire_count_down_) {
fire_count_down_--;
}
}

bool wcj_Tank::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 *wcj_Tank::UnitName() const {
return "wcj Tank";
}

const char *wcj_Tank::Author() const {
return "wcj";
}
}
23 changes: 23 additions & 0 deletions src/battle_game/core/units/wcj_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 wcj_Tank : public Unit {
public:
wcj_Tank(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};
};
}
1 change: 1 addition & 0 deletions xx
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
sterd