《UE5_C++多人TPS完整教程》学习笔记36 ——《P37 拾取组件(Pickup Widget)》

news2025/5/18 10:35:30

本文为B站系列教学视频 《UE5_C++多人TPS完整教程》 —— 《P37 拾取组件(Pickup Widget)》 的学习笔记,该系列教学视频为计算机工程师、程序员、游戏开发者、作家(Engineer, Programmer, Game Developer, Author) Stephen Ulibarri 发布在 Udemy 上的课程 《Unreal Engine 5 C++ Multiplayer Shooter》 的中文字幕翻译版,UP主(也是译者)为 游戏引擎能吃么。

在这里插入图片描述


文章目录

  • P37 拾取组件(Pickup Widget)
  • 37.1 创建拾取控件类
  • 37.2 创建拾取组件重叠事件
  • 37.3 Summary


P37 拾取组件(Pickup Widget)

本节课我们将为我们的武器类制作一个拾取组件(Pickup widget),该组件将显示可以拾取的武器,我们还将基于武器状态设置该组件的可见性(Visibility)。
在这里插入图片描述


37.1 创建拾取控件类

  1. 在虚幻引擎的内容浏览器 “/内容/Blueprints/HUD” 目录下新建一个 “控件蓝图” 类(Widget Blueprint) “WBP_PickupWidget”。
    在这里插入图片描述

  2. 双击 “WBP_PickupWidget”,进入用户控件设计器窗口。如果在左下 “层级” 面板中有 “画布画板” (Canvas Panel),需要将其删除,接着在 “控制板” 面板中将 “通用” 选项卡下的 “文本”(Text)组件拖拽到设计器中,重命名为 “PickupText”;然后在右侧 “细节” 面板的 “内容”(Content)选项卡下设置 “文本”(Text)为 “E-Pick Up”,在“字体”(Font)选项卡下设置 “字号“(Size)为 48,“字体样式”(Typeface)为 “常规”(Regular),设置 “对齐”(Justification)为 “居中对齐”(Align center text);最后调整其大小(Resize),并进行编译、保存。
    在这里插入图片描述

  3. 打开头文件 “Weapon.h”,在武器类私有属性中声明所有地方可见(VisibleAnywhere)、归类为武器属性 “Weapon Properties” 的拾取组件 ”PickupWidget“。

    /*** Weapon.h ***/
    
    ...
    
    UCLASS()
    class BLASTER_API AWeapon : public AActor
    {
    	GENERATED_BODY()
    	
    ...
    
    private:
    	UPROPERTY(VisibleAnywhere, Category = "Weapon Properties")	// 添加所有地方可见的骨骼网格组件,这样就可以通过蓝图进行编辑武器,归类为 Weapon Properties
    	class USkeletalMeshComponent* WeaponMesh;				
    
    	UPROPERTY(VisibleAnywhere, Category = "Weapon Properties")	// 添加一个重叠体积(Overlap Volume),这里使用球体组件 “包裹” 武器骨骼体组件,用于判定人物是否碰到球体,若碰到人物将会拾取该武器
    	class USphereComponent* AreaSphere;
    
    	UPROPERTY(VisibleAnywhere, Category = "Weapon Properties")	// 添加新声明的武器状态枚举类型
    	EWeaponState WeaponState;									
    
    	/* P37 拾取组件(Pickup Widget)*/
    	UPROPERTY(VisibleAnywhere, Category = "Weapon Properties")	// 添加拾取组件
    	class UWidgetComponent* PickupWidget;			
    	/* P37 拾取组件(Pickup Widget)*/
    };
    
  4. 打开源文件 “Weapon.cpp”,添加头文件 “Components//WidgetComponent.h” 在 “Weapon” 类构造函数中 “AWeapon()”,基于拾取组件类创建 “PickupWidget“ 对象,并将 PickupWidget 附加到武器类的根组件上。最后进行编译。

    /*** Weapon.cpp ***/
    
    // Fill out your copyright notice in the Description page of Project Settings.
    
    #include "Weapon.h"	// 原来自动生成的代码是 #include "HUD/OverheadWidget.h",这里需要把 "GameMode/" 去掉,否则找不到文件 "LobbyGameMode.h"
    #include "Components/SphereComponent.h"
    
    /* P37 拾取组件(Pickup Widget)*/
    #include "Components//WidgetComponent.h"
    /* P37 拾取组件(Pickup Widget)*/
    
    // Sets default values
    AWeapon::AWeapon()
    {
    	// Set this actor to call Tick() every frame.  You can turn this off to improve performance if you don't need it.
    	
    	...
    
    	AreaSphere = CreateDefaultSubobject<USphereComponent>(TEXT("AreaSphere"));		// 基于球体组件类创建 AreaSphere 对象
    	AreaSphere->SetupAttachment(RootComponent);										// 将 AreaSphere 附加到武器类的根组件上
    	AreaSphere->SetCollisionResponseToAllChannels(ECollisionResponse::ECR_Ignore);	// 忽略球体组件的碰撞响应,以便于当玩家靠近武器时我们可以检测玩家是否碰到武器(而不是武器被弹开)
    	AreaSphere->SetCollisionEnabled(ECollisionEnabled::NoCollision);				// 先暂时关闭碰撞检测和碰撞响应,物体之间不会发生任何碰撞,我们将在服务器端启用
    
    	/* P37 拾取组件(Pickup Widget)*/
    	PickupWidget = CreateDefaultSubobject<UWidgetComponent>(TEXT("PickupWidget"));	// 基于球体组件类创建 PickupWidget 对象
    	PickupWidget->SetupAttachment(RootComponent);									// 将 PickupWidget 附加到武器类的根组件上
    	/* P37 拾取组件(Pickup Widget)*/
    }
    
    ...
    

