using System.Collections.Generic;
using UnityEngine.Rendering;
using UnityEngine.Rendering.Universal;
using UnityEngine.Scripting.APIUpdating;
namespace UnityEngine.Experimental.Rendering.Universal
{
///
/// The Pixel Perfect Camera component ensures your pixel art remains crisp and clear at different resolutions, and stable in motion.
///
[ExecuteInEditMode]
[DisallowMultipleComponent]
[AddComponentMenu("Rendering/2D/Pixel Perfect Camera")]
[RequireComponent(typeof(Camera))]
[MovedFrom("UnityEngine.Experimental.Rendering")]
[HelpURL("https://docs.unity3d.com/Packages/com.unity.render-pipelines.universal@latest/index.html?subfolder=/manual/2d-pixelperfect.html%23properties")]
public class PixelPerfectCamera : MonoBehaviour, IPixelPerfectCamera, ISerializationCallbackReceiver
{
///
/// An enumeration of the types of display cropping.
///
public enum CropFrame
{
///
/// No cropping.
///
None,
///
/// Black borders added to the left and right of viewport to match Reference Resolution.
///
Pillarbox,
///
/// Black borders added to the top and bottom of viewport to match Reference Resolution.
///
Letterbox,
///
/// Black borders added to all sides of viewport to match Reference Resolution.
///
Windowbox,
///
/// Expands the viewport to fit the screen resolution while maintaining the viewport's aspect ratio.
///
StretchFill
}
///
/// Determines how pixels are snapped to the grid.
///
public enum GridSnapping
{
///
/// No snapping.
///
None,
///
/// Prevent subpixel movement and make Sprites appear to move in pixel-by-pixel increments.
///
PixelSnapping,
///
/// The scene is rendered to a temporary texture set as close as possible to the Reference Resolution, while maintaining the full screen aspect ratio. This temporary texture is then upscaled to fit the full screen.
///
UpscaleRenderTexture
}
///
/// Defines the filter mode use to render the final render target.
///
public enum PixelPerfectFilterMode
{
///
/// Uses point filter to upscale to closest multiple of Reference Resolution, followed by bilinear filter to the target resolution.
///
RetroAA,
///
/// Uses point filter to upscale to target resolution.
///
Point,
}
private enum ComponentVersions
{
Version_Unserialized = 0,
Version_1 = 1
}
#if UNITY_EDITOR
const ComponentVersions k_CurrentComponentVersion = ComponentVersions.Version_1;
[SerializeField] ComponentVersions m_ComponentVersion = ComponentVersions.Version_Unserialized;
#endif
///
/// Defines how the output display will be cropped.
///
public CropFrame cropFrame { get { return m_CropFrame; } set { m_CropFrame = value; } }
///
/// Defines if pixels will be locked to a grid determined by assetsPPU.
///
public GridSnapping gridSnapping { get { return m_GridSnapping; } set { m_GridSnapping = value; } }
///
/// The target orthographic size of the camera.
///
public float orthographicSize { get { return m_Internal.orthoSize; } }
///
/// Match this value to to the Pixels Per Unit values of all Sprites within the Scene.
///
public int assetsPPU { get { return m_AssetsPPU; } set { m_AssetsPPU = value > 0 ? value : 1; } }
///
/// The original horizontal resolution your Assets are designed for.
///
public int refResolutionX { get { return m_RefResolutionX; } set { m_RefResolutionX = value > 0 ? value : 1; } }
///
/// Original vertical resolution your Assets are designed for.
///
public int refResolutionY { get { return m_RefResolutionY; } set { m_RefResolutionY = value > 0 ? value : 1; } }
///
/// Set to true to have the Scene rendered to a temporary texture set as close as possible to the Reference Resolution,
/// while maintaining the full screen aspect ratio. This temporary texture is then upscaled to fit the full screen.
///
[System.Obsolete("Use gridSnapping instead", false)]
public bool upscaleRT
{
get
{
return m_GridSnapping == GridSnapping.UpscaleRenderTexture;
}
set
{
m_GridSnapping = value ? GridSnapping.UpscaleRenderTexture : GridSnapping.None;
}
}
///
/// Set to true to prevent subpixel movement and make Sprites appear to move in pixel-by-pixel increments.
/// Only applicable when upscaleRT is false.
///
[System.Obsolete("Use gridSnapping instead", false)]
public bool pixelSnapping
{
get
{
return m_GridSnapping == GridSnapping.PixelSnapping;
}
set
{
m_GridSnapping = value ? GridSnapping.PixelSnapping : GridSnapping.None;
}
}
///
/// Set to true to crop the viewport with black bars to match refResolutionX in the horizontal direction.
///
[System.Obsolete("Use cropFrame instead", false)]
public bool cropFrameX
{
get
{
return m_CropFrame == CropFrame.StretchFill || m_CropFrame == CropFrame.Windowbox || m_CropFrame == CropFrame.Pillarbox;
}
set
{
if (value)
{
if (m_CropFrame == CropFrame.None)
m_CropFrame = CropFrame.Pillarbox;
else if (m_CropFrame == CropFrame.Letterbox)
m_CropFrame = CropFrame.Windowbox;
}
else
{
if (m_CropFrame == CropFrame.Pillarbox)
m_CropFrame = CropFrame.None;
else if (m_CropFrame == CropFrame.Windowbox || m_CropFrame == CropFrame.StretchFill)
m_CropFrame = CropFrame.Letterbox;
}
}
}
///
/// Set to true to crop the viewport with black bars to match refResolutionY in the vertical direction.
///
[System.Obsolete("Use cropFrame instead", false)]
public bool cropFrameY
{
get
{
return m_CropFrame == CropFrame.StretchFill || m_CropFrame == CropFrame.Windowbox || m_CropFrame == CropFrame.Letterbox;
}
set
{
if (value)
{
if (m_CropFrame == CropFrame.None)
m_CropFrame = CropFrame.Letterbox;
else if (m_CropFrame == CropFrame.Pillarbox)
m_CropFrame = CropFrame.Windowbox;
}
else
{
if (m_CropFrame == CropFrame.Letterbox)
m_CropFrame = CropFrame.None;
else if (m_CropFrame == CropFrame.Windowbox || m_CropFrame == CropFrame.StretchFill)
m_CropFrame = CropFrame.Pillarbox;
}
}
}
///
/// Set to true to expand the viewport to fit the screen resolution while maintaining the viewport's aspect ratio.
/// Only applicable when both cropFrameX and cropFrameY are true.
///
[System.Obsolete("Use cropFrame instead", false)]
public bool stretchFill
{
get
{
return m_CropFrame == CropFrame.StretchFill;
}
set
{
if (value)
m_CropFrame = CropFrame.StretchFill;
else
m_CropFrame = CropFrame.Windowbox;
}
}
///
/// Ratio of the rendered Sprites compared to their original size (readonly).
///
public int pixelRatio
{
get
{
if (m_CinemachineCompatibilityMode)
{
if (m_GridSnapping == GridSnapping.UpscaleRenderTexture)
return m_Internal.zoom * m_Internal.cinemachineVCamZoom;
else
return m_Internal.cinemachineVCamZoom;
}
else
{
return m_Internal.zoom;
}
}
}
///
/// Returns if an upscale pass is required.
///
public bool requiresUpscalePass
{
get
{
return m_Internal.requiresUpscaling;
}
}
///
/// Round a arbitrary position to an integer pixel position. Works in world space.
///
/// The position you want to round.
///
/// The rounded pixel position.
/// Depending on the values of upscaleRT and pixelSnapping, it could be a screen pixel position or an art pixel position.
///
public Vector3 RoundToPixel(Vector3 position)
{
float unitsPerPixel = m_Internal.unitsPerPixel;
if (unitsPerPixel == 0.0f)
return position;
Vector3 result;
result.x = Mathf.Round(position.x / unitsPerPixel) * unitsPerPixel;
result.y = Mathf.Round(position.y / unitsPerPixel) * unitsPerPixel;
result.z = Mathf.Round(position.z / unitsPerPixel) * unitsPerPixel;
return result;
}
///
/// Find a pixel-perfect orthographic size as close to targetOrthoSize as possible. Used by Cinemachine to solve compatibility issues with Pixel Perfect Camera.
///
/// Orthographic size from the live Cinemachine Virtual Camera.
/// The corrected orthographic size.
public float CorrectCinemachineOrthoSize(float targetOrthoSize)
{
m_CinemachineCompatibilityMode = true;
if (m_Internal == null)
return targetOrthoSize;
else
return m_Internal.CorrectCinemachineOrthoSize(targetOrthoSize);
}
[SerializeField] int m_AssetsPPU = 100;
[SerializeField] int m_RefResolutionX = 320;
[SerializeField] int m_RefResolutionY = 180;
[SerializeField] CropFrame m_CropFrame;
[SerializeField] GridSnapping m_GridSnapping;
[SerializeField] PixelPerfectFilterMode m_FilterMode = PixelPerfectFilterMode.RetroAA;
// These are obsolete. They are here only for migration.
#if UNITY_EDITOR
[SerializeField] bool m_UpscaleRT;
[SerializeField] bool m_PixelSnapping;
[SerializeField] bool m_CropFrameX;
[SerializeField] bool m_CropFrameY;
[SerializeField] bool m_StretchFill;
#endif
Camera m_Camera;
PixelPerfectCameraInternal m_Internal;
bool m_CinemachineCompatibilityMode;
internal FilterMode finalBlitFilterMode
{
get
{
return m_FilterMode == PixelPerfectFilterMode.RetroAA ? FilterMode.Bilinear : FilterMode.Point;
}
}
internal Vector2Int offscreenRTSize
{
get
{
return new Vector2Int(m_Internal.offscreenRTWidth, m_Internal.offscreenRTHeight);
}
}
Vector2Int cameraRTSize
{
get
{
var targetTexture = m_Camera.targetTexture;
return targetTexture == null ? new Vector2Int(Screen.width, Screen.height) : new Vector2Int(targetTexture.width, targetTexture.height);
}
}
// Snap camera position to pixels using Camera.worldToCameraMatrix.
void PixelSnap()
{
Vector3 cameraPosition = m_Camera.transform.position;
Vector3 roundedCameraPosition = RoundToPixel(cameraPosition);
Vector3 offset = roundedCameraPosition - cameraPosition;
offset.z = -offset.z;
// Get world to local camera matrix without scale
var invPos = Matrix4x4.TRS(cameraPosition + offset, Quaternion.identity, Vector3.one).inverse;
var invRot = Matrix4x4.Rotate(m_Camera.transform.rotation).inverse;
var scaleMatrix = Matrix4x4.Scale(new Vector3(1.0f, 1.0f, -1.0f));
// Calculate inverse TRS
m_Camera.worldToCameraMatrix = scaleMatrix * invRot * invPos;
}
void Awake()
{
m_Camera = GetComponent();
m_Internal = new PixelPerfectCameraInternal(this);
// Case 1249076: Initialize internals immediately after the scene is loaded,
// as the Cinemachine extension may need them before OnBeginContextRendering is called.
UpdateCameraProperties();
}
void UpdateCameraProperties()
{
var rtSize = cameraRTSize;
m_Internal.CalculateCameraProperties(rtSize.x, rtSize.y);
if (m_Internal.useOffscreenRT)
m_Camera.pixelRect = m_Internal.CalculateFinalBlitPixelRect(rtSize.x, rtSize.y);
else
m_Camera.rect = new Rect(0.0f, 0.0f, 1.0f, 1.0f);
}
void OnBeginCameraRendering(ScriptableRenderContext context, Camera camera)
{
if (camera == m_Camera)
{
UpdateCameraProperties();
PixelSnap();
if (!m_CinemachineCompatibilityMode)
{
m_Camera.orthographicSize = m_Internal.orthoSize;
}
UnityEngine.U2D.PixelPerfectRendering.pixelSnapSpacing = m_Internal.unitsPerPixel;
}
}
void OnEndCameraRendering(ScriptableRenderContext context, Camera camera)
{
if (camera == m_Camera)
UnityEngine.U2D.PixelPerfectRendering.pixelSnapSpacing = 0.0f;
}
void OnEnable()
{
m_CinemachineCompatibilityMode = false;
RenderPipelineManager.beginCameraRendering += OnBeginCameraRendering;
RenderPipelineManager.endCameraRendering += OnEndCameraRendering;
}
internal void OnDisable()
{
RenderPipelineManager.beginCameraRendering -= OnBeginCameraRendering;
RenderPipelineManager.endCameraRendering -= OnEndCameraRendering;
m_Camera.rect = new Rect(0.0f, 0.0f, 1.0f, 1.0f);
m_Camera.ResetWorldToCameraMatrix();
}
#if DEVELOPMENT_BUILD || UNITY_EDITOR
// Show on-screen warning about invalid render resolutions.
void OnGUI()
{
Color oldColor = GUI.color;
GUI.color = Color.red;
Vector2Int renderResolution = Vector2Int.zero;
renderResolution.x = m_Internal.useOffscreenRT ? m_Internal.offscreenRTWidth : m_Camera.pixelWidth;
renderResolution.y = m_Internal.useOffscreenRT ? m_Internal.offscreenRTHeight : m_Camera.pixelHeight;
if (renderResolution.x % 2 != 0 || renderResolution.y % 2 != 0)
{
string warning = string.Format("Rendering at an odd-numbered resolution ({0} * {1}). Pixel Perfect Camera may not work properly in this situation.", renderResolution.x, renderResolution.y);
GUILayout.Box(warning);
}
var targetTexture = m_Camera.targetTexture;
Vector2Int rtSize = targetTexture == null ? new Vector2Int(Screen.width, Screen.height) : new Vector2Int(targetTexture.width, targetTexture.height);
if (rtSize.x < refResolutionX || rtSize.y < refResolutionY)
{
GUILayout.Box("Target resolution is smaller than the reference resolution. Image may appear stretched or cropped.");
}
GUI.color = oldColor;
}
#endif
///
/// OnBeforeSerialize implementation.
///
public void OnBeforeSerialize()
{
#if UNITY_EDITOR
m_ComponentVersion = k_CurrentComponentVersion;
#endif
}
///
/// OnAfterSerialize implementation.
///
public void OnAfterDeserialize()
{
#if UNITY_EDITOR
// Upgrade from no serialized version
if (m_ComponentVersion == ComponentVersions.Version_Unserialized)
{
if (m_UpscaleRT)
m_GridSnapping = GridSnapping.UpscaleRenderTexture;
else if (m_PixelSnapping)
m_GridSnapping = GridSnapping.PixelSnapping;
if (m_CropFrameX && m_CropFrameY)
{
if (m_StretchFill)
m_CropFrame = CropFrame.StretchFill;
else
m_CropFrame = CropFrame.Windowbox;
}
else if (m_CropFrameX)
{
m_CropFrame = CropFrame.Pillarbox;
}
else if (m_CropFrameY)
{
m_CropFrame = CropFrame.Letterbox;
}
else
{
m_CropFrame = CropFrame.None;
}
m_ComponentVersion = ComponentVersions.Version_1;
}
#endif
}
}
}