Unity3D仿星露谷物语开发46之种植/砍伐橡树
1、目标
种植一棵橡树,从种子变成大树。
然后可以使用斧头砍伐橡树。
2、删除totalGrowthDays字段
修改growthDays的含义,定义每个值为到达当前阶段的累加天数。此时最后一个阶段就是totalGrowthDays的含义。所以就可以删除totalGrowthDays字段。
(1)修改SO_CropDetailsList的配置信息
修改前的GrowthDays的配置信息:
修改后的GrowthDays的配置信息:
每个值换成了累加的方式。
(2)修改CropDetails.cs脚本
删除totalGrowthDays字段。
(3)修改GridPropertiesManager.cs脚本
找出目前所处的成长阶段换成如下的代码:
// 找出目前所处的成长阶段
for (int i = growthStages - 1; i >= 0; i--)
{if (gridPropertyDetails.growthDays >= cropDetails.growthDays[i]){currentGrowthStage = i;break;}
}
(3)修改GridCursor.cs脚本
Check if crop fully grown的逻辑修改如下:
3、修改CropStandard预制体的配置
修改CropSprite的Sprite Sort Point为Pivot。
修改CropHarvestedSprite的Sprite Sort Point为Pivot,Order in Layer为1。
调整完成后,收获的防风草会显示在Player之前,如下所示:
4、动画系统
已有的动画为:
查看TreeFallingLeft节点:
根级别的Sprite Renderer值为0,被禁用了。
子级别有Top和Trunk两个Sprite Renderer组件。
这就要求动画绑定的预设体有同样的结构。
5、创建橡树预设体
(1)创建CropTreeCanyonOak
在Hierarchy -> PersistentScene下新创建一个物体命名为CropTreeCanyonOak。
添加Box Collider 2D组件和Crop组件,相关配置信息如下:
(2)创建CropSprite
在CropTreeCanyonOak下创建子物体命名为CropSprite。
添加Sorting Group、Sprite Renderer、Animator组件,相应的配置信息如下:
双击Animator下的Controller(Tree),在Animation界面可以看到相关配置被黄色高亮显示,这表明animator识别不到相关的对象。
(3)创建Trunk
在CropSprite下创建物体命名为Trunk,添加组件信息如下:
Auto Tiling(自动贴片)的作用:要是你把 Box Collider 2D 组件添加到使用了瓦片精灵的 Sprite(精灵)上,并且开启了 Auto Tiling,这个组件就能够依据精灵的瓦片布局,自动对碰撞体的形状和大小进行调整。
- 贴合瓦片形状:它会按照精灵所使用的瓦片图集的边界,自动调整碰撞体的轮廓,让碰撞检测与精灵的视觉外观精准匹配。
- 处理多瓦片精灵:当精灵由多个瓦片组合而成时,Auto Tiling 能够自动生成一个覆盖所有瓦片的组合碰撞体,而无需你手动去调整每个瓦片的碰撞体。
- 提升效率:对于复杂的瓦片地图,开启 Auto Tiling 可以节省大量手动设置碰撞体的时间,避免逐个为瓦片添加碰撞体。
添加完成后,Animation下的Trunk立马置灰(正常状态)。
(4)创建Top
在CropSprite下创建子物体命名为Top,添加相关组件的配置信息如下:
添加完成后,Animation下的Top立马置灰(正常状态)。
(5)查看特效
点击CropSprite,点击Animation下的播放,可以看到特效。
(6)移动Prefab
将CropTreeCanyonOak从Hierarchy移动到Assets -> Prefabs -> Crop下。
并且删除Hierarchy下的CropTreeCanyonOak。
6、SO_CropDetailsList添加配置
因为Growth Prefab第5个Prefab已经有图片信息,所以Growth Sprite第5个不再需要图片信息。
7、运行游戏1
首先挖5块土,然后放入橡木种子。
走过未成熟的橡木时,树木会左右晃动。经过成熟的橡木后方时,会有褪色的效果。
接下来处理使用斧头砍伐成年橡树的逻辑。
8、修改GridCursor.cs脚本
修改IsCursorValidForTool函数,添加下面一行代码:
9、修改Player.cs脚本
在ProcessPlayerClickInput函数中添加一行代码:
在ProcessPlayerClickInputTool函数中,添加一个case如下:
case ItemType.Chopping_tool:if (gridCursor.CursorPositionIsValid){ChopInPlayerDirection(gridPropertyDetails, itemDetails, playerDirection);}break;
添加ChopInPlayerDirection函数如下:
private void ChopInPlayerDirection(GridPropertyDetails gridPropertyDetails, ItemDetails itemDetails, Vector3Int playerDirection)
{// Trigger animationStartCoroutine(ChopInPlayerDirectionRoutine(gridPropertyDetails, itemDetails, playerDirection));
}
添加ChopInPlayerDirectionRoutine函数如下:
private IEnumerator ChopInPlayerDirectionRoutine(GridPropertyDetails gridPropertyDetails, ItemDetails equippedItemDetails, Vector3Int playerDirection)
{PlayerInputIsDisabled = true;playerToolUseDisabled = true;// Set tool animation to axe in override animationtoolCharacterAttribute.partVariantType = PartVariantType.axe;characterAttributeCustomisationList.Clear();characterAttributeCustomisationList.Add(toolCharacterAttribute);animationOverrides.ApplyCharacterCustomisationParameters(characterAttributeCustomisationList);ProcessCropWithEquippedItemInPlayerDirection(playerDirection, equippedItemDetails, gridPropertyDetails);yield return useToolAnimationPause;// After animation pause yield return useToolAnimationPause;PlayerInputIsDisabled = false;playerToolUseDisabled = false;
}
修改ProcessCropWithEquippedItemInPlayerDirection如下,添加了对case ItemType.Chopping_tool的支持。
private void ProcessCropWithEquippedItemInPlayerDirection(Vector3Int playerDirection, ItemDetails equippedItemDetails, GridPropertyDetails gridPropertyDetails)
{switch (equippedItemDetails.itemType){case ItemType.Chopping_tool:if(playerDirection == Vector3Int.right){isUsingToolRight = true;}else if(playerDirection == Vector3Int.left){isUsingToolLeft = true;}else if(playerDirection == Vector3Int.up){isUsingToolUp = true;}else if(playerDirection == Vector3Int.down){isUsingToolDown = true;}break;case ItemType.Collecting_tool:if (playerDirection == Vector3Int.right){isPickingRight = true;}else if (playerDirection == Vector3Int.left){isPickingLeft = true;}else if (playerDirection == Vector3Int.up){isPickingUp = true;}else if (playerDirection == Vector3Int.down){isPickingDown = true;}break;case ItemType.none:break;}// Get crop at cursor grid locationCrop crop = GridPropertiesManager.Instance.GetCropObjectAtGridLocation(gridPropertyDetails);// Execute Process Tool Action For cropif(crop != null){switch (equippedItemDetails.itemType){case ItemType.Chopping_tool:crop.ProcessToolAction(equippedItemDetails, isUsingToolRight, isUsingToolLeft, isUsingToolDown, isUsingToolUp);break;case ItemType.Collecting_tool:crop.ProcessToolAction(equippedItemDetails, isPickingRight, isPickingLeft, isPickingDown, isPickingUp);break;}}
}
10、运行游戏2
存在一个问题:就是砍树的时候,用的不是斧头,而是锄头。这是因为使用了默认的Tool即Hoe。
之前,当玩家携带一个道具时,我们为动画创建了一些脚本对象资产,以便可以在携带道具和不携带之间切换。
现在,我们需要对use tool动画做类似的事情,以便在axe和hoe两个工具之间切换。
11、新建Animation Type
在Assets -> Scriptable Object Assets -> Animation -> Animation Types下新建tool目录,在其下再创建目录NoneAxe。
该目录右击,Create -> Scriptable Objects -> Animation -> Animation Type,重命名为ToolNoneAxeUseToolDown。
配置ToolNoneAxeUseToolDown的相关信息:
以同样的方式,分别创建ToolNoneAxeUseToolLeft、ToolNoneAxeUseToolRight、ToolNoneAxeUseToolUp。
比如ToolNoneAxeUseToolRight的参数如下:
在tool目录下再创建目录NoneHoe,相应地创建4个AnimationType,比如ToolNoneHoeUseToolDown如下:
在tool目录下再创建目录NoneScythe,相应地创建4个AnimationType,比如ToolNoneScytheSwingToolDown如下:
在tool目录下再创建目录NoneWateringCan,相应地创建4个AnimationType,比如ToolNoneWateringCanLiftToolUp如下:
新建完毕后,将上面16个添加到Player的CharacterCustomiser下:
添加完毕后,相应的数量从24变为40。
12、运行游戏3
再次砍树,发现正确使用了斧头这个工具
相关文章:

