【制作100个unity游戏之29】使用unity复刻经典游戏《愤怒的小鸟》(完结,附带项目源码)(上)

简介: 【制作100个unity游戏之29】使用unity复刻经典游戏《愤怒的小鸟》(完结,附带项目源码)

前言

欢迎来到【制作100个Unity游戏】系列!本系列将引导您一步步学习如何使用Unity开发各种类型的游戏。在这第29篇中,我们将探索如何用unity复刻经典游戏《愤怒的小鸟》,我会附带项目源码,以便你更好理解它。

素材下载

链接:https://pan.baidu.com/s/1hBbnRkGuf44jsQQBZSn96g?pwd=h73r

提取码:h73r

简单搭建环境

修改图片配置并切图,修改最大尺寸是为了让图片放大不那么模糊

背景图片和地面草地可能不够长,可以修改绘制模式改成平铺,修改宽度

控制小鸟

新增Bird,控制小鸟跟随鼠标移动

public enum BirdState
{
    Waiting,//等待
    BeforeShoot,//发射前
    AfterShoot//发射后
}
public class Bird : MonoBehaviour
{
    public BirdState state = BirdState.BeforeShoot;
    private bool isMouseDown = false; //是否按下
    void Update()
    {
        switch (state)
        {
            case BirdState.Waiting:
                break;
            case BirdState.BeforeShoot:
                MoveControl();
                break;
            case BirdState.AfterShoot:
                break;
            default:
                break;
        }
    }
    //按下触发事件
    private void OnMouseDown()
    {
        if (state == BirdState.BeforeShoot)
        {
            isMouseDown = true;
        }
    }
    //抬起触发事件
    private void OnMouseUp()
    {
        if (state == BirdState.BeforeShoot)
        {
            isMouseDown = false;
        }
    }
    //跟随鼠标
    private void MoveControl()
    {
        if (isMouseDown) transform.position = GetMousePosition();
    }
    //屏幕坐标转世界坐标
    private Vector3 GetMousePosition()
    {
        Vector3 position = Camera.main.ScreenToWorldPoint(Input.mousePosition);
        return new Vector2(position.x, position.y);
    }
}

配置,记得给小鸟加上碰撞体

效果

生成弹簧

新增Slingshot代码,控制弹簧得生成

public class Slingshot : MonoBehaviour
{
    public static Slingshot Instance;
    public LineRenderer leftLineRenderer;
    public LineRenderer rightLineRenderer;
    public Transform leftPoint;
    public Transform rightPoint;
    // private Transform centerPoint;
    private bool isDrawing = false;//是否画线
    private Transform birdTransform;//鸟
    private void Awake() {
        Instance = this;
    }
    private void Update()
    {
        if (isDrawing)
        {
            Draw();
        }
    }
    public void StartDraw(Transform birdTransform)
    {
        isDrawing = true;
        this.birdTransform = birdTransform;
    }
    public void EndDraw()
    {
        isDrawing = false;
    }
    public void Draw()
    {
        leftLineRenderer.SetPosition(0, birdTransform.position);
        leftLineRenderer.SetPosition(1, leftPoint.position);
        rightLineRenderer.SetPosition(0, birdTransform.position);
        rightLineRenderer.SetPosition(1, rightPoint.position);
    }
}

配置

效果

可以看到现在线得终点是在鸟的中位置,我们希望在鸟的后面位置生成

修改代码,按碰撞器的半径进行偏移

public void Draw()
{
    Vector2 birdPosition = birdTransform.position + birdTransform.GetComponent<CircleCollider2D>().radius *  (birdTransform.position-transform.position).normalized;
    leftLineRenderer.SetPosition(0, birdPosition);
    leftLineRenderer.SetPosition(1, leftPoint.position);
    rightLineRenderer.SetPosition(0, birdPosition);
    rightLineRenderer.SetPosition(1, rightPoint.position);
}

效果

限制小鸟的控制范围

修改Bird

public float maxDistance = 1.5f;//限制拉动距离
//屏幕坐标转世界坐标
private Vector3 GetMousePosition()
{
    Vector3 position = Camera.main.ScreenToWorldPoint(Input.mousePosition);
    position.z = 0;
    // 获取弹弓的中心位置
    Vector3 centerPosition = Slingshot.Instance.gameObject.transform.position;
    //计算鼠标指向的方向
    Vector3 mouseDir = position - centerPosition;
    //计算鼠标指向的距离
    float distance = mouseDir.magnitude;
    // 如果距离超过最大距离,则限制在最大距离范围内
    if(distance > maxDistance){
        position = mouseDir.normalized * maxDistance + centerPosition;
    }
    return position;
}

效果

弹簧线的显示隐藏

修改Slingshot

