【unity实战】一个通用的FPS枪支不同武器射击控制脚本

简介: 【unity实战】一个通用的FPS枪支不同武器射击控制脚本

前言

实现FPS枪支不同武器效果,比如手枪,喷子,狙击枪,机枪,其实我最开始的想法是先做一个基类脚本,写一些公共属性和方法,然后再起不同的武器脚本这个基础基类,实现不同的武器效果。

这样的实现思路其实是没什么问题的,直到我看到这个视频:https://www.youtube.com/watch?v=bqNW08Tac0Y,作者只用一个脚本就实现了不同的武器效果更加方便,下面我就参考一下作者的思路实现一下大致的效果。

顺带说一下,在第一人称射击(FPS)游戏中实现子弹射击效果,可以通过不同的技术和方法来完成。以下是几种常见的实现方式:

  1. 射线投射(Raycasting):这是最常用的方法之一。射线投射意味着从枪口发出一个虚拟的射线,并检测这个射线与游戏世界中的对象之间的交互。如果射线与某个对象相交,那么就可以认为子弹击中了该对象。实现步骤:
  • 从玩家的摄像机或枪口位置发出一条射线。
  • 使用物理引擎提供的射线投射功能来检测射线路径上的碰撞。
  • 如果射线与对象相交,根据交互结果执行相应的逻辑,比如扣除生命值、播放受击动画等。
  • 在射击点显示击中效果,如粒子效果或贴图。
  1. 抛射物模拟(Projectile Simulation):对于需要模拟子弹飞行轨迹的情况,比如远距离狙击、火箭筒或者抛射武器,可以使用抛射物模拟。实现步骤:
  • 创建一个子弹实体,并赋予它初始速度和方向。
  • 通过物理引擎模拟子弹的飞行轨迹,考虑重力、空气阻力等因素。
  • 检测子弹与其他对象的碰撞,并在碰撞发生时处理相应的逻辑。
  • 在子弹飞行过程中可以添加轨迹效果,如拖尾。

每种方法都有其适用场景和优缺点。射线投射适合快速射击和近距离交火,抛射物模拟适合远距离和弧线射击。在实际开发中,这些方法可以组合使用,以达到最佳的效果。

模型素材

不会配置模型可以看我之前的文章,进行下载和配置:

unity中导入下载的3D模型及albedo/baseColor、normal 、AO/Occlus、metallic、roughness贴图纹理设置

文章用到的粒子火光特效

https://assetstore.unity.com/packages/vfx/particles/legacy-particle-pack-73777

射击效果

[Tooltip("是否正在射击")]
bool shooting;
[Tooltip("是否允许按住射击")]
public bool allowButtonHold;
[Tooltip("是否可以射击")]
bool readyToShoot;
[Tooltip("是否在换弹")]
bool reloading;
[Tooltip("弹夹容量")]
public int magazineSize;
[Tooltip("当前弹夹容量")]
public int bulletsLeft;
[Tooltip("储备弹药容量")]
public int reservedAmmoCapacity = 300;
[Tooltip("当前剩余射击发射的子弹数")]
public int bulletsShot;
[Tooltip("枪口火焰特效")]
public ParticleSystem muzzleFlash;
[Tooltip("子弹击中效果")]
public GameObject bulletHoleGraphic;
[Tooltip("射击间隔时间")]
public float timeBetweenShooting;
[Tooltip("连发射击之间的间隔时间")]
public float timeBetweenShots;
[Tooltip("射击时的散布度")]
public float spread;
[Tooltip("射击的最大距离")]
public float range;
[Tooltip("每次射击发射的子弹数")]
public int bulletsPerTap;
[Tooltip("是否允许按住射击")]
public bool allowButtonHold;
[Tooltip("每次射击造成的伤害")]
public int damage;  // 伤害
public Camera fpsCam;
private void Awake()
{
    bulletsLeft = magazineSize;
    readyToShoot = true;
}
    
private void Update()
{
    MyInput();
}
    
private void MyInput()
{
    if (allowButtonHold)
        shooting = Input.GetKey(KeyCode.Mouse0);
    else
        shooting = Input.GetKeyDown(KeyCode.Mouse0);
    // 射击
    if (readyToShoot && shooting && !reloading && bulletsLeft > 0)
    {
        bulletsShot = bulletsPerTap;
        Shoot();
    }
}
    