37.2 创建拾取组件重叠事件

  1. 在虚幻引擎打开 “BP_Weapon” 蓝图编辑器,在左侧 “组件”(Component)面板中可以看到拾取组件 “Pickup Widget” 已经附加到根组件武器骨骼网格体 “Weapon Mesh (WeaponMesh)” 上,点击它,并在右侧 “细节”(Details)面板 “用户界面”(USER INTERFACE)选项卡下将 “空间”(Space)从 “世界”(World) 设置为 “屏幕”(Screen),将 “控件类”(Widget Class)设置为 “WBP_OverheadWidget”,勾选 “以所需大小绘制”(Draw at Desired Size),这样我们就不必手动设置(Manually set)这个部件的大小;接着在 “视口”(Viewport)面板调整 “Pickup Widget” 到骨骼网格体 “Weapon Mesh (WeaponMesh)” 的上方。
    在这里插入图片描述

  2. 打开游戏关卡 “BlasterMap”,点击工具栏的 “播放”()按钮启动运行,可以看到在监听服务器中和两个客户端中武器上方均出现了文本 “E-Pick Up”。
    在这里插入图片描述

  3. 我们想要在人物与球体组件 “AreaSphere” 发生重叠时才显示 “E-Pick Up”。现在暂时不需要头顶组件 “OverheadWidget” 上显示网络角色(之后将用来显示玩家昵称),进入 “WBP_OverheadWidget” 用户控件设计器窗口,在左下 “层级” 面板中点击 “DisplayText”,接着在右侧 “细节” 面板的 “内容”(Content)选项卡下设置 “文本”(Text)为
    在这里插入图片描述

  4. 打开头文件 “Weapon.h”,声明覆写球体重叠事件 “OnSphereOverlap“ 函数。

    /*** Weapon.cpp ***/
    
    ...
    
    UCLASS()
    class BLASTER_API AWeapon : public AActor
    {
    	GENERATED_BODY()
    	
    public:	
    	// Sets default values for this actor's properties
    	AWeapon();
    
    protected:
    	// Called when the game starts or when spawned
    	virtual void BeginPlay() override;
    
    public:	
    	// Called every frame
    	virtual void Tick(float DeltaTime) override;
    
    
    	/* P37 拾取组件(Pickup Widget)*/
    	UFUNCTION()					// 需要动态绑定到 OnComponentBeginOverlap 的委托上,因此需要将函数设置为 UFunction
    	virtual void OnSphereOverlap(
    		UPrimitiveComponent*  OverlappedComponent,
    		AActor* OtherActor,
    		UPrimitiveComponent* OhterComponent,
    		int32 OtherBodyIndex,
    		bool bFromSweep,
    		const FHitResult& SweepResult
    	);
    	/* P37 拾取组件(Pickup Widget)*/
    	
    	...
    
    }
    
  5. 在源文件 “Weapon.cpp” 中将其动态绑定到球体组件 “AreaSphere” 的委托 “OnComponentBeginOverlap” 上,“OnComponentBeginOverlap” 是 “FComponentBeginOverlapSignature“ 类型的变量,在 VS 中我们查看 “FComponentBeginOverlapSignature“ 的文档,可以发现为实现动态绑定到委托,“OnSphereOverlap“ 函数需要的入参和文档中要求的入参需要保持一致。
    为实现人物 “ABlasterCharacter” 与球体组件 “AreaSphere” 发生重叠时显示 “E-Pick Up”,首先在 “BeginPlay” 先设置拾取组件不可见,然后在 “OnSphereOverlap” 中设置当人物与球体组件发生重叠时拾取组件可见。

    /*** Weapon.cpp ***/
    
    // Fill out your copyright notice in the Description page of Project Settings.
    
    #include "Weapon.h"	// 原来自动生成的代码是 #include "HUD/OverheadWidget.h",这里需要把 "GameMode/" 去掉,否则找不到文件 "LobbyGameMode.h"
    #include "Components/SphereComponent.h"
    
    /* P37 拾取组件(Pickup Widget)*/
    #include "Components/WidgetComponent.h"
    #include "Blaster/Character/BlasterCharacter.h"
    /* P37 拾取组件(Pickup Widget)*/
    
    ...
    // Called when the game starts or when spawned
    void AWeapon::BeginPlay()
    {
    	Super::BeginPlay();
    
    	// 服务器将负责掌管(in charge of)所有武器权限
    	if (HasAuthority()) {															// 判断是否拥有权威角色,等同于 if (GetLocalRole() == ENetRole::ROLE_Authority) 
    		AreaSphere->SetCollisionEnabled(ECollisionEnabled::QueryAndPhysics);		// 既进行碰撞检测,也进行碰撞响应。物体之间会检测是否发生碰撞,同时会产生物理效果
    		AreaSphere->SetCollisionResponseToChannel(ECollisionChannel::ECC_Pawn, ECollisionResponse::ECR_Overlap);
    		
    		/* P37 拾取组件(Pickup Widget)*/
    		AreaSphere->OnComponentBeginOverlap.AddDynamic(this, &AWeapon::OnSphereOverlap);
    		/** Delegate for notification of start of overlap with a specific component */
    		/* DECLARE_DYNAMIC_MULTICAST_SPARSE_DELEGATE_SixParams(FComponentBeginOverlapSignature, 
    																UPrimitiveComponent, 
    																OnComponentBeginOverlap, 
    																UPrimitiveComponent*, 
    																OverlappedComponent, 
    																AActor*, OtherActor, 
    																UPrimitiveComponent*, 
    																OtherComp, int32, 
    																OtherBodyIndex, 
    																bool,
    																bFromSweep, 
    																FHitResult&, 
    																SweepResult);
    		*/
    		/* P37 拾取组件(Pickup Widget)*/
    	}
    
    	/* P37 拾取组件(Pickup Widget)*/
    	if (PickupWidget) {									// 确保 PickupWidget 不是空指针
    		PickupWidget->SetVisibility(false);				// 先设置拾取组件不可见
    	}
    	/* P37 拾取组件(Pickup Widget)*/
    }
    
    /* P37 拾取组件(Pickup Widget)*/
    void AWeapon::OnSphereOverlap(UPrimitiveComponent* OverlappedComponent, AActor* OtherActor, UPrimitiveComponent* OhterComponent, int32 OtherBodyIndex, bool bFromSweep, const FHitResult& SweepResult)
    {
    	// 需要先添加头文件 "Blaster/Character/BlasterCharacter.h"
    	ABlasterCharacter* BlasterCharacter = Cast<ABlasterCharacter>(OtherActor);	// 类型转换为 OtherActor
    	if (BlasterCharacter && PickupWidget) {										// 确保 BlasterCharacter 和 PickupWidget 都不是空指针
    		PickupWidget->SetVisibility(true);										// 设置拾取组件可见
    	}
    }
    /* P37 拾取组件(Pickup Widget)*/
    

    注意
    委托以及动态绑定的相关概念可以参阅笔者之前的学习笔记《UE5_C++多人TPS完整教程》学习笔记9 ——《P10 创建会话(Creating A Session)》以及《UE5_C++多人TPS完整教程》学习笔记20 ——《P21 更多的子系统委托(More Subsystem Delegates)》。

  6. 分别进行三次测试,可以看到无论是视口面板中的服务器端上的角色还是另外两个小窗中的客户端的角色,当他们分别与武器发生重叠时, “E-Pick Up” 均只显示在视口面板中的服务器端上,这是我们期待发生的,因为重叠事件只在服务器上调用。
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述


