【制作100个unity游戏之25】3D背包、库存、制作、快捷栏、存储系统、砍伐树木获取资源、随机战利品宝箱1(附带项目源码)

本文主要是介绍【制作100个unity游戏之25】3D背包、库存、制作、快捷栏、存储系统、砍伐树木获取资源、随机战利品宝箱1(附带项目源码),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

效果演示

在这里插入图片描述

文章目录

  • 效果演示
  • 系列目录
  • 前言
  • 人物和视角基本控制
  • 简单的背包系统和物品交互
    • 绘制背包UI
    • 脚本控制
  • 源码
  • 完结

系列目录

前言

欢迎来到【制作100个Unity游戏】系列!本系列将引导您一步步学习如何使用Unity开发各种类型的游戏。在这第25篇中,我们将探索如何用unity制作一个3D背包、库存、制作、快捷栏、存储系统、砍伐树木获取资源、随机战利品宝箱等功能,我会附带项目源码,以便你更好理解它。

人物和视角基本控制

具体可以看我这篇文章:
【unity小技巧】unity最完美的CharacterController 3d角色控制器,实现移动、跳跃、下蹲、奔跑、上下坡、物理碰撞效果,复制粘贴即用

这里我就直接贴出代码了

人物移动控制

[RequireComponent(typeof(CharacterController))]
public class MovementScript : MonoBehaviour
{[Tooltip("角色控制器")] public CharacterController characterController;[Tooltip("重力加速度")] private float Gravity = -19.8f;private float horizontal;private float vertical;[Header("移动")][Tooltip("角色行走的速度")] public float walkSpeed = 6f;[Tooltip("角色奔跑的速度")] public float runSpeed = 9f;[Tooltip("角色移动的方向")] private Vector3 moveDirection;[Tooltip("当前速度")] private float speed;[Tooltip("是否奔跑")] private bool isRun;[Header("地面检测")][Tooltip("是否在地面")] private bool isGround;[Header("跳跃")][Tooltip("角色跳跃的高度")] public float jumpHeight = 8f;private float _verticalVelocity;void Start(){speed = walkSpeed;}void Update(){horizontal = Input.GetAxis("Horizontal");vertical = Input.GetAxis("Vertical");//地面检测isGround = characterController.isGrounded;SetSpeed();SetRun();SetMove();SetJump();}//速度设置void SetSpeed(){if (isRun){speed = runSpeed;}else{speed = walkSpeed;}}//控制奔跑void SetRun(){if (Input.GetKey(KeyCode.LeftShift)){isRun = true;}else{isRun = false;}}//控制移动void SetMove(){moveDirection = transform.right * horizontal + transform.forward * vertical; // 计算移动方向moveDirection = moveDirection.normalized; // 归一化移动方向,避免斜向移动速度过快  }//控制跳跃void SetJump(){bool jump = Input.GetButtonDown("Jump");if (isGround){// 在着地时阻止垂直速度无限下降if (_verticalVelocity < 0.0f){_verticalVelocity = -2f;}if (jump){_verticalVelocity = jumpHeight;}}else{//随时间施加重力_verticalVelocity += Gravity * Time.deltaTime;}characterController.Move(moveDirection * speed * Time.deltaTime + new Vector3(0.0f, _verticalVelocity, 0.0f) * Time.deltaTime);}
}

视角控制

public class MouseLook : MonoBehaviour
{// 鼠标灵敏度public float mouseSensitivity = 500f;// 玩家的身体Transform组件,用于旋转public Transform playerBody;// x轴的旋转角度float xRotation = 0f;void Start(){// 锁定光标到屏幕中心,并隐藏光标Cursor.lockState = CursorLockMode.Locked;}// Update在每一帧调用void Update(){// 执行自由视角查看功能FreeLook();}// 自由视角查看功能的实现void FreeLook(){// 获取鼠标X轴和Y轴的移动量,乘以灵敏度和时间,得到平滑的移动速率float mouseX = Input.GetAxis("Mouse X") * mouseSensitivity * Time.deltaTime;float mouseY = Input.GetAxis("Mouse Y") * mouseSensitivity * Time.deltaTime;//限制旋转角度在-90到90度之间,防止过度翻转xRotation = Mathf.Clamp(xRotation, -90f, 90f);// 累计x轴上的旋转量xRotation -= mouseY;// 应用摄像头的x轴旋转transform.localRotation = Quaternion.Euler(xRotation, 0f, 0f);// 应用玩家身体的y轴旋转playerBody.Rotate(Vector3.up * mouseX);}
}

效果
在这里插入图片描述

简单的背包系统和物品交互

对UI知识还不太懂的小伙伴可以看这篇基础篇文件:【Unity游戏开发教程】零基础带你从小白到超神30——UI组件和布局的使用

绘制背包UI

物品插槽背景框
在这里插入图片描述
物品插槽,可以把物品插槽做出预制体,后面好修改
在这里插入图片描述

准星图像和文本
在这里插入图片描述

脚本控制

物品信息脚本

public class Item : MonoBehaviour
{public new string name = "New Item";//物品名称[TextArea]public string description = "New Description";//物品描述public Sprite icon;//物品图标public int currentQuantity = 1;//物品当前数量public int maxQuantity = 16;//物品最大堆叠数量
}

背包插槽脚本

public class Slot : MonoBehaviour, IPointerEnterHandler, IPointerExitHandler
{public bool hovered; // 鼠标是否悬停在该槽位上的标志private Item heldItem; // 当前槽位持有的物品private Color opaque = new Color(1, 1, 1, 1); // 不透明颜色private Color transparent = new Color(1, 1, 1, 0); // 透明颜色private Image thisSlotImage; // 该槽位的图像组件public TMP_Text thisSlotQuantityText; // 用于显示物品数量的文本组件// 初始化槽位public void initialiseSlot(){thisSlotImage = gameObject.GetComponent<Image>();thisSlotQuantityText = transform.GetChild(0).GetComponent<TMP_Text>();thisSlotImage.sprite = null;thisSlotImage.color = transparent;setItem(null);}// 设置槽位中的物品public void setItem(Item item){heldItem = item;if (item != null){thisSlotImage.sprite = heldItem.icon;thisSlotImage.color = opaque;updateData();}else {thisSlotImage.sprite = null;thisSlotImage.color = transparent;updateData();}}// 获取当前槽位持有的物品public Item getItem(){return heldItem;}// 当前槽位是否持有的物品public bool hasItem(){return heldItem ? true : false;}// 更新槽位显示的数据public void updateData(){if (heldItem != null) // 如果持有物品thisSlotQuantityText.text = heldItem.currentQuantity.ToString(); // 显示物品的数量else // 如果不持有物品thisSlotQuantityText.text = "";}// 当鼠标指针进入槽位区域时调用public void OnPointerEnter(PointerEventData pointerEventData){hovered = true;}// 当鼠标指针离开槽位区域时调用public void OnPointerExit(PointerEventData pointerEventData){hovered = false;}
}

库存系统脚本

public class Inventory : MonoBehaviour
{[Header("UI")]public GameObject inventory; // 游戏中的背包界面public List<Slot> allInventorySlots = new List<Slot>(); // 所有的槽位列表public List<Slot> inventorySloats = new List<Slot>();//背包的的槽位列表public Image crosshair; // 准星图像public TMP_Text itemHoverText; // 当中心悬停在物品上时显示物品名称的文本[Header("射线检测")]public float raycastDistance = 5f; // 射线检测的距离public LayerMask itemLayer; // 射线检测的目标层,用于识别物品public void Start(){toggleInventory(false); // 初始时关闭背包界面//合并槽位allInventorySlots.AddRange(inventorySloats);foreach (Slot uiSlot in allInventorySlots) // 初始化所有槽位{uiSlot.initialiseSlot();}}public void Update(){itemRaycast(Input.GetKeyDown(KeyCode.E)); // 显示物品名称和按E拾取物品if (Input.GetKeyDown(KeyCode.Tab)) // 按下tab键切换背包界面的显示状态toggleInventory(!inventory.activeInHierarchy);}private void itemRaycast(bool hasClicked = false){itemHoverText.text = ""; // 默认不显示任何物品名称Ray ray = Camera.main.ScreenPointToRay(crosshair.transform.position); // 从准星位置发出射线RaycastHit hit;if (Physics.Raycast(ray, out hit, raycastDistance, itemLayer)) // 如果射线检测到物品层的对象{if (hit.collider != null){if (hasClicked) // 如果是按了操作,尝试捡起物品{Item newItem = hit.collider.GetComponent<Item>();if (newItem){addItemToInventory(newItem); // 将物品添加到背包中}}else // 否则,仅获取物品名称以显示{Item newItem = hit.collider.GetComponent<Item>();if (newItem){itemHoverText.text = newItem.name; // 显示物品名称}}}}}//将物品添加到背包中private void addItemToInventory(Item itemToAdd){int leftoverQuantity = itemToAdd.currentQuantity; // 剩余需要添加到背包的物品数量Slot openSlot = null; // 记录一个空的槽位for (int i = 0; i < allInventorySlots.Count; i++) // 遍历所有槽位{Item heldItem = allInventorySlots[i].getItem();if (heldItem != null && itemToAdd.name == heldItem.name) // 如果槽位中有相同名称的物品{int freeSpaceInSlot = heldItem.maxQuantity - heldItem.currentQuantity; // 计算槽位中的剩余空间if (freeSpaceInSlot >= leftoverQuantity) // 如果剩余空间足够{heldItem.currentQuantity += leftoverQuantity; // 添加物品到该槽位Destroy(itemToAdd.gameObject); // 销毁场景中的物品对象allInventorySlots[i].updateData(); // 更新槽位显示的数据return;}else // 如果剩余空间不足{heldItem.currentQuantity = heldItem.maxQuantity; // 填满当前槽位leftoverQuantity -= freeSpaceInSlot; // 更新剩余需要添加的物品数量}}else if (heldItem == null) // 如果槽位为空{if (!openSlot)openSlot = allInventorySlots[i]; // 记录第一个空槽位}allInventorySlots[i].updateData(); // 更新槽位显示的数据}if (leftoverQuantity > 0 && openSlot) // 如果还有剩余物品且找到了空槽位{openSlot.setItem(itemToAdd); // 将物品添加到空槽位itemToAdd.currentQuantity = leftoverQuantity; // 更新物品的数量itemToAdd.gameObject.SetActive(false); // 隐藏场景中的物品对象}else{itemToAdd.currentQuantity = leftoverQuantity; // 更新物品的数量}}private void toggleInventory(bool enable){//关闭背包时,关闭所有鼠标悬停在该槽位上的标志if (!enable){foreach (Slot curSlot in allInventorySlots){curSlot.hovered = false;}}inventory.SetActive(enable); // 根据参数显示或隐藏背包界面Cursor.lockState = enable ? CursorLockMode.None : CursorLockMode.Locked; // 根据背包界面的状态锁定或解锁鼠标指针Cursor.visible = enable; // 设置鼠标指针的可见性// 禁用或启用相机的旋转控制Camera.main.GetComponent<MouseLook>().enabled = !enable;}
}

物品挂载Item脚本,配置参数,记得添加碰撞体并修改图层为Item
在这里插入图片描述
背包插槽挂载Slot脚本
在这里插入图片描述
角色上挂载Inventory脚本
在这里插入图片描述

拾取
在这里插入图片描述

源码

源码不出意外的话我会放在最后一节

完结

赠人玫瑰,手有余香!如果文章内容对你有所帮助,请不要吝啬你的点赞评论和关注,以便我第一时间收到反馈,你的每一次支持都是我不断创作的最大动力。当然如果你发现了文章中存在错误或者有更好的解决方法,也欢迎评论私信告诉我哦!

好了,我是向宇,https://xiangyu.blog.csdn.net

一位在小公司默默奋斗的开发者,出于兴趣爱好,最近开始自学unity,闲暇之余,边学习边记录分享,站在巨人的肩膀上,通过学习前辈们的经验总是会给我很多帮助和启发!php是工作,unity是生活!如果你遇到任何问题,也欢迎你评论私信找我, 虽然有些问题我也不一定会,但是我会查阅各方资料,争取给出最好的建议,希望可以帮助更多想学编程的人,共勉~

在这里插入图片描述

这篇关于【制作100个unity游戏之25】3D背包、库存、制作、快捷栏、存储系统、砍伐树木获取资源、随机战利品宝箱1(附带项目源码)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/702800

相关文章

Nginx实现高并发的项目实践

《Nginx实现高并发的项目实践》本文主要介绍了Nginx实现高并发的项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录使用最新稳定版本的Nginx合理配置工作进程(workers)配置工作进程连接数(worker_co

golang获取prometheus数据(prometheus/client_golang包)

《golang获取prometheus数据(prometheus/client_golang包)》本文主要介绍了使用Go语言的prometheus/client_golang包来获取Prometheu... 目录1. 创建链接1.1 语法1.2 完整示例2. 简单查询2.1 语法2.2 完整示例3. 范围值

Vue项目的甘特图组件之dhtmlx-gantt使用教程和实现效果展示(推荐)

《Vue项目的甘特图组件之dhtmlx-gantt使用教程和实现效果展示(推荐)》文章介绍了如何使用dhtmlx-gantt组件来实现公司的甘特图需求,并提供了一个简单的Vue组件示例,文章还分享了一... 目录一、首先 npm 安装插件二、创建一个vue组件三、业务页面内 引用自定义组件:四、dhtmlx

javaScript在表单提交时获取表单数据的示例代码

《javaScript在表单提交时获取表单数据的示例代码》本文介绍了五种在JavaScript中获取表单数据的方法:使用FormData对象、手动提取表单数据、使用querySelector获取单个字... 方法 1:使用 FormData 对象FormData 是一个方便的内置对象,用于获取表单中的键值

SpringBoot项目注入 traceId 追踪整个请求的日志链路(过程详解)

《SpringBoot项目注入traceId追踪整个请求的日志链路(过程详解)》本文介绍了如何在单体SpringBoot项目中通过手动实现过滤器或拦截器来注入traceId,以追踪整个请求的日志链... SpringBoot项目注入 traceId 来追踪整个请求的日志链路,有了 traceId, 我们在排

Rust中的Drop特性之解读自动化资源清理的魔法

《Rust中的Drop特性之解读自动化资源清理的魔法》Rust通过Drop特性实现了自动清理机制,确保资源在对象超出作用域时自动释放,避免了手动管理资源时可能出现的内存泄漏或双重释放问题,智能指针如B... 目录自动清理机制:Rust 的析构函数提前释放资源:std::mem::drop android的妙

部署Vue项目到服务器后404错误的原因及解决方案

《部署Vue项目到服务器后404错误的原因及解决方案》文章介绍了Vue项目部署步骤以及404错误的解决方案,部署步骤包括构建项目、上传文件、配置Web服务器、重启Nginx和访问域名,404错误通常是... 目录一、vue项目部署步骤二、404错误原因及解决方案错误场景原因分析解决方案一、Vue项目部署步骤

golang内存对齐的项目实践

《golang内存对齐的项目实践》本文主要介绍了golang内存对齐的项目实践,内存对齐不仅有助于提高内存访问效率,还确保了与硬件接口的兼容性,是Go语言编程中不可忽视的重要优化手段,下面就来介绍一下... 目录一、结构体中的字段顺序与内存对齐二、内存对齐的原理与规则三、调整结构体字段顺序优化内存对齐四、内

如何利用Java获取当天的开始和结束时间

《如何利用Java获取当天的开始和结束时间》:本文主要介绍如何使用Java8的LocalDate和LocalDateTime类获取指定日期的开始和结束时间,展示了如何通过这些类进行日期和时间的处... 目录前言1. Java日期时间API概述2. 获取当天的开始和结束时间代码解析运行结果3. 总结前言在J

配置springboot项目动静分离打包分离lib方式

《配置springboot项目动静分离打包分离lib方式》本文介绍了如何将SpringBoot工程中的静态资源和配置文件分离出来,以减少jar包大小,方便修改配置文件,通过在jar包同级目录创建co... 目录前言1、分离配置文件原理2、pom文件配置3、使用package命令打包4、总结前言默认情况下,