一、游戏介绍

        使用Unity3D引擎制作一个射箭游戏。玩家在场景中,可以以第一人称方式在地图上游走,到达特定的射击位可以进行射击,命中静止的或者移动的靶子得到分数。每一次游戏有n次机会。

二、实现

        1、游走

        使用标准资源库Standard Assets 中的第一人称控制器FPSController,将内置的FirstPersonCharacter设为主摄像机。该控制器支持游走、跳跃。

        然后,创建一个脚本Controllers / CameraFollow.cs,挂载到 弓 这个预制件上,使之时刻跟随主摄像机。(弓CrossBow 使用资源包 RyuGiKen)。

using UnityEngine;

public class CameraFollow : MonoBehaviour
{
    public Transform cameraTransform; // 要跟随的物体
    // public Vector3 offset = new Vector3(0f, 0.5f, -1f); // 相对位置偏移量
    public Vector3 offset = new Vector3(0f, -0.4f, 0.2f); // 相对位置偏移量,x,y,z
    private Vector3 targetPosition; // 目标位置
    private Quaternion targetRotation;// 目标角度

    public void Start(){
        cameraTransform = Camera.main.transform;
    }
    private void LateUpdate()
    {
        if (cameraTransform != null)
        {
            // 获取相机的旋转角度
            targetRotation = cameraTransform.rotation;

            // 根据相机旋转角度计算弓的目标位置
            targetPosition = cameraTransform.position + (targetRotation * offset);

            // 设置弓的位置和旋转
            transform.position = targetPosition;
            transform.rotation = targetRotation;
        }
    }
}

        创建脚本mainController.cs,挂载到FPSController上,作用是在场景中创建物体 弓 。由于弓自身挂载了CameraFollow,所以一旦在游戏中创建,就会跟随摄像机。

using System.Diagnostics;
using UnityEngine;

public class mainController : MonoBehaviour
{
    public GameObject bow;
    //public TreeGenController treeGenController;
    void Start(){

        bow=Instantiate(Resources.Load("Prefabs/myCrossbow")) as GameObject;
        UnityEngine.Debug.Log("corssBow gen");
        //后面部分介绍
        gameObject.AddComponent<ShootScoreController>();
        gameObject.AddComponent<TreeGenController>();
    }
}
        2、射击

        首先,使用Animator动画控制 弓 的 蓄力拉弓、保持、射击 的动画变化过程。

unity 编辑器下关闭游戏_游戏

        其中,Empty、Fill、Shoot均为资源包 RyuGiKen中弓的动画单元。Empty_Fill是 两个动画的混合。

        对应的,需要用脚本控制参数的变化进而控制动画。将下面的MouseController挂载到预制件 弓 (CrossBow)上。

using System.Diagnostics;
using UnityEngine;
using System;

public class MouseController : MonoBehaviour
{
    public Animator animator;
    private float clickStartTime;
    public ArrowFactory arrowFactory;
    public ShootScoreController shootScoreController;

