Unity中基于网格/平铺移动+碰撞?

Posted

技术标签:

【中文标题】Unity中基于网格/平铺移动+碰撞?【英文标题】:Grid-based/Tile Movement + Collisions in Unity? 【发布时间】:2018-10-09 06:40:06 【问题描述】:

在弄清楚如何进行基于网格的移动或基于图块的移动时遇到了很大的问题。有点类似于 Nitrome 的 Redungeon:不能真正发布图像:所以这是一个 gif。 http://www.eneminds.com/redungeon/img/gifs/gif_1.gif

我想我应该创建一个网格系统。我认为这很容易,就像旧游戏(口袋妖怪等)一样。虽然不知道该怎么做。但我也希望运动快。

【问题讨论】:

你尝试过做什么?您面临的具体问题是什么?您的研究向您展示了什么? 【参考方案1】:

Unity 有自己的 tilemaps 实现,它应该允许您构建背景,也就是您行走的地面: https://docs.unity3d.com/Manual/Tilemap.html

此外,Unity 提供了更多 2d 内容来帮助您在这里构建东西: https://github.com/Unity-Technologies/2d-extras

在此处找到的众多教程中概述了碰撞: https://unity3d.com/learn/tutorials/s/2d-game-creation

要制作一款像 GIF 动画那样的游戏,不容易。如果您是游戏开发新手,情况尤其如此,所以不要指望简单的解决方案。

您可能想查看 Unity 资源商店以快速获取基于图块的移动方式,但这总是要付出代价,并且不会帮助您学习。 https://assetstore.unity.com/

关于移动速度,这始终只是游戏中的一个变量,可以根据您的喜好进行调整。在基于图块的游戏的情况下,它将对应于角色在视觉上从一个图块转移到另一个图块所花费的时间。尝试阅读作为 Unity 引擎一部分的 Lerping。 https://docs.unity3d.com/ScriptReference/Vector3.Lerp.html

【讨论】:

谢谢。另外...我的意思是移动速度是...因为我已经尝试过使用 (pos += Vector3.DIRECTION) 进行 MoveTowards。但仅在 (pos == transform.position) 时才存储在 if 语句中)。 where pos = transform.position..这意味着你只有在你的目标位置时才能移动..它确实使移动不平滑并且没有碰撞。【参考方案2】:

我能够成功。 我刚刚修改了 Unity 学习部分的 RogueLike 教程中的输入。

所以这是负责移动的代码

    void Move(string direction_string)
    
        ChangeDirection(direction_string);

        Vector2 start = transform.position;
        Vector2 end = start + direction;

        boxCollider.enabled = false;
        hit = Physics2D.Linecast(start, end, blockingLayer);
        boxCollider.enabled = true;

        if (hit.transform == null) 
            StartCoroutine(Movement(end));
            animator.SetTrigger(direction_string);
            return;
         else 
            moveSequence.RemoveAt(0);
            animator.SetTrigger(direction_string);
            return;
        
    

    void ChangeDirection(string direction_string)
    
        switch (direction_string) 
            case "up":
                direction = dir_Up;
                break;
            case "down":
                direction = dir_Down;
                break;
            case "left":
                direction = dir_Left;
                break;
            case "right":
                direction = dir_Right;
                break;
        
    

    IEnumerator Movement(Vector3 end)
    
        moving = true;
        float sqrRemainingDistance = (transform.position - end).sqrMagnitude;

        while (sqrRemainingDistance > float.Epsilon) 
            Vector3 newPosition = Vector3.MoveTowards(rb2D.position, end, moveSpeed * Time.deltaTime);
            rb2D.MovePosition(newPosition);
            sqrRemainingDistance = (transform.position - end).sqrMagnitude;
            yield return null;
        

        currentPos = end;
        moveSequence.RemoveAt(0);
        moving = false;
    

这是负责输入的内容


    void FixedUpdate ()
    
        if (moveSequence.Count > 0 && !moving) 
            Move(moveSequence[0]);
        
    

然后只需将它连接到一个 Update() 函数,该函数会监听按钮按下并将一个列表项添加到 moveSequence List 中

moveSequence.Add("up");

【讨论】:

【参考方案3】:

刚刚在 reddit 上为某人做了一个这样的代码,来这里看看它之前是否被问过。我知道我迟到了 2 年,但如果有人需要,它就在这里。

    using System.Collections;
    using System.Collections.Generic;
    using UnityEngine;
     
    public class GameController : MonoBehaviour
    
     
        // a script created so that a player always moves one direction at a time on a grid.
        // It uses a plane with even width and height to make a grid
        //     - if needed, turn off the mesh renderer so that you dont see the tiled grid.
     
     
     
     
        // the game object we use to create a grid. Should be a plane with an even width and height.
        public  GameObject tile_object;
     
        // grid x and y sizes, set them to odd numbers
        public float x_grid_size, y_grid_size;
     
        // List of all our floor tiles
        Dictionary<(float x, float y), GameObject> floor_list = new Dictionary<(float x, float y), GameObject>();
     
        // Reference to the player prefab
        public GameObject player;
     
        // reference to player character
        private GameObject player_character;
     
        // a tool used for debugging
        bool debugging = true;
     
        //check if player is moving
        bool player_is_moving = false;
     
        (float x, float y) players_current_tile;
     
        // players speed
        public float player_speed = 35f;
     
        // used in calculating movements
        (float x, float y) to_tile;
     
        //used to calculate movement
        Vector3 to_pos;
     
     
        // Start is called before the first frame update
        void Start()
        
            // put the player into the scene
           player_character = Instantiate(player, Vector3.zero, player.transform.rotation);
     
            // Sets the Middle Tile and checks grid is odd
            SetMiddleTile();
     
            // create a floor grid to help us visualize our play field.
            CreateGrid();
     
        
     
        // Update is called once per frame
        void Update()
        
            PlayerMovement();
            MovePlayer();
        
     
        // gets inputs on A S W D keys for movement
        void PlayerMovement() 
            if (Input.GetKeyDown(KeyCode.W))
                PlayerInput((0, 1));
            
            if (Input.GetKeyDown(KeyCode.A))
            
                PlayerInput((-1, 0));
            
            if (Input.GetKeyDown(KeyCode.S))
            
                PlayerInput((0, -1));
            
            if (Input.GetKeyDown(KeyCode.D))
            
                PlayerInput((1, 0));
            
        
     
     
        // simplifies player entries, and gets input directional data
        void PlayerInput((float x, float y) input)
        
            if (!player_is_moving)
            
                (float x, float y) to_tile_check = (players_current_tile.x + input.x, players_current_tile.y + input.y);
                if (floor_list[to_tile_check] != null)
                
                    player_is_moving = true;
     
                    to_tile = (players_current_tile.x + input.x, players_current_tile.y + input.y);
                    to_pos = floor_list[to_tile].transform.position;
                
            
     
        
     
       // used to move the player
        void MovePlayer()
        
     
            if (player_is_moving)
            
                player_character.transform.position = Vector3.MoveTowards(player_character.transform.position, to_pos, Time.deltaTime * player_speed);
     
                if (player_character.transform.position == to_pos)
                
                    players_current_tile = to_tile;
                    player_is_moving = false;
                
            
        
     
        // used to create the floor grid
        void CreateGrid()
        
            // Create an empty to hold our floor grid
            GameObject grid_empty = new GameObject();
            grid_empty.name = "grid_floor";
     
            // get the size of our floor tile so we know how far to space them.
            float tile_size = tile_object.GetComponent<MeshRenderer>().bounds.size.x;
     
            // create an offset so the grid is always centered in game view
            Vector3 offSet = new Vector3((x_grid_size * tile_size) / 2 - (tile_size / 2), 0, (y_grid_size * tile_size)/2 - (tile_size / 2));
     
            // iterate to create tiles on x axis
            for (int x = 0; x < x_grid_size; x++)
            
                // iterate to create tiles on y axis
                for (int y = 0; y < y_grid_size; y++)
                
                    // instantiate new tile
                    GameObject floor_tile = Instantiate(tile_object, (new Vector3(x * tile_size, 0, y * tile_size) - offSet), tile_object.transform.rotation);
                    floor_tile.name = "flr_x: " + (x  + 1) + " y:" + (y + 1);
     
                    // set the parent to grid empty so the scene isnt filled with objects and all floor tiles are neatly centerd
                    floor_tile.transform.parent = grid_empty.transform;
     
                    // add the tile to our dictionary.
                    floor_list.Add(((x + 1),(y + 1)), floor_tile);
                
            
            if (debugging)  print("Created Floors: " + (x_grid_size * y_grid_size) + " Floor List Size: " + floor_list.Count);
        
     
        // sets the middle tile the player is spawned on, also checks to make sure grid size is set correctly
       void SetMiddleTile()
        
            // these check to make sure grid size isnt 0 and are odd
            if (x_grid_size == 0 && y_grid_size == 0)  print("Forgot to set grid size! setting default to 7x and 7y"); x_grid_size = 7; y_grid_size = 7; 
            if (x_grid_size % 2 == 0)  print("x_grid_size is set to an even number(" + x_grid_size + "), changing it to odd(" + (x_grid_size + 1) + ")"); x_grid_size += 1; 
            if (y_grid_size % 2 == 0)  print("y_grid_size is set to an even number(" + y_grid_size + "), changing it to odd(" + (y_grid_size + 1) + ")"); y_grid_size += 1; 
     
            // splits the grid and half
            float x = x_grid_size / 2 + 0.5f;
            float y = y_grid_size / 2 + 0.5f;
     
            // set the players current tile to middle tile
            players_current_tile = (x, y);
     
            // set the to tile to current tile to avoid null instances when first checking our dictionary
            to_tile = players_current_tile;
     
            // used for debugging
            if (debugging)  print("the middle tile is: x(" + x + ")" + " y(" + y + ")"); 
        
    
         

【讨论】:

以上是关于Unity中基于网格/平铺移动+碰撞?的主要内容,如果未能解决你的问题,请参考以下文章

Leaflet.js网格的白线

详解Unity中的网格碰撞器Mesh Collider

在Javascript中,平铺映射碰撞不正常

Unity入门计划基本概念-2D碰撞体Collider 2D

Unity 自定义编辑器,例如“层碰撞矩阵”

unity3d怎么做碰撞?unity3d如何添加碰撞体?求解!