private void Shoot()
{
    readyToShoot = false;
    
    // 散布
    float x = Random.Range(-spread, spread);
    float y = Random.Range(-spread, spread);
    // 计算带有散布的射击方向
    Vector3 direction = fpsCam.transform.forward + fpsCam.transform.TransformDirection(new Vector3(x, y, 0));
    // 射线检测
    if (Physics.Raycast(fpsCam.transform.position, direction, out RaycastHit rayHit, range))
    {
      //场景显示红线,方便调试查看
      Debug.DrawLine(fpsCam.transform.position, rayHit.point, Color.red, 10f);
        Debug.Log(rayHit.collider.name);
        muzzleFlash.Play();//枪口火焰/火光
        //TODO:相机震动
        if (rayHit.collider.CompareTag("Enemy"))
        {
            Debug.Log("击中敌人");
            Rigidbody rb = rayHit.transform.GetComponent<Rigidbody>();
            if (rb != null)
            {
                rb.constraints = RigidbodyConstraints.None; // 解除刚体约束
                rb.AddForce(transform.parent.transform.forward * 500); // 给敌人施加一个力
            }
            // 击中敌人特效
            //使用 LookRotation() 方法来让子弹孔特效朝向被击中表面的法线方向。其中 rayHit.normal 是表示被击中表面法线方向的向量
            var res1 = Instantiate(bulletHoleGraphic, rayHit.point, Quaternion.LookRotation(rayHit.normal));
            Destroy(res1, 0.5f);
            
      //TODO:扣血
        }
    }
    bulletsLeft--;
    bulletsShot--;
    Invoke("ResetShot", timeBetweenShooting);
    if (bulletsShot > 0 && bulletsLeft > 0)
        Invoke("Shoot", timeBetweenShots);
}
private void ResetShot()
{
    readyToShoot = true;
}

换弹

private void MyInput()
{
   //。。。
    if (Input.GetKeyDown(KeyCode.R) && bulletsLeft < magazineSize && !reloading)
        Reload();
}
//换弹
private void Reload()
{
    reloading = true;
    Invoke("ReloadFinished", reloadTime);
}
private void ReloadFinished()
{
    if (reservedAmmoCapacity <= 0) return;
    //计算需要填装的子弹数=1个弹匣子弹数-当前弹匣子弹数
    int bullectToLoad = magazineSize - bulletsLeft;
    //计算备弹需扣除子弹数
    int bullectToReduce = (reservedAmmoCapacity >= bullectToLoad) ? bullectToLoad : reservedAmmoCapacity;
    reservedAmmoCapacity -= bullectToReduce;//减少备弹数
    bulletsLeft += bullectToReduce;//当前子弹数增加
    bulletsLeft = magazineSize;
    reloading = false;
}

瞄准

private void MyInput()
{
    //。。。
    //瞄准
    DetermineAim();
}
void DetermineAim()
{
    Vector3 target = normalLocalPosition; // 默认目标位置为正常瞄准时的本地位置
    if (Input.GetMouseButton(1)){
      //spread = 0;//瞄准情况下我们通常可能会让射击散步值为0,这个看自己的情况而定
      target = aimingLocalPosition; // 如果按下鼠标右键,目标位置为瞄准时的本地位置
    }
    Vector3 desiredPosition = Vector3.Lerp(transform.localPosition, target, Time.deltaTime * aimSmoothing); // 使用插值平滑过渡到目标位置
    transform.localPosition = desiredPosition; // 更新枪支的本地位置
}

效果

开枪抖动效果

如果你的枪模型没有开枪动画的话,这个方法就很方便了

private void Shoot()
{
    transform.localPosition -= Vector3.forward * 0.1f; // 后坐力使枪支向后移动
    
  //。。。
}

设置显示文本

private void Update()
{
    //。。。
    SetUI();
}
    
// 设置文本
private void SetUI()
{
    text.SetText(bulletsLeft + " / " + reservedAmmoCapacity);
}

生成实体子弹

[Header("子弹")]
public float bulletForce = 100f;//子弹的力
public GameObject bulletPrefab;//子弹预制体
public GameObject BulletShootPoint;//子弹生成点
//实例化一个子弹
GameObject bullet = Instantiate(bulletPrefab, BulletShootPoint.transform.position, BulletShootPoint.transform.rotation);
//给子弹拖尾一个向前的速度力(加上射线打出去的偏移值)
bullet.GetComponent<Rigidbody>().velocity = (BulletShootPoint.transform.forward + direction) * bulletForce;

最终代码

