Pixel Adventure Unity2D开发完整指南

本文参考:2-2. Get and Setup Assets_哔哩哔哩_bilibili

1、下载资源

在Asset Store中下载Pix Adventure1 +2的资源:

在import的时候,不用到Scene import进来,如下图所示,Scenes目录反勾选一下。

两个资源都下载完成后,则下图所示:

在Assets下创建目录名为Graphiics,然后把Pixel Adventure1和Pixel Adventure2下Assets中的内容全部都copy到Graphics目录下,如下图所示:

2、Unity使用基础说明

(1)CSharp知识

1)virtual和abstract的区别

abstract方法只能声明在抽象类中,同时不能有方法体,必须被子类重写

virtual方法可以声明在任何非密封类中,它可以有方法体,可以被子类选择性重写

2)使用父类的方法

base.func()

(2)Collider 

我们经常使用的Collider是Box Collider,这个是矩形的碰撞体。实际上还有Circle Collider。这样,针对不同的物体形状,我们可以选择不同的Collider。

(3)Debug模式

在Normal模式下,我们看到的是对象所有组件的Inspector信息,如下:

在Debug模式下,我们可以看到组件中各个变量的详细信息,所以debug时该模式非常有用。

(4)SerializeField字段

在C#脚本中

如果我们希望公开某个变量,那么可以使用public进行标识

如果我们不希望公开某个变量,那么可以使用private进行标识

如果我们不希望公开某个变量,但是在Inspector中又可以看到该变量的值,那么可以使用private进行标识,同时加上[SerializeField]的标识,示例如下:

[SerializeField] private float speed;

(5)Header字段

用于标记下方在Inspector中显示字段的header名称,效果如下:

(6)ContextMenu字段

创建类的选项,本身继承自MonoBehaviour。

比如在脚本中创建ContextMenu信息如下:

  [ContextMenu("Debug here")]private void print(){Debug.Log("here!");}

此时当脚本挂载到某个对象后,右击该脚本,即可看到"Debug here"的选项,点击该选项即可执行对应的函数。

(7)Camera方式让Game视野变大

如果Game中物体过大,一种做法是分别缩小每个物体,如果物体特别多操作起来特别麻烦。另一种做法是,修改Camera的size大小,值越大视野也就越大。

(8)Flip左右翻转的方法

transform.Rotate(0, 180, 0);

通过rb.velocity.x判断朝向,如果该值小于0则朝左,否则朝右。

完整方法:

(9)例程(类似进程)

启动例程:StartCoroutine(func()), func的返回值为IEnumerator类型。

使用例程的好处:单独开进程,不会阻塞当前主进程。

 比如例程中含有sleep时间,主线程就不用卡在那里了。

等待的写法:yield return new WaitForSeconds(xx);

例程举例:

假如两个例程存在相互干扰,那么可以在例程开始时停止已有的所有例程,使用StopAllCoroutines()方法。

比如:

(10)物体移动的方式

方法1: transform.position = new Vector2(pos.x, pos.y);

方法2:transform.position = Vector2.MoveTowards(pos1, pos2, moveSpeed * Time.deltaTime);

(11)给物体施加力量rb.AddForce

方法1:AddForce(Vector3 force),使用Vector3类型参数,可以分别向刚体按Vector3对象指定x/y/z分量施加力。使用该方法会对刚体施加一个持续的力。rb.AddForce(new Vector3(10f, 0f, 0f));

方法2:AddForce(x, y, x),与方法1类似,只是把Vector3的3个值拆开了。

方法3:AddForce(Vector3 force, ForceMode mode),其中force是施加力的矢量,参数mode是一个枚举类型的参数,用于指定力的模式。模式有:

  • ForceMode.Force:施加一个持续的力
  • ForceMode.Impulse:施加一个瞬间的冲击力
  • ForceMode.Acceleration:施加一个持续的加速度
  • ForceMode.VelocityChange:施加一个改变刚体速度的力

方法4:AddForce(x, y, z, ForceMode mode),与方法3类似,只是把Vector3的3个值拆开了。

(12)创建预设体

GameObject newObj = Instantiate(prefab, position, Quaternion.identity);

其中,prefab也是GameObject的对象