37.3 Summary

本节课我们为武器类添加交互式拾取组件,该组件将显示可以拾取的武器。首先我们新建了 “WBP_PickupWidget” 控件蓝图,在武器 C++ 类中声明 “UWidgetComponent* PickupWidget”,通过构造函数初始化并附加到武器根组件。接着,为了使得游戏人物与球体组件 “AreaSphere” 发生重叠时才显示 “E-Pick Up”,我们在武器 C++ 类的 “BeginPlay()” 函数中通过 “PickupWidget->SetVisibility(false)" 设置初始时拾取组件不可见,然后在服务器端绑定重叠事件“OnSphereOverlap” 到球体组件 “AreaSphere” 的碰撞委托上,当人物进入球体范围时,通过 “PickupWidget->SetVisibility(true)” 显示 “E-Pick Up”,并且仅由服务器触发,客户端通过属性复制同步状态。在后续测试中,无论是视口面板中的服务器端上的角色还是另外两个小窗中的客户端的角色,当他们分别在测试中与武器发生重叠时, “E-Pick Up” 均只显示在视口面板中的服务器端上,因为重叠事件只在服务器上调用。
在这里插入图片描述
37.2 创建拾取组件重叠事件 中,关于委托以及动态绑定的相关概念可以参阅笔者之前的学习笔记《UE5_C++多人TPS完整教程》学习笔记9 ——《P10 创建会话(Creating A Session)》以及《UE5_C++多人TPS完整教程》学习笔记20 ——《P21 更多的子系统委托(More Subsystem Delegates)》。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2337639.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

