Conversión de coordenadas tridimensionales a coordenadas de pantalla

Actualización de la página :
Fecha de creación de la página :

resumen

Convierte las coordenadas en el espacio 3D en las coordenadas en la pantalla y muestra el nombre en la ubicación del modelo. En la muestra, la cámara se gira para que sea más fácil ver que el nombre sigue al modelo.

3次元座標からスクリーンへの座標変換

Entorno operativo

Prerrequisitos

Versiones de XNA compatibles
  • 4.0
Plataformas compatibles
  • Windows (XP SP2 o posterior, Vista, 7)
  • Xbox 360
  • Windows Phone 7
Versión requerida del sombreador de vértices de Windows 2.0
Versión de Pixel Shader requerida para Windows 2.0

Entorno operativo

plataforma
  • Windows 7
  • Xbox 360
  • Emulador de Windows Phone 7

sustancia

Conversión de coordenadas 3D en coordenadas de pantalla

Si desea mostrar el nombre sobre la cabeza del personaje, como en un juego en 3D, es necesario calcular las coordenadas de la pantalla en las que se debe mostrar el personaje desde la posición espacial tridimensional del personaje.

Puede parecer un cálculo difícil, pero en realidad, los datos de vértice de un polígono se obtienen de la misma manera que la fórmula utilizada para convertir los datos de vértice de un polígono de coordenadas 3D a coordenadas de pantalla, por lo que no se requieren nuevos conocimientos.

También hay métodos que son fáciles de calcular, por lo que no tienes que escribir ningún código adicional.

campo

En el ejemplo, se muestran tres modelos, por lo que definimos un modelo y un vector[3], que es la información de tres posiciones. También define un ángulo de rotación automático de la cámara para que el movimiento sea fácil de entender.

/// <summary>
/// モデル
/// </summary>
private Model model = null;

/// <summary>
/// 位置の配列
/// </summary>
private Vector3[] positions = new Vector3[3];

/// <summary>
/// カメラの水平回転角度
/// </summary>
private float theta = 0.0f;

Proceso de actualización

El ángulo de rotación para la rotación automática de la cámara se obtiene a partir del tiempo transcurrido del juego.

// カメラの水平角度を自動更新
this.theta = (float)gameTime.TotalGameTime.TotalSeconds / 2.0f;

Pre-Dibujo

Después de dibujar el sprite, es posible que se deshabilite el búfer de profundidad, por lo que debe establecer "DepthStencilState.Default" para habilitar el juicio de dibujo por valor de profundidad.

Al multiplicar la matriz de rotación del eje Y por la matriz de vista que ya ha creado, puede rotar la posición de la cámara alrededor del punto de interés. Presta atención al orden en el que se multiplica la matriz.

// Zバッファを有効にする
this.GraphicsDevice.DepthStencilState = DepthStencilState.Default;

// ビューマトリックスに回転を合成算出
Matrix rotatedView = Matrix.CreateRotationY(this.theta) * this.view;

Obtener ventanilla

La ventanilla proporciona un método para convertir de coordenadas espaciales 3D a coordenadas de pantalla, así que obténgalo de GraphicsDevice.

// ビューポート取得
Viewport viewport = this.GraphicsDevice.Viewport;

Cálculo de coordenadas de pantalla a partir de coordenadas 3D

Encontrar las coordenadas de la pantalla a partir de las coordenadas del espacio 3D es tan sencillo como utilizar el método Viewport.Project.

Especifique la posición espacial 3D del origen como primer argumento, la matriz de proyección como segundo argumento y la matriz de vista como tercer argumento. A continuación, puede obtener las coordenadas en la pantalla como un valor devuelto.

Después de eso, si extrae los valores X e Y de Vector3, puede usar la posición de visualización del texto. (Z es la posición de profundidad)

// 3次元座標からスクリーンの座標算出
Vector3 v3 = viewport.Project(this.positions[i],
                                this.projection,
                                rotatedView,
                                Matrix.Identity);
Vector2 screenPosition = new Vector2(v3.X, v3.Y);

