Files
integration/Source/Integration/LuprexGameModeBase.cpp

259 lines
7.3 KiB
C++
Raw Normal View History

2023-06-08 17:10:14 -04:00
// Copyright Epic Games, Inc. All Rights Reserved.
#include "LuprexGameModeBase.h"
#include "PlayerControllerBase.h"
#include "LockedWrapper.h"
#include "lpx-drvutil.hpp"
#include "Misc/Paths.h"
#include "Tangible.h"
2023-09-02 01:33:11 -04:00
#include "TangibleManager.h"
#include "LuaCall.h"
#include "Blueprint/UserWidget.h"
#include "Blueprint/WidgetBlueprintLibrary.h"
2025-04-07 16:48:27 -04:00
#include "Kismet/GameplayStatics.h"
2026-02-27 13:31:06 -05:00
#include "Engine/GameInstance.h"
2026-02-09 13:54:00 -05:00
#include "Common.h"
#include "AnimQueue.h"
2023-06-08 17:10:14 -04:00
#include <string>
#include <string_view>
2026-02-09 13:54:00 -05:00
using namespace LpxCommonTypes;
2026-02-27 17:49:29 -05:00
ALuprexGameModeBase::ALuprexGameModeBase() {}
ALuprexGameModeBase::~ALuprexGameModeBase() {}
2023-06-08 17:10:14 -04:00
void ALuprexGameModeBase::ResetToInitialState()
2023-06-09 16:47:46 -04:00
{
Playing = false;
TickEnabled = true;
2026-02-27 17:49:29 -05:00
// Stop the tick function.
FWorldDelegates::OnWorldPostActorTick.Remove(OnWorldPostActorTickHandle);
OnWorldPostActorTickHandle.Reset();
FlxLockedWrapper w;
2023-09-03 03:40:44 -04:00
2023-07-03 15:28:14 -04:00
// Release and close all sockets.
2023-09-03 02:01:32 -04:00
if (Sockets != nullptr)
{
Sockets->ForceCloseEverything(w);
Sockets.Reset();
}
2023-07-03 15:28:14 -04:00
2023-06-23 16:27:23 -04:00
// Delete the engine.
2023-09-03 02:01:32 -04:00
if (w->release != nullptr)
2023-06-23 16:27:23 -04:00
{
2023-09-03 02:01:32 -04:00
w->release(w.Get());
2023-06-23 16:27:23 -04:00
}
// Clear the lua call assembly buffer.
UlxEngineWrapper::GetLuaCallBuffer().clear();
// Stop trapping log errors to the debugger.
BreakToDebuggerLogVerbosityDevice.Reset();
// Clear the PlayerID
PlayerId = 0;
2023-06-23 16:27:23 -04:00
// Reset the clocks.
EngineSeconds = 0.0;
2023-06-23 16:27:23 -04:00
}
void ALuprexGameModeBase::UpdateConsoleOutput() {
// Copy Luprex Stdout into the console.
FlxLockedWrapper lockedwrap;
if (Playing) {
FString Text = lockedwrap.ChannelPrints();
2025-12-09 02:42:13 -05:00
if (!Text.IsEmpty())
{
ConsoleAddOutput(Text);
}
2023-06-08 17:10:14 -04:00
}
}
#pragma optimize("", off)
void ALuprexGameModeBase::UpdateTangibles() {
double radius = 1000.0; // Hardwired for now.
2023-09-26 19:26:09 -04:00
using TanArray = UlxTangibleManager::TanArray;
if (!Playing) return;
UlxTangibleManager *TM = GetGameInstance()->GetSubsystem<UlxTangibleManager>();
TanArray alltans;
{
FlxLockedWrapper w;
PlayerId = w.GetActor();
IdView nearids = w.GetNear(PlayerId, radius, radius, radius);
TM->UpdateNearAccordingToLuprex(nearids);
alltans = TM->GetAllTangibles();
IdArray allids = TM->GetIds(alltans);
StringViewVec allqueues = w.GetAnimationQueues(allids);
for (int i = 0; i < alltans.Num(); i++) {
alltans[i]->UpdateAnimationQueue(allqueues[i]);
}
// Debugging hook.
volatile bool printaqs = false;
if (printaqs) {
for (int i = 0; i < alltans.Num(); i++) {
UE_LOG(LogLuprex, Display, TEXT("\n--- AQ of %ld ---\n%s\n"), allids[i], *(alltans[i]->AnimTracker.DebugString()));
}
}
}
// This is where we run the blueprint code that updates animation
// states. Be aware that the blueprint code could call back
// into the gamemode, which could in turn access the FlxLockedWrapper.
// This is why we've released the FlxLockedWrapper before doing this.
2023-09-26 19:26:09 -04:00
for (int i = 0; i < alltans.Num(); i++) {
alltans[i]->MaybeExecuteAnimStateChanged();
}
TM->RecalcNearAccordingToUnreal(PlayerId, radius);
TM->DeleteFarawayTangibles();
}
void ALuprexGameModeBase::UpdatePossessedTangible() {
UlxTangibleManager *TM = GetGameInstance()->GetSubsystem<UlxTangibleManager>();
UlxTangible *ptan = TM->GetPossessedTangible();
UlxTangible *tan = TM->GetTangible(PlayerId);
2024-09-24 22:13:56 -04:00
APlayerController *ctrl = GetWorld()->GetFirstPlayerController();
APawn *pawn = nullptr;
if (tan != nullptr) pawn = Cast<APawn>(tan->GetActor());
if (pawn == nullptr) {
if (ptan != nullptr) {
TM->SetPossessedTangible(nullptr);
2026-03-02 17:17:23 -05:00
ctrl->UnPossess();
}
2024-09-24 22:13:56 -04:00
} else {
if (ptan != tan) {
TM->SetPossessedTangible(tan);
2024-09-24 22:13:56 -04:00
ctrl->Possess(pawn);
}
}
}
2026-02-27 17:49:29 -05:00
void ALuprexGameModeBase::UpdateLuaSourceCode() {
FlxLockedWrapper lockedwrap;
if (lockedwrap->get_rescan_lua_source(lockedwrap.Get()))
{
drvutil::ostringstream srcpak;
FString LuprexRoot = FPaths::Combine(FPaths::ProjectDir(), TEXT("luprex"));
std::string srcpakerr = drvutil::package_lua_source(TCHAR_TO_UTF8(*LuprexRoot), &srcpak);
if (!srcpakerr.empty())
{
FString FMessage((const UTF8CHAR *)(srcpakerr.c_str()));
UE_LOG(LogLuprexIntegration, Error, TEXT("Trying to read Lua source: %s"), *FMessage);
}
else
{
std::string_view srcpakv = srcpak.view();
lockedwrap->play_access(lockedwrap.Get(), AccessKind::INVOKE_LUA_SOURCE, 0, srcpakv.size(), srcpakv.data(), nullptr, nullptr);
}
}
}
void ALuprexGameModeBase::UpdateSocketsAndLua()
{
FlxLockedWrapper lockedwrap;
Sockets->Update(lockedwrap);
lockedwrap->play_update(lockedwrap.Get(), EngineSeconds);
Sockets->Update(lockedwrap);
}
2026-02-27 17:49:29 -05:00
void ALuprexGameModeBase::OnWorldPostActorTick(UWorld* InWorld, ELevelTick InLevelTick, float deltaseconds)
{
if(Playing && TickEnabled && (GetWorld() == InWorld) && (InLevelTick == LEVELTICK_All))
{
EngineSeconds += deltaseconds;
2026-02-27 17:49:29 -05:00
UpdateLuaSourceCode();
UpdateSocketsAndLua();
2024-02-16 15:48:22 -05:00
UpdateConsoleOutput();
UpdateTangibles();
UpdatePossessedTangible();
AlxPlayerControllerBase *PC = Cast<AlxPlayerControllerBase>(GetWorld()->GetFirstPlayerController());
if (PC != nullptr) PC->UpdateLookAt();
}
}
void ALuprexGameModeBase::BeginPlay()
2023-06-09 16:47:46 -04:00
{
2023-09-03 03:40:44 -04:00
ResetToInitialState();
InitializeGlobalState();
Super::BeginPlay();
}
2023-09-03 03:40:44 -04:00
void ALuprexGameModeBase::InitializeGlobalState()
{
FlxLockedWrapper w;
2023-09-03 02:01:32 -04:00
2023-09-03 03:40:44 -04:00
// Sanity checks. Make sure everything is clean.
2023-09-03 02:01:32 -04:00
checkf(w->engine == nullptr, TEXT("There should be no engine here."));
2023-06-23 12:45:23 -04:00
2023-06-23 16:27:23 -04:00
// If we failed to initialize the wrapper, print an error message.
2023-09-03 02:01:32 -04:00
if (w->play_initialize == nullptr)
2023-06-23 16:27:23 -04:00
{
UE_LOG(LogLuprexIntegration, Error, TEXT("Luprex wrapper initialization failed"));
2023-06-08 17:10:14 -04:00
}
2023-06-23 12:45:23 -04:00
// If wrapper is initialized, try to initialize the luprex engine.
2023-09-03 02:01:32 -04:00
if (w->play_initialize != nullptr)
2023-06-23 12:45:23 -04:00
{
w->play_initialize(w.Get(), "lpxclient", "");
if (w->error[0])
2023-06-23 12:45:23 -04:00
{
FString FMessage((const UTF8CHAR *)w->error);
UE_LOG(LogLuprexIntegration, Error, TEXT("Calling Luprex play_initialize: %s"), *FMessage);
2023-06-23 12:45:23 -04:00
}
if (w->engine != nullptr) {
UE_LOG(LogLuprexIntegration, Verbose, TEXT("Luprex initialization success."));
Playing = true;
2023-06-23 12:45:23 -04:00
}
2023-06-08 17:10:14 -04:00
}
2023-06-23 12:45:23 -04:00
// Possibly tell the engine to connect to a server.
if (Playing) {
FString LuprexServer;
FParse::Value(FCommandLine::Get(), TEXT("-LuprexServer="), LuprexServer);
LuprexServer = LuprexServer.ToLower();
UE_LOG(LogTemp, Display, TEXT("LuprexServer = %s"), *LuprexServer)
if (LuprexServer != TEXT("standalone"))
{
FTCHARToUTF8 utf8server(LuprexServer);
w->play_access(w.Get(), AccessKind::CONNECT_TO_SERVER, 0, utf8server.Length(), utf8server.Get(), nullptr, nullptr);
}
}
2026-02-27 17:49:29 -05:00
// If we successfully created a luprex engine, create a socket system.
if (Playing) {
2023-09-15 13:28:18 -04:00
Sockets.Reset(FlxSockets::Create(w));
2023-07-03 15:28:14 -04:00
std::string error = Sockets->GetError();
check(error.empty());
2023-06-08 17:10:14 -04:00
}
2023-08-29 20:05:10 -04:00
if (Playing) {
OnWorldPostActorTickHandle = FWorldDelegates::OnWorldPostActorTick.AddUObject(this, &ALuprexGameModeBase::OnWorldPostActorTick);
}
// If somebody generates a log message that's severe enough, break to debugger.
BreakToDebuggerLogVerbosityDevice.Reset(
new FlxBreakToDebuggerOutputDevice(BreakToDebuggerLogVerbosity));
2023-06-08 17:10:14 -04:00
}
void ALuprexGameModeBase::EndPlay(const EEndPlayReason::Type EndPlayReason)
2023-06-23 12:45:23 -04:00
{
2023-06-23 16:27:23 -04:00
ResetToInitialState();
2025-09-09 17:14:14 -04:00
Super::EndPlay(EndPlayReason);
2023-06-09 16:47:46 -04:00
}
int64 ALuprexGameModeBase::GetPlayerId() {
return PlayerId;
}
2024-08-31 16:42:07 -04:00
ALuprexGameModeBase *ALuprexGameModeBase::FromContext(const UObject *context) {
ALuprexGameModeBase *result = context->GetWorld()->GetAuthGameMode<ALuprexGameModeBase>();
2024-08-31 16:42:07 -04:00
if (result == nullptr) {
UE_LOG(LogBlueprint, Fatal, TEXT("Not currently using a Luprex Game Mode."));
2024-08-31 16:42:07 -04:00
}
return result;
}