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

devtools/2025/2/26 12:00:52/

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

学习了UE5的例子中的第三人称移动Demo,想实现几个NPC在场景内移动。
本来想自己写一个类,遇到一堆问题花费了好几天时间,所以我把问题写下来方便和我一样初学的同学不要卡住了。

遇到的问题

本来想写一个简单的移动类,发现这些移动可能在上斜坡,或者坐电梯的时候都存在物理问题,采用物理的写法可以上坡,但是在电梯上可能会不跟随电梯,无法达到预期。自己写一个非物理的又很费事。
经过AI咨询一本正经的给出了使用Pawn和UCharacterMovementComponent结合的方式,最后总是报错,AI又开始说用ACharacter,这不是绕了一圈。

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

使用ACharacter的问题

后来又采用了ACharacter作为基类制作第三人称类似的来控制NPC移动。
这里其实也卡了很长时间,因为身边实在没人会UE,AI的水平还是不怎么好,GPT,X.ai,深度,豆包等AI都请教了,得到的答复都差不多。

首先我建立好了类AVRPlayerConsole(用来控制npc)这个类基本上和第三人称相同,就是不需要Input部分,创建了蓝图BP_VRPlayerConsole。
我在蓝图里加了一个球体方便观察。
在这里插入图片描述
拖入场景中是这样的
在这里插入图片描述
然后通过一处脚本来创建这个npc。

 加载蓝图类
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求证才得到了需要创建对象后需要调用

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

这样才可以激活控制器。

就这么简单,需要添加AIController。
就这么一行代码把人折腾死了 。哈哈~~

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

源码

.h

// 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 propertiesAVRPlayerConsole();protected:// Called when the game starts or when spawnedvirtual void BeginPlay() override;public:	void MoveCharacter(const FVector& Direction);// Called every framevirtual void Tick(float DeltaTime) override;// Called to bind functionality to inputvirtual void SetupPlayerInputComponent(class UInputComponent* PlayerInputComponent) override;};

.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 capsuleGetCapsuleComponent()->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 movementGetCharacterMovement()->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 themGetCharacterMovement()->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函数来测试是否可以移动,直接用主玩家的函数调用过来。

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);}

最后测试可以传入方向可以移动了。
效果图如下:
请添加图片描述


http://www.ppmy.cn/devtools/162778.html

相关文章

【qt计算器】

qt计算器 目录注释部分模块配置目标配置模板配置源文件配置头文件配置UI 文件配置1. 头文件保护宏2. 包含必要的头文件3. 命名空间声明4. 类的定义5. 构造函数和析构函数6. 私有槽函数7. 私有成员变量8. 头文件保护宏结束1. 包含头文件2. 构造函数 MainWindow::MainWindow(QWid…

使用机器学习进行土地覆盖分类

土地利用和土地覆盖 (LULC) 分类在林业和农业领域发挥着重要作用&#xff0c;无论是种植园管理、生态系统恢复、碳市场计划还是其他应用。监测土地覆盖和土地利用变化是特许权所有者的一项强制性任务&#xff0c;需要对其特许权区域进行一致且准确的分析。 作为一名 GIS 分析师…

数据结构——排序2

今天&#xff0c;我们来讲解一下选择排序和冒泡排序还有堆排序。 选择排序的基本思想&#xff1a;每一次从待排序的数据元素中选出最小&#xff08;或最大&#xff09;的一个元素&#xff0c;存放在序列的起始位置&#xff0c;直到全部待排序的数据元素排完 。 下图中只选取了它…

显卡(Graphics Processing Unit,GPU)架构详细解读

显卡架构主要分为两大类&#xff1a;GPU 核心架构&#xff08;也称为图形处理单元架构&#xff09;和显卡的其他组件&#xff08;如内存、控制器、输出接口等&#xff09;。本篇文章将对显卡架构进行详细分析&#xff0c;重点介绍 GPU 核心架构、显卡计算单元、显存结构、显卡管…

实现 INFINI Console 与 GitHub 的单点登录集成:一站式身份验证解决方案

本文将为您详细解析如何通过 GitHub OAuth 2.0 协议&#xff0c;为 INFINI Console 实现高效、安全的单点登录&#xff08;Single Sign-On, SSO&#xff09;集成。通过此方案&#xff0c;用户可直接使用 GitHub 账户无缝登录 INFINI Console&#xff0c;简化身份验证流程&#…

Spring Boot + Redis 实现分布式锁

在 Spring Boot 中结合 Redis 实现分布式锁&#xff0c;可以通过 Redisson 或 Jedis 等客户端来操作 Redis&#xff0c;从而实现分布式锁。以下是使用 Redisson 实现分布式锁的示例。 1. 添加依赖 在 pom.xml 中添加 Redisson 依赖&#xff1a; 登录后复制 <dependency>&…

002简单MaterialApp主题和Scaffold脚手架

002最简单的MaterialApp主题和Scaffold脚手架使用导航栏_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV1RZ421p7BL?spm_id_from333.788.videopod.episodes&vd_source68aea1c1d33b45ca3285a52d4ef7365f&p1501.MaterialApp纯净的 /*MaterialApp 是主题,自带方向设…

FFmpeg 命令行全解析:高效音视频处理从入门到精通

FFmpeg FFmpeg 是一款开源的多媒体处理工具集,支持音视频编解码、格式转换、流媒体处理等全链路操作。核心功能与工具: 多媒体全链路支持 支持 1000+ 音视频编解码格式(如 H.264、HEVC、AV1)和协议(RTMP、RTSP、HLS),覆盖录制、转码、流化等全流程。提供三大核心工具: …