Unreal 5.x
As of the latest version, yes!
First and foremost, make sure you've saved all your levels. If you're playing in the editor (PIE) and your levels aren't saved, the data UE reports to SPUD isn't reliable and can cause problems.
Either:
- Always use "File > Save All Levels" before playing in the editor, or
- Enable auto-saving of all levels before PIE in "Project Settings > Plugins > SPUD"
The Output Log will report an error about this if it detects you're running with unsaved levels.
I have no idea why this happens, but it seems to be pre-4.26 only. Playing in Standalone mode from the editor randomly resets some properties sometimes and I haven't been able to figure out why. It's only playing Standalone from the editor:
- PIE mode works fine (if you save all levels, see above).
- Packaged games work fine too
Having discussed this with others who have experienced other weird problems specifically with Standalone, we've concluded that Standalone mode is just "bad". However in 4.26 everything seemed to start working in Standalone so YMMV.
To be safe, use PIE, or package the game.
This seems to be caused by a weird UE bug: if you have Widgets open in the Editor, screenshot requests will always time out, resulting in a blank screenshot. Close all your Widget blueprints in the editor to resolve this.
Correct, they won't. Unreal Engine's physics are non-deterministic, which means you won't necessarily get the exact same results every time, even if objects positions and velocities are restored correctly.
This is not a SPUD bug, it's just the way UE's physics are. You'd get the same result if you loaded your level and applied the same forces, each time things would play out slightly differently.
If it's a subclass of Pawns, GameModes, GameStates, PlayerStates and Characters then
SPUD will automatically figure out it doesn't have to respawn them. For anything
else, implement the GetSpudRespawnMode
method on ISpudObject
to change
respawn behaviour.
On spawning or loading a game, assuming your player pawn implements ISpudObject it will be moved to the correct location. However, level streaming can take a little time so the character could fall through the level until that's done.
One way to fix this is to put some sort of collider in the Persistent Level (which is always loaded) to catch things until the level is loaded, but that's only really practical for simple worlds.
A better way is to disable your character's movement and gravity until it can tell that the ground has appeared underneath it. This is what the example character does.
Firstly, on BeginPlay, and on TeleportTo and post-restore (by implementing ISpudObjectCallback)
we call BeginWaitingForStreaming
. This means whenever there's a chance the character
has moved quickly into a new area, we wait for the ground to appear:
void ASPUDExamplesCharacter::BeginWaitingForStreaming()
{
// Just after we've been spawned, or just after teleport, the level around us might not be streamed in yet
bIsWaitingForStreaming = true;
if (UCharacterMovementComponent* MoveComp = Cast<UCharacterMovementComponent>(GetMovementComponent()))
{
MoveComp->GravityScale = 0;
}
}
We also override IsMoveInputIgnored
to make sure the character isn't allowed to
move.
bool ASPUDExamplesCharacter::IsMoveInputIgnored() const
{
// Don't allow movement until streaming is ready
return Super::IsMoveInputIgnored() || bIsWaitingForStreaming;
}
Then, on Tick we check whether things are OK yet, by doing a sphere trace downwards. As soon as a static world is seen underneath us, we know it's ok to enable gravity and movement again.
void ASPUDExamplesCharacter::Tick(float DeltaSeconds)
{
Super::Tick(DeltaSeconds);
if (bIsWaitingForStreaming)
CheckStreamingOK();
}
void ASPUDExamplesCharacter::CheckStreamingOK()
{
FHitResult OutHit;
TArray<AActor*> ToIgnore;
if (UKismetSystemLibrary::SphereTraceSingle(GetWorld(),
GetActorLocation(), GetActorLocation() + FVector::DownVector * 200,
30, UEngineTypes::ConvertToTraceType(ECC_WorldStatic),
false, ToIgnore, EDrawDebugTrace::None, OutHit, true))
{
bIsWaitingForStreaming = false;
if (UCharacterMovementComponent* MoveComp = Cast<UCharacterMovementComponent>(GetMovementComponent()))
{
MoveComp->GravityScale = 1;
}
}
}
So that's one way to make your characters streaming / teleporting / loading friendly! If your characters move differently or it's possible you could save them above a large or even infinite drop, then you might need to tweak this a bit.
Persistent object cross-references are supported as saved state. However, they can only reference objects in the same level, or in the Persistent level, because you cannot guarantee which levels are loaded at once.
If objects needs to cross streaming level boundaries then you might consider putting them in the Persistent Level instead of the streamed level. However, be careful about making them respond to gravity if there's a chance they can be in an area that unloads.
See Gameplay Framework Spawned Actors
The most common reason for this is that you're spawning new actors during
one of the per-actor callbacks in ISpudObjectCallback
. This isn't allowed, because
we're currently going through the level actors to restore them. While SPUD could
duplicate the level actors array to prevent this, there can be a lot of actors in
a level so this would be an unnecessary cost.
Instead, simply delay the creation of actors in any post-restore hook. Note that you can spawn actors indirectly, such as playing a Level Sequence which uses spawnables. In all cases, simply delay spawning / playing the sequence by a small time so that it happens after the restore process.
In DefaultEngine.ini you can change these settings:
[/Script/SPUD.SpudSubsystem]
; The time delay after the last request for a streaming level is withdrawn, that the level will be unloaded
; This is used to reduce load/unload thrashing at boundaries
StreamLevelUnloadDelay=3
; Save game thumbnail sizes
ScreenshotWidth=240
ScreenshotHeight=135
; If true, use the show/hide events of streaming levels to save/load, which is compatible with World Partition
; You can set this to false to change to the legacy mode which requires ASpudStreamingVolume
bSupportWorldPartition=true
; Levels with a name that matches any entry in this list will not be saved/loaded
; Supports basic wildcard matching. Only considers the level name, not the path to the level.
; Case insensitive.
+ExcludeLevelNamePatterns=TitleScreen
+ExcludeLevelNamePatterns=TransientLevel_*