2020-01-02 02:41:43 +00:00
|
|
|
//
|
|
|
|
// File: foldhaus_animation.h
|
|
|
|
// Author: Peter Slattery
|
|
|
|
// Creation Date: 2020-01-01
|
|
|
|
//
|
|
|
|
#ifndef FOLDHAUS_ANIMATION
|
|
|
|
|
2020-09-07 04:10:36 +00:00
|
|
|
#define ANIMATION_PROC(name) void name(led_buffer* Leds, assembly Assembly, r32 Time, gs_memory_arena* Transient)
|
2019-12-26 16:11:48 +00:00
|
|
|
typedef ANIMATION_PROC(animation_proc);
|
|
|
|
|
2020-03-02 01:19:30 +00:00
|
|
|
struct frame_range
|
|
|
|
{
|
2020-03-08 00:06:10 +00:00
|
|
|
s32 Min;
|
|
|
|
s32 Max;
|
2020-03-02 01:19:30 +00:00
|
|
|
};
|
|
|
|
|
2019-11-29 05:12:57 +00:00
|
|
|
struct animation_block
|
|
|
|
{
|
2020-03-02 01:19:30 +00:00
|
|
|
frame_range Range;
|
2020-02-05 06:50:12 +00:00
|
|
|
u32 AnimationProcHandle;
|
2019-12-26 16:11:48 +00:00
|
|
|
u32 Layer;
|
|
|
|
};
|
|
|
|
|
2020-03-08 21:44:28 +00:00
|
|
|
enum blend_mode
|
|
|
|
{
|
|
|
|
BlendMode_Overwrite,
|
|
|
|
BlendMode_Add,
|
|
|
|
BlendMode_Multiply,
|
|
|
|
BlendMode_Count,
|
|
|
|
};
|
|
|
|
|
2020-03-08 00:06:10 +00:00
|
|
|
struct anim_layer
|
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
gs_string Name;
|
2020-03-08 21:44:28 +00:00
|
|
|
blend_mode BlendMode;
|
2020-03-08 00:06:10 +00:00
|
|
|
};
|
|
|
|
|
2020-10-10 05:08:51 +00:00
|
|
|
struct anim_layer_array
|
|
|
|
{
|
|
|
|
anim_layer* Values;
|
|
|
|
u32 Count;
|
|
|
|
u32 CountMax;
|
|
|
|
};
|
|
|
|
|
2020-10-05 20:17:33 +00:00
|
|
|
struct animation
|
|
|
|
{
|
2020-10-10 05:08:51 +00:00
|
|
|
anim_layer_array Layers;
|
2020-10-05 20:17:33 +00:00
|
|
|
gs_list<animation_block> Blocks;
|
|
|
|
|
|
|
|
frame_range PlayableRange;
|
|
|
|
};
|
|
|
|
|
2020-10-10 05:08:51 +00:00
|
|
|
struct animation_array
|
|
|
|
{
|
|
|
|
animation* Values;
|
|
|
|
u32 Count;
|
|
|
|
u32 CountMax;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct animation_frame
|
|
|
|
{
|
|
|
|
// NOTE(pjs): These are all parallel arrays of equal length
|
|
|
|
animation_block* Blocks;
|
|
|
|
b8* BlocksFilled;
|
|
|
|
|
|
|
|
u32 BlocksCountMax;
|
|
|
|
u32 BlocksCount;
|
|
|
|
};
|
|
|
|
|
2019-12-26 16:11:48 +00:00
|
|
|
#define ANIMATION_SYSTEM_LAYERS_MAX 128
|
|
|
|
#define ANIMATION_SYSTEM_BLOCKS_MAX 128
|
|
|
|
struct animation_system
|
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
gs_memory_arena* Storage;
|
2020-10-10 05:08:51 +00:00
|
|
|
animation_array Animations;
|
2020-03-08 00:06:10 +00:00
|
|
|
|
2020-10-10 05:08:51 +00:00
|
|
|
frame_range PlayableRange_;
|
|
|
|
gs_list<animation_block> Blocks_;
|
2019-12-26 16:11:48 +00:00
|
|
|
|
2020-03-01 01:11:15 +00:00
|
|
|
// NOTE(Peter): The frame currently being displayed/processed. you
|
|
|
|
// can see which frame you're on by looking at the time slider on the timeline
|
|
|
|
// panel
|
2020-03-08 00:06:10 +00:00
|
|
|
s32 CurrentFrame;
|
|
|
|
s32 LastUpdatedFrame;
|
2019-12-27 00:23:43 +00:00
|
|
|
r32 SecondsPerFrame;
|
2019-12-28 19:31:21 +00:00
|
|
|
b32 TimelineShouldAdvance;
|
|
|
|
|
2019-11-29 05:12:57 +00:00
|
|
|
};
|
2020-01-02 02:41:43 +00:00
|
|
|
|
2020-10-10 05:08:51 +00:00
|
|
|
//////////////////////////
|
|
|
|
//
|
|
|
|
// Anim Layers Array
|
|
|
|
|
|
|
|
internal anim_layer_array
|
|
|
|
AnimLayerArray_Create(gs_memory_arena* Storage, u32 CountMax)
|
|
|
|
{
|
|
|
|
anim_layer_array Result = {0};
|
|
|
|
Result.CountMax = CountMax;
|
|
|
|
Result.Values = PushArray(Storage, anim_layer, Result.CountMax);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal u32
|
|
|
|
AnimLayerArray_Push(anim_layer_array* Array, anim_layer Value)
|
|
|
|
{
|
|
|
|
Assert(Array->Count < Array->CountMax);
|
|
|
|
u32 Index = Array->Count++;
|
|
|
|
Array->Values[Index] = Value;
|
|
|
|
return Index;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal u32
|
|
|
|
AnimLayerArray_Remove(anim_layer_array* Array, u32 Index)
|
|
|
|
{
|
|
|
|
Assert(Index < Array->Count);
|
|
|
|
for (u32 i = Index; i < Array->Count - 1; i++)
|
|
|
|
{
|
|
|
|
Array->Values[i] = Array->Values[i + 1];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////
|
|
|
|
//
|
|
|
|
// Animation Array
|
|
|
|
|
|
|
|
internal animation_array
|
|
|
|
AnimationArray_Create(gs_memory_arena* Storage, u32 CountMax)
|
|
|
|
{
|
|
|
|
animation_array Result = {0};
|
|
|
|
Result.CountMax = CountMax;
|
|
|
|
Result.Values = PushArray(Storage, animation, Result.CountMax);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal u32
|
|
|
|
AnimationArray_Push(animation_array* Array, animation Value)
|
|
|
|
{
|
|
|
|
Assert(Array->Count < Array->CountMax);
|
|
|
|
u32 Index = Array->Count++;
|
|
|
|
Array->Values[Index] = Value;
|
|
|
|
return Index;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////
|
|
|
|
//
|
|
|
|
// Animation
|
|
|
|
|
|
|
|
internal u32
|
|
|
|
Animation_AddLayer(animation* Animation, anim_layer Layer)
|
|
|
|
{
|
|
|
|
return AnimLayerArray_Push(&Animation->Layers, Layer);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal u32
|
|
|
|
Animation_AddLayer (animation* Animation, gs_string Name, blend_mode BlendMode, animation_system* System)
|
|
|
|
{
|
|
|
|
anim_layer NewLayer = {0};
|
|
|
|
NewLayer.Name = PushStringF(System->Storage, 256, "%S", Name);
|
|
|
|
NewLayer.BlendMode = BlendMode;
|
|
|
|
|
|
|
|
return Animation_AddLayer(Animation, NewLayer);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal void
|
|
|
|
Animation_RemoveLayer (animation* Animation, u32 LayerIndex)
|
|
|
|
{
|
|
|
|
AnimLayerArray_Remove(&Animation->Layers, LayerIndex);
|
|
|
|
for (u32 i = Animation->Blocks.Used -= 1; i >= 0; i--)
|
|
|
|
{
|
|
|
|
gs_list_entry<animation_block>* Entry = Animation->Blocks.GetEntryAtIndex(i);
|
|
|
|
if (EntryIsFree(Entry)) { continue; }
|
|
|
|
|
|
|
|
animation_block* Block = &Entry->Value;
|
|
|
|
if (Block->Layer > LayerIndex)
|
|
|
|
{
|
|
|
|
Block->Layer -= 1;
|
|
|
|
}
|
|
|
|
else if (Block->Layer == LayerIndex)
|
|
|
|
{
|
|
|
|
Animation->Blocks.FreeElementAtIndex(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////
|
|
|
|
//
|
|
|
|
//
|
|
|
|
|
2020-03-01 01:11:15 +00:00
|
|
|
internal u32
|
|
|
|
SecondsToFrames(r32 Seconds, animation_system System)
|
|
|
|
{
|
|
|
|
u32 Result = Seconds * (1.0f / System.SecondsPerFrame);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2020-03-02 01:19:30 +00:00
|
|
|
inline b32
|
2020-10-10 05:08:51 +00:00
|
|
|
FrameIsInRange(frame_range Range, s32 Frame)
|
2020-03-02 01:19:30 +00:00
|
|
|
{
|
|
|
|
b32 Result = (Frame >= Range.Min) && (Frame <= Range.Max);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal u32
|
|
|
|
GetFrameCount(frame_range Range)
|
|
|
|
{
|
2020-07-18 19:00:14 +00:00
|
|
|
u32 Result = (u32)Max(0, Range.Max - Range.Min);
|
2020-03-02 01:19:30 +00:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal r32
|
|
|
|
FrameToPercentRange(s32 Frame, frame_range Range)
|
|
|
|
{
|
|
|
|
r32 Result = (r32)(Frame - Range.Min);
|
|
|
|
Result = Result / GetFrameCount(Range);
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2020-03-08 00:06:10 +00:00
|
|
|
internal s32
|
2020-03-02 01:19:30 +00:00
|
|
|
PercentToFrameInRange(r32 Percent, frame_range Range)
|
|
|
|
{
|
2020-03-08 00:06:10 +00:00
|
|
|
s32 Result = Range.Min + (s32)(Percent * GetFrameCount(Range));
|
2020-03-02 01:19:30 +00:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2020-06-15 22:36:50 +00:00
|
|
|
internal s32
|
2020-03-08 00:06:10 +00:00
|
|
|
ClampFrameToRange(s32 Frame, frame_range Range)
|
2020-03-02 01:19:30 +00:00
|
|
|
{
|
2020-03-08 00:06:10 +00:00
|
|
|
s32 Result = Frame;
|
2020-03-02 01:19:30 +00:00
|
|
|
if (Result < Range.Min)
|
|
|
|
{
|
|
|
|
Result = Range.Min;
|
|
|
|
}
|
|
|
|
else if (Result > Range.Max)
|
|
|
|
{
|
|
|
|
Result = Range.Max;
|
|
|
|
}
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2020-03-08 00:06:10 +00:00
|
|
|
// Blocks
|
|
|
|
|
|
|
|
internal gs_list_handle
|
2020-10-10 05:08:51 +00:00
|
|
|
Animation_AddBlock(animation* Animation, u32 StartFrame, s32 EndFrame, u32 AnimationProcHandle, u32 LayerIndex)
|
2020-03-08 00:06:10 +00:00
|
|
|
{
|
2020-10-10 05:08:51 +00:00
|
|
|
Assert(LayerIndex < Animation->Layers.Count);
|
|
|
|
|
2020-03-08 00:06:10 +00:00
|
|
|
animation_block NewBlock = {0};
|
|
|
|
NewBlock.Range.Min = StartFrame;
|
|
|
|
NewBlock.Range.Max = EndFrame;
|
|
|
|
NewBlock.AnimationProcHandle = AnimationProcHandle;
|
2020-10-10 05:08:51 +00:00
|
|
|
NewBlock.Layer = LayerIndex;
|
|
|
|
|
|
|
|
gs_list_handle Result = Animation->Blocks.PushElementOnList(NewBlock);
|
2020-03-08 00:06:10 +00:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
|
|
|
internal void
|
2020-10-10 05:08:51 +00:00
|
|
|
Animation_RemoveBlock(animation* Animation, gs_list_handle AnimationBlockHandle)
|
2020-03-08 00:06:10 +00:00
|
|
|
{
|
|
|
|
Assert(ListHandleIsValid(AnimationBlockHandle));
|
2020-10-10 05:08:51 +00:00
|
|
|
Animation->Blocks.FreeElementWithHandle(AnimationBlockHandle);
|
2020-03-08 00:06:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Layers
|
2020-10-10 05:08:51 +00:00
|
|
|
|
|
|
|
// System
|
|
|
|
|
|
|
|
internal animation*
|
|
|
|
AnimationSystem_GetActiveAnimation(animation_system* System)
|
2020-03-08 00:06:10 +00:00
|
|
|
{
|
2020-10-10 05:08:51 +00:00
|
|
|
// TODO(pjs): need a way to specify the active animation
|
|
|
|
return System->Animations.Values + 0;
|
2020-03-08 00:06:10 +00:00
|
|
|
}
|
|
|
|
|
2020-10-10 05:08:51 +00:00
|
|
|
internal animation_frame
|
|
|
|
AnimationSystem_CalculateAnimationFrame(animation_system* System, gs_memory_arena* Arena)
|
2020-03-08 00:06:10 +00:00
|
|
|
{
|
2020-10-10 05:08:51 +00:00
|
|
|
animation* ActiveAnim = AnimationSystem_GetActiveAnimation(System);
|
|
|
|
|
|
|
|
animation_frame Result = {0};
|
|
|
|
Result.BlocksCountMax = ActiveAnim->Layers.Count;
|
|
|
|
Result.Blocks = PushArray(Arena, animation_block, Result.BlocksCountMax);
|
|
|
|
Result.BlocksFilled = PushArray(Arena, b8, Result.BlocksCountMax);
|
|
|
|
|
|
|
|
for (u32 i = 0; i < ActiveAnim->Blocks.Used; i++)
|
2020-03-08 00:06:10 +00:00
|
|
|
{
|
2020-10-10 05:08:51 +00:00
|
|
|
gs_list_entry<animation_block>* BlockEntry = ActiveAnim->Blocks.GetEntryAtIndex(i);
|
|
|
|
if (EntryIsFree(BlockEntry)) { continue; }
|
|
|
|
|
|
|
|
animation_block Block = BlockEntry->Value;
|
|
|
|
|
|
|
|
if (FrameIsInRange(Block.Range, System->CurrentFrame)){ continue; }
|
|
|
|
|
|
|
|
Result.BlocksFilled[Block.Layer] = true;
|
|
|
|
Result.Blocks[Block.Layer] = Block;
|
|
|
|
Result.BlocksCount++;
|
2020-03-08 00:06:10 +00:00
|
|
|
}
|
2020-10-10 05:08:51 +00:00
|
|
|
|
|
|
|
return Result;
|
2020-03-08 00:06:10 +00:00
|
|
|
}
|
2020-10-10 05:08:51 +00:00
|
|
|
|
2020-01-02 02:41:43 +00:00
|
|
|
#define FOLDHAUS_ANIMATION
|
2020-03-02 01:19:30 +00:00
|
|
|
#endif // FOLDHAUS_ANIMATION
|