private void Awake()
{
    Instance = this;
    HideLine();
}
public void StartDraw(Transform birdTransform)
{
    isDrawing = true;
    this.birdTransform = birdTransform;
    ShowLine();
}
public void EndDraw()
{
    isDrawing = false;
    HideLine();
}
void HideLine()
{
    leftLineRenderer.enabled = false;
    rightLineRenderer.enabled = false;
}
void ShowLine()
{
    leftLineRenderer.enabled = true;
    rightLineRenderer.enabled = true;
}

效果

飞行

给鸟添加刚体,默认为静态

修改Slingshot,配置射击点

public Transform shootPoint;

配置

修改Bird

public float flySpeed;//飞行速度
public float force = 10f;//力大小
private Vector2 m_pushSpeed;//力向量
//抬起触发事件
private void OnMouseUp()
{
    if (state == BirdState.BeforeShoot)
    {
        isMouseDown = false;
        Slingshot.Instance.EndDraw();
        Fly();
    }
}
private void MoveControl()
{
    if (isMouseDown)
    {
        transform.position = GetMousePosition();//跟随鼠标
        Vector3 mouseDir = Slingshot.Instance.shootPoint.position - transform.position;
        m_pushSpeed = mouseDir.normalized * mouseDir.magnitude * force;//力向量
    }
}
void Fly(){
    rb.bodyType = RigidbodyType2D.Dynamic;
    rb.AddForce(m_pushSpeed, ForceMode2D.Impulse);
    state = BirdState.AfterShoot;
}

配置

效果

新增木头

添加刚体和碰撞体

木头销毁

新增Destructiable,控制木头销毁

public class Destructiable : MonoBehaviour
{
    public int maxHP = 100;// 最大生命值
    private int currentHP;// 当前生命值
    private void Start()
    {
        currentHP = maxHP;
    }
    private void OnCollisionEnter2D(Collision2D collision)
    {
        // 获取当前碰撞的相对速度
        Vector2 relativeVelocity = collision.relativeVelocity;
        // 计算相对速度的大小(标量值)
        float impactForce = relativeVelocity.magnitude;
        if(impactForce <= 1) return;//过滤小伤害
        // 根据相对速度大小计算伤害值,并减少当前生命值
        currentHP -= (int)(impactForce * 5);
        if (currentHP <= 0)Destroy(gameObject);
    }
}

效果

不同血量的木头状态

修改Destructiable,控制不同血量显示不同阶段的图片

public List<Sprite> spriteList;//不同阶段的图片
private SpriteRenderer spriteRenderer;
spriteRenderer = GetComponent<SpriteRenderer>();
private void OnCollisionEnter2D(Collision2D collision)
{
    // 获取当前碰撞的相对速度
    Vector2 relativeVelocity = collision.relativeVelocity;
    // 计算相对速度的大小(标量值)
    float impactForce = relativeVelocity.magnitude;
    if(impactForce <= 1) return;//过滤小伤害
    // 根据相对速度大小计算伤害值,并减少当前生命值
    currentHP -= (int)(impactForce * 5);
    if (currentHP <= 0)
    {
        Destroy(gameObject);
    }
    else
    {
        //计算剩余生命值的比例
        float healthRatio = (float)currentHP / maxHP;
        //计算阶段索引
        int index = (int)((1 - healthRatio) * spriteList.Count) - 1;
        if (index != -1) spriteRenderer.sprite = spriteList[index];
    }
}

配置

效果

配置更多物品

通过跳转血量,实现易碎的冰块和坚固的砖块

爆炸效果

配置

修改Destructiable,物体销毁时调用

//爆炸特效
GameObject prefabs = Resources.Load("Prefabs/VFX/爆炸烟雾特效") as GameObject;
GameObject go = Instantiate(prefabs, transform.position, Quaternion.identity);
Destroy(go, 1f);

效果

创建敌人的小猪

新增脚本pig,继承Destructiable

public class Pig : Destructiable {}

配置

效果

创建多个小鸟循环

修改Bird,

public enum BirdState
{
    Waiting,//等待
    BeforeShoot,//发射前
    AfterShoot,//发射后
    WaitToDie//死亡
}
void Update()
{
    switch (state)
    {
        case BirdState.Waiting:
            break;
        case BirdState.BeforeShoot:
            MoveControl();
            break;
        case BirdState.AfterShoot:
            StopControl();
            break;
        case BirdState.WaitToDie:
            break;
        default:
            break;
    }
}
void StopControl(){
    if(rb.velocity.magnitude < 0.1f){
        state = BirdState.WaitToDie;
        Invoke("LoadNextBird", 1f);
    }
}
//加载下一只鸟
protected void LoadNextBird(){
    Destroy(gameObject);
    //爆炸特效
    GameObject prefabs = Resources.Load("Prefabs/VFX/爆炸烟雾特效") as GameObject;
    GameObject go = Instantiate(prefabs, transform.position, Quaternion.identity);
    Destroy(go, 1f);
    GameManager.Instance.LoadNextBird();
}
//设置开始小鸟
public void SetStart(Vector3 position){
    state = BirdState.BeforeShoot;
    transform.position = position;
}