Unity3D仿星露谷物语开发46之种植/砍伐橡树
1、目标 种植一棵橡树,从种子变成大树。 然后可以使用斧头砍伐橡树。 2、删除totalGrowthDays字段 修改growthDays的含义,定义每个值为到达当前阶段的累加天数。此时最后一个阶段就是totalGrowthDays的含义。所以就可以删除totalGrowthDays字段。 &…...
STM32外设应用详解——从基础到高级应用的全面指南
目录 一、引言:为何选择STM32外设 二、主要外设类别与详细应用解析 1. GPIO(通用输入输出) 工作原理详解 高级应用设计 硬件连接建议 2. 定时器(TIM)详解 基本定时器原理 高级配置 实际应用 核心技巧 3. A…...
作业帮C++后台开发面试题及参考答案
Cookie 和 Session 的区别是什么? Cookie 和 Session 是 Web 开发中用于管理用户状态的两种机制,它们在存储位置、安全性、生命周期和数据类型等方面存在显著差异。 存储位置:Cookie 数据存储在客户端浏览器,而 Session 数据存储在服务器端。当浏览器向服务器发送请求时,…...
红队进阶实战
4.1 内网渗透(域渗透、横向移动) 域环境攻击链 初始立足点:通过钓鱼获取域用户凭据(如NTLM Hash)。信息收集: 使用BloodHound自动化分析域内关系。执行nltest /dclist:domain.com获取域控制器列表。横向移动: Pass-the-Hash:利用Mimikatz注入Hash到新会话。sekurlsa::…...
C语言中的指定初始化器
什么是指定初始化器? C99标准引入了一种更灵活、直观的初始化语法——指定初始化器(designated initializer), 可以在初始化列表中直接引用结构体或联合体成员名称的语法。通过这种方式,我们可以跳过某些不需要初始化的成员,并且可以以任意顺序对特定成员进行初始化。这…...
C/C++ 整数类型的长度
参考 cppreference.cn 在某些语言中,整数类型的长度是固定的,如java中 char 8short 16int 32long 64 可是C/C 与机器相关,整数类型长度与平台有关 先可以记一个简单的 按照C标准: char > 8short > 16int > 16long &g…...

