Unity之街机捕鱼

2024-03-05 19:20
文章标签 unity 捕鱼 街机

本文主要是介绍Unity之街机捕鱼,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

😪炮台系统

🎶炮口方向跟随鼠标

🎶切换炮台

😪战斗系统

🎮概述

🎮单例模式 

🎮开炮

🎮子弹脚本

🎮渔网脚本 

🎮鱼属性信息的脚本

😪奖励和等级的实现 

😪Unity数据持久化

📖保存游戏

📖继续游戏

📖开始新游戏

最近又跟着教程从头到尾完成了一个实例,虽然是好久之前的教程,但是老师讲的很好,我作为Unity初学者也收获很多,复盘后把收获写在这篇博客上,希望看到的你同样会有收获。


来看一下实例效果:切换炮台、捕鱼、加减金币(消耗炮弹的金币)、等级头衔设置、保存、继续游戏;作为一个捕鱼游戏基本功能都很齐全。

Unity中级案例 - 捕鱼达人哔哩哔哩_bilibili 教程分为上、中、下三部分。上主要对游戏UI进行设计和制作;中负责完成游戏主要玩法战斗系统;下对游戏进一步优化包括特效、音效、游戏保存、游戏发布。


UI和特效这里就先不说了,可以跟着教程实操一下。这里想记录核心玩法的实现和编码思路 ,学会思路和实现方式在其他想写的游戏上也能用到。

炮台系统

炮口方向跟随鼠标

所谓炮口方向跟随鼠标就是给炮口一个旋转朝向鼠标的方向。

我们要做的是获取 鼠标到炮台中心的连线 和 炮口所指位置的 夹角,然后将炮台 z轴旋转的数值 设置到和夹角一致就可以实现炮口方向跟随鼠标转动的目的。

using System.Collections;
using System.Collections.Generic;
using UnityEngine;//让炮口跟随鼠标
public class GunFollow : MonoBehaviour
{public RectTransform UGUICanvas;public Camera mainCamera;void Update(){//定义鼠标的当前位置Vector3 mousePos;//将屏幕坐标转换为世界坐标,结合Input.mousePosition获取鼠标在 当前Canvas(炮台所在的Canvas) 下的位置RectTransformUtility.ScreenPointToWorldPointInRectangle(UGUICanvas,new Vector2(Input.mousePosition.x, Input.mousePosition.y), mainCamera,out mousePos);//定义 z轴 的旋转值float z;//判断鼠标在炮口左边还是右边if (mousePos.x > transform.position.x){//Vector3.up  表示炮口的正方向//mousePos - transform.position  向量相减得到一条线;  与炮口正方向形成夹角//Vector3.Angle()  鼠标和炮台的夹角计算方法  返回值为正数  所以要判断鼠标在炮口左边还是右边z = -Vector3.Angle(Vector3.up, mousePos - transform.position);}else{z = Vector3.Angle(Vector3.up, mousePos - transform.position);}//设置炮口旋转transform.localRotation = Quaternion.Euler(0, 0, z);}
}

给所有炮台都挂载上跟随脚本:

切换炮台

炮台在游戏中一共有五种,同样也对应着五种子弹类型(每种子弹类型又有不同颜色与等级相对应),(每一炮所需要的金币数量)类型却有20种。也就是说每一种炮台分别有四挡所消耗的金币数。

来看看脚本是怎么实现的:

public class GameController : MonoBehaviour
{//所有炮台的预设体,用来更换炮台public GameObject[] gunGos;//炮弹生成点父物体public Transform bulletHolder;//五种炮弹预制体,对应五种炮台public GameObject[] bullet1Gos;public GameObject[] bullet2Gos;public GameObject[] bullet3Gos;public GameObject[] bullet4Gos;public GameObject[] bullet5Gos;//每一炮所消耗的金币数档位private int[] oneShootCosts =  { 5, 10, 20, 30, 40, 50, 60, 70, 80, 90, 100, 200, 300, 400, 500, 600, 700, 800, 900, 1000 };//使用的是第几档消耗金币数档位private int costIndex = 0;  //五种炮台对应20个消耗金币档位,也就是说每种炮台对应四个消耗金币档位void Update(){ChangeBulletCost();}//通过滚轮改变每发炮弹所消耗的钱void ChangeBulletCost(){if (Input.GetAxis("Mouse ScrollWheel") < 0){OnButtonMDown();}if (Input.GetAxis("Mouse ScrollWheel") > 0){OnButtonPDown();}}public void OnButtonPDown(){//costIndex  表示消耗金币档位,炮台和消耗金币档位是1对4的关系,那么 costIndex / 4  代表了我们在用哪个炮台//禁用武器          gunGos[costIndex / 4].SetActive(false);costIndex++;//播放音效AudioManager.Instance.PlayEffectSound(AudioManager.Instance.changeClip);//播放换枪特效Instantiate(changeEffect);//防止数组越界costIndex = (costIndex > oneShootCosts.Length - 1) ? 0 : costIndex;//启用武器gunGos[costIndex / 4].SetActive(true);//更新每发炮弹所消耗的钱oneShootCostText.text = "$" + oneShootCosts[costIndex];}public void OnButtonMDown(){//禁用武器gunGos[costIndex / 4].SetActive(false);costIndex--;AudioManager.Instance.PlayEffectSound(AudioManager.Instance.changeClip);//播放换枪特效Instantiate(changeEffect);//防止数组越界costIndex = (costIndex < 0) ? oneShootCosts.Length - 1 : costIndex;//启用武器gunGos[costIndex / 4].SetActive(true);//更新每发炮弹所消耗的钱oneShootCostText.text = "$" + oneShootCosts[costIndex];}
}

战斗系统

概述


当炮弹射出去之后碰到鱼;炮弹会消失生成渔网;渔网生成后也会在特定的时间消失。

同时渔网在消失前会对鱼进行碰撞检测;撞到渔网的鱼也会对自身生命值进行判断;

鱼死了(┏┛<・)))><<  墓┗┓) ——  播放死亡动画销毁自身并生成金币,对应玩家金币数量增加;鱼没死  ——  接着游,炮台接着开炮。



 单例模式 

