XNA 4.0碰撞检测导致精灵反弹

时间:2012-10-21 11:17:19

标签: c# collision-detection xna-4.0

我的碰撞检测似乎正在起作用,所有这些都是它检测到的一些碰撞是奇怪的,因为它会说碰撞是在平台的底部,而不是。然而,这不是主要问题。

当玩家与平台发生碰撞时,不是移动到平台的表面,而是移动到平台的表面上,并且在我的引力到位的情况下,看起来他不断地弹跳。我无法弄清楚原因。

如果你们能提供帮助,我们将不胜感激。因为我是XNA和C#的新手,所以我可能会非常简单。

我将在下面列出我的Game1,Player和Platoform课程。 (我的碰撞检测在玩家类中)

由于

Game1 Class

public class Game1 : Microsoft.Xna.Framework.Game
{
    GraphicsDeviceManager graphics;
    SpriteBatch spriteBatch;

    Texture2D background;
    Movement character;
    Platform[] platforms;
    //private Vector2 SnapePosition = Vector2.Zero;

    public Game1()
    {
        graphics = new GraphicsDeviceManager(this);
        Content.RootDirectory = "Content";
        graphics.PreferredBackBufferHeight = 440;
        graphics.PreferredBackBufferWidth = 782;

    }

    /// <summary>
    /// Allows the game to perform any initialization it needs to before starting to run.
    /// This is where it can query for any required services and load any non-graphic
    /// related content.  Calling base.Initialize will enumerate through any components
    /// and initialize them as well.
    /// </summary>
    protected override void Initialize()
    {
        // TODO: Add your initialization logic here
        platforms = new Platform[15];


        base.Initialize();
    }

    /// <summary>
    /// LoadContent will be called once per game and is the place to load
    /// all of your content.
    /// </summary>
    protected override void LoadContent()
    {
        // Create a new SpriteBatch, which can be used to draw textures.
        spriteBatch = new SpriteBatch(GraphicsDevice);


        for (int i = 0; i < platforms.Length; i++)
        {
            platforms[i] = new Platform(
                Content.Load<Texture2D>("Platforms/lvl2_platform"), new Rectangle(i*100, 410, 100, 30));
        }
        character = new Movement(Content.Load<Texture2D>("snape"), new Rectangle(0, 360, 50, 50), platforms);

        // TODO: use this.Content to load your game content here
        background = Content.Load<Texture2D>("Backgrounds/lvl2_background");
    }

    /// <summary>
    /// UnloadContent will be called once per game and is the place to unload
    /// all content.
    /// </summary>
    protected override void UnloadContent()
    {
        // TODO: Unload any non ContentManager content here
    }

    /// <summary>
    /// Allows the game to run logic such as updating the world,
    /// checking for collisions, gathering input, and playing audio.
    /// </summary>
    /// <param name="gameTime">Provides a snapshot of timing values.</param>
    protected override void Update(GameTime gameTime)
    {
        // Allows the game to exit
        if (GamePad.GetState(PlayerIndex.One).Buttons.Back == ButtonState.Pressed)
            this.Exit();

        //Allows the player to move
        character.Update();

        // TODO: Add your update logic here


        base.Update(gameTime);
    }

    /// <summary>
    /// This is called when the game should draw itself.
    /// </summary>
    /// <param name="gameTime">Provides a snapshot of timing values.</param>
    protected override void Draw(GameTime gameTime)
    {
        GraphicsDevice.Clear(Color.CornflowerBlue);

        // TODO: Add your drawing code here

        spriteBatch.Begin();
        spriteBatch.Draw(background, Vector2.Zero, Color.White);
        character.Draw(spriteBatch);

        foreach (Platform platform in platforms)
        {
            platform.Draw(spriteBatch);
        }
        spriteBatch.End();

        base.Draw(gameTime);
    }
}

玩家类

class Player
{
    public Texture2D Snape;
    public Rectangle SnapePosition;

    public enum CollisionPosition { None, Top, Bottom, Left, Right };
    public CollisionPosition collisionType;
    public bool inCollision;
    public int collisionDepth;
    public Platform[] plat;
    public int num;

    public virtual void Update()
    {

    }

    public void Draw(SpriteBatch spriteBatch)
    {
        spriteBatch.Draw(Snape, SnapePosition, Color.White);
    }
}

class Movement : Player
{
        public Movement(Texture2D newSnape, Rectangle newSnapePosition, Platform[] a)
    {
        Snape = newSnape;
        SnapePosition = newSnapePosition;
        plat = a; 
    }