gRPC开发指南:Visual Studio 2022 + Vcpkg + Windows全流程配置
前言 gRPC作为Google开源的高性能RPC框架,在微服务架构中扮演着重要角色。本文将详细介绍在Windows平台下,使用Visual Studio 2022和Vcpkg进行gRPC开发的完整流程,包括环境配置、项目搭建、常见问题解决等实用内容。 环境准备 1. 安装必要组…...

高密度服务器机柜散热方案:高风压风机在复杂风道中的关键作用与选型要点
随着云计算、人工智能等技术的飞速发展,数据中心内服务器机柜的集成度不断攀升,高密度部署成为常态。然而,高密度意味着单位空间内服务器数量剧增,发热量呈指数级上升,传统散热方案已难以满足需求。在复杂的机柜风道环…...
Android framework 问题记录
一、休眠唤醒,很快熄屏 1.1 问题描述 机器休眠唤醒后,没有按照约定的熄屏timeout 进行熄屏,很快就熄屏(约2s~3s左右) 1.2 原因分析: 抓取相关log,打印休眠背光 相关调用栈 //具体打印调用栈…...

框架之下再看HTTP请求对接后端method
在当今的软件开发中,各类框架如雨后春笋般不断涌现,极大地提升了开发效率。以 Java 开发为例,Spring 框架历经多次迭代演进,而 Spring Boot 更是将开发便捷性提升到了新高度。如今,开发者只需简单引入 Maven 包&#x…...
Oracle APEX IR报表列宽调整
目录 1. 问题:如何调整Oracle APEX IR报表列宽 2. 解决办法 1. 问题:如何调整Oracle APEX IR报表列宽 1-1. 防止因标题长而数据短,导致标题行的文字都立起来了,不好看。 1-2. 防止因数据太长而且中间还没有空格,把列…...