public class GunSystem : MonoBehaviour
{
    public Camera fpsCam;
    [Header("枪械状态")]
    [Tooltip("是否正在射击")]
    bool shooting;
    [Tooltip("是否可以射击")]
    bool readyToShoot;
    [Tooltip("是否在换弹")]
    bool reloading;
    [Header("弹夹")]
    [Tooltip("弹夹容量")]
    public int magazineSize;
    [Tooltip("当前弹夹容量")]
    public int bulletsLeft;
    [Tooltip("储备弹药容量")]
    public int reservedAmmoCapacity = 300;
    [Tooltip("当前剩余射击发射的子弹数")]
    public int bulletsShot;
    [Header("射击")]
    [Tooltip("射击间隔时间")]
    public float timeBetweenShooting;
    [Tooltip("射击时的散布度")]
    public float spread;
    [Tooltip("射击的最大距离")]
    public float range;
    [Tooltip("每次射击发射的子弹数")]
    public int bulletsPerTap;
    [Tooltip("是否允许按住射击")]
    public bool allowButtonHold;
    [Tooltip("每次射击造成的伤害")]
    public int damage;  // 伤害
    [Tooltip("装填弹药的时间")]
    public float reloadTime;
    [Tooltip("连发射击之间的间隔时间")]
    public float timeBetweenShots;
    [Header("瞄准")]
    [Tooltip("正常情况的本地位置")]
    public Vector3 normalLocalPosition;
    [Tooltip("瞄准时的本地位置")]
    public Vector3 aimingLocalPosition;
    [Tooltip("瞄准过程的平滑度")]
    public float aimSmoothing = 10;
    
    [Header("效果")]
    [Tooltip("枪口火焰特效")]
    public ParticleSystem muzzleFlash;
    [Tooltip("子弹击中效果")]
    public GameObject bulletHoleGraphic;
    [Header("UI")]
    public TextMeshProUGUI text;  // 弹药显示文本
    private void Awake()
    {
        bulletsLeft = magazineSize;
        readyToShoot = true;
    }
    private void Update()
    {
        MyInput();
        SetUI();
    }
    // 设置文本
    private void SetUI()
    {
        text.SetText(bulletsLeft + " / " + reservedAmmoCapacity);
    }
    private void MyInput()
    {
        if (allowButtonHold)
            shooting = Input.GetKey(KeyCode.Mouse0);
        else
            shooting = Input.GetKeyDown(KeyCode.Mouse0);
        // 射击
        if (readyToShoot && shooting && !reloading && bulletsLeft > 0)
        {
            bulletsShot = bulletsPerTap;
            Shoot();
        }
        //换弹
        if (Input.GetKeyDown(KeyCode.R) && bulletsLeft < magazineSize && !reloading)
            Reload();
        //瞄准
        DetermineAim();
    }
    private void Shoot()
    {
        readyToShoot = false;
        transform.localPosition -= Vector3.forward * 0.1f; // 后坐力使枪支向后移动
        // 散布
        float x = Random.Range(-spread, spread);
        float y = Random.Range(-spread, spread);
        // 计算带有散布的射击方向
        Vector3 direction = fpsCam.transform.forward + fpsCam.transform.TransformDirection(new Vector3(x, y, 0));
        // 射线检测
        if (Physics.Raycast(fpsCam.transform.position, direction, out RaycastHit rayHit, range))
        {
            Debug.Log(rayHit.collider.name);
            muzzleFlash.Play();//枪口火焰/火光
                               //相机震动
            if (rayHit.collider.CompareTag("Enemy"))
            {
              //场景显示红线,方便调试查看
          Debug.DrawLine(fpsCam.transform.position, rayHit.point, Color.red, 10f);
                Debug.Log("击中敌人");
                // Rigidbody rb = rayHit.transform.GetComponent<Rigidbody>();
                // if (rb != null)
                // {
                //     rb.constraints = RigidbodyConstraints.None; // 解除刚体约束
                //     rb.AddForce(transform.parent.transform.forward * 500); // 给敌人施加一个力
                // }
                // 击中敌人特效
                //使用 LookRotation() 方法来让子弹孔特效朝向被击中表面的法线方向。其中 rayHit.normal 是表示被击中表面法线方向的向量
                var res = Instantiate(bulletHoleGraphic, rayHit.point, Quaternion.LookRotation(rayHit.normal));
                res.transform.parent = rayHit.transform;//设置父类
                //TODO:扣血
            }
        }
    
    //实例化一个子弹
        GameObject bullet = Instantiate(bulletPrefab, BulletShootPoint.transform.position, BulletShootPoint.transform.rotation);
        //给子弹拖尾一个向前的速度力(加上射线打出去的偏移值)
        bullet.GetComponent<Rigidbody>().velocity = (BulletShootPoint.transform.forward + direction) * bulletForce;
        bulletsLeft--;
        bulletsShot--;
        Invoke("ResetShot", timeBetweenShooting);
        if (bulletsShot > 0 && bulletsLeft > 0)
            Invoke("Shoot", timeBetweenShots);
    }
    void DetermineAim()
    {
        Vector3 target = normalLocalPosition; // 默认目标位置为正常瞄准时的本地位置
        if (Input.GetMouseButton(1)) target = aimingLocalPosition; // 如果按下鼠标右键,目标位置为瞄准时的本地位置
        Vector3 desiredPosition = Vector3.Lerp(transform.localPosition, target, Time.deltaTime * aimSmoothing); // 使用插值平滑过渡到目标位置
        transform.localPosition = desiredPosition; // 更新枪支的本地位置
    }
    private void ResetShot()
    {
        readyToShoot = true;
    }
    //换弹
    private void Reload()
    {
        reloading = true;
        Invoke("ReloadFinished", reloadTime);
    }
    private void ReloadFinished()
    {
        if (reservedAmmoCapacity <= 0) return;
        //计算需要填装的子弹数=1个弹匣子弹数-当前弹匣子弹数
        int bullectToLoad = magazineSize - bulletsLeft;
        //计算备弹需扣除子弹数
        int bullectToReduce = (reservedAmmoCapacity >= bullectToLoad) ? bullectToLoad : reservedAmmoCapacity;
        reservedAmmoCapacity -= bullectToReduce;//减少备弹数
        bulletsLeft += bullectToReduce;//当前子弹数增加
        bulletsLeft = magazineSize;
        reloading = false;
    }
}

