20240116-【UNITY 学习】增加滑动功能

2024-01-17 20:20

本文主要是介绍20240116-【UNITY 学习】增加滑动功能,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

替换脚本PlayerMovement_02.cs

在这里插入图片描述

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class PlayerMovement_03 : MonoBehaviour
{private float moveSpeed; // 玩家移动速度public float walkSpeed = 7; // 行走速度public float sprintSpeed = 10; // 冲刺速度public float slideSpeed = 30; // 滑动速度private float desiredMoveSpeed; // 期望的移动速度private float lastDesiredMoveSpeed; // 上一次的期望移动速度public float speedIncreaseMultiplier = 1.5f; // 速度增加倍数public float slopeIncreaseMultiplier = 2.5f; // 斜坡增加倍数public float groundDrag = 5; // 地面时的阻力public float playerHeight = 2; // 玩家身高public LayerMask whatIsGround; // 地面的LayerMaskprivate bool grounded; // 是否在地面上public float jumpForce = 6; // 跳跃力度public float jumpCooldown = 0.25f; // 跳跃冷却时间public float airMultiplier = 0.4f; // 空中移动速度衰减private bool readyToJump = true; // 是否可以跳跃public float crouchSpeed = 3.5f; // 蹲伏时的移动速度public float crouchYScale = 0.5f; // 蹲伏时的Y轴缩放比例private float startYScale; // 初始Y轴缩放比例public float maxSlopAngle = 40; // 最大坡度角度private RaycastHit slopeHit; // 坡度检测的射线信息private bool exitingSlope = true; // 是否正在离开坡度public KeyCode jumpKey = KeyCode.Space; // 跳跃键public KeyCode sprintKey = KeyCode.LeftShift; // 冲刺键public KeyCode crouchKey = KeyCode.LeftControl; // 下蹲键public Transform orientation; // 玩家朝向的Transformprivate float h; // 水平输入private float v; // 垂直输入private Vector3 moveDirection; // 移动方向private Rigidbody rb; // 玩家刚体public MovementState state; // 当前玩家的移动状态public enum MovementState{walking,    // 行走sprinting,  // 冲刺crouching,  // 蹲伏sliding,    // 滑动air         // 空中}public bool sliding; // 是否正在滑动private void Start(){rb = GetComponent<Rigidbody>();rb.freezeRotation = true; // 防止刚体旋转startYScale = transform.localScale.y;  // 记录初始的Y轴缩放}private void Update(){grounded = Physics.Raycast(transform.position, Vector3.down, playerHeight * 0.5f + 0.2f, whatIsGround);MyInput();SpeedControl();StateHandler();if (grounded)rb.drag = groundDrag;elserb.drag = 0;}private void FixedUpdate(){MovePlayer();}private void MyInput(){// 获取水平和垂直输入h = Input.GetAxisRaw("Horizontal");v = Input.GetAxisRaw("Vertical");// 如果按下跳跃键且准备好跳,并且在地面上if (Input.GetKey(jumpKey) && readyToJump && grounded){readyToJump = false;Jump();Invoke(nameof(ResetJump), jumpCooldown);}if (Input.GetKeyDown(crouchKey)){// 调整玩家缩放以模拟蹲下效果transform.localScale = new Vector3(transform.localScale.x, crouchYScale, transform.localScale.z);rb.AddForce(Vector3.down * 5f, ForceMode.Impulse);}// 如果释放下蹲键if (Input.GetKeyUp(crouchKey)){// 恢复到原始Y轴缩放transform.localScale = new Vector3(transform.localScale.x, startYScale, transform.localScale.z);}}private void MovePlayer(){// 根据朝向计算移动方向moveDirection = orientation.forward * v + orientation.right * h;// 如果在斜坡上并且不是即将离开斜坡if (OnSlope() && !exitingSlope){// 在斜坡上施加力,以便更好地移动rb.AddForce(GetSlopeMoveDirection(moveDirection) * moveSpeed * 20f, ForceMode.Force);// 如果垂直速度为正(上升),则额外施加向下的力,以克服斜坡引起的垂直速度变慢if (rb.velocity.y > 0){rb.AddForce(Vector3.down * 80f, ForceMode.Force);}}else if (grounded) // 如果在地面上{rb.AddForce(moveDirection.normalized * moveSpeed * 10f, ForceMode.Force); // 在地面上施加移动力}else if (!grounded) // 如果在空中{// 在空中施加移动力,乘以空中移动速度衰减系数rb.AddForce(moveDirection.normalized * moveSpeed * 10f * airMultiplier, ForceMode.Force);}// 根据是否在斜坡上决定是否启用重力rb.useGravity = !OnSlope();}private void SpeedControl(){// 如果在斜坡上并且不是即将离开斜坡if (OnSlope() && !exitingSlope){// 如果速度的大小超过了设定的移动速度if (rb.velocity.magnitude > moveSpeed){// 将速度归一化,并乘以设定的移动速度,以限制速度在设定范围内rb.velocity = rb.velocity.normalized * moveSpeed;}}// 如果不在斜坡上else{// 获取水平方向的速度Vector3 flatVel = new Vector3(rb.velocity.x, 0f, rb.velocity.z);// 如果水平速度的大小超过了设定的移动速度if (flatVel.magnitude > moveSpeed){// 限制水平速度在设定范围内Vector3 limitedVel = flatVel.normalized * moveSpeed;// 更新刚体的速度,保持垂直速度不变rb.velocity = new Vector3(limitedVel.x, rb.velocity.y, limitedVel.z);}}}private void Jump(){exitingSlope = true;//rb.velocity = new Vector3(rb.velocity.x, 0f, rb.velocity.z);rb.velocity = Vector3.zero;// 添加向上的力以实现跳跃rb.AddForce(transform.up * jumpForce, ForceMode.Impulse);}private void ResetJump(){readyToJump = true;exitingSlope = false;}private void StateHandler(){if (sliding){state = MovementState.sliding;  // 设置当前状态为滑动状态if (OnSlope() && rb.velocity.y < 0.1f){desiredMoveSpeed = slideSpeed;  // 如果在斜坡上并且垂直速度小于0.1,则设置期望移动速度为滑动速度}else{desiredMoveSpeed = sprintSpeed;  // 否则,设置期望移动速度为冲刺速度}}// 如果按住蹲伏键else if (Input.GetKey(crouchKey)){// 设置当前状态为蹲伏状态state = MovementState.crouching;// 设置移动速度为蹲伏速度desiredMoveSpeed = crouchSpeed;}// 如果在地面上并且按住冲刺键else if (grounded && Input.GetKey(sprintKey)){// 设置当前状态为冲刺状态state = MovementState.sprinting;// 设置移动速度为冲刺速度desiredMoveSpeed = sprintSpeed;}// 如果在地面上但没有按住冲刺键else if (grounded){// 设置当前状态为行走状态state = MovementState.walking;// 设置移动速度为行走速度desiredMoveSpeed = walkSpeed;}// 如果不在地面上else{// 设置当前状态为空中状态state = MovementState.air;}if (Mathf.Abs(desiredMoveSpeed - lastDesiredMoveSpeed) > 4f && moveSpeed != 0){StopAllCoroutines();  // 停止所有协程StartCoroutine(SmoothlyLerpMoveSpeed());  // 启动平滑插值移动速度的协程}else{moveSpeed = desiredMoveSpeed;  // 否则,直接将移动速度设置为期望移动速度}lastDesiredMoveSpeed = desiredMoveSpeed;  // 更新上一次的期望移动速度}public bool OnSlope(){// 使用射线检测当前位置向下,获取击中信息存储在slopeHit中if (Physics.Raycast(transform.position, Vector3.down, out slopeHit, playerHeight * 0.5f + 0.3f)){// 计算斜坡的角度float angle = Vector3.Angle(Vector3.up, slopeHit.normal);// 如果角度小于最大允许斜坡角度且不等于0,表示在斜坡上return angle < maxSlopAngle && angle != 0;}// 如果没有击中信息,或者角度不符合条件,表示不在斜坡上return false;}public Vector3 GetSlopeMoveDirection(Vector3 direction){// 使用Vector3.ProjectOnPlane将移动方向投影到斜坡法线上,然后进行归一化return Vector3.ProjectOnPlane(direction, slopeHit.normal).normalized;}private IEnumerator SmoothlyLerpMoveSpeed(){float time = 0;  // 记录经过的时间float difference = Mathf.Abs(desiredMoveSpeed - moveSpeed);  // 计算期望移动速度与当前移动速度的差值float startValue = moveSpeed;  // 记录开始时的移动速度while (time < difference){moveSpeed = Mathf.Lerp(startValue, desiredMoveSpeed, time / difference);  // 使用插值平滑地改变移动速度if (OnSlope()){float slopeAngle = Vector3.Angle(Vector3.up, slopeHit.normal);  // 计算当前坡度的角度float slopeAngleIncrease = 1 + (slopeAngle / 90f);  // 根据坡度角度增加速度// 根据时间、速度增加倍数、坡度增加倍数进行平滑插值time += Time.deltaTime * speedIncreaseMultiplier * slopeIncreaseMultiplier * slopeAngleIncrease;}else{// 在平地上,只考虑时间和速度增加倍数time += Time.deltaTime * speedIncreaseMultiplier;}yield return null;  // 等待下一帧}moveSpeed = desiredMoveSpeed;  // 最终将移动速度设置为期望移动速度}
}

