Files
FlaxEngine/Source/Engine/Core/Math/Viewport.cs
2024-02-26 19:00:48 +01:00

336 lines
13 KiB
C#

// Copyright (c) 2012-2024 Wojciech Figat. All rights reserved.
// -----------------------------------------------------------------------------
// Original code from SharpDX project. https://github.com/sharpdx/SharpDX/
// Greetings to Alexandre Mutel. Original code published with the following license:
// -----------------------------------------------------------------------------
// Copyright (c) 2010-2014 SharpDX - Alexandre Mutel
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
using System;
using System.Globalization;
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
namespace FlaxEngine
{
/// <summary>
/// Defines the viewport dimensions using float coordinates for (X,Y,Width,Height).
/// </summary>
[StructLayout(LayoutKind.Sequential, Pack = 4)]
public struct Viewport : IEquatable<Viewport>
{
/// <summary>
/// Position of the pixel coordinate of the upper-left corner of the viewport.
/// </summary>
public float X;
/// <summary>
/// Position of the pixel coordinate of the upper-left corner of the viewport.
/// </summary>
public float Y;
/// <summary>
/// Width dimension of the viewport.
/// </summary>
public float Width;
/// <summary>
/// Height dimension of the viewport.
/// </summary>
public float Height;
/// <summary>
/// Gets or sets the minimum depth of the clip volume.
/// </summary>
public float MinDepth;
/// <summary>
/// Gets or sets the maximum depth of the clip volume.
/// </summary>
public float MaxDepth;
/// <summary>
/// Initializes a new instance of the <see cref="Viewport"/> struct.
/// </summary>
/// <param name="x">The x coordinate of the upper-left corner of the viewport in pixels.</param>
/// <param name="y">The y coordinate of the upper-left corner of the viewport in pixels.</param>
/// <param name="width">The width of the viewport in pixels.</param>
/// <param name="height">The height of the viewport in pixels.</param>
public Viewport(float x, float y, float width, float height)
{
X = x;
Y = y;
Width = width;
Height = height;
MinDepth = 0f;
MaxDepth = 1f;
}
/// <summary>
/// Initializes a new instance of the <see cref="Viewport"/> struct.
/// </summary>
/// <param name="x">The x coordinate of the upper-left corner of the viewport in pixels.</param>
/// <param name="y">The y coordinate of the upper-left corner of the viewport in pixels.</param>
/// <param name="width">The width of the viewport in pixels.</param>
/// <param name="height">The height of the viewport in pixels.</param>
/// <param name="minDepth">The minimum depth of the clip volume.</param>
/// <param name="maxDepth">The maximum depth of the clip volume.</param>
public Viewport(float x, float y, float width, float height, float minDepth, float maxDepth)
{
X = x;
Y = y;
Width = width;
Height = height;
MinDepth = minDepth;
MaxDepth = maxDepth;
}
/// <summary>
/// Initializes a new instance of the <see cref="Viewport"/> struct.
/// </summary>
/// <param name="bounds">A bounding box that defines the location and size of the viewport in a render target.</param>
public Viewport(Rectangle bounds)
{
X = bounds.X;
Y = bounds.Y;
Width = bounds.Width;
Height = bounds.Height;
MinDepth = 0f;
MaxDepth = 1f;
}
/// <summary>
/// Initializes a new instance of the <see cref="Viewport"/> struct.
/// </summary>
/// <param name="location">The location of the upper-left corner of the viewport in pixels.</param>
/// <param name="size">The size of the viewport in pixels.</param>
public Viewport(Float2 location, Float2 size)
{
X = location.X;
Y = location.Y;
Width = size.X;
Height = size.Y;
MinDepth = 0f;
MaxDepth = 1f;
}
/// <summary>
/// Gets the size of the viewport.
/// </summary>
/// <value>The bounds.</value>
public Rectangle Bounds
{
get => new Rectangle(X, Y, Width, Height);
set
{
X = value.X;
Y = value.Y;
Width = value.Width;
Height = value.Height;
}
}
/// <summary>
/// Gets or sets the size of the viewport (width and height).
/// </summary>
public Float2 Size
{
get => new Float2(Width, Height);
set
{
Width = value.X;
Height = value.Y;
}
}
/// <summary>
/// Gets the aspect ratio used by the viewport.
/// </summary>
public float AspectRatio => !Mathf.IsZero(Height) ? Width / Height : 0f;
/// <summary>
/// Determines whether the specified <see cref="Viewport"/> is equal to this instance.
/// </summary>
/// <param name="other">The <see cref="Viewport"/> to compare with this instance.</param>
/// <returns><c>true</c> if the specified <see cref="Viewport"/> is equal to this instance; otherwise, <c>false</c>.</returns>
public bool Equals(ref Viewport other)
{
return Mathf.NearEqual(X, other.X) &&
Mathf.NearEqual(Y, other.Y) &&
Mathf.NearEqual(Width, other.Width) &&
Mathf.NearEqual(Height, other.Height) &&
Mathf.NearEqual(MinDepth, other.MinDepth) &&
Mathf.NearEqual(MaxDepth, other.MaxDepth);
}
/// <summary>
/// Determines whether the specified <see cref="Viewport"/> is equal to this instance.
/// </summary>
/// <param name="other">The <see cref="Viewport"/> to compare with this instance.</param>
/// <returns><c>true</c> if the specified <see cref="Viewport"/> is equal to this instance; otherwise, <c>false</c>.</returns>
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public bool Equals(Viewport other)
{
return Equals(ref other);
}
/// <summary>
/// Determines whether the specified object is equal to this instance.
/// </summary>
/// <param name="obj">The object to compare with this instance.</param>
/// <returns><c>true</c> if the specified object is equal to this instance; otherwise, <c>false</c>.</returns>
public override bool Equals(object obj)
{
return obj is Viewport other && Equals(ref other);
}
/// <summary>
/// Returns a hash code for this instance.
/// </summary>
/// <returns>
/// A hash code for this instance, suitable for use in hashing algorithms and data structures like a hash table.
/// </returns>
public override int GetHashCode()
{
unchecked
{
int hashCode = X.GetHashCode();
hashCode = (hashCode * 397) ^ Y.GetHashCode();
hashCode = (hashCode * 397) ^ Width.GetHashCode();
hashCode = (hashCode * 397) ^ Height.GetHashCode();
hashCode = (hashCode * 397) ^ MinDepth.GetHashCode();
hashCode = (hashCode * 397) ^ MaxDepth.GetHashCode();
return hashCode;
}
}
/// <summary>
/// Implements the operator ==.
/// </summary>
/// <param name="left">The left.</param>
/// <param name="right">The right.</param>
/// <returns>The result of the operator.</returns>
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public static bool operator ==(Viewport left, Viewport right)
{
return left.Equals(ref right);
}
/// <summary>
/// Implements the operator !=.
/// </summary>
/// <param name="left">The left.</param>
/// <param name="right">The right.</param>
/// <returns>The result of the operator.</returns>
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public static bool operator !=(Viewport left, Viewport right)
{
return !left.Equals(ref right);
}
/// <summary>
/// Retrieves a string representation of this object.
/// </summary>
/// <returns>A <see cref="System.String"/> that represents this instance.</returns>
public override string ToString()
{
return string.Format(CultureInfo.CurrentCulture, "{{X:{0} Y:{1} Width:{2} Height:{3} MinDepth:{4} MaxDepth:{5}}}", X, Y, Width, Height, MinDepth, MaxDepth);
}
/// <summary>
/// Projects a 3D vector from object space into screen space.
/// </summary>
/// <param name="source">The vector to project.</param>
/// <param name="projection">The projection matrix.</param>
/// <param name="view">The view matrix.</param>
/// <param name="world">The world matrix.</param>
/// <returns>The projected vector.</returns>
public Vector3 Project(Vector3 source, Matrix projection, Matrix view, Matrix world)
{
Matrix.Multiply(ref world, ref view, out Matrix matrix);
Matrix.Multiply(ref matrix, ref projection, out matrix);
Project(ref source, ref matrix, out Vector3 vector);
return vector;
}
/// <summary>
/// Projects a 3D vector from object space into screen space.
/// </summary>
/// <param name="source">The vector to project.</param>
/// <param name="matrix">A combined WorldViewProjection matrix.</param>
/// <param name="vector">The projected vector.</param>
public void Project(ref Vector3 source, ref Matrix matrix, out Vector3 vector)
{
Vector3.Transform(ref source, ref matrix, out vector);
var w = source.X * matrix.M14 + source.Y * matrix.M24 + source.Z * matrix.M34 + matrix.M44;
if (!Mathf.IsZero(w))
{
vector /= w;
}
vector.X = (vector.X + 1f) * 0.5f * Width + X;
vector.Y = (-vector.Y + 1f) * 0.5f * Height + Y;
vector.Z = vector.Z * (MaxDepth - MinDepth) + MinDepth;
}
/// <summary>
/// Converts a screen space point into a corresponding point in world space.
/// </summary>
/// <param name="source">The vector to project.</param>
/// <param name="projection">The projection matrix.</param>
/// <param name="view">The view matrix.</param>
/// <param name="world">The world matrix.</param>
/// <returns>The unprojected Vector.</returns>
public Vector3 Unproject(Vector3 source, Matrix projection, Matrix view, Matrix world)
{
Matrix.Multiply(ref world, ref view, out Matrix matrix);
Matrix.Multiply(ref matrix, ref projection, out matrix);
Matrix.Invert(ref matrix, out matrix);
Unproject(ref source, ref matrix, out Vector3 vector);
return vector;
}
/// <summary>
/// Converts a screen space point into a corresponding point in world space.
/// </summary>
/// <param name="source">The vector to project.</param>
/// <param name="matrix">An inverted combined WorldViewProjection matrix.</param>
/// <param name="vector">The unprojected vector.</param>
public void Unproject(ref Vector3 source, ref Matrix matrix, out Vector3 vector)
{
vector.X = (source.X - X) / Width * 2f - 1f;
vector.Y = -((source.Y - Y) / Height * 2f - 1f);
vector.Z = (source.Z - MinDepth) / (MaxDepth - MinDepth);
var w = vector.X * matrix.M14 + vector.Y * matrix.M24 + vector.Z * matrix.M34 + matrix.M44;
Vector3.Transform(ref vector, ref matrix, out vector);
if (!Mathf.IsZero(w))
{
vector /= w;
}
}
}
}