【笔记】与PyCharm官方沟通解决开发环境问题
#工作记录 2025年5月20日 星期二 背景 在此前的笔记中,我们提到了向PyCharm官方反馈了几个关于Conda环境自动激活、远程解释器在社区版中的同步问题以及Shell脚本执行时遇到的问题。这些问题对日常开发流程产生了一定影响,因此决定联系官方支持寻求解…...
深入解析:如何基于开源OpENer开发EtherNet/IP从站服务
一、EtherNet/IP协议概述 EtherNet/IP(Industrial Protocol)是一种基于以太网的工业自动化通信协议,它将CIP(Common Industrial Protocol)封装在标准以太网帧中,通过TCP/IP和UDP/IP实现工业设备间的通信。作为ODVA(Open DeviceNet Vendors Association)组织的核心协议…...

node.js文件系统(fs) - 创建文件、打开文件、写入数据、追加数据、读取数据、创建目录、删除目录
注意:以下所有示例均是异步语法! 注意:以下所有示例均是异步语法! 创建文件 node.js 允许我们在计算机本地创建文件,例如创建一个 word 文件: // 引入核心模块(fs) var fs require(fs)// API fs.writeF…...
SQL:MySQL函数:空值处理函数(NULL Handling Functions)
目录 什么是空值(NULL)? 常用空值处理函数总览 1️⃣ IFNULL() – 空值替换函数(If Null) 2️⃣ COALESCE() – 多参数空值判断(返回第一个非 NULL 值) 3️⃣ NULLIF() – 相等则返回 NULL…...

利用ffmpeg截图和生成gif
从视频中截取指定数量的图片 ffmpeg -i input.mp4 -ss 00:00:10 -vframes 1 output.jpgffmpeg -i input.mp4 -ss 00:00:10 -vframes 180 output.jpg -vframes 180代表截取180帧, 实测后发现如果视频是60fps,那么会从第10秒截取到第13秒-i input.mp4:指定输入视频文…...

初始化一个Springboot项目
初始化一个Springboot项目 文章目录 初始化一个Springboot项目1、新建项目2、配置yml3、自定义异常4、通用相应类5、全局跨域配置6、总结 1、新建项目 首先,我们需要创建一个新的 Spring Boot 项目。这里我们使用 IntelliJ IDEA 作为开发工具,它提供了方…...

YOLOv8在单目向下多车辆目标检测中的应用
大家读完觉得我有帮助记得关注!!! 摘要 自动驾驶技术正逐步改变传统的汽车驾驶方式,标志着现代交通运输的一个重要里程碑。目标检测是自主系统的基石,在提高驾驶安全性、实现自主功能、提高交通效率和促进有效的应急…...
23种设计模式解释+记忆
一、创建型模式(5种)—— “怎么造对象?” 单例模式(Singleton) 场景:公司的CEO只能有一个。 核心:确保一个类只有一个实例,全局访问。 关键词:唯一、全局访问。 工厂方…...