    void Start(){
        animator = GetComponent<Animator>();
        UnityEngine.Debug.Log(animator);
        gameObject.AddComponent<ArrowFactory>();
        arrowFactory = gameObject.GetComponent<ArrowFactory>();
        shootScoreController = Singleton<ShootScoreController>.Instance;

        GameObject place = GameObject.Find("shootPlace");
        UnityEngine.Debug.Log(place.transform.position.x);
    }
    private void Update()
    {
        // 当鼠标点击时设置触发器为 true
        if (Input.GetMouseButtonDown(0))
        {
            // 设置触发器为 true
            animator.SetBool("prepared",true);
            UnityEngine.Debug.Log(animator.GetBool("prepared"));

            clickStartTime = Time.time;
        }

        if (Input.GetMouseButton(0))
        {
            float elapsedTime = Time.time - clickStartTime;

            // 将经过的时间映射到0到1的范围
            float PowerValue = Mathf.Clamp01(elapsedTime / 1.5f);

            // 设置Animator Controller中的浮点数参数
            animator.SetFloat("Power", PowerValue);
        }

        if (Input.GetMouseButtonUp(0))
        {
            // 将参数A重置为0
            //animator.SetFloat("Power", 0f);
        }

        if (Input.GetMouseButtonDown(1))
        {
            if(animator.GetFloat("Power") == 0f) return;
            if(!InShootPlace()) return;

            // 设置触发器为 true
            animator.SetTrigger("Fire");
            UnityEngine.Debug.Log("shoot");
            animator.SetBool("prepared",false);


            //箭神,启动!!!!
            GameObject arrow = arrowFactory.GetArrow();

            UnityEngine.Debug.Log("启动");
            //初始化箭的基本参数
            // 初始位置
            Vector3 offset = new Vector3(0f, 0f, 1.5f); 
            Transform cameraTransform = Camera.main.transform;
            arrow.transform.position = cameraTransform.position + (transform.rotation * offset);
            arrow.transform.rotation = cameraTransform.rotation;
            // arrow.transform.position = transform.position + (transform.rotation * offset);
            // arrow.transform.rotation = transform.rotation;
            //  力
            float maxShootForce = 1f;
            // 获取Power
            float power = animator.GetFloat("Power");
            float shootForce = power * maxShootForce;
            Vector3 shootDirection = arrow.transform.forward;
            arrow.GetComponent<Rigidbody>().AddForce(shootForce * shootDirection, ForceMode.Impulse);
            arrow.AddComponent<ArrowStop>();
            animator.SetFloat("Power", 0f);
        }

        if (Input.GetKey(KeyCode.O) && Input.GetKey(KeyCode.K))
        {
            // "O" 和 "K" 同时按下的处理逻辑
            UnityEngine.Debug.Log("按下了 O 和 K 键");
            shootScoreController.ResetAll();
        }
            
    }

    public bool InShootPlace(){

        GameObject place = GameObject.Find("shootPlace");

        // 判断 damnObject 是否存在,并获取其位置
        if(place != null){
            Vector3 placePosition = place.transform.position;
            if(gameObject.transform.position.y>=placePosition.y + 3.5 && 
                Mathf.Abs(gameObject.transform.position.x - placePosition.x)<=3 &&
                Mathf.Abs(gameObject.transform.position.z - placePosition.z)<=3)
                {
                    shootScoreController.DelCount(0);
                    return true;
                }
                
        }

        GameObject placeMoving = GameObject.Find("shootPlaceMoving");

        if(placeMoving != null){
            Vector3 placeMovingPosition = placeMoving.transform.position;
            if(gameObject.transform.position.y>=placeMovingPosition.y + 10 && 
                Mathf.Abs(gameObject.transform.position.x - placeMovingPosition.x)<=10 &&
                Mathf.Abs(gameObject.transform.position.z - placeMovingPosition.z)<=10)
                {
                    shootScoreController.DelCount(1);
                    return true;
                }
        }

        return false;
    }

}

上面实现了弓的动画,下面要在弓射击时,实现箭的发射。

        创建脚本ArrowFactory.cs,(弓箭工厂),生产 箭。

using System.IO;
using System.Diagnostics;
using System.Security.AccessControl;
using System.Collections;
using System.Collections.Generic;
using UnityEngine;


public class ArrowFactory : MonoBehaviour
{
    List<GameObject> arrow_used;
    List<GameObject> arrow_free;
    // Start is called before the first frame update
    void Start()
    {  
        arrow_used = new List<GameObject>();
        arrow_free = new List<GameObject>();
    }
    public GameObject GetArrow() {
        GameObject arrow;
        if (arrow_free.Count != 0) {
            arrow = arrow_free[0];
            arrow_free.Remove(arrow);
        }
        else {
            //
            arrow = Instantiate(Resources.Load("Prefabs/Arrow")) as GameObject;
        }
        arrow_used.Add(arrow);
        arrow.SetActive(true);
        UnityEngine.Debug.Log("generate arrow");
        return arrow;
    }

    

}

        箭的发射在上面的MouseController中实现。

