《游戏-02_2D-开发》

基于《游戏-01_2D-开发》,

继续制作游戏:

首先给人物添加一个2D重力效果

在编辑的项目设置中,

可以看出unity默认给的2D重力数值是-9.81,模拟现实社会中的重力效果

下方可以设置帧率

而Gravity Scale代表 这个数值会 * 重力 

还需要将玩家添加碰撞器Collider,而一般人物均添加胶囊碰撞器Capsule Collider 2D

在编辑场景中人物身上的绿色圆圈就是碰撞范围,我们进行修改

修改成与人物差不多大小

Shift + F 可以快速帮我们找到需要用到的物体

在场景中找到这个物体

添加不规则复合碰撞器有三步

勾选合成为1块 Used By Composite

最后将场景重力效果设置为静态

我们在运行测试前拖动玩家位置

运行可见玩家通过重力落在了地面

我们避免人物因为场景碰撞体的不均匀而摔倒,需要冻结玩家屏幕z坐标

我们还可以通过设置Mass来设定玩家的质量,如果玩家的质量大于怪物就可以撞开怪物

在重力设置中可以在碰撞检测方式上不选间歇性Discrete 而选择持续Continuous

运行即下落

接下来我们开始配置新输入系统

在编写代码之前,我们首先在项目管理中创建两个文件夹,

表示脚本下的玩家文件夹

在玩家管理文件夹中创建一个代码

命名为PlayerController

旧输入系统已经淘汰,因为移植性不好,跨平台性差,例如移植到手机或xbox平台很复杂

所以我们需要用到新输入系统

打开项目设置

找到Other Settings

我们可以修改Api,.NET Framework可以让我们利用更多的c#特性

【切记】:接下来我们要保存好项目,因为接下来的设置新输入系统会自动重启unity项目

我们在Player 的 Other Settings 的 Active Input Handing 中将输入管理Old设置为Both

注意:在点击Apply之前一定要先保存好项目

更新成新输入系统后我们开始使用新输入系统

我们在Windows下的Package Manager下找到新输入系统

选择Unity Registry 在右侧输入Input 即可看到新输入系统点击下载

创建文件夹

创建Input Actions

命名为InputControls

双击我们就可以打开新输入系统了

将新输入系统放置Scene右侧便于操作,创建一个Action map

Action Maps提供选择控制哪些物品操作的途径,

命名为运动Movement

设置

添加 上/下/左/右 绑定

删除无功能选项

设置

重命名WASD

键盘设置完成之后如果还想设置手柄操作的话

可以继续设置

这样就成功实现了两种输入系统

当然除了手动设置新操作系统也可以系统默认生成

首先在Player下增加组件Player Input

点击新建新输入系统

选择刚刚设置好的路径

选择替换

再次双击打开就可以看见系统为我们创建的新输入系统了

节省了很多设置操作

当然我们需要修改这些系统生成的设置比如,

确认设置成了Any  后面会修改它的设置

创建好新输入系统后我们就可以把Player Input这个组件删除了 ,因为我们想通过生成代码的方式实现

要求系统自动生成代码

双击打开代码

编写代码,

这里InputControls 类型 应与自定义新输入系统名字相符合

注意:

函数周期表,

awake > enable > start > physicsUpdate > Update > fixedUpdate > diable > destroy 

人物在编辑场景取消右上角显示勾选时,就会执行OnDisable()函数

勾选启动时就会执行OnEnable()这个函数

编写代码,

代码中的inputDirection可以存储一个新输入系统中设定的Vector2变量,从而代码控制新输入系统

编写代码:

定义公开变量inputControl用来存储我们在

下的【InputControls【自定义新输入系统类型】】,

定义公开变量rb存储用来控制玩家移动的【Rigidbody2D【刚体类型】】,

定义公开变量inputDirection存储用来二维转向的【Vector2【二维向量类型】】,

定义公开变量speed存储用来控制移动速度的【float【字符类型】】,

最先调用Awake()函数创建new新输入系统类型内存,再获取刚体组件分别分配给inputControl与rb变量,

然后调用OnEnable()函数与OnDisable()函数中的inputControl的Enabe()与Disable区别是否持续按键,

然后调用Update()函数,获取新输入系统中的二维向量读值,

最后调用FixedUpdate()函数中的Moved函数

Moved()函数中(int)transform.locakScale.x是将transform变量转换成int值

其.locakScale.x即

语句int faceDir = (int)transform.localScale.x; 即将x的缩放比例转换成值存储到faceDir变量中