新增GameManager,

public class GameManager : MonoBehaviour
{
    public static GameManager Instance { get; private set; }
    public Bird[] birdList;
    private int index = -1;
    
    private void Awake()
    {
        Instance = this;
    }
    
    private void Start()
    {
        //在当前场景中查找所有带Bird脚本的对象,不进行排序
        birdList = FindObjectsByType<Bird>(FindObjectsSortMode.None);
        LoadNextBird();
    }
    public void LoadNextBird()
    {
        index++;
        GameEnd();
    }
    public void OnPigDead()
    {
        pigTotalCount--;
        if (pigTotalCount <= 0)
        {
            GameEnd();
        }
    }
    void GameEnd()
    {
        print("游戏结束");
    }
}

效果

游戏结束

修改GameManager,实现猪全部死亡或者鸟全部用完结束游戏

private int pigTotalCount;//剩余猪的数量
private void Start()
{
    //在当前场景中查找所有带Bird脚本的对象,不进行排序
    birdList = FindObjectsByType<Bird>(FindObjectsSortMode.None);
    pigTotalCount = FindObjectsByType<Pig>(FindObjectsSortMode.None).Length;
    LoadNextBird();
}
public void LoadNextBird()
{
    index++;
    if (index >= birdList.Length)
    {
        GameEnd();
    }
    else
    {
        birdList[index].SetStart(Slingshot.Instance.shootPoint.transform.position);
    }
}
public void OnPigDead()
{
    pigTotalCount--;
    if (pigTotalCount <= 0)
    {
        GameEnd();
    }
}
void GameEnd()
{
    print("游戏结束");
}

修改Destructiable

public virtual void Dead(){
    Destroy(gameObject);
    //爆炸特效
    GameObject prefabs = Resources.Load("Prefabs/VFX/爆炸烟雾特效") as GameObject;
    GameObject go = Instantiate(prefabs, transform.position, Quaternion.identity);
    Destroy(go, 1f);
}
• 1
• 2
• 3
• 4
• 5
• 6
• 7

修改Pig

public class Pig : Destructiable {
    public override void Dead()
    {
        base.Dead();
        GameManager.Instance.OnPigDead();
    }
}

效果

