Files
modular-vr/Assets/Engine/Runtime/Utilities/Placement.cs

152 lines
6.3 KiB
C#

using System;
using System.Collections.Generic;
using UnityEngine;
namespace EscapeRoomEngine.Engine.Runtime.Utilities
{
/// <summary>
/// A placement defines the position, size and orientation of a module or space.
/// </summary>
public struct Placement
{
/// <summary>
/// The placement position is always considered to be at <c>(0.5, 0.5)</c> relative to the placement.
///
/// <example>In a placement of size <c>(1, 1)</c>, the position is at the center of the placement. In a larger placement, the position is in the center of the bottom left square.</example>
/// </summary>
public Vector3Int position;
public Vector2Int size;
public Orientation orientation;
/// <summary>
/// Return the bottom left corner position of the module.
/// </summary>
public Vector3Int BottomLeft {
get
{
var sizeMinusOne = size - Vector2Int.one;
return orientation switch
{
Orientation.North => position,
Orientation.East => position - new Vector3Int(0, 0, sizeMinusOne.x),
Orientation.South => position - new Vector3Int(sizeMinusOne.x, 0, sizeMinusOne.y),
Orientation.West => position - new Vector3Int(sizeMinusOne.y, 0, 0),
_ => throw new ArgumentOutOfRangeException()
};
}
}
/// <summary>
/// Get the geometric center of this placement taking into account its orientation.
/// </summary>
public Vector3 GeometricCenter
{
get
{
var sizeMinusOne = size - Vector2Int.one;
var half = new Vector2(sizeMinusOne.x/2f, sizeMinusOne.y/2f);
var horizontal = new Vector3(half.x, 0, half.y);
var vertical = new Vector3(half.y, 0, -half.x);
var geometricPosition = position + new Vector3(0.5f, 0, 0.5f);
#pragma warning disable CS8524
return orientation switch
#pragma warning restore CS8524
{
Orientation.North => geometricPosition + horizontal,
Orientation.East => geometricPosition + vertical,
Orientation.South => geometricPosition - horizontal,
Orientation.West => geometricPosition - vertical
};
}
}
/// <summary>
/// Get the integer positions of the two back corners. The first one is just the position of the placement.
/// </summary>
public (Vector3Int, Vector3Int) BackCorners {
get
{
var width = size.x - 1;
var horizontal = new Vector3Int(width, 0, 0);
var vertical = new Vector3Int(0, 0, width);
#pragma warning disable CS8524
var otherCorner = orientation switch
#pragma warning restore CS8524
{
Orientation.North => position + horizontal,
Orientation.East => position - vertical,
Orientation.South => position - horizontal,
Orientation.West => position + vertical
};
return (position, otherCorner);
}
}
/// <summary>
/// Create a set with every possible combination of position and orientation of placements that fit in a given space.
/// The size of the placements are given by this placement.
/// </summary>
/// <param name="space">The generated placements will fit inside the placement of the given space.</param>
public List<Placement> EveryPlacementInSpace(Space space)
{
var placements = new List<Placement>();
var sizeMinusOne = size - Vector2Int.one;
var spaceSize = space.rrPlacement.size;
placements.AddRange(ConstrainedPlacements(Orientation.North, size,
new Range(0, spaceSize.x - sizeMinusOne.x), new Range(0, spaceSize.y - sizeMinusOne.y)));
placements.AddRange(ConstrainedPlacements(Orientation.East, size,
new Range(0, spaceSize.x - sizeMinusOne.y), new Range(sizeMinusOne.x, spaceSize.y)));
placements.AddRange(ConstrainedPlacements(Orientation.South, size,
new Range(sizeMinusOne.x, spaceSize.x), new Range(sizeMinusOne.y, spaceSize.y)));
placements.AddRange(ConstrainedPlacements(Orientation.West, size,
new Range(sizeMinusOne.y, spaceSize.x), new Range(0, spaceSize.y - sizeMinusOne.x)));
return placements;
}
private static IEnumerable<Placement> ConstrainedPlacements(
Orientation orientation,
Vector2Int size,
Range xRange, Range zRange)
{
var placements = new List<Placement>(xRange.Length * zRange.Length);
for (var z = zRange.min; z < zRange.max; z++)
{
for (var x = xRange.min; x < xRange.max; x++)
{
placements.Add(new Placement
{
position = new Vector3Int(x, 0, z),
size = size,
orientation = orientation
});
}
}
return placements;
}
/// <summary>
/// Performs the specified action for each position inside the size of this placement.
///
/// <example>For the placement {(2, 0, 3), (1, 2), East} the action would be called twice with the parameters (2, 0, 3) and (3, 0, 3).</example>
/// </summary>
public void ForEachPosition(Action<Vector3Int> action)
{
var bottomLeft = BottomLeft;
for (var z = 0; z < (orientation is Orientation.North or Orientation.South ? size.y : size.x); z++)
{
for (var x = 0; x < (orientation is Orientation.North or Orientation.South ? size.x : size.y); x++)
{
action(bottomLeft + new Vector3Int(x, 0, z));
}
}
}
public string ToStringShort() => $"{{{position}, {orientation}}}";
public override string ToString() => $"{{{position}, {size}, {orientation}}}";
}
}