因为键盘判断左方向与右方向是通过1与-1设定的所以通过

transform.localScale = new Vector3(facwDir,1,1);

其中

是根据

而设定的

根据逻辑判断设定转向

if(inputDirection.x > 0)

    faceDir = 1;

if(inputDirection.x < 0)

    faceDir = -1;

最后设定刚体的速度

rb.velocity = new Vector2(inputDirection.x * speed * Time.deltaTime,rb.velocity.y);

using UnityEngine;
public class PlayerController : MonoBehaviour{
    public  InputControls inputControl;
    public Rigidbody2D rb;
    public Vector2 inputDirection;
    public float speed;
    private void Awake(){
        inputControl = new InputControls();
        rb = GetComponent<Rigidbody2D>(); 
    }
    private void OnEnable(){
        inputControl.Enable();
    }
    private void OnDisable(){
        inputControl.Disable();
    }
    private void Update(){
        inputDirection = inputControl.Player.Move.ReadValue<Vector2>();
    }
    private void FixedUpdate(){
        Moved();
    }
    private void Moved() {
        int faceDir = (int)transform.localScale.x;
        if (inputDirection.x > 0)
            faceDir = 1;
        if (inputDirection.x < 0)
            faceDir = -1;
        transform.localScale = new Vector3(faceDir, 1, 1);
        //键盘
        rb.velocity = new Vector2(inputDirection.x * speed * Time.deltaTime,rb.velocity.y);
    }
}
注意这里

先判断转向,再进行翻转

运行即完成了翻转及水平移动

接下来要完成跳跃同样,我们需要在新输入系统中自己设定

修改代码,为代码在unity中易读,可在代码中添加特性

保存代码我们在unity中即可看到显示中文 

我们首先添加一个 float类型 的 跳跃力 字段

我们想对纵轴施加一个 jumpForce跳跃力

我们需要修改代码:

using UnityEngine;
using UnityEngine.InputSystem;
public class PlayerController : MonoBehaviour{
    public  InputControls inputControl;
    public Rigidbody2D rb;
    public Vector2 inputDirection;
    [Header("基本参数")]
    public float speed;
    public float jumpForce;
    private void Awake(){
        inputControl = new InputControls();
        rb = GetComponent<Rigidbody2D>();
        inputControl.Player.Jump.started += Jump;
    }
    private void Jump(InputAction.CallbackContext obj){
        rb.AddForce(transform.up * jumpForce, ForceMode2D.Impulse);
    }

    private void OnEnable(){
        inputControl.Enable();
    }
    private void OnDisable(){
        inputControl.Disable();
    }
    private void Update(){
        inputDirection = inputControl.Player.Move.ReadValue<Vector2>();
    }
    private void FixedUpdate(){
        Moved();
    }
    private void Moved() {
        int faceDir = (int)transform.localScale.x;
        if (inputDirection.x > 0)
            faceDir = 1;
        if (inputDirection.x < 0)
            faceDir = -1;
        transform.localScale = new Vector3(faceDir, 1, 1);
        rb.velocity = new Vector2(inputDirection.x * speed * Time.deltaTime,rb.velocity.y);
    }
}
【解析】:

        玩家输入系统中我们添加了公开的float类型字段用来存储一个跳跃时纵方向获取的数值,

在调用Awake()函数时 首先获取新输入系统的Jump 然后 打点. 获取 Jump下的 started,

注意:按住可以执行 performed ,为了快速很精准执行一次 我们采取用started 表示按键按下的那一刻 执行了

一般对于单次执行的函数我们放进started中进行,因为started是一个事件方法,所以我们需要添加一个事件注册函数,而注册的内容 就用 += 加号等号 这个符号进行注册,

inpitControl.Player.Jump.started += Jump;

意思就是把 Jump这个函数 添加到 started 按键按下的那一刻来执行,

然后我们可以看到Jump报红,

我们选中Jump 按Alt + Enter 键即可弹出选项我们点击生成Jump方法

即可生成一个带参数的Jump函数我们修改函数中的内容

表示这个Jump函数在按键按下要被执行时 利用刚体rb 打点. 添加一个力,因为这个rb刚体的类型是 Rigidbody2D 所以括号中有两个参数,其中transform.up 表示 世界坐标的上方向,我们通过 * 乘以这个跳跃数值,第二个参数ForceMode2D.Impulse 代表 这个力是瞬时的

Impulse 瞬时力