        public override void Update()
        {
            // Check for collision
            Collisions();

            // Determine collision type
            //DetermineCollisionType();

            // Separate snape
            //SeparateSnape();

            KeyboardState keyBoard = Keyboard.GetState();

            if (keyBoard.IsKeyDown(Keys.A))
            {
                SnapePosition.X -= 5;
            }
            if (keyBoard.IsKeyDown(Keys.D))
            {
                SnapePosition.X += 5;
            }
            if (keyBoard.IsKeyDown(Keys.W))
            {
                SnapePosition.Y -= 5;
            }
            //if (keyBoard.IsKeyDown(Keys.S))
            //{
                SnapePosition.Y += 5;
           // }



          //  if (SnapePosition.X < 0)
          //      SnapePosition.X = 0;
          //  if (SnapePosition.Y < 0)
          //      SnapePosition.Y = 0;
          //  if (SnapePosition.X > GraphicsDevice.Viewport.Width - Snape.Width)
          //      SnapePosition.X = GraphicsDevice.Viewport.Width - Snape.Width;
          //  if (SnapePosition.Y > GraphicsDevice.Viewport.Height - Snape.Height)
          //      SnapePosition.Y = GraphicsDevice.Viewport.Height - Snape.Height;

        }


        public void Collisions()
        {
            for (int i = 0; i < plat.Length; i++)
            {
                if (plat[i].rectangle.Intersects(SnapePosition))
                {
                    inCollision = true;
                    num = i;
                    DetermineCollisionType();
                }

            }

        }

        public void DetermineCollisionType()
        {
            if (inCollision == false)
            {
                collisionType = CollisionPosition.None;
                collisionDepth = 0;
            }
            else
            {
                // Determine the side of *least intersection* for snape
                int minColDepth = int.MaxValue;

                // Check the top side                    
                int tColDepth = (plat[num].rectangle.Y + plat[num].texture.Height / 2) - (SnapePosition.Y - Snape.Height / 2);

                if (tColDepth > 0 && tColDepth < minColDepth)
                {
                    collisionType = CollisionPosition.Top;
                    minColDepth = tColDepth;
                }

                // Check the bottom side            
                int bColDepth = (SnapePosition.Y + Snape.Height / 2) - (plat[num].rectangle.Y - plat[num].texture.Height / 2);

                if (bColDepth > 0 && bColDepth < minColDepth)
                {
                    collisionType = CollisionPosition.Bottom;
                    minColDepth = bColDepth;
                }

                // Check the right overlap
                int rColDepth = (SnapePosition.X + Snape.Width / 2) - (plat[num].rectangle.X - plat[num].texture.Width / 2);

                if (rColDepth > 0 && rColDepth < minColDepth)
                {
                    collisionType = CollisionPosition.Right;
                    minColDepth = rColDepth;
                }

                // Check the left overlap
                int lColDepth = (plat[num].rectangle.X + plat[num].texture.Width / 2) - (SnapePosition.X - Snape.Width / 2);

                if (lColDepth > 0 && lColDepth < minColDepth)
                {
                    collisionType = CollisionPosition.Left;
                    minColDepth = lColDepth;
                }

                // Update the collision depth
                collisionDepth = minColDepth;
                SeparateSnape();
            }
        }



        public void SeparateSnape()
        {
            switch (collisionType)
            {
                case CollisionPosition.None:
                    break;

                case CollisionPosition.Top:
                    SnapePosition.Y += (collisionDepth);
                    break;

                case CollisionPosition.Bottom:
                    SnapePosition.Y -= collisionDepth;
                    break;

                case CollisionPosition.Right:
                    SnapePosition.X -= collisionDepth;
                    break;

                case CollisionPosition.Left:
                    SnapePosition.X += collisionDepth;
                    break;
            }
        }
}

平台类

class Platform
{
    public Texture2D texture;
    public Rectangle rectangle;

    public Platform(Texture2D newTexture, Rectangle newRectangle)
    {
        texture = newTexture;
        rectangle = newRectangle;
    }

    public void Update()
    {

    }

    public void Draw(SpriteBatch spriteBatch)
    {
        spriteBatch.Draw(texture, rectangle, Color.White);
    }


}

1 个答案:

答案 0 :(得分:0)

我修复了你的碰撞。

您的所有深度计算都不正确。你是根据实体的中心(玩家/平台)并使用他们的纹理大小而不是他们的矩形大小来做的。

您可能一直认为所有内容都来自中间,但默认情况下它是从XNA的左上角绘制的。

以下是位于“DetermineCollisionType()”函数下“Movement”类中的新深度计算。

最高深度需要在平台顶部(Y)和snape(Y +高度)底部工作。

int tColDepth = 
    (SnapePosition.Y + SnapePosition.Height) - (plat[num].rectangle.Y);

底部深度需要在游戏形式的底部(Y +高度)和snape(Y)的顶部。

int bColDepth = 
    (plat[num].rectangle.Y + plat[num].rectangle.Height) - (SnapePosition.Y);

正确的深度需要在平台左侧(X)和右侧snape(X +宽度)上工作。

int rColDepth = 
    (SnapePosition.X + SnapePosition.Width) - (plat[num].rectangle.X);

左侧深度需要在平台右侧(X +宽度)和Snape(X)左侧工作。

int lColDepth = 
    (plat[num].rectangle.X + plat[num].rectangle.Width) - (SnapePosition.X);

从现在开始,这将有效。但是,您可能希望为您的平台使用列表。当您执行关卡编辑器时,列表更容易,因为您不必说“我想要15个平台”。