Lost_Edge/Source/Lost_Edge/Private/Minigame/CrossyRoad/CrossyRoadManager.cpp
2024-09-26 15:15:45 +02:00

249 lines
7.5 KiB
C++

// Oleg Petruny proprietary.
#include "CrossyRoadManager.h"
#include "Camera/CameraComponent.h"
#include "Components/SkeletalMeshComponent.h"
#include "Components/SplineComponent.h"
#include "EnhancedInputComponent.h"
#include "InputMappingContext.h"
#include "CrossyRoadObstacle.h"
#include "CustomGameInstanceBase.h"
#include "MainGameModeBase.h"
#include "PlayerBase.h"
#include "Widgets/WidgetsManager.h"
ACrossyRoadManager::ACrossyRoadManager()
: AMinigame()
{
static ConstructorHelpers::FObjectFinder<UInputMappingContext> asset{ TEXT("/Script/EnhancedInput.InputMappingContext'/Game/Input/Minigame/IMC_Minigame_CrossyRoad.IMC_Minigame_CrossyRoad'") };
input = asset.Object;
auto root = CreateDefaultSubobject<USceneComponent>(TEXT("Scene"));
playerPos = CreateDefaultSubobject<USceneComponent>(TEXT("PlayerPos"));
playerPos->AttachToComponent(root, FAttachmentTransformRules::KeepRelativeTransform);
camera = CreateDefaultSubobject<UCameraComponent>(TEXT("Camera"));
camera->AttachToComponent(root, FAttachmentTransformRules::KeepRelativeTransform);
mannequin = CreateDefaultSubobject<USkeletalMeshComponent>(TEXT("Mannequin"));
mannequin->AttachToComponent(root, FAttachmentTransformRules::KeepRelativeTransform);
}
void ACrossyRoadManager::Start(APlayerBase* playerPawn, FMinigameEndCallback delegate)
{
if(!playerPawn)
return;
AMinigame::Start(playerPawn, delegate);
player->LockPlayer(FPlayerLock::All());
player->SwitchToView(this);
for(auto& mapping : input.LoadSynchronous()->GetMappings())
{
if(mapping.Action->ActionDescription.EqualTo(FText::AsCultureInvariant(TEXT("Up"))))
inputHandlers.Add(player->inputComponent->BindAction(mapping.Action, ETriggerEvent::Triggered, this, &ACrossyRoadManager::Up).GetHandle());
else if(mapping.Action->ActionDescription.EqualTo(FText::AsCultureInvariant(TEXT("Down"))))
inputHandlers.Add(player->inputComponent->BindAction(mapping.Action, ETriggerEvent::Triggered, this, &ACrossyRoadManager::Down).GetHandle());
else if(mapping.Action->ActionDescription.EqualTo(FText::AsCultureInvariant(TEXT("Left"))))
inputHandlers.Add(player->inputComponent->BindAction(mapping.Action, ETriggerEvent::Triggered, this, &ACrossyRoadManager::Left).GetHandle());
else if(mapping.Action->ActionDescription.EqualTo(FText::AsCultureInvariant(TEXT("Right"))))
inputHandlers.Add(player->inputComponent->BindAction(mapping.Action, ETriggerEvent::Triggered, this, &ACrossyRoadManager::Right).GetHandle());
}
playerPos->SetWorldLocation(lines[0]->GetLocationAtTime(lines[0]->Duration * playerLineTime, ESplineCoordinateSpace::World));
mannequin->SetWorldLocation(playerPos->GetComponentLocation());
for(auto line : lines)
if(FMath::RandRange(0, 1))
RotateLine(line);
PrimaryActorTick.SetTickFunctionEnable(true);
}
void ACrossyRoadManager::End()
{
PrimaryActorTick.SetTickFunctionEnable(false);
for(int32 handler : inputHandlers)
player->inputComponent->RemoveBindingByHandle(handler);
player->UnlockPlayer(FPlayerLock::All());
player->ReturnPlayerView();
AMinigame::End();
}
void ACrossyRoadManager::Hit()
{
if(ended)
return;
ended = true;
if(score > 1000)
{
Finish();
return;
}
result = EMinigameResult::Loss;
End();
}
void ACrossyRoadManager::Finish()
{
result = EMinigameResult::Win;
End();
}
void ACrossyRoadManager::PostInitializeComponents()
{
AMinigame::PostInitializeComponents();
GetComponents<USplineComponent>(lines);
for(int32 i = 0; i < lines.Num(); ++i)
{
int32 id = FCString::Atoi(*lines[i]->GetName().Mid(6));
if(i != id - 1)
{
auto l = lines[i];
lines[i] = lines[id - 1];
lines[id - 1] = l;
}
linesRotated.Add(lines[i], false);
}
}
void ACrossyRoadManager::Tick(float deltaTime)
{
AMinigame::Tick(deltaTime);
time += deltaTime;
if(time >= spawnDelay / speed)
{
time = 0; //FMath::RandRange(0.0f, spawnDelay / 4 * speed); // (time -= spawnDelay) is not good on lags
CreateObstacle();
}
for(int32 i = 0; i < obstacles.Num(); ++i)
{
if(obstacles[i].time > obstacles[i].line->Duration)
{
obstacles[i].actor->Destroy();
obstacles.RemoveAt(i);
//CreateObstacle();
}
else
{
obstacles[i].time += deltaTime * speed * obstacles[i].speed;
obstacles[i].actor->SetActorLocation(obstacles[i].line->GetWorldLocationAtTime(obstacles[i].time));
}
}
}
void ACrossyRoadManager::CreateObstacle(int32 index)
{
auto obstacleClass = obstacleClasses.CreateIterator();
for(int32 i = 0; i < FMath::RandRange(0, obstacleClasses.Num() - 1); ++i) // std::advance not works :(
++obstacleClass;
auto spline = lines[FMath::RandRange(0, lines.Num() - 1)];
FActorSpawnParameters spawnParams{};
spawnParams.Owner = this;
auto obstacle = GetWorld()->SpawnActor<ACrossyRoadObstacle>(*obstacleClass, spline->GetComponentLocation(), GetActorRotation(), spawnParams);
obstacle->manager = this;
if(index < 0)
{
obstacles.Add({ .actor = obstacle, .line = spline, .time = 0, .speed = FMath::RandRange(0.2f, 1.0f) * speed });
}
else
{
obstacles[index].actor->Destroy();
obstacles[index].actor = obstacle;
obstacles[index].line = spline;
obstacles[index].time = 0;
obstacles[index].speed = FMath::RandRange(1.0f, 2.0f) * speed;
}
}
void ACrossyRoadManager::RotateLine(USplineComponent* line)
{
auto pos = line->GetComponentLocation();
auto rot = line->GetComponentRotation();
auto dist = line->GetWorldLocationAtSplinePoint(1) - line->GetWorldLocationAtSplinePoint(0);
if(linesRotated[line])
{
line->SetWorldLocationAndRotation(pos - dist, FRotator{ rot.Pitch, rot.Yaw - 180, rot.Roll });
linesRotated[line] = false;
}
else
{
line->SetWorldLocationAndRotation(pos + dist, FRotator{ rot.Pitch, rot.Yaw + 180, rot.Roll });
linesRotated[line] = true;
}
}
void ACrossyRoadManager::Up()
{
auto line = lines[0];
auto diffY = lines[0]->GetComponentLocation().Y - lines[1]->GetComponentLocation().Y;
for(int32 i = 1; i < lines.Num(); ++i)
{
lines[i]->AddRelativeLocation(FVector{ 0, diffY, 0 });
lines[i - 1] = lines[i];
}
line->AddRelativeLocation(FVector{ 0, (lines.Num() - 1) * diffY * -1, 0 });
lines[lines.Num() - 1] = line;
//if(FMath::RandRange(0, 1))
// RotateLine(line);
speed += 0.15;
score += 100 * speed;
}
void ACrossyRoadManager::Down()
{
//
}
void ACrossyRoadManager::Left()
{
if(playerLineTime >= 1.0f)
return;
playerLineTime += 0.1f;
float lineTime = playerLineTime;
if(linesRotated[lines[0]])
lineTime = 1 - lineTime;
playerPos->SetWorldLocation(lines[0]->GetLocationAtTime(lines[0]->Duration * lineTime, ESplineCoordinateSpace::World));
mannequin->SetWorldLocation(playerPos->GetComponentLocation());
}
void ACrossyRoadManager::Right()
{
if(playerLineTime <= 0.0f)
return;
playerLineTime -= 0.1f;
float lineTime = playerLineTime;
if(linesRotated[lines[0]])
lineTime = 1 - lineTime;
playerPos->SetWorldLocation(lines[0]->GetLocationAtTime(lines[0]->Duration * lineTime, ESplineCoordinateSpace::World));
mannequin->SetWorldLocation(playerPos->GetComponentLocation());
}