写好代码,我们通过unity场景中的Player公开的跳跃力字段输入合适的值进行设定找到一个舒服合理的跳跃程度

运行实现跳跃

但这其中还有很多不合理的地方,比如跳跃后松手就会垂直下落,跳跃中前方有障碍物会卡住,在跳跃中可以持续跳跃等,

接下来我们在代码中进行修改,完善成合理的跳跃逻辑

首先我们先解决连续跳跃的问题:

我们新建脚本 物理检查PhysicsCheck.cs

编写代码:

using UnityEngine;
public class PhysicsCheck : MonoBehaviour{
    public bool isGround;
    public float checkRadius;
    public LayerMask groundLayer;
    private void Update(){
        Check();
    }
    public void Check(){
        isGround = Physics2D.OverlapCircle(transform.position, checkRadius, groundLayer);
    }
}
【解析】:

        定义一个公开bool类型变量存储到 isGround 变量中,用来判断是否接触到地面,

定义一个公开float类型变量存储到 checkRadius 变量中,用来表示碰撞范围,

定义一个公开 LayerMask层级面具变量存储到 groundLayer 变量中,用来表示对哪个层进行操作,

这里需要在unity场景中手动添加Layer 层:

保存代码后,在unity场景中可见

这两部分的层级是相同的,

我们首先添加层级

设置层级

设置作用层级

这样通过代码就可以影响 被设置的Ground层级的 地面了

语句表示 将进行物理2D 的 OverlapCircle 叠层圆形 检测

检测三个方面 

transform.position 检测点

checkRadius 检测范围

groundLayer 检测层级

这样我们回到unity中将检测半径设为 0.2

再将监测点设为脚下

运行即可看到 跳跃前 接触地面的选项被自动 勾选

跳跃后 接触地面的选项被自动 取消

PhysicsCheck脚本写好之后我们进行组合,

用到写好的类PhyisicsCheck脚本这个类型 就可以拿到这个类的所有公开变量与公开方法

【很重要】:拿到之后需要在Awake()函数中激活,

using UnityEngine;
using UnityEngine.InputSystem;
public class PlayerController : MonoBehaviour{
    public  InputControls inputControl;
    public Rigidbody2D rb;
    public Vector2 inputDirection;
    public PhysicsCheck physicsCheck;
    [Header("基本参数")]
    public float speed;
    public float jumpForce;
    private void Awake(){
        inputControl = new InputControls();
        rb = GetComponent<Rigidbody2D>();
        inputControl.Player.Jump.started += Jump;

        physicsCheck = GetComponent<PhysicsCheck>();

    }
    private void Jump(InputAction.CallbackContext obj){
        if(physicsCheck.isGround)
            rb.AddForce(transform.up * jumpForce, ForceMode2D.Impulse);
    }
    private void OnEnable(){
        inputControl.Enable();
    }
    private void OnDisable(){
        inputControl.Disable();
    }
    private void Update(){
        inputDirection = inputControl.Player.Move.ReadValue<Vector2>();
    }
    private void FixedUpdate(){
        Moved();
    }
    private void Moved() {
        int faceDir = (int)transform.localScale.x;
        if (inputDirection.x > 0)
            faceDir = 1;
        if (inputDirection.x < 0)
            faceDir = -1;
        transform.localScale = new Vector3(faceDir, 1, 1);
        rb.velocity = new Vector2(inputDirection.x * speed * Time.deltaTime,rb.velocity.y);
    }
}

我们回到unity场景中运行游戏,即可发现连续跳跃问题解决了

当然我们的触碰范围 = 0.2 没有在unity中显示,我们可以自己用代码画出来

using UnityEngine;
public class PhysicsCheck : MonoBehaviour{
    public bool isGround;
    public float checkRadius;
    public LayerMask groundLayer;
    public Vector2 bottomOffset;
    private void Update(){
        Check();
    }
    public void Check(){
        isGround = Physics2D.OverlapCircle(
            (Vector2)transform.position + bottomOffset, checkRadius, groundLayer);
    }
    private void OnDrawGizmosSelected(){
        Gizmos.DrawWireSphere(
            (Vector2)transform.position + bottomOffset, checkRadius
        );
    }
}

调用OnDrawGizmosSelected()函数将以

为中心点  以

为半径在人物脚下显示碰撞范围

【解析】

定义一个公开 Vector2二维向量类型的变量存储到 bottomOffset变量中,用作偏移量

在Check()检测函数中 将偏移量bottomOffset 与 检测点位置相加,

