2019-07-19 20:56:21 +00:00
|
|
|
#include "foldhaus_platform.h"
|
|
|
|
#include "foldhaus_app.h"
|
|
|
|
|
2019-12-26 21:14:00 +00:00
|
|
|
internal void
|
|
|
|
SetPanelDefinitionExternal(panel* Panel, s32 OldPanelDefinitionIndex, s32 NewPanelDefinitionIndex)
|
|
|
|
{
|
|
|
|
if(OldPanelDefinitionIndex >= 0)
|
|
|
|
{
|
|
|
|
GlobalPanelDefs[OldPanelDefinitionIndex].Cleanup(Panel);
|
|
|
|
}
|
|
|
|
GlobalPanelDefs[NewPanelDefinitionIndex].Init(Panel);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
internal void
|
|
|
|
DrawPanelFooter(panel* Panel, render_command_buffer* RenderBuffer, v2 FooterMin, v2 FooterMax, interface_config Interface, mouse_state Mouse)
|
|
|
|
{
|
|
|
|
PushRenderQuad2D(RenderBuffer, FooterMin, v2{FooterMax.x, FooterMin.y + 25}, v4{.5f, .5f, .5f, 1.f});
|
|
|
|
PushRenderQuad2D(RenderBuffer, FooterMin, FooterMin + v2{25, 25}, WhiteV4);
|
|
|
|
|
|
|
|
v2 PanelSelectButtonMin = FooterMin + v2{30, 1};
|
|
|
|
v2 PanelSelectButtonMax = PanelSelectButtonMin + v2{100, 23};
|
|
|
|
|
|
|
|
if (Panel->PanelSelectionMenuOpen)
|
|
|
|
{
|
|
|
|
v2 ButtonDimension = v2{100, 25};
|
|
|
|
v2 ButtonMin = v2{PanelSelectButtonMin.x, FooterMax.y};
|
|
|
|
|
|
|
|
v2 MenuMin = ButtonMin;
|
|
|
|
v2 MenuMax = v2{ButtonMin.x + ButtonDimension.x, ButtonMin.y + (ButtonDimension.y * GlobalPanelDefsCount)};
|
|
|
|
if (MouseButtonTransitionedDown(Mouse.LeftButtonState)
|
|
|
|
&& !PointIsInRange(Mouse.DownPos, MenuMin, MenuMax))
|
|
|
|
{
|
|
|
|
Panel->PanelSelectionMenuOpen = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
for (s32 i = 0; i < GlobalPanelDefsCount; i++)
|
|
|
|
{
|
|
|
|
panel_definition Def = GlobalPanelDefs[i];
|
|
|
|
string DefName = MakeString(Def.PanelName, Def.PanelNameLength);
|
|
|
|
button_result DefinitionButton = EvaluateButton(RenderBuffer,
|
|
|
|
ButtonMin, ButtonMin + ButtonDimension,
|
|
|
|
DefName, Interface, Mouse);
|
|
|
|
if (DefinitionButton.Pressed)
|
|
|
|
{
|
|
|
|
SetPanelDefinition(Panel, i);
|
|
|
|
Panel->PanelSelectionMenuOpen = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ButtonMin.y += ButtonDimension.y;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
button_result ButtonResult = EvaluateButton(RenderBuffer,
|
|
|
|
PanelSelectButtonMin,
|
|
|
|
PanelSelectButtonMax,
|
|
|
|
MakeStringLiteral("Select"), Interface, Mouse);
|
|
|
|
if (ButtonResult.Pressed)
|
|
|
|
{
|
|
|
|
Panel->PanelSelectionMenuOpen = !Panel->PanelSelectionMenuOpen;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
internal void
|
|
|
|
RenderPanel(panel* Panel, v2 PanelMin, v2 PanelMax, v2 WindowMin, v2 WindowMax, render_command_buffer* RenderBuffer, app_state* State, context Context, mouse_state Mouse)
|
|
|
|
{
|
|
|
|
Assert(Panel->PanelDefinitionIndex >= 0);
|
|
|
|
|
|
|
|
v2 FooterMin = PanelMin;
|
|
|
|
v2 FooterMax = v2{PanelMax.x, PanelMin.y + 25};
|
|
|
|
v2 PanelViewMin = v2{PanelMin.x, FooterMax.y};
|
|
|
|
v2 PanelViewMax = PanelMax;
|
|
|
|
|
|
|
|
panel_definition Definition = GlobalPanelDefs[Panel->PanelDefinitionIndex];
|
2019-12-26 22:45:27 +00:00
|
|
|
|
|
|
|
|
|
|
|
Definition.Render(*Panel, PanelMin, PanelMax, RenderBuffer, State, Context, Mouse);
|
2019-12-26 21:14:00 +00:00
|
|
|
|
|
|
|
PushRenderOrthographic(RenderBuffer, WindowMin.x, WindowMin.y, WindowMax.x, WindowMax.y);
|
|
|
|
DrawPanelFooter(Panel, RenderBuffer, FooterMin, FooterMax, State->Interface, Mouse);
|
|
|
|
}
|
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
internal v4
|
|
|
|
MouseToWorldRay(r32 MouseX, r32 MouseY, camera* Camera, r32 WindowWidth, r32 WindowHeight)
|
|
|
|
{
|
|
|
|
DEBUG_TRACK_SCOPE(MouseToWorldRay);
|
|
|
|
r32 X = ((2.0f * MouseX) / WindowWidth) - 1;
|
|
|
|
r32 Y = ((2.0f * MouseY) / WindowHeight) - 1;
|
|
|
|
|
|
|
|
v4 ScreenPos = v4{X, Y, -1, 1};
|
|
|
|
|
|
|
|
m44 InverseProjection = {};
|
|
|
|
Inverse(GetCameraPerspectiveProjectionMatrix(*Camera), &InverseProjection);
|
|
|
|
|
|
|
|
m44 InverseModelView = {};
|
|
|
|
Inverse(GetCameraModelViewMatrix(*Camera), &InverseModelView);
|
|
|
|
InverseModelView = Transpose(InverseModelView);
|
|
|
|
|
|
|
|
v4 ClipSpacePos = InverseProjection * ScreenPos;
|
|
|
|
v4 WorldPosition = InverseModelView * ClipSpacePos;
|
|
|
|
return WorldPosition;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct send_sacn_job_data
|
|
|
|
{
|
2019-11-23 00:07:25 +00:00
|
|
|
|
|
|
|
platform_socket_handle SendSocket;
|
|
|
|
platform_send_to* SendTo;
|
|
|
|
dmx_buffer_list* DMXBuffers;
|
2019-07-19 20:56:21 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
internal void
|
2019-11-23 00:07:25 +00:00
|
|
|
SACNSendDMXBufferListJob (s32 ThreadID, void* JobData)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
|
|
|
DEBUG_TRACK_FUNCTION;
|
|
|
|
|
|
|
|
send_sacn_job_data* Data = (send_sacn_job_data*)JobData;
|
2019-11-23 00:07:25 +00:00
|
|
|
platform_socket_handle SendSocket = Data->SendSocket;
|
|
|
|
platform_send_to* SendTo = Data->SendTo;
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-11-23 00:07:25 +00:00
|
|
|
dmx_buffer_list* DMXBufferAt = Data->DMXBuffers;
|
|
|
|
while (DMXBufferAt)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-11-23 00:07:25 +00:00
|
|
|
dmx_buffer Buffer = DMXBufferAt->Buffer;
|
|
|
|
|
|
|
|
u_long V4SendAddress = SACNGetUniverseSendAddress(Buffer.Universe);
|
2019-12-04 06:40:22 +00:00
|
|
|
|
|
|
|
platform_network_address SendAddress = {};
|
|
|
|
SendAddress.Family = AF_INET;
|
|
|
|
SendAddress.Port = DEFAULT_STREAMING_ACN_PORT;
|
|
|
|
SendAddress.Address = V4SendAddress;
|
2019-11-23 00:07:25 +00:00
|
|
|
|
|
|
|
SendTo(SendSocket, SendAddress, (const char*)Buffer.Base, Buffer.TotalSize, 0);
|
|
|
|
|
|
|
|
DMXBufferAt = DMXBufferAt->Next;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
RELOAD_STATIC_DATA(ReloadStaticData)
|
|
|
|
{
|
|
|
|
app_state* State = (app_state*)Context.MemoryBase;
|
|
|
|
|
|
|
|
GlobalDebugServices = DebugServices;
|
2019-11-12 04:34:56 +00:00
|
|
|
GSAlloc = Alloc;
|
|
|
|
GSFree = Free;
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-11-02 17:29:51 +00:00
|
|
|
if (State->DefaultInputCommandRegistry.Size > 0)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-11-02 17:29:51 +00:00
|
|
|
RegisterKeyPressCommand(&State->DefaultInputCommandRegistry, KeyCode_U, Command_Began, KeyCode_Invalid, OpenUniverseView);
|
2019-12-26 16:11:48 +00:00
|
|
|
RegisterKeyPressCommand(&State->DefaultInputCommandRegistry, KeyCode_X, Command_Ended, KeyCode_Invalid, DeleteAnimationBlock);
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
INITIALIZE_APPLICATION(InitializeApplication)
|
|
|
|
{
|
|
|
|
app_state* State = (app_state*)Context.MemoryBase;
|
2019-12-23 01:47:26 +00:00
|
|
|
State->Permanent = {};
|
|
|
|
State->Permanent.Alloc = (gs_memory_alloc*)Context.PlatformAlloc;
|
|
|
|
State->Permanent.Realloc = (gs_memory_realloc*)Context.PlatformRealloc;
|
|
|
|
State->Transient = {};
|
2019-12-26 20:42:55 +00:00
|
|
|
State->Transient.Alloc = (gs_memory_alloc*)Context.PlatformAlloc;
|
2019-12-23 01:47:26 +00:00
|
|
|
State->Transient.Realloc = (gs_memory_realloc*)Context.PlatformRealloc;
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-12-23 01:47:26 +00:00
|
|
|
InitializeInputCommandRegistry(&State->DefaultInputCommandRegistry, 32, &State->Permanent);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-10-30 14:28:02 +00:00
|
|
|
s32 CommandQueueSize = 32;
|
2019-12-23 01:47:26 +00:00
|
|
|
command_queue_entry* CommandQueueMemory = PushArray(&State->Permanent,
|
2019-10-30 14:28:02 +00:00
|
|
|
command_queue_entry,
|
|
|
|
CommandQueueSize);
|
|
|
|
State->CommandQueue = InitializeCommandQueue(CommandQueueMemory, CommandQueueSize);
|
|
|
|
|
2019-12-23 01:47:26 +00:00
|
|
|
State->ActiveTextEntry.Buffer = MakeString(PushArray(&State->Permanent, char, 256), 0, 256);
|
2019-09-02 06:03:38 +00:00
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
// TODO(Peter): put in InitializeInterface?
|
|
|
|
r32 FontSize = 14;
|
|
|
|
{
|
|
|
|
platform_memory_result FontFile = Context.PlatformReadEntireFile("Anonymous Pro.ttf");
|
|
|
|
if (FontFile.Size)
|
|
|
|
{
|
2019-12-23 01:47:26 +00:00
|
|
|
bitmap_font* Font = PushStruct(&State->Permanent, bitmap_font);
|
2019-07-28 21:31:05 +00:00
|
|
|
|
|
|
|
Font->BitmapWidth = 512;
|
|
|
|
Font->BitmapHeight = 512;
|
|
|
|
Font->BitmapBytesPerPixel = 4;
|
2019-12-23 01:47:26 +00:00
|
|
|
Font->BitmapMemory = PushArray(&State->Permanent, u8, Font->BitmapWidth * Font->BitmapHeight * Font->BitmapBytesPerPixel);
|
2019-07-28 21:31:05 +00:00
|
|
|
Font->BitmapStride = Font->BitmapWidth * Font->BitmapBytesPerPixel;
|
|
|
|
GSMemSet(Font->BitmapMemory, 0, Font->BitmapStride * Font->BitmapHeight);
|
|
|
|
|
2019-10-30 16:10:15 +00:00
|
|
|
platform_font_info FontInfo = Context.PlatformGetFontInfo("Anonymous Pro", FontSize, FontWeight_Normal, false, false, false);
|
2019-07-28 21:31:05 +00:00
|
|
|
Font->PixelHeight = FontInfo.PixelHeight;
|
|
|
|
Font->Ascent = FontInfo.Ascent;
|
|
|
|
Font->Descent = FontInfo.Descent;
|
|
|
|
Font->Leading = FontInfo.Leading;
|
|
|
|
Font->MaxCharWidth = FontInfo.MaxCharWidth;
|
|
|
|
|
|
|
|
Font->CodepointDictionarySize = (FontInfo.CodepointOnePastLast - FontInfo.CodepointStart);
|
|
|
|
Font->CodepointDictionaryCount = 0;
|
2019-12-23 01:47:26 +00:00
|
|
|
Font->CodepointKeys = PushArray(&State->Permanent, char, Font->CodepointDictionarySize);
|
|
|
|
Font->CodepointValues = PushArray(&State->Permanent, codepoint_bitmap, Font->CodepointDictionarySize);
|
2019-07-28 21:31:05 +00:00
|
|
|
|
2019-07-29 00:24:10 +00:00
|
|
|
for (s32 Codepoint = FontInfo.CodepointStart;
|
|
|
|
Codepoint < FontInfo.CodepointOnePastLast;
|
|
|
|
Codepoint++)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-07-22 06:30:53 +00:00
|
|
|
|
2019-07-28 21:31:05 +00:00
|
|
|
u32 CodepointX, CodepointY;
|
|
|
|
GetNextCodepointOffset(Font, &CodepointX, &CodepointY);
|
2019-07-22 06:30:53 +00:00
|
|
|
|
2019-07-28 21:31:05 +00:00
|
|
|
u32 CodepointW, CodepointH;
|
|
|
|
Context.PlatformDrawFontCodepoint(
|
2019-12-26 20:42:55 +00:00
|
|
|
Font->BitmapMemory,
|
|
|
|
Font->BitmapWidth,
|
|
|
|
Font->BitmapHeight,
|
|
|
|
CodepointX, CodepointY,
|
|
|
|
Codepoint, FontInfo,
|
|
|
|
&CodepointW, &CodepointH);
|
2019-07-22 06:30:53 +00:00
|
|
|
|
2019-07-28 21:31:05 +00:00
|
|
|
AddCodepointToFont(Font, Codepoint, 0, 0, CodepointW, CodepointH, CodepointX, CodepointY);
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
2019-07-28 21:31:05 +00:00
|
|
|
|
|
|
|
State->Interface.Font = Font;
|
|
|
|
State->Font = Font;
|
|
|
|
|
|
|
|
Font->BitmapTextureHandle = Context.PlatformGetGPUTextureHandle(Font->BitmapMemory,
|
|
|
|
Font->BitmapWidth, Font->BitmapHeight);
|
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
} else {}
|
|
|
|
}
|
|
|
|
|
|
|
|
State->Interface.FontSize = FontSize;
|
|
|
|
State->Interface.PanelBGColors[0] = v4{.3f, .3f, .3f, 1};
|
|
|
|
State->Interface.PanelBGColors[1] = v4{.4f, .4f, .4f, 1};
|
|
|
|
State->Interface.PanelBGColors[2] = v4{.5f, .5f, .5f, 1};
|
|
|
|
State->Interface.PanelBGColors[3] = v4{.6f, .6f, .6f, 1};
|
|
|
|
State->Interface.ButtonColor_Inactive = BlackV4;
|
|
|
|
State->Interface.ButtonColor_Active = v4{.1f, .1f, .1f, 1};
|
|
|
|
State->Interface.ButtonColor_Selected = v4{.1f, .1f, .3f, 1};
|
|
|
|
State->Interface.TextColor = WhiteV4;
|
|
|
|
State->Interface.Margin = v2{5, 5};
|
|
|
|
|
2019-11-23 00:07:25 +00:00
|
|
|
State->SACN = InitializeSACN(Context);
|
|
|
|
State->NetworkProtocolHeaderSize = STREAM_HEADER_SIZE;
|
2019-07-19 20:56:21 +00:00
|
|
|
|
|
|
|
State->Camera.FieldOfView = DegreesToRadians(45.0f);
|
|
|
|
State->Camera.AspectRatio = (r32)Context.WindowWidth / (r32)Context.WindowHeight;
|
|
|
|
State->Camera.Near = 1.0f;
|
|
|
|
State->Camera.Far = 100.0f;
|
|
|
|
State->Camera.Position = v3{0, 0, -250};
|
|
|
|
State->Camera.LookAt = v3{0, 0, 0};
|
|
|
|
|
2019-11-29 05:12:57 +00:00
|
|
|
State->AssemblyList.BucketSize = 32;
|
|
|
|
State->AssemblyList.FreeList.Next = &State->AssemblyList.FreeList;
|
|
|
|
State->ActiveAssemblyIndecies.BucketSize = 32;
|
2019-07-19 20:56:21 +00:00
|
|
|
#if 1
|
2019-12-23 01:47:26 +00:00
|
|
|
char Path[] = "radialumia.fold";
|
2019-07-19 20:56:21 +00:00
|
|
|
LoadAssembly(State, Context, Path);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
State->PixelsToWorldScale = .01f;
|
|
|
|
|
|
|
|
GlobalDebugServices->Interface.RenderSculpture = true;
|
|
|
|
|
2019-11-12 04:34:56 +00:00
|
|
|
ReloadStaticData(Context, GlobalDebugServices, Alloc, Free);
|
2019-11-01 12:46:40 +00:00
|
|
|
|
2019-12-26 16:11:48 +00:00
|
|
|
// Setup Operation Modes
|
2019-12-26 20:42:55 +00:00
|
|
|
State->Modes.ActiveModesCount = 0;
|
|
|
|
State->Modes.Arena = {};
|
|
|
|
State->Modes.Arena.Alloc = (gs_memory_alloc*)Context.PlatformAlloc;
|
|
|
|
State->Modes.Arena.Realloc = (gs_memory_realloc*)Context.PlatformRealloc;
|
|
|
|
State->Modes.Arena.FindAddressRule = FindAddress_InLastBufferOnly;
|
2019-12-26 16:11:48 +00:00
|
|
|
|
|
|
|
{ // MODES PLAYGROUND
|
|
|
|
InitializeAnimationSystem(&State->AnimationSystem);
|
2019-12-27 02:40:14 +00:00
|
|
|
State->AnimationSystem.SecondsPerFrame = 1.f / 24.f;
|
2019-12-26 20:42:55 +00:00
|
|
|
|
|
|
|
animation_block BlockOne = {0};
|
2019-12-27 02:40:14 +00:00
|
|
|
BlockOne.StartTime = 0;
|
|
|
|
BlockOne.EndTime = 8;
|
2019-12-26 16:11:48 +00:00
|
|
|
BlockOne.Proc = TestPatternTwo;
|
2019-12-26 20:42:55 +00:00
|
|
|
AddAnimationBlock(BlockOne, &State->AnimationSystem);
|
|
|
|
|
|
|
|
animation_block BlockTwo = {0};
|
2019-12-27 02:40:14 +00:00
|
|
|
BlockTwo.StartTime = 8;
|
|
|
|
BlockTwo.EndTime = 15;
|
2019-12-26 20:42:55 +00:00
|
|
|
BlockTwo.Proc = TestPatternThree;
|
2019-12-26 16:11:48 +00:00
|
|
|
AddAnimationBlock(BlockTwo, &State->AnimationSystem);
|
|
|
|
|
2019-12-27 02:40:14 +00:00
|
|
|
State->AnimationSystem.AnimationStart = 0;
|
|
|
|
State->AnimationSystem.AnimationEnd = 15;
|
2019-12-26 16:11:48 +00:00
|
|
|
} // End Animation Playground
|
2019-12-26 20:42:55 +00:00
|
|
|
|
|
|
|
|
|
|
|
{ // Panels Playground
|
|
|
|
InitializePanelLayout(&State->PanelLayout);
|
|
|
|
panel* Panel = TakeNewPanel(&State->PanelLayout);
|
2019-12-26 21:14:00 +00:00
|
|
|
SetPanelDefinition(Panel, 0);
|
2019-12-26 20:42:55 +00:00
|
|
|
} // End Panels Playground
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2019-11-11 20:02:24 +00:00
|
|
|
internal void
|
|
|
|
HandleInput (app_state* State, input_queue InputQueue, mouse_state Mouse)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-11-11 20:02:24 +00:00
|
|
|
DEBUG_TRACK_FUNCTION;
|
|
|
|
|
|
|
|
input_command_registry ActiveCommands = State->DefaultInputCommandRegistry;
|
|
|
|
if (State->Modes.ActiveModesCount > 0)
|
|
|
|
{
|
|
|
|
ActiveCommands = State->Modes.ActiveModes[State->Modes.ActiveModesCount - 1].Commands;
|
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-11-11 20:02:24 +00:00
|
|
|
for (s32 EventIdx = 0; EventIdx < InputQueue.QueueUsed; EventIdx++)
|
2019-08-18 12:56:18 +00:00
|
|
|
{
|
2019-11-11 20:02:24 +00:00
|
|
|
input_entry Event = InputQueue.Entries[EventIdx];
|
|
|
|
|
|
|
|
// NOTE(Peter): These are in the order Down, Up, Held because we want to privalege
|
|
|
|
// Down and Up over Held. In other words, we don't want to call a Held command on the
|
|
|
|
// frame when the button was released, even if the command is registered to both events
|
|
|
|
if (KeyTransitionedDown(Event))
|
2019-11-01 12:46:40 +00:00
|
|
|
{
|
2019-11-11 20:02:24 +00:00
|
|
|
FindAndPushExistingCommand(ActiveCommands, Event, Command_Began, &State->CommandQueue);
|
2019-11-01 12:46:40 +00:00
|
|
|
}
|
2019-11-11 20:02:24 +00:00
|
|
|
else if (KeyTransitionedUp(Event))
|
2019-10-30 14:28:02 +00:00
|
|
|
{
|
2019-11-11 20:02:24 +00:00
|
|
|
FindAndPushExistingCommand(ActiveCommands, Event, Command_Ended, &State->CommandQueue);
|
2019-10-30 14:28:02 +00:00
|
|
|
}
|
2019-11-11 20:02:24 +00:00
|
|
|
else if (KeyHeldDown(Event))
|
2019-10-30 14:28:02 +00:00
|
|
|
{
|
2019-11-11 20:02:24 +00:00
|
|
|
FindAndPushExistingCommand(ActiveCommands, Event, Command_Held, &State->CommandQueue);
|
2019-10-30 14:28:02 +00:00
|
|
|
}
|
2019-08-18 12:56:18 +00:00
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-11-11 20:02:24 +00:00
|
|
|
// Execute all commands in CommandQueue
|
|
|
|
for (s32 CommandIdx = State->CommandQueue.Used - 1; CommandIdx >= 0; CommandIdx--)
|
|
|
|
{
|
|
|
|
command_queue_entry* Entry = &State->CommandQueue.Commands[CommandIdx];
|
|
|
|
Entry->Command.Proc(State, Entry->Event, Mouse);
|
|
|
|
}
|
|
|
|
|
|
|
|
ClearCommandQueue(&State->CommandQueue);
|
|
|
|
}
|
|
|
|
|
2019-11-23 00:07:25 +00:00
|
|
|
internal dmx_buffer_list*
|
|
|
|
CreateDMXBuffers(assembly Assembly, s32 BufferHeaderSize, memory_arena* Arena)
|
|
|
|
{
|
2019-11-23 09:08:59 +00:00
|
|
|
DEBUG_TRACK_FUNCTION;
|
|
|
|
|
2019-11-23 00:07:25 +00:00
|
|
|
dmx_buffer_list* Result = 0;
|
|
|
|
dmx_buffer_list* Head = 0;
|
|
|
|
|
|
|
|
s32 BufferSize = BufferHeaderSize + 512;
|
|
|
|
|
2019-11-23 07:33:48 +00:00
|
|
|
for (s32 Range = 0; Range < Assembly.LEDUniverseMapCount; Range++)
|
2019-11-23 00:07:25 +00:00
|
|
|
{
|
2019-11-23 07:33:48 +00:00
|
|
|
leds_in_universe_range LEDUniverseRange = Assembly.LEDUniverseMap[Range];
|
|
|
|
|
|
|
|
dmx_buffer_list* NewBuffer = PushStruct(Arena, dmx_buffer_list);
|
2019-12-27 02:40:14 +00:00
|
|
|
NewBuffer->Buffer.Universe = LEDUniverseRange.Universe;
|
2019-11-23 07:33:48 +00:00
|
|
|
NewBuffer->Buffer.Base = PushArray(Arena, u8, BufferSize);
|
|
|
|
NewBuffer->Buffer.TotalSize = BufferSize;
|
|
|
|
NewBuffer->Buffer.HeaderSize = BufferHeaderSize;
|
2019-12-27 02:40:14 +00:00
|
|
|
NewBuffer->Next = 0;
|
2019-11-23 00:07:25 +00:00
|
|
|
|
2019-11-23 07:33:48 +00:00
|
|
|
// Append
|
|
|
|
if (!Result) {
|
|
|
|
Result = NewBuffer;
|
|
|
|
Head = Result;
|
|
|
|
}
|
|
|
|
Head->Next = NewBuffer;
|
|
|
|
Head = NewBuffer;
|
2019-11-23 00:07:25 +00:00
|
|
|
|
2019-12-04 06:40:22 +00:00
|
|
|
u8* DestChannel = Head->Buffer.Base + BufferHeaderSize;
|
2019-11-23 07:33:48 +00:00
|
|
|
for (s32 LEDIdx = LEDUniverseRange.RangeStart;
|
|
|
|
LEDIdx < LEDUniverseRange.RangeOnePastLast;
|
|
|
|
LEDIdx++)
|
2019-11-23 00:07:25 +00:00
|
|
|
{
|
2019-11-23 07:33:48 +00:00
|
|
|
led LED = Assembly.LEDs[LEDIdx];
|
|
|
|
pixel Color = Assembly.Colors[LED.Index];
|
|
|
|
|
2019-12-04 06:40:22 +00:00
|
|
|
|
|
|
|
DestChannel[0] = Color.R;
|
|
|
|
DestChannel[1] = Color.G;
|
|
|
|
DestChannel[2] = Color.B;
|
|
|
|
DestChannel += 3;
|
2019-11-23 00:07:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2019-11-11 20:02:24 +00:00
|
|
|
UPDATE_AND_RENDER(UpdateAndRender)
|
|
|
|
{
|
|
|
|
DEBUG_TRACK_FUNCTION;
|
|
|
|
app_state* State = (app_state*)Context.MemoryBase;
|
|
|
|
|
|
|
|
// NOTE(Peter): We do this at the beginning because all the render commands are stored in Transient,
|
|
|
|
// and need to persist beyond the end of the UpdateAndRender call. In the release version, we won't
|
|
|
|
// zero the Transient arena when we clear it so it wouldn't be a problem, but it is technically
|
|
|
|
// incorrect to clear the arena, and then access the memory later.
|
2019-12-23 01:47:26 +00:00
|
|
|
ClearArena(&State->Transient);
|
2019-11-11 20:02:24 +00:00
|
|
|
|
|
|
|
HandleInput(State, InputQueue, Mouse);
|
|
|
|
|
2019-12-26 16:11:48 +00:00
|
|
|
if (State->AnimationSystem.TimelineShouldAdvance) {
|
2019-12-26 20:42:55 +00:00
|
|
|
State->AnimationSystem.Time += Context.DeltaTime;
|
|
|
|
if (State->AnimationSystem.Time > State->AnimationSystem.AnimationEnd)
|
|
|
|
{
|
|
|
|
State->AnimationSystem.Time -= State->AnimationSystem.AnimationEnd;
|
|
|
|
}
|
2019-12-27 00:23:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
s32 CurrentFrame = (s32)(State->AnimationSystem.Time / State->AnimationSystem.SecondsPerFrame);
|
|
|
|
if (CurrentFrame != State->AnimationSystem.LastUpdatedFrame)
|
|
|
|
{
|
|
|
|
State->AnimationSystem.LastUpdatedFrame = CurrentFrame;
|
|
|
|
r32 FrameTime = CurrentFrame * State->AnimationSystem.SecondsPerFrame;
|
|
|
|
|
2019-12-26 16:11:48 +00:00
|
|
|
for (u32 i = 0; i < State->AnimationSystem.BlocksCount; i++)
|
2019-12-04 06:40:22 +00:00
|
|
|
{
|
2019-12-26 16:11:48 +00:00
|
|
|
animation_block_entry BlockEntry = State->AnimationSystem.Blocks[i];
|
|
|
|
if (!AnimationBlockIsFree(BlockEntry))
|
|
|
|
{
|
|
|
|
animation_block Block = BlockEntry.Block;
|
2019-12-26 20:42:55 +00:00
|
|
|
if (State->AnimationSystem.Time >= Block.StartTime
|
|
|
|
&& State->AnimationSystem.Time <= Block.EndTime)
|
|
|
|
{
|
2019-12-27 02:40:14 +00:00
|
|
|
for (s32 j = 0; j < State->ActiveAssemblyIndecies.Used; j++)
|
|
|
|
{
|
|
|
|
array_entry_handle* AssemblyHandle = GetElementAtIndex(j, State->ActiveAssemblyIndecies);
|
|
|
|
assembly* Assembly = GetElementWithHandle(*AssemblyHandle, State->AssemblyList);
|
|
|
|
Block.Proc(Assembly, FrameTime - Block.StartTime);
|
|
|
|
}
|
2019-12-26 16:11:48 +00:00
|
|
|
}
|
|
|
|
}
|
2019-12-27 00:23:43 +00:00
|
|
|
}
|
2019-12-26 20:42:55 +00:00
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-11-23 09:08:59 +00:00
|
|
|
s32 HeaderSize = State->NetworkProtocolHeaderSize;
|
|
|
|
dmx_buffer_list* DMXBuffers = 0;
|
2019-12-27 02:40:14 +00:00
|
|
|
if (State->ActiveAssemblyIndecies.Used > 1)
|
|
|
|
{
|
|
|
|
s32 f = 4;
|
|
|
|
}
|
|
|
|
|
2019-11-29 05:12:57 +00:00
|
|
|
for (s32 i = 0; i < State->ActiveAssemblyIndecies.Used; i++)
|
2019-11-23 09:08:59 +00:00
|
|
|
{
|
2019-12-27 02:40:14 +00:00
|
|
|
array_entry_handle* AssemblyHandle = GetElementAtIndex(i, State->ActiveAssemblyIndecies);
|
|
|
|
assembly* Assembly = GetElementWithHandle(*AssemblyHandle, State->AssemblyList);
|
|
|
|
dmx_buffer_list* NewDMXBuffers = CreateDMXBuffers(*Assembly, HeaderSize, &State->Transient);
|
|
|
|
DMXBuffers = DMXBufferListAppend(DMXBuffers, NewDMXBuffers);
|
2019-11-23 09:08:59 +00:00
|
|
|
}
|
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
DEBUG_IF(GlobalDebugServices->Interface.SendSACNData)
|
|
|
|
{
|
2019-11-23 00:07:25 +00:00
|
|
|
switch (State->NetworkProtocol)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-11-23 00:07:25 +00:00
|
|
|
case NetworkProtocol_SACN:
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-11-23 00:07:25 +00:00
|
|
|
SACNUpdateSequence(&State->SACN);
|
|
|
|
|
|
|
|
dmx_buffer_list* CurrentDMXBuffer = DMXBuffers;
|
|
|
|
while (CurrentDMXBuffer)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-11-23 00:07:25 +00:00
|
|
|
dmx_buffer Buffer = CurrentDMXBuffer->Buffer;
|
|
|
|
SACNPrepareBufferHeader(Buffer.Universe, Buffer.Base, Buffer.TotalSize, Buffer.HeaderSize, State->SACN);
|
|
|
|
CurrentDMXBuffer = CurrentDMXBuffer->Next;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
2019-11-23 00:07:25 +00:00
|
|
|
|
2019-12-23 01:47:26 +00:00
|
|
|
send_sacn_job_data* Job = PushStruct(&State->Transient, send_sacn_job_data);
|
2019-11-23 00:07:25 +00:00
|
|
|
Job->SendSocket = State->SACN.SendSocket;
|
|
|
|
Job->SendTo = Context.PlatformSendTo;
|
|
|
|
Job->DMXBuffers = DMXBuffers;
|
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
Context.GeneralWorkQueue->PushWorkOnQueue(
|
2019-12-26 20:42:55 +00:00
|
|
|
Context.GeneralWorkQueue,
|
|
|
|
SACNSendDMXBufferListJob,
|
|
|
|
Job);
|
2019-11-23 00:07:25 +00:00
|
|
|
}break;
|
|
|
|
|
|
|
|
InvalidDefaultCase;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-26 20:42:55 +00:00
|
|
|
PushRenderOrthographic(RenderBuffer, 0, 0, Context.WindowWidth, Context.WindowHeight);
|
|
|
|
PushRenderClearScreen(RenderBuffer);
|
|
|
|
v2 WindowMin = v2{0, 0};
|
|
|
|
v2 WindowMax = v2{Context.WindowWidth, Context.WindowHeight};
|
|
|
|
HandleMousePanelInteraction(&State->PanelLayout, WindowMin, WindowMax, Mouse);
|
|
|
|
DrawAllPanels(State->PanelLayout, WindowMin, WindowMax, RenderBuffer, State->Interface, Mouse, State, Context);
|
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
////////////////////////////////
|
|
|
|
// Render Assembly
|
|
|
|
///////////////////////////////
|
|
|
|
if (Context.WindowIsVisible)
|
|
|
|
{
|
2019-12-26 20:42:55 +00:00
|
|
|
#if 0
|
2019-07-19 20:56:21 +00:00
|
|
|
///////////////////////////////////////
|
|
|
|
// Interface
|
|
|
|
//////////////////////////////////////
|
|
|
|
|
|
|
|
DEBUG_TRACK_SCOPE(DrawInterface);
|
|
|
|
|
2019-07-22 06:30:53 +00:00
|
|
|
PushRenderOrthographic(RenderBuffer, 0, 0, Context.WindowWidth, Context.WindowHeight);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
|
|
|
///////////////////////////////////////
|
|
|
|
// Menu Bar
|
|
|
|
//////////////////////////////////////
|
|
|
|
r32 TopBarHeight = 40;
|
|
|
|
{
|
|
|
|
panel_result TopBarPanel = EvaluatePanel(RenderBuffer,
|
|
|
|
v2{0, Context.WindowHeight - TopBarHeight},
|
|
|
|
v2{Context.WindowWidth, Context.WindowHeight},
|
2019-10-30 14:28:02 +00:00
|
|
|
0, State->Interface);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-07-22 06:30:53 +00:00
|
|
|
v2 ButtonDim = v2{200, (r32)NewLineYOffset(*State->Interface.Font) + 10};
|
2019-07-19 20:56:21 +00:00
|
|
|
v2 ButtonPos = v2{State->Interface.Margin.x, Context.WindowHeight - (ButtonDim.y + 10)};
|
|
|
|
button_result LoadAssemblyBtn = EvaluateButton(RenderBuffer, ButtonPos, ButtonPos + ButtonDim,
|
|
|
|
MakeStringLiteral("Load Assembly"),
|
2019-11-01 14:38:44 +00:00
|
|
|
State->Interface, Mouse);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-12-23 01:47:26 +00:00
|
|
|
string InterfaceString = MakeString(PushArray(&State->Transient, char, 256), 256);
|
2019-11-29 05:12:57 +00:00
|
|
|
for (s32 i = 0; i < State->ActiveAssemblyIndecies.Used; i++)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-11-29 05:12:57 +00:00
|
|
|
array_entry_handle AssemblyHandle = *GetElementAtIndex(i, State->ActiveAssemblyIndecies);
|
|
|
|
assembly Assembly = *GetElementWithHandle(AssemblyHandle, State->AssemblyList);
|
|
|
|
PrintF(&InterfaceString, "Unload %.*s", Assembly.Name.Length, Assembly.Name.Memory);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
|
|
|
ButtonPos.x += ButtonDim.x + 10;
|
|
|
|
button_result UnloadAssemblyBtn = EvaluateButton(RenderBuffer, ButtonPos, ButtonPos + ButtonDim,
|
2019-11-01 14:38:44 +00:00
|
|
|
InterfaceString, State->Interface, Mouse);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
|
|
|
if (UnloadAssemblyBtn.Pressed)
|
|
|
|
{
|
|
|
|
UnloadAssembly(i, State, Context);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (LoadAssemblyBtn.Pressed)
|
|
|
|
{
|
|
|
|
char FilePath[256];
|
2019-08-18 12:56:18 +00:00
|
|
|
b32 Success = Context.PlatformGetFilePath(FilePath, 256, "Foldhaus Files\0*.fold\0\0");
|
2019-07-19 20:56:21 +00:00
|
|
|
if (Success)
|
|
|
|
{
|
|
|
|
LoadAssembly(State, Context, FilePath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-26 21:14:00 +00:00
|
|
|
DrawDebugInterface(RenderBuffer, 25,
|
|
|
|
State->Interface, Context.WindowWidth, Context.WindowHeight - TopBarHeight,
|
|
|
|
Context.DeltaTime, State, State->Camera, Mouse, &State->Transient);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-11-01 12:46:40 +00:00
|
|
|
for (s32 m = 0; m < State->Modes.ActiveModesCount; m++)
|
|
|
|
{
|
|
|
|
operation_mode OperationMode = State->Modes.ActiveModes[m];
|
2019-11-01 13:46:06 +00:00
|
|
|
if (OperationMode.Render != 0)
|
|
|
|
{
|
2019-11-01 14:38:44 +00:00
|
|
|
OperationMode.Render(State, RenderBuffer, OperationMode, Mouse);
|
2019-11-01 13:46:06 +00:00
|
|
|
}
|
2019-11-01 12:46:40 +00:00
|
|
|
}
|
|
|
|
|
2019-12-23 01:47:26 +00:00
|
|
|
|
|
|
|
// Checking for overflows
|
|
|
|
{
|
|
|
|
DEBUG_TRACK_SCOPE(OverflowChecks);
|
2019-12-26 20:42:55 +00:00
|
|
|
AssertAllocationsNoOverflow(State->Permanent);
|
2019-12-27 02:40:14 +00:00
|
|
|
for (s32 i = 0; i < State->ActiveAssemblyIndecies.Used; i++)
|
2019-12-26 20:42:55 +00:00
|
|
|
{
|
2019-12-27 02:40:14 +00:00
|
|
|
array_entry_handle* AssemblyHandle = GetElementAtIndex(i, State->ActiveAssemblyIndecies);
|
|
|
|
assembly* Assembly = GetElementWithHandle(*AssemblyHandle, State->AssemblyList);
|
2019-12-26 20:42:55 +00:00
|
|
|
AssertAllocationsNoOverflow(Assembly->Arena);
|
2019-12-23 01:47:26 +00:00
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CLEANUP_APPLICATION(CleanupApplication)
|
|
|
|
{
|
|
|
|
app_state* State = (app_state*)Context.MemoryBase;
|
|
|
|
SACNCleanup(&State->SACN, Context);
|
|
|
|
}
|