我们来看一下实例中单例模式的应用:

public class GameController : MonoBehaviour
{//单例模式private static GameController _instance;public static GameController Instance{get{return _instance;}}//在Awake()生命周期函数中给 _instance 赋值void Awake(){_instance = this;}//定义初始金币和经验值public int exp = 0;public int gold = 500;
}//=================================================================
//当鱼死亡后通过单例模式增加 GameController 脚本中的金币和经验值
//=================================================================//当鱼死亡加金币和经验值GameController.Instance.gold += gold;GameController.Instance.exp += exp;

开炮

每个炮台都有一个空对象用来标记子弹发射点;来看看脚本是怎么实现开炮功能的:

  //开炮void Fire(){//当前炮台用的什么子弹,默认是第五种炮使用的炮弹GameObject[] useBullets = bullet5Gos;int bulletIndex;//当按下发射键并没有碰到其他UI按键(比如加减钱按钮、设置按钮)时才会发射炮弹if (Input.GetMouseButtonDown(0) && EventSystem.current.IsPointerOverGameObject() == false){//判断当前金币够不够开炮if (gold - oneShootCosts[costIndex] >= 0){switch (costIndex / 4){case 0: useBullets = bullet1Gos; break;case 1: useBullets = bullet2Gos; break;case 2: useBullets = bullet3Gos; break;case 3: useBullets = bullet4Gos; break;case 4: useBullets = bullet5Gos; break;}//根据等级分配哪一套子弹里的哪一种颜色bulletIndex = (lv % 10 >= 9) ? 9 : lv % 10;//扣钱gold -= oneShootCosts[costIndex];//播放音效AudioManager.Instance.PlayEffectSound(AudioManager.Instance.fireClip);//播放开火特效Instantiate(fireEffect);//实例化子弹GameObject bullet = Instantiate(useBullets[bulletIndex]);bullet.transform.SetParent(bulletHolder, false);//让子弹的朝向和旋转都和炮口开炮位置一样bullet.transform.position = gunGos[costIndex / 4].transform.Find("FirePos").transform.position;bullet.transform.rotation = gunGos[costIndex / 4].transform.Find("FirePos").transform.rotation;//设置子弹的伤害值bullet.GetComponent<BulletAttr>().damage = oneShootCosts[costIndex];  //把价格的序号传过去//让子弹移动bullet.AddComponent<Ef_AutoMove>().dir = Vector3.up; //改变方向bullet.GetComponent<Ef_AutoMove>().speed = bullet.GetComponent<BulletAttr>().speed ;     //给子弹速度    }else{//TODO Flash The Text;如果金币不足,通过闪烁金币数值提醒玩家StartCoroutine(GoldNotEnough());  //以协程的方式开启}}}//金币不足开炮时进行闪烁提示IEnumerator GoldNotEnough(){goldText.color = goldColor;goldText.color = Color.red;//让程序在这等待0.5秒然后从这接着运行yield return new WaitForSeconds(0.5f);goldText.color = goldColor;}