//箭神,启动!!!!
            GameObject arrow = arrowFactory.GetArrow();

            UnityEngine.Debug.Log("启动");
            //初始化箭的基本参数
            // 初始位置
            Vector3 offset = new Vector3(0f, 0f, 1.5f); 
            Transform cameraTransform = Camera.main.transform;
            arrow.transform.position = cameraTransform.position + (transform.rotation * offset);
            arrow.transform.rotation = cameraTransform.rotation;
            // arrow.transform.position = transform.position + (transform.rotation * offset);
            // arrow.transform.rotation = transform.rotation;
            //  力
            float maxShootForce = 1f;
            // 获取Power
            float power = animator.GetFloat("Power");
            float shootForce = power * maxShootForce;
            Vector3 shootDirection = arrow.transform.forward;
            arrow.GetComponent<Rigidbody>().AddForce(shootForce * shootDirection, ForceMode.Impulse);
            arrow.AddComponent<ArrowStop>();

        箭在触碰到其他物体时,需停下来。将下面的ArrowStop.cs挂载到 箭 预制件上。

using UnityEngine;
public class ArrowStop : MonoBehaviour
{
    private Rigidbody arrowRigidbody;
    private ShootScoreController shootScoreController;

    private void Start()
    {
        arrowRigidbody = GetComponent<Rigidbody>();
        shootScoreController = Singleton<ShootScoreController>.Instance;
    }

    private void OnCollisionEnter(Collision collision)
    {

        // 停止箭的运动
        arrowRigidbody.velocity = Vector3.zero;
        arrowRigidbody.isKinematic = true;

        
        // 启用或禁用其他需要的组件或脚本
        // ...
        // if (collision.gameObject.CompareTag("targetMid")
        //     || collision.gameObject.CompareTag("targetLeft")
        //     || collision.gameObject.CompareTag("targetRight")){
        //     UnityEngine.Debug.Log("hit");
        // }
        if (collision.gameObject.name == "targetMid"
            || collision.gameObject.name == "targetLeft"
            || collision.gameObject.name == "targetRight"){
            UnityEngine.Debug.Log("hit");

            if(shootScoreController.GetCount(0)>0)
                shootScoreController.AddScore(1);
        }

        if (collision.gameObject.name == "targetMovingUp" || collision.gameObject.name == "targetMovingDown" || collision.gameObject.name == "targetMovingRound") {
            UnityEngine.Debug.Log("hitMove");
            transform.SetParent(collision.transform);
            
            if(shootScoreController.GetCount(1)>0)
                shootScoreController.AddScore(3);

        }
    }
}
        3、分数、次数、射击位

        实现碰撞检测,当弓箭命中靶子时,分数增加。同时,每一次发射消耗一次次数。

        其中,碰撞检测在ArrowStop.cs

using UnityEngine;

public class ShootScoreController : MonoBehaviour
{
    private int score;
    private ShootGUI shootGUI;

    void Start(){
        score = 0;
        shootGUI = Singleton<ShootGUI>.Instance;
    }

    void Update(){
        shootGUI.SetScore(score);

    }
    public void AddScore(int s){
        score += s;
    }
    public int GetCount(int type){
        return shootGUI.GetCount(type);
    }
    public void ResetAll(){
        score = 0;
        shootGUI.ResetAll();
    }
    public void DelCount(int type){
        shootGUI.DelCount(type);
    }
}

        射击位:到达射击位处才能射击。在MouseController.cs实现。原理为弓和射击位之间的坐标对比。

        4、场景--天空盒、树木

        地形、天空 使用 资源 Fantasy Skybox FREE中的地形。树木同样适使用现有资源包。

        实现天空随着时间变化:SkyboxController.cs挂载到主摄像机:

using UnityEngine;
using System.Collections;
public class SkyboxController : MonoBehaviour
{
    public string skyboxMaterialPath_Sunrise = "SkyBox/FS017_Sunrise";
    public string skyboxMaterialPath_Day = "SkyBox/FS000_Day_01";  // Skybox材质的资源路径
    public string skyboxMaterialPath_Night = "SkyBox/FS000_Night_02";
    public string skyboxMaterialPath_Sunset = "SkyBox/FS002_Sunset_Cubemap";
    public string skyboxMaterialPath_Sunset2 = "SkyBox/FS017_Sunset";

    public float changeInterval = 7f;  // 更换材质的时间间隔
    private Material[] skyboxMaterials;
    private int currentMaterialIndex = 0;

    void Start()
    {
        // 使用资源路径加载Skybox材质
        Material skyboxMaterial_Sunrise = Resources.Load<Material>(skyboxMaterialPath_Sunrise);
        Material skyboxMaterial_Day = Resources.Load<Material>(skyboxMaterialPath_Day);
        Material skyboxMaterial_Night = Resources.Load<Material>(skyboxMaterialPath_Night);
        Material skyboxMaterial_Sunset = Resources.Load<Material>(skyboxMaterialPath_Sunset);
        Material skyboxMaterial_Sunset2 = Resources.Load<Material>(skyboxMaterialPath_Sunset2);
        skyboxMaterials = new Material[] { skyboxMaterial_Sunrise, skyboxMaterial_Day, skyboxMaterial_Sunset, skyboxMaterial_Sunset2, skyboxMaterial_Night };

        
        for(int i=0;i<skyboxMaterials.Length;i++){
            if (skyboxMaterials[i] == null)
            {
                Debug.LogError("Failed to load Skybox material at path: " + skyboxMaterials[i]);
                return;
            }
                
        }
        StartCoroutine(ChangeSkyboxMaterial());

    }

    IEnumerator ChangeSkyboxMaterial()
    {
        while (true)
        {
            yield return new WaitForSeconds(changeInterval);

            currentMaterialIndex++;
            if (currentMaterialIndex >= skyboxMaterials.Length)
            {
                currentMaterialIndex = 0;
            }

            RenderSettings.skybox = skyboxMaterials[currentMaterialIndex];
        }
    }
}

        地图中随机生成树木:TreeGenController挂载到FPSController

using UnityEngine;

public class TreeGenController : MonoBehaviour
{
    private int treeNums = 120;
    private Terrain terrain;
    private void Start()
    {
        terrain = Terrain.activeTerrain;
        GenerateTrees();
        UnityEngine.Debug.Log(terrain.transform.position.x);
        UnityEngine.Debug.Log(terrain.terrainData.size.x);
    }

    private void GenerateTrees(){
        for(int i = 0;i<treeNums;i++){
            Vector3 randomPosition = GetRandomPosition();
            bool isPositionValid = IsPositionValid(randomPosition);

            // If the position is not valid, find a new position within the valid area
            while (!isPositionValid)
            {
                randomPosition = GetRandomPosition();
                isPositionValid = IsPositionValid(randomPosition);
            }
            Instantiate(Resources.Load("Prefabs/Tree9_2"), randomPosition, Quaternion.identity);
        }

        for(int i = 0;i<treeNums;i++){
            Vector3 randomPosition = GetRandomPosition();
            bool isPositionValid = IsPositionValid(randomPosition);

            // If the position is not valid, find a new position within the valid area
            while (!isPositionValid)
            {
                randomPosition = GetRandomPosition();
                isPositionValid = IsPositionValid(randomPosition);
            }
            Instantiate(Resources.Load("Prefabs/Tree9_3"), randomPosition, Quaternion.identity);
        }

        for(int i = 0;i<treeNums;i++){
            Vector3 randomPosition = GetRandomPosition();
            bool isPositionValid = IsPositionValid(randomPosition);

            // If the position is not valid, find a new position within the valid area
            while (!isPositionValid)
            {
                randomPosition = GetRandomPosition();
                isPositionValid = IsPositionValid(randomPosition);
            }
            Instantiate(Resources.Load("Prefabs/Tree9_2"), randomPosition, Quaternion.identity);
        }
    }