position是Vector3的位置信息

Quaternion.identity表示新复制的对象不会有初始旋转,假如使用prefab.transform.rotation表示新复制的对象会保留原来的旋转。

(13)Invoke委托函数

它是U3D的一种委托机制。

Invoke("SendMsg", 5); 意思是:5秒钟之后调用SendMsg()方法。

注意:

1)只能在脚本的生命周期里的(Start、Update、OnGUI、FixedUpdate、LateUpdate)中被调用

2)Invoke不接受含有参数的方法

3)当Time.ScaleTime = 0 时 invoke()方法无效

(14)Raycast检测光线

Physics2D.Raycast是用于检测2D光线命中图层的函数,可以用于检测光线是否与指定图层中的物体相交,并返回相交点的信息。

常见的使用方法:

RaycastHit2D hit = Physics2D.Raycast(rayOrigin, rayDirection, rayLength, layerMask);

rayOrigin:光线的起点

rayDirection:光线的方向

rayLength:光线的长度

layerMask:需要检测的图层

(15)设置物体间的碰撞矩阵

在Edit -> Project Settings -> Physics 2D -> Layer Collision Matrix

通过设置该矩阵,可以让反选的两个物体之间不会发生碰撞。

3、Animator和Animation的使用

(1)创建Idle动画

Animation用来将图片生成动画,Animator可控制动画的状态。

首先,在Assets下创建Animations的目录,并在该目录下右击创建Animator Controller,如下图所示:

其次,在SampleScene中Create Empty,命名为Player。将Graphics -> Main Characters -> Virtual Guy中Idle的一张图片拖入Scene,该对象重命名为Animator,并移到Player的下一级。同时将上一步的Animator Controller拖到Animator下作为组件。如下图所示:

此时,双击Animator组件的Controller的Player,会进入Animator的编辑界面。

点击Window -> Animation -> Animation,就出来Animation的编辑界面。

点击Create,重命名为PlayerIdle,此时在Animator界面中会自动出现PlayerIdle的节点,并且与Entry相连。

然后将Assets的Graphics下所有的Idle图片拖入Animation中,点击播放就可以看到Idle状态下的动画了。

播放动画时会发现对象运动过快,此时需要调整Sample Rate的值,但是目前Animation界面中没有操作该值的选项。需要点击Animation界面右上角的三个点,然后点击"Show Sample Rate"。

默认情况下Samples的值为60,我们调整到15.

(2)创建Move动画

在Animation界面,点击PlayerIdle下的Create NewClip,并且重命名为PlayerMove。

  ->   

然后,把Run的所有动画拖入Animation中,并且设置Samples的值为15。

此时在Animator下会有两个节点,然后分别右击两个节点Make Transition。

点击Animator的Parameters,添加变量isRunning。后续通过该变量实现两个动画之间的切换。

然后设置Transition的条件,Idle -> Move的isRunning为true,Move -> Idle的isRunning为false。

并且反选Has Exist Time。如下图所示:

(3)编写脚本

创建一个Squre,然后挂载Box Collider 2D的组件。

给Animator对象挂载Rigidbody 2D + Capsule Collider 2D这两个组件,Rigidbody 2d-> Constraints -> Freeze Rotation 勾选Z,否则物体移动时会滚动着移动。

然后在Assets下创建Scripts的目录,并创建AnimatorPlay的C#脚本,挂载到Animator对象下,完整的C#脚本如下:

using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class AnimatorPlay : MonoBehaviour
{Rigidbody2D rb;public float x;Animator anim;private void Awake(){rb = GetComponent<Rigidbody2D>();anim = GetComponent<Animator>();}void Start(){}void Update(){x = Input.GetAxisRaw("Horizontal");anim.SetBool("isRunning", rb.velocity.x != 0);rb.velocity = new Vector2(x, rb.velocity.y);}
}

(4)创建Blend Tree混合树

混合树:就是将多个动画混合在一起,通过阈值控制动画的变换。也需要通过变量控制别的节点transition到该Blend Tree。

首先,点中Animator这个角色,然后在Animation下Create New Clip,命名为PlayerJump如下:

然后把Jump的图片拖进去,只有一帧。