子弹脚本

public class BulletAttr : MonoBehaviour
{//子弹速度public int speed;//子弹伤害值public int damage;//子弹碰到鱼后生成网的预制体public GameObject webPrefab;private void OnTriggerEnter2D(Collider2D collision){//如果撞到边界(屏幕之外设定了一个UI边界),直接销毁自身if (collision.tag == "Border"){Destroy(gameObject);}//如果撞到鱼,销毁自身并生成网if (collision.tag == "Fish"){//生成网GameObject web = Instantiate(webPrefab);web.transform.SetParent(gameObject.transform.parent, false);//网的位置等于当前子弹的位置web.transform.position = gameObject.transform.position;//将子弹的伤害赋值给网的伤害web.GetComponent<WebAttr>().damage = damage;Destroy(gameObject);}}
}

渔网脚本 

public class WebAttr : MonoBehaviour
{//网消失的时间public float disapperTime;//子弹的伤害public int damage;void Start(){//隔多长时间销毁网自身Destroy(gameObject, disapperTime);}//检测是否碰到鱼private void OnTriggerEnter2D(Collider2D collision){if (collision.tag=="Fish"){//发送消息 给 FishAttr脚本 的 TakeDamage() 方法//通知鱼 我(渔网)打到你了 伤害值为 damagecollision.SendMessage("TakeDamage", damage);}}
}

鱼属性信息的脚本

//记录鱼的属性信息
public class FishAttr : MonoBehaviour
{//每种鱼的这些属性值都不一样,自己来设定public int hp;    //血量public int exp;   //经验值public int gold;  //金币值public int maxNum;    //生成最大数量public int maxSpeed;  //最大速度 //鱼死亡预设体public GameObject diePrefab;//金币预设体public GameObject goldPrefab;private void OnTriggerEnter2D(Collider2D collision){//撞到边界(屏幕之外设定了一个UI边界)就销毁鱼if (collision.tag == "Border"){Destroy(gameObject);}}//鱼受伤void TakeDamage(int value)  //value是从WebAttr脚本传过来的伤害值{hp -= value;//鱼死亡if (hp <= 0){//当鱼死亡加金币和经验值 —— 单例模式的应用GameController.Instance.gold += gold;GameController.Instance.exp += exp;//播放鱼死亡动画GameObject die = Instantiate(diePrefab);die.transform.SetParent(gameObject.transform.parent, false);die.transform.position = transform.position;die.transform.rotation = transform.rotation;//鱼死亡实例化金币GameObject goldGo = Instantiate(goldPrefab);goldGo.transform.SetParent(gameObject.transform.parent, false);goldGo.transform.position = transform.position;goldGo.transform.rotation = transform.rotation;//判断鱼身上有没有播放特效的脚本 Ef_PlayEffectif (gameObject.GetComponent<Ef_PlayEffect>() != null){AudioManager.Instance.PlayEffectSound(AudioManager.Instance.rewardClip);gameObject.GetComponent<Ef_PlayEffect>().PlayEffect();}Destroy(gameObject);}}
}

奖励和等级的实现 

