2016-11-09 11:36:41 +00:00
|
|
|
#include "Dengine/Asteroid.h"
|
|
|
|
#include "Dengine/Debug.h"
|
|
|
|
|
|
|
|
void initAssetManager(GameState *state)
|
|
|
|
{
|
|
|
|
AssetManager *assetManager = &state->assetManager;
|
|
|
|
MemoryArena_ *arena = &state->persistentArena;
|
|
|
|
|
|
|
|
i32 audioEntries = 32;
|
|
|
|
assetManager->audio.size = audioEntries;
|
|
|
|
assetManager->audio.entries =
|
|
|
|
memory_pushBytes(arena, audioEntries * sizeof(HashTableEntry));
|
|
|
|
|
|
|
|
i32 texAtlasEntries = 8;
|
|
|
|
assetManager->texAtlas.size = texAtlasEntries;
|
|
|
|
assetManager->texAtlas.entries =
|
|
|
|
memory_pushBytes(arena, texAtlasEntries * sizeof(HashTableEntry));
|
|
|
|
|
|
|
|
i32 texEntries = 32;
|
|
|
|
assetManager->textures.size = texEntries;
|
|
|
|
assetManager->textures.entries =
|
|
|
|
memory_pushBytes(arena, texEntries * sizeof(HashTableEntry));
|
|
|
|
|
|
|
|
i32 animEntries = 1024;
|
|
|
|
assetManager->anims.size = animEntries;
|
|
|
|
assetManager->anims.entries =
|
|
|
|
memory_pushBytes(arena, animEntries * sizeof(HashTableEntry));
|
|
|
|
|
|
|
|
/* Create empty 1x1 4bpp black texture */
|
|
|
|
u32 bitmap = (0xFF << 24) | (0xFF << 16) | (0xFF << 8) | (0xFF << 0);
|
|
|
|
Texture *tex = asset_getFreeTexSlot(assetManager, arena, "nullTex");
|
|
|
|
*tex = texture_gen(1, 1, 4, CAST(u8 *)(&bitmap));
|
|
|
|
|
|
|
|
/* Load shaders */
|
2016-11-11 14:17:30 +00:00
|
|
|
asset_loadShaderFiles(
|
|
|
|
assetManager, arena, "data/shaders/default_tex.vert.glsl",
|
|
|
|
"data/shaders/default_tex.frag.glsl", shaderlist_default);
|
|
|
|
|
|
|
|
asset_loadShaderFiles(
|
|
|
|
assetManager, arena, "data/shaders/default_no_tex.vert.glsl",
|
|
|
|
"data/shaders/default_no_tex.frag.glsl", shaderlist_default_no_tex);
|
2016-11-09 11:36:41 +00:00
|
|
|
|
|
|
|
i32 result =
|
|
|
|
asset_loadTTFont(assetManager, arena, "C:/Windows/Fonts/Arialbd.ttf");
|
2016-11-16 02:00:59 +00:00
|
|
|
if (result) ASSERT(TRUE);
|
2016-11-09 11:36:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void initRenderer(GameState *state, v2 windowSize) {
|
|
|
|
AssetManager *assetManager = &state->assetManager;
|
|
|
|
Renderer *renderer = &state->renderer;
|
|
|
|
renderer->size = windowSize;
|
|
|
|
|
|
|
|
// NOTE(doyle): Value to map a screen coordinate to NDC coordinate
|
|
|
|
renderer->vertexNdcFactor =
|
|
|
|
V2(1.0f / renderer->size.w, 1.0f / renderer->size.h);
|
|
|
|
|
|
|
|
const mat4 projection =
|
|
|
|
mat4_ortho(0.0f, renderer->size.w, 0.0f, renderer->size.h, 0.0f, 1.0f);
|
2016-11-11 14:17:30 +00:00
|
|
|
for (i32 i = 0; i < shaderlist_count; i++)
|
|
|
|
{
|
|
|
|
renderer->shaderList[i] = asset_getShader(assetManager, i);
|
|
|
|
shader_use(renderer->shaderList[i]);
|
|
|
|
shader_uniformSetMat4fv(renderer->shaderList[i], "projection",
|
|
|
|
projection);
|
|
|
|
GL_CHECK_ERROR();
|
|
|
|
}
|
|
|
|
|
|
|
|
renderer->activeShaderId = renderer->shaderList[shaderlist_default];
|
2016-11-09 11:36:41 +00:00
|
|
|
GL_CHECK_ERROR();
|
|
|
|
|
|
|
|
/* Create buffers */
|
2016-11-11 10:43:03 +00:00
|
|
|
glGenVertexArrays(ARRAY_COUNT(renderer->vao), renderer->vao);
|
|
|
|
glGenBuffers(ARRAY_COUNT(renderer->vbo), renderer->vbo);
|
2016-11-09 11:36:41 +00:00
|
|
|
GL_CHECK_ERROR();
|
|
|
|
|
2016-11-16 02:00:59 +00:00
|
|
|
// Bind buffers and configure vao, vao automatically intercepts
|
|
|
|
// glBindCalls and associates the state with that buffer for us
|
|
|
|
for (enum RenderMode mode = 0; mode < rendermode_count; mode++)
|
2016-11-11 10:43:03 +00:00
|
|
|
{
|
2016-11-16 02:00:59 +00:00
|
|
|
glBindVertexArray(renderer->vao[mode]);
|
|
|
|
glBindBuffer(GL_ARRAY_BUFFER, renderer->vbo[mode]);
|
2016-11-11 10:43:03 +00:00
|
|
|
|
|
|
|
glEnableVertexAttribArray(0);
|
|
|
|
u32 numVertexElements = 4;
|
|
|
|
u32 stride = sizeof(Vertex);
|
|
|
|
|
|
|
|
glVertexAttribPointer(0, numVertexElements, GL_FLOAT,
|
|
|
|
GL_FALSE, stride, (GLvoid *)0);
|
|
|
|
glBindBuffer(GL_ARRAY_BUFFER, 0);
|
|
|
|
glBindVertexArray(0);
|
|
|
|
}
|
2016-11-09 11:36:41 +00:00
|
|
|
|
|
|
|
/* Unbind */
|
|
|
|
GL_CHECK_ERROR();
|
|
|
|
|
|
|
|
// TODO(doyle): Lazy allocate render group capacity
|
|
|
|
renderer->groupCapacity = 4096;
|
|
|
|
for (i32 i = 0; i < ARRAY_COUNT(renderer->groups); i++)
|
|
|
|
{
|
|
|
|
renderer->groups[i].vertexList = memory_pushBytes(
|
|
|
|
&state->persistentArena, renderer->groupCapacity * sizeof(Vertex));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum ReadKeyType
|
|
|
|
{
|
|
|
|
readkeytype_oneShot,
|
|
|
|
readkeytype_delayedRepeat,
|
|
|
|
readkeytype_repeat,
|
|
|
|
readkeytype_count,
|
|
|
|
};
|
|
|
|
|
|
|
|
#define KEY_DELAY_NONE 0.0f
|
|
|
|
INTERNAL b32 getKeyStatus(KeyState *key, enum ReadKeyType readType,
|
|
|
|
f32 delayInterval, f32 dt)
|
|
|
|
{
|
|
|
|
|
|
|
|
if (!key->endedDown) return FALSE;
|
|
|
|
|
|
|
|
switch(readType)
|
|
|
|
{
|
|
|
|
case readkeytype_oneShot:
|
|
|
|
{
|
|
|
|
if (key->newHalfTransitionCount > key->oldHalfTransitionCount)
|
|
|
|
return TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case readkeytype_repeat:
|
|
|
|
case readkeytype_delayedRepeat:
|
|
|
|
{
|
|
|
|
if (key->newHalfTransitionCount > key->oldHalfTransitionCount)
|
|
|
|
{
|
|
|
|
if (readType == readkeytype_delayedRepeat)
|
|
|
|
{
|
|
|
|
// TODO(doyle): Let user set arbitrary delay after initial input
|
|
|
|
key->delayInterval = 2 * delayInterval;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
key->delayInterval = delayInterval;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else if (key->delayInterval <= 0.0f)
|
|
|
|
{
|
|
|
|
key->delayInterval = delayInterval;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
key->delayInterval -= dt;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
#ifdef DENGINE_DEBUG
|
|
|
|
DEBUG_LOG("getKeyStatus() error: Invalid ReadKeyType enum");
|
|
|
|
ASSERT(INVALID_CODE_PATH);
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2016-11-16 13:50:49 +00:00
|
|
|
typedef struct Basis
|
|
|
|
{
|
|
|
|
v2 basis;
|
|
|
|
v2 pivotPoint;
|
|
|
|
} Basis;
|
|
|
|
|
|
|
|
enum RectBaseline
|
|
|
|
{
|
|
|
|
rectbaseline_top,
|
|
|
|
rectbaseline_topLeft,
|
|
|
|
rectbaseline_topRight,
|
|
|
|
rectbaseline_bottom,
|
|
|
|
rectbaseline_bottomRight,
|
|
|
|
rectbaseline_bottomLeft,
|
|
|
|
rectbaseline_left,
|
|
|
|
rectbaseline_right,
|
|
|
|
rectbaseline_center,
|
|
|
|
rectbaseline_count,
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
Basis getBasis(Entity *entity, enum RectBaseline baseline)
|
|
|
|
{
|
|
|
|
ASSERT(baseline < rectbaseline_count);
|
|
|
|
|
|
|
|
v2 basis = v2_sub(entity->pos, entity->offset);
|
|
|
|
v2 pivotPoint = v2_scale(entity->size, 0.5f);
|
|
|
|
v2 size = entity->size;
|
|
|
|
switch (baseline)
|
|
|
|
{
|
|
|
|
case rectbaseline_top:
|
|
|
|
basis.y += (size.h);
|
|
|
|
basis.x += (size.w * 0.5f);
|
|
|
|
break;
|
|
|
|
case rectbaseline_topLeft:
|
|
|
|
basis.y += (size.h);
|
|
|
|
break;
|
|
|
|
case rectbaseline_topRight:
|
|
|
|
basis.y += (size.h);
|
|
|
|
basis.x += (size.w);
|
|
|
|
break;
|
|
|
|
case rectbaseline_bottom:
|
|
|
|
basis.x += (size.w * 0.5f);
|
|
|
|
break;
|
|
|
|
case rectbaseline_bottomRight:
|
|
|
|
basis.x += (size.w);
|
|
|
|
break;
|
|
|
|
case rectbaseline_left:
|
|
|
|
basis.y += (size.h * 0.5f);
|
|
|
|
break;
|
|
|
|
case rectbaseline_right:
|
|
|
|
basis.x += (size.w);
|
|
|
|
basis.y += (size.h * 0.5f);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case rectbaseline_bottomLeft:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
DEBUG_LOG(
|
|
|
|
"getPosRelativeToRect() warning: baseline enum not recognised");
|
|
|
|
break;
|
|
|
|
}
|
2016-11-09 11:36:41 +00:00
|
|
|
|
2016-11-16 13:50:49 +00:00
|
|
|
Basis result = {0};
|
|
|
|
result.basis = basis;
|
|
|
|
result.pivotPoint = pivotPoint;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Basis getDefaultBasis(Entity *entity)
|
|
|
|
{
|
|
|
|
Basis result = getBasis(entity, rectbaseline_bottomLeft);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <time.h>
|
2016-11-17 07:15:36 +00:00
|
|
|
v2 *createAsteroidVertexList(MemoryArena_ *arena, i32 iterations,
|
|
|
|
i32 asteroidRadius)
|
2016-11-16 13:50:49 +00:00
|
|
|
{
|
|
|
|
f32 iterationAngle = 360.0f / iterations;
|
|
|
|
iterationAngle = DEGREES_TO_RADIANS(iterationAngle);
|
|
|
|
v2 *result =
|
|
|
|
memory_pushBytes(arena, iterations * sizeof(v2));
|
|
|
|
|
|
|
|
for (i32 i = 0; i < iterations; i++)
|
|
|
|
{
|
2016-11-17 07:15:36 +00:00
|
|
|
i32 randValue = rand();
|
2016-11-16 13:50:49 +00:00
|
|
|
result[i] = V2(math_cosf(iterationAngle * i) * asteroidRadius,
|
|
|
|
math_sinf(iterationAngle * i) * asteroidRadius);
|
|
|
|
|
|
|
|
#if 1
|
2016-11-17 07:15:36 +00:00
|
|
|
f32 displacementDist = 0.50f * asteroidRadius;
|
2016-11-16 13:50:49 +00:00
|
|
|
i32 vertexDisplacement =
|
2016-11-17 07:15:36 +00:00
|
|
|
randValue % (i32)displacementDist + (i32)(displacementDist * 0.25f);
|
2016-11-16 13:50:49 +00:00
|
|
|
|
|
|
|
i32 quadrantSize = iterations / 4;
|
|
|
|
|
|
|
|
i32 firstQuadrant = quadrantSize;
|
|
|
|
i32 secondQuadrant = quadrantSize * 2;
|
|
|
|
i32 thirdQuadrant = quadrantSize * 3;
|
|
|
|
i32 fourthQuadrant = quadrantSize * 4;
|
|
|
|
|
|
|
|
if (i < firstQuadrant)
|
|
|
|
{
|
|
|
|
result[i].x += vertexDisplacement;
|
|
|
|
result[i].y += vertexDisplacement;
|
|
|
|
}
|
|
|
|
else if (i < secondQuadrant)
|
|
|
|
{
|
|
|
|
result[i].x -= vertexDisplacement;
|
|
|
|
result[i].y += vertexDisplacement;
|
|
|
|
}
|
|
|
|
else if (i < thirdQuadrant)
|
|
|
|
{
|
|
|
|
result[i].x -= vertexDisplacement;
|
|
|
|
result[i].y -= vertexDisplacement;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
result[i].x += vertexDisplacement;
|
|
|
|
result[i].y -= vertexDisplacement;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-11-09 11:36:41 +00:00
|
|
|
void asteroid_gameUpdateAndRender(GameState *state, Memory *memory,
|
|
|
|
v2 windowSize, f32 dt)
|
|
|
|
{
|
2016-11-16 13:50:49 +00:00
|
|
|
i32 iterations = 16;
|
2016-11-17 07:15:36 +00:00
|
|
|
|
|
|
|
memory_arenaInit(&state->transientArena, memory->transient,
|
|
|
|
memory->transientSize);
|
|
|
|
|
2016-11-16 02:00:59 +00:00
|
|
|
if (!state->init)
|
|
|
|
{
|
2016-11-17 07:15:36 +00:00
|
|
|
srand(time(NULL));
|
2016-11-09 11:36:41 +00:00
|
|
|
memory_arenaInit(&state->persistentArena, memory->persistent,
|
|
|
|
memory->persistentSize);
|
|
|
|
initAssetManager(state);
|
|
|
|
initRenderer(state, windowSize);
|
|
|
|
|
2016-11-11 10:43:03 +00:00
|
|
|
state->pixelsPerMeter = 70.0f;
|
|
|
|
|
2016-11-09 11:36:41 +00:00
|
|
|
{ // Init ship entity
|
2016-11-17 07:15:36 +00:00
|
|
|
Entity *ship = &state->entityList[state->entityIndex];
|
|
|
|
ship->id = state->entityIndex++;
|
2016-11-16 05:57:24 +00:00
|
|
|
ship->pos = V2(0, 0);
|
2016-11-16 02:00:59 +00:00
|
|
|
ship->size = V2(25.0f, 50.0f);
|
|
|
|
ship->hitbox = ship->size;
|
|
|
|
ship->offset = v2_scale(ship->size, 0.5f);
|
|
|
|
ship->scale = 1;
|
|
|
|
ship->type = entitytype_ship;
|
|
|
|
ship->direction = direction_null;
|
|
|
|
ship->renderMode = rendermode_triangle;
|
|
|
|
ship->tex = NULL;
|
2016-11-17 07:15:36 +00:00
|
|
|
ship->collides = TRUE;
|
|
|
|
|
|
|
|
i32 numAsteroids = 10;
|
|
|
|
for (i32 i = 0; i < numAsteroids; i++)
|
|
|
|
{
|
|
|
|
Entity *asteroid = &state->entityList[state->entityIndex];
|
|
|
|
asteroid->id = state->entityIndex++;
|
|
|
|
|
|
|
|
i32 randValue = rand();
|
|
|
|
i32 randX = (randValue % (i32)windowSize.w);
|
|
|
|
i32 randY = (randValue % (i32)windowSize.h);
|
|
|
|
asteroid->pos = V2i(randX, randY);
|
|
|
|
|
|
|
|
asteroid->size = V2(100.0f, 100.0f);
|
|
|
|
asteroid->hitbox = asteroid->size;
|
|
|
|
asteroid->offset = v2_scale(asteroid->size, 0.5f);
|
|
|
|
asteroid->scale = 1;
|
|
|
|
asteroid->type = entitytype_asteroid;
|
|
|
|
asteroid->direction = direction_null;
|
|
|
|
asteroid->renderMode = rendermode_polygon;
|
|
|
|
|
|
|
|
asteroid->numVertexPoints = 16;
|
|
|
|
asteroid->vertexPoints = createAsteroidVertexList(
|
|
|
|
&state->persistentArena, asteroid->numVertexPoints,
|
|
|
|
(i32)(asteroid->size.x * 0.5f));
|
|
|
|
|
|
|
|
asteroid->tex = NULL;
|
|
|
|
asteroid->collides = TRUE;
|
|
|
|
}
|
2016-11-09 11:36:41 +00:00
|
|
|
}
|
|
|
|
|
2016-11-16 05:57:24 +00:00
|
|
|
state->camera.min = V2(0, 0);
|
|
|
|
state->camera.max = state->renderer.size;
|
|
|
|
state->init = TRUE;
|
2016-11-09 11:36:41 +00:00
|
|
|
|
2016-11-16 02:00:59 +00:00
|
|
|
state->worldSize = windowSize;
|
2016-11-11 10:43:03 +00:00
|
|
|
|
2016-11-16 02:00:59 +00:00
|
|
|
debug_init(&state->persistentArena, windowSize,
|
|
|
|
state->assetManager.font);
|
2016-11-16 13:50:49 +00:00
|
|
|
}
|
|
|
|
|
2016-11-09 11:36:41 +00:00
|
|
|
{
|
|
|
|
KeyState *keys = state->input.keys;
|
|
|
|
for (enum KeyCode code = 0; code < keycode_count; code++)
|
|
|
|
{
|
|
|
|
KeyState *keyState = &keys[code];
|
|
|
|
|
|
|
|
u32 halfTransitionCount = keyState->newHalfTransitionCount -
|
|
|
|
keyState->oldHalfTransitionCount;
|
|
|
|
|
|
|
|
if (halfTransitionCount > 0)
|
|
|
|
{
|
|
|
|
b32 transitionCountIsOdd = ((halfTransitionCount & 1) == 1);
|
|
|
|
|
|
|
|
if (transitionCountIsOdd)
|
|
|
|
{
|
|
|
|
if (keyState->endedDown) keyState->endedDown = FALSE;
|
|
|
|
else keyState->endedDown = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
keyState->oldHalfTransitionCount =
|
|
|
|
keyState->newHalfTransitionCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i32 i = 0; i < state->entityIndex; i++)
|
|
|
|
{
|
|
|
|
Entity *entity = &state->entityList[i];
|
|
|
|
ASSERT(entity->type != entitytype_invalid);
|
|
|
|
|
2016-11-16 02:00:59 +00:00
|
|
|
v2 pivotPoint = {0};
|
2016-11-09 11:36:41 +00:00
|
|
|
|
2016-11-16 13:50:49 +00:00
|
|
|
// Loop entity around world
|
|
|
|
if (entity->pos.y >= state->worldSize.h)
|
|
|
|
entity->pos.y = 0;
|
|
|
|
else if (entity->pos.y < 0)
|
|
|
|
entity->pos.y = state->worldSize.h;
|
|
|
|
|
|
|
|
if (entity->pos.x >= state->worldSize.w)
|
|
|
|
entity->pos.x = 0;
|
|
|
|
else if (entity->pos.x < 0)
|
|
|
|
entity->pos.x = state->worldSize.w;
|
|
|
|
|
|
|
|
if (entity->type == entitytype_ship)
|
|
|
|
{
|
2016-11-16 02:00:59 +00:00
|
|
|
v2 ddP = {0};
|
2016-11-11 10:43:03 +00:00
|
|
|
if (getKeyStatus(&state->input.keys[keycode_up], readkeytype_repeat,
|
|
|
|
0.0f, dt))
|
|
|
|
{
|
2016-11-16 02:00:59 +00:00
|
|
|
// TODO(doyle): Renderer creates upfacing triangles by default,
|
|
|
|
// but we need to offset rotation so that our base "0 degrees"
|
|
|
|
// is right facing for trig to work
|
2016-11-16 13:50:49 +00:00
|
|
|
Radians rotation =
|
|
|
|
DEGREES_TO_RADIANS((entity->rotation + 90.0f));
|
|
|
|
|
|
|
|
v2 direction = V2(math_cosf(rotation), math_sinf(rotation));
|
|
|
|
ddP = direction;
|
2016-11-09 11:36:41 +00:00
|
|
|
}
|
|
|
|
|
2016-11-17 07:15:36 +00:00
|
|
|
Degrees rotationsPerSecond = 180.0f;
|
2016-11-11 10:43:03 +00:00
|
|
|
if (getKeyStatus(&state->input.keys[keycode_left],
|
|
|
|
readkeytype_repeat, 0.0f, dt))
|
|
|
|
{
|
2016-11-17 07:15:36 +00:00
|
|
|
entity->rotation += (rotationsPerSecond) * dt;
|
2016-11-11 10:43:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (getKeyStatus(&state->input.keys[keycode_right],
|
|
|
|
readkeytype_repeat, 0.0f, dt))
|
|
|
|
{
|
2016-11-17 07:15:36 +00:00
|
|
|
entity->rotation -= (rotationsPerSecond) * dt;
|
2016-11-11 10:43:03 +00:00
|
|
|
}
|
|
|
|
|
2016-11-17 07:15:36 +00:00
|
|
|
if (ddP.x > 0.0f && ddP.y > 0.0f)
|
2016-11-11 10:43:03 +00:00
|
|
|
{
|
|
|
|
// NOTE(doyle): Cheese it and pre-compute the vector for
|
|
|
|
// diagonal using pythagoras theorem on a unit triangle 1^2
|
|
|
|
// + 1^2 = c^2
|
2016-11-16 02:00:59 +00:00
|
|
|
ddP = v2_scale(ddP, 0.70710678118f);
|
2016-11-11 10:43:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Assuming acceleration A over t time, then integrate twice to get
|
|
|
|
|
|
|
|
newVelocity = a*t + oldVelocity
|
|
|
|
newPos = (a*t^2)/2 + oldVelocity*t + oldPos
|
2016-11-16 02:00:59 +00:00
|
|
|
|
2016-11-11 10:43:03 +00:00
|
|
|
*/
|
|
|
|
|
2016-11-16 02:00:59 +00:00
|
|
|
ddP = v2_scale(ddP, state->pixelsPerMeter * 25);
|
|
|
|
|
2016-11-17 07:15:36 +00:00
|
|
|
v2 oldDp = entity->dP;
|
2016-11-16 02:00:59 +00:00
|
|
|
v2 resistance = v2_scale(oldDp, 2.0f);
|
2016-11-17 07:15:36 +00:00
|
|
|
ddP = v2_sub(ddP, resistance);
|
2016-11-11 10:43:03 +00:00
|
|
|
|
2016-11-16 02:00:59 +00:00
|
|
|
entity->dP = v2_add(v2_scale(ddP, dt), oldDp);
|
2016-11-11 10:43:03 +00:00
|
|
|
|
2016-11-16 02:00:59 +00:00
|
|
|
v2 ddPHalf = v2_scale(ddP, 0.5f);
|
|
|
|
v2 ddPHalfDtSquared = v2_scale(ddPHalf, (SQUARED(dt)));
|
|
|
|
v2 oldDpDt = v2_scale(oldDp, dt);
|
|
|
|
v2 oldPos = entity->pos;
|
|
|
|
entity->pos = v2_add(v2_add(ddPHalfDtSquared, oldDpDt), oldPos);
|
|
|
|
|
|
|
|
pivotPoint = v2_scale(entity->size, 0.5f);
|
2016-11-11 10:43:03 +00:00
|
|
|
|
2016-11-16 13:50:49 +00:00
|
|
|
DEBUG_PUSH_VAR("Pos: %5.2f, %5.2f", entity->pos, "v2");
|
|
|
|
DEBUG_PUSH_VAR("Velocity: %5.2f, %5.2f", entity->dP, "v2");
|
|
|
|
DEBUG_PUSH_VAR("Rotation: %5.2f", entity->rotation, "f32");
|
2016-11-17 07:15:36 +00:00
|
|
|
}
|
|
|
|
else if (entity->type == entitytype_asteroid)
|
|
|
|
{
|
2016-11-09 11:36:41 +00:00
|
|
|
|
2016-11-17 07:15:36 +00:00
|
|
|
i32 randValue = rand();
|
|
|
|
if (entity->direction == direction_null)
|
|
|
|
{
|
|
|
|
entity->direction = randValue % direction_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
v2 ddP = {0};
|
|
|
|
switch (entity->direction)
|
|
|
|
{
|
|
|
|
case direction_north:
|
|
|
|
{
|
|
|
|
ddP.y = 1.0f;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case direction_northwest:
|
|
|
|
{
|
|
|
|
ddP.x = 1.0f;
|
|
|
|
ddP.y = 1.0f;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case direction_west:
|
|
|
|
{
|
|
|
|
ddP.x = -1.0f;
|
|
|
|
}
|
|
|
|
break;
|
2016-11-16 02:00:59 +00:00
|
|
|
|
2016-11-17 07:15:36 +00:00
|
|
|
case direction_southwest:
|
|
|
|
{
|
|
|
|
ddP.x = -1.0f;
|
|
|
|
ddP.y = -1.0f;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case direction_south:
|
|
|
|
{
|
|
|
|
ddP.y = -1.0f;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case direction_southeast:
|
|
|
|
{
|
|
|
|
ddP.x = 1.0f;
|
|
|
|
ddP.y = -1.0f;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case direction_east:
|
|
|
|
{
|
|
|
|
ddP.x = 1.0f;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case direction_northeast:
|
|
|
|
{
|
|
|
|
ddP.x = 1.0f;
|
|
|
|
ddP.y = 1.0f;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
ASSERT(INVALID_CODE_PATH);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
f32 dirOffset = (randValue % 10) / 100.0f;
|
|
|
|
v2_scale(ddP, dirOffset);
|
|
|
|
ASSERT(ddP.x <= 1.0f && ddP.y <= 1.0f);
|
|
|
|
|
|
|
|
entity->dP = v2_scale(ddP, state->pixelsPerMeter * 2);
|
|
|
|
v2 ddPHalf = v2_scale(ddP, 0.5f);
|
|
|
|
v2 ddPHalfDtSquared = v2_scale(ddPHalf, (SQUARED(dt)));
|
|
|
|
|
|
|
|
v2 dPDt = v2_scale(entity->dP, dt);
|
|
|
|
|
|
|
|
entity->pos = v2_add(v2_add(ddPHalfDtSquared, dPDt), entity->pos);
|
2016-11-16 13:50:49 +00:00
|
|
|
}
|
2016-11-09 11:36:41 +00:00
|
|
|
|
2016-11-17 07:15:36 +00:00
|
|
|
RenderFlags flags = renderflag_wireframe | renderflag_no_texture;
|
|
|
|
renderer_entity(&state->renderer, &state->transientArena, state->camera,
|
|
|
|
entity, pivotPoint, 0, V4(0.4f, 0.8f, 1.0f, 1.0f),
|
|
|
|
flags);
|
|
|
|
|
|
|
|
Basis entityBasis = getDefaultBasis(entity);
|
|
|
|
renderer_rect(&state->renderer, state->camera, entityBasis.basis,
|
|
|
|
V2(4, 4), entityBasis.pivotPoint,
|
|
|
|
DEGREES_TO_RADIANS(entity->rotation), NULL,
|
|
|
|
V4(1.0f, 0, 0, 1.0f), flags);
|
|
|
|
}
|
2016-11-16 13:50:49 +00:00
|
|
|
|
2016-11-11 10:43:03 +00:00
|
|
|
TrianglePoints triangle = {0};
|
|
|
|
triangle.points[0] = V2(100, 200);
|
|
|
|
triangle.points[1] = V2(200, 100);
|
2016-11-16 02:00:59 +00:00
|
|
|
triangle.points[2] = V2(100, 300);
|
2016-11-11 13:22:09 +00:00
|
|
|
|
|
|
|
LOCAL_PERSIST Radians rotation = 0.0f;
|
|
|
|
rotation += DEGREES_TO_RADIANS(((60.0f) * dt));
|
|
|
|
|
2016-11-16 02:00:59 +00:00
|
|
|
RenderFlags flags = renderflag_wireframe | renderflag_no_texture;
|
2016-11-11 13:22:09 +00:00
|
|
|
renderer_triangle(&state->renderer, state->camera, triangle, V2(0, 0),
|
|
|
|
rotation, NULL, V4(1, 1, 1, 1), flags);
|
2016-11-11 10:43:03 +00:00
|
|
|
|
2016-11-16 02:00:59 +00:00
|
|
|
debug_drawUi(state, dt);
|
|
|
|
debug_clearCounter();
|
|
|
|
|
2016-11-09 11:36:41 +00:00
|
|
|
renderer_renderGroups(&state->renderer);
|
|
|
|
}
|