当前位置: 首页 > article >正文

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。

 加载蓝图类
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 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

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

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.kler.cn/a/562415.html

相关文章:

  • FFMPEG编码容错处理解决办法之途径----升级库文件
  • 一周掌握Flutter开发--3、布局与 UI 组件
  • 基于“桌面云+RPA”技术金融应用实施建议
  • STM32--SPI通信讲解
  • ​腾讯云 轻量云对象存储
  • Redis实战篇《黑马点评》5
  • vue3封装一个悬浮操作固定列表格组件(性能版)
  • 声明式UI差分刷新机制的原理,应用场景及优缺点
  • 计组笔记day-01
  • 点云配准技术的演进与前沿探索:从传统算法到深度学习融合(1)
  • DeepSeek+谷云科技智能体,快速构建企业知识问答
  • CryptoJS库中WordArray对象支持哪些输出格式?除了toString() 方法还有什么方法可以输出吗?WordArray对象的作用是什么?
  • Vue 3中的路由和Vue Router 4有哪些变化?
  • 计算机网络————(三)
  • Java Web应用中的跨站脚本攻击(XSS)防护策略
  • linux-c 字节序问题--大小端
  • Html 5简介(学习笔记)
  • BIO系统调用strace查看IO阻塞
  • 【Java 基础】-- Java 接口中的 @Public 和 @FunctionalInterface 注解详解
  • SpringBoot整合sharding-jdbc 实现分库分表操作