游戏中为避免玩家金币耗光有两种金币奖励功能:小奖励通过每60s倒计时自动给玩家50金币;大奖励240s倒计时结束后变为按钮,玩家点击按钮会获得500金币然后会重新倒计时。

 游戏设定等级称号为:"新手", "入门", "钢铁", "青铜", "白银", "黄金", "白金", "钻石", "大师", "宗师"。根据等级来对应称号

提示玩家升到多少级的UI文本: 

 脚本实现:

public class GameController : MonoBehaviour
{//等级public int lv = 0;//经验public int exp = 0;//初始金币值public int gold = 500;public const int bigCountdown = 240;   //240秒大奖励public const int smallCountdown = 60;  //60秒小奖励public float bigTimer = bigCountdown;  //计时器public float smallTimer = smallCountdown;private string[] lvName = { "新手", "入门", "钢铁", "青铜", "白银", "黄金", "白金", "钻石", "大师", "宗师" };void Update(){//更新等级UpdateUI();}void UpdateUI(){bigTimer -= Time.deltaTime;   //倒计时smallTimer -= Time.deltaTime; //倒计时//如果小计时器小于0,给玩家发金币if (smallTimer <= 0){//计时器重新计时smallTimer = smallCountdown;gold += 50;}//大计时器小于0 并且 当按钮没有显示出来才会执行if (bigTimer <= 0 && bigCountdownButton.gameObject.activeSelf == false){//倒计时结束隐藏计时器bigCountdownText.gameObject.SetActive(false);//显示领取金币按钮bigCountdownButton.gameObject.SetActive(true);}//经验等级换算公式:升级所需经验=1000+200*当前等级while (exp >= 1000 + 20*lv){exp = exp - (1000 + 200 * lv);lv++;//提示玩家升到多少级lvUpTips.SetActive(true);lvUpTips.transform.Find("Text").GetComponent<Text>().text = lv.ToString();//启动协程把提示关闭StartCoroutine(lvUpTips.GetComponent<Ef_HideSelf>().HideSelf(0.6f));//播放音效AudioManager.Instance.PlayEffectSound(AudioManager.Instance.lvUpClip);//播放升级特效Instantiate(lvEffect);}goldText.text = "$" + gold;lvText.text = lv.ToString();//如果玩家等级超过99级,就一直是宗师if ((lv / 10) <= 9){lvNameText.text = lvName[lv / 10];}else{lvNameText.text = lvName[9];}//小计时器                           拿到十位数                       拿到个位smallCountdownText.text = "  " + (int)smallTimer / 10 + "  " + (int)smallTimer % 10;//大计时器bigCountdownText.text = (int)bigTimer + "s";//滑动条显示的是比例 —— 滑动条用来显示等级进度expSlider.value = ((float)exp) / (1000 + 20 * lv);}//大计时器点击领取金币按钮public void OnBigCountdownButtonDown(){gold += 500;AudioManagerFwl.Instance.PlayEffectSound(AudioManagerFwl.Instance.rewardClip);//显示加金币特效Instantiate(goldEffect);//领取完隐藏按钮bigCountdownButton.gameObject.SetActive(false);//显示倒计时bigCountdownText.gameObject.SetActive(true);//给倒计时重新赋值bigTimer = bigCountdown;}
}

Unity数据持久化

Unity的这种数据持久化是通过 键值对 的方式存储在 Windows注册表 

unity PlayerPrefs数据存储位置_playerprefer保存的数据位置-CSDN博客文章浏览阅读1.4k次,点赞4次,收藏8次。1、首先查看自己当前工程的名字:可以在Unity->Edit->Project Settings->Player中设置与查看,如图所示:2、按下键盘Win+R键,输入regedit,打开注册表编辑器,找到相应位置查看如下图:_playerprefer保存的数据位置https://blog.csdn.net/Monkey_Xuan/article/details/115518561unity3d--PlayerPrefs 游戏存档_unity 修改playerprefs.setfloat-CSDN博客文章浏览阅读5.7k次,点赞4次,收藏42次。Unity3D游戏开发之数据持久化PlayerPrefs的使用 转载自 本文作者:秦元培,本文出处:http://blog.csdn.net/qinyuanpei/article/details/24195977 博主今天研究了在Unity3D中的数据持久化问题。数据持久化在任何一个开发领域都是一个值得关注的问题,小到一个_unity 修改playerprefs.setfloathttps://blog.csdn.net/acmer_sly/article/details/52675954

