经典游戏案例:仿植物大战僵尸
学习目标:仿植物大战僵尸核心玩法实现
游戏画面


项目结构目录

部分核心代码
using System;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.SceneManagement;
using Random = UnityEngine.Random;public enum ZombieType
{Zombie1, ConeHeadZombie,BucketHeadZombie,
}[Serializable]
public struct Wave
{[Serializable]public struct Data{public ZombieType zombieType;public uint count;}public bool isLargeWave;[Range(0f,1f)]public float percentage;public Data[] zombieData;
}public class GameController : MonoBehaviour
{public GameObject zombie1;public GameObject BucketheadZombie;public GameObject ConeheadZombie;private GameModel model;public GameObject progressBar;public GameObject gameLabel;public GameObject sunPrefab;public GameObject cardDialog;public GameObject sunLabel;public GameObject shovelBG;public GameObject btnSubmitObj;public GameObject btnResetObj;public string nextStage;public float readyTime;public float elapsedTime;public float playTime;public float sunInterval;public AudioClip readySound;public AudioClip zombieComing;public AudioClip hugeWaveSound;public AudioClip finalWaveSound;public AudioClip loseMusic;public AudioClip winMusic;public Wave[] waves;public int initSun;private bool isLostGame = false;void Awake(){model = GameModel.GetInstance();}void Start (){model.Clear();model.sun = initSun;ArrayList flags=new ArrayList();for (int i = 0; i < waves.Length; i++){if (waves[i].isLargeWave){flags.Add(waves[i].percentage);}}progressBar.GetComponent<ProgressBar>().InitWithFlag((float[])flags.ToArray(typeof(float)));progressBar.SetActive(false);cardDialog.SetActive(false);sunLabel.SetActive(false);shovelBG.SetActive(false);btnResetObj.SetActive(false);btnSubmitObj.SetActive(false);GetComponent<HandlerForShovel>().enabled = false;GetComponent<HandlerForPlants>().enabled = false;StartCoroutine(GameReady());}Vector3 origin{get{return new Vector3(-2f,-2.6f);}}void OnDrawGizmos(){// DeBugDrawGrid(origin,0.8f,1f,9,5,Color.blue);}void DeBugDrawGrid(Vector3 _orgin,float x,float y,int col,int row,Color color){for (int i = 0; i < col+1; i++){Vector3 startPoint = _orgin + Vector3.right*i*x;Vector3 endPoint = startPoint + Vector3.up*row*y;Debug.DrawLine(startPoint,endPoint,color);}for (int i = 0; i < row+1; i++){Vector3 startPoint = _orgin + Vector3.up * i * y;Vector3 endPoint = startPoint + Vector3.right * col * x;Debug.DrawLine(startPoint, endPoint, color);}}public void AfterSelectCard(){btnResetObj.SetActive(false);btnSubmitObj.SetActive(false);Destroy(cardDialog);GetComponent<HandlerForShovel>().enabled = true;GetComponent<HandlerForPlants>().enabled = true;Camera.main.transform.position=new Vector3(1.1f,0,-1f);StartCoroutine(WorkFlow());InvokeRepeating("ProduceSun", sunInterval, sunInterval);}IEnumerator GameReady(){yield return new WaitForSeconds(0.5f);MoveBy move = Camera.main.gameObject.AddComponent<MoveBy>();move.offset=new Vector3(3.55f,0,0);move.time = 1f;move.Begin();yield return new WaitForSeconds(1.5f);sunLabel.SetActive(true);shovelBG.SetActive(true);cardDialog.SetActive(true);btnResetObj.SetActive(true);btnSubmitObj.SetActive(true);}void Update(){if (Input.GetKeyDown(KeyCode.S)){model.sun += 50;}if (!isLostGame){for (int row = 0; row < model.zombieList.Length; row++){foreach (GameObject zombie in model.zombieList[row]){if (zombie.transform.position.x<(StageMap.GRID_LEFT-0.4f)){LoseGame();isLostGame = true;return;}}} }}IEnumerator WorkFlow(){gameLabel.GetComponent<GameTips>().ShowStartTip();AudioManager.GetInstance().PlaySound(readySound);yield return new WaitForSeconds(readyTime);ShowProgressBar();AudioManager.GetInstance().PlaySound(zombieComing);for (int i = 0; i < waves.Length; i++){yield return StartCoroutine(WaitForWavePercentage(waves[i].percentage));if (waves[i].isLargeWave){StopCoroutine(UpdateProgress());yield return StartCoroutine(WaitForZombieClear());yield return new WaitForSeconds(3.0f);gameLabel.GetComponent<GameTips>().ShowApproachingTip();AudioManager.GetInstance().PlaySound(hugeWaveSound);yield return new WaitForSeconds(3.0f);StartCoroutine(UpdateProgress());}if (i+1==waves.Length){gameLabel.GetComponent<GameTips>().ShowFinalTip();AudioManager.GetInstance().PlaySound(finalWaveSound);}yield return StartCoroutine(WaitForZombieClear());CreatZombies(ref waves[i]);}yield return StartCoroutine(WaitForZombieClear());yield return new WaitForSeconds(2f);WinGame();}IEnumerator WaitForZombieClear(){while (true){bool hasZombie = false;for (int row = 0; row < StageMap.ROW_MAX; row++){if (model.zombieList[row].Count!=0){hasZombie = true;break;}}if (hasZombie){yield return new WaitForSeconds(0.1f);}else{break;}}}IEnumerator WaitForWavePercentage(float percentage){while (true){if ((elapsedTime/playTime)>=percentage){break;}else{yield return 0;}}}IEnumerator UpdateProgress(){while (true){elapsedTime += Time.deltaTime; progressBar.GetComponent<ProgressBar>().SetProgress(elapsedTime/playTime); yield return 0;}}void ShowProgressBar(){progressBar.SetActive(true);StartCoroutine(UpdateProgress());}void CreatZombies(ref Wave wave){foreach (Wave.Data data in wave.zombieData){for (int i = 0; i < data.count; i++){CreatOneZombie(data.zombieType);}}}void CreatOneZombie(ZombieType type){GameObject zombie=null;switch (type){case ZombieType.Zombie1:zombie = Instantiate(zombie1);break; case ZombieType.ConeHeadZombie:zombie = Instantiate(ConeheadZombie);break;case ZombieType.BucketHeadZombie:zombie = Instantiate(BucketheadZombie);break; }int row = Random.Range(0, StageMap.ROW_MAX); zombie.transform.position = StageMap.SetZombiePos(row);zombie.GetComponent<ZombieMove>().row = row;zombie.GetComponent<SpriteDisplay>().SetOrderByRow(row);model.zombieList[row].Add(zombie);}void ProduceSun(){float x = Random.Range(StageMap.GRID_LEFT, StageMap.GRID_RIGTH);float y = Random.Range(StageMap.GRID_BOTTOM, StageMap.GRID_TOP);float startY = StageMap.GRID_TOP + 1.5f;GameObject sun = Instantiate(sunPrefab);sun.transform.position=new Vector3(x,startY,0);MoveBy move = sun.AddComponent<MoveBy>();move.offset=new Vector3(0,y-startY,0);move.time = (startY - y)/1.0f;move.Begin();}void LoseGame(){gameLabel.GetComponent<GameTips>().ShowLostTip();GetComponent<HandlerForPlants>().enabled = false;CancelInvoke("ProduceSun");AudioManager.GetInstance().PlayMusic(loseMusic,false);}void WinGame(){CancelInvoke("ProduceSun");AudioManager.GetInstance().PlayMusic(winMusic, false);Invoke("GotoNextStage",3.0f);}void GotoNextStage(){ SceneManager.LoadScene(nextStage);}
}
下载链接:PlantsVsZombies: 经典游戏:植物大战僵尸
相关文章:
经典游戏案例:仿植物大战僵尸
学习目标:仿植物大战僵尸核心玩法实现 游戏画面 项目结构目录 部分核心代码 using System; using System.Collections; using System.Collections.Generic; using UnityEngine; using UnityEngine.SceneManagement; using Random UnityEngine.Random;public enum…...
[Day 18] 區塊鏈與人工智能的聯動應用:理論、技術與實踐
強化學習與生成對抗網絡(GAN) 引言 強化學習 (Reinforcement Learning, RL) 和生成對抗網絡 (Generative Adversarial Networks, GANs) 是現代人工智能中的兩大關鍵技術。強化學習使得智能體可以通過與環境交互學習最佳行動策略,而生成對抗網絡則通過兩個相互競爭…...
【Mac】DMG Canvas for mac(DMG镜像制作工具)软件介绍
软件介绍 DMG Canvas 是一款专门用于创建 macOS 磁盘映像文件(DMG)的软件。它的主要功能是让用户可以轻松地设计、定制和生成 macOS 上的安装器和磁盘映像文件,以下是它的一些主要特点和功能。 主要特点和功能 1. 用户界面设计 DMG Canva…...
RAG分块方法 从固定大小到自然语言处理分块——深入研究文本分块技术
发掘文本分块-准确的搜索结果和更智能的语言模型背后的秘诀,通过了解如何有效地分块文本,我们可以改进索引文档、处理用户查询和利用搜索结果的方式。准备好揭开文本分块的秘密了吗? 一、了解分块 分块是一种旨在嵌入尽可能少噪音的内容,同…...
FFmpeg 系列
📚 此篇文章是先引入ffmpeg的概念以及主要的功能,后面会根据每一个特点进行详解,喜欢ffmpeg的可以持续关注。 ffmpeg是什么? FFmpeg 是一个开源的跨平台音视频处理工具,它可以用来录制、转换以及流化音视频内容。具体…...
240626_昇思学习打卡-Day8-稀疏矩阵
240626_昇思学习打卡-Day8-稀疏矩阵 稀疏矩阵 在一些应用场景中,比如训练二值化图像分割时,图像的特征是稀疏的,使用一堆0和极个别的1表示这些特征即费事又难看,此时就可以使用稀疏矩阵。通过参考大佬博文,结合个人理…...
Docker: 使用容器化数据库
使用容器化数据库 使用本地容器化数据库提供了灵活性和简易的设置,使您能够在不需要传统数据库安装开销的情况下,紧密模拟生产环境。Docker 简化了这一过程,只需几条命令就可以在隔离的容器中部署、管理和扩展数据库。 在本指南中,您将学习如何: 运行本地容器化数据库访…...
Oracle对用户敏感数据进行编码处理
由于系统运行时间比较长,没有对用户的身份证号、邮箱、手机号进行脱敏处理,后期对数据进行了编码。 更新表数据 sql UPDATE sys_staff SET MOBIL_PHONE CASEWHEN MOBIL_PHONE IS NULL THEN ELSE utl_raw.cast_to_varchar2(utl_encode.base64_encode(ut…...
VXLAN详解:概念、架构、原理、搭建过程、常用命令与实战案例
一、VXLAN概述 1.1 VXLAN的定义 VXLAN(Virtual Extensible LAN,虚拟可扩展局域网)是一种网络虚拟化技术,通过在现有IP网络上创建虚拟网络,使数据中心可以实现大规模的网络隔离和扩展。VXLAN使用MAC-in-UDP封装技术&a…...
Redis-数据类型-Hash
文章目录 1、查看redis是否启动2、通过客户端连接redis3、切换到db3数据库4、插入新数据返回15、获取指定哈希(hash)对象的所有字段(field)名6、获取存储在指定哈希(hash)对象中的所有字段(fiel…...
基于redisson实现tomcat集群session共享
目录 1、环境 2、修改server.xml 3、修改context.xml 4、新增redisson配置文件 5、下载并复制2个Jar包到Tomcat Lib目录中 6、 安装redis 7、配置nginx负载均衡 8、配置测试页面 9、session共享测试验证 前言: 上篇中,Tomcat session复制及ses…...
postgres数据库的流复制
1. 流复制和逻辑复制的差异 逻辑复制和流复制最直观的不同是,逻辑复制支持表级别复制区分点事原理不同 逻辑日志是在wal日志产生的数据库上,由逻辑解析模块对wal日志进行初步的解析,解析结果是ReorderBufferChange(理解为HeapTup…...
Dxf库中的DL_Extrusion类
类DL_Extrusion DL_Extrusion 是 DXF 库中的一个类,用于表示三维实体的扩展信息。在 DXF 文件中,DL_Extrusion 类通常用于表示具有高度的三维图形实体,如立方体、圆柱体等,以及其它具有体积的几何对象。 以下是一个简单的示例代…...
“ONLYOFFICE 8.1版本评测:功能更强大,用户体验更佳”
最新版本的在线编辑器已经发布 ONLYOFFICE在线编辑器的最新版本8.1已经发布,整个套件带来了30多个新功能和432个bug修复。这个强大的文档编辑器支持处理文本文档、电子表格、演示文稿、可填写的表单和PDF,并允许多人在线协作,同时支持AI集成…...
搜维尔科技:【研究】触觉手套比控制器更能带来身临其境、更安全、更高效的虚拟体验
自然交互可提高VR模拟的有效性。研究表明,触觉手套比控制器更能带来身临其境、更安全、更高效的虚拟体验。 以下是验证 医疗培训中的触觉技术 “ 95.5%的参与者表示触摸是 XR 教育的重要组成部分,90.9% 的参与者表示 XR 触觉将提供一个安全的学习场所。…...
【小学期】实体类设计——以学生管理系统为例
项目目录中的位置 将Student.java文件放在src/model目录中,即: student_management │ ├── src │ ├── model │ │ ├── Student.java // 这里是Student实体类 │ │ └── StudentDAO.java │ │ │ ├── view │ │ …...
Java测试类
在Java中,为了编写测试类,通常使用JUnit框架。 1. 首先,创建一个名为Calculator的简单Java类,它包含一个方法add用于计算两个整数的和: public class Calculator {public int add(int a, int b) {return a b;} } 2.…...
python 中面向对象编程:深入理解封装、继承和多态
在本章中,我们将深入探讨Python中的高级面向对象编程概念,包括封装、继承和多态。让我们开始吧! 目录 面向对象简介类和实例属性和方法继承和多态 高级面向对象概念私有变量使用 property使用 __slots__类的特殊成员__doc____call____str____…...
OpenCV练习(2)图像校正
1、傅里叶变换 霍夫变换 直线 角度 旋转2、边缘检测 霍夫变换 直线角度 旋转3、四点透视 角度 旋转4、检测矩形轮廓 角度 旋转 1.目的 实现类似全能扫面王的图像校正功能 2. 基于轮廓提取和透射变换 基于轮廓提取和透射变换的矫正算法更适用于车牌、身份证、人民…...
Excel中的“点选输入”——次级下拉列表创建
在Excel中,用“数据验证”功能可以设置下拉列表,二级下拉列表需要设置公式。 (笔记模板由python脚本于2024年06月16日 18:36:37创建,本篇笔记适合经常使用Excel处理数据的coder翻阅) 【学习的细节是欢悦的历程】 Python 官网:http…...
零依赖STL转STEP工具:5分钟实现3D格式无缝转换的完整指南
零依赖STL转STEP工具:5分钟实现3D格式无缝转换的完整指南 【免费下载链接】stltostp Convert stl files to STEP brep files 项目地址: https://gitcode.com/gh_mirrors/st/stltostp 在数字化制造和工程设计的交叉领域,STL到STEP格式转换已成为连…...
AI智能体操作系统Agent-OS:架构、实现与生产部署指南
1. 项目概述:一个为AI智能体设计的操作系统最近在AI智能体开发领域,一个名为“Agent-OS”的项目引起了我的注意。这个项目由 factspark23-hash 团队开源,它不是一个传统意义上的操作系统,比如Windows或Linux,而是一个专…...
从自动化到智能代理:构建家庭智能中枢的架构与实践
1. 项目概述与核心价值最近在折腾智能家居和自动化流程,发现市面上的很多方案要么太“重”,需要依赖特定品牌的生态闭环;要么太“散”,各种工具和脚本堆在一起,管理起来一团乱麻。直到我遇到了一个名为“Home-agent-as…...
本事同根生,相煎何太急
简 介: 【轮腿组比赛难度调整建议】针对智能车竞赛轮腿穿越组室外赛道的视觉识别难题,参赛选手提出以下建议:1.科目三元素应避开塑胶跑道线干扰区域;2.当前轮腿组任务量(机械、控制、导航、视觉等)已远超往…...
cliclick 开发者指南:从源码编译到自定义Action开发
cliclick 开发者指南:从源码编译到自定义Action开发 【免费下载链接】cliclick macOS CLI tool for emulating mouse and keyboard events 项目地址: https://gitcode.com/gh_mirrors/cl/cliclick cliclick 是一款强大的 macOS 命令行工具,用于模…...
第5章 集群初始化
本章说明: 集群初始化是 Kubernetes 部署过程中最核心的一步。本章使用 kubeadm 在 master01 节点上初始化高可用集群控制平面。初始化时需要指定 VIP(192.168.3.59:6443)作为控制平面统一入口,这样后续加入的其他 Master 节点和 Worker 节点都通过 VIP 访问 API Server,…...
基于规则引擎的自动化文件管理工具smartcat实战指南
1. 项目概述:一个智能化的文件分类与归档工具最近在整理个人电脑和服务器上的文件时,我又一次陷入了混乱。下载文件夹里塞满了各种格式的文档、图片、压缩包,项目目录下混杂着不同版本的代码、设计稿和会议记录。手动分类不仅耗时,…...
本地AI对话伴侣实战:Electron+llama.cpp部署与调优指南
1. 项目概述:一个桌面端的本地AI对话伴侣最近在折腾本地大语言模型(LLM)的时候,发现了一个挺有意思的项目:ItsPi3141/alpaca-electron。简单来说,这是一个用 Electron 框架打包的桌面应用程序,它…...
面向高校的基于算法的发明专利申请写作方法
发明专利作为国家和高校认可的成果形式之一,其申请和授权一直受到教师和学生们的高度重视;基于算法的发明专利作为发明专利的重要分支,每年都有大量的算法专利被授权或者拒绝。虽然高校的教师对论文写作非常熟悉,但是发明专利的写…...
700MHz 5G网络DTMB干扰实战:从测量到规避的完整解决方案
1. 项目概述:直面700MHz网络中的DTMB干扰挑战在5G网络的深度覆盖战役中,700MHz频段因其卓越的穿透能力和广阔的覆盖范围,被寄予厚望,成为解决偏远地区和室内深度覆盖难题的“黄金频段”。然而,理想很丰满,现…...
