2020-01-02 02:41:43 +00:00
|
|
|
//
|
|
|
|
// File: foldhaus_app.cpp
|
|
|
|
// Author: Peter Slattery
|
|
|
|
// Creation Date: 2020-01-01
|
|
|
|
//
|
|
|
|
#ifndef FOLDHAUS_APP_CPP
|
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
#include "foldhaus_platform.h"
|
|
|
|
#include "foldhaus_app.h"
|
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
RELOAD_STATIC_DATA(ReloadStaticData)
|
|
|
|
{
|
|
|
|
app_state* State = (app_state*)Context.MemoryBase;
|
|
|
|
|
|
|
|
GlobalDebugServices = DebugServices;
|
|
|
|
}
|
|
|
|
|
|
|
|
INITIALIZE_APPLICATION(InitializeApplication)
|
|
|
|
{
|
|
|
|
app_state* State = (app_state*)Context.MemoryBase;
|
2020-06-15 22:36:50 +00:00
|
|
|
*State = {};
|
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
State->Permanent = CreateMemoryArena(Context.ThreadContext.Allocator);
|
2020-09-07 20:42:00 +00:00
|
|
|
State->Transient = Context.ThreadContext.Transient;
|
2020-06-15 22:36:50 +00:00
|
|
|
|
|
|
|
State->Assemblies.CountMax = 8;
|
|
|
|
State->Assemblies.Values = PushArray(&State->Permanent, assembly, State->Assemblies.CountMax);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2020-09-07 20:42:00 +00:00
|
|
|
State->GlobalLog = PushStruct(State->Transient, event_log);
|
2020-02-06 04:24:34 +00:00
|
|
|
*State->GlobalLog = {0};
|
|
|
|
|
2019-10-30 14:28:02 +00:00
|
|
|
s32 CommandQueueSize = 32;
|
2020-03-22 05:44:44 +00:00
|
|
|
command_queue_entry* CommandQueueMemory = PushArray(&State->Permanent,
|
|
|
|
command_queue_entry,
|
2019-10-30 14:28:02 +00:00
|
|
|
CommandQueueSize);
|
|
|
|
State->CommandQueue = InitializeCommandQueue(CommandQueueMemory, CommandQueueSize);
|
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
// TODO(Peter): put in InitializeInterface?
|
|
|
|
r32 FontSize = 14;
|
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
gs_file FontFile = ReadEntireFile(Context.ThreadContext.FileHandler, ConstString("data/Anonymous Pro.ttf"));
|
|
|
|
if (FileNoError(FontFile))
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
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;
|
2020-07-18 19:00:14 +00:00
|
|
|
ZeroMemoryBlock(Font->BitmapMemory, Font->BitmapStride * Font->BitmapHeight);
|
2019-07-28 21:31:05 +00:00
|
|
|
|
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,
|
2020-03-22 05:44:44 +00:00
|
|
|
Font->BitmapWidth,
|
2019-12-26 20:42:55 +00:00
|
|
|
Font->BitmapHeight,
|
2020-03-22 05:44:44 +00:00
|
|
|
CodepointX, CodepointY,
|
2019-12-26 20:42:55 +00:00
|
|
|
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
|
|
|
|
2020-06-23 00:39:58 +00:00
|
|
|
State->Interface.Style.Font = Font;
|
2019-07-28 21:31:05 +00:00
|
|
|
|
2020-03-22 05:44:44 +00:00
|
|
|
Font->BitmapTextureHandle = Context.PlatformGetGPUTextureHandle(Font->BitmapMemory,
|
2019-07-28 21:31:05 +00:00
|
|
|
Font->BitmapWidth, Font->BitmapHeight);
|
2020-03-22 05:44:44 +00:00
|
|
|
}
|
|
|
|
else
|
2020-02-05 08:03:56 +00:00
|
|
|
{
|
2020-02-06 04:24:34 +00:00
|
|
|
LogError(State->GlobalLog, "Unable to load font");
|
2020-02-05 08:03:56 +00:00
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2020-06-23 00:39:58 +00:00
|
|
|
State->Interface.Style.FontSize = FontSize;
|
|
|
|
State->Interface.Style.PanelBGColors[0] = v4{.3f, .3f, .3f, 1};
|
|
|
|
State->Interface.Style.PanelBGColors[1] = v4{.4f, .4f, .4f, 1};
|
|
|
|
State->Interface.Style.PanelBGColors[2] = v4{.5f, .5f, .5f, 1};
|
|
|
|
State->Interface.Style.PanelBGColors[3] = v4{.6f, .6f, .6f, 1};
|
|
|
|
State->Interface.Style.ButtonColor_Inactive = BlackV4;
|
|
|
|
State->Interface.Style.ButtonColor_Active = v4{.1f, .1f, .1f, 1};
|
|
|
|
State->Interface.Style.ButtonColor_Selected = v4{.1f, .1f, .3f, 1};
|
|
|
|
State->Interface.Style.TextColor = WhiteV4;
|
|
|
|
State->Interface.Style.ListBGColors[0] = v4{ .16f, .16f, .16f, 1.f };
|
|
|
|
State->Interface.Style.ListBGColors[1] = v4{ .18f, .18f, .18f, 1.f };
|
|
|
|
State->Interface.Style.ListBGHover = v4{ .22f, .22f, .22f, 1.f };
|
|
|
|
State->Interface.Style.ListBGSelected = v4{.44f, .44f, .44f, 1.f };
|
|
|
|
State->Interface.Style.Margin = v2{5, 5};
|
|
|
|
State->Interface.Style.RowHeight = ui_GetTextLineHeight(State->Interface);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-11-23 00:07:25 +00:00
|
|
|
State->SACN = InitializeSACN(Context);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
State->Camera.FieldOfView = 45.0f;
|
|
|
|
State->Camera.AspectRatio = RectAspectRatio(State->WindowBounds);
|
|
|
|
State->Camera.Near = .1f;
|
|
|
|
State->Camera.Far = 800.0f;
|
|
|
|
State->Camera.Position = v3{0, 0, 400};
|
|
|
|
State->Camera.LookAt = v3{0, 0, 0
|
|
|
|
};
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
State->LedSystem = LedSystemInitialize(Context.ThreadContext.Allocator, 128);
|
2020-06-15 22:36:50 +00:00
|
|
|
|
2019-07-19 20:56:21 +00:00
|
|
|
#if 1
|
2020-09-07 04:10:36 +00:00
|
|
|
gs_const_string SculpturePath = ConstString("data/blumen_lumen_v2.fold");
|
2020-09-07 20:42:00 +00:00
|
|
|
LoadAssembly(&State->Assemblies, &State->LedSystem, State->Transient, Context, SculpturePath, State->GlobalLog);
|
2020-09-07 19:37:32 +00:00
|
|
|
|
|
|
|
SculpturePath = ConstString("data/radialumia_v2.fold");
|
2020-09-07 20:42:00 +00:00
|
|
|
LoadAssembly(&State->Assemblies, &State->LedSystem, State->Transient, Context, SculpturePath, State->GlobalLog);
|
2019-07-19 20:56:21 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
State->PixelsToWorldScale = .01f;
|
|
|
|
|
|
|
|
GlobalDebugServices->Interface.RenderSculpture = true;
|
|
|
|
|
2020-02-06 04:33:12 +00:00
|
|
|
ReloadStaticData(Context, GlobalDebugServices);
|
2019-11-01 12:46:40 +00:00
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
State->Modes = OperationModeSystemInit(&State->Permanent, Context.ThreadContext);
|
2019-12-26 16:11:48 +00:00
|
|
|
|
2019-12-28 18:51:47 +00:00
|
|
|
{ // Animation PLAYGROUND
|
2019-12-31 21:15:28 +00:00
|
|
|
State->AnimationSystem = {};
|
2020-06-15 22:36:50 +00:00
|
|
|
State->AnimationSystem.Storage = &State->Permanent;
|
2019-12-28 18:51:47 +00:00
|
|
|
State->AnimationSystem.SecondsPerFrame = 1.f / 24.f;
|
2020-03-02 01:19:30 +00:00
|
|
|
State->AnimationSystem.PlayableRange.Min = 0;
|
|
|
|
State->AnimationSystem.PlayableRange.Max = SecondsToFrames(15, State->AnimationSystem);
|
2020-03-08 00:06:10 +00:00
|
|
|
State->AnimationSystem.LayersMax = 32;
|
|
|
|
State->AnimationSystem.Layers = PushArray(&State->Permanent, anim_layer, State->AnimationSystem.LayersMax);
|
2020-07-18 19:00:14 +00:00
|
|
|
AddLayer(MakeString("Base Layer"), &State->AnimationSystem, BlendMode_Overwrite);
|
|
|
|
AddLayer(MakeString("Color Layer"), &State->AnimationSystem, BlendMode_Multiply);
|
|
|
|
AddLayer(MakeString("Sparkles"), &State->AnimationSystem, BlendMode_Add);
|
2019-12-26 16:11:48 +00:00
|
|
|
} // End Animation Playground
|
2019-12-26 20:42:55 +00:00
|
|
|
|
|
|
|
|
2019-12-28 21:02:19 +00:00
|
|
|
InitializePanelSystem(&State->PanelSystem);
|
|
|
|
panel* Panel = TakeNewPanel(&State->PanelSystem);
|
2020-09-07 20:42:00 +00:00
|
|
|
SetPanelDefinition(Panel, PanelType_SculptureView, State, Context);
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2019-11-11 20:02:24 +00:00
|
|
|
internal void
|
2020-09-07 20:42:00 +00:00
|
|
|
HandleInput (app_state* State, rect2 WindowBounds, input_queue InputQueue, mouse_state Mouse, context Context)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-11-11 20:02:24 +00:00
|
|
|
DEBUG_TRACK_FUNCTION;
|
|
|
|
|
2019-12-28 22:00:09 +00:00
|
|
|
b32 PanelSystemHandledInput = HandleMousePanelInteraction(&State->PanelSystem, State->WindowBounds, Mouse, State);
|
|
|
|
if (!PanelSystemHandledInput)
|
2019-08-18 12:56:18 +00:00
|
|
|
{
|
2019-12-28 22:00:09 +00:00
|
|
|
input_command_registry ActiveCommands = {};
|
|
|
|
if (State->Modes.ActiveModesCount > 0)
|
2019-11-01 12:46:40 +00:00
|
|
|
{
|
2019-12-28 22:00:09 +00:00
|
|
|
ActiveCommands = State->Modes.ActiveModes[State->Modes.ActiveModesCount - 1].Commands;
|
2019-11-01 12:46:40 +00:00
|
|
|
}
|
2019-12-28 22:00:09 +00:00
|
|
|
else
|
2019-10-30 14:28:02 +00:00
|
|
|
{
|
2019-12-28 22:00:09 +00:00
|
|
|
panel_and_bounds PanelWithMouseOverIt = GetPanelContainingPoint(Mouse.Pos, &State->PanelSystem, WindowBounds);
|
|
|
|
if (!PanelWithMouseOverIt.Panel) { return; }
|
2019-12-30 02:28:59 +00:00
|
|
|
State->HotPanel = PanelWithMouseOverIt.Panel;
|
2019-12-28 22:00:09 +00:00
|
|
|
|
|
|
|
panel_definition PanelDefinition = GlobalPanelDefs[PanelWithMouseOverIt.Panel->PanelDefinitionIndex];
|
|
|
|
if (!PanelDefinition.InputCommands) { return; }
|
|
|
|
|
|
|
|
ActiveCommands.Commands = PanelDefinition.InputCommands;
|
|
|
|
ActiveCommands.Size = sizeof(*PanelDefinition.InputCommands) / sizeof(PanelDefinition.InputCommands[0]);
|
|
|
|
ActiveCommands.Used = ActiveCommands.Size;
|
2019-10-30 14:28:02 +00:00
|
|
|
}
|
2019-12-28 22:00:09 +00:00
|
|
|
|
|
|
|
for (s32 EventIdx = 0; EventIdx < InputQueue.QueueUsed; EventIdx++)
|
2019-10-30 14:28:02 +00:00
|
|
|
{
|
2019-12-28 22:00:09 +00:00
|
|
|
input_entry Event = InputQueue.Entries[EventIdx];
|
|
|
|
|
2020-03-22 05:44:44 +00:00
|
|
|
// 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
|
2019-12-28 22:00:09 +00:00
|
|
|
// frame when the button was released, even if the command is registered to both events
|
|
|
|
if (KeyTransitionedDown(Event))
|
|
|
|
{
|
2020-03-22 05:44:44 +00:00
|
|
|
FindAndPushExistingCommand(ActiveCommands, Event, Command_Began, &State->CommandQueue);
|
2019-12-28 22:00:09 +00:00
|
|
|
}
|
|
|
|
else if (KeyTransitionedUp(Event))
|
|
|
|
{
|
2020-03-22 05:44:44 +00:00
|
|
|
FindAndPushExistingCommand(ActiveCommands, Event, Command_Ended, &State->CommandQueue);
|
2019-12-28 22:00:09 +00:00
|
|
|
}
|
|
|
|
else if (KeyHeldDown(Event))
|
|
|
|
{
|
2020-03-22 05:44:44 +00:00
|
|
|
FindAndPushExistingCommand(ActiveCommands, Event, Command_Held, &State->CommandQueue);
|
2019-12-28 22:00:09 +00:00
|
|
|
}
|
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];
|
2020-09-07 20:42:00 +00:00
|
|
|
Entry->Command.Proc(State, Entry->Event, Mouse, Context);
|
2019-11-11 20:02:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ClearCommandQueue(&State->CommandQueue);
|
|
|
|
}
|
|
|
|
|
|
|
|
UPDATE_AND_RENDER(UpdateAndRender)
|
|
|
|
{
|
|
|
|
DEBUG_TRACK_FUNCTION;
|
2020-03-13 05:42:59 +00:00
|
|
|
app_state* State = (app_state*)Context->MemoryBase;
|
2019-11-11 20:02:24 +00:00
|
|
|
|
|
|
|
// 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
|
2020-03-22 05:44:44 +00:00
|
|
|
// zero the Transient arena when we clear it so it wouldn't be a problem, but it is technically
|
2019-11-11 20:02:24 +00:00
|
|
|
// incorrect to clear the arena, and then access the memory later.
|
2020-09-07 20:42:00 +00:00
|
|
|
ClearArena(State->Transient);
|
2020-03-13 05:42:59 +00:00
|
|
|
Context->Mouse.CursorType = CursorType_Arrow;
|
2019-11-11 20:02:24 +00:00
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
PushRenderClearScreen(RenderBuffer);
|
|
|
|
State->Camera.AspectRatio = RectAspectRatio(Context->WindowBounds);
|
|
|
|
|
2020-09-07 20:42:00 +00:00
|
|
|
HandleInput(State, State->WindowBounds, InputQueue, Context->Mouse, *Context);
|
2019-11-11 20:02:24 +00:00
|
|
|
|
2020-03-22 05:44:44 +00:00
|
|
|
if (State->AnimationSystem.TimelineShouldAdvance) {
|
2020-03-01 01:11:15 +00:00
|
|
|
// TODO(Peter): Revisit this. This implies that the framerate of the animation system
|
|
|
|
// is tied to the framerate of the simulation. That seems correct to me, but I'm not sure
|
2020-03-22 05:44:44 +00:00
|
|
|
State->AnimationSystem.CurrentFrame += 1;
|
2020-03-01 01:11:15 +00:00
|
|
|
// Loop back to the beginning
|
2020-03-02 01:19:30 +00:00
|
|
|
if (State->AnimationSystem.CurrentFrame > State->AnimationSystem.PlayableRange.Max)
|
2019-12-26 20:42:55 +00:00
|
|
|
{
|
2020-03-22 05:44:44 +00:00
|
|
|
State->AnimationSystem.CurrentFrame = 0;
|
2019-12-26 20:42:55 +00:00
|
|
|
}
|
2019-12-27 00:23:43 +00:00
|
|
|
}
|
2019-12-28 18:51:47 +00:00
|
|
|
|
2020-03-08 00:06:10 +00:00
|
|
|
s32 CurrentFrame = State->AnimationSystem.CurrentFrame;
|
2019-12-28 18:51:47 +00:00
|
|
|
if (CurrentFrame != State->AnimationSystem.LastUpdatedFrame)
|
|
|
|
{
|
|
|
|
State->AnimationSystem.LastUpdatedFrame = CurrentFrame;
|
|
|
|
r32 FrameTime = CurrentFrame * State->AnimationSystem.SecondsPerFrame;
|
|
|
|
|
2020-03-08 00:29:07 +00:00
|
|
|
u32 CurrentBlocksMax = State->AnimationSystem.LayersCount;
|
2020-09-07 20:42:00 +00:00
|
|
|
b8* CurrentBlocksFilled = PushArray(State->Transient, b8, CurrentBlocksMax);
|
2020-07-18 19:00:14 +00:00
|
|
|
ZeroArray(CurrentBlocksFilled, b8, CurrentBlocksMax);
|
2020-09-07 20:42:00 +00:00
|
|
|
animation_block* CurrentBlocks = PushArray(State->Transient, animation_block, CurrentBlocksMax);
|
2020-03-08 00:29:07 +00:00
|
|
|
|
2019-12-31 04:26:28 +00:00
|
|
|
for (u32 i = 0; i < State->AnimationSystem.Blocks.Used; i++)
|
2019-12-04 06:40:22 +00:00
|
|
|
{
|
2019-12-31 04:26:28 +00:00
|
|
|
gs_list_entry<animation_block>* BlockEntry = State->AnimationSystem.Blocks.GetEntryAtIndex(i);
|
2020-03-08 00:06:10 +00:00
|
|
|
if (EntryIsFree(BlockEntry)) { continue; }
|
|
|
|
animation_block Block = BlockEntry->Value;
|
|
|
|
if (CurrentFrame < Block.Range.Min || CurrentFrame > Block.Range.Max) { continue; }
|
2020-03-08 00:29:07 +00:00
|
|
|
CurrentBlocksFilled[Block.Layer] = true;
|
|
|
|
CurrentBlocks[Block.Layer] = Block;
|
|
|
|
}
|
|
|
|
|
2020-09-07 20:42:00 +00:00
|
|
|
led_buffer* LayerLEDBuffers = PushArray(State->Transient, led_buffer, CurrentBlocksMax);
|
2020-06-15 22:36:50 +00:00
|
|
|
for (u32 AssemblyIndex = 0; AssemblyIndex < State->Assemblies.Count; AssemblyIndex++)
|
2020-03-08 21:44:28 +00:00
|
|
|
{
|
2020-06-15 22:36:50 +00:00
|
|
|
assembly* Assembly = &State->Assemblies.Values[AssemblyIndex];
|
|
|
|
led_buffer* AssemblyLedBuffer = LedSystemGetBuffer(&State->LedSystem, Assembly->LedBufferIndex);
|
2020-03-08 21:44:28 +00:00
|
|
|
|
|
|
|
for (u32 Layer = 0; Layer < CurrentBlocksMax; Layer++)
|
|
|
|
{
|
|
|
|
if (!CurrentBlocksFilled[Layer]) { continue; }
|
|
|
|
animation_block Block = CurrentBlocks[Layer];
|
|
|
|
|
|
|
|
// Prep Temp Buffer
|
2020-06-15 22:36:50 +00:00
|
|
|
LayerLEDBuffers[Layer] = *AssemblyLedBuffer;
|
2020-09-07 20:42:00 +00:00
|
|
|
LayerLEDBuffers[Layer].Colors = PushArray(State->Transient, pixel, AssemblyLedBuffer->LedCount);
|
2020-03-08 21:44:28 +00:00
|
|
|
|
|
|
|
u32 FramesIntoBlock = CurrentFrame - Block.Range.Min;
|
|
|
|
r32 SecondsIntoBlock = FramesIntoBlock * State->AnimationSystem.SecondsPerFrame;
|
2020-09-07 04:18:49 +00:00
|
|
|
|
|
|
|
u32 AnimationProcIndex = Block.AnimationProcHandle - 1;
|
|
|
|
animation_proc* AnimationProc = GlobalAnimationClips[AnimationProcIndex].Proc;
|
2020-09-07 20:42:00 +00:00
|
|
|
AnimationProc(&LayerLEDBuffers[Layer], *Assembly, SecondsIntoBlock, State->Transient);
|
2020-03-08 21:44:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Consolidate Temp Buffers
|
|
|
|
// We do this in reverse order so that they go from top to bottom
|
|
|
|
for (u32 Layer = 0; Layer < CurrentBlocksMax; Layer++)
|
|
|
|
{
|
|
|
|
if (!CurrentBlocksFilled[Layer]) { continue; }
|
|
|
|
|
|
|
|
switch (State->AnimationSystem.Layers[Layer].BlendMode)
|
|
|
|
{
|
|
|
|
case BlendMode_Overwrite:
|
|
|
|
{
|
2020-06-15 22:36:50 +00:00
|
|
|
for (u32 LED = 0; LED < AssemblyLedBuffer->LedCount; LED++)
|
2020-03-08 21:44:28 +00:00
|
|
|
{
|
2020-06-15 22:36:50 +00:00
|
|
|
AssemblyLedBuffer->Colors[LED] = LayerLEDBuffers[Layer].Colors[LED];
|
2020-03-08 21:44:28 +00:00
|
|
|
}
|
|
|
|
}break;
|
|
|
|
|
|
|
|
case BlendMode_Add:
|
|
|
|
{
|
2020-06-15 22:36:50 +00:00
|
|
|
for (u32 LED = 0; LED < AssemblyLedBuffer->LedCount; LED++)
|
2020-03-08 21:44:28 +00:00
|
|
|
{
|
2020-06-15 22:36:50 +00:00
|
|
|
u32 R = (u32)AssemblyLedBuffer->Colors[LED].R + (u32)LayerLEDBuffers[Layer].Colors[LED].R;
|
|
|
|
u32 G = (u32)AssemblyLedBuffer->Colors[LED].G + (u32)LayerLEDBuffers[Layer].Colors[LED].G;
|
|
|
|
u32 B = (u32)AssemblyLedBuffer->Colors[LED].B + (u32)LayerLEDBuffers[Layer].Colors[LED].B;
|
2020-03-08 21:44:28 +00:00
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
AssemblyLedBuffer->Colors[LED].R = (u8)Min(R, (u32)255);
|
|
|
|
AssemblyLedBuffer->Colors[LED].G = (u8)Min(G, (u32)255);
|
|
|
|
AssemblyLedBuffer->Colors[LED].B = (u8)Min(B, (u32)255);
|
2020-03-08 21:44:28 +00:00
|
|
|
}
|
|
|
|
}break;
|
|
|
|
|
|
|
|
case BlendMode_Multiply:
|
|
|
|
{
|
2020-06-15 22:36:50 +00:00
|
|
|
for (u32 LED = 0; LED < AssemblyLedBuffer->LedCount; LED++)
|
2020-03-08 21:44:28 +00:00
|
|
|
{
|
2020-06-15 22:36:50 +00:00
|
|
|
r32 DR = (r32)AssemblyLedBuffer->Colors[LED].R / 255.f;
|
|
|
|
r32 DG = (r32)AssemblyLedBuffer->Colors[LED].G / 255.f;
|
|
|
|
r32 DB = (r32)AssemblyLedBuffer->Colors[LED].B / 255.f;
|
2020-03-08 21:44:28 +00:00
|
|
|
|
|
|
|
r32 SR = (r32)LayerLEDBuffers[Layer].Colors[LED].R / 255.f;
|
|
|
|
r32 SG = (r32)LayerLEDBuffers[Layer].Colors[LED].G / 255.f;
|
|
|
|
r32 SB = (r32)LayerLEDBuffers[Layer].Colors[LED].B / 255.f;
|
|
|
|
|
2020-06-15 22:36:50 +00:00
|
|
|
AssemblyLedBuffer->Colors[LED].R = (u8)((DR * SR) * 255.f);
|
|
|
|
AssemblyLedBuffer->Colors[LED].G = (u8)((DG * SG) * 255.f);
|
|
|
|
AssemblyLedBuffer->Colors[LED].B = (u8)((DB * SB) * 255.f);
|
2020-03-08 21:44:28 +00:00
|
|
|
}
|
|
|
|
}break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-12-28 18:51:47 +00:00
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2020-10-01 22:30:24 +00:00
|
|
|
addressed_data_buffer_list OutputData = {0};
|
|
|
|
switch (State->NetworkProtocol)
|
2019-11-23 09:08:59 +00:00
|
|
|
{
|
2020-10-01 22:30:24 +00:00
|
|
|
case NetworkProtocol_SACN:
|
|
|
|
{
|
|
|
|
SACN_BuildOutputData(&State->SACN, &OutputData, State->Assemblies, &State->LedSystem, State->Transient);
|
|
|
|
}break;
|
|
|
|
|
|
|
|
case NetworkProtocol_UART:
|
|
|
|
{
|
|
|
|
//UART_BuildOutputData(&OutputData, State, State->Transient);
|
|
|
|
}break;
|
|
|
|
|
|
|
|
case NetworkProtocol_ArtNet:
|
|
|
|
InvalidDefaultCase;
|
2019-11-23 09:08:59 +00:00
|
|
|
}
|
|
|
|
|
2020-10-01 22:30:24 +00:00
|
|
|
if (0)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-10-01 22:30:24 +00:00
|
|
|
// TODO(pjs): This should happen on another thread
|
|
|
|
AddressedDataBufferList_SendAll(OutputData, State->SACN.SendSocket, *Context);
|
|
|
|
|
|
|
|
/*
|
|
|
|
Saved this lien as an example of pushing onto a queue
|
|
|
|
Context->GeneralWorkQueue->PushWorkOnQueue(Context->GeneralWorkQueue, SACNSendDMXBufferListJob, Job, "SACN Send Data Job");
|
|
|
|
*/
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
2020-10-01 22:30:24 +00:00
|
|
|
|
|
|
|
// Skipped for performance at the moment
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2020-06-27 18:23:50 +00:00
|
|
|
|
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
PushRenderOrthographic(RenderBuffer, State->WindowBounds);
|
2019-12-28 18:51:47 +00:00
|
|
|
PushRenderClearScreen(RenderBuffer);
|
|
|
|
|
2020-03-20 07:55:13 +00:00
|
|
|
State->WindowBounds = Context->WindowBounds;
|
2020-06-23 00:39:58 +00:00
|
|
|
State->Interface.RenderBuffer = RenderBuffer;
|
|
|
|
State->Interface.Mouse = Context->Mouse;
|
2020-03-20 07:55:13 +00:00
|
|
|
|
2020-09-07 20:42:00 +00:00
|
|
|
panel_layout PanelsToRender = GetPanelLayout(&State->PanelSystem, State->WindowBounds, State->Transient);
|
2020-03-13 05:42:59 +00:00
|
|
|
DrawAllPanels(PanelsToRender, RenderBuffer, &Context->Mouse, State, *Context);
|
2019-12-28 18:51:47 +00:00
|
|
|
|
|
|
|
for (s32 m = 0; m < State->Modes.ActiveModesCount; m++)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2019-12-28 18:51:47 +00:00
|
|
|
operation_mode OperationMode = State->Modes.ActiveModes[m];
|
|
|
|
if (OperationMode.Render != 0)
|
2019-07-19 20:56:21 +00:00
|
|
|
{
|
2020-09-07 20:42:00 +00:00
|
|
|
OperationMode.Render(State, RenderBuffer, OperationMode, Context->Mouse, *Context);
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
2019-12-26 21:14:00 +00:00
|
|
|
}
|
2019-12-23 01:47:26 +00:00
|
|
|
|
2020-07-18 19:00:14 +00:00
|
|
|
Context->GeneralWorkQueue->CompleteQueueWork(Context->GeneralWorkQueue, Context->ThreadContext);
|
2020-03-13 05:42:59 +00:00
|
|
|
Context->GeneralWorkQueue->ResetWorkQueue(Context->GeneralWorkQueue);
|
2020-02-29 23:07:56 +00:00
|
|
|
|
2019-12-23 01:47:26 +00:00
|
|
|
// Checking for overflows
|
2020-06-20 01:53:23 +00:00
|
|
|
#if 0
|
2019-12-23 01:47:26 +00:00
|
|
|
{
|
|
|
|
DEBUG_TRACK_SCOPE(OverflowChecks);
|
2019-12-26 20:42:55 +00:00
|
|
|
AssertAllocationsNoOverflow(State->Permanent);
|
2020-06-15 22:36:50 +00:00
|
|
|
for (u32 i = 0; i < State->Assemblies.Count; i++)
|
2019-12-26 20:42:55 +00:00
|
|
|
{
|
2020-06-15 22:36:50 +00:00
|
|
|
assembly* Assembly = &State->Assemblies.Values[i];
|
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
|
|
|
}
|
2020-06-20 01:53:23 +00:00
|
|
|
#endif
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
CLEANUP_APPLICATION(CleanupApplication)
|
|
|
|
{
|
|
|
|
app_state* State = (app_state*)Context.MemoryBase;
|
|
|
|
SACNCleanup(&State->SACN, Context);
|
2020-01-02 02:41:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#define FOLDHAUS_APP_CPP
|
|
|
|
#endif // FOLDHAUS_APP_CPP
|