UE(虚幻)学习(五)初学创建NPC移动和遇到的问题

最近在学习UE中遇到一些问题,把这些问题记录一下,因为实在废了很大功夫。

在学习了UE5的例子中的第三人称移动Demo,想实现几个NPC在场景内移动。

本来想自己写一个类,遇到一堆问题花费了好几天时间,所以我把问题写下来方便和我一样初学的同学不要卡住了。

遇到的问题

本来想写一个简单的移动类,发现这些移动可能在上斜坡,或者坐电梯的时候都存在物理问题,采用物理的写法可以上坡,但是在电梯上可能会不跟随电梯,无法达到预期。自己写一个非物理的又很费事。

经过AI咨询一本正经的给出了使用Pawn和UCharacterMovementComponent结合的方式,最后总是报错,AI又开始说用ACharacter,这不是绕了一圈。

上面的错误没绕出来的代码就不贴了(早删掉了) :)

使用ACharacter的问题

后来又采用了ACharacter作为基类制作第三人称类似的来控制NPC移动。

这里其实也卡了很长时间,因为身边实在没人会UE,AI的水平还是不怎么好,GPT,X.ai,深度,豆包等AI都请教了,得到的答复都差不多。

首先我建立好了类AVRPlayerConsole(用来控制npc)这个类基本上和第三人称相同,就是不需要Input部分,创建了蓝图BP_VRPlayerConsole。

我在蓝图里加了一个球体方便观察。

拖入场景中是这样的

然后通过一处脚本来创建这个npc。

cpp 复制代码
 加载蓝图类
FString BlueprintPath = TEXT("/Game/BP/BP_VRPlayerConsole.BP_VRPlayerConsole_C");
UClass* BlueprintClass = StaticLoadClass(UObject::StaticClass(), nullptr, *BlueprintPath);

if (!BlueprintClass)
{
	UE_LOG(LogTemp, Error, TEXT("Failed to load blueprint class at %s"), *BlueprintPath);
	return;
}

// 设置生成位置和旋转
FVector SpawnLocation(310.f, 350.f, 300.f);
FRotator SpawnRotation(0.0f, 0.0f, 0.0f);

if (BlueprintClass) // 确保蓝图类有效
{

	charConsole = GetWorld()->SpawnActor<AVRPlayerConsole>(
		BlueprintClass,
		SpawnLocation,
		SpawnRotation,
		SpawnParams
	);

	if (charConsole)
	{
		
		
		UE_LOG(LogTemp, Log, TEXT("成功生成 BP_VRPlayerConsole 实例!"));
	}
}

当我编译后运行,发现创建出来了,但是他在空中不会随着重力掉下去。

AI解决

AI给的说法一大堆

Movement Mode 是否正确
UpdatedComponent 是否有效
Velocity 是否被更新
是否被 Root Motion 锁定
Tick 是否在执行
强制刷新 Movement
检查是否被其他状态阻止移动
强制启用 Movement Component
检查 Movement Component 是否允许移动
直接修改位置SetActorLocation,验证角色是否能移动
检查 Custom Movement Mode
检查角色的 CanMove() 是否被禁用
尝试强制设置 MovementMode
是否受到物理影响
强制切换 MovementMode
是否禁用了 MovementComponent
确保 charConsole 不是 Physics 角色
运动速度 (MaxWalkSpeed) 是否有效
AddMovementInput 依赖于 RootComponent
charConsole 是否具有控制器 (Controller)

看着就觉得有问题,但是又找不到问题。花费了大把时间研究,都一一排除了。

其实AI提到了控制器,但是初次接触UE的是想不到还有AI控制器等等的说法。

发现

一直觉得是脚本的问题。

后来发现如果把蓝图拖入场景中运行,对象因为重力影响会掉落到地板。当拖入多个也都会落到地板上,只有自己实例化的不行。

再问AI为什么拖入场景的可以落地,自己实例化的不行,又给出实例化要这样要那样,都不对。

最后发现拖入场景的对象运行后会有一个AIControler对象生成。

再次AI求证才得到了需要创建对象后需要调用

cpp 复制代码
charConsole->SpawnDefaultController(); // 可选:生成AI控制器

这样才可以激活控制器。

就这么简单,需要添加AIController。

就这么一行代码把人折腾死了 。哈哈~~

最后放上这个类的源码做一个记录

源码

.h

cpp 复制代码
// Fill out your copyright notice in the Description page of Project Settings.

#pragma once

#include "CoreMinimal.h"
#include "GameFramework/Character.h"

#include "VRPlayerConsole.generated.h"

UCLASS(config = Game)
class UE5TECHNOLOGY_API AVRPlayerConsole : public ACharacter
{
	GENERATED_BODY()

public:
	// Sets default values for this character's properties
	AVRPlayerConsole();

protected:
	// Called when the game starts or when spawned
	virtual void BeginPlay() override;

public:	


	void MoveCharacter(const FVector& Direction);

	// Called every frame
	virtual void Tick(float DeltaTime) override;

	// Called to bind functionality to input
	virtual void SetupPlayerInputComponent(class UInputComponent* PlayerInputComponent) override;


};

.cpp

cpp 复制代码
// Fill out your copyright notice in the Description page of Project Settings.