同样的方法创建PlayerFall,然后把Fall的图片拖进去,同样也只有一帧。

接着,在Animator界面中,将其他Animator动画都删除掉,只剩一个PlayerIdle,右击Create State -> From New Blend Tree。

创建的Blend Tree重命名为Jump/Fall。

双击该Blend Tree进入编辑界面

1)增加Parameters:yVelocity

2)在右边Inspector -> Parameter中选择yVelocity

3)点击"+",Add Motion Field,分别把PlayerJump和PlayerFall添加进去

4)反勾选Automate Thresholds,然后修改PlayerJump和PlayerFall的Threshold分别为-1和1

5)增加一个isGround的Bool变量,通过该变量控制PlayerIdle到Jump/Fall之间的切换,到isGround=false时变为Jump/Fall,否则变为PlayerIdle。

6)代码中控制transition的变化(isGound的判断并不是很严谨,只是演示使用)

using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;public class AnimatorPlay : MonoBehaviour
{Rigidbody2D rb;public float x;Animator anim;Boolean isGround;private void Awake(){rb = GetComponent<Rigidbody2D>();anim = GetComponent<Animator>();}void Start(){}void Update(){if (Input.GetKeyDown(KeyCode.Space)){rb.velocity = new Vector2(rb.velocity.x, 5);}anim.SetFloat("yVelocity", rb.velocity.y);anim.SetBool("isGround", isGround);if(rb.velocity.y != 0){isGround = false;}else{isGround = true;}}}

(5)触发类型Trigger

该类型没有任何状态值,通过trigger可以直接触发某个动画。

演示:

1)创建PlayerKnockback的动画,并把所有的Hit动画拖入其中,设置Samples为12。

2)在Animator中,创建Any State 到PlayerKnockback的transition,条件中反勾选“Can Transition to S”(不能重新回到原来状态,否则会导致循环了),设置Transition Duration为0。

增加knockback的trigger类型变量,在transition中也指定一下该变量。

3)在Animator中,创建PlayerKnockback到PlayerIdle的transition,设置Exit Time为1(表示knockback的动画需要存在1秒钟),设置Transition Duration为0。

4)脚本中编写使用逻辑:

在Update()函数中,

if(Input.GetKeyDown(KeyCode.K)){

        Knockback():

}

编写和Update平级的函数Knockback:

public void Knockback(){

        anim.SetTrigger("knockback");

}

(6)Add Event

在Animation界面,可以选择一帧Add Event,如下图所示:

add Event的函数只能选择当前gameobject挂载脚本中的方法。

4、Tile Palette瓦片调色板

(1)打开Tile Palette

Window -> 2D -> Tile Palette,点击Create New Palette。

将新的Palette命名为:Tile Palette

保存到Assets的新建的Tile Palette目录下:

(2)使用Tile Palette

打开Assets -> Graphics,选择Terrain Slice -> Sprite Editor

点击后可以看到各种切片好的图片:

选择Pixels Per Unit为16,然后点击Apply。

将Terrain Slice拖入到Tile Palette 下,然后选择保存目录为新建的 Assets -> Tile Paletter -> Assets:

此时,在Tile Palette界面可以看到切片后的的图片,同时在Assets -> Tile Palette -> Assets下也可以看到一个个切片后的图片。

在Hierachry下2D Object -> Tilemap -> Rectangular,此时在Hierachry下会出现Grid -> Tilemap,重命名为Ground。

点击Brush画笔,在Scene界面左击即可添加元素。

按Shift左击可以删除元素

(3)创建基于Tile的组件

1)添加Tilemap Collider 2D组件

ps:Tilemap:瓦片地图

此时角色可以站在我们创建的Ground之上。

但是此时存在一个问题,就是Ground上创建了很多的Collider检测器如下:

我们需要合并这些Collider以提高性能。

2)添加Composite Collider 2D组件

添加完之后会自动添加了Rigidbody 2D组件,

在Rigidbody 2D -> Body Type中选择Static,表明这个对象不需要动态计算刚体。

然后在Tilemap Collider 2D的Used By Composite打勾,此时Collider就会合并。

对比如下:

合并前:

合并后:

(4)创建Background

