// // File: blumen_lumen.cpp // Author: Peter Slattery // Creation Date: 2021-01-23 // #ifndef BLUMEN_LUMEN_CPP internal animation_handle_array LoadAllAnimationsInDir(gs_const_string Path, blumen_lumen_state* BLState, app_state* State, context Context) { animation_handle_array Result = {}; gs_thread_context Ctx = Context.ThreadContext; gs_file_info_array FilesInDir = EnumerateDirectory(Ctx.FileHandler, State->Transient, Path, 0); Result.Count = FilesInDir.Count; Result.Handles = PushArray(&State->Permanent, animation_handle, Result.Count); for (u32 i = 0; i < FilesInDir.Count; i++) { gs_file_info File = FilesInDir.Values[i]; Result.Handles[i] = AnimationSystem_LoadAnimationFromFile(&State->AnimationSystem, State->Patterns, Context, File.Path); } return Result; } internal s32 GetCCIndex (assembly Assembly, blumen_lumen_state* BLState) { s32 Result = 0; u64 AssemblyNameHash = HashDJB2ToU32(StringExpand(Assembly.Name)); for (u32 i = 0; i < BLState->AssemblyNameToClearCoreMapCount; i++) { if (AssemblyNameHash == BLState->AssemblyNameToClearCore_Names[i]) { Result = (s32)i; break; } } return Result; } internal void DEBUG_AppendText(gs_string Str, gs_thread_context Ctx) { gs_const_string DebugPath = ConstString("data/debug_motor_changes.txt"); gs_file DebugFile = ReadEntireFile(Ctx.FileHandler, DebugPath); gs_string NewString = PushString(Ctx.Transient, DebugFile.Size + Str.Size + 16); if (DebugFile.Size > 0) { PrintF(&NewString, "%.*s\nENTRY:\n", DebugFile.Size, (char*)DebugFile.Memory); } AppendPrintF(&NewString, "%S\n", Str.ConstString); NullTerminate(&NewString); if (!WriteEntireFile(Ctx.FileHandler, DebugPath, StringToData(NewString))) { InvalidCodePath; } } internal void DEBUG_SentMotorCommand(motor_packet Packet, gs_thread_context Ctx) { gs_string Str = PushStringF(Ctx.Transient, 256, "Motor Command Sent\nRequested Positions: %d %d %d\n", Packet.FlowerPositions[0], Packet.FlowerPositions[1], Packet.FlowerPositions[2]); DEBUG_AppendText(Str, Ctx); NullTerminate(&Str); OutputDebugStringA(Str.Str); } internal void DEBUG_ReceivedMotorPositions(motor_packet NewPos, motor_packet LastPos, gs_thread_context Ctx) { bool PosChanged = (LastPos.FlowerPositions[0] != NewPos.FlowerPositions[0] || LastPos.FlowerPositions[1] != NewPos.FlowerPositions[1] || LastPos.FlowerPositions[2] != NewPos.FlowerPositions[2]); if (PosChanged) { gs_string Str = PushStringF(Ctx.Transient, 256, "Motor Status Received\nCurrent Positions: %d %d %d\n", NewPos.FlowerPositions[0], NewPos.FlowerPositions[1], NewPos.FlowerPositions[2]); DEBUG_AppendText(Str, Ctx); NullTerminate(&Str); OutputDebugStringA(Str.Str); } } internal void DEBUG_ReceivedTemperature(temp_packet Temp, gs_thread_context Ctx) { gs_string TempStr = PushStringF(Ctx.Transient, 256, "\nTemperature: %d\n", Temp.Temperature); NullTerminate(&TempStr); OutputDebugStringA(TempStr.Str); } internal void BlumenLumen_MicListenJob(gs_thread_context* Ctx, u8* UserData) { mic_listen_job_data* Data = (mic_listen_job_data*)UserData; gs_data Msg = {}; u8 WeathermanIPAddr[4] = {}; WeathermanIPAddr[0] = 127; WeathermanIPAddr[1] = 0; WeathermanIPAddr[2] = 0; WeathermanIPAddr[3] = 1; u32 WeathermanIPV4 = (u32)UpackB4(WeathermanIPAddr); u32 WeathermanPort = 20185; platform_socket_handle_ ListenSocket = {0}; while (*Data->Running) { if (!SocketQueryStatus(Data->SocketManager, ListenSocket)) { Data->IsConnected = false; if (SocketHandleIsValid(ListenSocket)) { OutputDebugStringA("Disconnected from Python Server\n"); CloseSocket(Data->SocketManager, ListenSocket); } ListenSocket = CreateSocket(Data->SocketManager, "127.0.0.1", "20185"); if (ListenSocket.Index != 0) { OutputDebugStringA("Connected to Python Server\n"); Data->IsConnected = true; } } if (SocketQueryStatus(Data->SocketManager, ListenSocket)) { if (SocketPeek(Data->SocketManager, ListenSocket)) { Msg = SocketRecieve(Data->SocketManager, ListenSocket, Ctx->Transient); if (Msg.Size > 0) { MessageQueue_Write(Data->IncomingMsgQueue, Msg); } } while (MessageQueue_CanRead(*Data->OutgoingMsgQueue)) { Msg = MessageQueue_Peek(Data->OutgoingMsgQueue); u32 Address = WeathermanIPV4; u32 Port = WeathermanPort; s32 Flags = 0; s32 LengthSent = SocketSend(Data->SocketManager, ListenSocket, Address, Port, Msg, Flags); if (LengthSent != 0) { // if we actually sent the message, THEN we pull it off the // message queue MessageQueue_Read(Data->OutgoingMsgQueue); } else { break; } } } } CloseSocket(Data->SocketManager, ListenSocket); } internal void BlumenLumen_SetPatternMode(bl_pattern_mode Mode, r32 FadeDuration, animation_system* System, blumen_lumen_state* BLState) { BLState->PatternMode = Mode; animation_handle_array Playlist = BLState->ModeAnimations[Mode]; System->RepeatMode = AnimationRepeat_Loop; System->PlaylistFadeTime = FadeDuration; AnimationSystem_FadeToPlaylist(System, Playlist); } internal void BlumenLumen_LoadPatterns(app_state* State) { animation_pattern_array* Patterns = &State->Patterns; if (Patterns->CountMax == 0) { *Patterns = Patterns_Create(&State->Permanent, 32); } Patterns->Count = 0; Patterns_PushPattern(Patterns, Pattern_HueShift, PATTERN_MULTITHREADED); Patterns_PushPattern(Patterns, Pattern_Rainbow, PATTERN_MULTITHREADED); Patterns_PushPattern(Patterns, Pattern_BasicFlowers, PATTERN_MULTITHREADED); Patterns_PushPattern(Patterns, Pattern_Wavy, PATTERN_MULTITHREADED); Patterns_PushPattern(Patterns, Pattern_Patchy, PATTERN_MULTITHREADED); Patterns_PushPattern(Patterns, Pattern_Leafy, PATTERN_MULTITHREADED); Patterns_PushPattern(Patterns, Pattern_LeafyPatchy, PATTERN_MULTITHREADED); Patterns_PushPattern(Patterns, Pattern_WavyPatchy, PATTERN_SINGLETHREADED); Patterns_PushPattern(Patterns, Pattern_VerticalLines, PATTERN_MULTITHREADED); Patterns_PushPattern(Patterns, Pattern_Rotary, PATTERN_MULTITHREADED); Patterns_PushPattern(Patterns, Pattern_AllOnMask, PATTERN_MULTITHREADED); Patterns_PushPattern(Patterns, Pattern_BulbMask, PATTERN_MULTITHREADED); Patterns_PushPattern(Patterns, Pattern_VoicePattern, PATTERN_MULTITHREADED); } internal void AppendPrintDate(gs_string* WriteStr, system_time Time) { AppendPrintF(WriteStr, "%d-%d-%d : %d:%d:%d\n\n", Time.Year, Time.Month, Time.Day, Time.Hour, Time.Minute, Time.Second); } internal void BlumenLumen_AppendBootupLog(app_state* State, blumen_lumen_state* BLState, context Context) { gs_thread_context Ctx = Context.ThreadContext; gs_const_string BootupLogPath = ConstString("lumenarium_boot_log.log"); gs_file BootLog = ReadEntireFile(Ctx.FileHandler, BootupLogPath); gs_string WriteStr = {}; // we don't want the log file getting huge // if it gets above some threshold, instead of appending, // copy what there is to an _old file, and start this one over // // The thinking is that without the copy operation, when we reached // our threshold, we'd overwrite the whole log. If something went // wrong at that point, we'd have nothing to go on. This way, there is // always some historical data present on the system // if (BootLog.Size < MB(4)) { WriteStr = PushString(State->Transient, BootLog.Size + 1024); } else { if (!WriteEntireFile(Ctx.FileHandler, ConstString("lumenarium_boot_log_old.log"), BootLog.Data)) { InvalidCodePath; } WriteStr = PushString(State->Transient, 1024); } // Copy old entries in if (BootLog.Size > 0) { PrintF(&WriteStr, "%.*s", BootLog.Size, BootLog.Memory); } // New Entry AppendPrintF(&WriteStr, "Lumenarium Restarted\n"); AppendPrintF(&WriteStr, "* Time: "); AppendPrintDate(&WriteStr, Context.SystemTime_Current); gs_data Data = StringToData(WriteStr); WriteEntireFile(Ctx.FileHandler, BootupLogPath, Data); } internal void BlumenLumen_UpdateLog(app_state* State, blumen_lumen_state* BLState, context Context) { if (!BLState->ShouldUpdateLog) return; gs_string FileStr = PushString(State->Transient, 1024); AppendPrintF(&FileStr, "Lumenarium Status\n"); AppendPrintF(&FileStr, "Last Updated At:"); AppendPrintDate(&FileStr, Context.SystemTime_Current); AppendPrintF(&FileStr, "\n\n"); animation* CurrAnim = AnimationSystem_GetActiveAnimation(&State->AnimationSystem); AppendPrintF(&FileStr, "Curr Animation: %S\n", CurrAnim->Name); char* Connected = BLState->MicListenJobData.IsConnected ? "Connected" : "Disconnected"; AppendPrintF(&FileStr, "Connected to Python: %s\n", Connected); u8 MP0 = BLState->LastKnownMotorState.FlowerPositions[0]; u8 MP1 = BLState->LastKnownMotorState.FlowerPositions[1]; u8 MP2 = BLState->LastKnownMotorState.FlowerPositions[2]; AppendPrintF(&FileStr, "Last Known Motor State: %d %d %d\n", MP0, MP1, MP2); char* PatternMode = 0; switch (BLState->PatternMode) { case BlumenPattern_Standard: { PatternMode = "Standard"; } break; case BlumenPattern_VoiceCommand: { PatternMode = "Voice Command"; } break; } AppendPrintF(&FileStr, "Pattern Mode: %s\n", PatternMode); phrase_hue LastHuePhrase = BLState->LastHuePhrase; AppendPrintF(&FileStr, "Last Mic Phrase: %S\n", LastHuePhrase.Phrase); AppendPrintF(&FileStr, "Pattern Speed: %f\n", BLState->PatternSpeed); AppendPrintF(&FileStr, "Pattern Brightness: %f\n", BLState->BrightnessPercent); AppendPrintF(&FileStr, "Last Temp Received: %d\n", BLState->LastTemperatureReceived); gs_data LogMem = StringToData(FileStr); if (!WriteEntireFile(Context.ThreadContext.FileHandler, ConstString("lumenarium_status.log"), LogMem)) { InvalidCodePath; } } internal gs_data BlumenLumen_CustomInit(app_state* State, context Context) { // This is memory for any custom data that we want to use // as a part of a particular sculpture. // By returning it from here, it will be sent as an argument to // the sculpture's CustomUpdate function; gs_data Result = {}; Result = PushSizeToData(&State->Permanent, sizeof(blumen_lumen_state)); blumen_lumen_state* BLState = (blumen_lumen_state*)Result.Memory; BLState->Running = true; BLState->BrightnessPercent = 1; MessageQueue_Init(&BLState->IncomingMsgQueue, &State->Permanent); MessageQueue_Init(&BLState->OutgoingMsgQueue, &State->Permanent); BLState->MicListenJobData.Running = &BLState->Running; BLState->MicListenJobData.SocketManager = Context.SocketManager; BLState->MicListenJobData.IncomingMsgQueue = &BLState->IncomingMsgQueue; BLState->MicListenJobData.OutgoingMsgQueue = &BLState->OutgoingMsgQueue; BLState->PatternSpeed = GlobalAnimSpeed; #if 1 BLState->MicListenThread = CreateThread(Context.ThreadManager, BlumenLumen_MicListenJob, (u8*)&BLState->MicListenJobData, Context.ThreadContext); #endif assembly* Flower0 = LoadAssembly(Flower0AssemblyPath, State, Context); assembly* Flower1 = LoadAssembly(Flower1AssemblyPath, State, Context); assembly* Flower2 = LoadAssembly(Flower2AssemblyPath, State, Context); BLState->AssemblyNameToClearCoreMapCount = 3; BLState->AssemblyNameToClearCore_Names = PushArray(&State->Permanent, u64, BLState->AssemblyNameToClearCoreMapCount); BLState->AssemblyNameToClearCore_Names[0] = HashDJB2ToU32(StringExpand(Flower2->Name)); BLState->AssemblyNameToClearCore_Names[1] = HashDJB2ToU32(StringExpand(Flower1->Name)); BLState->AssemblyNameToClearCore_Names[2] = HashDJB2ToU32(StringExpand(Flower0->Name)); gs_file_handler FileHandler = Context.ThreadContext.FileHandler; gs_file ColorPhraseCSVFile = ReadEntireFile(FileHandler, PhraseMapCSVPath); gs_const_string ColorPhraseMapStr = DataToString(ColorPhraseCSVFile.Data); gscsv_sheet ColorPhraseSheet = CSV_Parse(ColorPhraseMapStr, { PhraseMapCSVSeparator }, State->Transient); BLState->PhraseHueMap = PhraseHueMap_GenFromCSV(ColorPhraseSheet, &State->Permanent); #if 0 animation_handle DemoPatternsAnim = AnimationSystem_LoadAnimationFromFile(&State->AnimationSystem, State->Patterns, Context, ConstString("data/demo_patterns.foldanim")); State->AnimationSystem.ActiveFadeGroup.From = DemoPatternsAnim; #else BLState->ModeAnimations[BlumenPattern_Standard] = LoadAllAnimationsInDir(AmbientPatternFolder, BLState, State, Context); BLState->ModeAnimations[BlumenPattern_VoiceCommand] = LoadAllAnimationsInDir(VoicePatternFolder, BLState, State, Context); AnimationSystem_LoadAnimationFromFile(&State->AnimationSystem, State->Patterns, Context, ConstString("data/blumen_animations/anim_demo.foldanim")); BlumenLumen_SetPatternMode(BlumenPattern_VoiceCommand, 5, &State->AnimationSystem, BLState); #endif State->AnimationSystem.TimelineShouldAdvance = true; BlumenLumen_AppendBootupLog(State, BLState, Context); return Result; } internal void BlumenLumen_CustomUpdate(gs_data UserData, app_state* State, context* Context) { blumen_lumen_state* BLState = (blumen_lumen_state*)UserData.Memory; BLState->ShouldUpdateLog = false; bool SendMotorCommand = false; blumen_packet MotorCommand = {}; while (MessageQueue_CanRead(BLState->IncomingMsgQueue)) { gs_data PacketData = MessageQueue_Read(&BLState->IncomingMsgQueue); blumen_packet Packet = *(blumen_packet*)PacketData.Memory; switch (Packet.Type) { case PacketType_PatternCommand: { microphone_packet Mic = Packet.MicPacket; u64 NameHash = HashDJB2ToU32(Mic.AnimationFileName); u32 NameLen = CStringLength(Mic.AnimationFileName); phrase_hue NewHue = PhraseHueMap_Get(BLState->PhraseHueMap, NameHash); if (BLState->NextHotHue.Phrase.Length < NewHue.Phrase.Length) { BLState->NextHotHue = NewHue; if (SecondsElapsed(BLState->TimePhraseReceptionBegan, Context->SystemTime_Current) > PhrasePriorityMessageGroupingTime) { BLState->TimePhraseReceptionBegan = Context->SystemTime_Current; BLState->InPhraseReceptionMode = true; } BLState->TimeLastPhraseReceived = Context->SystemTime_Current; } }break; case PacketType_MotorState: { motor_status_packet Motor = Packet.MotorStatusPacket; // NOTE(pjs): Python sends multi-byte integers in little endian // order. Have to unpack u8* T = (u8*)&Motor.Temperature; Motor.Temperature = (T[0] << 8 | T[1] << 0); motor_packet CurrPos = Motor.Pos; motor_packet LastPos = BLState->LastKnownMotorState; DEBUG_ReceivedMotorPositions(LastPos, Motor.Pos, Context->ThreadContext); for (u32 i = 0; i < BL_FLOWER_COUNT; i++) { if (LastPos.FlowerPositions[i] != CurrPos.FlowerPositions[i]) { BLState->LastTimeMotorStateChanged[i] = Context->SystemTime_Current.NanosSinceEpoch; } } BLState->LastKnownMotorState = Motor.Pos; BLState->ShouldUpdateLog = true; }break; case PacketType_Temperature: { temp_packet Temp = Packet.TempPacket; if (Temp.Temperature > MinHighTemperature) { BLState->BrightnessPercent = HighTemperatureBrightnessPercent; } else { BLState->BrightnessPercent = FullBrightnessPercent; } BLState->LastTemperatureReceived = Temp.Temperature; DEBUG_ReceivedTemperature(Temp, Context->ThreadContext); BLState->ShouldUpdateLog = true; }break; InvalidDefaultCase; } } if (BLState->InPhraseReceptionMode) { r32 SecondsSincePhraseBegan = SecondsElapsed(BLState->TimePhraseReceptionBegan, Context->SystemTime_Current); if (SecondsSincePhraseBegan > PhrasePriorityMessageGroupingTime) { // if we are in standard color mode, shift all flowers to the new color // otherwise, only shift the next flower in the sequence to the new color phrase_hue NewHue = BLState->NextHotHue; if (BLState->PatternMode == BlumenPattern_Standard) { BLState->AssemblyColors[0] = NewHue; BLState->AssemblyColors[1] = NewHue; BLState->AssemblyColors[2] = NewHue; } else { u32 AssemblyIdx = BLState->LastAssemblyColorSet; BLState->AssemblyColors[AssemblyIdx] = NewHue; BLState->LastAssemblyColorSet = (BLState->LastAssemblyColorSet + 1) % 3; } BlumenLumen_SetPatternMode(BlumenPattern_VoiceCommand, 5, &State->AnimationSystem, BLState); BLState->TimeLastSetToVoiceMode = Context->SystemTime_Current; BLState->LastHuePhrase = NewHue; BLState->ShouldUpdateLog = true; BLState->InPhraseReceptionMode = false; } } // Update next frames Hues r32 AnimTime = AnimationSystem_GetCurrentTime(State->AnimationSystem); AnimTime = (r32)Context->TotalTime; r32 BaseTime = AnimTime * BLState->PatternSpeed; r32 ColorSpeed = 1; //.001; r32 ColorOscSpeed = .05 * ColorSpeed; r32 ColorRelOscSpeed = 1 * ColorSpeed;; r32 ColorOscillation = (SinR32(BaseTime * ColorOscSpeed) + 1) / 2; r32 ColorRelationship = 30 + (((1 + SinR32(BaseTime * ColorRelOscSpeed)) / 2) * 300); BLState->StandardPatternHues.Hue0 = ModR32(ColorOscillation * 360, 360); BLState->StandardPatternHues.Hue1 = ModR32(BaseTime + ColorRelationship, 360); BLState->StandardPatternHues.Hue2 = LerpR32(.3f, BLState->StandardPatternHues.Hue0, BLState->StandardPatternHues.Hue1); // Transition back to standard mode after some time if (BLState->PatternMode == BlumenPattern_VoiceCommand) { u64 LastChangeClock = BLState->TimeLastSetToVoiceMode.NanosSinceEpoch; u64 NowClocks = Context->SystemTime_Current.NanosSinceEpoch; s64 NanosSinceChange = NowClocks - LastChangeClock; r64 SecondsSinceChange = (r64)NanosSinceChange * NanosToSeconds; if (SecondsSinceChange > VoiceCommandSustainDuration) { BLState->PatternMode = BlumenPattern_Standard; animation_handle NewAnim = BLState->ModeAnimations[BlumenPattern_Standard].Handles[0]; AnimationFadeGroup_FadeTo(&State->AnimationSystem.ActiveFadeGroup, NewAnim, VoiceCommandFadeDuration); BLState->ShouldUpdateLog = true; } } // Open / Close the Motor if (MessageQueue_CanWrite(BLState->OutgoingMsgQueue)) { for (u32 i = 0; i < MotorOpenTimesCount; i++) { time_range Range = MotorOpenTimes[i]; bool CurrTimeInRange = SystemTimeIsInTimeRange(Context->SystemTime_Current, Range); bool LastSendTimeInRange = SystemTimeIsInTimeRange(BLState->LastSendTime, Range); bool LastTimeInRange = SystemTimeIsInTimeRange(Context->SystemTime_Last, Range); bool SendOpen = CurrTimeInRange && !LastSendTimeInRange; bool SendClose = !CurrTimeInRange && LastSendTimeInRange; //SendOpen = SecondsSinceLastSend > 2; if (SendOpen) { SendMotorCommand = true; BLState->LastSendTime = Context->SystemTime_Current; OutputDebugString("Motors: Open\n"); blumen_packet Packet = {}; Packet.Type = PacketType_MotorState; Packet.MotorPacket.FlowerPositions[0] = 2; Packet.MotorPacket.FlowerPositions[1] = 2; Packet.MotorPacket.FlowerPositions[2] = 2; MotorCommand = Packet; } else if (SendClose) { SendMotorCommand = true; BLState->LastSendTime = Context->SystemTime_Current; OutputDebugString("Motors: Close\n"); blumen_packet Packet = {}; Packet.Type = PacketType_MotorState; Packet.MotorPacket.FlowerPositions[0] = 1; Packet.MotorPacket.FlowerPositions[1] = 1; Packet.MotorPacket.FlowerPositions[2] = 1; MotorCommand = Packet; } } if (SendMotorCommand) { gs_data Msg = StructToData(&MotorCommand, blumen_packet); MessageQueue_Write(&BLState->OutgoingMsgQueue, Msg); DEBUG_SentMotorCommand(MotorCommand.MotorPacket, Context->ThreadContext); } } // When a motor state changes to being open, wait to turn Upper Leds on // in order to hide the fact that they are turning off motor_packet CurrMotorPos = BLState->LastKnownMotorState; u64 NowNanos = Context->SystemTime_Current.NanosSinceEpoch; for (u32 i = 0; i < BL_FLOWER_COUNT; i++) { // have to map from "assembly load order" to // the order that the clear core is referencing the // motors by assembly Assembly = State->Assemblies.Values[i]; u64 AssemblyCCIndex = GetCCIndex(Assembly, BLState); u8 MotorPos = CurrMotorPos.FlowerPositions[AssemblyCCIndex]; if ((MotorPos == MotorState_Open || MotorPos == MotorState_MostlyOpen) && !BLState->ShouldDimUpperLeds[i]) { u64 ChangedNanos = BLState->LastTimeMotorStateChanged[i]; u64 NanosSinceChanged = NowNanos - ChangedNanos; r64 SecondsSinceChanged = (r64)NanosSinceChanged * NanosToSeconds; if (SecondsSinceChanged > TurnUpperLedsOffAfterMotorCloseCommandDelay) { BLState->ShouldDimUpperLeds[i] = true; } } } // NOTE(PS): If the flowers are mostly open or full open // we mask off the top leds to prevent them from overheating // while telescoped inside the flower for (u32 a = 0; a < BL_FLOWER_COUNT; a++) { assembly Assembly = State->Assemblies.Values[a]; if (!BLState->ShouldDimUpperLeds[a]) continue; led_buffer Buffer = State->LedSystem.Buffers[Assembly.LedBufferIndex]; led_strip_list TopStrips = AssemblyStripsGetWithTagValue(Assembly, ConstString("section"), ConstString("inner_bloom"), State->Transient); for (u32 s = 0; s < TopStrips.Count; s++) { u32 SIndex = TopStrips.StripIndices[s]; v2_strip Strip = Assembly.Strips[SIndex]; for (u32 l = 0; l < Strip.LedCount; l++) { u32 LIndex = Strip.LedLUT[l]; Buffer.Colors[LIndex] = {0}; } } } // Dim the leds based on temp data if (!BLState->DEBUG_IgnoreWeatherDimmingLeds) { for (u32 i = 0; i < State->LedSystem.BuffersCount; i++) { led_buffer Buffer = State->LedSystem.Buffers[i]; for (u32 j = 0; j < Buffer.LedCount; j++) { pixel* Color = Buffer.Colors + j; Color->R = Color->R * BLState->BrightnessPercent; Color->G = Color->G * BLState->BrightnessPercent; Color->B = Color->B * BLState->BrightnessPercent; } } } // Send Status Packet { system_time LastSendTime = BLState->LastStatusUpdateTime; r64 NanosSinceLastSend = ((r64)Context->SystemTime_Current.NanosSinceEpoch - (r64)LastSendTime.NanosSinceEpoch); r64 SecondsSinceLastSend = NanosSinceLastSend * NanosToSeconds; if (SecondsSinceLastSend >= STATUS_PACKET_FREQ_SECONDS) { BLState->LastStatusUpdateTime = Context->SystemTime_Current; OutputDebugString("Attempting to Send Lumenarium Status\n"); blumen_packet Packet = {}; Packet.Type = PacketType_LumenariumStatus; Packet.StatusPacket.NextMotorEventType = 0; Packet.StatusPacket.NextEventTime = 0; animation* ActiveAnim = AnimationSystem_GetActiveAnimation(&State->AnimationSystem); if (ActiveAnim) { CopyMemoryTo(ActiveAnim->Name.Str, Packet.StatusPacket.AnimFileName, Min(ActiveAnim->Name.Length, 32)); Packet.StatusPacket.AnimFileName[ActiveAnim->Name.Length] = 0; } gs_data Msg = StructToData(&Packet, blumen_packet); MessageQueue_Write(&BLState->OutgoingMsgQueue, Msg); // there's no new information here, but by updating the log here, // we're updating it at some infrequent but regular period that isnt // every single frame BLState->ShouldUpdateLog = true; } } BlumenLumen_UpdateLog(State, BLState, *Context); } US_CUSTOM_DEBUG_UI(BlumenLumen_DebugUI) { blumen_lumen_state* BLState = (blumen_lumen_state*)UserData.Memory; ui_interface* I = &State->Interface; // NOTE(PS): Motors { motor_packet PendingPacket = BLState->DEBUG_PendingMotorPacket; for (u32 MotorIndex = 0; MotorIndex < BL_FLOWER_COUNT; MotorIndex++) { gs_string Label = PushStringF(State->Transient, 32, "Motor %d", MotorIndex); ui_BeginRow(I, 5); { ui_Label(I, Label); bool IsClosed = PendingPacket.FlowerPositions[MotorIndex] == 1; if (ui_ToggleText(I, MakeString("Closed (1)"), IsClosed)) { PendingPacket.FlowerPositions[MotorIndex] = 1; } bool IsHOpen = PendingPacket.FlowerPositions[MotorIndex] == 3; if (ui_ToggleText(I, MakeString("Half Open (3)"), IsHOpen)) { PendingPacket.FlowerPositions[MotorIndex] = 3; } bool IsMOpen = PendingPacket.FlowerPositions[MotorIndex] == 4; if (ui_ToggleText(I, MakeString("Mostly Open (4)"), IsMOpen)) { PendingPacket.FlowerPositions[MotorIndex] = 4; } bool IsOpen = PendingPacket.FlowerPositions[MotorIndex] == 2; if (ui_ToggleText(I, MakeString("Open (2)"), IsOpen)) { PendingPacket.FlowerPositions[MotorIndex] = 2; } } ui_EndRow(I); } BLState->DEBUG_PendingMotorPacket = PendingPacket; if (ui_Button(I, MakeString("Send Motor Packet"))) { blumen_packet Packet = {}; Packet.Type = PacketType_MotorState; Packet.MotorPacket = BLState->DEBUG_PendingMotorPacket; gs_data Msg = StructToData(&Packet, blumen_packet); MessageQueue_Write(&BLState->OutgoingMsgQueue, Msg); DEBUG_SentMotorCommand(Packet.MotorPacket, Context.ThreadContext); } motor_packet MotorPos = BLState->LastKnownMotorState; ui_Label(I, MakeString("Current Motor Positions")); { for (u32 i = 0; i < BL_FLOWER_COUNT; i++) { ui_BeginRow(I, 2); gs_string MotorStr = PushStringF(State->Transient, 32, "Motor %d", i); ui_Label(I, MotorStr); gs_string StateStr = {}; switch (MotorPos.FlowerPositions[i]) { case MotorState_Closed: { StateStr = MakeString("Closed"); } break; case MotorState_HalfOpen: { StateStr = MakeString("Half Open"); } break; case MotorState_MostlyOpen: { StateStr = MakeString("Mostly Open"); } break; case MotorState_Open: { StateStr = MakeString("Open"); } break; } ui_Label(I, StateStr); ui_EndRow(I); } } BLState->DEBUG_IgnoreWeatherDimmingLeds = ui_LabeledToggle(I, MakeString("Ignore Weather Dimming Leds"), BLState->DEBUG_IgnoreWeatherDimmingLeds); } } US_CUSTOM_CLEANUP(BlumenLumen_CustomCleanup) { blumen_lumen_state* BLState = (blumen_lumen_state*)UserData.Memory; BLState->Running = false; } internal user_space_desc BlumenLumen_UserSpaceCreate() { user_space_desc Result = {}; Result.LoadPatterns = BlumenLumen_LoadPatterns; Result.CustomInit = BlumenLumen_CustomInit; Result.CustomUpdate = BlumenLumen_CustomUpdate; Result.CustomDebugUI = BlumenLumen_DebugUI; Result.CustomCleanup = BlumenLumen_CustomCleanup; return Result; } #define BLUMEN_LUMEN_CPP #endif // BLUMEN_LUMEN_CPP