// テキスト描画
this.spriteBatch.DrawString(this.font, "Model " + (i + 1).ToString(),
    screenPosition, Color.White);

Viewport.Project método

Proyecta un vector 3D desde el espacio de objetos en el espacio de la pantalla.

fuente Vector3 Vector de coordenadas espaciales 3D para proyección en coordenadas de pantalla
proyección Matriz Matriz proyectiva
vista Matriz Ver Matriz
mundo Matriz Especifica la primera transformación que se va a realizar
Valores devueltos Vector3 Obtiene un vector en el espacio de la pantalla

Todos los códigos

using System;
using System.Collections.Generic;
using System.Linq;
using Microsoft.Xna.Framework;
using Microsoft.Xna.Framework.Audio;
using Microsoft.Xna.Framework.Content;
using Microsoft.Xna.Framework.GamerServices;
using Microsoft.Xna.Framework.Graphics;
using Microsoft.Xna.Framework.Input;
using Microsoft.Xna.Framework.Media;
#if WINDOWS_PHONE
using Microsoft.Xna.Framework.Input.Touch;
#endif

namespace TransformToScreen
{
    /// <summary>
    /// ゲームメインクラス
    /// </summary>
    public class GameMain : Microsoft.Xna.Framework.Game
    {
        /// <summary>
        /// グラフィックデバイス管理クラス
        /// </summary>
        private GraphicsDeviceManager graphics = null;

        /// <summary>
        /// スプライトのバッチ化クラス
        /// </summary>
        private SpriteBatch spriteBatch = null;

        /// <summary>
        /// スプライトでテキストを描画するためのフォント
        /// </summary>
        private SpriteFont font = null;

        /// <summary>
        /// モデル
        /// </summary>
        private Model model = null;

        /// <summary>
        /// 位置の配列
        /// </summary>
        private Vector3[] positions = new Vector3[3];

        /// <summary>
        /// カメラの水平回転角度
        /// </summary>
        private float theta = 0.0f;

        /// <summary>
        /// ビューマトリックス
        /// </summary>
        private Matrix view = Matrix.Identity;

        /// <summary>
        /// プロジェクションマトリックス
        /// </summary>
        private Matrix projection = Matrix.Identity;


        /// <summary>
        /// GameMain コンストラクタ
        /// </summary>
        public GameMain()
        {
            // グラフィックデバイス管理クラスの作成
            this.graphics = new GraphicsDeviceManager(this);

            // ゲームコンテンツのルートディレクトリを設定
            this.Content.RootDirectory = "Content";

#if WINDOWS_PHONE
            // Windows Phone のデフォルトのフレームレートは 30 FPS
            this.TargetElapsedTime = TimeSpan.FromTicks(333333);

            // バックバッファサイズの設定
            this.graphics.PreferredBackBufferWidth = 480;
            this.graphics.PreferredBackBufferHeight = 800;

            // フルスクリーン表示
            this.graphics.IsFullScreen = true;
#endif
        }

        /// <summary>
        /// ゲームが始まる前の初期化処理を行うメソッド
        /// グラフィック以外のデータの読み込み、コンポーネントの初期化を行う
        /// </summary>
        protected override void Initialize()
        {
            // 位置をランダムに設定
            Random rand = new Random();
            for (int i = 0; i < this.positions.Length; i++)
            {
                this.positions[i] =
                    new Vector3((float)(rand.NextDouble() - 0.5) * 10.0f,
                                0.0f,
                                (float)(rand.NextDouble() - 0.5) * 10.0f);
            }

            // ビューマトリックス
            this.view = Matrix.CreateLookAt(new Vector3(0.0f, 10.0f, 20.0f),
                                            Vector3.Zero,
                                            Vector3.Up);

            // プロジェクションマトリックス
            this.projection = Matrix.CreatePerspectiveFieldOfView(
                        MathHelper.ToRadians(45.0f),
                        (float)this.GraphicsDevice.Viewport.Width /
                            (float)this.GraphicsDevice.Viewport.Height,
                        1.0f,
                        100.0f
                    );

            // コンポーネントの初期化などを行います
            base.Initialize();
        }

