地面和模型闪烁,有些不可见

时间:2012-05-30 17:52:27

标签: c# xna camera game-engine

我真的不知道这里出了什么问题。游戏接缝很有趣。我注意到的第一件事是,当玩家移动时,相机会在化身前移动,因此相机会从头像中移出一点。然后我注意到,当你从侧面看到头像时,你可以看到他的肩膀和他的cheast腔(它在我制作的动画播放器和Maya中都没有这样做)。我注意到的下一件事是当玩家向前移动时,地面开始闪烁。我是否将绘制方法称为错误或什么?

这是我正在使用的相机类

public class Camera : Microsoft.Xna.Framework.GameComponent 
{ 
    //Cam Matrices 
    //viw is composed of position, direction, and up vectors 
    public Matrix view { get; protected set; } 
    public Matrix projection { get; protected set; } 

    //to move the camera 
    //used to recreate view matrix each frame 
    public Vector3 cameraPosition { get; protected set; } 
    public Vector3 cameraDirection; //not the target/ point camera is looking at a relative direction camrea is faceing 
    //to find target add the cameraPosition and cameraDirection togeater 
    public Vector3 cameraUp; 
    public Vector3 target; 

    //camera controls 
    float speed = 3; 


    MouseState prevMouseState; 

    // Pi/180 = 1 degree 
    // Max yaw/pitch variables 
    //this is for moving the head only not the body 
    float totalYaw = MathHelper.PiOver2; 
    public float currentYaw = 0; 
    float totalPitch = MathHelper.PiOver2; 
    public float currentPitch = 0; 


    public float leftTrigYaw; 
    public float rightTrigYaw; 

    //constructor 
    public Camera(Game game, Vector3 pos, Vector3 target, Vector3 up) 
        : base(game) 
    { 
        //build camera view matrix 
        cameraPosition = pos; 
        cameraDirection = target - pos; 
        cameraDirection.Normalize(); //Convert to magintue 1 

        //make it easy to apply differt speeds to movement of camera 
        cameraUp = up; 
        CreateLookAt(); 



        projection = Matrix.CreatePerspectiveFieldOfView 
            ( 
            MathHelper.PiOver4, 
            (float)Game.Window.ClientBounds.Width / (float)Game.Window.ClientBounds.Height, 
            1, 3000); 
    } 

    private void CreateLookAt() 
    { 
        //middle variable is the target! 
        view = Matrix.CreateLookAt(cameraPosition, cameraPosition + cameraDirection, cameraUp); 

    } 


    public override void Initialize() 
    { 

        Mouse.SetPosition(Game.Window.ClientBounds.Width / 2, 
            Game.Window.ClientBounds.Height / 2); 
        prevMouseState = Mouse.GetState(); 
        base.Initialize(); 
    } 

    public override void Update(GameTime gameTime) 
    { 
        GamePadState currentState = GamePad.GetState(PlayerIndex.One); 

        //move forward 
        if (currentState.ThumbSticks.Left.Y > 0 || Keyboard.GetState().IsKeyDown(Keys.W)) 
        { 
            //move just on x and x axis Y is controled other places. 
            cameraPosition += new Vector3(cameraDirection.X, 0, cameraDirection.Z) * speed; 
        } 
        //move backward 
        if (currentState.ThumbSticks.Left.Y < 0 || Keyboard.GetState().IsKeyDown(Keys.S)) 
        { 
            //move just on x and x axis Y is controled other places. 
            cameraPosition -= new Vector3(cameraDirection.X, 0, cameraDirection.Z) * speed; 
        } 
        //move left 
        if (currentState.ThumbSticks.Left.X < 0 || Keyboard.GetState().IsKeyDown(Keys.A)) 
        { 
            //cross product of the up and direction vectors can give the side direction 
            cameraPosition += Vector3.Cross(cameraUp, cameraDirection) * speed; 
        } 
        //move right 
        if (currentState.ThumbSticks.Left.X > 0 || Keyboard.GetState().IsKeyDown(Keys.D)) 
        { 
            cameraPosition -= Vector3.Cross(cameraUp, cameraDirection) * speed; 
        } 


        //mouse movements 

        // Yaw rotation 
        float yawAngle = (-MathHelper.PiOver4 / 150) * 
                (Mouse.GetState().X - prevMouseState.X); 

        float padYaw = (-MathHelper.PiOver2 / 50) * 
             (currentState.ThumbSticks.Right.X); 

        if (Math.Abs(currentYaw + yawAngle) < totalYaw) 
        { 
            cameraDirection = Vector3.Transform(cameraDirection, 
                Matrix.CreateFromAxisAngle(cameraUp, yawAngle)); 
            currentYaw += yawAngle; 
        } 

        //rotate left and right  
        if (currentState.ThumbSticks.Right.X < 0 || currentState.ThumbSticks.Right.X > 0) 
        { 
            if (Math.Abs(currentYaw + padYaw) < totalYaw) 
            { 
                cameraDirection = Vector3.Transform(cameraDirection, 
                Matrix.CreateFromAxisAngle(cameraUp, padYaw)); 
                currentYaw += padYaw; 
            } 

        } 

         leftTrigYaw = (MathHelper.PiOver2 / 50) * 
           (currentState.Triggers.Left); 

        if (currentState.Triggers.Left > 0.0f) 
        { 
            cameraDirection = Vector3.Transform(cameraDirection, 
            Matrix.CreateFromAxisAngle(cameraUp, leftTrigYaw)); 
        } 

         rightTrigYaw = (-MathHelper.PiOver2 / 50) * 
           (currentState.Triggers.Right); 

        if (currentState.Triggers.Right > 0.0f) 
        { 
            cameraDirection = Vector3.Transform(cameraDirection, 
            Matrix.CreateFromAxisAngle(cameraUp, rightTrigYaw)); 
        } 


        // Pitch rotation 
        float pitchAngle = (MathHelper.PiOver4 / 150) * 
            (Mouse.GetState().Y - prevMouseState.Y); 

        float padPitch = (-MathHelper.PiOver2 / 50) * 
             (currentState.ThumbSticks.Right.Y); 


        if (Math.Abs(currentPitch + pitchAngle) < totalPitch) 
        { 
            cameraDirection = Vector3.Transform(cameraDirection, 
                Matrix.CreateFromAxisAngle( 
                    Vector3.Cross(cameraUp, cameraDirection), 
                pitchAngle)); 

            currentPitch += pitchAngle; 
        } 

        if (currentState.ThumbSticks.Right.Y < 0 || currentState.ThumbSticks.Right.Y > 0) 
        { 
            if (Math.Abs(currentPitch + padPitch) < totalPitch) 
            { 
                cameraDirection = Vector3.Transform(cameraDirection, 
                Matrix.CreateFromAxisAngle( 
                    Vector3.Cross(cameraUp, cameraDirection), 
                padPitch)); 

                currentPitch += padPitch; 
            } 

        } 


        //reset mouse state 
        target = cameraPosition + cameraDirection; 

        prevMouseState = Mouse.GetState(); 

        //for testing only!! get rid of this when done 
        if (currentState.Buttons.A == ButtonState.Pressed) 
        { 
            //fly upwar 
            cameraPosition += new Vector3(0, 15, 0); 
        } 

        if (currentState.Buttons.B == ButtonState.Pressed) 
        { 
            //fly upwar 
            cameraPosition -= new Vector3(0, 15, 0); 
        } 



        //call camera creat look at method to build new view matrix 
        CreateLookAt(); 

        if (Keyboard.GetState().IsKeyDown(Keys.Escape)) 
        { 
            Game.Exit(); 

        } 
        base.Update(gameTime); 
    } 
} 
 } 