在Assets -> Graphics -> Background下选择图片,并设置Sprite mode为multiple, Pixels Per Unit为16.

点击Sprite Editor, Slice -> Grid By Cell Size,并设置Pixel Size为16 * 16,与之前的参数吧保持一致。

将切片后的图片拖入Tile Palette,然后根据提示保存到Assets -> Tile Palette -> Assets下。

在Hierachry的Grid下创建新的Tilemap命名为Background。

选中背景图,然后点击paint,即可在scene中画出一条。

选择Pick,然后在Scene中选中那一条,即可整条的进行draw,而不用一块一块的。

如下,整条进行draw。

画完之后,之前的ground不见了,只剩下角色和背景了。

(5)调整显示优先级

在Hierachy中点击Ground,点击Tilemap Renderer,在Sorting Layer中,点击Add Sorting Layer。

新增3个Sorting Layers如下:

Layer值越大,显示时越靠前。

然后Animator角色、Ground、Background分别选择对应的Sorting Layer,就可以正确显示如下:

5、GameManager

GameManager游戏管理器,可以做很多的事情:

  • 可以用于获取玩家的参考
  • 用于重生玩家
  • 记录您收集的水果或硬币的分数
  • 重新启动游戏、保存游戏

我们需要在Hierachy中Create Empty命名为***GameManager***,加*号是为了区分普通的对象。同时创建GameManager的脚本挂载到该对象下。

正常情况下,在一个游戏中我们只需要一个GameManager。

6、创建类秋千的SpikedBall

(1)创建秋千的支点

将Terrain的一个图标拖入scene

设置Scale大小均为0.5,并且设置Sorting Layer为Ground

在Hierarchy中重命名为Base

在这个过程中,点击下方小按钮即可让xyz的数值保持一致,修改其中一个,其他两个同步修改。

添加组件Distance Joint 2D,则会自动添加Rigidbody 2D的组件。

设置Rigidbody的Body Type为Static,表示支点不可移动。

(2)创建Empty Parent并命名为SpikedBall

这个是秋千的根元素,而秋千实际上是由多个部件组成的。

(3)插入秋千元素SpikedBall

选择Assets -> Graphics -> Traps -> Spiked Ball,设置Pixels Per Unit为16

将Spiked Ball拖入Scene中,并使其成为SpikedBall的子节点

同时新增Rigidbody 2D组件 ,设置Collision Detection和interpolate的值如下。

(4)配置支点和ball之间的铰链

点击Base,然后将Spiked Ball拖入Base的Distance Joint 2D组件的Connected Rigid Body中,使得Base和Spiked Ball形成距离关节。

然后执行程序,在Scene界面拉一下Spiked Ball,便可使其左右摆动

(5)通过脚本初始化Spiked Ball的摆动

在此之前,程序执行时需要手工拉一下ball才能使其摆动,现在我们希望通过在脚本初始化时推一下ball使其可以自行进行摆动。

public class SpikedBall : MonoBehaviour
{[SerializeField] private Rigidbody2D rb;[SerializeField] private float pushForce;void Start(){Vector2 pushVector = new Vector2(pushForce, 0);rb.AddForce(pushVector, ForceMode2D.Impulse);}}

(6)增加秋千的链子

同样在Traps -> Spiked Ball下

设置Chain的Pixels Per Unit 为16,拖入Scene中,并且再复制3份,将这4个Chain同时拖到Hierarchy的SpikedBall父节点下方。

给这四个Chain同时添加Hinge Joint 2D组件。

然后依次设置Joint 2D中Connected Rigid Body的值:

Base设置Chain

Chain设置Chain(1)

Chain(1)设置Chain(2)

Chain(2)设置Chain(3)

Chain(3)设置Spiked Ball

此时再次执行程序,可以看到Spiked Ball带动Chain一起摆动了。

(7)Joint之间的区别

可以参考:Unity 2DJoint 物理关节功能与总结_target joint 2d-CSDN博客

Distance Joint距离关节:该关节使得两物体保持一定的距离。

Hinge Joint铰链关节:允许rb对象围绕空间中的点火另一个对象上的点旋转的关节,常用于绳子模拟、开门等铰链。

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

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

相关文章