Baklib构建AI就绪型知识中台实践
Baklib驱动企业知识资产重构 在数字化转型浪潮中,企业知识中台的构建已成为激活数据价值的关键路径。Baklib通过结构化存储与智能分类引擎,将分散于邮件、文档、IM工具中的碎片化信息转化为可检索、可复用的数字资产。其核心能力体现在三个维度…...

JS逆向-某易云音乐下载器
文章目录 介绍下载链接Robots文件搜索功能JS逆向**函数a:生成随机字符串****函数b:AES-CBC加密****函数c:RSA公钥加密** 歌曲下载总结 介绍 在某易云音乐中,很多歌曲听是免费的,但下载需要VIP,此程序旨在“…...
FreeRTOS全攻略:从入门到精通
目录 一、FreeRTOS 基础概念1.1 FreeRTOS 是什么1.2 为什么选择 FreeRTOS 二、与裸机开发的区别2.1 任务管理2.2 中断处理2.3 资源管理 三、FreeRTOS 入门篇3.1 内存管理3.2 任务创建3.3 任务状态3.4 任务优先级3.5 空闲任务和钩子函数3.6 同步与互斥3.7 队列3.8 信号量3…...

服务器的基础知识
什么是服务器 配置牛、运行稳、价格感人的高级计算机,家用电脑不能比拟的。 服务器的组成:电源、raid卡、网卡、内存、cpu、主板、风扇、硬盘。 服务器的分类 按计算能力分类 超级计算机 小型机AIX x86服务器(服务器cpu架构) …...
AGI大模型(25):LangChain提示词模版
我们也可以创建prompt template, 并引入一些变量到prompt template中,这样在应用的时候更加灵活。 1 代码实现 # 我们也可以创建prompt template, 并引入一些变量到prompt template中,这样在应用的时候更加灵活 from langchain_core.prompts import ChatPromptTemplate from…...

Python连接redis
第一步安装redis Releases microsoftarchive/redis 安装时勾上所有能勾上的选项下一步即可 在CMD中pip install redis 安装redis pip install redis -i https://pypi.tuna.tsinghua.edu.cn/simple 配置redis 在redis安装目录下找到 修改 line 57 bind 0.0.0.0 line…...

使用exceljs将excel文件转化为html预览最佳实践(完整源码)
前言 在企业应用中,我们时常会遇到需要上传并展示 Excel 文件的需求,以实现文件内容的在线预览。经过一番探索与尝试,笔者最终借助 exceljs 这一库成功实现了该功能。本文将以 Vue 3 为例,演示如何实现该功能,代码示例…...

前端面经12 函数柯里化
<script>function sum(num){return function(num2){return numnum2}}console.log(sum(1)(2))</script>面试考察 只要参数够了 达到某个数量就输出 <script>let nums[]function sum(...args){nums.push(...args)if(nums.length>5){const out (nums.slice…...
企业级 Hosts 自动化管理实战:基于 HTTP 检测的高可用域名解析方案
摘要 本文针对企业级域名解析稳定性需求,提供一套从IP 检测到Hosts 更新的完整自动化解决方案。通过 HTTP 状态码检测、权威 DNS 解析、原子化文件操作等核心技术,结合多行业真实案例,详细阐述方案设计、脚本实现与生产部署,帮助…...

告别蜘蛛池!PHP 打造你的网站专属蜘蛛导航仪
在网站优化的赛道上,吸引搜索引擎蜘蛛来访一直是站长和开发者关注的重点。以往借助蜘蛛池、软件等工具引蜘蛛,不仅存在成本高、易违规的风险,效果也参差不齐。现在,有一种更高效、更安全的方式 —— 利用 PHP 代码,无需…...

ubuntu kubeasz 部署高可用k8s 集群
ubuntu kubeasz 部署高可用k8s 集群 测试环境主机列表软件清单kubeasz 部署高可用 kubernetes配置源配置host文件安装 ansible 并进行 ssh 免密登录:下载 kubeasz 项⽬及组件部署集群部署各组件开始安装修改 config 配置文件增加 master 节点增加 kube_node 节点登录dashboard…...