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;
|
2020-11-03 20:49:16 +00:00
|
|
|
State->PanelSystem.PanelDefs = GlobalPanelDefs;
|
|
|
|
State->PanelSystem.PanelDefsCount = GlobalPanelDefsCount;
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
2020-10-03 15:46:14 +00:00
|
|
|
State->Assemblies = AssemblyArray_Create(8, &State->Permanent);
|
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};
|
|
|
|
|
2020-11-08 07:15:39 +00:00
|
|
|
State->CommandQueue = CommandQueue_Create(&State->Permanent, 32);
|
2019-10-30 14:28:02 +00:00
|
|
|
|
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};
|
2020-11-08 06:54:59 +00:00
|
|
|
State->Interface.Style.ButtonColor_Selected = v4{.3f, .3f, .3f, 1};
|
2020-06-23 00:39:58 +00:00
|
|
|
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};
|
2020-11-15 22:48:04 +00:00
|
|
|
State->Interface.Style.RowHeight = ui_GetTextLineHeight(State->Interface) + (2 * State->Interface.Style.Margin.y);
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2020-11-08 06:54:59 +00:00
|
|
|
State->Interface.WidgetsCountMax = 4096;
|
|
|
|
State->Interface.Widgets = PushArray(&State->Permanent, ui_widget, State->Interface.WidgetsCountMax);
|
2020-11-15 01:18:38 +00:00
|
|
|
State->Interface.PerFrameMemory = PushStruct(&State->Permanent, gs_memory_arena);
|
|
|
|
*State->Interface.PerFrameMemory = CreateMemoryArena(Context.ThreadContext.Allocator);
|
2020-11-08 06:54:59 +00:00
|
|
|
|
2020-10-01 22:41:32 +00:00
|
|
|
State->SACN = SACN_Initialize(Context);
|
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-10-12 03:54:38 +00:00
|
|
|
gs_const_string SculpturePath = ConstString("data/blumen_lumen_silver_spring.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;
|
2020-10-10 05:08:51 +00:00
|
|
|
State->AnimationSystem.Animations = AnimationArray_Create(State->AnimationSystem.Storage, 32);
|
|
|
|
|
2019-12-28 18:51:47 +00:00
|
|
|
State->AnimationSystem.SecondsPerFrame = 1.f / 24.f;
|
2020-10-10 05:08:51 +00:00
|
|
|
|
|
|
|
animation Anim = {0};
|
2020-10-10 07:10:51 +00:00
|
|
|
Anim.Name = PushStringF(&State->Permanent, 256, "test_anim_one");
|
2020-10-18 20:57:04 +00:00
|
|
|
Anim.Layers = AnimLayerArray_Create(State->AnimationSystem.Storage, 8);
|
|
|
|
Anim.Blocks_ = AnimBlockArray_Create(State->AnimationSystem.Storage, 8);
|
2020-10-10 05:08:51 +00:00
|
|
|
Anim.PlayableRange.Min = 0;
|
|
|
|
Anim.PlayableRange.Max = SecondsToFrames(15, State->AnimationSystem);
|
|
|
|
Animation_AddLayer(&Anim, MakeString("Base Layer"), BlendMode_Overwrite, &State->AnimationSystem);
|
|
|
|
Animation_AddLayer(&Anim, MakeString("Color Layer"), BlendMode_Multiply, &State->AnimationSystem);
|
|
|
|
Animation_AddLayer(&Anim, MakeString("Sparkles"), BlendMode_Add, &State->AnimationSystem);
|
|
|
|
|
2020-11-15 06:08:13 +00:00
|
|
|
Animation_AddBlock(&Anim, 0, Anim.PlayableRange.Max, 4, 0);
|
2020-10-10 05:08:51 +00:00
|
|
|
|
2020-10-10 07:10:51 +00:00
|
|
|
AnimationArray_Push(&State->AnimationSystem.Animations, Anim);
|
2020-11-15 06:08:13 +00:00
|
|
|
|
|
|
|
State->AnimationSystem.TimelineShouldAdvance = true;
|
2019-12-26 16:11:48 +00:00
|
|
|
} // End Animation Playground
|
2019-12-26 20:42:55 +00:00
|
|
|
|
2020-11-03 20:49:16 +00:00
|
|
|
PanelSystem_Init(&State->PanelSystem, GlobalPanelDefs, GlobalPanelDefsCount, &State->Permanent);
|
2020-10-18 22:31:53 +00:00
|
|
|
PanelSystem_PushPanel(&State->PanelSystem, PanelType_SculptureView, State, Context);
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
|
|
|
|
2019-11-11 20:02:24 +00:00
|
|
|
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);
|
2019-11-11 20:02:24 +00:00
|
|
|
|
2020-10-25 01:54:47 +00:00
|
|
|
Editor_Update(State, Context, InputQueue);
|
2019-11-11 20:02:24 +00:00
|
|
|
|
2020-11-14 20:19:36 +00:00
|
|
|
AnimationSystem_Update(&State->AnimationSystem);
|
|
|
|
if (AnimationSystem_NeedsRender(State->AnimationSystem))
|
2020-10-10 05:08:51 +00:00
|
|
|
{
|
2020-11-14 20:19:36 +00:00
|
|
|
State->AnimationSystem.LastUpdatedFrame = State->AnimationSystem.CurrentFrame;
|
|
|
|
AnimationSystem_RenderToLedBuffers(&State->AnimationSystem,
|
|
|
|
State->Assemblies,
|
|
|
|
&State->LedSystem,
|
|
|
|
GlobalAnimationPatterns,
|
|
|
|
State->Transient);
|
2019-12-28 18:51:47 +00:00
|
|
|
}
|
2019-07-19 20:56:21 +00:00
|
|
|
|
2019-11-23 09:08:59 +00:00
|
|
|
{
|
2020-10-01 22:41:32 +00:00
|
|
|
// NOTE(pjs): Building data buffers to be sent out to the sculpture
|
2020-10-03 15:46:14 +00:00
|
|
|
// This array is used on the platform side to actually send the information
|
|
|
|
assembly_array SACNAssemblies = AssemblyArray_Filter(State->Assemblies, AssemblyFilter_OutputsViaSACN, State->Transient);
|
|
|
|
assembly_array UARTAssemblies = AssemblyArray_Filter(State->Assemblies, AssemblyFilter_OutputsViaUART, State->Transient);
|
|
|
|
SACN_BuildOutputData(&State->SACN, OutputData, SACNAssemblies, &State->LedSystem);
|
|
|
|
UART_BuildOutputData(OutputData, UARTAssemblies, &State->LedSystem);
|
2019-07-19 20:56:21 +00:00
|
|
|
}
|
2020-10-01 22:30:24 +00:00
|
|
|
|
2020-10-25 01:54:47 +00:00
|
|
|
Editor_Render(State, Context, RenderBuffer);
|
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;
|
2020-10-01 22:41:32 +00:00
|
|
|
SACN_Cleanup(&State->SACN, Context);
|
2020-01-02 02:41:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#define FOLDHAUS_APP_CPP
|
|
|
|
#endif // FOLDHAUS_APP_CPP
|