Unity 使用 NewtonSoft Json插件报错

JsonReaderException: Unexpected character encountered while parsing value: . Path , line 0, position 0. 通过断点发现&#xff0c;头有一串ZWNBSP&#xff0c;这个是BOM格式的JSON。在文件下看不到。 解决方法&#xff1a;改编码格式&#xff0c;Remove BOM.

(回溯) LeetCode 51. N 皇后

原题链接 一. 题目描述 按照国际象棋的规则&#xff0c;皇后可以攻击与之处在同一行或同一列或同一斜线上的棋子。 n 皇后问题 研究的是如何将 n 个皇后放置在 nn 的棋盘上&#xff0c;并且使皇后彼此之间不能相互攻击。 给你一个整数 n &#xff0c;返回所有不同的 n 皇后…

腾讯云AI代码助手:智能AI代码助手 ,新一代的高效代码开发辅助工具

前言 近些年是一个科技大爆发的时代&#xff0c;自从大模型发布以来越来越多的科技产品出现。例如去年的智能编码助手自出现以来&#xff0c;各大老牌大厂腾讯&#xff0c;百度 阿里也都紧随其后&#xff0c;智能编码助手的出现可以说大大的节省了我们写一些冗余代码的时间成本…

十七、访问者模式

文章目录 1 基本介绍2 案例2.1 Element 接口2.2 Vehicle 抽象类2.3 Car 类2.4 Jeep 类2.5 VehicleCollection 类2.6 Action 抽象类2.7 Repair 类2.8 Drive 类2.9 Client 类2.10 Client 类的运行结果2.11 总结 3 各角色之间的关系3.1 角色3.1.1 Element ( 元素 )3.1.2 ConcreteE…

靓图!多点创新!CEEMDAN-Kmeans-VMD-CNN-LSTM-Attention双重分解+卷积长短期+注意力多元时间序列预测

靓图&#xff01;多点创新&#xff01;CEEMDAN-Kmeans-VMD-CNN-LSTM-Attention双重分解卷积长短期注意力多元时间序列预测 目录 靓图&#xff01;多点创新&#xff01;CEEMDAN-Kmeans-VMD-CNN-LSTM-Attention双重分解卷积长短期注意力多元时间序列预测效果一览基本介绍程序设计…

LVS 调度器 nat和DR模式

lvs-nat 修改请求报文的目标IP,多目标IP的DNAT 配置网络 LVS主机 注意网卡的顺序 &#xff08;nat和主机模式&#xff09; [rootlvs ~]# cat /etc/NetworkManager/system-connections/ens160.nmconnection [connection] idens160 typeethernet interface-nameens160 ​ [ip…

Linux使用学习笔记3 系统运维监控基础

系统运维监控类命令 查询每个进程的线程数 for pid in $(ps -ef | grep -v grep|grep "systemd" |awk {print $2});do echo ${pid} > /tmp/a.txt;cat /proc/${pid}/status|grep Threads > /tmp/b.txt;paste /tmp/a.txt /tmp/b.txt;done|sort -k3 -rn for pid…

数据结构与算法-16高级数据结构_图论(图论基础)

图论基础 1 什么是图 1.1 基础定义 图&#xff08;Graph&#xff09;是一个用于描述一组对象之间关系的数学结构。这些对象被称为顶点&#xff08;Vertex&#xff09;&#xff0c;也称为节点&#xff08;Node&#xff09;或点&#xff08;Point&#xff09;&#xff0c;而对…

2024国赛Word论文模板【一键生成式操作】

一、比赛介绍 该竞赛创办于1992年&#xff0c;每年一届&#xff0c;是首批列入“高校学科竞赛排行榜”的19项竞赛之一。2023年&#xff0c;来自全国及美国、澳大利亚、马来西亚的1685所院校/校区、59611队(本科54158队、专科5453队)、近18万人报名参赛。 而今年的国赛马上就要…

【CTF | WEB】001、攻防世界WEB题目之backup

文章目录 backup题目描述:解题思路&#xff1a;解题过程&#xff1a; backup 题目描述: X老师忘记删除备份文件&#xff0c;他派小宁同学去把备份文件找出来,一起来帮小宁同学吧&#xff01; 进入题目后显示&#xff1a; 解题思路&#xff1a; 在进行网站安全检查时&#xf…