注意:因为检测点是三维向量 所以 二维向量 与 三维向量相加需要用到强制转换 (Vector2) 强制转换为二维向量Vector2类型

最后通过OnDrawGizmosSelected()函数将 检测点画出来

注意:Gizmos就是辅助线等小控件可以选择是否显示

这里调用OnDrawGizmosSelected()函数就是选择性将小控件的哪个控件画出来,

显而易见

Gizmos.DrawWireSphere(
            (Vector2)transform.position + bottomOffset, checkRadius
        );

我们画出来的是 检测范围

在unity场景中通过改变位移差值x/y可以改变检测范围的位置

接下来我们优化跳跃中不会贴在前方障碍物这个问题,

首先在unity场景中创建一个文件夹PhysicsMaterials,

然后创建一个光滑的材质,

起名为Normal

这里表示摩擦力是0.4

我们只需要把摩擦力改为0,就创建了一个光滑的材质

将光滑材质放进玩家的 胶囊碰撞器组件 的 Material框选中

即解决了跳跃中贴前方障碍物的问题

最后运行

实现完整的逻辑跳跃

End

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

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

相关文章

// doesn‘t exist

- // doesnt exist 13.3 赋给派生类引用,将发生什么情况呢?派生类引用能够为基对象调用派生类方法,这样做将出现问题。例 如,将RatedPlayer :: Rating()方法用于TableTennisPlayer对象是没有意义的,因为TableTennisPlayer对象没 有rating成员。 如果基类引用和指针可以指向…

webpack 中的loader 和plugin的区别

Loader: 作用&#xff1a; Loader 用于在模块加载时对文件进行转换。它是一个转换器&#xff0c;将文件从一种形式转换为另一种形式&#xff0c;例如&#xff0c;将 ES6 语法的 JavaScript 文件转换为能够在浏览器中运行的普通 JavaScript。使用场景&#xff1a; Loader通常被配…

Addressables(2) ResourceLocation和AssetReference

IResourceLocation var op Addressables.LoadResourceLocationsAsync(key); var result op.WaitForCompletion(); 把加载的Key塞进去&#xff0c;不难看出&#xff0c;IResourceLocation可以用来获得资源的详细信息 很适合用于更新分析&#xff0c;或者一些检查工具 AssetR…

Eureka使用详解

介绍主要特点主要功能与常用服务注册中心的比较Eureka与Zookeeper的区别和联系Eureka与Nacos的区别与联系Eureka与Consul的区别与联系 安装部署Eureka与CAP理论Eureka实现实时上下线Eureka常用注解Eureka架构模式 介绍 Eureka是一个基于REST的服务&#xff0c;主要用于AWS云中…

logstack 日志技术栈-05-windows10 安装 Elasticsearch elasticsearch-8.11.1 实战笔记

安装 Elasticsearch elasticsearch-8.11.1 下载 访问 Elasticsearch 下载页面 解压下载的压缩文件到你选择的目录。 运行 进入 Elasticsearch 目录&#xff0c;运行 bin/elasticsearch.bat 启动 Elasticsearch。 验证 elaasticsearch的默认访问路径是localhost:9200&…

HTTP 认证方式

目录 1.HTTP认证方式2.Python中 HTTP 认证方式的使用 本文主要介绍HTTP 认证方式有哪些及在Python中的基本使用方式。 1.HTTP认证方式 HTTP认证方式主要有以下几种&#xff1a; 基本认证&#xff08;Basic Authentication&#xff09;&#xff1a;客户端将用户名和密码进行B…

一文读懂「RAG,Retrieval-Augmented Generation」检索增强生成

Retrieval-Augmented Generation&#xff08;RAG&#xff09;作为机器学习和自然语言处理领域的一大创新&#xff0c;不仅代表了技术的进步&#xff0c;更在实际应用中展示了其惊人的潜力。 RAG结合了检索&#xff08;Retrieval&#xff09;和生成&#xff08;Generation&#…

Flutter 页面嵌入 Android原生 View

前言 文章主要讲解Flutter页面如何使用Android原生View&#xff0c;但用到了Flutter 和 Android原生 相互通信知识&#xff0c;建议先看完这篇讲解通信的文章 Flutter 与 Android原生 相互通信&#xff1a;BasicMessageChannel、MethodChannel、EventChannel-CSDN博客 数据观…

pdf拆分成各个小pdf的方法

