3520 lines
201 KiB
C++
3520 lines
201 KiB
C++
#if defined(__clang__)
|
|
#pragma once
|
|
#include "feely_pona_unity.h"
|
|
#endif
|
|
|
|
Dqn_f32 const PHYSICS_STEP = 1 / 60.f;
|
|
|
|
Dqn_Rect FP_Game_GetBuildingPlacementRectForEntity(FP_Game *game, FP_GamePlaceableBuilding placeable_building, FP_GameEntityHandle handle)
|
|
{
|
|
Dqn_Rect result = {};
|
|
FP_GameEntity *entity = FP_Game_GetEntity(game, handle);
|
|
if (FP_Game_IsNilEntity(entity))
|
|
return result;
|
|
|
|
FP_EntityRenderData render_data = FP_Entity_GetRenderData(game, placeable_building.type, placeable_building.state, entity->direction);
|
|
|
|
Dqn_Rect box = FP_Game_CalcEntityWorldHitBox(game, entity->handle);
|
|
Dqn_V2 build_p = {};
|
|
switch (entity->direction) {
|
|
case FP_GameDirection_Up: {
|
|
build_p = Dqn_Rect_InterpolatedPoint(box, Dqn_V2_InitNx2(0.5f, 0.f)) - Dqn_V2_InitNx2(0.f, render_data.render_size.h * .5f + 10.f);
|
|
} break;
|
|
|
|
case FP_GameDirection_Down: {
|
|
build_p = Dqn_Rect_InterpolatedPoint(box, Dqn_V2_InitNx2(0.5f, 1.f)) + Dqn_V2_InitNx2(0.f, render_data.render_size.h * .5f + 10.f);
|
|
} break;
|
|
|
|
case FP_GameDirection_Left: {
|
|
build_p = Dqn_Rect_InterpolatedPoint(box, Dqn_V2_InitNx2(0.0f, 0.5f)) - Dqn_V2_InitNx2(render_data.render_size.w * .5f + 10.f, 0);
|
|
} break;
|
|
|
|
case FP_GameDirection_Right: {
|
|
build_p = Dqn_Rect_InterpolatedPoint(box, Dqn_V2_InitNx2(1.f, 0.5f)) + Dqn_V2_InitNx2(render_data.render_size.w * .5f + 10.f, 0);
|
|
} break;
|
|
|
|
case FP_GameDirection_Count: DQN_INVALID_CODE_PATH; break;
|
|
}
|
|
|
|
result.size = render_data.render_size;
|
|
result.pos = build_p - (render_data.render_size * .5f);
|
|
return result;
|
|
}
|
|
|
|
TELY_AssetSpriteSheet FP_LoadSpriteSheetFromSpec(TELY_Platform *platform, TELY_Assets *assets, Dqn_Arena *arena, Dqn_String8 sheet_name)
|
|
{
|
|
TELY_AssetSpriteSheet result = {};
|
|
Dqn_ThreadScratch scratch = Dqn_Thread_GetScratch(arena);
|
|
result.sprite_size = Dqn_V2I_InitNx2(185, 170);
|
|
result.type = TELY_AssetSpriteSheetType_Rects;
|
|
|
|
// NOTE: Load the sprite meta file =========================================================
|
|
Dqn_String8 sprite_spec_path = Dqn_FsPath_ConvertF(scratch.arena, "%.*s/%.*s.txt", DQN_STRING_FMT(assets->textures_dir), DQN_STRING_FMT(sheet_name));
|
|
Dqn_String8 sprite_spec_buffer = platform->func_load_file(scratch.arena, sprite_spec_path);
|
|
Dqn_String8SplitAllocResult lines = Dqn_String8_SplitAlloc(scratch.allocator, sprite_spec_buffer, DQN_STRING8("\n"));
|
|
Dqn_usize sprite_rect_index = 0;
|
|
Dqn_usize sprite_anim_index = 0;
|
|
|
|
DQN_FOR_UINDEX(line_index, lines.size) {
|
|
Dqn_String8 line = lines.data[line_index];
|
|
Dqn_String8SplitAllocResult line_splits = Dqn_String8_SplitAlloc(scratch.allocator, line, DQN_STRING8(";"));
|
|
|
|
if (line_index == 0) {
|
|
DQN_ASSERTF(line_splits.size == 4, "Expected 4 splits for @file lines");
|
|
DQN_ASSERT(Dqn_String8_StartsWith(line_splits.data[0], DQN_STRING8("@file"), Dqn_String8EqCase_Sensitive));
|
|
|
|
// NOTE: Sprite sheet path
|
|
Dqn_String8 sprite_sheet_path = Dqn_FsPath_ConvertF(scratch.arena, "%.*s/%.*s", DQN_STRING_FMT(assets->textures_dir), DQN_STRING_FMT(line_splits.data[1]));
|
|
result.tex_handle = platform->func_load_texture(assets, sheet_name, sprite_sheet_path);
|
|
DQN_ASSERTF(Dqn_Fs_Exists(sprite_sheet_path), "Required file does not exist '%.*s'", DQN_STRING_FMT(sprite_sheet_path));
|
|
|
|
// NOTE: Total sprite frame count
|
|
Dqn_String8ToU64Result total_frame_count = Dqn_String8_ToU64(line_splits.data[2], 0);
|
|
DQN_ASSERT(total_frame_count.success);
|
|
result.rects = Dqn_Slice_Alloc<Dqn_Rect>(arena, total_frame_count.value, Dqn_ZeroMem_No);
|
|
|
|
// NOTE: Total animation count
|
|
Dqn_String8ToU64Result total_anim_count = Dqn_String8_ToU64(line_splits.data[3], 0);
|
|
DQN_ASSERT(total_anim_count.success);
|
|
result.anims = Dqn_Slice_Alloc<TELY_AssetSpriteAnimation>(arena, total_anim_count.value, Dqn_ZeroMem_No);
|
|
|
|
// TODO: Sprite size?
|
|
|
|
// TODO: Texture name?
|
|
continue;
|
|
}
|
|
|
|
if (Dqn_String8_StartsWith(line, DQN_STRING8("@anim"))) {
|
|
DQN_ASSERTF(line_splits.size == 4, "Expected 4 splits for @anim lines");
|
|
Dqn_String8 anim_name = line_splits.data[1];
|
|
Dqn_String8ToU64Result frames_per_second = Dqn_String8_ToU64(line_splits.data[2], 0);
|
|
Dqn_String8ToU64Result frame_count = Dqn_String8_ToU64(line_splits.data[3], 0);
|
|
|
|
DQN_ASSERT(anim_name.size);
|
|
DQN_ASSERT(frame_count.success);
|
|
DQN_ASSERT(frames_per_second.success);
|
|
|
|
Dqn_Allocator allocator = Dqn_Arena_Allocator(arena);
|
|
TELY_AssetSpriteAnimation *anim = result.anims.data + sprite_anim_index++;
|
|
anim->label = Dqn_String8_Copy(allocator, anim_name);
|
|
anim->index = DQN_CAST(uint16_t)sprite_rect_index;
|
|
anim->count = DQN_CAST(uint16_t)frame_count.value;
|
|
anim->ms_per_frame = DQN_CAST(uint32_t)(1000.f / frames_per_second.value);
|
|
DQN_ASSERT(anim->ms_per_frame != 0);
|
|
} else {
|
|
DQN_ASSERTF(line_splits.size == 5, "Expected 5 splits for sprite frame lines");
|
|
Dqn_String8ToU64Result x = Dqn_String8_ToU64(line_splits.data[0], 0);
|
|
Dqn_String8ToU64Result y = Dqn_String8_ToU64(line_splits.data[1], 0);
|
|
Dqn_String8ToU64Result w = Dqn_String8_ToU64(line_splits.data[2], 0);
|
|
Dqn_String8ToU64Result h = Dqn_String8_ToU64(line_splits.data[3], 0);
|
|
|
|
DQN_ASSERT(x.success);
|
|
DQN_ASSERT(y.success);
|
|
DQN_ASSERT(w.success);
|
|
DQN_ASSERT(h.success);
|
|
|
|
result.rects.data[sprite_rect_index++] =
|
|
Dqn_Rect_InitNx4(DQN_CAST(Dqn_f32) x.value,
|
|
DQN_CAST(Dqn_f32) y.value,
|
|
DQN_CAST(Dqn_f32) w.value,
|
|
DQN_CAST(Dqn_f32) h.value);
|
|
}
|
|
}
|
|
|
|
DQN_ASSERT(result.rects.size == sprite_rect_index);
|
|
DQN_ASSERT(result.anims.size == sprite_anim_index);
|
|
|
|
return result;
|
|
}
|
|
|
|
static void FP_Game_MoveEntity(FP_Game *game, FP_GameEntityHandle entity_handle, Dqn_V2 acceleration_meters_per_s)
|
|
{
|
|
// f"(t) = a
|
|
// f'(t) = at + v
|
|
// f (t) = 0.5f*a(t^2) + vt + p
|
|
|
|
FP_GameEntity *entity = FP_Game_GetEntity(game, entity_handle);
|
|
if (FP_Game_IsNilEntity(entity))
|
|
return;
|
|
|
|
Dqn_f32 t = DQN_CAST(Dqn_f32)DQN_SQUARED(PHYSICS_STEP);
|
|
Dqn_f32 t_squared = DQN_SQUARED(t);
|
|
|
|
Dqn_f32 velocity_falloff_coefficient = 0.82f;
|
|
Dqn_f32 acceleration_feel_good_factor = 15'000.f;
|
|
Dqn_V2 acceleration = FP_Game_MetersToPixelsV2(game->play, acceleration_meters_per_s) * acceleration_feel_good_factor;
|
|
entity->velocity = (acceleration * t) + entity->velocity * velocity_falloff_coefficient;
|
|
|
|
// NOTE: Zero out velocity with epsilon
|
|
if (DQN_ABS(entity->velocity.x) < 5.f)
|
|
entity->velocity.x = 0.f;
|
|
if (DQN_ABS(entity->velocity.y) < 5.f)
|
|
entity->velocity.y = 0.f;
|
|
|
|
Dqn_V2 delta_pos = (acceleration * 0.5f * t_squared) + (entity->velocity * t);
|
|
Dqn_Rect entity_world_hit_box = FP_Game_CalcEntityWorldHitBox(game, entity->handle);
|
|
Dqn_V2 entity_pos = FP_Game_CalcEntityWorldPos(game, entity->handle);
|
|
Dqn_V2 entity_new_pos = entity_pos + delta_pos;
|
|
|
|
Dqn_f32 const SENTINEL_T = 999.f;
|
|
Dqn_f32 global_earliest_t = SENTINEL_T;
|
|
Dqn_V2 global_earliest_pos_just_before_collide = {};
|
|
|
|
if ((entity->flags & FP_GameEntityFlag_NoClip) == 0) {
|
|
for (FP_GameEntityIterator collider_it = {}; FP_Game_DFSPostOrderWalkEntityTree(game, &collider_it, game->play.root_entity); ) {
|
|
FP_GameEntity *collider = collider_it.entity;
|
|
if (collider->handle == entity->handle)
|
|
continue;
|
|
|
|
// TODO(doyle): Calculate the list of collidables at the start of the frame
|
|
if ((collider->flags & FP_GameEntityFlag_NonTraversable) == 0)
|
|
continue;
|
|
|
|
bool entity_collides_with_collider = true;
|
|
switch (entity->type) {
|
|
case FP_EntityType_Catfish: /*FALLTHRU*/
|
|
case FP_EntityType_Smoochie: /*FALLTHRU*/
|
|
case FP_EntityType_Clinger: {
|
|
if (collider->type == FP_EntityType_Smoochie || collider->type == FP_EntityType_Clinger || collider->type == FP_EntityType_Catfish) {
|
|
entity_collides_with_collider = false;
|
|
} else if (FP_Entity_IsBuildingForMobs(collider)) {
|
|
#if 0
|
|
// NOTE: We disable collision on buildings we have visited to avoid some
|
|
// problems ...
|
|
if (FP_SentinelList_Find(&entity->buildings_visited, collider->handle)) {
|
|
entity_collides_with_collider = false;
|
|
}
|
|
#else
|
|
entity_collides_with_collider = false;
|
|
#endif
|
|
} else if (collider->type == FP_EntityType_Heart ||
|
|
collider->type == FP_EntityType_MerchantGym ||
|
|
collider->type == FP_EntityType_MerchantTerry ||
|
|
collider->type == FP_EntityType_MerchantGraveyard ||
|
|
collider->type == FP_EntityType_MerchantPhoneCompany) {
|
|
entity_collides_with_collider = false;
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_Terry: {
|
|
// NOTE: Don't collide with mobs when dashing (e.g. phase through)
|
|
FP_EntityTerryState state = *DQN_CAST(FP_EntityTerryState *)&entity->action.state;
|
|
if (state == FP_EntityTerryState_Dash || state == FP_EntityTerryState_DeadGhost) {
|
|
if (collider->type == FP_EntityType_Smoochie || collider->type == FP_EntityType_Clinger || collider->type == FP_EntityType_Catfish)
|
|
entity_collides_with_collider = false;
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_Nil: break;
|
|
case FP_EntityType_MerchantTerry: break;
|
|
case FP_EntityType_Count: break;
|
|
case FP_EntityType_ClubTerry: break;
|
|
case FP_EntityType_Map: break;
|
|
case FP_EntityType_MerchantGraveyard: break;
|
|
case FP_EntityType_MerchantGym: break;
|
|
case FP_EntityType_MerchantPhoneCompany: break;
|
|
case FP_EntityType_Heart: break;
|
|
case FP_EntityType_AirportTerry: break;
|
|
case FP_EntityType_ChurchTerry: break;
|
|
case FP_EntityType_KennelTerry: break;
|
|
case FP_EntityType_PhoneMessageProjectile: break;
|
|
}
|
|
|
|
if (!entity_collides_with_collider)
|
|
continue;
|
|
|
|
// NOTE: Sweep collider with half the radius of the source entity
|
|
Dqn_Rect collider_world_hit_box = FP_Game_CalcEntityWorldHitBox(game, collider->handle);
|
|
if (Dqn_V2_Area(collider_world_hit_box.size) <= 0)
|
|
continue;
|
|
|
|
Dqn_Rect swept_collider_world_hit_box = collider_world_hit_box;
|
|
swept_collider_world_hit_box.pos -= (entity_world_hit_box.size * .5f);
|
|
swept_collider_world_hit_box.size += entity_world_hit_box.size;
|
|
|
|
if (!Dqn_Rect_ContainsPoint(swept_collider_world_hit_box, entity_new_pos))
|
|
continue;
|
|
|
|
Dqn_f32 collider_left_wall_x = swept_collider_world_hit_box.pos.x;
|
|
Dqn_f32 collider_right_wall_x = swept_collider_world_hit_box.pos.x + swept_collider_world_hit_box.size.w;
|
|
Dqn_f32 collider_top_wall_y = swept_collider_world_hit_box.pos.y;
|
|
Dqn_f32 collider_bottom_wall_y = swept_collider_world_hit_box.pos.y + swept_collider_world_hit_box.size.h;
|
|
|
|
Dqn_V2 o = entity_pos;
|
|
Dqn_V2 d = delta_pos;
|
|
|
|
// NOTE: Solve collision by determining the 't' value at which
|
|
// we hit one of the walls of the collider and move the entity
|
|
// at exactly that point.
|
|
// O + td = x
|
|
// td = x - O
|
|
// t = (x - O) / d
|
|
|
|
Dqn_f32 earliest_t = SENTINEL_T;
|
|
if (d.x != 0.f) {
|
|
Dqn_f32 left_t = (collider_left_wall_x - o.x) / d.x;
|
|
Dqn_f32 right_t = (collider_right_wall_x - o.x) / d.x;
|
|
if (left_t >= 0.f && left_t <= 1.f)
|
|
earliest_t = DQN_MIN(earliest_t, left_t);
|
|
if (right_t >= 0.f && right_t <= 1.f)
|
|
earliest_t = DQN_MIN(earliest_t, right_t);
|
|
}
|
|
|
|
if (d.y != 0.f) {
|
|
Dqn_f32 top_t = (collider_top_wall_y - o.y) / d.y;
|
|
Dqn_f32 bottom_t = (collider_bottom_wall_y - o.y) / d.y;
|
|
if (top_t >= 0.f && top_t <= 1.f)
|
|
earliest_t = DQN_MIN(earliest_t, top_t);
|
|
if (bottom_t >= 0.f && bottom_t <= 1.f)
|
|
earliest_t = DQN_MIN(earliest_t, bottom_t);
|
|
}
|
|
|
|
if (earliest_t < global_earliest_t) {
|
|
global_earliest_t = earliest_t;
|
|
global_earliest_pos_just_before_collide = entity_pos + (d * earliest_t);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
if (global_earliest_t == SENTINEL_T) {
|
|
entity->local_pos += delta_pos;
|
|
} else {
|
|
Dqn_V2 new_delta_pos = global_earliest_pos_just_before_collide - entity_pos;
|
|
entity->local_pos += new_delta_pos;
|
|
}
|
|
}
|
|
|
|
static void FP_PlayReset(FP_Game *game, TELY_Platform *platform)
|
|
{
|
|
FP_GamePlay *play = &game->play;
|
|
if (game->play.root_entity)
|
|
FP_Game_DeleteEntity(game, game->play.root_entity->handle);
|
|
|
|
Dqn_VArray<FP_GameEntity> shallow_entities_copy = play->entities;
|
|
*play = {};
|
|
|
|
play->chunk_pool = &platform->chunk_pool;
|
|
play->meters_to_pixels = 65.416f;
|
|
play->entities = shallow_entities_copy;
|
|
|
|
if (play->entities.data) {
|
|
Dqn_VArray_Clear(&play->entities, Dqn_ZeroMem_Yes);
|
|
} else {
|
|
play->entities = Dqn_VArray_Init<FP_GameEntity>(&play->arena, 1024 * 8);
|
|
}
|
|
|
|
play->root_entity = Dqn_VArray_Make(&play->entities, Dqn_ZeroMem_No);
|
|
Dqn_FArray_Add(&play->parent_entity_stack, play->root_entity->handle);
|
|
Dqn_PCG32_Seed(&play->rng, 0xABCDEF);
|
|
|
|
// NOTE: Map ===================================================================================
|
|
{
|
|
TELY_AssetSpriteAnimation *sprite_anim = TELY_Asset_GetSpriteAnimation(&game->atlas_sprite_sheet, g_anim_names.map);
|
|
Dqn_Rect sprite_rect = game->atlas_sprite_sheet.rects.data[sprite_anim->index];
|
|
|
|
FP_GameEntity *entity = FP_Game_MakeEntityPointerF(game, "Map");
|
|
entity->type = FP_EntityType_Map;
|
|
FP_EntityRenderData render_data = FP_Entity_GetRenderData(game, entity->type, 0 /*state*/, FP_GameDirection_Down);
|
|
entity->sprite_height = render_data.height;
|
|
entity->local_pos = {};
|
|
|
|
Dqn_f32 size_scale = FP_Entity_CalcSpriteScaleForDesiredHeight(game, entity->sprite_height, sprite_rect);
|
|
Dqn_V2 sprite_rect_scaled = sprite_rect.size * size_scale;
|
|
|
|
entity->local_hit_box_size = sprite_rect_scaled;
|
|
FP_Entity_AddDebugEditorFlags(game, entity->handle);
|
|
FP_Game_EntityActionReset(game, entity->handle, FP_GAME_ENTITY_ACTION_INFINITE_TIMER, render_data.sprite);
|
|
play->map = entity;
|
|
}
|
|
|
|
// NOTE: Map walls =============================================================================
|
|
FP_GameEntity const *map = play->map;
|
|
Dqn_Rect const map_hit_box = FP_Game_CalcEntityWorldHitBox(game, map->handle);
|
|
{
|
|
Dqn_f32 wall_thickness = FP_Game_MetersToPixelsNx1(game->play, 1.f);
|
|
Dqn_f32 half_wall_thickness = wall_thickness * .5f;
|
|
|
|
FP_Entity_CreateWallAtPos(game,
|
|
DQN_STRING8("Left Wall"),
|
|
Dqn_Rect_InterpolatedPoint(map_hit_box, Dqn_V2_InitNx2(0.f, 0.5f)) - Dqn_V2_InitNx2(half_wall_thickness, 0.f),
|
|
Dqn_V2_InitNx2(wall_thickness, map_hit_box.size.h));
|
|
|
|
FP_Entity_CreateWallAtPos(game,
|
|
DQN_STRING8("Right Wall"),
|
|
Dqn_Rect_InterpolatedPoint(map_hit_box, Dqn_V2_InitNx2(1.f, 0.5f)) + Dqn_V2_InitNx2(half_wall_thickness, 0.f),
|
|
Dqn_V2_InitNx2(wall_thickness, map_hit_box.size.h));
|
|
|
|
FP_Entity_CreateWallAtPos(game,
|
|
DQN_STRING8("Top Wall"),
|
|
Dqn_Rect_InterpolatedPoint(map_hit_box, Dqn_V2_InitNx2(0.5f, 0.f)) - Dqn_V2_InitNx2(0.f, half_wall_thickness),
|
|
Dqn_V2_InitNx2(map_hit_box.size.w, wall_thickness));
|
|
|
|
FP_Entity_CreateWallAtPos(game,
|
|
DQN_STRING8("Bottom Wall"),
|
|
Dqn_Rect_InterpolatedPoint(map_hit_box, Dqn_V2_InitNx2(0.5f, 1.f)) + Dqn_V2_InitNx2(0.f, half_wall_thickness),
|
|
Dqn_V2_InitNx2(map_hit_box.size.w, wall_thickness));
|
|
}
|
|
|
|
// NOTE: Map building zones
|
|
{
|
|
{
|
|
FP_Entity_CreatePermittedBuildZone(game,
|
|
Dqn_V2_InitNx2(0.f, -1206),
|
|
Dqn_V2_InitNx2(map_hit_box.size.w, 335),
|
|
"Building Zone");
|
|
}
|
|
|
|
{
|
|
FP_Entity_CreatePermittedBuildZone(game,
|
|
Dqn_V2_InitNx2(-839.9, -460),
|
|
Dqn_V2_InitNx2(2991.3, 670),
|
|
"Building Zone");
|
|
}
|
|
|
|
{
|
|
FP_Entity_CreatePermittedBuildZone(game,
|
|
Dqn_V2_InitNx2(-839.9, 460),
|
|
Dqn_V2_InitNx2(2991.3, 670),
|
|
"Building Zone");
|
|
}
|
|
|
|
{
|
|
FP_Entity_CreatePermittedBuildZone(game,
|
|
Dqn_V2_InitNx2(0.f, 1200),
|
|
Dqn_V2_InitNx2(map_hit_box.size.w, 335),
|
|
"Building Zone");
|
|
}
|
|
}
|
|
|
|
Dqn_V2 base_mid_p = Dqn_V2_InitNx2(1580, 0.f);
|
|
{
|
|
// NOTE: Mid lane mob spawner ==================================================================
|
|
Dqn_V2 mid_lane_mob_spawner_pos = Dqn_V2_InitNx2(play->map->local_hit_box_size.w * -0.5f + 128.f, 0.f);
|
|
Dqn_usize spawn_cap = 16;
|
|
{
|
|
FP_GameEntityHandle mob_spawner = FP_Entity_CreateMobSpawner(game, mid_lane_mob_spawner_pos, spawn_cap, "Mob spawner");
|
|
FP_Game_PushParentEntity(game, mob_spawner);
|
|
FP_Entity_CreateWaypointF(game, Dqn_V2_InitNx2(-mid_lane_mob_spawner_pos.x + base_mid_p.x, base_mid_p.y), "Waypoint");
|
|
FP_Game_PopParentEntity(game);
|
|
Dqn_FArray_Add(&play->mob_spawners, mob_spawner);
|
|
}
|
|
|
|
// NOTE: Bottom lane spawner ===================================================================
|
|
#if 1
|
|
Dqn_V2 bottom_lane_mob_spawner_pos = Dqn_V2_InitNx2(mid_lane_mob_spawner_pos.x, mid_lane_mob_spawner_pos.y + 932.f);
|
|
{
|
|
FP_GameEntityHandle mob_spawner = FP_Entity_CreateMobSpawner(game, bottom_lane_mob_spawner_pos, spawn_cap, "Mob spawner");
|
|
FP_Game_PushParentEntity(game, mob_spawner);
|
|
FP_Entity_CreateWaypointF(game, Dqn_V2_InitNx2(-bottom_lane_mob_spawner_pos.x + base_mid_p.x, 0.f), "Waypoint");
|
|
FP_Entity_CreateWaypointF(game, Dqn_V2_InitNx2(-bottom_lane_mob_spawner_pos.x + base_mid_p.x, -932.f), "Waypoint");
|
|
FP_Game_PopParentEntity(game);
|
|
Dqn_FArray_Add(&play->mob_spawners, mob_spawner);
|
|
}
|
|
|
|
// NOTE: Top lane spawner ===================================================================
|
|
Dqn_V2 top_lane_mob_spawner_pos = Dqn_V2_InitNx2(mid_lane_mob_spawner_pos.x, mid_lane_mob_spawner_pos.y - 915.f);
|
|
{
|
|
FP_GameEntityHandle mob_spawner = FP_Entity_CreateMobSpawner(game, top_lane_mob_spawner_pos, spawn_cap, "Mob spawner");
|
|
FP_Game_PushParentEntity(game, mob_spawner);
|
|
FP_Entity_CreateWaypointF(game, Dqn_V2_InitNx2(-top_lane_mob_spawner_pos.x + base_mid_p.x, 0.f), "Waypoint");
|
|
FP_Entity_CreateWaypointF(game, Dqn_V2_InitNx2(-top_lane_mob_spawner_pos.x + base_mid_p.x, +915.f), "Waypoint");
|
|
FP_Game_PopParentEntity(game);
|
|
Dqn_FArray_Add(&play->mob_spawners, mob_spawner);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// NOTE: Monkey ============================================================
|
|
{
|
|
Dqn_V2 monkey_base_p = Dqn_V2_InitNx2(base_mid_p.x + 400.f, base_mid_p.y + 16.f);
|
|
FP_GameEntityHandle monkey_a = FP_Entity_CreatePortalMonkey(game, Dqn_V2_InitNx2(monkey_base_p.x, monkey_base_p.y), "Monkey A");
|
|
FP_GameEntityHandle monkey_b = FP_Entity_CreatePortalMonkey(game, Dqn_V2_InitNx2(monkey_base_p.x, monkey_base_p.y + 100.f), "Monkey B");
|
|
FP_GameEntityHandle monkey_c = FP_Entity_CreatePortalMonkey(game, Dqn_V2_InitNx2(monkey_base_p.x, monkey_base_p.y - 100.f), "Monkey C");
|
|
|
|
Dqn_FArray_Add(&play->portal_monkeys, monkey_a);
|
|
Dqn_FArray_Add(&play->portal_monkeys, monkey_b);
|
|
Dqn_FArray_Add(&play->portal_monkeys, monkey_c);
|
|
}
|
|
|
|
// NOTE: Hero ==================================================================================
|
|
{
|
|
FP_GameEntityHandle terry = FP_Entity_CreateTerry(game, Dqn_V2_InitNx2(1434, 11), "Terry");
|
|
play->clicked_entity = terry;
|
|
play->player = terry;
|
|
}
|
|
|
|
{
|
|
Dqn_V2 base_top_left_pos = Dqn_V2_InitNx2(1018, -335);
|
|
Dqn_V2 base_bottom_right_pos = Dqn_V2_InitNx2(2050, +351);
|
|
|
|
Dqn_V2 base_top_left = base_top_left_pos;
|
|
Dqn_V2 base_top_right = Dqn_V2_InitNx2(base_bottom_right_pos.x, base_top_left_pos.y);
|
|
Dqn_V2 base_bottom_left = Dqn_V2_InitNx2(base_top_left_pos.x, base_bottom_right_pos.y);
|
|
Dqn_V2 base_bottom_right = Dqn_V2_InitNx2(base_bottom_right_pos.x, base_bottom_right_pos.y);
|
|
|
|
play->merchant_terry = FP_Entity_CreateMerchantTerry(game, base_top_left, "Merchant");
|
|
play->merchant_graveyard = FP_Entity_CreateMerchantGraveyard(game, base_bottom_left, "Graveyard");
|
|
play->merchant_gym = FP_Entity_CreateMerchantGym(game, base_bottom_right, "Gym");
|
|
play->merchant_phone_company = FP_Entity_CreateMerchantPhoneCompany(game, base_top_right, "PhoneCompany");
|
|
}
|
|
|
|
#if 0
|
|
FP_Entity_CreateClubTerry(game, Dqn_V2_InitNx2(+500, -191), "Club Terry");
|
|
FP_Entity_CreateKennelTerry(game, Dqn_V2_InitNx2(-300, -191), "Kennel Terry");
|
|
FP_Entity_CreateChurchTerry(game, Dqn_V2_InitNx2(-800, -191), "Church Terry");
|
|
FP_Entity_CreateAirportTerry(game, Dqn_V2_InitNx2(-1200, -191), "Airport Terry");
|
|
#endif
|
|
|
|
play->tile_size = 37;
|
|
Dqn_V2I max_tile = platform->core.window_size / play->tile_size;
|
|
|
|
// NOTE: Heart
|
|
game->play.heart = FP_Entity_CreateHeart(game, base_mid_p, "Heart");
|
|
play->camera.world_pos = base_mid_p - Dqn_V2_InitV2I(platform->core.window_size * .5f);
|
|
play->camera.scale = Dqn_V2_InitNx1(1);
|
|
}
|
|
|
|
extern "C" __declspec(dllexport)
|
|
void TELY_DLL_Reload(void *user_data)
|
|
{
|
|
TELY_Platform *platform = DQN_CAST(TELY_Platform *)user_data;
|
|
Dqn_Library_SetPointer(platform->core.dqn_lib);
|
|
|
|
}
|
|
|
|
extern "C" __declspec(dllexport)
|
|
void TELY_DLL_Init(void *user_data)
|
|
{
|
|
TELY_Platform *platform = DQN_CAST(TELY_Platform *)user_data;
|
|
TELY_DLL_Reload(user_data);
|
|
|
|
FP_UnitTests(platform);
|
|
|
|
// NOTE: TELY Game =============================================================================
|
|
TELY_Assets *assets = &platform->assets;
|
|
FP_Game *game = Dqn_Arena_New(&platform->arena, FP_Game, Dqn_ZeroMem_Yes);
|
|
|
|
uint16_t font_size = 18;
|
|
game->inter_regular_font_large = platform->func_load_font(assets, DQN_STRING8("Inter (Regular)"), DQN_STRING8("Data/Inter-Regular.otf"), DQN_CAST(uint16_t)(font_size * 5.f));
|
|
game->inter_regular_font = platform->func_load_font(assets, DQN_STRING8("Inter (Regular)"), DQN_STRING8("Data/Inter-Regular.otf"), DQN_CAST(uint16_t)(font_size));
|
|
game->inter_italic_font = platform->func_load_font(assets, DQN_STRING8("Inter (Italic)"), DQN_STRING8("Data/Inter-Italic.otf"), font_size);
|
|
game->jetbrains_mono_font = platform->func_load_font(assets, DQN_STRING8("JetBrains Mono NL (Regular)"), DQN_STRING8("Data/JetBrainsMonoNL-Regular.ttf"), font_size);
|
|
game->talkco_font = platform->func_load_font(assets, DQN_STRING8("Talkco"), DQN_STRING8("Data/Talkco.otf"), font_size);
|
|
game->talkco_font_large = platform->func_load_font(assets, DQN_STRING8("Talkco"), DQN_STRING8("Data/Talkco.otf"), DQN_CAST(uint16_t)(font_size * 1.5f));
|
|
game->audio[FP_GameAudio_TestAudio] = platform->func_load_audio(assets, DQN_STRING8("Test Audio"), DQN_STRING8("Data/Audio/Purrple Cat - Moonwinds.qoa"));
|
|
game->audio[FP_GameAudio_TerryHit] = platform->func_load_audio(assets, DQN_STRING8("Terry Hit"), DQN_STRING8("Data/Audio/terry_hit.ogg"));
|
|
game->audio[FP_GameAudio_Smooch] = platform->func_load_audio(assets, DQN_STRING8("Smooch"), DQN_STRING8("Data/Audio/smooch.mp3"));
|
|
game->audio[FP_GameAudio_Woosh] = platform->func_load_audio(assets, DQN_STRING8("Woosh"), DQN_STRING8("Data/Audio/woosh.ogg"));
|
|
game->audio[FP_GameAudio_Ching] = platform->func_load_audio(assets, DQN_STRING8("Ching"), DQN_STRING8("Data/Audio/ching.ogg"));
|
|
game->audio[FP_GameAudio_Church] = platform->func_load_audio(assets, DQN_STRING8("Church"), DQN_STRING8("Data/Audio/church.ogg"));
|
|
game->audio[FP_GameAudio_Plane] = platform->func_load_audio(assets, DQN_STRING8("Plane"), DQN_STRING8("Data/Audio/airport.ogg"));
|
|
game->audio[FP_GameAudio_Club] = platform->func_load_audio(assets, DQN_STRING8("Club"), DQN_STRING8("Data/Audio/club_terry.ogg"));
|
|
game->audio[FP_GameAudio_Dog] = platform->func_load_audio(assets, DQN_STRING8("Dog"), DQN_STRING8("Data/Audio/dog.ogg"));
|
|
game->audio[FP_GameAudio_MerchantTerry] = platform->func_load_audio(assets, DQN_STRING8("Door"), DQN_STRING8("Data/Audio/merchant_terry.ogg"));
|
|
game->audio[FP_GameAudio_MerchantGhost] = platform->func_load_audio(assets, DQN_STRING8("Ghost"), DQN_STRING8("Data/Audio/merchant_ghost.ogg"));
|
|
game->audio[FP_GameAudio_MerchantGym] = platform->func_load_audio(assets, DQN_STRING8("Gym"), DQN_STRING8("Data/Audio/merchant_gym.ogg"));
|
|
game->audio[FP_GameAudio_MerchantPhone] = platform->func_load_audio(assets, DQN_STRING8("Phone"), DQN_STRING8("Data/Audio/merchant_tech.ogg"));
|
|
game->audio[FP_GameAudio_Message] = platform->func_load_audio(assets, DQN_STRING8("Message"), DQN_STRING8("Data/Audio/message.ogg"));
|
|
|
|
platform->user_data = game;
|
|
{
|
|
TELY_AssetSpriteSheet *sheet = &game->hero_sprite_sheet;
|
|
Dqn_ThreadScratch scratch = Dqn_Thread_GetScratch(nullptr);
|
|
Dqn_String8 sheet_path = Dqn_FsPath_ConvertF(scratch.arena, "%.*s/adventurer-v1.5-sheet.png", DQN_STRING_FMT(assets->textures_dir));
|
|
sheet->tex_handle = platform->func_load_texture(assets, DQN_STRING8("Hero"), sheet_path);
|
|
sheet->sprite_count = 109;
|
|
sheet->sprites_per_row = 7;
|
|
sheet->sprite_size = Dqn_V2I_InitNx2(50, 37);
|
|
|
|
TELY_AssetSpriteAnimation hero_anims[] = {
|
|
{DQN_STRING8("Everything"), /*index*/ 0, /*count*/ sheet->sprite_count, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Idle"), /*index*/ 0, /*count*/ 3, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 4.f)},
|
|
{DQN_STRING8("Run"), /*index*/ 8, /*count*/ 6, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 8.f)},
|
|
{DQN_STRING8("Jump"), /*index*/ 14, /*count*/ 10, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Floor slide"), /*index*/ 24, /*count*/ 5, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Unknown"), /*index*/ 29, /*count*/ 9, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Attack A"), /*index*/ 42, /*count*/ 7, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Attack B"), /*index*/ 49, /*count*/ 4, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 8.f)},
|
|
{DQN_STRING8("Attack C"), /*index*/ 53, /*count*/ 6, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Hurt A"), /*index*/ 59, /*count*/ 5, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Hurt B"), /*index*/ 64, /*count*/ 5, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Unsheath sword"), /*index*/ 69, /*count*/ 4, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Sheath sword"), /*index*/ 73, /*count*/ 4, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Air drift"), /*index*/ 77, /*count*/ 2, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Air drop"), /*index*/ 79, /*count*/ 2, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Ladder climb"), /*index*/ 81, /*count*/ 4, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Chi push"), /*index*/ 85, /*count*/ 8, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Leap slice A"), /*index*/ 93, /*count*/ 7, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Leap slice B"), /*index*/ 100, /*count*/ 3, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
{DQN_STRING8("Leap slice C"), /*index*/ 103, /*count*/ 6, /*ms_per_frame*/ DQN_CAST(uint32_t)(1000.f / 12.f)},
|
|
};
|
|
|
|
game->hero_sprite_anims = Dqn_Slice_Alloc<TELY_AssetSpriteAnimation>(&platform->arena, DQN_ARRAY_UCOUNT(hero_anims), Dqn_ZeroMem_No);
|
|
DQN_MEMCPY(game->hero_sprite_anims.data, &hero_anims, sizeof(hero_anims[0]) * DQN_ARRAY_UCOUNT(hero_anims));
|
|
}
|
|
|
|
// NOTE: Load sprite sheets ====================================================================
|
|
game->atlas_sprite_sheet = FP_LoadSpriteSheetFromSpec(platform, assets, &platform->arena, DQN_STRING8("atlas"));
|
|
FP_PlayReset(game, platform);
|
|
}
|
|
|
|
struct FP_GetClosestPortalMonkeyResult
|
|
{
|
|
FP_GameEntityHandle entity;
|
|
Dqn_f32 dist;
|
|
};
|
|
|
|
FP_GetClosestPortalMonkeyResult FP_GetClosestPortalMonkey(FP_Game *game, FP_GameEntityHandle handle)
|
|
{
|
|
// NOTE: Check if we are nearby a monkey and picking it up
|
|
FP_GetClosestPortalMonkeyResult result = {};
|
|
result.dist = DQN_F32_MAX;
|
|
FP_GameEntityHandle best_portal_monkey = {};
|
|
for (FP_GameEntityHandle portal_monkey_handle : game->play.portal_monkeys) {
|
|
FP_GameEntity *portal_monkey = FP_Game_GetEntity(game, portal_monkey_handle);
|
|
if (FP_Game_IsNilEntity(portal_monkey))
|
|
continue;
|
|
Dqn_V2 entity_pos = FP_Game_CalcEntityWorldPos(game, handle);
|
|
Dqn_V2 portal_monkey_pos = FP_Game_CalcEntityWorldPos(game, portal_monkey_handle);
|
|
Dqn_f32 dist_squared = Dqn_V2_LengthSq_V2x2(entity_pos, portal_monkey_pos);
|
|
if (dist_squared < result.dist) {
|
|
result.dist = dist_squared;
|
|
result.entity = portal_monkey_handle;
|
|
}
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void FP_EntityActionStateMachine(FP_Game *game, TELY_Audio *audio, TELY_PlatformInput *input, FP_GameEntity *entity, Dqn_V2 *acceleration_meters_per_s)
|
|
{
|
|
TELY_AssetSpriteSheet *sheet = &game->atlas_sprite_sheet;
|
|
FP_GameEntityAction *action = &entity->action;
|
|
bool const we_are_clicked_entity = entity->handle == game->play.clicked_entity;
|
|
bool const entity_has_velocity = entity->velocity.x || entity->velocity.y;
|
|
bool const entering_new_state = entity->alive_time_s == 0.f || action->state != action->next_state;
|
|
bool const action_has_finished = !entering_new_state && game->play.clock_ms >= action->end_at_clock_ms;
|
|
action->state = action->next_state;
|
|
|
|
FP_EntityRenderData render_data = FP_Entity_GetRenderData(game, entity->type, entity->action.state, entity->direction);
|
|
switch (entity->type) {
|
|
case FP_EntityType_Terry: {
|
|
FP_EntityTerryState *state = DQN_CAST(FP_EntityTerryState *) & action->state;
|
|
|
|
{
|
|
FP_GameEntity *portal_monkey = FP_Game_GetEntity(game, entity->carried_monkey);
|
|
if (!FP_Game_IsNilEntity(portal_monkey)) {
|
|
Dqn_Rect hit_box = FP_Game_CalcEntityWorldHitBox(game, entity->handle);
|
|
Dqn_f32 trig_t = DQN_CAST(Dqn_f32)input->timer_s * 2.f;
|
|
Dqn_V2 offset = Dqn_V2_InitNx2(DQN_COSF(trig_t), DQN_SINF(trig_t)) * 18.f;
|
|
portal_monkey->local_pos = Dqn_Rect_InterpolatedPoint(hit_box, Dqn_V2_InitNx2(0.5f, -0.75f)) + offset;
|
|
}
|
|
}
|
|
|
|
if (entity->hp <= 0 && *state != FP_EntityTerryState_DeadGhost) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_DeadGhost);
|
|
break;
|
|
}
|
|
|
|
switch (*state) {
|
|
case FP_EntityTerryState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (we_are_clicked_entity) {
|
|
if (game->play.in_game_menu == FP_GameInGameMenu_Nil) {
|
|
bool picked_up_monkey_this_frame = false;
|
|
if (FP_Game_IsNilEntityHandle(game, entity->carried_monkey)) {
|
|
// NOTE: Check if we are nearby a monkey and picking it up
|
|
FP_GetClosestPortalMonkeyResult closest_monkey = FP_GetClosestPortalMonkey(game, entity->handle);
|
|
if (closest_monkey.dist < DQN_SQUARED(FP_Game_MetersToPixelsNx1(game->play, 2.f))) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J)) {
|
|
entity->carried_monkey = closest_monkey.entity;
|
|
picked_up_monkey_this_frame = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (FP_Game_IsNilEntityHandle(game, entity->carried_monkey)) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J) ||
|
|
TELY_Platform_InputGamepadKeyIsPressed(input, 0, TELY_PlatformInputGamepadKey_X)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_Attack);
|
|
} else if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_K) ||
|
|
TELY_Platform_InputGamepadKeyIsPressed(input, 0, TELY_PlatformInputGamepadKey_Y)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_RangeAttack);
|
|
}
|
|
} else {
|
|
if (!picked_up_monkey_this_frame && TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J)) {
|
|
FP_GameEntity *portal_monkey = FP_Game_GetEntity(game, entity->carried_monkey);
|
|
portal_monkey->local_pos = FP_Game_CalcEntityWorldPos(game, entity->handle);
|
|
entity->carried_monkey = {};
|
|
}
|
|
}
|
|
}
|
|
|
|
if (action->next_state == action->state && (acceleration_meters_per_s->x || acceleration_meters_per_s->y)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_Run);
|
|
}
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityTerryState_Attack: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = render_data.sprite.anim->count * render_data.sprite.anim->ms_per_frame;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (action_has_finished) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_Idle);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityTerryState_RangeAttack: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = render_data.sprite.anim->count * render_data.sprite.anim->ms_per_frame;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
entity->terry_mobile_data_plan -= FP_TERRY_MOBILE_DATA_PER_RANGE_ATTACK;
|
|
}
|
|
|
|
if (action_has_finished) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_Idle);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityTerryState_Run: {
|
|
if (entering_new_state || action->sprite.anim->label != render_data.anim_name) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (we_are_clicked_entity) {
|
|
bool picked_up_monkey_this_frame = false;
|
|
if (game->play.in_game_menu == FP_GameInGameMenu_Nil) {
|
|
if (FP_Game_IsNilEntityHandle(game, entity->carried_monkey)) {
|
|
// NOTE: Check if we are nearby a monkey and picking it up
|
|
FP_GetClosestPortalMonkeyResult closest_monkey = FP_GetClosestPortalMonkey(game, entity->handle);
|
|
if (closest_monkey.dist < DQN_SQUARED(FP_Game_MetersToPixelsNx1(game->play, 2.f))) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J)) {
|
|
entity->carried_monkey = closest_monkey.entity;
|
|
picked_up_monkey_this_frame = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (FP_Game_IsNilEntityHandle(game, entity->carried_monkey)) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J) ||
|
|
TELY_Platform_InputGamepadKeyIsPressed(input, 0, TELY_PlatformInputGamepadKey_X)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_Attack);
|
|
} else if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_K) ||
|
|
TELY_Platform_InputGamepadKeyIsPressed(input, 0, TELY_PlatformInputGamepadKey_Y)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_RangeAttack);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (FP_Game_IsNilEntityHandle(game, entity->carried_monkey)) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_LeftControl) ||
|
|
TELY_Platform_InputGamepadKeyIsPressed(input, 0, TELY_PlatformInputGamepadKey_A)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_Dash);
|
|
}
|
|
} else {
|
|
if (!picked_up_monkey_this_frame && TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J)) {
|
|
FP_GameEntity *portal_monkey = FP_Game_GetEntity(game, entity->carried_monkey);
|
|
portal_monkey->local_pos = FP_Game_CalcEntityWorldPos(game, entity->handle);
|
|
entity->carried_monkey = {};
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!entity_has_velocity) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_Idle);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityTerryState_Dash: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = 250;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
*acceleration_meters_per_s *= 35.f;
|
|
|
|
entity->stamina -= FP_TERRY_DASH_STAMINA_COST;
|
|
TELY_Audio_Play(audio, game->audio[FP_GameAudio_Woosh], 1.f);
|
|
|
|
#if 0
|
|
FP_GameRenderSprite *cosmetic_sprite = Dqn_FArray_Make(&entity->extra_cosmetic_anims, Dqn_ZeroMem_Yes);
|
|
if (cosmetic_sprite) {
|
|
cosmetic_sprite->asset = TELY_Asset_MakeAnimatedSprite(sheet, g_anim_names.terry_ghost, TELY_AssetFlip_No);
|
|
cosmetic_sprite->started_at_clock_ms = game->play.clock_ms;
|
|
cosmetic_sprite->height.meters = entity->sprite_height.meters;
|
|
|
|
uint32_t max_rng_dist_x = DQN_CAST(uint32_t)(FP_Game_MetersToPixelsNx1(game, entity->sprite_height.meters) * 1.f);
|
|
uint32_t rng_x = Dqn_PCG32_Range(&game->play.rng, DQN_CAST(uint32_t)0, max_rng_dist_x);
|
|
cosmetic_sprite->offset.x = rng_x - (max_rng_dist_x * .5f);
|
|
|
|
uint32_t max_rng_dist_y = DQN_CAST(uint32_t)(FP_Game_MetersToPixelsNx1(game, entity->sprite_height.meters) * .25f);
|
|
uint32_t rng_y = Dqn_PCG32_Range(&game->play.rng, DQN_CAST(uint32_t)0, max_rng_dist_y);
|
|
cosmetic_sprite->offset.y = -DQN_CAST(Dqn_f32)rng_y;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
entity->action.sprite_alpha = Dqn_PCG32_NextF32(&game->play.rng);
|
|
|
|
if (action_has_finished) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_Run);
|
|
entity->action.sprite_alpha = 1.f;
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityTerryState_DeadGhost: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
entity->action.sprite_play_once = true;
|
|
entity->faction = FP_GameEntityFaction_Nil;
|
|
}
|
|
|
|
Dqn_f32 hp_t = entity->hp / DQN_CAST(Dqn_f32)entity->hp_cap;
|
|
if (hp_t >= 0.9f)
|
|
entity->action.sprite_alpha = Dqn_PCG32_NextF32(&game->play.rng);
|
|
|
|
if (entity->hp >= entity->hp_cap) {
|
|
entity->faction = FP_GameEntityFaction_Friendly;
|
|
entity->action.sprite_alpha = 1.f;
|
|
entity->action.sprite_play_once = false;
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_Idle);
|
|
}
|
|
} break;
|
|
}
|
|
} break;
|
|
|
|
|
|
case FP_EntityType_Smoochie: {
|
|
FP_EntitySmoochieState *state = DQN_CAST(FP_EntitySmoochieState *) & action->state;
|
|
switch (*state) {
|
|
case FP_EntitySmoochieState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (we_are_clicked_entity) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J) ||
|
|
TELY_Platform_InputGamepadKeyIsPressed(input, 0, TELY_PlatformInputGamepadKey_X)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntitySmoochieState_Attack);
|
|
} else if (acceleration_meters_per_s->x || acceleration_meters_per_s->y) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntitySmoochieState_Run);
|
|
}
|
|
}
|
|
|
|
if (entity_has_velocity) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntitySmoochieState_Run);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntitySmoochieState_Attack: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = render_data.sprite.anim->count * render_data.sprite.anim->ms_per_frame;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
|
|
// NOTE: Deal with this further down with the gameplay attack code.
|
|
TELY_Audio_Play(audio, game->audio[FP_GameAudio_Smooch], 1.f);
|
|
}
|
|
|
|
// NOTE: Check if the heart animation is playing
|
|
bool has_heart_flourish = false;
|
|
DQN_FOR_UINDEX(anim_index, entity->extra_cosmetic_anims.size) {
|
|
FP_GameRenderSprite *sprite = entity->extra_cosmetic_anims.data + anim_index;
|
|
if (sprite->asset.anim->label == g_anim_names.smoochie_attack_heart) {
|
|
has_heart_flourish = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// NOTE: If we don't have the anim playing make one
|
|
if (!has_heart_flourish) {
|
|
FP_GameRenderSprite *cosmetic_sprite = Dqn_FArray_Make(&entity->extra_cosmetic_anims, Dqn_ZeroMem_Yes);
|
|
if (cosmetic_sprite) {
|
|
cosmetic_sprite->asset = TELY_Asset_MakeAnimatedSprite(sheet, g_anim_names.smoochie_attack_heart, TELY_AssetFlip_No);
|
|
cosmetic_sprite->started_at_clock_ms = game->play.clock_ms;
|
|
cosmetic_sprite->height.meters = entity->sprite_height.meters * .35f;
|
|
cosmetic_sprite->loop = true;
|
|
|
|
uint32_t max_rng_dist_x = DQN_CAST(uint32_t)(FP_Game_MetersToPixelsNx1(game->play, entity->sprite_height.meters) * 1.f);
|
|
uint32_t rng_x = Dqn_PCG32_Range(&game->play.rng, DQN_CAST(uint32_t)0, max_rng_dist_x);
|
|
cosmetic_sprite->offset.x = rng_x - (max_rng_dist_x * .5f);
|
|
|
|
uint32_t max_rng_dist_y = DQN_CAST(uint32_t)(FP_Game_MetersToPixelsNx1(game->play, entity->sprite_height.meters) * .25f);
|
|
uint32_t rng_y = Dqn_PCG32_Range(&game->play.rng, DQN_CAST(uint32_t)0, max_rng_dist_y);
|
|
cosmetic_sprite->offset.y = -DQN_CAST(Dqn_f32)rng_y;
|
|
}
|
|
}
|
|
|
|
if (action_has_finished) {
|
|
// NOTE: Ensure the heart animation terminates by removing the loop
|
|
DQN_FOR_UINDEX(anim_index, entity->extra_cosmetic_anims.size) {
|
|
FP_GameRenderSprite *sprite = entity->extra_cosmetic_anims.data + anim_index;
|
|
if (sprite->asset.anim->label == g_anim_names.smoochie_attack_heart)
|
|
sprite->loop = false;
|
|
}
|
|
|
|
// NOTE: Transition out of the action
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntitySmoochieState_Idle);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntitySmoochieState_HurtSide: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = render_data.sprite.anim->count * render_data.sprite.anim->ms_per_frame;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (action_has_finished)
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntitySmoochieState_Idle);
|
|
} break;
|
|
|
|
case FP_EntitySmoochieState_Death: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = render_data.sprite.anim->count * render_data.sprite.anim->ms_per_frame;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
entity->local_hit_box_size = {};
|
|
}
|
|
|
|
if (action_has_finished)
|
|
FP_Game_DeleteEntity(game, entity->handle);
|
|
} break;
|
|
|
|
|
|
case FP_EntitySmoochieState_Run: {
|
|
if (entering_new_state || action->sprite.anim->label != render_data.anim_name) {
|
|
TELY_AssetAnimatedSprite sprite = TELY_Asset_MakeAnimatedSprite(sheet, render_data.anim_name, render_data.flip);
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, sprite);
|
|
}
|
|
|
|
if (we_are_clicked_entity) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J) ||
|
|
TELY_Platform_InputGamepadKeyIsPressed(input, 0, TELY_PlatformInputGamepadKey_X)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntitySmoochieState_Attack);
|
|
}
|
|
}
|
|
|
|
if (!entity_has_velocity) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntitySmoochieState_Idle);
|
|
}
|
|
} break;
|
|
}
|
|
|
|
if (entity->is_dying && *state != FP_EntitySmoochieState_Death) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntitySmoochieState_Death);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_Clinger: {
|
|
FP_EntityClingerState *state = DQN_CAST(FP_EntityClingerState *)&action->state;
|
|
switch (*state) {
|
|
case FP_EntityClingerState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (we_are_clicked_entity) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J) ||
|
|
TELY_Platform_InputGamepadKeyIsPressed(input, 0, TELY_PlatformInputGamepadKey_X)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityClingerState_Attack);
|
|
} else if (acceleration_meters_per_s->x || acceleration_meters_per_s->y) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityClingerState_Run);
|
|
}
|
|
}
|
|
|
|
if (entity_has_velocity) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityClingerState_Run);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityClingerState_Attack: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = render_data.sprite.anim->count * render_data.sprite.anim->ms_per_frame;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (action_has_finished)
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityClingerState_Idle);
|
|
} break;
|
|
|
|
case FP_EntityClingerState_Death: {
|
|
if (entering_new_state) {
|
|
TELY_AssetAnimatedSprite sprite = TELY_Asset_MakeAnimatedSprite(sheet, g_anim_names.clinger_death, TELY_AssetFlip_No);
|
|
uint64_t duration_ms = sprite.anim->count * sprite.anim->ms_per_frame;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, sprite);
|
|
entity->local_hit_box_size = {};
|
|
}
|
|
|
|
if (action_has_finished)
|
|
FP_Game_DeleteEntity(game, entity->handle);
|
|
|
|
} break;
|
|
|
|
case FP_EntityClingerState_Run: {
|
|
if (entering_new_state || action->sprite.anim->label != render_data.anim_name) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (we_are_clicked_entity) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J) ||
|
|
TELY_Platform_InputGamepadKeyIsPressed(input, 0, TELY_PlatformInputGamepadKey_X)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityClingerState_Attack);
|
|
}
|
|
}
|
|
|
|
if (!entity_has_velocity) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityClingerState_Idle);
|
|
}
|
|
} break;
|
|
}
|
|
|
|
if (entity->is_dying && *state != FP_EntityClingerState_Death) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityClingerState_Death);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_MerchantTerry: {
|
|
FP_EntityMerchantTerryState *state = DQN_CAST(FP_EntityMerchantTerryState *)&action->state;
|
|
switch (*state) {
|
|
case FP_EntityMerchantTerryState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
} break;
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_MerchantPhoneCompany: {
|
|
FP_EntityMerchantPhoneCompanyState *state = DQN_CAST(FP_EntityMerchantPhoneCompanyState *)&action->state;
|
|
switch (*state) {
|
|
case FP_EntityMerchantPhoneCompanyState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
} break;
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_MerchantGym: {
|
|
FP_EntityMerchantGymState *state = DQN_CAST(FP_EntityMerchantGymState *)&action->state;
|
|
switch (*state) {
|
|
case FP_EntityMerchantGymState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
} break;
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_MerchantGraveyard: {
|
|
FP_EntityMerchantGraveyardState *state = DQN_CAST(FP_EntityMerchantGraveyardState *)&action->state;
|
|
switch (*state) {
|
|
case FP_EntityMerchantGraveyardState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
} break;
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_ClubTerry: {
|
|
FP_EntityClubTerryState *state = DQN_CAST(FP_EntityClubTerryState *)&action->state;
|
|
switch (*state) {
|
|
case FP_EntityClubTerryState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityClubTerryState_PartyTime: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = 5000;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (action_has_finished) {
|
|
if (!FP_Game_IsNilEntityHandle(game, entity->building_patron)) {
|
|
FP_GameEntity *patron = FP_Game_GetEntity(game, entity->building_patron);
|
|
patron->flags &= ~FP_GameEntityFlag_OccupiedInBuilding;
|
|
patron->base_acceleration_per_s.meters *= .5f;
|
|
}
|
|
entity->building_patron = {};
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityClubTerryState_Idle);
|
|
}
|
|
} break;
|
|
}
|
|
|
|
} break;
|
|
|
|
case FP_EntityType_Map: {
|
|
FP_EntityMapState *state = DQN_CAST(FP_EntityMapState *) & action->state;
|
|
switch (*state) {
|
|
case FP_EntityMapState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
} break;
|
|
}
|
|
|
|
} break;
|
|
|
|
case FP_EntityType_Heart: {
|
|
FP_EntityHeartState *state = DQN_CAST(FP_EntityHeartState *) & action->state;
|
|
switch (*state) {
|
|
case FP_EntityHeartState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
} break;
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_AirportTerry: {
|
|
FP_EntityAirportTerryState *state = DQN_CAST(FP_EntityAirportTerryState *)&action->state;
|
|
switch (*state) {
|
|
case FP_EntityAirportTerryState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityAirportTerryState_FlyPassenger: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = 5000;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (action_has_finished) {
|
|
Dqn_V2 world_pos = FP_Game_CalcEntityWorldPos(game, entity->handle);
|
|
FP_GameEntityHandle plane_handle = FP_Entity_CreateAirportTerryPlane(game, world_pos, "Aiport Terry Plane");
|
|
FP_GameEntity *plane = FP_Game_GetEntity(game, plane_handle);
|
|
|
|
// NOTE: Transfer the entity to the plane
|
|
plane->building_patron = entity->building_patron;
|
|
entity->building_patron = {};
|
|
|
|
// NOTE: Add a waypoint for the plane to the mob spawn
|
|
{
|
|
uint32_t mob_spawner_index = Dqn_PCG32_Range(&game->play.rng, 0, DQN_CAST(uint32_t)game->play.mob_spawners.size);
|
|
FP_GameEntityHandle mob_spawner = game->play.mob_spawners.data[mob_spawner_index];
|
|
Dqn_V2 target_pos = FP_Game_CalcEntityWorldPos(game, mob_spawner);
|
|
plane->waypoints = FP_SentinelList_Init<FP_GameWaypoint>(game->play.chunk_pool);
|
|
FP_SentinelListLink<FP_GameWaypoint> *link = FP_SentinelList_MakeBefore(&plane->waypoints,
|
|
FP_SentinelList_Front(&plane->waypoints),
|
|
game->play.chunk_pool);
|
|
FP_GameWaypoint *waypoint = &link->data;
|
|
waypoint->entity = mob_spawner;
|
|
waypoint->type = FP_GameWaypointType_ClosestSide;
|
|
}
|
|
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityAirportTerryState_Idle);
|
|
}
|
|
} break;
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_AirportTerryPlane: {
|
|
FP_EntityAirportTerryPlaneState *state = DQN_CAST(FP_EntityAirportTerryPlaneState *)&action->state;
|
|
switch (*state) {
|
|
case FP_EntityAirportTerryPlaneState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (!FP_Game_IsNilEntityHandle(game, entity->building_patron))
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityAirportTerryPlaneState_FlyPassenger);
|
|
} break;
|
|
|
|
case FP_EntityAirportTerryPlaneState_FlyPassenger: {
|
|
if (entity->waypoints.size == 0) {
|
|
FP_GameEntity *patron = FP_Game_GetEntity(game, entity->building_patron);
|
|
patron->local_pos = entity->local_pos;
|
|
patron->flags &= ~(FP_GameEntityFlag_OccupiedInBuilding);
|
|
FP_Game_DeleteEntity(game, entity->handle);
|
|
return;
|
|
}
|
|
} break;
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_Catfish: {
|
|
FP_EntityCatfishState *state = DQN_CAST(FP_EntityCatfishState *) & action->state;
|
|
switch (*state) {
|
|
case FP_EntityCatfishState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (we_are_clicked_entity) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J) ||
|
|
TELY_Platform_InputGamepadKeyIsPressed(input, 0, TELY_PlatformInputGamepadKey_X)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityCatfishState_Attack);
|
|
} else if (acceleration_meters_per_s->x || acceleration_meters_per_s->y) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityCatfishState_Run);
|
|
}
|
|
}
|
|
|
|
if (entity_has_velocity) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityCatfishState_Run);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityCatfishState_Attack: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = render_data.sprite.anim->count * render_data.sprite.anim->ms_per_frame;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (action_has_finished)
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityCatfishState_Idle);
|
|
} break;
|
|
|
|
case FP_EntityCatfishState_Death: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = render_data.sprite.anim->count * render_data.sprite.anim->ms_per_frame;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
entity->local_hit_box_size = {};
|
|
}
|
|
|
|
if (action_has_finished)
|
|
FP_Game_DeleteEntity(game, entity->handle);
|
|
} break;
|
|
|
|
case FP_EntityCatfishState_Run: {
|
|
if (entering_new_state || action->sprite.anim->label != render_data.anim_name) {
|
|
TELY_AssetAnimatedSprite sprite = TELY_Asset_MakeAnimatedSprite(sheet, render_data.anim_name, render_data.flip);
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, sprite);
|
|
}
|
|
|
|
if (we_are_clicked_entity) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J) ||
|
|
TELY_Platform_InputGamepadKeyIsPressed(input, 0, TELY_PlatformInputGamepadKey_X)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityCatfishState_Attack);
|
|
}
|
|
}
|
|
|
|
if (!entity_has_velocity) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityCatfishState_Idle);
|
|
}
|
|
} break;
|
|
}
|
|
|
|
if (entity->is_dying && *state != FP_EntityCatfishState_Death) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityCatfishState_Death);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_ChurchTerry: {
|
|
FP_EntityChurchTerryState *state = DQN_CAST(FP_EntityChurchTerryState *)&action->state;
|
|
switch (*state) {
|
|
case FP_EntityChurchTerryState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityChurchTerryState_ConvertPatron: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = 5000;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (action_has_finished) {
|
|
if (!FP_Game_IsNilEntityHandle(game, entity->building_patron)) {
|
|
FP_GameEntity *patron = FP_Game_GetEntity(game, entity->building_patron);
|
|
patron->flags &= ~(FP_GameEntityFlag_OccupiedInBuilding | FP_GameEntityFlag_RespondsToBuildings);
|
|
patron->faction = FP_GameEntityFaction_Friendly;
|
|
patron->converted_faction = true;
|
|
}
|
|
entity->building_patron = {};
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityChurchTerryState_Idle);
|
|
}
|
|
} break;
|
|
}
|
|
|
|
} break;
|
|
|
|
case FP_EntityType_KennelTerry: {
|
|
FP_EntityKennelTerryState *state = DQN_CAST(FP_EntityKennelTerryState *)&action->state;
|
|
switch (*state) {
|
|
case FP_EntityKennelTerryState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
} break;
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_Nil: break;
|
|
|
|
case FP_EntityType_Count: DQN_INVALID_CODE_PATH; break;
|
|
case FP_EntityType_PhoneMessageProjectile: break;
|
|
|
|
case FP_EntityType_MobSpawner: {
|
|
FP_EntityMobSpawnerState *state = DQN_CAST(FP_EntityMobSpawnerState *)&action->state;
|
|
switch (*state) {
|
|
case FP_EntityMobSpawnerState_Idle: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
}
|
|
|
|
if (!FP_Game_IsNilEntityHandle(game, entity->carried_monkey)) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityMobSpawnerState_Shutdown);
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityMobSpawnerState_Shutdown: {
|
|
if (entering_new_state) {
|
|
uint64_t duration_ms = FP_GAME_ENTITY_ACTION_INFINITE_TIMER;
|
|
FP_Game_EntityActionReset(game, entity->handle, duration_ms, render_data.sprite);
|
|
entity->action.sprite_play_once = true;
|
|
FP_Game_DeleteEntity(game, entity->carried_monkey);
|
|
}
|
|
} break;
|
|
}
|
|
} break;
|
|
}
|
|
|
|
switch (entity->type) {
|
|
case FP_EntityType_Nil:
|
|
case FP_EntityType_AirportTerry:
|
|
case FP_EntityType_ChurchTerry:
|
|
case FP_EntityType_ClubTerry:
|
|
case FP_EntityType_Heart:
|
|
case FP_EntityType_KennelTerry:
|
|
case FP_EntityType_Map:
|
|
case FP_EntityType_MerchantGraveyard:
|
|
case FP_EntityType_MerchantGym:
|
|
case FP_EntityType_MerchantPhoneCompany:
|
|
case FP_EntityType_MerchantTerry:
|
|
case FP_EntityType_PhoneMessageProjectile:
|
|
case FP_EntityType_Count: break;
|
|
|
|
case FP_EntityType_Terry: /*FALLTHRU*/
|
|
case FP_EntityType_Catfish: /*FALLTHRU*/
|
|
case FP_EntityType_Clinger: /*FALLTHRU*/
|
|
case FP_EntityType_Smoochie: {
|
|
|
|
bool is_attacking = false;
|
|
bool is_range_attack = false;
|
|
if (entity->type == FP_EntityType_Catfish) {
|
|
is_attacking = entity->action.state == FP_EntityCatfishState_Attack;
|
|
} else if (entity->type == FP_EntityType_Clinger) {
|
|
is_attacking = entity->action.state == FP_EntityClingerState_Attack;
|
|
} else if (entity->type == FP_EntityType_Smoochie) {
|
|
is_attacking = entity->action.state == FP_EntitySmoochieState_Attack;
|
|
} else {
|
|
DQN_ASSERT(entity->type == FP_EntityType_Terry);
|
|
is_range_attack = entity->action.state == FP_EntityTerryState_RangeAttack;
|
|
is_attacking = is_range_attack || entity->action.state == FP_EntityTerryState_Attack;
|
|
}
|
|
|
|
if (!is_attacking) {
|
|
entity->attack_processed = false;
|
|
entity->attack_box_size = {};
|
|
break;
|
|
}
|
|
|
|
// NOTE: Position the attack box
|
|
uint64_t duration_ms = action->sprite.anim->count * action->sprite.anim->ms_per_frame;
|
|
uint64_t midpoint_clock_ms = action->started_at_clock_ms + (duration_ms / 2);
|
|
DQN_ASSERT(duration_ms >= PHYSICS_STEP);
|
|
DQN_ASSERT(action->sprite.anim);
|
|
|
|
// NOTE: Adding an attack_processed bool to make sure things only fire once
|
|
if (!entity->attack_processed && game->play.clock_ms >= midpoint_clock_ms) {
|
|
|
|
// NOTE: Position the attack box
|
|
if (is_range_attack) {
|
|
Dqn_V2 dir_vector = {};
|
|
switch (entity->direction) {
|
|
case FP_GameDirection_Left: dir_vector.x = -1.f; break;
|
|
case FP_GameDirection_Right: dir_vector.x = +1.f; break;
|
|
case FP_GameDirection_Up: dir_vector.y = -1.f; break;
|
|
case FP_GameDirection_Down: dir_vector.y = +1.f; break;
|
|
case FP_GameDirection_Count: break;
|
|
}
|
|
|
|
Dqn_V2 projectile_pos = FP_Game_CalcEntityWorldPos(game, entity->handle);
|
|
Dqn_V2 projectile_acceleration = FP_Game_MetersToPixelsV2(game->play, dir_vector * 0.25f);
|
|
FP_Entity_CreatePhoneMessageProjectile(game,
|
|
entity->handle,
|
|
projectile_pos,
|
|
projectile_acceleration,
|
|
"Phone Message Projectile");
|
|
TELY_Audio_Play(audio, game->audio[FP_GameAudio_Message], 1.f /*volume*/);
|
|
} else {
|
|
Dqn_FArray<Dqn_Rect, FP_GameDirection_Count> attack_boxes = FP_Game_CalcEntityMeleeAttackBoxes(game, entity->handle);
|
|
entity->attack_box_size = attack_boxes.data[entity->direction].size;
|
|
entity->attack_box_offset = attack_boxes.data[entity->direction].pos - FP_Game_CalcEntityWorldPos(game, entity->handle);
|
|
TELY_Audio_Play(audio, game->audio[FP_GameAudio_TerryHit], 1.f);
|
|
}
|
|
entity->attack_processed = true;
|
|
} else {
|
|
entity->attack_box_size = {};
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_AirportTerryPlane: break;
|
|
}
|
|
}
|
|
|
|
void FP_Update(TELY_Platform *platform, FP_Game *game, TELY_PlatformInput *input, TELY_Audio *audio)
|
|
{
|
|
Dqn_Profiler_ZoneScopeWithIndex("FP_Update", FP_ProfileZone_FPUpdate);
|
|
|
|
game->play.update_counter++;
|
|
game->play.clock_ms = DQN_CAST(uint64_t)(platform->input.timer_s * 1000.f);
|
|
Dqn_ProfilerZone update_zone = Dqn_Profiler_BeginZoneWithIndex(DQN_STRING8("FP_Update: Entity loop"), FP_ProfileZone_FPUpdate_EntityLoop);
|
|
if (game->play.state == FP_GameState_Play) {
|
|
Dqn_V2 dir_vector = {};
|
|
if (TELY_Platform_InputKeyIsReleased(input->mouse_left))
|
|
game->play.clicked_entity = game->play.prev_active_entity;
|
|
|
|
// NOTE: Keyboard movement input
|
|
if (TELY_Platform_InputScanCodeIsDown(input, TELY_PlatformInputScanCode_W))
|
|
dir_vector.y = -1.f;
|
|
if (TELY_Platform_InputScanCodeIsDown(input, TELY_PlatformInputScanCode_A))
|
|
dir_vector.x = -1.f;
|
|
if (TELY_Platform_InputScanCodeIsDown(input, TELY_PlatformInputScanCode_S))
|
|
dir_vector.y = +1.f;
|
|
if (TELY_Platform_InputScanCodeIsDown(input, TELY_PlatformInputScanCode_D))
|
|
dir_vector.x = +1.f;
|
|
|
|
// NOTE: Gamepad movement input
|
|
// NOTE: button_codes 0 should be the first gamepad connected, we can
|
|
// get this working with other gamepads later
|
|
uint32_t gamepad = 0;
|
|
if (input->button_codes[gamepad]) {
|
|
dir_vector.x += input->left_stick[gamepad].x;
|
|
dir_vector.y += input->left_stick[gamepad].y;
|
|
}
|
|
|
|
// TODO(doyle): Some bug, diagonal is still faster
|
|
if (dir_vector.x && dir_vector.y) {
|
|
dir_vector.x *= 0.7071067811865475244f;
|
|
dir_vector.y *= 0.7071067811865475244f;
|
|
}
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_Escape))
|
|
game->play.state = FP_GameState_Pause;
|
|
|
|
if (game->play.clicked_entity.id) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_Delete))
|
|
FP_Game_DeleteEntity(game, game->play.clicked_entity);
|
|
|
|
// NOTE: Building selector
|
|
Dqn_usize last_building_index = DQN_ARRAY_UCOUNT(PLACEABLE_BUILDINGS) - 1;
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_Q)) {
|
|
if (game->play.build_mode_building_index <= 0) {
|
|
game->play.build_mode_building_index = last_building_index;
|
|
} else {
|
|
game->play.build_mode_building_index -= 1;
|
|
}
|
|
}
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_E)) {
|
|
if (game->play.build_mode_building_index >= last_building_index) {
|
|
game->play.build_mode_building_index = 0;
|
|
} else {
|
|
game->play.build_mode_building_index += 1;
|
|
}
|
|
}
|
|
|
|
} else {
|
|
Dqn_f32 pan_speed = 5.f;
|
|
if (TELY_Platform_InputScanCodeIsDown(input, TELY_PlatformInputScanCode_Space))
|
|
pan_speed *= 2.5f;
|
|
game->play.camera.world_pos += dir_vector * pan_speed;
|
|
}
|
|
|
|
// NOTE: Handle input ==========================================================================
|
|
for (FP_GameEntityIterator it = {}; FP_Game_DFSPostOrderWalkEntityTree(game, &it, game->play.root_entity); ) {
|
|
FP_GameEntity *entity = it.entity;
|
|
entity->alive_time_s += PHYSICS_STEP;
|
|
|
|
if (entity->flags & FP_GameEntityFlag_OccupiedInBuilding)
|
|
continue;
|
|
|
|
// NOTE: Move entity by keyboard and gamepad ===============================================
|
|
Dqn_V2 acceleration_meters_per_s = entity->constant_acceleration_per_s;
|
|
if (game->play.clicked_entity == entity->handle) {
|
|
if (entity->flags & (FP_GameEntityFlag_MoveByKeyboard | FP_GameEntityFlag_MoveByGamepad)) {
|
|
bool move_entity = true;
|
|
switch (entity->type) {
|
|
case FP_EntityType_Terry: {
|
|
auto *state = DQN_CAST(FP_EntityTerryState *)&entity->action.state;
|
|
move_entity = *state == FP_EntityTerryState_Run || *state == FP_EntityTerryState_Idle;
|
|
if (*state == FP_EntityTerryState_DeadGhost) {
|
|
FP_GameEntityAction const *action = &entity->action;
|
|
uint64_t const elapsed_ms = game->play.clock_ms - action->started_at_clock_ms;
|
|
uint16_t const raw_anim_frame = DQN_CAST(uint16_t)(elapsed_ms / action->sprite.anim->ms_per_frame);
|
|
if (raw_anim_frame >= action->sprite.anim->count) {
|
|
move_entity = true;
|
|
}
|
|
}
|
|
} break;
|
|
|
|
case FP_EntityType_Smoochie: {
|
|
auto *state = DQN_CAST(FP_EntitySmoochieState *)&entity->action.state;
|
|
move_entity = *state == FP_EntitySmoochieState_Run || *state == FP_EntitySmoochieState_Idle;
|
|
} break;
|
|
|
|
case FP_EntityType_Clinger: {
|
|
auto *state = DQN_CAST(FP_EntityClingerState *)&entity->action.state;
|
|
move_entity = *state == FP_EntityClingerState_Run || *state == FP_EntityClingerState_Idle;
|
|
} break;
|
|
|
|
case FP_EntityType_Nil: break;
|
|
case FP_EntityType_ClubTerry: break;
|
|
case FP_EntityType_Count: break;
|
|
case FP_EntityType_Map: break;
|
|
case FP_EntityType_MerchantTerry: break;
|
|
case FP_EntityType_MerchantGraveyard: break;
|
|
case FP_EntityType_MerchantPhoneCompany: break;
|
|
case FP_EntityType_Heart: break;
|
|
case FP_EntityType_MerchantGym: break;
|
|
case FP_EntityType_AirportTerry: break;
|
|
case FP_EntityType_Catfish: break;
|
|
case FP_EntityType_ChurchTerry: break;
|
|
case FP_EntityType_KennelTerry: break;
|
|
case FP_EntityType_PhoneMessageProjectile: break;
|
|
case FP_EntityType_AirportTerryPlane: break;
|
|
}
|
|
|
|
if (move_entity) {
|
|
acceleration_meters_per_s = dir_vector * entity->base_acceleration_per_s.meters;
|
|
if (TELY_Platform_InputScanCodeIsDown(input, TELY_PlatformInputScanCode_Space))
|
|
acceleration_meters_per_s *= 2.5f;
|
|
}
|
|
}
|
|
} else {
|
|
if (entity->velocity.x)
|
|
entity->direction = entity->velocity.x > 0.f ? FP_GameDirection_Right : FP_GameDirection_Left;
|
|
else if (entity->velocity.y)
|
|
entity->direction = entity->velocity.y > 0.f ? FP_GameDirection_Down : FP_GameDirection_Up;
|
|
}
|
|
|
|
// NOTE: Determine AI movement =============================================================
|
|
Dqn_V2 entity_pos = FP_Game_CalcEntityWorldPos(game, entity->handle);
|
|
if (entity->flags & FP_GameEntityFlag_Aggros && entity->faction != FP_GameEntityFaction_Nil) {
|
|
FP_GameFindClosestEntityResult closest_defender = {};
|
|
closest_defender.dist_squared = DQN_F32_MAX;
|
|
closest_defender.pos = Dqn_V2_InitNx1(DQN_F32_MAX);
|
|
|
|
FP_GameEntityFaction enemy_faction =
|
|
entity->faction == FP_GameEntityFaction_Friendly
|
|
? FP_GameEntityFaction_Foe
|
|
: FP_GameEntityFaction_Friendly;
|
|
|
|
for (FP_GameEntityIterator defender_it = {}; FP_Game_DFSPostOrderWalkEntityTree(game, &defender_it, game->play.root_entity); ) {
|
|
FP_GameEntity *it_entity = defender_it.entity;
|
|
Dqn_V2 pos = FP_Game_CalcEntityWorldPos(game, it_entity->handle);
|
|
Dqn_f32 dist = Dqn_V2_LengthSq_V2x2(pos, entity_pos);
|
|
|
|
if (it_entity->faction != enemy_faction)
|
|
continue;
|
|
|
|
if (dist < closest_defender.dist_squared) {
|
|
closest_defender.pos = pos;
|
|
closest_defender.dist_squared = dist;
|
|
closest_defender.entity = it_entity->handle;
|
|
}
|
|
}
|
|
|
|
Dqn_f32 aggro_dist_threshold = FP_Game_MetersToPixelsNx1(game->play, 4.f);
|
|
Dqn_f32 dist_to_defender = DQN_SQRTF(closest_defender.dist_squared);
|
|
if (dist_to_defender > (aggro_dist_threshold * 1.5f)) {
|
|
for (FP_SentinelListLink<FP_GameWaypoint> *link = nullptr; FP_SentinelList_Iterate<FP_GameWaypoint>(&entity->waypoints, &link); ) {
|
|
FP_GameEntity *maybe_terry = FP_Game_GetEntity(game, link->data.entity);
|
|
if (maybe_terry->type == FP_EntityType_Terry) {
|
|
link = FP_SentinelList_Erase(&entity->waypoints, link, game->play.chunk_pool);
|
|
}
|
|
}
|
|
} else if (dist_to_defender < aggro_dist_threshold) {
|
|
bool has_waypoint_to_defender = false;
|
|
for (FP_SentinelListLink<FP_GameWaypoint> *link = nullptr;
|
|
!has_waypoint_to_defender && FP_SentinelList_Iterate<FP_GameWaypoint>(&entity->waypoints, &link); ) {
|
|
has_waypoint_to_defender = link->data.entity == closest_defender.entity;
|
|
}
|
|
|
|
if (!has_waypoint_to_defender) {
|
|
FP_GameEntity *defender = FP_Game_GetEntity(game, closest_defender.entity);
|
|
FP_SentinelListLink<FP_GameWaypoint> *link = FP_SentinelList_MakeBefore(&entity->waypoints,
|
|
FP_SentinelList_Front(&entity->waypoints),
|
|
game->play.chunk_pool);
|
|
FP_GameWaypoint *waypoint = &link->data;
|
|
waypoint->entity = defender->handle;
|
|
waypoint->type = FP_GameWaypointType_ClosestSide;
|
|
}
|
|
}
|
|
}
|
|
|
|
// NOTE: Make waypoint to building =====================================================
|
|
// We can queue up to ente a building if we respond to the building AND we aren't
|
|
// already queuing up in a building already.
|
|
if (entity->flags & FP_GameEntityFlag_RespondsToBuildings && FP_Game_IsNilEntityHandle(game, entity->queued_at_building)) {
|
|
FP_GameFindClosestEntityResult closest_building = {};
|
|
closest_building.dist_squared = DQN_F32_MAX;
|
|
closest_building.pos = Dqn_V2_InitNx1(DQN_F32_MAX);
|
|
|
|
for (FP_GameEntityIterator building_it = {}; FP_Game_DFSPostOrderWalkEntityTree(game, &building_it, game->play.root_entity); ) {
|
|
FP_GameEntity *it_entity = building_it.entity;
|
|
if (it_entity->type != FP_EntityType_ClubTerry &&
|
|
it_entity->type != FP_EntityType_AirportTerry &&
|
|
it_entity->type != FP_EntityType_ChurchTerry)
|
|
continue;
|
|
|
|
// NOTE: Already converted, we cannot attend church again
|
|
if (entity->converted_faction && it_entity->type == FP_EntityType_ChurchTerry)
|
|
continue;
|
|
|
|
// NOTE: The queue to enter the building is completely full skip
|
|
if (it_entity->building_queue.size == Dqn_FArray_Max(&it_entity->building_queue))
|
|
continue;
|
|
|
|
// NOTE: Entity is already in the building queue, skip
|
|
if (Dqn_FArray_Find<FP_GameEntityHandle>(&it_entity->building_queue, entity->handle).data)
|
|
continue;
|
|
|
|
bool already_visited_building = false;
|
|
for (FP_SentinelListLink<FP_GameEntityHandle> *link_it = {};
|
|
!already_visited_building && FP_SentinelList_Iterate(&entity->buildings_visited, &link_it);
|
|
) {
|
|
FP_GameEntityHandle visit_item = link_it->data;
|
|
already_visited_building = visit_item == it_entity->handle;
|
|
}
|
|
|
|
if (already_visited_building)
|
|
continue;
|
|
|
|
Dqn_V2 pos = FP_Game_CalcEntityWorldPos(game, it_entity->handle);
|
|
Dqn_f32 dist = Dqn_V2_LengthSq_V2x2(pos, entity_pos);
|
|
if (dist < closest_building.dist_squared) {
|
|
closest_building.pos = pos;
|
|
closest_building.dist_squared = dist;
|
|
closest_building.entity = it_entity->handle;
|
|
}
|
|
}
|
|
|
|
if (!FP_Game_IsNilEntityHandle(game, closest_building.entity) &&
|
|
closest_building.dist_squared < DQN_SQUARED(FP_Game_MetersToPixelsNx1(game->play, 5.f))) {
|
|
|
|
bool has_waypoint_to_building = false;
|
|
for (FP_SentinelListLink<FP_GameWaypoint> *link = nullptr;
|
|
!has_waypoint_to_building && FP_SentinelList_Iterate<FP_GameWaypoint>(&entity->waypoints, &link); ) {
|
|
has_waypoint_to_building = link->data.entity == closest_building.entity;
|
|
}
|
|
|
|
if (!has_waypoint_to_building) {
|
|
Dqn_Rect hit_box = FP_Game_CalcEntityWorldHitBox(game, closest_building.entity);
|
|
Dqn_V2 top_left = Dqn_Rect_TopLeft(hit_box);
|
|
Dqn_V2 top_right = Dqn_Rect_TopRight(hit_box);
|
|
|
|
FP_SentinelListLink<FP_GameWaypoint> *link = FP_SentinelList_MakeBefore(&entity->waypoints, FP_SentinelList_Front(&entity->waypoints), game->play.chunk_pool);
|
|
FP_GameWaypoint *waypoint = &link->data;
|
|
waypoint->entity = closest_building.entity;
|
|
waypoint->type = FP_GameWaypointType_Queue;
|
|
|
|
// NOTE: Add the entity to the building queue
|
|
FP_GameEntity *building = FP_Game_GetEntity(game, closest_building.entity);
|
|
Dqn_FArray_Add(&building->building_queue, entity->handle);
|
|
|
|
// NOTE: Remember the building we are queued at
|
|
entity->queued_at_building = building->handle;
|
|
}
|
|
}
|
|
}
|
|
|
|
// NOTE: Building queue ================================================================
|
|
for (Dqn_usize index = 0; index < entity->building_queue.size; index++) {
|
|
FP_GameEntityHandle queue_entity_handle = entity->building_queue.data[index];
|
|
// NOTE: Delete dead entities
|
|
if (FP_Game_IsNilEntityHandle(game, queue_entity_handle)) {
|
|
index = Dqn_FArray_EraseRange(&entity->building_queue, index, 1 /*count*/, Dqn_ArrayErase_Stable).it_index;
|
|
continue;
|
|
}
|
|
|
|
// NOTE: Delete far away entities
|
|
FP_GameEntity *queue_entity = FP_Game_GetEntity(game, queue_entity_handle);
|
|
Dqn_V2 queue_entity_p = FP_Game_CalcEntityWorldPos(game, queue_entity_handle);
|
|
Dqn_V2 building_p = FP_Game_CalcEntityWorldPos(game, entity->handle);
|
|
Dqn_f32 dist_sq = Dqn_V2_LengthSq_V2x2(queue_entity_p, building_p);
|
|
Dqn_f32 threshold_sq = DQN_SQUARED(FP_Game_MetersToPixelsNx1(game->play, 10.f));
|
|
if (dist_sq >= threshold_sq || queue_entity->converted_faction) {
|
|
// NOTE: Remove the entity from the building
|
|
queue_entity->queued_at_building = {};
|
|
|
|
// NOTE: Remove it from the queue
|
|
index = Dqn_FArray_EraseRange(&entity->building_queue, index, 1 /*count*/, Dqn_ArrayErase_Stable).it_index;
|
|
|
|
// NOTE: Make sure the entity doesnt' try and revisit
|
|
FP_SentinelList_Add(&queue_entity->buildings_visited, game->play.chunk_pool, entity->handle);
|
|
|
|
// NOTE: Remove the waypoint from the entity
|
|
if (queue_entity->waypoints.size) {
|
|
for (FP_SentinelListLink<FP_GameWaypoint> *link = nullptr;
|
|
FP_SentinelList_Iterate<FP_GameWaypoint>(&queue_entity->waypoints, &link); ) {
|
|
if (link->data.entity == entity->handle) {
|
|
FP_SentinelList_Erase(&queue_entity->waypoints, link, game->play.chunk_pool);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
// NOTE: Bubble sort entities in the building queue by distance to the building ========
|
|
// For example the closest entity will be assigned the first queue slot to the building
|
|
if (entity->building_queue.size && game->play.clock_ms >= entity->building_queue_next_sort_timestamp_ms) {
|
|
|
|
// NOTE: We only sort the queue for the building every second. This prevents the queue
|
|
// from stagnating incase some entity has locked a position in the queue but had
|
|
// some dodgy physics that sent it far away
|
|
entity->building_queue_next_sort_timestamp_ms = game->play.clock_ms + 1000;
|
|
for (bool swapped = true; swapped; ) {
|
|
swapped = false;
|
|
for (Dqn_usize index = 0; index < (entity->building_queue.size - 1); index++) {
|
|
FP_GameEntityHandle left_handle = entity->building_queue.data[index + 0];
|
|
FP_GameEntityHandle right_handle = entity->building_queue.data[index + 1];
|
|
Dqn_V2 left_world_p = FP_Game_CalcEntityWorldPos(game, left_handle);
|
|
Dqn_V2 right_world_p = FP_Game_CalcEntityWorldPos(game, right_handle);
|
|
Dqn_f32 left_dist_sq = Dqn_V2_LengthSq_V2x2(entity_pos, left_world_p);
|
|
Dqn_f32 right_dist_sq = Dqn_V2_LengthSq_V2x2(entity_pos, right_world_p);
|
|
|
|
if (left_dist_sq > (right_dist_sq * 1.1f)) {
|
|
DQN_SWAP(entity->building_queue.data[index + 0], entity->building_queue.data[index + 1]);
|
|
swapped = true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// NOTE: Handle waypoints ==============================================================
|
|
while (entity->waypoints.size) {
|
|
FP_SentinelListLink<FP_GameWaypoint> *waypoint_link = entity->waypoints.sentinel->next;
|
|
FP_GameWaypoint const *waypoint = &waypoint_link->data;
|
|
FP_GameEntity *waypoint_entity = FP_Game_GetEntity(game, waypoint_link->data.entity);
|
|
if (FP_Game_IsNilEntity(waypoint_entity)) {
|
|
FP_SentinelList_Erase(&entity->waypoints, waypoint_link, game->play.chunk_pool);
|
|
continue;
|
|
}
|
|
|
|
// NOTE: We found a waypoint that is valid to move towards
|
|
Dqn_V2 target_pos = FP_Game_CalcWaypointWorldPos(game, entity->handle, waypoint);
|
|
Dqn_V2 entity_to_waypoint = target_pos - entity_pos;
|
|
|
|
// NOTE: Check if we've arrived at the waypoint
|
|
Dqn_f32 dist_to_waypoint_sq = Dqn_V2_LengthSq(entity_to_waypoint);
|
|
|
|
// NOTE: Calculate the approaching direction
|
|
FP_GameDirection approach_dir = FP_GameDirection_Up;
|
|
{
|
|
Dqn_V2 dir_vectors[FP_GameDirection_Count] = {};
|
|
dir_vectors[FP_GameDirection_Up] = Dqn_V2_InitNx2(+0, -1);
|
|
dir_vectors[FP_GameDirection_Down] = Dqn_V2_InitNx2(+0, +1);
|
|
dir_vectors[FP_GameDirection_Left] = Dqn_V2_InitNx2(-1, +0);
|
|
dir_vectors[FP_GameDirection_Right] = Dqn_V2_InitNx2(+1, +0);
|
|
|
|
Dqn_V2 target_entity_pos = FP_Game_CalcEntityWorldPos(game, waypoint_entity->handle);
|
|
Dqn_V2 entity_to_target = target_entity_pos - entity_pos;
|
|
Dqn_V2 entity_to_target_norm = Dqn_V2_Normalise(entity_to_target);
|
|
Dqn_f32 approach_dir_scalar_projection_onto_entity_to_waypoint_vector = -1.1f;
|
|
DQN_FOR_UINDEX (dir_index, FP_GameDirection_Count) {
|
|
Dqn_V2 attack_dir = dir_vectors[dir_index];
|
|
Dqn_f32 scalar_projection = Dqn_V2_Dot(attack_dir, entity_to_target_norm);
|
|
if (scalar_projection > approach_dir_scalar_projection_onto_entity_to_waypoint_vector) {
|
|
approach_dir = DQN_CAST(FP_GameDirection)dir_index;
|
|
approach_dir_scalar_projection_onto_entity_to_waypoint_vector = scalar_projection;
|
|
}
|
|
}
|
|
}
|
|
|
|
Dqn_f32 arrival_threshold = {};
|
|
switch (waypoint->arrive) {
|
|
case FP_GameWaypointArrive_Default: {
|
|
Dqn_Rect waypoint_hit_box = FP_Game_CalcEntityWorldHitBox(game, waypoint_entity->handle);
|
|
if (approach_dir == FP_GameDirection_Up || approach_dir == FP_GameDirection_Down)
|
|
arrival_threshold = 10.f;
|
|
else
|
|
arrival_threshold = 10.f;
|
|
} break;
|
|
|
|
case FP_GameWaypointArrive_WhenWithinEntitySize: {
|
|
arrival_threshold = DQN_MAX(waypoint_entity->local_hit_box_size.w, waypoint_entity->local_hit_box_size.h) * waypoint_link->data.value;
|
|
} break;
|
|
}
|
|
|
|
// NOTE: We haven't arrived yet, calculate an acceleration vector to the waypoint
|
|
if (dist_to_waypoint_sq > DQN_SQUARED(arrival_threshold)) {
|
|
Dqn_V2 entity_to_waypoint_norm = Dqn_V2_Normalise(entity_to_waypoint);
|
|
acceleration_meters_per_s = entity_to_waypoint_norm * (entity->base_acceleration_per_s.meters * 0.25f);
|
|
|
|
if (entity->type == FP_EntityType_Smoochie) {
|
|
if (waypoint_entity->type == FP_EntityType_Terry) {
|
|
if (dist_to_waypoint_sq < DQN_SQUARED(FP_Game_MetersToPixelsNx1(game->play, 2.f)) &&
|
|
!entity->smoochie_has_teleported) {
|
|
|
|
if (entity->smoochie_teleport_timestamp == 0) {
|
|
entity->smoochie_teleport_timestamp = game->play.clock_ms + DQN_CAST(Dqn_usize)(Dqn_PCG32_NextF32(&game->play.rng) * 5000);
|
|
}
|
|
|
|
if (game->play.clock_ms < entity->smoochie_teleport_timestamp) {
|
|
Dqn_usize time_to_teleport_ms = entity->smoochie_teleport_timestamp - game->play.clock_ms;
|
|
if (time_to_teleport_ms < 2000)
|
|
entity->action.sprite_alpha = Dqn_PCG32_NextF32(&game->play.rng);
|
|
} else if (game->play.clock_ms >= entity->smoochie_teleport_timestamp) {
|
|
Dqn_Rect waypoint_rect = FP_Game_CalcEntityWorldHitBox(game, waypoint_entity->handle);
|
|
entity->smoochie_has_teleported = true;
|
|
entity->action.sprite_alpha = 1.f;
|
|
|
|
Dqn_FArray<Dqn_V2, FP_GameDirection_Count> teleport_pos = {};
|
|
|
|
if (waypoint_entity->direction != FP_GameDirection_Up) {
|
|
Dqn_FArray_Add(&teleport_pos, Dqn_Rect_InterpolatedPoint(waypoint_rect, Dqn_V2_InitNx2(0.5f, -1.2f)));
|
|
}
|
|
|
|
if (waypoint_entity->direction != FP_GameDirection_Down) {
|
|
Dqn_FArray_Add(&teleport_pos, Dqn_Rect_InterpolatedPoint(waypoint_rect, Dqn_V2_InitNx2(0.5f, +1.2f)));
|
|
}
|
|
|
|
if (waypoint_entity->direction != FP_GameDirection_Left) {
|
|
Dqn_FArray_Add(&teleport_pos, Dqn_Rect_InterpolatedPoint(waypoint_rect, Dqn_V2_InitNx2(-1.2f, +0.5f)));
|
|
}
|
|
|
|
if (waypoint_entity->direction != FP_GameDirection_Right) {
|
|
Dqn_FArray_Add(&teleport_pos, Dqn_Rect_InterpolatedPoint(waypoint_rect, Dqn_V2_InitNx2(+1.2f, +0.5f)));
|
|
}
|
|
|
|
uint32_t teleport_index = Dqn_PCG32_Range(&game->play.rng, 0, DQN_CAST(uint32_t)teleport_pos.size);
|
|
entity->local_pos = teleport_pos.data[teleport_index];
|
|
}
|
|
}
|
|
}
|
|
} else if (entity->type == FP_EntityType_Clinger) {
|
|
if (game->play.clock_ms >= entity->clinger_next_dash_timestamp && dist_to_waypoint_sq > (DQN_SQUARED(arrival_threshold * 4.f))) {
|
|
entity->clinger_next_dash_timestamp = game->play.clock_ms + 2000;
|
|
acceleration_meters_per_s = entity_to_waypoint_norm * entity->base_acceleration_per_s.meters * 45.f;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
// NOTE: We have arrived at the waypoint
|
|
bool aggro = false;
|
|
if (entity->flags & FP_GameEntityFlag_Aggros) {
|
|
aggro |= entity->faction == FP_GameEntityFaction_Friendly && waypoint_entity->faction & FP_GameEntityFaction_Foe;
|
|
aggro |= entity->faction == FP_GameEntityFaction_Foe && waypoint_entity->faction & FP_GameEntityFaction_Friendly;
|
|
}
|
|
|
|
bool building_response = (entity->flags & FP_GameEntityFlag_RespondsToBuildings) &&
|
|
waypoint_entity->type == FP_EntityType_ClubTerry ||
|
|
waypoint_entity->type == FP_EntityType_AirportTerry ||
|
|
waypoint_entity->type == FP_EntityType_ChurchTerry;
|
|
|
|
if (aggro || building_response) {
|
|
bool can_attack = !entity->is_dying; // TODO(doyle): State transition needs to check if it's valid to move to making this not necessary
|
|
if (building_response) {
|
|
FP_GameEntity *building = waypoint_entity;
|
|
can_attack = false;
|
|
|
|
DQN_ASSERTF(
|
|
waypoint->type == FP_GameWaypointType_Queue,
|
|
"There's nothing stopping us from supporting other "
|
|
"waypoint types to buildings, but, for this game "
|
|
"we only ever make mobs queue at the building");
|
|
|
|
DQN_ASSERTF(
|
|
building->building_queue.size,
|
|
"An entity should only be forming a waypoint to "
|
|
"the building if there was space in the queue and "
|
|
"they were added the the queue");
|
|
|
|
if (FP_Game_IsNilEntityHandle(game, building->building_patron)) {
|
|
if (waypoint_entity->building_queue.data[0] == entity->handle) {
|
|
// NOTE: This entity is front-in-line in the queue to enter the building, we can enter!
|
|
building->building_patron = entity->handle;
|
|
entity->queued_at_building = {};
|
|
|
|
// NOTE: Remove them from the queue
|
|
Dqn_FArray_EraseRange(&waypoint_entity->building_queue, 0 /*index*/, 1 /*count*/, Dqn_ArrayErase_Stable);
|
|
|
|
Dqn_Rect hit_box = FP_Game_CalcEntityWorldHitBox(game, building->handle);
|
|
Dqn_V2 exit_pos = Dqn_Rect_InterpolatedPoint(hit_box, Dqn_V2_InitNx2(0.5f, 1.1f));
|
|
if (building->type == FP_EntityType_ClubTerry) {
|
|
FP_Game_EntityTransitionState(game, building, FP_EntityClubTerryState_PartyTime);
|
|
entity->local_pos = exit_pos; // TODO(doyle): Only works when parent world pos is 0,0
|
|
} else if (building->type == FP_EntityType_AirportTerry) {
|
|
FP_Game_EntityTransitionState(game, building, FP_EntityAirportTerryState_FlyPassenger);
|
|
entity->local_pos = exit_pos; // TODO(doyle): Only works when parent world pos is 0,0
|
|
} else {
|
|
DQN_ASSERT(building->type == FP_EntityType_ChurchTerry);
|
|
FP_Game_EntityTransitionState(game, building, FP_EntityChurchTerryState_ConvertPatron);
|
|
}
|
|
|
|
entity->flags |= FP_GameEntityFlag_OccupiedInBuilding;
|
|
FP_SentinelList_Erase(&entity->waypoints, waypoint_link, game->play.chunk_pool);
|
|
|
|
// NOTE: Add the building to the entity's visit list to prevent them from re-entering
|
|
FP_SentinelList_Add(&entity->buildings_visited, game->play.chunk_pool, building->handle);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (can_attack) {
|
|
switch (entity->type) {
|
|
case FP_EntityType_Terry: /*FALLTHRU*/
|
|
case FP_EntityType_Smoochie: /*FALLTHRU*/
|
|
case FP_EntityType_Catfish: /*FALLTHRU*/
|
|
case FP_EntityType_Clinger: {
|
|
// TODO(doyle): We should check if it's valid to enter this new state
|
|
// from the entity's current state
|
|
if (entity->type == FP_EntityType_Terry) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityTerryState_Attack);
|
|
} else if (entity->type == FP_EntityType_Smoochie) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntitySmoochieState_Attack);
|
|
} else if (entity->type == FP_EntityType_Catfish) {
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityCatfishState_Attack);
|
|
} else {
|
|
DQN_ASSERT(entity->type == FP_EntityType_Clinger);
|
|
FP_Game_EntityTransitionState(game, entity, FP_EntityClingerState_Attack);
|
|
}
|
|
entity->direction = approach_dir;
|
|
} break;
|
|
|
|
case FP_EntityType_Nil: break;
|
|
case FP_EntityType_ClubTerry: break;
|
|
case FP_EntityType_Map: break;
|
|
case FP_EntityType_Heart: break;
|
|
case FP_EntityType_MerchantTerry: break;
|
|
case FP_EntityType_MerchantGraveyard: break;
|
|
case FP_EntityType_MerchantGym: break;
|
|
case FP_EntityType_MerchantPhoneCompany: break;
|
|
case FP_EntityType_AirportTerry: break;
|
|
case FP_EntityType_ChurchTerry: break;
|
|
case FP_EntityType_KennelTerry: break;
|
|
case FP_EntityType_PhoneMessageProjectile: break;
|
|
case FP_EntityType_Count: DQN_INVALID_CODE_PATH; break;
|
|
}
|
|
}
|
|
|
|
// NOTE: Aggro makes the entity attack, we will exit here preserving the waypoint in the entity.
|
|
break;
|
|
} else {
|
|
FP_SentinelList_Erase(&entity->waypoints, waypoint_link, game->play.chunk_pool);
|
|
}
|
|
}
|
|
|
|
// NOTE: Move entity by mouse ==============================================================
|
|
if (game->play.active_entity == entity->handle && entity->flags & FP_GameEntityFlag_MoveByMouse) {
|
|
entity->velocity = {};
|
|
acceleration_meters_per_s = {};
|
|
entity->local_pos += input->mouse_p_delta;
|
|
}
|
|
|
|
if (game->play.clicked_entity == entity->handle) {
|
|
if (game->play.in_game_menu == FP_GameInGameMenu_Nil || game->play.in_game_menu == FP_GameInGameMenu_Build) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_H))
|
|
game->play.in_game_menu = DQN_CAST(FP_GameInGameMenu)(DQN_CAST(uint32_t)game->play.in_game_menu ^ FP_GameInGameMenu_Build);
|
|
}
|
|
|
|
if (entity->flags & FP_GameEntityFlag_CameraTracking)
|
|
game->play.camera.world_pos = FP_Game_CalcEntityWorldPos(game, entity->handle) - Dqn_V2_InitV2I(platform->core.window_size) * .5f;
|
|
|
|
FP_GamePlaceableBuilding placeable_building = PLACEABLE_BUILDINGS[game->play.build_mode_building_index];
|
|
game->play.build_mode_can_place_building = false;
|
|
|
|
uint8_t *inventory_count = nullptr;
|
|
if (placeable_building.type == FP_EntityType_ChurchTerry)
|
|
inventory_count = &entity->inventory.churchs;
|
|
else if (placeable_building.type == FP_EntityType_KennelTerry)
|
|
inventory_count = &entity->inventory.kennels;
|
|
else if (placeable_building.type == FP_EntityType_ClubTerry)
|
|
inventory_count = &entity->inventory.clubs;
|
|
else if (placeable_building.type == FP_EntityType_AirportTerry)
|
|
inventory_count = &entity->inventory.airports;
|
|
|
|
bool have_building_inventory = inventory_count && (*inventory_count) > 0;
|
|
if (have_building_inventory && game->play.in_game_menu == FP_GameInGameMenu_Build) {
|
|
Dqn_Rect dest_rect = FP_Game_GetBuildingPlacementRectForEntity(game, placeable_building, entity->handle);
|
|
Dqn_V2 placement_pos = Dqn_Rect_Center(dest_rect);
|
|
|
|
for (FP_GameEntityIterator zone_it = {};
|
|
FP_Game_DFSPreOrderWalkEntityTree(game, &zone_it, game->play.root_entity);
|
|
) {
|
|
|
|
FP_GameEntity *zone = zone_it.entity;
|
|
bool is_building = zone->type == FP_EntityType_KennelTerry ||
|
|
zone->type == FP_EntityType_AirportTerry ||
|
|
zone->type == FP_EntityType_ChurchTerry ||
|
|
zone->type == FP_EntityType_ClubTerry;
|
|
|
|
Dqn_Rect zone_hit_box = FP_Game_CalcEntityWorldHitBox(game, zone->handle);
|
|
if (is_building) {
|
|
if (Dqn_Rect_Intersects(zone_hit_box, dest_rect)) {
|
|
game->play.build_mode_can_place_building = false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ((zone->flags & FP_GameEntityFlag_BuildZone) == 0)
|
|
continue;
|
|
|
|
zone_hit_box.pos += dest_rect.size * .5f;
|
|
zone_hit_box.size -= dest_rect.size;
|
|
zone_hit_box.size = Dqn_V2_Max(zone_hit_box.size, Dqn_V2_Zero);
|
|
|
|
game->play.build_mode_can_place_building |= Dqn_Rect_ContainsPoint(zone_hit_box, placement_pos);
|
|
}
|
|
|
|
if (game->play.build_mode_can_place_building &&
|
|
TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_J)) {
|
|
if (placeable_building.type == FP_EntityType_ClubTerry) {
|
|
FP_Entity_CreateClubTerry(game, placement_pos, "Club Terry");
|
|
TELY_Audio_Play(audio, game->audio[FP_GameAudio_Club], 1.f);
|
|
} else if (placeable_building.type == FP_EntityType_ChurchTerry) {
|
|
FP_Entity_CreateChurchTerry(game, placement_pos, "Church Terry");
|
|
TELY_Audio_Play(audio, game->audio[FP_GameAudio_Church], 1.f);
|
|
} else if (placeable_building.type == FP_EntityType_AirportTerry) {
|
|
FP_Entity_CreateAirportTerry(game, placement_pos, "Airport Terry");
|
|
TELY_Audio_Play(audio, game->audio[FP_GameAudio_Plane], 1.f);
|
|
} else {
|
|
DQN_ASSERT(placeable_building.type == FP_EntityType_KennelTerry);
|
|
FP_Entity_CreateKennelTerry(game, placement_pos, "Kennel Terry");
|
|
TELY_Audio_Play(audio, game->audio[FP_GameAudio_Dog], 1.f);
|
|
}
|
|
|
|
(*inventory_count)--;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!FP_Game_IsNilEntityHandle(game, entity->carried_monkey) && entity->type != FP_EntityType_MobSpawner) {
|
|
FP_GameFindClosestEntityResult closest_portal = FP_Game_FindClosestEntityWithType(game, entity->carried_monkey, game->play.mob_spawners.data, game->play.mob_spawners.size);
|
|
if (closest_portal.dist_squared < DQN_SQUARED(FP_Game_MetersToPixelsNx1(game->play, 1.f))) {
|
|
FP_GameEntity *portal = FP_Game_GetEntity(game, closest_portal.entity);
|
|
portal->carried_monkey = entity->carried_monkey;
|
|
entity->carried_monkey = {};
|
|
}
|
|
acceleration_meters_per_s *= 0.5f; // TODO(doyle): Penalise the player
|
|
} else {
|
|
acceleration_meters_per_s *= 1.f; // TODO(doyle): Penalise the player
|
|
}
|
|
|
|
// NOTE: Left-shift lets us strafe in the same direction
|
|
if (!TELY_Platform_InputScanCodeIsDown(input, TELY_PlatformInputScanCode_LeftShift)) {
|
|
if (acceleration_meters_per_s.x)
|
|
entity->direction = acceleration_meters_per_s.x > 0.f ? FP_GameDirection_Right : FP_GameDirection_Left;
|
|
else if (acceleration_meters_per_s.y)
|
|
entity->direction = acceleration_meters_per_s.y > 0.f ? FP_GameDirection_Down : FP_GameDirection_Up;
|
|
}
|
|
|
|
// NOTE: Tick the state machine
|
|
// NOTE: This can delete the entity! Take caution
|
|
FP_GameEntityHandle entity_handle = entity->handle;
|
|
FP_EntityActionStateMachine(game, &platform->audio, input, entity, &acceleration_meters_per_s);
|
|
|
|
// NOTE: Core equations of motion ==========================================================
|
|
FP_Game_MoveEntity(game, entity_handle, acceleration_meters_per_s);
|
|
}
|
|
|
|
// NOTE: If all enemies for the current wave have been spawned and the cooldown has elapsed
|
|
// start the next wave.
|
|
if (game->play.enemies_spawned_this_wave >= game->play.enemies_per_wave && game->play.clock_ms >= game->play.wave_cooldown_timestamp_ms) {
|
|
game->play.enemies_per_wave = DQN_MAX(5 * DQN_CAST(uint32_t)game->play.mob_spawners.size, DQN_CAST(uint32_t)(game->play.enemies_per_wave * 3));
|
|
game->play.enemies_spawned_this_wave = 0;
|
|
game->play.current_wave++;
|
|
}
|
|
}
|
|
|
|
// NOTE: Update entity =========================================================================
|
|
for (FP_GameEntityIterator it = {}; FP_Game_DFSPostOrderWalkEntityTree(game, &it, game->play.root_entity); ) {
|
|
FP_GameEntity *entity = it.entity;
|
|
|
|
if (entity->flags & FP_GameEntityFlag_TTL) {
|
|
if (game->play.clock_ms >= entity->ttl_end_timestamp) {
|
|
FP_Game_DeleteEntity(game, entity->handle);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
// NOTE: Recover mobile data
|
|
entity->terry_mobile_data_plan =
|
|
DQN_MIN(entity->terry_mobile_data_plan + DQN_CAST(Dqn_usize)(FP_TERRY_MOBILE_DATA_PER_RANGE_ATTACK * .25f * PHYSICS_STEP),
|
|
entity->terry_mobile_data_plan_cap);
|
|
|
|
// NOTE: Recover hp & stamina
|
|
if (game->play.update_counter % 4 == 0) {
|
|
entity->stamina = DQN_MIN(entity->stamina + 1, entity->stamina_cap);
|
|
}
|
|
|
|
if (entity->flags & FP_GameEntityFlag_RecoversHP) {
|
|
if (game->play.update_counter % entity->hp_recover_every_n_ticks == 0) {
|
|
entity->hp = DQN_MIN(entity->hp + 1, entity->hp_cap);
|
|
}
|
|
}
|
|
|
|
// NOTE: Derive dynmamic bounding boxes ====================================================
|
|
if (entity->flags & FP_GameEntityFlag_DeriveHitBoxFromChildrenBoundingBox) {
|
|
Dqn_Rect children_bbox = {};
|
|
|
|
// TODO(doyle): Is the hit box supposed to include the containing
|
|
// entity itself? Not sure
|
|
children_bbox.pos = FP_Game_CalcEntityWorldPos(game, entity->handle);
|
|
|
|
for (FP_GameEntityIterator child_it = {}; FP_Game_DFSPreOrderWalkEntityTree(game, &child_it, entity);) {
|
|
FP_GameEntity *child = child_it.entity;
|
|
DQN_ASSERT(child != entity);
|
|
|
|
Dqn_Rect bbox = FP_Game_CalcEntityWorldBoundingBox(game, child->handle);
|
|
children_bbox = Dqn_Rect_Union(children_bbox, bbox);
|
|
}
|
|
|
|
Dqn_Rect padded_bbox = Dqn_Rect_Expand(children_bbox, 16.f);
|
|
entity->local_hit_box_offset = padded_bbox.pos - entity->local_pos + (padded_bbox.size * .5f);
|
|
entity->local_hit_box_size = padded_bbox.size;
|
|
}
|
|
|
|
// NOTE: Mob spawner =======================================================================
|
|
if (entity->type == FP_EntityType_MobSpawner) {
|
|
// NOTE: Flush any spawn entities that are dead
|
|
for (FP_SentinelListLink<FP_GameEntityHandle> *link = nullptr; FP_SentinelList_Iterate<FP_GameEntityHandle>(&entity->spawn_list, &link); ) {
|
|
FP_GameEntity *spawned_entity = FP_Game_GetEntity(game, link->data);
|
|
if (FP_Game_IsNilEntity(spawned_entity)) // NOTE: Entity is dead remove it from the linked list
|
|
link = FP_SentinelList_Erase(&entity->spawn_list, link, game->play.chunk_pool);
|
|
}
|
|
|
|
if (entity->action.state != FP_EntityMobSpawnerState_Shutdown &&
|
|
game->play.enemies_spawned_this_wave < game->play.enemies_per_wave && entity->spawn_list.size < entity->spawn_cap) { // NOTE: Spawn new entities
|
|
if (input->timer_s >= entity->next_spawn_timestamp_s) {
|
|
Dqn_usize spawn_count = DQN_MIN(game->play.current_wave + 1, 8);
|
|
for (Dqn_usize spawn_index = 0; spawn_index < spawn_count; spawn_index++) {
|
|
uint16_t hp_adjustment = DQN_CAST(uint16_t)game->play.current_wave;
|
|
entity->next_spawn_timestamp_s = DQN_CAST(uint64_t)(input->timer_s + 5.f);
|
|
|
|
FP_SentinelListLink<FP_GameEntityHandle> *link = FP_SentinelList_Make(&entity->spawn_list, game->play.chunk_pool);
|
|
|
|
Dqn_f32 one_meter = FP_Game_MetersToPixelsNx1(game->play, 1.f);
|
|
Dqn_V2 entity_world_pos = FP_Game_CalcEntityWorldPos(game, entity->handle);
|
|
Dqn_Rect entity_hit_box = FP_Game_CalcEntityWorldHitBox(game, entity->handle);
|
|
|
|
Dqn_f32 step_y = (entity_hit_box.size.h / spawn_count) * 1.5f;
|
|
Dqn_f32 mob_y_offset = (Dqn_PCG32_NextF32(&game->play.rng) * step_y) + (step_y * spawn_index);
|
|
if (Dqn_PCG32_NextF32(&game->play.rng) >= .5f)
|
|
mob_y_offset *= -1;
|
|
|
|
Dqn_V2 mob_world_pos = Dqn_V2_InitNx2(entity_world_pos.x, entity_world_pos.y + mob_y_offset);
|
|
Dqn_f32 mob_choice = Dqn_PCG32_NextF32(&game->play.rng);
|
|
if (mob_choice <= 0.33f)
|
|
link->data = FP_Entity_CreateClinger(game, mob_world_pos, "Clinger");
|
|
else if (mob_choice <= 0.66f)
|
|
link->data = FP_Entity_CreateSmoochie(game, mob_world_pos, "Smoochie");
|
|
else
|
|
link->data = FP_Entity_CreateCatfish(game, mob_world_pos, "Catfish");
|
|
|
|
// NOTE: Setup the mob with waypoints
|
|
FP_GameEntity *mob = FP_Game_GetEntity(game, link->data);
|
|
mob->waypoints = FP_SentinelList_Init<FP_GameWaypoint>(game->play.chunk_pool);
|
|
mob->flags |= FP_GameEntityFlag_Aggros;
|
|
mob->flags |= FP_GameEntityFlag_RespondsToBuildings;
|
|
mob->hp_cap += hp_adjustment;
|
|
mob->hp = mob->hp_cap;
|
|
|
|
for (FP_GameEntity *waypoint_entity = entity->first_child; waypoint_entity; waypoint_entity = waypoint_entity->next) {
|
|
if ((waypoint_entity->flags & FP_GameEntityFlag_MobSpawnerWaypoint) == 0)
|
|
continue;
|
|
|
|
// NOTE: Add the waypoint
|
|
FP_SentinelListLink<FP_GameWaypoint> *waypoint = FP_SentinelList_Make(&mob->waypoints, game->play.chunk_pool);
|
|
waypoint->data.entity = waypoint_entity->handle;
|
|
waypoint->data.arrive = FP_GameWaypointArrive_WhenWithinEntitySize;
|
|
waypoint->data.value = 1.5f;
|
|
|
|
uint32_t min_vary = DQN_CAST(uint32_t)(one_meter * .5f);
|
|
uint32_t max_vary = DQN_CAST(uint32_t)(one_meter * 2.f);
|
|
waypoint->data.offset += Dqn_V2_InitNx2(DQN_CAST(Dqn_f32)Dqn_PCG32_Range(&game->play.rng, min_vary, max_vary),
|
|
DQN_CAST(Dqn_f32)Dqn_PCG32_Range(&game->play.rng, min_vary, max_vary));
|
|
|
|
if (Dqn_PCG32_NextF32(&game->play.rng) >= .5f)
|
|
waypoint->data.offset *= -1;
|
|
}
|
|
|
|
game->play.enemies_spawned_this_wave++;
|
|
if (game->play.enemies_spawned_this_wave >= game->play.enemies_per_wave)
|
|
game->play.wave_cooldown_timestamp_ms = game->play.clock_ms + 30'000;
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
// NOTE: Do attacks ============================================================================
|
|
{
|
|
Dqn_Profiler_ZoneScopeWithIndex("FP_Update: Attacks", FP_ProfileZone_FPUpdate_Attacks);
|
|
FP_GameEntity *attacker = entity;
|
|
|
|
// NOTE: Resolve attack boxes
|
|
if (!Dqn_V2_Area(attacker->attack_box_size))
|
|
continue;
|
|
|
|
Dqn_Rect attacker_box = FP_Game_CalcEntityAttackWorldHitBox(game, attacker->handle);
|
|
Dqn_V2 attacker_world_pos = FP_Game_CalcEntityWorldPos(game, attacker->handle);
|
|
|
|
FP_GameEntityFaction enemy_faction =
|
|
entity->faction == FP_GameEntityFaction_Friendly
|
|
? FP_GameEntityFaction_Foe
|
|
: FP_GameEntityFaction_Friendly;
|
|
|
|
for (FP_GameEntityIterator defender_it = {}; FP_Game_DFSPostOrderWalkEntityTree(game, &defender_it, game->play.root_entity); ) {
|
|
FP_GameEntity *defender = defender_it.entity;
|
|
if (defender->handle == attacker->handle)
|
|
continue;
|
|
|
|
if ((defender->flags & FP_GameEntityFlag_Attackable) == 0)
|
|
continue;
|
|
|
|
// NOTE: Projectiles can't hurt the owner that spawned it
|
|
if (attacker->projectile_owner == defender->handle)
|
|
continue;
|
|
|
|
if (defender->faction != enemy_faction)
|
|
continue;
|
|
|
|
Dqn_Rect defender_box = FP_Game_CalcEntityWorldHitBox(game, defender->handle);
|
|
if (!Dqn_Rect_Intersects(attacker_box, defender_box))
|
|
continue;
|
|
|
|
// NOTE: Do HP =========================================================================
|
|
if (game->play.player == defender->handle || game->play.heart == defender->handle) {
|
|
if (game->play.god_mode)
|
|
continue;
|
|
}
|
|
|
|
defender->hp = defender->hp >= attacker->base_attack ? defender->hp - attacker->base_attack : 0;
|
|
if (defender->hp <= 0) {
|
|
if (!defender->is_dying) {
|
|
FP_GameEntity *coin_receiver = FP_Game_GetEntity(game, attacker->projectile_owner);
|
|
if (FP_Game_IsNilEntity(coin_receiver))
|
|
coin_receiver = attacker;
|
|
coin_receiver->coins += 1;
|
|
}
|
|
defender->is_dying = true;
|
|
}
|
|
|
|
// NOTE: Kickback ======================================================================
|
|
#if 0
|
|
Dqn_V2 defender_world_pos = Dqn_Rect_Center(defender_box);
|
|
Dqn_V2 attack_dir_vector = {};
|
|
if (attacker_world_pos.x < defender_world_pos.x)
|
|
attack_dir_vector.x = 1.f;
|
|
else
|
|
attack_dir_vector.x = -1.f;
|
|
|
|
Dqn_V2 attack_acceleration_meters_per_s = attack_dir_vector * 60.f;
|
|
FP_Game_MoveEntity(game, defender->handle, attack_acceleration_meters_per_s);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
bool all_portals_shutdown = true;
|
|
for (FP_GameEntityHandle portal_handle : game->play.mob_spawners) {
|
|
FP_GameEntity *portal = FP_Game_GetEntity(game, portal_handle);
|
|
all_portals_shutdown &= portal->action.state == FP_EntityMobSpawnerState_Shutdown;
|
|
}
|
|
|
|
if (all_portals_shutdown) {
|
|
game->play.state = FP_GameState_WinGame;
|
|
}
|
|
|
|
{
|
|
FP_GameEntity *heart = FP_Game_GetEntity(game, game->play.heart);
|
|
FP_GameEntity *player = FP_Game_GetEntity(game, game->play.player);
|
|
if (heart->hp <= 0) {
|
|
player->hp = 0;
|
|
game->play.state = FP_GameState_LoseGame;
|
|
}
|
|
}
|
|
|
|
if (!FP_Game_IsNilEntityHandle(game, game->play.clicked_entity)) {
|
|
TELY_AssetSpriteAnimation *sprite_anim = TELY_Asset_GetSpriteAnimation(&game->atlas_sprite_sheet, g_anim_names.map);
|
|
Dqn_Rect sprite_rect = game->atlas_sprite_sheet.rects.data[sprite_anim->index];
|
|
|
|
const Dqn_usize target_width = 1800;
|
|
const Dqn_usize target_height = 1046;
|
|
|
|
game->play.camera.world_pos.x = DQN_MIN(game->play.camera.world_pos.x, game->play.map->local_hit_box_size.w * +0.5f - target_width);
|
|
game->play.camera.world_pos.x = DQN_MAX(game->play.camera.world_pos.x, game->play.map->local_hit_box_size.w * -0.5f);
|
|
game->play.camera.world_pos.y = DQN_MAX(game->play.camera.world_pos.y, game->play.map->local_hit_box_size.h * -0.5f);
|
|
game->play.camera.world_pos.y = DQN_MIN(game->play.camera.world_pos.y, game->play.map->local_hit_box_size.h * +0.5f - target_height);
|
|
}
|
|
|
|
Dqn_Profiler_EndZone(update_zone);
|
|
}
|
|
|
|
void FP_Render(FP_Game *game, TELY_Platform *platform, TELY_Renderer *renderer, TELY_Audio *audio)
|
|
{
|
|
Dqn_Profiler_ZoneScopeWithIndex("FP_Render", FP_ProfileZone_FPRender);
|
|
TELY_PlatformInput *input = &platform->input;
|
|
TELY_RFui *rfui = &game->rfui;
|
|
TELY_Assets *assets = &platform->assets;
|
|
|
|
TELY_Render_ClearColourV3(renderer, TELY_COLOUR_BLACK_MIDNIGHT_V4.rgb);
|
|
TELY_Render_PushFont(renderer, game->jetbrains_mono_font);
|
|
|
|
TELY_RFui_FrameSetup(rfui, &platform->frame_arena);
|
|
TELY_RFui_PushFont(rfui, game->jetbrains_mono_font);
|
|
TELY_RFui_PushLabelColourV4(rfui, TELY_COLOUR_BLACK_MIDNIGHT_V4);
|
|
|
|
Dqn_M2x3 model_view = FP_Game_CameraModelViewM2x3(game->play.camera, platform);
|
|
TELY_Render_PushTransform(renderer, model_view);
|
|
Dqn_V2 world_mouse_p = input->mouse_p + game->play.camera.world_pos;
|
|
|
|
// NOTE: Draw tiles ============================================================================
|
|
Dqn_usize tile_count_x = DQN_CAST(Dqn_usize)(platform->core.window_size.w / game->play.tile_size);
|
|
Dqn_usize tile_count_y = DQN_CAST(Dqn_usize)(platform->core.window_size.h / game->play.tile_size);
|
|
|
|
for (Dqn_usize x = 0; x < tile_count_x; x++) {
|
|
Dqn_V2 start = Dqn_V2_InitNx2((x + 1) * game->play.tile_size, 0);
|
|
Dqn_V2 end = Dqn_V2_InitNx2(start.x, platform->core.window_size.h);
|
|
TELY_Render_LineColourV4(renderer, start, end, TELY_Colour_V4Alpha(TELY_COLOUR_WHITE_V4, .25f), 1.f);
|
|
}
|
|
|
|
for (Dqn_usize y = 0; y < tile_count_y; y++) {
|
|
Dqn_V2 start = Dqn_V2_InitNx2(0, (y + 1) * game->play.tile_size);
|
|
Dqn_V2 end = Dqn_V2_InitNx2(platform->core.window_size.w, start.y);
|
|
TELY_Render_LineColourV4(renderer, start, end, TELY_Colour_V4Alpha(TELY_COLOUR_WHITE_V4, .25f), 1.f);
|
|
}
|
|
|
|
TELY_Render_PushColourV4(renderer, TELY_COLOUR_BLACK_V4);
|
|
|
|
// NOTE: Draw entities =========================================================================
|
|
for (FP_GameEntityIterator it = {}; FP_Game_DFSPostOrderWalkEntityTree(game, &it, game->play.root_entity); ) {
|
|
FP_GameEntity *entity = it.entity;
|
|
if (entity->flags & FP_GameEntityFlag_OccupiedInBuilding)
|
|
continue;
|
|
|
|
// NOTE: Render shapes in entity ===========================================================
|
|
Dqn_Rect world_hit_box = FP_Game_CalcEntityWorldHitBox(game, entity->handle);
|
|
Dqn_V2 world_pos = FP_Game_CalcEntityWorldPos(game, entity->handle);
|
|
|
|
for (FP_GameShape const &shape_ : entity->shapes) {
|
|
FP_GameShape const *shape = &shape_;
|
|
Dqn_V2 local_to_world_p1 = world_pos + shape->p1;
|
|
Dqn_V2 local_to_world_p2 = world_pos + shape->p2;
|
|
switch (shape->type) {
|
|
case FP_GameShapeType_None: {
|
|
} break;
|
|
|
|
case FP_GameShapeType_Circle: {
|
|
TELY_Render_CircleColourV4(renderer, local_to_world_p1, shape->circle_radius, shape->render_mode, shape->colour);
|
|
} break;
|
|
|
|
case FP_GameShapeType_Rect: {
|
|
Dqn_Rect rect = Dqn_Rect_InitV2x2(local_to_world_p1, local_to_world_p2 - local_to_world_p1);
|
|
rect.pos -= rect.size * .5f;
|
|
TELY_Render_RectColourV4(renderer, rect, shape->render_mode, shape->colour);
|
|
} break;
|
|
|
|
case FP_GameShapeType_Line: {
|
|
TELY_Render_LineColourV4(renderer, local_to_world_p1, local_to_world_p2, shape->colour, shape->line_thickness);
|
|
} break;
|
|
}
|
|
}
|
|
|
|
// NOTE: Render entity sprites =============================================================
|
|
if (entity->action.sprite.anim) {
|
|
FP_GameEntityAction const *action = &entity->action;
|
|
TELY_AssetAnimatedSprite const sprite = action->sprite;
|
|
uint64_t const elapsed_ms = game->play.clock_ms - action->started_at_clock_ms;
|
|
uint16_t const raw_anim_frame = DQN_CAST(uint16_t)(elapsed_ms / sprite.anim->ms_per_frame);
|
|
DQN_ASSERTF(sprite.anim->count, "We will modulo by 0 or overflow to UINT64_MAX");
|
|
|
|
// TODO(doyle): So many ways to create and get sprite data .. its a mess
|
|
// I want to override per sprite anim height, we currently use the one
|
|
// in the entity which is not correct.
|
|
FP_EntityRenderData render_data = FP_Entity_GetRenderData(game, entity->type, action->state, entity->direction);
|
|
|
|
uint16_t anim_frame = 0;
|
|
if (action->sprite_play_once)
|
|
anim_frame = DQN_MIN(raw_anim_frame, (sprite.anim->count - 1));
|
|
else
|
|
anim_frame = raw_anim_frame % sprite.anim->count;
|
|
|
|
Dqn_usize sprite_index = sprite.anim->index + anim_frame;
|
|
Dqn_Rect src_rect = {};
|
|
switch (sprite.sheet->type) {
|
|
case TELY_AssetSpriteSheetType_Uniform: {
|
|
Dqn_usize sprite_sheet_row = sprite_index / sprite.sheet->sprites_per_row;
|
|
Dqn_usize sprite_sheet_column = sprite_index % sprite.sheet->sprites_per_row;
|
|
src_rect.pos.x = DQN_CAST(Dqn_f32)(sprite_sheet_column * sprite.sheet->sprite_size.w);
|
|
src_rect.pos.y = DQN_CAST(Dqn_f32)(sprite_sheet_row * sprite.sheet->sprite_size.y);
|
|
src_rect.size.w = DQN_CAST(Dqn_f32)sprite.sheet->sprite_size.w;
|
|
src_rect.size.h = DQN_CAST(Dqn_f32)sprite.sheet->sprite_size.h;
|
|
} break;
|
|
|
|
case TELY_AssetSpriteSheetType_Rects: {
|
|
DQN_ASSERT(sprite_index < sprite.sheet->rects.size);
|
|
src_rect = sprite.sheet->rects.data[sprite_index];
|
|
} break;
|
|
}
|
|
|
|
Dqn_f32 sprite_in_meters = FP_Game_PixelsToMetersNx1(game->play, src_rect.size.y);
|
|
Dqn_f32 size_scale = render_data.height.meters / sprite_in_meters;
|
|
|
|
Dqn_Rect dest_rect = {};
|
|
dest_rect.size = src_rect.size * size_scale;
|
|
dest_rect.pos = world_pos - (dest_rect.size * .5f);
|
|
|
|
if (sprite.flip & TELY_AssetFlip_X)
|
|
dest_rect.size.w *= -1.f; // NOTE: Flip the texture horizontally
|
|
|
|
if (sprite.flip & TELY_AssetFlip_Y)
|
|
dest_rect.size.h *= -1.f; // NOTE: Flip the texture vertically
|
|
|
|
TELY_Render_TextureColourV4(renderer,
|
|
sprite.sheet->tex_handle,
|
|
src_rect,
|
|
dest_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotate radians*/,
|
|
TELY_Colour_V4Alpha(TELY_COLOUR_WHITE_V4, entity->action.sprite_alpha));
|
|
|
|
if (entity->converted_faction) {
|
|
Dqn_V2 label_p = Dqn_Rect_InterpolatedPoint(dest_rect, Dqn_V2_InitNx2(0, -0.25f));
|
|
TELY_Render_TextF(renderer, label_p, Dqn_V2_InitNx2(0.f, 0.1f), "CONVERTED");
|
|
}
|
|
|
|
}
|
|
|
|
DQN_FOR_UINDEX(anim_index, entity->extra_cosmetic_anims.size) {
|
|
FP_GameRenderSprite *sprite = entity->extra_cosmetic_anims.data + anim_index;
|
|
uint64_t elapsed_ms = game->play.clock_ms - sprite->started_at_clock_ms;
|
|
uint16_t raw_anim_frame = DQN_CAST(uint16_t)(elapsed_ms / sprite->asset.anim->ms_per_frame);
|
|
if (raw_anim_frame > sprite->asset.anim->count && !sprite->loop) {
|
|
anim_index = Dqn_FArray_EraseRange(&entity->extra_cosmetic_anims, anim_index, 1, Dqn_ArrayErase_Unstable).it_index;
|
|
continue;
|
|
}
|
|
|
|
uint16_t anim_frame = raw_anim_frame % sprite->asset.anim->count;
|
|
Dqn_usize sprite_index = sprite->asset.anim->index + anim_frame;
|
|
Dqn_Rect src_rect = {};
|
|
switch (sprite->asset.sheet->type) {
|
|
case TELY_AssetSpriteSheetType_Uniform: {
|
|
Dqn_usize sprite_sheet_row = sprite_index / sprite->asset.sheet->sprites_per_row;
|
|
Dqn_usize sprite_sheet_column = sprite_index % sprite->asset.sheet->sprites_per_row;
|
|
src_rect.pos.x = DQN_CAST(Dqn_f32)(sprite_sheet_column * sprite->asset.sheet->sprite_size.w);
|
|
src_rect.pos.y = DQN_CAST(Dqn_f32)(sprite_sheet_row * sprite->asset.sheet->sprite_size.y);
|
|
src_rect.size.w = DQN_CAST(Dqn_f32)sprite->asset.sheet->sprite_size.w;
|
|
src_rect.size.h = DQN_CAST(Dqn_f32)sprite->asset.sheet->sprite_size.h;
|
|
} break;
|
|
|
|
case TELY_AssetSpriteSheetType_Rects: {
|
|
DQN_ASSERT(sprite_index < sprite->asset.sheet->rects.size);
|
|
src_rect = sprite->asset.sheet->rects.data[sprite_index];
|
|
} break;
|
|
}
|
|
|
|
Dqn_f32 sprite_in_meters = FP_Game_PixelsToMetersNx1(game->play, src_rect.size.y);
|
|
Dqn_f32 size_scale = sprite->height.meters / sprite_in_meters;
|
|
|
|
Dqn_Rect dest_rect = {};
|
|
dest_rect.size = src_rect.size * size_scale;
|
|
dest_rect.pos = world_pos - (dest_rect.size * .5f) + sprite->offset;
|
|
|
|
if (sprite->asset.flip & TELY_AssetFlip_X)
|
|
dest_rect.size.w *= -1.f; // NOTE: Flip the texture horizontally
|
|
|
|
if (sprite->asset.flip & TELY_AssetFlip_Y)
|
|
dest_rect.size.h *= -1.f; // NOTE: Flip the texture vertically
|
|
|
|
TELY_Render_TextureColourV4(renderer,
|
|
sprite->asset.sheet->tex_handle,
|
|
src_rect,
|
|
dest_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
TELY_COLOUR_WHITE_V4);
|
|
}
|
|
|
|
if (entity->handle != game->play.player && entity->handle != game->play.heart && entity->hp != entity->hp_cap && entity->hp) {
|
|
Dqn_f32 bar_height = 12.f;
|
|
TELY_AssetSpriteAnimation *anim = TELY_Asset_GetSpriteAnimation(&game->atlas_sprite_sheet, g_anim_names.icon_health);
|
|
Dqn_Rect icon_tex_rect = game->atlas_sprite_sheet.rects.data[anim->index];
|
|
|
|
Dqn_V2 draw_p = Dqn_Rect_InterpolatedPoint(world_hit_box, Dqn_V2_InitNx2(0.f, -0.2f));
|
|
Dqn_f32 health_t = entity->hp / DQN_CAST(Dqn_f32)entity->hp_cap;
|
|
Dqn_Rect health_rect = Dqn_Rect_InitNx4(draw_p.x, draw_p.y, DQN_CAST(Dqn_f32)entity->hp_cap, bar_height);
|
|
Dqn_Rect curr_health_rect = Dqn_Rect_InitNx4(draw_p.x, draw_p.y, DQN_CAST(Dqn_f32)entity->hp_cap * health_t, bar_height);
|
|
|
|
TELY_Render_RectColourV4(renderer, curr_health_rect, TELY_RenderShapeMode_Fill, TELY_COLOUR_RED_TOMATO_V4);
|
|
TELY_RenderCommandRect *cmd = TELY_Render_RectColourV4(renderer, health_rect, TELY_RenderShapeMode_Line, TELY_COLOUR_BLACK_V4);
|
|
cmd->thickness = 1.f;
|
|
}
|
|
|
|
if (entity->type == FP_EntityType_ClubTerry ||
|
|
entity->type == FP_EntityType_AirportTerry ||
|
|
entity->type == FP_EntityType_ChurchTerry) {
|
|
|
|
FP_GameEntityAction const *action = &entity->action;
|
|
bool draw_timer = entity->type == FP_EntityType_ClubTerry && action->state == FP_EntityClubTerryState_PartyTime ||
|
|
entity->type == FP_EntityType_AirportTerry && action->state == FP_EntityAirportTerryState_FlyPassenger ||
|
|
entity->type == FP_EntityType_ChurchTerry && action->state == FP_EntityChurchTerryState_ConvertPatron;
|
|
if (draw_timer) {
|
|
Dqn_f32 duration = action->end_at_clock_ms - DQN_CAST(Dqn_f32)action->started_at_clock_ms;
|
|
Dqn_f32 elapsed = DQN_CAST(Dqn_f32)(game->play.clock_ms - action->started_at_clock_ms);
|
|
Dqn_f32 t01 = DQN_MIN(1.f, elapsed / duration);
|
|
Dqn_Rect rect = {};
|
|
rect.pos = Dqn_Rect_InterpolatedPoint(world_hit_box, Dqn_V2_InitNx2(0, -0.3f));
|
|
rect.size = Dqn_V2_InitNx2(world_hit_box.size.w * t01, 16.f);
|
|
TELY_Render_RectColourV4(renderer, rect, TELY_RenderShapeMode_Fill, TELY_Colour_V4Alpha(TELY_COLOUR_RED_TOMATO_V4, 1.0f));
|
|
}
|
|
}
|
|
|
|
if (game->play.in_game_menu == FP_GameInGameMenu_Build) {
|
|
if (entity->flags & FP_GameEntityFlag_BuildZone)
|
|
TELY_Render_RectColourV4(renderer, world_hit_box, TELY_RenderShapeMode_Fill, TELY_Colour_V4Alpha(TELY_COLOUR_BLUE_CADET_V4, 0.5f));
|
|
}
|
|
|
|
if (game->play.debug_ui) {
|
|
// NOTE: Render waypoint entities ======================================================
|
|
if (entity->type == FP_EntityType_MobSpawner) {
|
|
Dqn_V2 start = world_pos;
|
|
for (FP_GameEntity *waypoint_entity = entity->first_child; waypoint_entity; waypoint_entity = waypoint_entity->next) {
|
|
if ((waypoint_entity->flags & FP_GameEntityFlag_MobSpawnerWaypoint) == 0)
|
|
continue;
|
|
|
|
Dqn_V2 end = FP_Game_CalcEntityWorldPos(game, waypoint_entity->handle);
|
|
TELY_Render_LineColourV4(renderer, start, end, TELY_COLOUR_BLUE_CADET_V4, 2.f);
|
|
start = end;
|
|
}
|
|
TELY_Render_RectColourV4(renderer, world_hit_box, TELY_RenderShapeMode_Line, TELY_COLOUR_BLUE_CADET_V4);
|
|
}
|
|
|
|
// NOTE: Draw the waypoints that the entity is moving along
|
|
Dqn_V2 start = world_pos;
|
|
for (FP_SentinelListLink<FP_GameWaypoint> *link = nullptr; FP_SentinelList_Iterate(&entity->waypoints, &link); ) {
|
|
Dqn_V2 end = FP_Game_CalcWaypointWorldPos(game, entity->handle, &link->data);
|
|
TELY_Render_LineColourV4(renderer, start, end, TELY_COLOUR_WHITE_PALE_GOLDENROD_V4, 2.f);
|
|
start = end;
|
|
}
|
|
|
|
if (entity->flags & FP_GameEntityFlag_MobSpawnerWaypoint)
|
|
TELY_Render_CircleColourV4(renderer, world_pos, 16.f /*radius*/, TELY_RenderShapeMode_Line, TELY_COLOUR_BLUE_CADET_V4);
|
|
|
|
if (entity->flags & FP_GameEntityFlag_BuildZone) {
|
|
{
|
|
Dqn_V2 line_p1 = Dqn_Rect_InterpolatedPoint(world_hit_box, Dqn_V2_InitNx2(0, 0));
|
|
Dqn_V2 line_p2 = Dqn_Rect_InterpolatedPoint(world_hit_box, Dqn_V2_InitNx2(1, 1));
|
|
TELY_Render_LineColourV4(renderer, line_p1, line_p2, TELY_COLOUR_BLACK_V4, 2.f);
|
|
}
|
|
{
|
|
Dqn_V2 line_p1 = Dqn_Rect_InterpolatedPoint(world_hit_box, Dqn_V2_InitNx2(0, 1));
|
|
Dqn_V2 line_p2 = Dqn_Rect_InterpolatedPoint(world_hit_box, Dqn_V2_InitNx2(1, 0));
|
|
TELY_Render_LineColourV4(renderer, line_p1, line_p2, TELY_COLOUR_BLACK_V4, 2.f);
|
|
}
|
|
}
|
|
|
|
// NOTE: Render attack box =================================================================
|
|
{
|
|
Dqn_FArray<Dqn_Rect, FP_GameDirection_Count> attack_boxes = FP_Game_CalcEntityMeleeAttackBoxes(game, entity->handle);
|
|
for (Dqn_Rect box : attack_boxes)
|
|
TELY_Render_RectColourV4(renderer, box, TELY_RenderShapeMode_Line, TELY_COLOUR_RED_TOMATO_V4);
|
|
}
|
|
|
|
// NOTE: Render world position =============================================================
|
|
TELY_Render_CircleColourV4(renderer, world_pos, 4.f, TELY_RenderShapeMode_Fill, TELY_COLOUR_RED_TOMATO_V4);
|
|
|
|
// NOTE: Render hot/active entity ==========================================================
|
|
if (game->play.clicked_entity == entity->handle) {
|
|
TELY_Render_RectColourV4(renderer, world_hit_box, TELY_RenderShapeMode_Line, TELY_COLOUR_WHITE_PALE_GOLDENROD_V4);
|
|
|
|
} else if (game->play.hot_entity == entity->handle || (entity->flags & FP_GameEntityFlag_DrawHitBox)) {
|
|
Dqn_V4 hot_colour = game->play.hot_entity == entity->handle ? TELY_COLOUR_RED_TOMATO_V4 : TELY_Colour_V4Alpha(TELY_COLOUR_YELLOW_SANDY_V4, .5f);
|
|
TELY_Render_RectColourV4(renderer, world_hit_box, TELY_RenderShapeMode_Line, hot_colour);
|
|
}
|
|
|
|
if (game->play.hot_entity == entity->handle) {
|
|
if (entity->name.size) {
|
|
Dqn_V2I player_tile = Dqn_V2I_InitNx2(world_pos.x / game->play.tile_size, world_pos.y / game->play.tile_size);
|
|
Dqn_f32 line_height = TELY_Render_FontHeight(renderer, &platform->assets);
|
|
Dqn_V2 draw_p = world_mouse_p;
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_InitNx2(0.f, 1), "%.*s", DQN_STRING_FMT(entity->name)); draw_p.y += line_height;
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_InitNx2(0.f, 1), "World Pos: (%.1f, %.1f)", world_pos.x, world_pos.y); draw_p.y += line_height;
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_InitNx2(0.f, 1), "Hit Box Size: %.1fx%.1f", world_hit_box.size.x, world_hit_box.size.y); draw_p.y += line_height;
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_InitNx2(0.f, 1), "Tile: %I32dx%I32d", player_tile.x, player_tile.y); draw_p.y += line_height;
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_InitNx2(0.f, 1), "World Mouse Pos: (%.1f, %.1f)", world_mouse_p.x, world_mouse_p.y); draw_p.y += line_height;
|
|
|
|
Dqn_String8 faction = {};
|
|
switch (entity->faction) {
|
|
case FP_GameEntityFaction_Nil: faction = DQN_STRING8("Nil"); break;
|
|
case FP_GameEntityFaction_Friendly: faction = DQN_STRING8("Friendly"); break;
|
|
case FP_GameEntityFaction_Foe: faction = DQN_STRING8("Foe"); break;
|
|
}
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_InitNx2(0.f, 1), "Faction: %.*s", DQN_STRING_FMT(faction)); draw_p.y += line_height;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// NOTE: Render overlay UI =====================================================================
|
|
if (game->play.state == FP_GameState_Pause || game->play.state == FP_GameState_Play) {
|
|
|
|
// NOTE: Render the merchant menus =========================================================
|
|
FP_GameEntity *player = FP_Game_GetEntity(game, game->play.player);
|
|
Dqn_V2 player_pos = FP_Game_CalcEntityWorldPos(game, game->play.player);
|
|
{
|
|
static bool sound_played_flags[4] = {false, false, false, false};
|
|
FP_GameInventory *invent = &player->inventory;
|
|
struct FP_MerchantToMenuMapping {
|
|
FP_GameEntityHandle merchant;
|
|
Dqn_String8 menu_anim;
|
|
Dqn_String8 building;
|
|
Dqn_V2 building_offset01;
|
|
uint8_t *inventory_count;
|
|
uint32_t *building_base_price;
|
|
uint32_t *upgrade_base_price;
|
|
FP_GameAudio audio_type;
|
|
bool *sound_played;
|
|
} merchants[] = {
|
|
{game->play.merchant_terry, g_anim_names.merchant_terry_menu, g_anim_names.club_terry_dark, Dqn_V2_InitNx2(0.015f, +0.04f), &invent->clubs, &invent->clubs_base_price, &invent->health_base_price, FP_GameAudio_MerchantTerry, &sound_played_flags[0]},
|
|
{game->play.merchant_graveyard, g_anim_names.merchant_graveyard_menu, g_anim_names.church_terry_dark, Dqn_V2_InitNx2(0.04f, -0.15f), &invent->churchs, &invent->churchs_base_price, &invent->stamina_base_price, FP_GameAudio_MerchantGhost, &sound_played_flags[1]},
|
|
{game->play.merchant_gym, g_anim_names.merchant_gym_menu, g_anim_names.kennel_terry, Dqn_V2_InitNx2(0, +0), &invent->kennels, &invent->kennels_base_price, &invent->attack_base_price, FP_GameAudio_MerchantGym, &sound_played_flags[2]},
|
|
{game->play.merchant_phone_company, g_anim_names.merchant_phone_company_menu, g_anim_names.airport_terry, Dqn_V2_InitNx2(0, -0.1f), &invent->airports, &invent->airports_base_price, &invent->mobile_plan_base_price, FP_GameAudio_MerchantPhone, &sound_played_flags[3]},
|
|
};
|
|
|
|
bool activated_merchant = false;
|
|
for (FP_MerchantToMenuMapping mapping : merchants) {
|
|
FP_GameEntityHandle merchant_handle = mapping.merchant;
|
|
Dqn_V2 world_pos = FP_Game_CalcEntityWorldPos(game, merchant_handle);
|
|
Dqn_f32 dist_squared = Dqn_V2_LengthSq_V2x2(world_pos, player_pos);
|
|
|
|
if (dist_squared > DQN_SQUARED(FP_Game_MetersToPixelsNx1(game->play, 4))) {
|
|
*mapping.sound_played = false;
|
|
continue;
|
|
}
|
|
|
|
// NOTE: Render animated merchant menu =============================
|
|
activated_merchant = true;
|
|
Dqn_Rect merchant_menu_rect = {};
|
|
{
|
|
FP_GameRenderSprite *sprite = &game->play.player_merchant_menu;
|
|
if (!sprite->asset.anim || sprite->asset.anim->label != mapping.menu_anim) {
|
|
sprite->asset = TELY_Asset_MakeAnimatedSprite(&game->atlas_sprite_sheet, mapping.menu_anim, TELY_AssetFlip_No);
|
|
sprite->started_at_clock_ms = game->play.clock_ms;
|
|
}
|
|
|
|
uint64_t elapsed_ms = game->play.clock_ms - sprite->started_at_clock_ms;
|
|
uint16_t raw_anim_frame = DQN_CAST(uint16_t)(elapsed_ms / sprite->asset.anim->ms_per_frame);
|
|
uint16_t anim_frame = raw_anim_frame % sprite->asset.anim->count;
|
|
|
|
Dqn_usize sprite_index = sprite->asset.anim->index + anim_frame;
|
|
Dqn_Rect src_rect = sprite->asset.sheet->rects.data[sprite_index];
|
|
|
|
merchant_menu_rect.size = src_rect.size;
|
|
merchant_menu_rect.pos = world_pos - (merchant_menu_rect.size * .5f) - Dqn_V2_InitNx2(0.f, src_rect.size.y);
|
|
Dqn_f32 sin_t = DQN_SINF(DQN_CAST(Dqn_f32)input->timer_s * 3.f);
|
|
merchant_menu_rect.pos.y += sin_t * 4.f;
|
|
|
|
TELY_Render_TextureColourV4(renderer,
|
|
sprite->asset.sheet->tex_handle,
|
|
src_rect,
|
|
merchant_menu_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
TELY_COLOUR_WHITE_V4);
|
|
|
|
if (activated_merchant && !*mapping.sound_played) {
|
|
TELY_Audio_Play(audio, game->audio[mapping.audio_type], 1.f);
|
|
*mapping.sound_played = true;
|
|
}
|
|
}
|
|
|
|
TELY_Render_PushColourV4(renderer, TELY_COLOUR_WHITE_V4);
|
|
TELY_Render_PushFont(renderer, game->talkco_font_large);
|
|
DQN_DEFER {
|
|
TELY_Render_PopFont(renderer);
|
|
TELY_Render_PopColourV4(renderer);
|
|
};
|
|
|
|
// NOTE: Render the merchant button for buildings ==================
|
|
uint64_t const buy_duration_ms = 500;
|
|
{
|
|
bool const have_enough_coins = player->coins >= *mapping.building_base_price;
|
|
// NOTE: Buy trigger + animation ===============================
|
|
{
|
|
TELY_PlatformInputScanCode key = TELY_PlatformInputScanCode_J;
|
|
bool trigger_buy_anim = false;
|
|
|
|
if (have_enough_coins) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, key)) {
|
|
game->play.player_trigger_purchase_building_timestamp = game->play.clock_ms + buy_duration_ms;
|
|
} else if (TELY_Platform_InputScanCodeIsDown(input, key)) {
|
|
trigger_buy_anim = true;
|
|
if (game->play.clock_ms > game->play.player_trigger_purchase_building_timestamp)
|
|
game->play.player_trigger_purchase_building_timestamp = game->play.clock_ms;
|
|
} else if (TELY_Platform_InputScanCodeIsReleased(input, key)) {
|
|
if (game->play.clock_ms > game->play.player_trigger_purchase_building_timestamp) {
|
|
if (mapping.inventory_count) {
|
|
player->coins -= *mapping.building_base_price;
|
|
TELY_Audio_Play(audio, game->audio[FP_GameAudio_Ching], 1.f);
|
|
*mapping.building_base_price *= 2;
|
|
|
|
// NOTE: Raise the prices of everything else
|
|
invent->airports_base_price *= 2;
|
|
invent->clubs_base_price *= 2;
|
|
invent->kennels_base_price *= 2;
|
|
invent->churchs_base_price *= 2;
|
|
|
|
(*mapping.inventory_count)++;
|
|
}
|
|
} else {
|
|
game->play.player_trigger_purchase_building_timestamp = UINT64_MAX;
|
|
}
|
|
}
|
|
|
|
if (trigger_buy_anim) {
|
|
uint64_t start_buy_time = game->play.player_trigger_purchase_building_timestamp - buy_duration_ms;
|
|
uint64_t elapsed_time = game->play.clock_ms - start_buy_time;
|
|
|
|
Dqn_f32 buy_t = DQN_MIN(elapsed_time / DQN_CAST(Dqn_f32)buy_duration_ms, 1.f);
|
|
Dqn_Rect buy_lerp_rect = {};
|
|
buy_lerp_rect.pos = Dqn_Rect_InterpolatedPoint(merchant_menu_rect, Dqn_V2_InitNx2(0.297f, 0.215f));
|
|
buy_lerp_rect.size.w = (merchant_menu_rect.size.w * 0.38f) * buy_t;
|
|
buy_lerp_rect.size.h = merchant_menu_rect.size.h * .611f;
|
|
|
|
TELY_Render_RectColourV4(renderer, buy_lerp_rect, TELY_RenderShapeMode_Fill, TELY_Colour_V4Alpha(TELY_COLOUR_BLUE_CADET_V4, 0.5f));
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// NOTE: Render the (A) button =================================
|
|
Dqn_V4 tex_mod_colour = have_enough_coins ? TELY_COLOUR_WHITE_V4 : TELY_Colour_V4Alpha(TELY_COLOUR_RED_TOMATO_V4, .5f);
|
|
{
|
|
TELY_AssetSpriteAnimation *anim = TELY_Asset_GetSpriteAnimation(&game->atlas_sprite_sheet, g_anim_names.merchant_button_a);
|
|
Dqn_Rect button_rect = game->atlas_sprite_sheet.rects.data[anim->index];
|
|
Dqn_Rect dest_rect = {};
|
|
dest_rect.size = button_rect.size * 1.5f;
|
|
dest_rect.pos = Dqn_Rect_InterpolatedPoint(merchant_menu_rect, Dqn_V2_InitNx2(0.345f, 0.5f));
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
button_rect,
|
|
dest_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
tex_mod_colour);
|
|
|
|
TELY_Render_TextF(renderer, Dqn_Rect_InterpolatedPoint(dest_rect, Dqn_V2_InitNx2(0.5f, -1)), Dqn_V2_InitNx2(0.5, 0.f), "$%u", *mapping.building_base_price);
|
|
}
|
|
|
|
// NOTE: Render the merchant shop item building ================
|
|
{
|
|
TELY_AssetSpriteAnimation *anim = TELY_Asset_GetSpriteAnimation(&game->atlas_sprite_sheet, mapping.building);
|
|
Dqn_Rect tex_rect = game->atlas_sprite_sheet.rects.data[anim->index];
|
|
Dqn_Rect dest_rect = {};
|
|
dest_rect.size = tex_rect.size * 0.35f;
|
|
dest_rect.pos = Dqn_Rect_InterpolatedPoint(merchant_menu_rect, Dqn_V2_InitNx2(0.42f, 0.25f) + mapping.building_offset01);
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
tex_rect,
|
|
dest_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
tex_mod_colour);
|
|
}
|
|
}
|
|
|
|
// NOTE: Render the merchant button for buildings
|
|
{
|
|
bool const have_enough_coins = player->coins >= *mapping.upgrade_base_price;
|
|
// NOTE: Buy trigger + animation ===============================
|
|
{
|
|
TELY_PlatformInputScanCode key = TELY_PlatformInputScanCode_K;
|
|
bool trigger_buy_anim = false;
|
|
|
|
if (have_enough_coins) {
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, key)) {
|
|
game->play.player_trigger_purchase_upgrade_timestamp = game->play.clock_ms + buy_duration_ms;
|
|
} else if (TELY_Platform_InputScanCodeIsDown(input, key)) {
|
|
trigger_buy_anim = true;
|
|
if (game->play.clock_ms > game->play.player_trigger_purchase_upgrade_timestamp)
|
|
game->play.player_trigger_purchase_upgrade_timestamp = game->play.clock_ms;
|
|
} else if (TELY_Platform_InputScanCodeIsReleased(input, key)) {
|
|
if (game->play.clock_ms > game->play.player_trigger_purchase_upgrade_timestamp) {
|
|
player->coins -= *mapping.upgrade_base_price;
|
|
*mapping.upgrade_base_price *= 2;
|
|
TELY_Audio_Play(audio, game->audio[FP_GameAudio_Ching], 1.f);
|
|
|
|
if (mapping.merchant == game->play.merchant_terry) {
|
|
// TODO(doyle): Attack damage? Or increase attack range?
|
|
player->base_attack += DQN_CAST(uint32_t)(FP_DEFAULT_DAMAGE * .3f);
|
|
} else if (mapping.merchant == game->play.merchant_graveyard) {
|
|
player->stamina_cap += DQN_CAST(uint32_t)(FP_TERRY_DASH_STAMINA_COST * .5f);
|
|
} else if (mapping.merchant == game->play.merchant_gym) {
|
|
player->hp_cap += FP_DEFAULT_DAMAGE;
|
|
player->hp = player->hp_cap;
|
|
} else if (mapping.merchant == game->play.merchant_phone_company) {
|
|
player->terry_mobile_data_plan_cap += DQN_KILOBYTES(1);
|
|
}
|
|
} else {
|
|
game->play.player_trigger_purchase_upgrade_timestamp = UINT64_MAX;
|
|
}
|
|
}
|
|
|
|
if (trigger_buy_anim) {
|
|
uint64_t start_buy_time = game->play.player_trigger_purchase_upgrade_timestamp - buy_duration_ms;
|
|
uint64_t elapsed_time = game->play.clock_ms - start_buy_time;
|
|
|
|
Dqn_f32 buy_t = DQN_MIN(elapsed_time / DQN_CAST(Dqn_f32)buy_duration_ms, 1.f);
|
|
Dqn_Rect buy_lerp_rect = {};
|
|
buy_lerp_rect.pos = Dqn_Rect_InterpolatedPoint(merchant_menu_rect, Dqn_V2_InitNx2(0.68f, 0.215f));
|
|
buy_lerp_rect.size.w = (merchant_menu_rect.size.w * 0.211f) * buy_t;
|
|
buy_lerp_rect.size.h = merchant_menu_rect.size.h * .611f;
|
|
|
|
TELY_Render_RectColourV4(renderer, buy_lerp_rect, TELY_RenderShapeMode_Fill, TELY_Colour_V4Alpha(TELY_COLOUR_RED_TOMATO_V4, 0.5f));
|
|
}
|
|
}
|
|
}
|
|
|
|
// NOTE: Render the (B) button =================================
|
|
Dqn_V4 tex_mod_colour = have_enough_coins ? TELY_COLOUR_WHITE_V4 : TELY_Colour_V4Alpha(TELY_COLOUR_RED_TOMATO_V4, .5f);
|
|
{
|
|
TELY_AssetSpriteAnimation *anim = TELY_Asset_GetSpriteAnimation(&game->atlas_sprite_sheet, g_anim_names.merchant_button_b);
|
|
Dqn_Rect button_rect = game->atlas_sprite_sheet.rects.data[anim->index];
|
|
Dqn_Rect dest_rect = {};
|
|
dest_rect.size = button_rect.size * 1.5f;
|
|
dest_rect.pos = Dqn_Rect_InterpolatedPoint(merchant_menu_rect, Dqn_V2_InitNx2(0.75f, 0.5f));
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
button_rect,
|
|
dest_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
tex_mod_colour);
|
|
|
|
TELY_Render_TextF(renderer, Dqn_Rect_InterpolatedPoint(dest_rect, Dqn_V2_InitNx2(0.5f, -1)), Dqn_V2_InitNx2(0.5, 0.f), "$%u", *mapping.upgrade_base_price);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (activated_merchant) {
|
|
game->play.in_game_menu = FP_GameInGameMenu_Merchant;
|
|
} else {
|
|
if (game->play.in_game_menu == FP_GameInGameMenu_Merchant) {
|
|
game->play.in_game_menu = FP_GameInGameMenu_Nil;
|
|
}
|
|
}
|
|
}
|
|
|
|
// NOTE: Render player avatar HUD ==========================================================
|
|
Dqn_Rect player_avatar_rect = {};
|
|
player_avatar_rect.pos = Dqn_V2_InitNx1(32.f);
|
|
Dqn_V2 next_pos = {};
|
|
{
|
|
TELY_Render_PushTransform(renderer, Dqn_M2x3_Identity());
|
|
DQN_DEFER { TELY_Render_PopTransform(renderer); };
|
|
|
|
FP_EntityRenderData render_data = FP_Entity_GetRenderData(game, FP_EntityType_Terry, FP_EntityTerryState_Idle, FP_GameDirection_Down);
|
|
player_avatar_rect.size = render_data.render_size;
|
|
|
|
TELY_Render_TextureColourV4(renderer,
|
|
render_data.sheet->tex_handle,
|
|
render_data.sheet_rect,
|
|
player_avatar_rect,
|
|
Dqn_V2_Zero,
|
|
0.f,
|
|
TELY_COLOUR_WHITE_V4);
|
|
|
|
TELY_Render_PushFont(renderer, game->talkco_font);
|
|
DQN_DEFER { TELY_Render_PopFont(renderer); };
|
|
|
|
next_pos = Dqn_Rect_InterpolatedPoint(player_avatar_rect, Dqn_V2_InitNx2(1.f, 0));
|
|
Dqn_f32 font_height = TELY_Render_FontHeight(renderer, &platform->assets);
|
|
|
|
// TELY_Render_TextF(renderer, next_pos, Dqn_V2_Zero, "Terry");
|
|
// next_pos.y += font_height;
|
|
|
|
// NOTE: Health bar ====================================================
|
|
Dqn_f32 bar_height = font_height * .75f;
|
|
Dqn_Rect health_icon_rect = {};
|
|
{
|
|
TELY_AssetSpriteAnimation *anim = TELY_Asset_GetSpriteAnimation(&game->atlas_sprite_sheet, g_anim_names.icon_health);
|
|
Dqn_Rect icon_tex_rect = game->atlas_sprite_sheet.rects.data[anim->index];
|
|
{
|
|
health_icon_rect.size = icon_tex_rect.size * .4f;
|
|
health_icon_rect.pos = Dqn_V2_InitNx2(next_pos.x - health_icon_rect.size.x * .25f, next_pos.y - (health_icon_rect.size.y * .35f));
|
|
}
|
|
|
|
Dqn_f32 bar_x = next_pos.x + (health_icon_rect.size.x * .25f);
|
|
Dqn_f32 health_t = player->hp / DQN_CAST(Dqn_f32)player->hp_cap;
|
|
Dqn_Rect health_rect = Dqn_Rect_InitNx4(bar_x, next_pos.y, DQN_CAST(Dqn_f32)player->hp_cap, bar_height);
|
|
Dqn_Rect curr_health_rect = Dqn_Rect_InitNx4(bar_x, next_pos.y, DQN_CAST(Dqn_f32)player->hp_cap * health_t, bar_height);
|
|
|
|
TELY_Render_RectColourV4(renderer, curr_health_rect, TELY_RenderShapeMode_Fill, TELY_COLOUR_RED_TOMATO_V4);
|
|
|
|
TELY_RenderCommandRect *cmd = TELY_Render_RectColourV4(
|
|
renderer,
|
|
health_rect,
|
|
TELY_RenderShapeMode_Line,
|
|
TELY_COLOUR_BLACK_V4);
|
|
cmd->thickness = 4.f;
|
|
|
|
// NOTE: Draw the heart icon shadow
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
icon_tex_rect,
|
|
Dqn_Rect_InitV2x2(health_icon_rect.pos - (cmd->thickness * 2.f), health_icon_rect.size + (cmd->thickness * 4.f)),
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
TELY_COLOUR_BLACK_V4);
|
|
|
|
// NOTE: Draw the heart icon
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
icon_tex_rect,
|
|
health_icon_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
TELY_COLOUR_WHITE_V4);
|
|
}
|
|
|
|
// NOTE: Stamina bar ===================================================
|
|
next_pos.y += health_icon_rect.size.h * .8f;
|
|
Dqn_Rect stamina_icon_rect = {};
|
|
{
|
|
TELY_AssetSpriteAnimation *anim = TELY_Asset_GetSpriteAnimation(&game->atlas_sprite_sheet, g_anim_names.icon_stamina);
|
|
Dqn_Rect icon_tex_rect = game->atlas_sprite_sheet.rects.data[anim->index];
|
|
{
|
|
stamina_icon_rect.size = icon_tex_rect.size * .35f;
|
|
stamina_icon_rect.pos = Dqn_V2_InitNx2(next_pos.x - stamina_icon_rect.size.x * .25f, next_pos.y - (stamina_icon_rect.size.y * .35f));
|
|
}
|
|
|
|
Dqn_f32 bar_x = next_pos.x + (stamina_icon_rect.size.x * .25f);
|
|
Dqn_f32 stamina_t = player->stamina / DQN_CAST(Dqn_f32)player->stamina_cap;
|
|
Dqn_Rect stamina_rect = Dqn_Rect_InitNx4(bar_x, next_pos.y, DQN_CAST(Dqn_f32)player->stamina_cap, bar_height);
|
|
Dqn_Rect curr_stamina_rect = Dqn_Rect_InitNx4(bar_x, next_pos.y, DQN_CAST(Dqn_f32)player->stamina_cap * stamina_t, bar_height);
|
|
TELY_Render_RectColourV4(
|
|
renderer,
|
|
curr_stamina_rect,
|
|
TELY_RenderShapeMode_Fill,
|
|
TELY_COLOUR_YELLOW_SANDY_V4);
|
|
|
|
TELY_RenderCommandRect *cmd = TELY_Render_RectColourV4(
|
|
renderer,
|
|
stamina_rect,
|
|
TELY_RenderShapeMode_Line,
|
|
TELY_COLOUR_BLACK_V4);
|
|
cmd->thickness = 4.f;
|
|
|
|
// NOTE: Draw the icon shadow
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
icon_tex_rect,
|
|
Dqn_Rect_InitV2x2(stamina_icon_rect.pos - (cmd->thickness * 2.f), stamina_icon_rect.size + (cmd->thickness * 4.f)),
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
TELY_COLOUR_BLACK_V4);
|
|
|
|
// NOTE: Draw the icon
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
icon_tex_rect,
|
|
stamina_icon_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
TELY_COLOUR_WHITE_V4);
|
|
}
|
|
|
|
// NOTE: Mobile data bar ===================================================
|
|
next_pos.y += stamina_icon_rect.size.h * .8f;
|
|
Dqn_Rect phone_icon_rect = {};
|
|
{
|
|
TELY_AssetSpriteAnimation *anim = TELY_Asset_GetSpriteAnimation(&game->atlas_sprite_sheet, g_anim_names.icon_phone);
|
|
Dqn_Rect icon_tex_rect = game->atlas_sprite_sheet.rects.data[anim->index];
|
|
{
|
|
phone_icon_rect.size = icon_tex_rect.size * .35f;
|
|
phone_icon_rect.pos = Dqn_V2_InitNx2(next_pos.x - phone_icon_rect.size.x * .25f, next_pos.y - (phone_icon_rect.size.y * .35f));
|
|
}
|
|
|
|
Dqn_f32 pixels_per_kb = 15.5f;
|
|
Dqn_f32 bar_width = DQN_CAST(Dqn_f32)player->terry_mobile_data_plan_cap / DQN_KILOBYTES(1) * pixels_per_kb;
|
|
|
|
Dqn_f32 bar_x = next_pos.x + (phone_icon_rect.size.x * .25f);
|
|
Dqn_f32 data_plan_t = player->terry_mobile_data_plan / DQN_CAST(Dqn_f32)player->terry_mobile_data_plan_cap;
|
|
Dqn_Rect data_plan_rect = Dqn_Rect_InitNx4(bar_x, next_pos.y, bar_width, bar_height);
|
|
Dqn_Rect curr_data_plan_rect = Dqn_Rect_InitNx4(bar_x, next_pos.y, bar_width * data_plan_t, bar_height);
|
|
TELY_Render_RectColourV4(
|
|
renderer,
|
|
curr_data_plan_rect,
|
|
TELY_RenderShapeMode_Fill,
|
|
TELY_COLOUR_BLUE_CADET_V4);
|
|
|
|
TELY_RenderCommandRect *cmd = TELY_Render_RectColourV4(
|
|
renderer,
|
|
data_plan_rect,
|
|
TELY_RenderShapeMode_Line,
|
|
TELY_COLOUR_BLACK_V4);
|
|
cmd->thickness = 4.f;
|
|
|
|
// NOTE: Draw the icon shadow
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
icon_tex_rect,
|
|
Dqn_Rect_InitV2x2(phone_icon_rect.pos - (cmd->thickness * 2.f), phone_icon_rect.size + (cmd->thickness * 4.f)),
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
TELY_COLOUR_BLACK_V4);
|
|
|
|
// NOTE: Draw the icon
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
icon_tex_rect,
|
|
phone_icon_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
TELY_COLOUR_WHITE_V4);
|
|
}
|
|
|
|
next_pos.y += phone_icon_rect.size.h * .8f;
|
|
Dqn_Rect money_icon_rect = {};
|
|
{
|
|
TELY_AssetSpriteAnimation *anim = TELY_Asset_GetSpriteAnimation(&game->atlas_sprite_sheet, g_anim_names.icon_money);
|
|
Dqn_Rect icon_tex_rect = game->atlas_sprite_sheet.rects.data[anim->index];
|
|
{
|
|
money_icon_rect.size = icon_tex_rect.size * .35f;
|
|
money_icon_rect.pos = Dqn_V2_InitNx2(next_pos.x - money_icon_rect.size.x * .25f, next_pos.y - (money_icon_rect.size.y * .15f));
|
|
}
|
|
|
|
TELY_Render_TextF(renderer,
|
|
Dqn_V2_InitNx2(next_pos.x + money_icon_rect.size.x * .75f, money_icon_rect.pos.y),
|
|
Dqn_V2_InitNx2(0, -0.5),
|
|
"$%I64u",
|
|
player->coins);
|
|
|
|
// NOTE: Draw the icon shadow
|
|
Dqn_f32 thickness = 4.f;
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
icon_tex_rect,
|
|
Dqn_Rect_InitV2x2(money_icon_rect.pos - (thickness * 2.f), money_icon_rect.size + (thickness * 4.f)),
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
TELY_COLOUR_BLACK_V4);
|
|
|
|
// NOTE: Draw the icon
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
icon_tex_rect,
|
|
money_icon_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
TELY_COLOUR_WHITE_V4);
|
|
}
|
|
|
|
next_pos.y += money_icon_rect.size.h;
|
|
|
|
#if 0
|
|
next_pos.y += font_height;
|
|
TELY_Render_TextF(renderer, next_pos, Dqn_V2_Zero, "[H] Build Menu");
|
|
|
|
next_pos.y += font_height;
|
|
TELY_Render_TextF(renderer, next_pos, Dqn_V2_Zero, "[Shift+WASD] Strafe");
|
|
|
|
next_pos.y += font_height;
|
|
TELY_Render_TextF(renderer, next_pos, Dqn_V2_Zero, "[Ctrl+WASD] Dash");
|
|
|
|
next_pos.y += font_height;
|
|
TELY_Render_TextF(renderer, next_pos, Dqn_V2_Zero, "[J|K] Melee/Range");
|
|
#endif
|
|
}
|
|
|
|
// NOTE: Render the wave ===================================================================
|
|
{
|
|
TELY_Render_PushTransform(renderer, Dqn_M2x3_Identity());
|
|
DQN_DEFER { TELY_Render_PopTransform(renderer); };
|
|
|
|
TELY_Render_PushFont(renderer, game->talkco_font_large);
|
|
DQN_DEFER { TELY_Render_PopFont(renderer); };
|
|
|
|
uint64_t time_until_next_wave_ms = 0;
|
|
if (game->play.enemies_spawned_this_wave >= game->play.enemies_per_wave) {
|
|
if (game->play.wave_cooldown_timestamp_ms > game->play.clock_ms) {
|
|
time_until_next_wave_ms = game->play.wave_cooldown_timestamp_ms - game->play.clock_ms;
|
|
}
|
|
}
|
|
|
|
Dqn_f32 mid_x = platform->core.window_size.x * .5f;
|
|
if (time_until_next_wave_ms) {
|
|
TELY_Render_TextF(renderer,
|
|
Dqn_V2_InitNx2(mid_x, player_avatar_rect.pos.y),
|
|
Dqn_V2_InitNx1(0.5f),
|
|
"%.1fs remaining until Wave %u!",
|
|
(time_until_next_wave_ms / 1000.f), game->play.current_wave + 1);
|
|
} else {
|
|
TELY_Render_TextF(renderer, Dqn_V2_InitNx2(mid_x, player_avatar_rect.pos.y), Dqn_V2_InitNx1(0.5f), "Wave %u", game->play.current_wave);
|
|
}
|
|
}
|
|
|
|
if (!FP_Game_IsNilEntityHandle(game, game->play.clicked_entity)) {
|
|
// NOTE: Render building blueprint =====================================================
|
|
if (game->play.in_game_menu == FP_GameInGameMenu_Build) {
|
|
FP_GameEntity *entity = FP_Game_GetEntity(game, game->play.clicked_entity);
|
|
FP_GamePlaceableBuilding placeable_building = PLACEABLE_BUILDINGS[game->play.build_mode_building_index];
|
|
FP_EntityRenderData render_data = FP_Entity_GetRenderData(game, placeable_building.type, placeable_building.state, entity->direction);
|
|
Dqn_Rect dest_rect = FP_Game_GetBuildingPlacementRectForEntity(game, placeable_building, entity->handle);
|
|
|
|
Dqn_V4 colour = game->play.build_mode_can_place_building ?
|
|
TELY_Colour_V4Alpha(TELY_COLOUR_WHITE_V4, 0.5f) :
|
|
TELY_Colour_V4Alpha(TELY_COLOUR_RED_V4, 0.5f);
|
|
|
|
TELY_Render_RectColourV4(renderer, dest_rect, TELY_RenderShapeMode_Fill, TELY_Colour_V4Alpha(TELY_COLOUR_BLUE_CADET_V4, 0.5f));
|
|
TELY_Render_TextureColourV4(renderer,
|
|
render_data.sheet->tex_handle,
|
|
render_data.sheet_rect,
|
|
dest_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
colour);
|
|
}
|
|
|
|
// NOTE: Render the building selector UI ===============================================
|
|
{
|
|
TELY_Render_PushTransform(renderer, Dqn_M2x3_Identity());
|
|
DQN_DEFER { TELY_Render_PopTransform(renderer); };
|
|
game->play.build_mode_building_index = DQN_CLAMP(game->play.build_mode_building_index, 0, DQN_ARRAY_UCOUNT(PLACEABLE_BUILDINGS) - 1);
|
|
|
|
Dqn_f32 building_ui_size = 64.f;
|
|
Dqn_f32 padding = 10.f;
|
|
Dqn_f32 start_x = player_avatar_rect.pos.x;
|
|
DQN_FOR_UINDEX (building_index, DQN_ARRAY_UCOUNT(PLACEABLE_BUILDINGS)) {
|
|
FP_GamePlaceableBuilding building = PLACEABLE_BUILDINGS[building_index];
|
|
FP_EntityRenderData render_data = FP_Entity_GetRenderData(game, building.type, building.state, FP_GameDirection_Down);
|
|
Dqn_Rect rect = Dqn_Rect_InitNx4(start_x + (building_index * building_ui_size) + (padding * building_index),
|
|
next_pos.y,
|
|
building_ui_size,
|
|
building_ui_size);
|
|
|
|
Dqn_V4 texture_colour = TELY_Colour_V4Alpha(TELY_COLOUR_WHITE_V4, .5f);
|
|
Dqn_V4 outline_colour = TELY_COLOUR_WHITE_PALE_GOLDENROD_V4;
|
|
if (game->play.build_mode_building_index == building_index) {
|
|
outline_colour = TELY_COLOUR_RED_TOMATO_V4;
|
|
texture_colour.a = 1.f;
|
|
}
|
|
|
|
TELY_Render_RectColourV4(renderer, rect, TELY_RenderShapeMode_Fill, TELY_Colour_V4Alpha(TELY_COLOUR_WHITE_V4, 0.5f));
|
|
TELY_Render_TextureColourV4(renderer,
|
|
render_data.sheet->tex_handle,
|
|
render_data.sheet_rect,
|
|
rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
texture_colour);
|
|
|
|
uint32_t building_count = 0;
|
|
if (building.type == FP_EntityType_ClubTerry)
|
|
building_count = player->inventory.clubs;
|
|
else if (building.type == FP_EntityType_AirportTerry)
|
|
building_count = player->inventory.airports;
|
|
else if (building.type == FP_EntityType_KennelTerry)
|
|
building_count = player->inventory.kennels;
|
|
else if (building.type == FP_EntityType_ChurchTerry)
|
|
building_count = player->inventory.churchs;
|
|
|
|
TELY_Render_PushFont(renderer, game->talkco_font);
|
|
DQN_DEFER { TELY_Render_PopFont(renderer); };
|
|
|
|
Dqn_V2 label_p = Dqn_Rect_InterpolatedPoint(rect, Dqn_V2_InitNx2(0.5f, 1.25f));
|
|
TELY_Render_TextF(renderer, label_p, Dqn_V2_InitNx2(0.5f, 0.5f), "x %u", building_count);
|
|
|
|
TELY_RenderCommandRect *cmd = TELY_Render_RectColourV4(renderer, rect, TELY_RenderShapeMode_Line, outline_colour);
|
|
cmd->thickness = 2.f;
|
|
}
|
|
}
|
|
}
|
|
|
|
// NOTE: Render the heart health ===========================================================
|
|
{
|
|
TELY_Render_PushTransform(renderer, Dqn_M2x3_Identity());
|
|
DQN_DEFER { TELY_Render_PopTransform(renderer); };
|
|
|
|
Dqn_f32 font_height = TELY_Render_FontHeight(renderer, assets);
|
|
Dqn_f32 bar_height = font_height * 1.25f;
|
|
{
|
|
TELY_AssetSpriteAnimation *anim = TELY_Asset_GetSpriteAnimation(&game->atlas_sprite_sheet, g_anim_names.heart);
|
|
FP_GameEntity *heart = FP_Game_GetEntity(game, game->play.heart);
|
|
Dqn_f32 max_width = platform->core.window_size.x * .5f;
|
|
Dqn_V2 draw_p = Dqn_V2_InitNx2(platform->core.window_size.x * .25f, player_avatar_rect.pos.y + bar_height);
|
|
Dqn_f32 health_t = heart->hp / DQN_CAST(Dqn_f32)heart->hp_cap;
|
|
Dqn_Rect health_rect = Dqn_Rect_InitNx4(draw_p.x, draw_p.y, max_width, bar_height);
|
|
Dqn_Rect curr_health_rect = Dqn_Rect_InitNx4(draw_p.x, draw_p.y, max_width * health_t, bar_height);
|
|
|
|
TELY_Render_RectColourV4(renderer, curr_health_rect, TELY_RenderShapeMode_Fill, TELY_COLOUR_RED_TOMATO_V4);
|
|
|
|
TELY_RenderCommandRect *cmd = TELY_Render_RectColourV4(
|
|
renderer,
|
|
health_rect,
|
|
TELY_RenderShapeMode_Line,
|
|
TELY_COLOUR_BLACK_V4);
|
|
cmd->thickness = 4.f;
|
|
|
|
// NOTE: Draw the heart icon
|
|
Dqn_Rect icon_tex_rect = game->atlas_sprite_sheet.rects.data[anim->index];
|
|
Dqn_Rect heart_icon_rect = {};
|
|
heart_icon_rect.size = icon_tex_rect.size * .4f;
|
|
heart_icon_rect.pos = Dqn_V2_InitNx2(draw_p.x - (heart_icon_rect.size.w * .7f), draw_p.y - (heart_icon_rect.size.y * .4f));
|
|
|
|
TELY_Render_TextureColourV4(renderer,
|
|
game->atlas_sprite_sheet.tex_handle,
|
|
icon_tex_rect,
|
|
heart_icon_rect,
|
|
Dqn_V2_Zero /*rotate origin*/,
|
|
0.f /*rotation*/,
|
|
TELY_COLOUR_WHITE_V4);
|
|
|
|
TELY_Render_PushFont(renderer, game->talkco_font_large);
|
|
DQN_DEFER { TELY_Render_PopFont(renderer); };
|
|
TELY_Render_TextF(renderer, Dqn_Rect_InterpolatedPoint(heart_icon_rect, Dqn_V2_InitNx2(1.f, 0.65f)), Dqn_V2_Zero, "Terry's Heart");
|
|
}
|
|
}
|
|
}
|
|
|
|
// NOTE: Add scanlines into the game for A E S T H E T I C S ===================================
|
|
if (game->play.state == FP_GameState_Play) {
|
|
Dqn_V2 screen_size = Dqn_V2_InitNx2(platform->core.window_size.w, platform->core.window_size.h);
|
|
Dqn_f32 scanline_gap = 4.0f;
|
|
Dqn_f32 scanline_thickness = 3.0f;
|
|
FP_GameRenderCameraFollowScanlines(renderer, screen_size, game->play.camera.world_pos, scanline_gap, scanline_thickness);
|
|
}
|
|
|
|
// NOTE: Render the other game state modes =====================================================
|
|
if (game->play.state == FP_GameState_IntroScreen || game->play.state == FP_GameState_WinGame || game->play.state == FP_GameState_LoseGame || game->play.state == FP_GameState_Pause) {
|
|
TELY_Render_PushTransform(renderer, Dqn_M2x3_Identity());
|
|
DQN_DEFER { TELY_Render_PopTransform(renderer); };
|
|
TELY_Render_RectColourV4(
|
|
renderer,
|
|
Dqn_Rect_InitNx4(0, 0, DQN_CAST(Dqn_f32)platform->core.window_size.x, DQN_CAST(Dqn_f32)platform->core.window_size.y),
|
|
TELY_RenderShapeMode_Fill,
|
|
TELY_Colour_V4Alpha(TELY_COLOUR_BLACK_V4, 0.8f));
|
|
|
|
Dqn_V2I inset = platform->core.window_size * .05f;
|
|
Dqn_f32 min_inset = DQN_CAST(Dqn_f32)DQN_MIN(inset.x, inset.y);
|
|
|
|
Dqn_V2 draw_p = Dqn_V2_InitNx2(min_inset, min_inset);
|
|
TELY_Render_PushColourV4(renderer, TELY_COLOUR_WHITE_V4);
|
|
|
|
if (game->play.state == FP_GameState_IntroScreen) {
|
|
TELY_Render_PushFont(renderer, game->inter_regular_font_large);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "Feely Pona"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_PopFont(renderer);
|
|
|
|
TELY_Render_PushFont(renderer, game->inter_regular_font);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "Press enter to start"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_PopFont(renderer);
|
|
TELY_Render_PopColourV4(renderer);
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_Return))
|
|
game->play.state = FP_GameState_Play;
|
|
} else if (game->play.state == FP_GameState_LoseGame) {
|
|
TELY_Render_PushFont(renderer, game->inter_regular_font_large);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "Terry's heart has been crushed"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "from the hoard of monsters"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_PopFont(renderer);
|
|
|
|
TELY_Render_PushFont(renderer, game->inter_regular_font);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "Sayounara amigo"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "Press enter to restart"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_PopFont(renderer);
|
|
TELY_Render_PopColourV4(renderer);
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_Return))
|
|
FP_PlayReset(game, platform);
|
|
} else if (game->play.state == FP_GameState_Pause) {
|
|
TELY_Render_PushFont(renderer, game->inter_regular_font_large);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "Paused"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_PopFont(renderer);
|
|
|
|
TELY_Render_PushFont(renderer, game->inter_regular_font);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "Press enter to resume"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_PopFont(renderer);
|
|
TELY_Render_PopColourV4(renderer);
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_Return))
|
|
game->play.state = FP_GameState_Play;
|
|
} else {
|
|
DQN_ASSERT(game->play.state == FP_GameState_WinGame);
|
|
TELY_Render_PushFont(renderer, game->inter_regular_font_large);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "Terry has been saved"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "from his terrible calamity"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_PopFont(renderer);
|
|
|
|
TELY_Render_PushFont(renderer, game->inter_regular_font);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "He lives for yet another day and another love"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "Press enter to restart"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_PopFont(renderer);
|
|
TELY_Render_PopColourV4(renderer);
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_Return))
|
|
FP_PlayReset(game, platform);
|
|
}
|
|
}
|
|
|
|
// NOTE: Debug UI ==============================================================================
|
|
if (game->play.debug_ui) {
|
|
TELY_Render_PushTransform(renderer, Dqn_M2x3_Identity());
|
|
DQN_DEFER { TELY_Render_PopTransform(renderer); };
|
|
// NOTE: Info bar ==========================================================================
|
|
{
|
|
TELY_RFuiResult info_bar = TELY_RFui_Row(rfui, DQN_STRING8("Info Bar"));
|
|
info_bar.widget->semantic_position[TELY_RFuiAxis_X].kind = TELY_RFuiPositionKind_Absolute;
|
|
info_bar.widget->semantic_position[TELY_RFuiAxis_X].value = 10.f;
|
|
info_bar.widget->semantic_position[TELY_RFuiAxis_Y].kind = TELY_RFuiPositionKind_Absolute;
|
|
info_bar.widget->semantic_position[TELY_RFuiAxis_Y].value = 10.f;
|
|
|
|
TELY_RFui_PushParent(rfui, info_bar.widget);
|
|
DQN_DEFER { TELY_RFui_PopParent(rfui); };
|
|
|
|
Dqn_ThreadScratch scratch = Dqn_Thread_GetScratch(nullptr);
|
|
TELY_RFui_TextF(rfui, "TELY");
|
|
if (Dqn_String8_IsValid(platform->core.os_name)) {
|
|
TELY_RFui_TextF(rfui, " | %.*s", DQN_STRING_FMT(platform->core.os_name));
|
|
}
|
|
|
|
TELY_RFui_TextF(rfui,
|
|
" | %dx%d %.1fHz | TSC %.1f GHz",
|
|
platform->core.window_size.w,
|
|
platform->core.window_size.h,
|
|
platform->core.display.refresh_rate,
|
|
platform->core.tsc_per_second / 1'000'000'000.0);
|
|
|
|
if (platform->core.ram_mb)
|
|
TELY_RFui_TextF(rfui, " | RAM %.1fGB", platform->core.ram_mb / 1024.0);
|
|
|
|
TELY_RFui_TextF(rfui,
|
|
" | Work %04.1fms/f (%04.1f%%) | %05.1f FPS | Frame %'I64u | Timer %.1fs",
|
|
input->work_ms,
|
|
input->work_ms * 100.0 / input->delta_ms,
|
|
1000.0 / input->delta_ms,
|
|
input->frame_counter,
|
|
input->timer_s);
|
|
|
|
}
|
|
|
|
// NOTE: Other
|
|
{
|
|
TELY_RFuiResult profiler_layout = TELY_RFui_Column(rfui, DQN_STRING8("Profiler Bar"));
|
|
profiler_layout.widget->semantic_position[TELY_RFuiAxis_X].kind = TELY_RFuiPositionKind_Absolute;
|
|
profiler_layout.widget->semantic_position[TELY_RFuiAxis_X].value = 10.f;
|
|
profiler_layout.widget->semantic_position[TELY_RFuiAxis_Y].kind = TELY_RFuiPositionKind_Absolute;
|
|
profiler_layout.widget->semantic_position[TELY_RFuiAxis_Y].value = TELY_Asset_GetFont(assets, TELY_RFui_ActiveFont(rfui))->pixel_height * 1.5f;
|
|
|
|
TELY_RFui_PushParent(rfui, profiler_layout.widget);
|
|
DQN_DEFER { TELY_RFui_PopParent(rfui); };
|
|
|
|
TELY_RFui_TextF(rfui, "Camera: %.1f, %.1f", game->play.camera.world_pos.x, game->play.camera.world_pos.y);
|
|
|
|
Dqn_ProfilerAnchor *anchors = Dqn_Profiler_AnchorBuffer(Dqn_ProfilerAnchorBuffer_Back);
|
|
for (size_t anchor_index = 1; anchor_index < DQN_PROFILER_ANCHOR_BUFFER_SIZE; anchor_index++) {
|
|
Dqn_ProfilerAnchor const *anchor = anchors + anchor_index;
|
|
if (!anchor->hit_count)
|
|
continue;
|
|
|
|
uint64_t tsc_exclusive = anchor->tsc_exclusive;
|
|
uint64_t tsc_inclusive = anchor->tsc_inclusive;
|
|
Dqn_f64 tsc_exclusive_milliseconds = tsc_exclusive * 1000 / DQN_CAST(Dqn_f64)platform->core.tsc_per_second;
|
|
if (tsc_exclusive == tsc_inclusive) {
|
|
TELY_RFui_TextF(rfui,
|
|
"%.*s[%u]: %.1fms",
|
|
DQN_STRING_FMT(anchor->name),
|
|
anchor->hit_count,
|
|
tsc_exclusive_milliseconds);
|
|
} else {
|
|
Dqn_f64 tsc_inclusive_milliseconds = tsc_inclusive * 1000 / DQN_CAST(Dqn_f64)platform->core.tsc_per_second;
|
|
TELY_RFui_TextF(rfui,
|
|
"%.*s[%u]: %.1f/%.1fms",
|
|
DQN_STRING_FMT(anchor->name),
|
|
anchor->hit_count,
|
|
tsc_exclusive_milliseconds,
|
|
tsc_inclusive_milliseconds);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (game->play.state == FP_GameState_Play) {
|
|
TELY_Render_PushTransform(renderer, Dqn_M2x3_Identity());
|
|
DQN_DEFER { TELY_Render_PopTransform(renderer); };
|
|
|
|
FP_GameEntity *player = FP_Game_GetEntity(game, game->play.player);
|
|
Dqn_V2 draw_p = Dqn_V2_InitNx2(32.f, platform->core.window_size.h * .5f);
|
|
TELY_Render_PushFont(renderer, game->inter_regular_font);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, "DEBUG MENU"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, " F1 Debug info"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, " F2 Add coins x10,000"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, " F3 Win game"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, " F4 Reset game"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, " F5 Increase health"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, " F6 Increase stamina"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, " F7 Increase mobile data"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, " F8 %s god mode", game->play.god_mode ? "Disable" : "Enable"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, " F9 %s noclip", player->flags & FP_GameEntityFlag_NoClip ? "Disable" : "Enable"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, " F10 Building inventory +1"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_TextF(renderer, draw_p, Dqn_V2_Zero, " F11 %s by enemies", player->faction == FP_GameEntityFaction_Nil ? "Attacked" : "Ignored"); draw_p.y += TELY_Render_FontHeight(renderer, assets);
|
|
TELY_Render_PopFont(renderer);
|
|
TELY_Render_PopColourV4(renderer);
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_F1))
|
|
game->play.debug_ui = !game->play.debug_ui;
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_F2)) {
|
|
if (!FP_Game_IsNilEntity(player))
|
|
player->coins += 10'000;
|
|
}
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_F3))
|
|
game->play.state = FP_GameState_WinGame;
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_F4))
|
|
FP_PlayReset(game, platform);
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_F5)) {
|
|
if (!FP_Game_IsNilEntity(player))
|
|
player->hp_cap += FP_DEFAULT_DAMAGE;
|
|
player->hp = player->hp_cap;
|
|
}
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_F6)) {
|
|
if (!FP_Game_IsNilEntity(player)) {
|
|
player->stamina_cap += DQN_CAST(uint16_t)(FP_TERRY_DASH_STAMINA_COST * .5f);
|
|
player->stamina = player->stamina_cap;
|
|
}
|
|
}
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_F7)) {
|
|
if (!FP_Game_IsNilEntity(player)) {
|
|
player->terry_mobile_data_plan_cap += DQN_KILOBYTES(1);
|
|
player->terry_mobile_data_plan = player->terry_mobile_data_plan_cap;
|
|
}
|
|
}
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_F8))
|
|
game->play.god_mode = !game->play.god_mode;
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_F9))
|
|
player->flags ^= FP_GameEntityFlag_NoClip;
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_F10)) {
|
|
player->inventory.clubs += 1;
|
|
player->inventory.airports += 1;
|
|
player->inventory.churchs += 1;
|
|
player->inventory.kennels += 1;
|
|
}
|
|
|
|
if (TELY_Platform_InputScanCodeIsPressed(input, TELY_PlatformInputScanCode_F11)) {
|
|
player->faction = player->faction == FP_GameEntityFaction_Nil
|
|
? FP_GameEntityFaction_Friendly
|
|
: FP_GameEntityFaction_Nil;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
extern "C" __declspec(dllexport)
|
|
void TELY_DLL_FrameUpdate(void *user_data)
|
|
{
|
|
TELY_Platform *platform = DQN_CAST(TELY_Platform *) user_data;
|
|
TELY_PlatformInput *input = &platform->input;
|
|
TELY_Assets *assets = &platform->assets;
|
|
TELY_Renderer *renderer = &platform->renderer;
|
|
FP_Game *game = DQN_CAST(FP_Game *) platform->user_data;
|
|
TELY_RFui *rfui = &game->rfui;
|
|
|
|
// =============================================================================================
|
|
|
|
game->play.prev_clicked_entity = game->play.clicked_entity;
|
|
game->play.prev_hot_entity = game->play.hot_entity;
|
|
game->play.prev_active_entity = game->play.active_entity;
|
|
game->play.hot_entity = {};
|
|
game->play.active_entity = {};
|
|
Dqn_FArray_Clear(&game->play.parent_entity_stack);
|
|
Dqn_FArray_Add(&game->play.parent_entity_stack, game->play.root_entity->handle);
|
|
|
|
// =============================================================================================
|
|
|
|
TELY_Audio *audio = &platform->audio;
|
|
|
|
#if 0
|
|
if (audio->playback_size == 0) {
|
|
TELY_Audio_Play(audio, game->audio[FP_GameAudio_TestAudio], 1.f /*volume*/);
|
|
}
|
|
#endif
|
|
|
|
// =============================================================================================
|
|
|
|
if (game->play.state == FP_GameState_Play) {
|
|
if (TELY_Platform_InputKeyWasDown(input->mouse_left) && TELY_Platform_InputKeyIsDown(input->mouse_left)) {
|
|
if (game->play.prev_active_entity.id)
|
|
game->play.active_entity = game->play.prev_active_entity;
|
|
} else {
|
|
Dqn_V2 world_mouse_p = input->mouse_p + game->play.camera.world_pos;
|
|
for (FP_GameEntityIterator it = {}; FP_Game_DFSPreOrderWalkEntityTree(game, &it, game->play.root_entity); ) {
|
|
FP_GameEntity *entity = it.entity;
|
|
if (entity->local_hit_box_size.x <= 0 || entity->local_hit_box_size.y <= 0)
|
|
continue;
|
|
|
|
if ((entity->flags & FP_GameEntityFlag_Clickable) == 0)
|
|
continue;
|
|
|
|
Dqn_Rect world_hit_box = FP_Game_CalcEntityWorldHitBox(game, entity->handle);
|
|
if (!Dqn_Rect_ContainsPoint(world_hit_box, world_mouse_p))
|
|
continue;
|
|
|
|
game->play.hot_entity = entity->handle;
|
|
if (TELY_Platform_InputKeyIsPressed(input->mouse_left)) {
|
|
game->play.active_entity = entity->handle;
|
|
game->play.clicked_entity = entity->handle;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (game->play.delta_s_accumulator += DQN_CAST(Dqn_f32)input->delta_s;
|
|
game->play.delta_s_accumulator > PHYSICS_STEP;
|
|
game->play.delta_s_accumulator -= PHYSICS_STEP) {
|
|
FP_Update(platform, game, input, audio);
|
|
}
|
|
|
|
FP_Render(game, platform, renderer, audio);
|
|
|
|
TELY_RFui_Flush(rfui, renderer, input, assets);
|
|
TELY_Audio_MixPlaybackSamples(audio, assets);
|
|
}
|