目录
相关文章
|
2月前
|
测试技术 C# 图形学
掌握Unity调试与测试的终极指南:从内置调试工具到自动化测试框架,全方位保障游戏品质不踩坑,打造流畅游戏体验的必备技能大揭秘!
【9月更文挑战第1天】在开发游戏时,Unity 引擎让创意变为现实。但软件开发中难免遇到 Bug,若不解决,将严重影响用户体验。调试与测试成为确保游戏质量的最后一道防线。本文介绍如何利用 Unity 的调试工具高效排查问题,并通过 Profiler 分析性能瓶颈。此外,Unity Test Framework 支持自动化测试,提高开发效率。结合单元测试与集成测试,确保游戏逻辑正确无误。对于在线游戏,还需进行压力测试以验证服务器稳定性。总之,调试与测试贯穿游戏开发全流程,确保最终作品既好玩又稳定。
98 4
|
2月前
|
前端开发 图形学 开发者
【独家揭秘】那些让你的游戏瞬间鲜活起来的Unity UI动画技巧:从零开始打造动态按钮,提升玩家交互体验的绝招大公开!
【9月更文挑战第1天】在游戏开发领域,Unity 是最受欢迎的游戏引擎之一,其强大的跨平台发布能力和丰富的功能集让开发者能够迅速打造出高质量的游戏。优秀的 UI 设计对于游戏至关重要,尤其是在手游市场,出色的 UI 能给玩家留下深刻的第一印象。Unity 的 UGUI 系统提供了一整套解决方案,包括 Canvas、Image 和 Button 等组件,支持添加各种动画效果。
132 3
|
2月前
|
设计模式 存储 人工智能
深度解析Unity游戏开发:从零构建可扩展与可维护的游戏架构,让你的游戏项目在模块化设计、脚本对象运用及状态模式处理中焕发新生,实现高效迭代与团队协作的完美平衡之路
【9月更文挑战第1天】游戏开发中的架构设计是项目成功的关键。良好的架构能提升开发效率并确保项目的长期可维护性和可扩展性。在使用Unity引擎时,合理的架构尤为重要。本文探讨了如何在Unity中实现可扩展且易维护的游戏架构,包括模块化设计、使用脚本对象管理数据、应用设计模式(如状态模式)及采用MVC/MVVM架构模式。通过这些方法,可以显著提高开发效率和游戏质量。例如,模块化设计将游戏拆分为独立模块。
180 3
|
3月前
|
图形学 C#
超实用!深度解析Unity引擎,手把手教你从零开始构建精美的2D平面冒险游戏,涵盖资源导入、角色控制与动画、碰撞检测等核心技巧,打造沉浸式游戏体验完全指南
【8月更文挑战第31天】本文是 Unity 2D 游戏开发的全面指南,手把手教你从零开始构建精美的平面冒险游戏。首先,通过 Unity Hub 创建 2D 项目并导入游戏资源。接着,编写 `PlayerController` 脚本来实现角色移动,并添加动画以增强视觉效果。最后,通过 Collider 2D 组件实现碰撞检测等游戏机制。每一步均展示 Unity 在 2D 游戏开发中的强大功能。
172 6
|
3月前
|
图形学 缓存 算法
掌握这五大绝招,让您的Unity游戏瞬间加载完毕,从此告别漫长等待,大幅提升玩家首次体验的满意度与留存率!
【8月更文挑战第31天】游戏的加载时间是影响玩家初次体验的关键因素,特别是在移动设备上。本文介绍了几种常见的Unity游戏加载优化方法,包括资源的预加载与异步加载、使用AssetBundles管理动态资源、纹理和模型优化、合理利用缓存系统以及脚本优化。通过具体示例代码展示了如何实现异步加载场景,并提出了针对不同资源的优化策略。综合运用这些技术可以显著缩短加载时间,提升玩家满意度。
136 5
|
3月前
|
图形学 开发者 存储
超越基础教程:深度拆解Unity地形编辑器的每一个隐藏角落,让你的游戏世界既浩瀚无垠又细节满满——从新手到高手的全面技巧升级秘籍
【8月更文挑战第31天】Unity地形编辑器是游戏开发中的重要工具,可快速创建复杂多变的游戏环境。本文通过比较不同地形编辑技术,详细介绍如何利用其功能构建广阔且精细的游戏世界,并提供具体示例代码,展示从基础地形绘制到植被与纹理添加的全过程。通过学习这些技巧,开发者能显著提升游戏画面质量和玩家体验。
147 3
|
3月前
|
图形学 机器学习/深度学习 人工智能
颠覆传统游戏开发,解锁未来娱乐新纪元:深度解析如何运用Unity引擎结合机器学习技术,打造具备自我进化能力的智能游戏角色,彻底改变你的游戏体验——从基础设置到高级应用全面指南
【8月更文挑战第31天】本文探讨了如何在Unity中利用机器学习增强游戏智能。作为领先的游戏开发引擎,Unity通过ML-Agents Toolkit等工具支持AI代理的强化学习训练,使游戏角色能自主学习完成任务。文章提供了一个迷宫游戏示例及其C#脚本,展示了环境观察、动作响应及奖励机制的设计,并介绍了如何设置训练流程。此外,还提到了Unity与其他机器学习框架(如TensorFlow和PyTorch)的集成,以实现更复杂的游戏玩法。通过这些技术,游戏的智能化程度得以显著提升,为玩家带来更丰富的体验。
64 1
|
3月前
|
图形学 数据可视化 开发者
超实用Unity Shader Graph教程:从零开始打造令人惊叹的游戏视觉特效,让你的作品瞬间高大上,附带示例代码与详细步骤解析!
【8月更文挑战第31天】Unity Shader Graph 是 Unity 引擎中的强大工具,通过可视化编程帮助开发者轻松创建复杂且炫酷的视觉效果。本文将指导你使用 Shader Graph 实现三种效果:彩虹色渐变着色器、动态光效和水波纹效果。首先确保安装最新版 Unity 并启用 Shader Graph。创建新材质和着色器图谱后,利用节点库中的预定义节点,在编辑区连接节点定义着色器行为。
244 0
|
3月前
|
图形学
小功能⭐️获取Unity游戏物体上,所挂载组件的名称
小功能⭐️获取Unity游戏物体上,所挂载组件的名称
|
2月前
|
图形学 C++ C#
Unity插件开发全攻略:从零起步教你用C++扩展游戏功能,解锁Unity新玩法的详细步骤与实战技巧大公开
【8月更文挑战第31天】Unity 是一款功能强大的游戏开发引擎,支持多平台发布并拥有丰富的插件生态系统。本文介绍 Unity 插件开发基础,帮助读者从零开始编写自定义插件以扩展其功能。插件通常用 C++ 编写,通过 Mono C# 运行时调用,需在不同平台上编译。文中详细讲解了开发环境搭建、简单插件编写及在 Unity 中调用的方法,包括创建 C# 封装脚本和处理跨平台问题,助力开发者提升游戏开发效率。
200 0