    private Vector3 GetRandomPosition()
    {
        // Generate a random position within the terrain bounds
        float x = Random.Range(terrain.transform.position.x, terrain.transform.position.x + terrain.terrainData.size.x);
        float z = Random.Range(terrain.transform.position.z, terrain.transform.position.z + terrain.terrainData.size.z);
        float y = terrain.SampleHeight(new Vector3(x, 0, z));

        return new Vector3(x, y, z);
    }

    private bool IsPositionValid(Vector3 position)
    {
        // Check if the position is within the restricted areas or too close to them
        // You can use any method that suits your needs to define and check the restricted areas

        // Example: Check distance from restricted areas

        // Collider[] colliders = Physics.OverlapSphere(position, minDistanceFromRestrictedAreas);
        // foreach (Collider collider in colliders)
        // {
        //     if (collider.CompareTag("RestrictedArea"))
        //     {
        //         return false;
        //     }
        // }



        return true;
    }
}
        4、GUI

        显示分数、次数、提示、准心,提供函数以实时修改分数等参数

using UnityEngine;

public class ShootGUI : MonoBehaviour
{
    private int score;
    private int count_static;
    private int count_Moving;
    void Start(){
        score = 0;
        count_static = 10;
        count_Moving = 10;
    }
    public void SetScore(int score){
        this.score = score;
    }
    public void DelCount(int type){
        if(type == 0){
            if(count_static > 0)
            count_static-=1;
        }
        else if(type == 1){
            if(count_Moving > 0)
            count_Moving-=1;
        }
    }
    public int GetCount(int type){
        if(type == 0){
            return count_static;
        }
        else if(type == 1){
            return count_Moving;
        }
        return 0;
    }
    public void ResetAll(){
        score = 0;
        count_static = 10;
        count_Moving = 10;
    }
    private void OnGUI()
    {
        // 获取屏幕中心的位置
        Vector3 screenCenter = new Vector3(Screen.width / 2f, Screen.height / 2f, 0f);
        
        // 设置GUI样式
        GUIStyle style = new GUIStyle(GUI.skin.label);
        style.fontSize = 20;
        style.fontStyle = FontStyle.Bold;
        style.alignment = TextAnchor.MiddleCenter;
        style.normal.textColor = Color.red;
        
        // 在屏幕中心绘制字母 "O"
        GUI.Label(new Rect(screenCenter.x - style.fontSize/2, screenCenter.y - style.fontSize/2, style.fontSize, style.fontSize), "o", style);
        // 在屏幕左上方绘制分数文本
        GUI.Label(new Rect(10f, 10f, 100f, 30f), "Score: " + score.ToString(), style);
        GUI.Label(new Rect(10f, 30f, 150f, 30f), "Case_1: " + count_static.ToString(), style);
        GUI.Label(new Rect(10f, 50f, 150f, 30f), "Case_2: " + count_Moving.ToString(), style);
        // 在屏幕右上方绘制三行文本
        style.fontStyle = FontStyle.Normal;
        GUI.Label(new Rect(Screen.width - 200f, 10f, 200f, 30f), "左键长按蓄力", style);
        GUI.Label(new Rect(Screen.width - 200f, 40f, 200f, 30f), "左键点击取消", style);
        GUI.Label(new Rect(Screen.width - 200f, 70f, 200f, 30f), "右键点击射击", style);
        GUI.Label(new Rect(Screen.width - 200f, 100f, 200f, 30f), "ok键重置", style);
    }
}