新增脚本Sliding.cs

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class Sliding : MonoBehaviour
{public Transform orientation;  // 玩家方向的Transformpublic Transform playerObj;   // 玩家对象的Transform,用于在滑动期间调整缩放private Rigidbody rb;          // 玩家的刚体组件private PlayerMovement_03 pm_03; // PlayerMovement_03脚本的引用public float maxSlideTime = 0.75f; // 滑动的最大持续时间public float slideForce = 200;     // 滑动期间施加的力private float slideTimer;          // 用于跟踪滑动持续时间的计时器public float slideYScale = 0.5f;   // 滑动期间玩家的Y轴缩放private float startYScale;          // 玩家的初始Y轴缩放public KeyCode slideKey = KeyCode.F; // 启动滑动的按键private float h;  // 水平输入private float v;  // 垂直输入private void Start(){rb = GetComponent<Rigidbody>();pm_03 = GetComponent<PlayerMovement_03>();startYScale = playerObj.localScale.y;}private void Update(){h = Input.GetAxisRaw("Horizontal");v = Input.GetAxisRaw("Vertical");// 检查是否按下滑动键且存在水平或垂直输入if (Input.GetKeyDown(slideKey) && (h != 0 || v != 0)){StartSlide(); // 启动滑动}// 检查是否释放了滑动键且玩家当前正在滑动if (Input.GetKeyUp(slideKey) && pm_03.sliding){StopSlide(); // 停止滑动}}private void FixedUpdate(){// 检查玩家当前是否在滑动if (pm_03.sliding){SlidingMovement(); // 处理滑动运动}}private void StartSlide(){pm_03.sliding = true; // 在PlayerMovement_03脚本中设置滑动标志// 调整玩家的缩放以创建蹲伏效果playerObj.localScale = new Vector3(playerObj.localScale.x, slideYScale, playerObj.localScale.z);// 应用向下的力以模拟蹲伏rb.AddForce(Vector3.down * 5f, ForceMode.Impulse);slideTimer = maxSlideTime; // 初始化滑动计时器}private void SlidingMovement(){Vector3 inputDirection = orientation.forward * v + orientation.right * h; // 计算输入方向// 检查玩家是否不在斜坡上或向上移动if (!pm_03.OnSlope() || rb.velocity.y > -0.1f){// 在滑动期间在输入方向上施加力rb.AddForce(inputDirection.normalized * slideForce, ForceMode.Force);slideTimer -= Time.deltaTime; // 减少滑动计时器}else{// 在斜坡上滑动时根据斜坡的方向调整力rb.AddForce(pm_03.GetSlopeMoveDirection(inputDirection) * slideForce, ForceMode.Force);}// 检查滑动持续时间是否已过期if (slideTimer <= 0){StopSlide(); // 停止滑动}}private void StopSlide(){pm_03.sliding = false; // 在PlayerMovement_03脚本中重置滑动标志// 将玩家的缩放恢复到初始大小playerObj.localScale = new Vector3(playerObj.localScale.x, startYScale, playerObj.localScale.z);}
}

这篇关于20240116-【UNITY 学习】增加滑动功能的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

【学习笔记】 陈强-机器学习-Python-Ch15 人工神经网络(1)sklearn

系列文章目录 监督学习:参数方法 【学习笔记】 陈强-机器学习-Python-Ch4 线性回归 【学习笔记】 陈强-机器学习-Python-Ch5 逻辑回归 【课后题练习】 陈强-机器学习-Python-Ch5 逻辑回归(SAheart.csv) 【学习笔记】 陈强-机器学习-Python-Ch6 多项逻辑回归 【学习笔记 及 课后题练习】 陈强-机器学习-Python-Ch7 判别分析 【学

系统架构师考试学习笔记第三篇——架构设计高级知识(20)通信系统架构设计理论与实践

本章知识考点:         第20课时主要学习通信系统架构设计的理论和工作中的实践。根据新版考试大纲,本课时知识点会涉及案例分析题(25分),而在历年考试中,案例题对该部分内容的考查并不多,虽在综合知识选择题目中经常考查,但分值也不高。本课时内容侧重于对知识点的记忆和理解,按照以往的出题规律,通信系统架构设计基础知识点多来源于教材内的基础网络设备、网络架构和教材外最新时事热点技术。本课时知识