#include "game.hpp" namespace game { world::cmodel_t unit_t::make_cmodel(v2f_t at) { world::cmodel_t cmodel; cmodel.bounds[0] = at + size[0]; cmodel.bounds[1] = at + size[1]; cmodel.cflags = cflags; return cmodel; } void unit_t::compute_bounds() { render_bounds[0] = x + render_size[0]; render_bounds[1] = x + render_size[1]; } unit_t::unit_t(game::state_t *game_, unit_t::type_t type_) : entity_t(ET_UNIT) { type = type_; game = game_; } void unit_t::render_to(render::state_t *render) { if (selected == selection_cookie) render->render_hlrect(render_bounds, sf::Color::Blue); if (!dead && health < max_health) { rectf_t bar; float frac; sf::Color color; bar[0][0] = (render_bounds[0][0] + render_bounds[1][0]) / 2; bar[0][1] = render_bounds[0][1] - 0.05f; bar[0][0] -= 0.3f; bar[1][0] = bar[0][0] + 0.6f; bar[1][1] = bar[0][1] + 0.05f; render->render_rect(bar, sf::Color::Black); frac = (float)health / max_health; if (frac < 0.3f) color = sf::Color::Red; else if (frac < 0.75f) color = sf::Color::Yellow; else color = sf::Color::Green; bar[1][0] = lerp(bar[0][0], bar[1][0], frac); render->render_rect(bar, color); } if (move.moving && debug_draw_paths) render->debug_path(&move.path); } void unit_t::say(std::string str) { say_text = str; say_time = game->now; game->interface->print(name + ": " + str); } void unit_t::place(world::world_t *world_, v2f_t x_) { world = world_; x = x_; move.moving = false; unlink(); cmodel = make_cmodel(x); compute_bounds(); link(world); } bool unit_t::keep_moving(double speed) { float time; bool rv = true; if (!move.moving) return true; if (move.blocked && game->now < move.next_attempt) return true; time = game->dt * speed; while (time > 0.0f) { v2f_t delta, next, x_new; world::cmodel_t cmodel_next; if (!move.path.size()) { move.moving = false; break; } next = *move.path.begin(); delta = next - x; move.angle = delta.angle(); if (delta.len() >= time) { x_new = x + delta * time / delta.len(); time -= delta.len(); } else { x_new = next; time -= delta.len(); move.path.pop_front(); } cmodel_next = make_cmodel(x_new); if (!world->test_rect(&cmodel_next, this)) { x = x_new; cmodel = cmodel_next; continue; } if (move.attempts_left) { move.blocked = true; move.attempts_left--; move.next_attempt = game->now + 0.2f; } else { if ((x - move.dst).len() > 1.5f) rv = false; move.moving = false; } break; } unlink(); compute_bounds(); link(world); return rv; } bool unit_t::start_moving(v2f_t dst, world::cflags_t cflags) { world::cmodel_t rep; if (!world) { printf("unit_t::start_moving: entity is not linked\n"); return false; } move.dst = dst; move.path.clear(); rep.cflags = cflags; rep.bounds = cmodel.bounds; if (!world->find_path(x, move.dst, &rep, this, &move.path)) { move.moving = false; return false; } move.moving = true; move.blocked = false; move.attempts_left = 10; move.next_attempt = -INFINITY; return true; } void unit_t::damage(int points, unit_t *attacker) { health -= points; if (health < 0) { game->interface->print(name + " " + text::get(text::UNIT_DEATH) + "."); dead = true; death_time = game->now; cflags = 0; die(); } } void unit_t::try_attack(unit_t *target) { std::stringstream ss; int hit_roll, hit_dc = 10 /* FIXME */; int dmg_roll, dmg_bonus, dmg_total; ss << name << " " << text::get(text::UNIT_ATTACK) << " " << target->name << ": "; hit_roll = roll(1, 20); ss << hit_roll << " vs " << hit_dc; if (hit_roll == 1 || hit_roll < hit_dc) { ss << " (" << text::get((hit_roll == 1 ? text::UNIT_CRITICAL_MISS : text::UNIT_MISS)) << ")"; game->interface->print(ss.str()); return; } ss << " (" << text::get((hit_roll == 20 ? text::UNIT_CRITICAL_HIT : text::UNIT_HIT)) << ")"; game->interface->print(ss.str()); dmg_roll = roll(2, 6); dmg_bonus = 2; dmg_total = dmg_roll + dmg_bonus; ss.str(std::string()); ss << name << " " << text::get(text::UNIT_DAMAGE) << " " << target->name << ": "; ss << dmg_roll << " + " << dmg_bonus << " = " << dmg_total; game->interface->print(ss.str()); target->damage(dmg_total, this); } static unit_t *find_target(world::world_t *world, v2f_t x, float r, unit_t::type_t type) { rectf_t rect; float nearest_dist = INFINITY; unit_t *nearest = NULL; rect[0] = x - v2f_t(r, r); rect[1] = x + v2f_t(r, r); for (world::entity_t *ent : world->get_entities(rect, -1)) { unit_t *unit; float dist; if (ent->type != ET_UNIT) continue; unit = (unit_t*)ent; if (unit->type != type) continue; if (unit->dead) continue; dist = (unit->x - x).len(); if (dist < nearest_dist) { nearest_dist = dist; nearest = unit; } } return nearest; } human_t::human_t(game::state_t *game) : unit_t(game, UNIT_HUMAN) { cflags = CF_BODY; health = max_health = 20; size[0] = v2f_t(-0.4f, -0.4f); size[1] = v2f_t(+0.4f, +0.4f); render_size[0] = v2f_t(-0.5f, -1.0f); render_size[1] = v2f_t(+0.5f, +0.5f); name = text::get(text::UNIT_HUMAN); } void human_t::wake(unit_t *by_whom) { } void human_t::sleep(void) { } void human_t::think(void) { if (game->now > next_targetting) { unit_t *target; target = find_target(world, x, 10.0f, UNIT_ALIEN); if (target) { last_target_time = game->now; last_target_x = target->x; if (last_attack + 0.5 < game->now) { last_attack = game->now; try_attack(target); } //start_moving(target->x, CF_SOLID); } next_targetting = game->now + 0.2; } if (!keep_moving(4.0)) say(text::get(text::SAY_BLOCKED)); } void human_t::die(void) { render_size[0] = v2f_t(-0.75f, -0.5f); render_size[1] = v2f_t(+0.75f, +0.5f); render_layer = -1; unlink(); compute_bounds(); link(world); } void human_t::render_to(render::state_t *render) { if (!dead) { render::oriented_sprite_t *legs, *body; float body_angle; if (move.moving && !move.blocked) legs = &assets::human.legs_walking; else legs = &assets::human.legs_idle; if (last_target_time + 3 > game->now) { if (last_attack + 0.1 > game->now) body = &assets::human.body_firing; else body = &assets::human.body_aiming; body_angle = (last_target_x - x).angle(); } else { body = &assets::human.body_idle; body_angle = move.angle; } render->render(legs, render_bounds, move.angle); render->render(body, render_bounds, body_angle); render->render(&assets::human.head_idle, render_bounds, move.angle); } else render->render(&assets::human.dead, render_bounds); if (say_time + 5.0 > game->now) { v2f_t text_pos; float height; text_pos = render_bounds[0] + v2f_t(render_bounds.dim(0) / 2, -render_bounds.dim(1) * 0.1); height = size.dim_min() * 0.20f; render->render_text(text_pos, height, say_text, render::ALIGN_CENTER_BOTTOM, sf::Color::White); } unit_t::render_to(render); } alien_t::alien_t(game::state_t *game) : unit_t(game, UNIT_ALIEN) { cflags = CF_BODY_SMALL; health = max_health = 4; size[0] = v2f_t(-0.2f, -0.2f); size[1] = v2f_t(+0.2f, +0.2f); render_size[0] = v2f_t(-0.3f, -0.3f); render_size[1] = v2f_t(+0.3f, +0.3f); name = text::get(text::UNIT_ALIEN); } void alien_t::wake(unit_t *by_whom) { start_moving(by_whom->x, CF_SOLID); next_targetting = game->now + 0.4; } void alien_t::sleep(void) { } void alien_t::die(void) { render_layer = -1; } void alien_t::attack(unit_t *target, float range) { world::trace_t trace; if (game->now < next_attack) return; if ((x - target->x).len() > range) return; trace = world->trace(x, target->x, CF_SOLID); if (!trace.hit) try_attack(target); next_attack = game->now + 1.0; } void alien_t::think(void) { if (game->now > next_targetting) { unit_t *target; target = find_target(world, x, 10.0f, UNIT_HUMAN); if (target) { attack(target, 1.0f); start_moving(target->x, CF_SOLID); } next_targetting = game->now + 0.2; } keep_moving(7.0); } void alien_t::render_to(render::state_t *render) { bool moving; moving = move.moving && !move.blocked; if (!dead) render->render((moving ? &assets::alien.walking : &assets::alien.idle), render_bounds, move.angle); else render->render(&assets::alien.dead, render_bounds); unit_t::render_to(render); } } // namespace game