《空间复杂度(C语言)》

文章目录 前言一、什么是空间复杂度&#xff1f;通俗理解&#xff1a; 二、空间复杂度的数学定义三、常见空间复杂度举例&#xff08;含C语言代码&#xff09;&#x1f539; O(1)&#xff1a;常数空间&#x1f539; O(n)&#xff1a;线性空间&#x1f539; O(n^2)&#xff1a;平…

智能合约安全审计平台——以太坊虚拟机安全沙箱

目录 以太坊虚拟机安全沙箱 —— 理论、设计与实战1. 引言2. 理论背景与安全原理2.1 以太坊虚拟机(EVM)概述2.2 安全沙箱的基本概念2.3 安全证明与形式化验证3. 系统架构与模块设计3.1 模块功能说明3.2 模块之间的数据流与安全性4. 安全性与密码学考量4.1 密码学保障在沙箱中…

【MCP教程】Claude Desktop 如何连接部署在远程的remote mcp server服务器(remote host)

前言 最近MCP特别火热&#xff0c;笔者自己也根据官方文档尝试了下。 官方文档给的Demo是在本地部署一个weather.py&#xff0c;然后用本地的Claude Desktop去访问该mcp服务器&#xff0c;从而完成工具的调用&#xff1a; 但是&#xff0c;问题来了&#xff0c;Claude Deskto…