#include "VRPlayerConsole.h"
#include "Components/CapsuleComponent.h"
#include "GameFramework/CharacterMovementComponent.h"
#include "Engine/LocalPlayer.h"
// Sets default values
AVRPlayerConsole::AVRPlayerConsole()
{
	// Set this character to call Tick() every frame.  You can turn this off to improve performance if you don't need it.
	PrimaryActorTick.bCanEverTick = false;

	// Set size for collision capsule
	GetCapsuleComponent()->InitCapsuleSize(42.f, 96.0f);

	// Don't rotate when the controller rotates. Let that just affect the camera.
	bUseControllerRotationPitch = false;
	bUseControllerRotationYaw = false;
	bUseControllerRotationRoll = false;

	// Configure character movement
	GetCharacterMovement()->bOrientRotationToMovement = true; // Character moves in the direction of input...	
	GetCharacterMovement()->RotationRate = FRotator(0.0f, 500.0f, 0.0f); // ...at this rotation rate

	// Note: For faster iteration times these variables, and many more, can be tweaked in the Character Blueprint
	// instead of recompiling to adjust them
	GetCharacterMovement()->JumpZVelocity = 700.f;
	GetCharacterMovement()->AirControl = 0.35f;
	GetCharacterMovement()->MaxWalkSpeed = 500.f;
	GetCharacterMovement()->MinAnalogWalkSpeed = 20.f;
	GetCharacterMovement()->BrakingDecelerationWalking = 2000.f;
	GetCharacterMovement()->BrakingDecelerationFalling = 1500.0f;


}

// Called when the game starts or when spawned
void AVRPlayerConsole::BeginPlay()
{
	Super::BeginPlay();

	//GetCharacterMovement()->SetMovementMode(EMovementMode::MOVE_Walking);
	//UE_LOG(LogTemp, Warning, TEXT("IsActive: %d"), GetCharacterMovement()->IsActive());


}

// Called every frame
void AVRPlayerConsole::Tick(float DeltaTime)
{
	Super::Tick(DeltaTime);

}

// Called to bind functionality to input
void AVRPlayerConsole::SetupPlayerInputComponent(UInputComponent* PlayerInputComponent)
{
	Super::SetupPlayerInputComponent(PlayerInputComponent);

}

void AVRPlayerConsole::MoveCharacter(const FVector& Direction)
{
	if (Direction.IsNearlyZero()) return;

	// 确保 Movement 是激活的
	UCharacterMovementComponent* MoveComp = GetCharacterMovement();
	if (MoveComp && MoveComp->IsActive())
	{
		MoveComp->AddInputVector(Direction);
	}
	else
	{
		UE_LOG(LogTemp, Error, TEXT("Character Movement is Invalid!"));
	}
}

我在我的主玩家控制力添加了一个MoveCharacter函数来测试是否可以移动,直接用主玩家的函数调用过来。

cpp 复制代码
void AVRPlayer::Move(const FInputActionValue& Value)
{
	// 获取 2D-Axis 值
	FVector2D MovementVector = Value.Get<FVector2D>();
	if (MovementVector.IsNearlyZero())
	{
		return; // 如果没有输入,就直接返回
	}
	if (vrCamera == nullptr)
	{
		UE_LOG(LogTemp, Log, TEXT("vrCamera null"));
		return;
	}
	// 获取相机的方向
	FVector ForwardDirection = vrCamera->GetForwardVector(); // 相机的前向向量
	FVector RightDirection = vrCamera->GetRightVector();     // 相机的右向向量

	ForwardDirection.Z = 0;
	// 将 MovementVector 的 X 和 Y 分别映射到相机的方向上
	FVector MoveDirection = ((ForwardDirection * MovementVector.Y) + (RightDirection * MovementVector.X)).GetSafeNormal();

	if (charConsole == nullptr)
	{
		UE_LOG(LogTemp, Log, TEXT("charConsole null !!!"));
		//UE_LOG(LogTemp, Log, TEXT("VRRoot address: %p"), VRRoot);
		return;
	}

	charConsole->MoveCharacter(MoveDirection);

}

最后测试可以传入方向可以移动了。

效果图如下:

相关推荐
网络安全指导员39 分钟前
如何在JMeter中配置断言,将非200状态码视为测试成功
网络·学习·jmeter·安全·web安全·架构
浪淘沙jkp43 分钟前
大模型学习四:‌DeepSeek Janus-Pro 多模态理解和生成模型 本地部署指南(折腾版)
python·学习·deepseek
Kx…………1 小时前
Uni-app入门到精通:uni-app的基础组件
前端·css·学习·uni-app·html
~樱小路~1 小时前
网络:华为数通HCIA学习:IP路由基础
网络·学习·华为
吴梓穆2 小时前
UE5学习笔记 FPS游戏制作42 按钮添加回调函数
笔记·学习·ue5
吴梓穆2 小时前
UE5学习笔记 FPS游戏制作39 制作一个带有背景的预制面板 使用overlay和nameSlot
笔记·学习·ue5
AgilityBaby2 小时前
UE5把动画导出为视频格式
ue5·游戏引擎·unreal engine
向宇it2 小时前
【零基础入门unity游戏开发——2D篇】SortingGroup(排序分组)组件
开发语言·unity·c#·游戏引擎·材质
云上艺旅1 天前
K8S学习之基础七十四:部署在线书店bookinfo
学习·云原生·容器·kubernetes
你觉得2051 天前
哈尔滨工业大学DeepSeek公开课:探索大模型原理、技术与应用从GPT到DeepSeek|附视频与讲义下载方法
大数据·人工智能·python·gpt·学习·机器学习·aigc