保存游戏

游戏一共两个场景:一个Start游戏开始界面场景和Main主要玩法场景。在Main中有一个 返回 按钮点击会返回到开始界面场景中,当我们按下这个按钮保存当前游戏的功能就发挥了作用。

我们需要保存的数据有金币数量、当前等级、当前倒计时数值、还有背景音乐的开关。


当我们按下返回按钮后 :

    //返回按钮public void OnBackButtonDown(){//TODO 保存当前游戏  PlayerPrefs只支持 int float string 三种数据类型//保存金币值PlayerPrefs.SetInt("gold", GameController.Instance.gold);//保存等级PlayerPrefs.SetInt("lv", GameController.Instance.lv); //小计时器PlayerPrefs.SetFloat("scd", GameController.Instance.smallTimer);//大计时器PlayerPrefs.SetFloat("bcd", GameController.Instance.bigTimer);//保存经验值PlayerPrefs.SetInt("exp", GameController.Instance.exp);int temp = (AudioManager.Instance.IsMute == false) ? 0 : 1;//保存当前游戏场景的背景音乐是开还是关PlayerPrefs.SetInt("mute", temp);//跳转到开始界面场景UnityEngine.SceneManagement.SceneManager.LoadScene(2);}

继续游戏

当我们点击完返回按钮来到游戏开始界面后,点击 继续游戏 按钮会获取保存好的玩家信息。

public class GameController : MonoBehaviour
{//等级public int lv = 0;//经验public int exp = 0;public int gold = 500;public const int bigCountdown = 240;   //240大奖励public const int smallCountdown = 60;  //60秒小奖励public float bigTimer = bigCountdown;  //计时器public float smallTimer = smallCountdown;//读取游戏void Start(){//通过键值对的形式获取保存好的数值gold = PlayerPrefs.GetInt("gold", gold);lv = PlayerPrefs.GetInt("lv", lv);exp = PlayerPrefs.GetInt("exp", exp);smallTimer = PlayerPrefs.GetFloat("scd", smallCountdown);bigTimer = PlayerPrefs.GetFloat("bcd", bigCountdown);UpdateUI();}
}

 开始新游戏

保存好玩家信息后如果点击 开始游戏 ,通过脚本会删除保存好的玩家信息开始新游戏。

using UnityEngine;
using UnityEngine.SceneManagement;public class StartSceneUI : MonoBehaviour
{//开始游戏按钮清空数据后在加载游戏场景public void NewGame(){//根据键删除保存好的值PlayerPrefs.DeleteKey("gold");PlayerPrefs.DeleteKey("lv");PlayerPrefs.DeleteKey("exp");PlayerPrefs.DeleteKey("scd");PlayerPrefs.DeleteKey("bcd");SceneManager.LoadScene(1);}//继续游戏直接加载游戏场景,在 GameController 脚本获取保存好的玩家信息public void OldGame(){SceneManager.LoadScene(1);}public void OnCloseButton(){//退出游戏Application.Quit();}
}

想要完整源码就在评论区评论吧,本篇到这里就先结束了,希望我们都能有所收获,拜拜┏(^0^)┛ 

这篇关于Unity之街机捕鱼的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Unity Post Process Unity后处理学习日志

Unity Post Process Unity后处理学习日志 在现代游戏开发中,后处理(Post Processing)技术已经成为提升游戏画面质量的关键工具。Unity的后处理栈(Post Processing Stack)是一个强大的插件,它允许开发者为游戏场景添加各种视觉效果,如景深、色彩校正、辉光、模糊等。这些效果不仅能够增强游戏的视觉吸引力,还能帮助传达特定的情感和氛围。 文档

Unity协程搭配队列开发Tips弹窗模块

概述 在Unity游戏开发过程中,提示系统是提升用户体验的重要组成部分。一个设计良好的提示窗口不仅能及时传达信息给玩家,还应当做到不干扰游戏流程。本文将探讨如何使用Unity的协程(Coroutine)配合队列(Queue)数据结构来构建一个高效且可扩展的Tips弹窗模块。 技术模块介绍 1. Unity协程(Coroutines) 协程是Unity中的一种特殊函数类型,允许异步操作的实现