        /// <summary>
        /// ゲームが始まるときに一回だけ呼ばれ
        /// すべてのゲームコンテンツを読み込みます
        /// </summary>
        protected override void LoadContent()
        {
            // テクスチャーを描画するためのスプライトバッチクラスを作成します
            this.spriteBatch = new SpriteBatch(this.GraphicsDevice);

            // フォントをコンテンツパイプラインから読み込む
            this.font = this.Content.Load<SpriteFont>("Font");

            // モデルを作成
            this.model = this.Content.Load<Model>("Model");

            // ライトとプロジェクションはあらかじめ設定しておく
            foreach (ModelMesh mesh in this.model.Meshes)
            {
                foreach (BasicEffect effect in mesh.Effects)
                {
                    // デフォルトのライト適用
                    effect.EnableDefaultLighting();

                    // プロジェクションマトリックスをあらかじめ設定
                    effect.Projection = this.projection;
                }
            }
        }

        /// <summary>
        /// ゲームが終了するときに一回だけ呼ばれ
        /// すべてのゲームコンテンツをアンロードします
        /// </summary>
        protected override void UnloadContent()
        {
            // TODO: ContentManager で管理されていないコンテンツを
            //       ここでアンロードしてください
        }

        /// <summary>
        /// 描画以外のデータ更新等の処理を行うメソッド
        /// 主に入力処理、衝突判定などの物理計算、オーディオの再生など
        /// </summary>
        /// <param name="gameTime">このメソッドが呼ばれたときのゲーム時間</param>
        protected override void Update(GameTime gameTime)
        {
            // Xbox 360 コントローラ、Windows Phone の BACK ボタンを押したときに
            // ゲームを終了させます
            if (GamePad.GetState(PlayerIndex.One).Buttons.Back == ButtonState.Pressed)
            {
                this.Exit();
            }

            // カメラの水平角度を自動更新
            this.theta = (float)gameTime.TotalGameTime.TotalSeconds / 2.0f;

            // 登録された GameComponent を更新する
            base.Update(gameTime);
        }

        /// <summary>
        /// 描画処理を行うメソッド
        /// </summary>
        /// <param name="gameTime">このメソッドが呼ばれたときのゲーム時間</param>
        protected override void Draw(GameTime gameTime)
        {
            // 画面を指定した色でクリアします
            this.GraphicsDevice.Clear(Color.CornflowerBlue);

            // Zバッファを有効にする
            this.GraphicsDevice.DepthStencilState = DepthStencilState.Default;

            // ビューマトリックスに回転を合成算出
            Matrix rotatedView = Matrix.CreateRotationY(this.theta) * this.view;

            for (int i = 0; i < this.positions.Length; i++)
            {
                foreach (ModelMesh mesh in this.model.Meshes)
                {
                    foreach (BasicEffect effect in mesh.Effects)
                    {
                        // ビューマトリックス
                        effect.View = rotatedView;

                        // モデルの位置設定
                        effect.World = Matrix.CreateTranslation(this.positions[i]);
                    }

                    // モデルを描画
                    mesh.Draw();
                }
            }

            // ビューポート取得
            Viewport viewport = this.GraphicsDevice.Viewport;

            // スプライトの描画準備
            this.spriteBatch.Begin();

            // 各モデルの位置にテキストを描画
            for (int i = 0; i < this.positions.Length; i++)
            {
                // 3次元座標からスクリーンの座標算出
                Vector3 v3 = viewport.Project(this.positions[i],
                                              this.projection,
                                              rotatedView,
                                              Matrix.Identity);
                Vector2 screenPosition = new Vector2(v3.X, v3.Y);

                // テキスト描画
                this.spriteBatch.DrawString(this.font, "Model " + (i + 1).ToString(),
                    screenPosition, Color.White);
            }

            // スプライトの一括描画
            this.spriteBatch.End();

            // 登録された DrawableGameComponent を描画する
            base.Draw(gameTime);
        }
    }
}