Unity2D学习 Fox Game制作 过程1:基本的游戏角色控制,动画效果,镜头控制,物品收集,bug优化
教程学习地址 有进行部分修改,与教材不是完全一致
1.素材导入
搜索
Unity素材商店 在素材商店搜索Sunny Land data:image/s3,"s3://crabby-images/3079c/3079c62c9b82cf20b84d3292b3730cd308776af3" alt="Sunny Land"
导入
data:image/s3,"s3://crabby-images/505e8/505e8adec01f05d3a94a1ba3d41f6eea442b8fed" alt="在Unity中打开" 点击在Unity中打开
data:image/s3,"s3://crabby-images/6ab0d/6ab0dd1fa5524f29412fd99df9a52cda656ed15b" alt="" 点击右下角Import 若没有下载需要先下载
data:image/s3,"s3://crabby-images/f8502/f85026e401d5e6e7d9d20a354bb29a5f4d625813" alt="Import Unity Package" 全部选择之后点击Import即可
2.编辑素材
添加背景图
data:image/s3,"s3://crabby-images/23375/23375a878ded3f470054651cae628809768ebb46" alt="pixels per unit" 修改Pixels Per Unit为16 即场景中一格为16像素 之后所有素材的Unit都改为16
data:image/s3,"s3://crabby-images/246bc/246bcafeb5ccdcd60c8c9ceb55ebcd33c5af095f" alt="back" 将back拖入即可
创建tilemap
data:image/s3,"s3://crabby-images/93ee0/93ee03279bd89f8a0726682cdb59f6a7f090aae9" alt="创建tilemap" 在Hierarchy窗口中创建tilemap
data:image/s3,"s3://crabby-images/98472/98472fc61c452257ba1d5881117c4abd9a37af84" alt="Tile Palette" 打开Tile Palette
data:image/s3,"s3://crabby-images/04abb/04abb703cd9f4c83241f117f27fa8066efa5ee81" alt="new palette" 在Tile Palette中创建新Palette
data:image/s3,"s3://crabby-images/ced51/ced518a443cd8395cdad5216baea195f146dcebb" alt="设置Sprite Mode" 选择tileset设置sprite mode为Multiple,点击Sprite Editor
data:image/s3,"s3://crabby-images/53408/534080145d0bad1e9d35eaf375f8bc8038fd8471" alt="示意图" 使用自定义尺寸进行切割,保证每个单元可以独立使用 若使用自动切割,无法将每个图块分割成多个单元 Pixel Size是由素材的Pixes Per Unite决定的
data:image/s3,"s3://crabby-images/77f1b/77f1b76ac390d24fadb5ce7c3b181eb9c6f8fb14" alt="" 将分割好的素材拖拽进入Tile Palette即可
data:image/s3,"s3://crabby-images/b5c16/b5c16dd33162205776b554afc565f032f53d4fec" alt="绘制地图" 使用Tile Palette的笔刷等工具绘制地图即可
data:image/s3,"s3://crabby-images/2da5c/2da5c6bd247a0c13405a2aae2da06f5d247a4cba" alt="地图"
3.图层Layer
data:image/s3,"s3://crabby-images/1e2ea/1e2ea33ea0f1398ecf79c7ce123163f058e68d74" alt="sorting layer" 选择和添加图层 越下面越先显示
4.游戏角色添加
添加载体
data:image/s3,"s3://crabby-images/b25fd/b25fd6f169acdaa6fd8afbf85bea6564bbefdcc7" alt="添加载体" 右键添加sprites载体
data:image/s3,"s3://crabby-images/63a1c/63a1c4ca9351136aea6ff0e301bc9edcee80ca94" alt="" 选中载体,拖拽角色素材到sprite
Rigidbody
data:image/s3,"s3://crabby-images/b7345/b73455fcae1abc56c0ca3e1c259d4dbabec45ed6" alt="Add Component" data:image/s3,"s3://crabby-images/cb79e/cb79e980cfc0023c4e01a8a2da5f3ad5550d54a2" alt="" data:image/s3,"s3://crabby-images/05739/05739afaeac2d032d89e52561be1e62bcd749f06" alt="" 给角色添加2d刚体属性
Collider
data:image/s3,"s3://crabby-images/a1b82/a1b8204ec2df6d41f6d157958d1e2078a55220b0" alt="Collider" data:image/s3,"s3://crabby-images/afcaa/afcaa27aa5bc2414bfad9b206551ad0b3fbe0a2e" alt="" 给角色添加2d碰撞体属性
data:image/s3,"s3://crabby-images/9ce5a/9ce5ac78bd7710458fd29d4dc006ed1626569322" alt="" data:image/s3,"s3://crabby-images/0f934/0f934c8fc383f0602293b260893b21d70dac1b19" alt="" 为tilemap创建碰撞体属性
5.角色移动
data:image/s3,"s3://crabby-images/10e75/10e75cf322d815b8badacfa1381b1d22d0572dd2" alt="Add Component" data:image/s3,"s3://crabby-images/3a734/3a734a3b0b48e69a4d0a6db8db5c761b848f8fd3" alt="" 为角色创建脚本
data:image/s3,"s3://crabby-images/c7f03/c7f039d777872a01abb7e7ad1a6a2418d4af5aac" alt="" 打开脚本添加两个属性后保存
data:image/s3,"s3://crabby-images/5ad1d/5ad1d58cfad3c7613544019ecac3e4f533260618" alt="" 选中角色obj为两个属性赋初值,将rigidbody2d拖动到rb
void Movement()
{
float horizontalMove;
horizontalMove = Input.GetAxis("Horizontal");
if(horizontalMove != 0)
{
rb.velocity = new Vector2(horizontalMove * speed, rb.velocity.y);
}
}
构造简单的移动函数,在update中调用
6.角色朝向&跳跃
void Movement()
{
float horizontalMove = Input.GetAxis("Horizontal");
float faceDirection = Input.GetAxisRaw("Horizontal");
if(horizontalMove != 0)
{
rb.velocity = new Vector2(horizontalMove * speed * Time.fixedDeltaTime, rb.velocity.y);
}
if(faceDirection != 0)
{
transform.localScale = new Vector3(faceDirection, 1, 1);
}
if (Input.GetButtonDown("Jump"))
{
rb.velocity = new Vector2(rb.velocity.x, jumpForce * Time.fixedDeltaTime);
}
}
7.动画效果Animator
添加Animator
data:image/s3,"s3://crabby-images/bf0e6/bf0e6972d1069cc6cfb37df23e68a8235f75ecb7" alt="Add Component" data:image/s3,"s3://crabby-images/b28c3/b28c3dae296280a24c3ce6d4a18c8bd82ef87819" alt="" 添加Animator属性
data:image/s3,"s3://crabby-images/e5180/e51808916d645cce96956cefa3d5395be0d408d8" alt="" 在Project窗口创建Animator Controller
data:image/s3,"s3://crabby-images/beb6e/beb6e352df6246fe03b152654955005e6836bd76" alt="" data:image/s3,"s3://crabby-images/9eb37/9eb371b2d965bef81eba9bd214eaac9e330095a5" alt="" 拖动Animator Controller至Animator中的Controller进行绑定
角色idle动画
data:image/s3,"s3://crabby-images/db8b0/db8b0605a7c7da2a3c5a086fe5a636fca9290212" alt="" 选中角色后打开Animation窗口
data:image/s3,"s3://crabby-images/17e38/17e380d0dfa0947f539421f76152b8c301a9c4d7" alt="" 选中idle图片拖拽进入Animation,自动生成关键帧 data:image/s3,"s3://crabby-images/bc355/bc35568112ba76e32c826fbec2a504d152815606" alt="" 可以通过时间轴长度和修改Samples改变播放速度,samples由此处打开
角色run动画
data:image/s3,"s3://crabby-images/5244c/5244ce5ec0d3eca2750fdec9aafd336355ec88b9" alt="" 选中角色obj,创建新动画
data:image/s3,"s3://crabby-images/02e15/02e1559f43ae8a919a7ebf3ea705b0073c2bbe61" alt="" 选中run图片拖拽进入Animation,自动生成关键帧 可以通过时间轴长度和修改Samples改变播放速度,samples打开方式见上
data:image/s3,"s3://crabby-images/4aff5/4aff564efc4b3ad72c30cfb27812356c6c52febf" alt="" 打开animator,为idle和run动画建立transition
data:image/s3,"s3://crabby-images/b1efd/b1efdfbfa4e120f2aa1de1156418acfea4e7c191" alt="" 在animator窗口左侧添加参数
data:image/s3,"s3://crabby-images/06999/06999db65f3968f0f2962055e035fbc285856787" alt="" 点击transition的箭头,修改参数,保证动画可以立刻切换
data:image/s3,"s3://crabby-images/ccac1/ccac16d076b0d548a3a7e2b394292ac557603dd6" alt="" data:image/s3,"s3://crabby-images/fb87f/fb87f71a1d652ff05908b9a30e5cca8da3f2d3cf" alt="" 点击transition的箭头,添加条件判断,进行动画转换
public Rigidbody2D rb;
public Animator animator;
public float speed;
public float jumpForce;
添加public Animator animator,用于绑定角色动画的Animator
data:image/s3,"s3://crabby-images/02852/0285212c419e411c78e104aa646c72da45df2a57" alt="" 拖拽Animator到C#Script的Animator进行绑定
void Movement()
{
float horizontalMove = Input.GetAxis("Horizontal");
float faceDirection = Input.GetAxisRaw("Horizontal");
if (horizontalMove != 0)
{
rb.velocity = new Vector2(horizontalMove * speed * Time.fixedDeltaTime, rb.velocity.y);
}
if(faceDirection != 0)
{
transform.localScale = new Vector3(faceDirection, 1, 1);
animator.SetBool("isRun", true);
}
else
{
animator.SetBool("isRun", false);
}
if (Input.GetButtonDown("Jump"))
{
rb.velocity = new Vector2(rb.velocity.x, jumpForce * Time.fixedDeltaTime);
}
}
修改Movement(),通过animator.SetBool()方法设置Animator的参数
角色jump动画
data:image/s3,"s3://crabby-images/09483/094838361c46210b958938b7ce18316c1b3f2a57" alt="" data:image/s3,"s3://crabby-images/f2851/f285109e1f3f482ed2833608d9f250918f16778d" alt="" 创建跳起和下落的动画
data:image/s3,"s3://crabby-images/c5e19/c5e1994236fbccc05bbdb647572615aa2bc46920" alt="" 创建各个动画之间的transition,创建新的属性值,调整参数,添加转换条件
void Movement()
{
float horizontalMove = Input.GetAxis("Horizontal");
float faceDirection = Input.GetAxisRaw("Horizontal");
if (horizontalMove != 0)
{
rb.velocity = new Vector2(horizontalMove * speed, rb.velocity.y);
}
if (faceDirection != 0)
{
transform.localScale = new Vector3(faceDirection, 1, 1);
animator.SetBool("isRun", true);
}
else
{
animator.SetBool("isRun", false);
}
if (Input.GetButtonDown("Jump"))
{
rb.velocity = new Vector2(rb.velocity.x, jumpForce);
animator.SetBool("isJump", true);
}
}
void SwitchAnim()
{
if (animator.GetBool("isJump"))
{
if (rb.velocity.y <= 0)
{
animator.SetBool("isJump", false);
animator.SetBool("isFall", true);
}
}
else if (collider.IsTouchingLayers(ground))
{
animator.SetBool("isFall", false);
}
}
修改角色脚本,实现动画转换
8.代码及碰撞体优化
碰撞体优化
data:image/s3,"s3://crabby-images/44ebc/44ebc8f58ee1770cd0fc0f510c1ca44aafb61891" alt="fox" 修改Box Collider为Box Collider+Circle Collider 防止角色横向移动时被tilemap的单位格碰撞体卡住和无法上坡
代码优化
private Rigidbody2D rb;
private Animator animator;
private Collider2D coll;
public float speed;
public float jumpForce;
public LayerMask ground;
void Start()
{
rb = GetComponent<Rigidbody2D>();
animator = GetComponent<Animator>();
coll = GetComponent<CircleCollider2D>();
}
修改属性为私有,在Start()方法中初始化
9.镜头控制
方法1:为Camera添加Script
public Transform player;
void Update()
{
transform.position = new Vector3(player.position.x, player.position.y, transform.position.z);
}
该脚本实现镜头实时跟随角色移动 不推荐
方法2:Cinemachine插件
功能详解 data:image/s3,"s3://crabby-images/ef5e1/ef5e1b7d380ad67eea1485f5c32d25a7d19f0c4a" alt="" 打开Package Manager,安装Cinemachine插件
data:image/s3,"s3://crabby-images/4c40a/4c40affb8c01d982bc0e306dcf49d28a18afa41d" alt="" 安装完成后添加Cinemachine 2D Camera组件
data:image/s3,"s3://crabby-images/4a652/4a6520fb375ef02705e9711d901ff025d690c41c" alt="" 创建成功后会与Main Camera绑定
镜头跟踪
data:image/s3,"s3://crabby-images/7b22d/7b22d3f99ff5ad2d284d56d78aef2c5e862dca34" alt="" 设置cinemachine的follow为player
data:image/s3,"s3://crabby-images/9633c/9633c29ee23e12ccf76a9c78da52eba61562df58" alt="" 根据需求自行设置Body的参数
镜头移动限制
data:image/s3,"s3://crabby-images/71669/71669bfb340ddf1c09af49a3e522aec62e8a9ce9" alt="" 为cinemachine添加confinder组件
data:image/s3,"s3://crabby-images/3085a/3085a71590a03edf41e5b6b90a46c508cbf1ae68" alt="" 为背景添加Polygon Collider组件,自行调整大小,设置为Trigger
data:image/s3,"s3://crabby-images/8d4f1/8d4f11f58632273f125d70791e604b6a9d00003d" alt="" Confider组件的Bounding Shape 2D属性设置为BackGround,镜头即被限制在BackGround的Polygon Collider中移动
10.物品收集
樱桃创建
data:image/s3,"s3://crabby-images/67b51/67b51489e9048e6b34b0e102cbfc45f3b90bb4bc" alt="" 创建存放Cherry的obj
data:image/s3,"s3://crabby-images/46701/46701b26f0c69f3ba1bfc399007420109c7b64a7" alt="" 为Cherry添加碰撞体组件,更改为trigger属性
data:image/s3,"s3://crabby-images/1f8b1/1f8b135b825e3cb6bcb9bff9ff1a52331fad6196" alt="" 为Cherry创建动画 过程同第七步
物品收集脚本
data:image/s3,"s3://crabby-images/f54ce/f54ce7521301b2e0a8c20bcac9178acf98ba4731" alt="" 添加新的tag,切换Cherry的tag属性为新添加的Collection
public int cherryCount = 0;
private void OnTriggerEnter2D(Collider2D collision)
{
if(collision.tag == "Collection")
{
Destroy(collision.gameObject);
cherryCount += 1;
}
}
在角色控制脚本中添加新的属性和方法,实现角色收集樱桃
Prefabs
data:image/s3,"s3://crabby-images/dac9c/dac9c416e5e0acea17e318e3442005dbab6949ef" alt="" 创建Prefabs文件夹
data:image/s3,"s3://crabby-images/f5721/f5721c18418f4230045fa6d0f06951d74c766bee" alt="" 将场景中已经创建好的obj拖入Prefabs文件夹,自动生成Prefab 预制件
11.优化Player Controller代码
原本代码对于速度的控制放在update内,可能出现电脑性能不同player速度不同的情况
void Update()
{
SwitchAnim();
checkHorizontalMove();
checkJump();
checkGround();
}
void FixedUpdate()
{
horizontalMove();
jump();
}
修改代码,将对于玩家输入的判断全部放在Update内,对游戏物理上的控制放在FixedUpdate内 Update每帧调用一次,FixedUpdate每0.02秒执行一次,所以FixedUpdate中控制物理变化与电脑性能无关,但是若在FixedUpdate中判断玩家输入可能出现FixedUpdate执行时控制输入结束,导致无法实时控制角色,所以对玩家输入的判断放在Update中 详细可以网上查阅资料
void checkHorizontalMove()
{
faceDirection = Input.GetAxisRaw("Horizontal");
}
void checkJump()
{
if (Input.GetButtonDown("Jump")) isJump = true;
}
void checkGround()
{
isOnGround = feetColl.IsTouchingLayers(ground);
}
void horizontalMove()
{
if (faceDirection != 0)
{
rb.velocity = new Vector2(faceDirection * speed, rb.velocity.y);
transform.localScale = new Vector3(faceDirection, 1, 1);
animator.SetBool("isRun", true);
}
else
{
rb.velocity = new Vector2(0, rb.velocity.y);
animator.SetBool("isRun", false);
}
}
void jump()
{
if (isOnGround)
{
if (isJump)
{
rb.velocity = new Vector2(rb.velocity.x, jumpForce);
animator.SetBool("isJump", true);
}
}
isJump = false;
}
void SwitchAnim()
{
if (faceDirection != 0)
{
animator.SetBool("isRun", true);
}
else
{
animator.SetBool("isRun", false);
}
if (animator.GetBool("isJump"))
{
if (rb.velocity.y <= 0)
{
animator.SetBool("isJump", false);
animator.SetBool("isFall", true);
}
}
else if (feetColl.IsTouchingLayers(ground))
{
animator.SetBool("isFall", false);
}
}
详细代码
12.限制角色跳跃次数
角色可以无限制的跳跃,不合常理
void checkJump()
{
if (Input.GetButtonDown("Jump")) isJump = true;
}
void checkGround()
{
isOnGround = feetColl.IsTouchingLayers(ground);
}
void jump()
{
if (isOnGround)
{
if (isJump)
{
rb.velocity = new Vector2(rb.velocity.x, jumpForce);
animator.SetBool("isJump", true);
}
}
isJump = false;
}
通过这3个方法实现角色只有在地上时能够跳跃 checkJump,checkGround放在Update中,jump放在FixedUpdate中
13.解决角色有时无法跳跃的bug
经过测试,发现是由于tilemap每格之间为独立碰撞体,有时会将角色弹起,导致isOnGround判断为false而无法跳跃 data:image/s3,"s3://crabby-images/264a0/264a058f83deed627e6494c4e27d9f2293c9d12e" alt="" 为tilemap添加Composite Collider 2D组件
data:image/s3,"s3://crabby-images/0c557/0c5570dfe5d9300018f373a2cfe356204c0d7c8a" alt="" Tilemap Collider 2D勾选Used By Composite,Rigibody 2D的Body Type更改为Static
14.解决撞墙不掉落的bug
data:image/s3,"s3://crabby-images/fc7b1/fc7b1362f9859882e072faf70d62ad1b057bc1a7" alt="" 创建Material 2D
data:image/s3,"s3://crabby-images/ce26c/ce26c3334f27c5dd26c941ab5a21f6966156b725" alt="" 修改Friction为0,即摩擦力为零
data:image/s3,"s3://crabby-images/8ae8a/8ae8a9eeea91100b39af05c735bf85a8dd9cb7a9" alt="" 为Player的两个Collider添加为新建的Material
|