多个路由器互通(静态路由)无单臂路由(简单版)

多个路由器互通&#xff08;静态路由&#xff09;无单臂路由&#xff08;简单版&#xff09; 开启端口并配ip地址 维护1 Router>en Router#conf t Router(config)#int g0/0 Router(config-if)#no shutdown Router(config-if)#ip address 192.168.10.254 255.255.255.0 Ro…

OpenCV 图形API(38)图像滤波-----Sobel 算子操作函数Sobel()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 cv::gapi::Sobel 函数是 OpenCV 的 G-API 模块中用于执行 Sobel 算子操作的一个函数&#xff0c;主要用于图像的边缘检测。Sobel 算子通过计算图…

windows系统安装驱动、cuda和cudnn

一、首先在自己的电脑里安装了nvidia的独立显卡 显卡的查找方式&#xff1a; CtrlShiftEsc打开任务管理器&#xff0c;点击性能&#xff0c;点击GPU 0查看显卡型号&#xff0c;如下图所示&#xff1a; 只要电脑中有nvidia的独立显卡&#xff0c;就可以暗转显卡驱动、cuda和cu…

嵌入式开发--STM32软件和硬件CRC的使用--续篇

本文是《嵌入式开发–STM32软件和硬件CRC的使用》的续篇&#xff0c;又踩到一个坑&#xff0c;发出来让大家避一下坑。 按照G0系列的设置&#xff0c;得出错误的结果 前文对应的是STM32G0系列&#xff0c;今天在用STM32G4系列时&#xff0c;按照前文的设置&#xff0c;用硬件…

【Git】git的简单使用

文章目录 1. 基础概念2. 简单使用2.1 git配置2.1.1 git的配置文件2.1.2 .gitignore文件 2.2 创建仓库2.2.1 创建本地仓库2.2.2 github创建远程仓库step1&#xff1a;github新建一个代码仓step2&#xff1a;创建密钥远程仓库相关指令2.2.3 本地仓库 关联 远程仓库 2.3 分支2.3.1…

[Web 安全] Web 信息收集 —— 信息收集流程

&#x1f31f; 想系统化学习 Web 渗透&#xff1f;看看这个&#xff1a;[Web 安全] Web 安全攻防 学习手册 提示&#xff1a;本章不涉及任何具体信息收集技术&#xff0c;仅仅是讲解收集这些信息我能干啥&#xff0c;以及如何才能比较全面的收集信息。 0x01&#xff1a;信息收…

内部聊天软件,BeeWorks-安全的企业内部通讯软件

企业在享受数据便利的同时&#xff0c;如何保障企业数据安全已经成为无法回避的重要课题。BeeWorks作为一款专为企业设计的内部通讯软件&#xff0c;通过全链路的安全能力升维&#xff0c;为企业提供了一个安全、高效、便捷的沟通协作平台&#xff0c;全面保障企业数据安全。 …

应用篇02-镜头标定(上)

本节主要介绍相机的标定方法&#xff0c;包括其内、外参数的求解&#xff0c;以及如何使用HALCON标定助手实现标定。 计算机视觉——相机标定(Camera Calibration)_摄像机标定-CSDN博客 1. 原理 本节介绍与相机标定相关的理论知识&#xff0c;不一定全&#xff0c;可以参考相…