Unity 资源 之 Super Confetti FX:点亮项目的璀璨粒子之光

Unity 资源 之 Super Confetti FX:点亮项目的璀璨粒子之光 一,前言二,资源包内容三,免费获取资源包 一,前言 在创意的世界里,每一个细节都能决定一个项目的独特魅力。今天,要向大家介绍一款令人惊艳的粒子效果包 ——Super Confetti FX。 二,资源包内容 💥充满活力与动态,是 Super Confetti FX 最显著的标签。它宛如一位

Unity数据持久化 之 一个通过2进制读取Excel并存储的轮子(4)

本文仅作笔记学习和分享,不用做任何商业用途 本文包括但不限于unity官方手册,unity唐老狮等教程知识,如有不足还请斧正​​ Unity数据持久化 之 一个通过2进制读取Excel并存储的轮子(3)-CSDN博客  这节就是真正的存储数据了   理清一下思路: 1.存储路径并检查 //2进制文件类存储private static string Data_Binary_Pa

Unity Adressables 使用说明(一)概述

使用 Adressables 组织管理 Asset Addressables 包基于 Unity 的 AssetBundles 系统,并提供了一个用户界面来管理您的 AssetBundles。当您使一个资源可寻址(Addressable)时,您可以使用该资源的地址从任何地方加载它。无论资源是在本地应用程序中可用还是存储在远程内容分发网络上,Addressable 系统都会定位并返回该资源。 您

Unity Adressables 使用说明(六)加载(Load) Addressable Assets

【概述】Load Addressable Assets Addressables类提供了加载 Addressable assets 的方法。你可以一次加载一个资源或批量加载资源。为了识别要加载的资源,你需要向加载方法传递一个键或键列表。键可以是以下对象之一: Address:包含你分配给资源的地址的字符串。Label:包含分配给一个或多个资源的标签的字符串。AssetReference Obj

在Unity环境中使用UTF-8编码

为什么要讨论这个问题         为了避免乱码和更好的跨平台         我刚开始开发时是使用VS开发,Unity自身默认使用UTF-8 without BOM格式,但是在Unity中创建一个脚本,使用VS打开,VS自身默认使用GB2312(它应该是对应了你电脑的window版本默认选取了国标编码,或者是因为一些其他的原因)读取脚本,默认是看不到在VS中的编码格式,下面我介绍一种简单快

Unity数据持久化 之 一个通过2进制读取Excel并存储的轮子(3)

本文仅作笔记学习和分享,不用做任何商业用途 本文包括但不限于unity官方手册,unity唐老狮等教程知识,如有不足还请斧正​​ Unity数据持久化 之 一个通过2进制读取Excel并存储的轮子(2) (*****生成数据结构类的方式特别有趣****)-CSDN博客 做完了数据结构类,该做一个存储类了,也就是生成一个字典类(只是声明)  实现和上一节的数据结构类的方式大同小异,所

【Unity小技巧】URP管线遮挡高亮效果

前言 在URP渲染管线环境下实现物体遮挡高亮显示效果,效果如下: Unity URP遮挡高亮 实现步骤 创建层级,为需要显示高亮效果的物体添加层级,比如Player 创建一个材质球,也就是高亮效果显示的材质球找到Universal Renderer Data Assets 4.在Assets上添加两个Render Objects组件 第一个做如下三处设置 指定遮挡层级指

【Unity面经】实习篇:面试官常问的一百个面试题

👨‍💻个人主页:@元宇宙-秩沅 👨‍💻 hallo 欢迎 点赞👍 收藏⭐ 留言📝 加关注✅! 👨‍💻 本文由 秩沅 原创 👨‍💻 专栏交流🧧🟥Unity100个实战基础✨🎁🟦 Unity100个精华一记✨🎁🟩 Unity50个demo案例教程✨🎁🟨 Unity100个精华细节BUG✨🎁🟨 Unity100个面试题✨🎁 文章