背景:由于某些缘故,一个大的pdf需要拆分成页数少的pdf,或者pdf需要去掉指定页,那么就有必要对pdf进行重新编辑,这里需要用到一个库,直接进行操作即可。 当使用Python时,可以使用PyMuPDF库来拆分PDF文件。以下是一个示例代码, import fitz # PyMuPDF def split_pdf(i…

禅道的安装及使用流程

目录 一.安装 1.下载禅道安装包​http://www.zentao.net/ 2.选择禅道开源版 3.选择需要下载的安装包&#xff08;注意&#xff1a;下载的安装包必须放在根目录&#xff0c;不能移动&#xff09; 4.将下载的安装包双击进行解压&#xff1b;解压后的文件为d:\xampp&#xff1b…

sfml使用opengl着色器实现2d水面波浪

SFML中使用GLSL着色器来绘制水波。 效果 代码 #include <SFML/Graphics.hpp> #include <iostream>int main() {const int WIDTH = 800;

(南京观海微电子)——TCON介绍

TCON板详细介绍 TCON又称&#xff1a;逻辑板&#xff0c;控制板&#xff0c;在液晶电视里的作用和CRT中的视放板相当&#xff0c;但有本质的区别&#xff0c;逻辑板不是一个纯粹的信号放大器&#xff0c;它输入是LVDS格式信号&#xff0c;而不是RGB。逻辑板的作用是把数字板送来…

python进程间通信——命名管道(Named Pipe、FIFO)

文章目录 Python中的命名管道&#xff1a;深入理解进程间通信1. 命名管道简介2. 创建和删除命名管道3. 写入命名管道4. 读取命名管道5. 示例&#xff1a;进程间通信write_to_pipe.pyread_from_pipe.py测试运行 6. 注意事项和限制命名管道的半双工机制命名管道读写任意一方未打开…

记录一次QT乱码问题

问题描述 在敲陆文周的书《QT5开发及实例》的示例代码时&#xff0c;出现乱码&#xff0c;如下图所示 具体代码如下 Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);ui->treeWidget->clear();int groupSize 2;int ite…

Pyro —— Velocity Voxel Scale

Velocity Voxel Scale是H19.5引入的新参数&#xff0c;该参数可单独定义volume和速度体素&#xff1b;根据参数设置&#xff0c;可观察到模拟时间的显著变化&#xff1b; Velocity Voxel Scale对DOP和SOP均可用&#xff1b;对DOP设置&#xff0c;该参数在Smoke Object&#xf…

js 小数精确计算 (mathjs / decimal.js)

mathjs import { all, create } from "mathjs";let mathjs create(all, {precision: 14,number: "BigNumber", });/**加法 */ export function jia(left: any, right: any) {return mathjs.floor(Number(mathjs.add(mathjs.bignumber(left), mathjs.bign…

Docker(四)操作容器

作者主页&#xff1a; 正函数的个人主页 文章收录专栏&#xff1a; Docker 欢迎大家点赞 &#x1f44d; 收藏 ⭐ 加关注哦&#xff01; 操作 Docker 容器 容器是 Docker 又一核心概念。 简单的说&#xff0c;容器是独立运行的一个或一组应用&#xff0c;以及它们的运行态环境…

FreeType和HarfBuzz入门示例

最近在了解字体渲染的一些东西&#xff0c;其中不可避免的需要到这两个库。现在写个入门示例记录一下。 一、FreeType和HarfBuzz介绍 1.1 FreeType FreeType 是一个开源的字体引擎&#xff0c;它提供了一套用于渲染字体的 API。FreeType 支持多种字体格式&#xff0c;包括 True…

LLMs之Cheshire-Cat :Cheshire-Cat (Stregatto)的简介(构建自定义人工智能的框架)、安装、使用方法之详细攻略

LLMs之Cheshire-Cat &#xff1a;Cheshire-Cat (Stregatto)的简介(构建自定义人工智能的框架)、安装、使用方法之详细攻略 目录 Cheshire-Cat (Stregatto)的简介 1、文档和资源 2、为什么使用Cat Cheshire-Cat (Stregatto)的安装和使用方法 1、安装 2、最小插件示例 Che…

Centos使用Docker搭建自己的Gitlab(社区版和设置汉化、修改密码、设置SSH秘钥)

根据我的经验 部署Gitlab&#xff08;社区版&#xff09; 至少需要2核4g的服务器 带宽3~4M 1. 在自己电脑上安装终端&#xff1a;宝塔ssl终端 或者 FinalShell&#xff0c;根据喜好安装即可 http://www.hostbuf.com/t/988.html http://www.hostbuf.com/downloads/finalshell_w…