【UE5 C++】“ProceduralMeshComponent”的使用记录

效果 如下所示&#xff0c;通过“ProceduralMeshComponent”创建了一个自定义形状的Mesh&#xff0c;并且该Mesh包含碰撞信息&#xff0c;然后2s后更新Mesh形状。 步骤 1. 在“xxx.Build.cs”中引入“ProceduralMeshComponent”模块 2. 新建一个Actor类&#xff0c;这里命名为…

(leetcode算法题)309. 买卖股票的最佳时机含冷冻期

按照题目要求&#xff0c;研究对象是最后一天结束后获得的最大利润 那么就可以把问题拆分成 第 1 天结束后获得的最大利润&#xff0c; 第 2 天结束后获得的最大利润&#xff0c; 第 i 天结束后获得的最大利润&#xff0c; 由于规则中强调不能同时参与多笔交易&#xff0c…

Chrome漏洞可窃取数据并获得未经授权的访问权限

在发现两个关键漏洞后,谷歌发布了Chrome浏览器的紧急安全更新。这些漏洞可能允许攻击者窃取敏感数据并未经授权访问用户系统。 这些缺陷被识别为CVE-2025-3619和CVE-2025-3620,在Windows和Mac的135.0.7049.95/.96之前影响Chrome版本,影响Linux的135.0.7049.95/.96。该更新将在…

.net core 项目快速接入Coze智能体-开箱即用-全局说明

目录 一、Coze智能体的核心价值 二、开箱即用-效果如下 三 流程与交互设计 为什么要分析意图&#xff0c;而不是全部交由AI处理。 四 接入前的准备工作 五&#xff1a;代码实现----字节Coze 签署 JWT和获取Token .net core 项目快速接入Coze智能体-开箱即用 .net core快…

风丘年度活动:2025年横滨汽车工程展览会

| 展会简介&#xff1a; 2025年横滨汽车工程展览会&#xff0c;是由日本汽车工程师学会&#xff08;JSAE&#xff09;精心主办的一场行业盛会。预计届时将汇聚超550家参展商&#xff0c;设置1300个展位&#xff0c;展览面积超过20000平方米。展会受众广泛&#xff0c;面向汽车…

Redis线上操作最佳实践有哪些?

大家好&#xff0c;我是锋哥。今天分享关于【Redis线上操作最佳实践有哪些?】面试题。希望对大家有帮助&#xff1b; Redis线上操作最佳实践有哪些? 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 在使用 Redis 时&#xff0c;尤其是在生产环境中&#xff0c;合理…

Redis——五种数据类型

目录 前言 1.String 1.1RAW编码 1.2EMBSTR编码 1.3 INT编码 2.List 3.Set 3.1 InSet编码转化成Dict编码 4.ZSet 4.1结合SkipList和HT实现 4.2使用ZipList实现 4.3编码转换 4.4 ZipList排序功能 5.Hash 5.1Hash底层存储结构 6.Redis数据结构和数据类型关系图 前言…

Godot学习-创建简单动画

文章目录 1、准备工作Godot资源 2、创建项目3、创建结点4、创建动画1、创建动画2、添加轨道3、创建关键帧3.1 第一个关键帧3.2 第二个关键帧 5、加载后自动播放6、动画循环7、轨道设置1、轨道更新模式2、轨迹插值3、其他属性的关键帧4、编辑关键帧5、使用 RESET 轨道6、洋葱皮 …

论文阅读VACE: All-in-One Video Creation and Editing

code&#xff1a;https://github.com/ali-vilab/VACE 核心 单个模型同时处理多种视频生成和视频编辑任务通过VCU&#xff08;视频条件单元&#xff09;进行实现 方法 视频任务 所有的视频相关任务可以分为4类 文本生视频 参考图片生视频 视频生视频 视频mask生视频 VCU …