我也在使用自定义模型处理程序绘制地面和头像

public class CModel 
{ 
    public Vector3 Position { get; set; } 
    public Vector3 Rotation { get; set; } 
    public Vector3 Scale { get; set; } 

    public Model Model { get; private set; } 
    private Matrix[] modelTransforms; 

    private GraphicsDevice graphicsDevice; 

    public CModel(Model Model, Vector3 Position, Vector3 Rotation, 
        Vector3 Scale, GraphicsDevice graphicsDevice) 
    { 
        this.Model = Model; 

        modelTransforms = new Matrix[Model.Bones.Count]; 
        Model.CopyAbsoluteBoneTransformsTo(modelTransforms); 

        this.Position = Position; 
        this.Rotation = Rotation; 
        this.Scale = Scale; 

        this.graphicsDevice = graphicsDevice; 
    } 

    public void Draw(Matrix View, Matrix Projection) 
    { 
        // Calculate the base transformation by combining 
        // translation, rotation, and scaling 
        Matrix baseWorld = Matrix.CreateScale(Scale) 
            * Matrix.CreateFromYawPitchRoll( 
                Rotation.Y, Rotation.X, Rotation.Z) 
            * Matrix.CreateTranslation(Position); 

        foreach (ModelMesh mesh in Model.Meshes) 
        { 
            Matrix localWorld = modelTransforms[mesh.ParentBone.Index] 
                * baseWorld; 

            foreach (ModelMeshPart meshPart in mesh.MeshParts) 
            { 
                BasicEffect effect = (BasicEffect)meshPart.Effect; 

                effect.World = localWorld; 
                effect.View = View; 
                effect.Projection = Projection; 

                effect.EnableDefaultLighting(); 
            } 

            mesh.Draw(); 
        } 
    } 
} 

} 

我不知道问题的起源在哪里,但这些是我能想到的唯一可以解决问题的两件事。 感谢您对此的任何帮助。

以下是模型的内容。我确信maya中的正常情况是正确的,因为我使用了微软在教程中使用的简单模型抽屉,但是你可以稍微看一下那些家伙的肩膀和手臂。我无法得到地面闪烁的图片,因为它发生得太快而无法真正捕捉它。但每次我移动相机时,地面都会上下跳动很多。当我停止移动相机环顾四周时停止。我没有调用camera.Update()方法,因为它是一个游戏组件并且与game1类相关联。 See though model

See the inside of his arm though the top!

这就是我绘制模型的方式 它是自定义模型类中的draw方法

public void Draw(Matrix View, Matrix Projection)
    {
        // Calculate the base transformation by combining
        // translation, rotation, and scaling
        Matrix baseWorld = Matrix.CreateScale(Scale)
            * Matrix.CreateFromYawPitchRoll(
                Rotation.Y, Rotation.X, Rotation.Z)
            * Matrix.CreateTranslation(Position);

        foreach (ModelMesh mesh in Model.Meshes)
        {
            Matrix localWorld = modelTransforms[mesh.ParentBone.Index]
                * baseWorld;

            foreach (ModelMeshPart meshPart in mesh.MeshParts)
            {
                BasicEffect effect = (BasicEffect)meshPart.Effect;

                effect.World = localWorld;
                effect.View = View;
                effect.Projection = Projection;

                effect.EnableDefaultLighting();
            }

            mesh.Draw();
        }
    }

1 个答案:

答案 0 :(得分:1)

对于模型上的工件:模型看起来不像是正确渲染的。您的DrawPrimitives调用可能不会为图形卡提供正确的数据(您的顶点/索引缓冲区填充错误或者您的基本类型/计数错误)。尝试在线框模式下渲染它,你应该能够看到变形的三角形,产生这些伪像。

至于理由 - 我将通过代码查看更多信息,看看我是否能想到任何事情。