网络协议四 物理层,数据链路层

从这一节开始学习 五层模型。学习方法是从最底层物理层开始学习 七层模型 五层模型 各个层用的协议&#xff0c;以及加上协议后的称谓 各个层的作用 应用层&#xff1a;可以认为是原始数据&#xff0c;该数据称为 报文&#xff0c;用户数据。 运输层&#xff1a;也叫传输层&am…

全网超详细攻略——LVS原理详解及部署

目录 一、LVS原理 1.LVS简介 2.LVS结构 3.IP负载均衡技术 4.LVS相关术语 二、LVS负载均衡四种工作模式 1.LVS-DR模式 2.LVS-NAT模式 3.LVS-TUN模式&#xff08;了解&#xff09; 4.FULL-NAT模式&#xff08;了解&#xff09; 三、LVS负载均衡十种调度算法 四、LVS部…

米思奇安装——Mac版本

米思奇安装——Mac版本 1.下载 访问米思奇官网https://mixly.org/bnu-maker/mixl2.0rc 打开官网后在首页点击导航栏的软件平台&#xff0c;选择Mixly离线版 点击Mixly2.0RC4发布下载。 进入百度网盘分享的文件&#xff0c;选择Mac一键更新版本&#xff0c;等待下载完成。 …

尚品汇-ES(三十一)

目录&#xff1a; &#xff08;1&#xff09;封装搜索相关实体对象 &#xff08;2&#xff09;搜索接口封装 &#xff08;3&#xff09;在service-list-client模块添加远程接口 &#xff08;1&#xff09;封装搜索相关实体对象 搜索参数实体&#xff1a;SearchParam 搜索参…

第七节 流编辑器sed(stream editor)(7.1)

一,sed简介 sed是一种流编辑器,处理时,把当前处理的行存储在临时缓冲区中,称为模式空间,接着用sed命令处理缓冲区中的内容,处理完成后,把缓冲区的内容送往屏幕,接着处理下一行,这样不断重复,直到文件末尾,文件内容并没有改变 二,sed的语法 2,1,基本语法 sed options ... […

AI学习记录 - gpt如何进行token化,理论知识,以GPT2为举例

AI学习记录已经发了十几篇&#xff0c;大佬们可以看看&#xff0c;如果有帮助动动小手点赞 token入门版&#xff0c;有空会更新具体代码操作 GPT4当中&#xff0c;我们提问问题是按照token进行扣费的&#xff0c;那到底什么是token&#xff1f; 在不同的语言模型当中&#x…

gradio之进度条

输出控件显示进度&#xff0c;进度结束显示控件结果 import gradio as gr import timedef slowly_reverse(word, progressgr.Progress()):progress(0, desc"Starting")time.sleep(1)progress(0.05)new_string ""for letter in progress.tqdm(word, desc&…

C++ 特性之vector详解 + 联合opencv使用

C 特性之vector详解 联合opencv使用 在C中&#xff0c;遍历vector并删除元素需要小心处理迭代器失效的问题。通常推荐的方法是使用迭代器进行遍历&#xff0c;并在需要删除元素时使用erase函数。这里给出一个示例代码&#xff0c;演示如何安全地遍历vector并删除特定条件的元…

计算机毕业设计 家电销售展示平台 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试

&#x1f34a;作者&#xff1a;计算机编程-吉哥 &#x1f34a;简介&#xff1a;专业从事JavaWeb程序开发&#xff0c;微信小程序开发&#xff0c;定制化项目、 源码、代码讲解、文档撰写、ppt制作。做自己喜欢的事&#xff0c;生活就是快乐的。 &#x1f34a;心愿&#xff1a;点…

智能识别,2024年SD卡数据恢复软件的智能进化

除了手机之外现在有不少的设备还是依靠SD卡来存储数据&#xff0c;比如相机、摄像头、无人机等。有的时候会因为一些意外的情况导致数据丢失&#xff0c;那是真的丢失了吗&#xff1f;大部分情况还是可以依靠sd卡数据恢复工具来找回这些“消失”的数据哦。 1.福昕数据恢复 链…