不同武器射击效果

注意:这里为了方便,我就用一把枪做演示了

1. 手枪

参数配置

效果

2. 机枪

参数

效果

3. 狙击枪

参数,狙击枪其实和手枪参数差不多,可以就需要修改射击间隔时间、换弹时间和伤害

效果

4. 霰弹枪

参数

效果

5. 加特林

参数

效果

其他

可以看到其实还有很多功能没有实现,比如后座力或者放大镜等等效果,这篇文章说的已经够多了,后面我再单独做其他内容的探究吧!

感谢

【视频】https://www.youtube.com/watch?v=bqNW08Tac0Y

目录
相关文章
|
3月前
|
设计模式 存储 人工智能
深度解析Unity游戏开发:从零构建可扩展与可维护的游戏架构,让你的游戏项目在模块化设计、脚本对象运用及状态模式处理中焕发新生,实现高效迭代与团队协作的完美平衡之路
【9月更文挑战第1天】游戏开发中的架构设计是项目成功的关键。良好的架构能提升开发效率并确保项目的长期可维护性和可扩展性。在使用Unity引擎时,合理的架构尤为重要。本文探讨了如何在Unity中实现可扩展且易维护的游戏架构,包括模块化设计、使用脚本对象管理数据、应用设计模式(如状态模式)及采用MVC/MVVM架构模式。通过这些方法,可以显著提高开发效率和游戏质量。例如,模块化设计将游戏拆分为独立模块。
217 3
|
4月前
|
开发者 图形学 Java
揭秘Unity物理引擎核心技术:从刚体动力学到关节连接,全方位教你如何在虚拟世界中重现真实物理现象——含实战代码示例与详细解析
【8月更文挑战第31天】Unity物理引擎对于游戏开发至关重要,它能够模拟真实的物理效果,如刚体运动、碰撞检测及关节连接等。通过Rigidbody和Collider组件,开发者可以轻松实现物体间的互动与碰撞。本文通过具体代码示例介绍了如何使用Unity物理引擎实现物体运动、施加力、使用关节连接以及模拟弹簧效果等功能,帮助开发者提升游戏的真实感与沉浸感。
113 1
|
3月前
|
图形学 C++ C#
Unity插件开发全攻略:从零起步教你用C++扩展游戏功能,解锁Unity新玩法的详细步骤与实战技巧大公开
【8月更文挑战第31天】Unity 是一款功能强大的游戏开发引擎,支持多平台发布并拥有丰富的插件生态系统。本文介绍 Unity 插件开发基础,帮助读者从零开始编写自定义插件以扩展其功能。插件通常用 C++ 编写,通过 Mono C# 运行时调用,需在不同平台上编译。文中详细讲解了开发环境搭建、简单插件编写及在 Unity 中调用的方法,包括创建 C# 封装脚本和处理跨平台问题,助力开发者提升游戏开发效率。
296 0
|
4月前
|
开发者 图形学 API
从零起步,深度揭秘:运用Unity引擎及网络编程技术,一步步搭建属于你的实时多人在线对战游戏平台——详尽指南与实战代码解析,带你轻松掌握网络化游戏开发的核心要领与最佳实践路径
【8月更文挑战第31天】构建实时多人对战平台是技术与创意的结合。本文使用成熟的Unity游戏开发引擎,从零开始指导读者搭建简单的实时对战平台。内容涵盖网络架构设计、Unity网络API应用及客户端与服务器通信。首先,创建新项目并选择适合多人游戏的模板,使用推荐的网络传输层。接着,定义基本玩法,如2D多人射击游戏,创建角色预制件并添加Rigidbody2D组件。然后,引入网络身份组件以同步对象状态。通过示例代码展示玩家控制逻辑,包括移动和发射子弹功能。最后,设置服务器端逻辑,处理客户端连接和断开。本文帮助读者掌握构建Unity多人对战平台的核心知识,为进一步开发打下基础。
154 0
|
4月前
|
开发者 图形学 C#
揭秘游戏沉浸感的秘密武器:深度解析Unity中的音频设计技巧,从背景音乐到动态音效,全面提升你的游戏氛围艺术——附实战代码示例与应用场景指导
【8月更文挑战第31天】音频设计在游戏开发中至关重要,不仅能增强沉浸感,还能传递信息,构建氛围。Unity作为跨平台游戏引擎,提供了丰富的音频处理功能,助力开发者轻松实现复杂音效。本文将探讨如何利用Unity的音频设计提升游戏氛围,并通过具体示例代码展示实现过程。例如,在恐怖游戏中,阴森的背景音乐和突然的脚步声能增加紧张感;在休闲游戏中,轻快的旋律则让玩家感到愉悦。
124 0
|
4月前
|
图形学 C# 开发者
全面掌握Unity游戏开发核心技术:C#脚本编程从入门到精通——详解生命周期方法、事件处理与面向对象设计,助你打造高效稳定的互动娱乐体验
【8月更文挑战第31天】Unity 是一款强大的游戏开发平台,支持多种编程语言,其中 C# 最为常用。本文介绍 C# 在 Unity 中的应用,涵盖脚本生命周期、常用函数、事件处理及面向对象编程等核心概念。通过具体示例,展示如何编写有效的 C# 脚本,包括 Start、Update 和 LateUpdate 等生命周期方法,以及碰撞检测和类继承等高级技巧,帮助开发者掌握 Unity 脚本编程基础,提升游戏开发效率。
101 0
|
6月前
|
存储 JSON 关系型数据库
【unity实战】制作unity数据保存和加载系统——大型游戏存储的最优解
【unity实战】制作unity数据保存和加载系统——大型游戏存储的最优解
189 2
|
6月前
|
人工智能 定位技术 图形学
【unity实战】制作敌人的AI,使用有限状态机、继承和抽象类多态 定义不同状态的敌人行为
【unity实战】制作敌人的AI,使用有限状态机、继承和抽象类多态 定义不同状态的敌人行为
172 1
|
4月前
|
图形学 C#
超实用!深度解析Unity引擎,手把手教你从零开始构建精美的2D平面冒险游戏,涵盖资源导入、角色控制与动画、碰撞检测等核心技巧,打造沉浸式游戏体验完全指南
【8月更文挑战第31天】本文是 Unity 2D 游戏开发的全面指南,手把手教你从零开始构建精美的平面冒险游戏。首先,通过 Unity Hub 创建 2D 项目并导入游戏资源。接着,编写 `PlayerController` 脚本来实现角色移动,并添加动画以增强视觉效果。最后,通过 Collider 2D 组件实现碰撞检测等游戏机制。每一步均展示 Unity 在 2D 游戏开发中的强大功能。
221 6
|
4月前
|
图形学 缓存 算法
掌握这五大绝招,让您的Unity游戏瞬间加载完毕,从此告别漫长等待,大幅提升玩家首次体验的满意度与留存率!
【8月更文挑战第31天】游戏的加载时间是影响玩家初次体验的关键因素,特别是在移动设备上。本文介绍了几种常见的Unity游戏加载优化方法,包括资源的预加载与异步加载、使用AssetBundles管理动态资源、纹理和模型优化、合理利用缓存系统以及脚本优化。通过具体示例代码展示了如何实现异步加载场景,并提出了针对不同资源的优化策略。综合运用这些技术可以显著缩短加载时间,提升玩家满意度。
295 5