//======= Copyright (c) Valve Corporation, All rights reserved. ===============
|
|
//
|
|
// Purpose: Handles rendering of all SteamVR_Cameras
|
|
//
|
|
//=============================================================================
|
|
|
|
using UnityEngine;
|
|
using System.Collections;
|
|
using Valve.VR;
|
|
|
|
public class SteamVR_Render : MonoBehaviour
|
|
{
|
|
public bool pauseGameWhenDashboardIsVisible = true;
|
|
public bool lockPhysicsUpdateRateToRenderFrequency = true;
|
|
|
|
public SteamVR_ExternalCamera externalCamera;
|
|
public string externalCameraConfigPath = "externalcamera.cfg";
|
|
|
|
public ETrackingUniverseOrigin trackingSpace = ETrackingUniverseOrigin.TrackingUniverseStanding;
|
|
|
|
static public EVREye eye { get; private set; }
|
|
|
|
static private SteamVR_Render _instance;
|
|
static public SteamVR_Render instance
|
|
{
|
|
get
|
|
{
|
|
if (_instance == null)
|
|
{
|
|
_instance = GameObject.FindObjectOfType<SteamVR_Render>();
|
|
|
|
if (_instance == null)
|
|
_instance = new GameObject("[SteamVR]").AddComponent<SteamVR_Render>();
|
|
}
|
|
return _instance;
|
|
}
|
|
}
|
|
|
|
void OnDestroy()
|
|
{
|
|
_instance = null;
|
|
}
|
|
|
|
static private bool isQuitting;
|
|
void OnApplicationQuit()
|
|
{
|
|
isQuitting = true;
|
|
SteamVR.SafeDispose();
|
|
}
|
|
|
|
static public void Add(SteamVR_Camera vrcam)
|
|
{
|
|
if (!isQuitting)
|
|
instance.AddInternal(vrcam);
|
|
}
|
|
|
|
static public void Remove(SteamVR_Camera vrcam)
|
|
{
|
|
if (!isQuitting && _instance != null)
|
|
instance.RemoveInternal(vrcam);
|
|
}
|
|
|
|
static public SteamVR_Camera Top()
|
|
{
|
|
if (!isQuitting)
|
|
return instance.TopInternal();
|
|
|
|
return null;
|
|
}
|
|
|
|
private SteamVR_Camera[] cameras = new SteamVR_Camera[0];
|
|
|
|
void AddInternal(SteamVR_Camera vrcam)
|
|
{
|
|
var camera = vrcam.GetComponent<Camera>();
|
|
var length = cameras.Length;
|
|
var sorted = new SteamVR_Camera[length + 1];
|
|
int insert = 0;
|
|
for (int i = 0; i < length; i++)
|
|
{
|
|
var c = cameras[i].GetComponent<Camera>();
|
|
if (i == insert && c.depth > camera.depth)
|
|
sorted[insert++] = vrcam;
|
|
|
|
sorted[insert++] = cameras[i];
|
|
}
|
|
if (insert == length)
|
|
sorted[insert] = vrcam;
|
|
|
|
cameras = sorted;
|
|
}
|
|
|
|
void RemoveInternal(SteamVR_Camera vrcam)
|
|
{
|
|
var length = cameras.Length;
|
|
int count = 0;
|
|
for (int i = 0; i < length; i++)
|
|
{
|
|
var c = cameras[i];
|
|
if (c == vrcam)
|
|
++count;
|
|
}
|
|
if (count == 0)
|
|
return;
|
|
|
|
var sorted = new SteamVR_Camera[length - count];
|
|
int insert = 0;
|
|
for (int i = 0; i < length; i++)
|
|
{
|
|
var c = cameras[i];
|
|
if (c != vrcam)
|
|
sorted[insert++] = c;
|
|
}
|
|
|
|
cameras = sorted;
|
|
}
|
|
|
|
SteamVR_Camera TopInternal()
|
|
{
|
|
if (cameras.Length > 0)
|
|
return cameras[cameras.Length - 1];
|
|
|
|
return null;
|
|
}
|
|
|
|
public TrackedDevicePose_t[] poses = new TrackedDevicePose_t[OpenVR.k_unMaxTrackedDeviceCount];
|
|
public TrackedDevicePose_t[] gamePoses = new TrackedDevicePose_t[0];
|
|
|
|
static private bool _pauseRendering;
|
|
static public bool pauseRendering
|
|
{
|
|
get { return _pauseRendering; }
|
|
set
|
|
{
|
|
_pauseRendering = value;
|
|
|
|
var compositor = OpenVR.Compositor;
|
|
if (compositor != null)
|
|
compositor.SuspendRendering(value);
|
|
}
|
|
}
|
|
|
|
private WaitForEndOfFrame waitForEndOfFrame = new WaitForEndOfFrame();
|
|
|
|
private IEnumerator RenderLoop()
|
|
{
|
|
while (Application.isPlaying)
|
|
{
|
|
yield return waitForEndOfFrame;
|
|
|
|
if (pauseRendering)
|
|
continue;
|
|
|
|
var compositor = OpenVR.Compositor;
|
|
if (compositor != null)
|
|
{
|
|
if (!compositor.CanRenderScene())
|
|
continue;
|
|
|
|
compositor.SetTrackingSpace(trackingSpace);
|
|
}
|
|
|
|
var overlay = SteamVR_Overlay.instance;
|
|
if (overlay != null)
|
|
overlay.UpdateOverlay();
|
|
|
|
RenderExternalCamera();
|
|
}
|
|
}
|
|
|
|
void RenderExternalCamera()
|
|
{
|
|
if (externalCamera == null)
|
|
return;
|
|
|
|
if (!externalCamera.gameObject.activeInHierarchy)
|
|
return;
|
|
|
|
var frameSkip = (int)Mathf.Max(externalCamera.config.frameSkip, 0.0f);
|
|
if (Time.frameCount % (frameSkip + 1) != 0)
|
|
return;
|
|
|
|
// Keep external camera relative to the most relevant vr camera.
|
|
externalCamera.AttachToCamera(TopInternal());
|
|
|
|
externalCamera.RenderNear();
|
|
externalCamera.RenderFar();
|
|
}
|
|
|
|
float sceneResolutionScale = 1.0f, timeScale = 1.0f;
|
|
|
|
private void OnInputFocus(bool hasFocus)
|
|
{
|
|
if (hasFocus)
|
|
{
|
|
if (pauseGameWhenDashboardIsVisible)
|
|
{
|
|
Time.timeScale = timeScale;
|
|
}
|
|
|
|
SteamVR_Camera.sceneResolutionScale = sceneResolutionScale;
|
|
}
|
|
else
|
|
{
|
|
if (pauseGameWhenDashboardIsVisible)
|
|
{
|
|
timeScale = Time.timeScale;
|
|
Time.timeScale = 0.0f;
|
|
}
|
|
|
|
sceneResolutionScale = SteamVR_Camera.sceneResolutionScale;
|
|
SteamVR_Camera.sceneResolutionScale = 0.5f;
|
|
}
|
|
}
|
|
|
|
void OnQuit(VREvent_t vrEvent)
|
|
{
|
|
#if UNITY_EDITOR
|
|
foreach (System.Reflection.Assembly a in System.AppDomain.CurrentDomain.GetAssemblies())
|
|
{
|
|
var t = a.GetType("UnityEditor.EditorApplication");
|
|
if (t != null)
|
|
{
|
|
t.GetProperty("isPlaying").SetValue(null, false, null);
|
|
break;
|
|
}
|
|
}
|
|
#else
|
|
Application.Quit();
|
|
#endif
|
|
}
|
|
|
|
private string GetScreenshotFilename(uint screenshotHandle, EVRScreenshotPropertyFilenames screenshotPropertyFilename)
|
|
{
|
|
var error = EVRScreenshotError.None;
|
|
var capacity = OpenVR.Screenshots.GetScreenshotPropertyFilename(screenshotHandle, screenshotPropertyFilename, null, 0, ref error);
|
|
if (error != EVRScreenshotError.None && error != EVRScreenshotError.BufferTooSmall)
|
|
return null;
|
|
if (capacity > 1)
|
|
{
|
|
var result = new System.Text.StringBuilder((int)capacity);
|
|
OpenVR.Screenshots.GetScreenshotPropertyFilename(screenshotHandle, screenshotPropertyFilename, result, capacity, ref error);
|
|
if (error != EVRScreenshotError.None)
|
|
return null;
|
|
return result.ToString();
|
|
}
|
|
return null;
|
|
}
|
|
|
|
private void OnRequestScreenshot(VREvent_t vrEvent)
|
|
{
|
|
var screenshotHandle = vrEvent.data.screenshot.handle;
|
|
var screenshotType = (EVRScreenshotType)vrEvent.data.screenshot.type;
|
|
|
|
if (screenshotType == EVRScreenshotType.StereoPanorama)
|
|
{
|
|
string previewFilename = GetScreenshotFilename(screenshotHandle, EVRScreenshotPropertyFilenames.Preview);
|
|
string VRFilename = GetScreenshotFilename(screenshotHandle, EVRScreenshotPropertyFilenames.VR);
|
|
|
|
if (previewFilename == null || VRFilename == null)
|
|
return;
|
|
|
|
// Do the stereo panorama screenshot
|
|
// Figure out where the view is
|
|
GameObject screenshotPosition = new GameObject("screenshotPosition");
|
|
screenshotPosition.transform.position = SteamVR_Render.Top().transform.position;
|
|
screenshotPosition.transform.rotation = SteamVR_Render.Top().transform.rotation;
|
|
screenshotPosition.transform.localScale = SteamVR_Render.Top().transform.lossyScale;
|
|
SteamVR_Utils.TakeStereoScreenshot(screenshotHandle, screenshotPosition, 32, 0.064f, ref previewFilename, ref VRFilename);
|
|
|
|
// and submit it
|
|
OpenVR.Screenshots.SubmitScreenshot(screenshotHandle, screenshotType, previewFilename, VRFilename);
|
|
}
|
|
}
|
|
|
|
void OnEnable()
|
|
{
|
|
StartCoroutine(RenderLoop());
|
|
SteamVR_Events.InputFocus.Listen(OnInputFocus);
|
|
SteamVR_Events.System(EVREventType.VREvent_Quit).Listen(OnQuit);
|
|
SteamVR_Events.System(EVREventType.VREvent_RequestScreenshot).Listen(OnRequestScreenshot);
|
|
#if UNITY_2017_1_OR_NEWER
|
|
Application.onBeforeRender += OnBeforeRender;
|
|
#else
|
|
Camera.onPreCull += OnCameraPreCull;
|
|
#endif
|
|
var vr = SteamVR.instance;
|
|
if (vr == null)
|
|
{
|
|
enabled = false;
|
|
return;
|
|
}
|
|
var types = new EVRScreenshotType[] { EVRScreenshotType.StereoPanorama };
|
|
OpenVR.Screenshots.HookScreenshot(types);
|
|
}
|
|
|
|
void OnDisable()
|
|
{
|
|
StopAllCoroutines();
|
|
SteamVR_Events.InputFocus.Remove(OnInputFocus);
|
|
SteamVR_Events.System(EVREventType.VREvent_Quit).Remove(OnQuit);
|
|
SteamVR_Events.System(EVREventType.VREvent_RequestScreenshot).Remove(OnRequestScreenshot);
|
|
#if UNITY_2017_1_OR_NEWER
|
|
Application.onBeforeRender -= OnBeforeRender;
|
|
#else
|
|
Camera.onPreCull -= OnCameraPreCull;
|
|
#endif
|
|
}
|
|
|
|
void Awake()
|
|
{
|
|
if (externalCamera == null && System.IO.File.Exists(externalCameraConfigPath))
|
|
{
|
|
var prefab = Resources.Load<GameObject>("SteamVR_ExternalCamera");
|
|
var instance = Instantiate(prefab);
|
|
instance.gameObject.name = "External Camera";
|
|
|
|
externalCamera = instance.transform.GetChild(0).GetComponent<SteamVR_ExternalCamera>();
|
|
externalCamera.configPath = externalCameraConfigPath;
|
|
externalCamera.ReadConfig();
|
|
}
|
|
}
|
|
|
|
public void UpdatePoses()
|
|
{
|
|
var compositor = OpenVR.Compositor;
|
|
if (compositor != null)
|
|
{
|
|
compositor.GetLastPoses(poses, gamePoses);
|
|
SteamVR_Events.NewPoses.Send(poses);
|
|
SteamVR_Events.NewPosesApplied.Send();
|
|
}
|
|
}
|
|
|
|
#if UNITY_2017_1_OR_NEWER
|
|
void OnBeforeRender() { UpdatePoses(); }
|
|
#else
|
|
void OnCameraPreCull(Camera cam)
|
|
{
|
|
#if !( UNITY_5_4 )
|
|
if (cam.cameraType != CameraType.VR)
|
|
return;
|
|
#endif
|
|
// Only update poses on the first camera per frame.
|
|
if (Time.frameCount != lastFrameCount)
|
|
{
|
|
lastFrameCount = Time.frameCount;
|
|
UpdatePoses();
|
|
}
|
|
}
|
|
static int lastFrameCount = -1;
|
|
#endif
|
|
|
|
void Update()
|
|
{
|
|
// Force controller update in case no one else called this frame to ensure prevState gets updated.
|
|
SteamVR_Controller.Update();
|
|
|
|
// Dispatch any OpenVR events.
|
|
var system = OpenVR.System;
|
|
if (system != null)
|
|
{
|
|
var vrEvent = new VREvent_t();
|
|
var size = (uint)System.Runtime.InteropServices.Marshal.SizeOf(typeof(VREvent_t));
|
|
for (int i = 0; i < 64; i++)
|
|
{
|
|
if (!system.PollNextEvent(ref vrEvent, size))
|
|
break;
|
|
|
|
switch ((EVREventType)vrEvent.eventType)
|
|
{
|
|
case EVREventType.VREvent_InputFocusCaptured: // another app has taken focus (likely dashboard)
|
|
if (vrEvent.data.process.oldPid == 0)
|
|
{
|
|
SteamVR_Events.InputFocus.Send(false);
|
|
}
|
|
break;
|
|
case EVREventType.VREvent_InputFocusReleased: // that app has released input focus
|
|
if (vrEvent.data.process.pid == 0)
|
|
{
|
|
SteamVR_Events.InputFocus.Send(true);
|
|
}
|
|
break;
|
|
case EVREventType.VREvent_ShowRenderModels:
|
|
SteamVR_Events.HideRenderModels.Send(false);
|
|
break;
|
|
case EVREventType.VREvent_HideRenderModels:
|
|
SteamVR_Events.HideRenderModels.Send(true);
|
|
break;
|
|
default:
|
|
SteamVR_Events.System((EVREventType)vrEvent.eventType).Send(vrEvent);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Ensure various settings to minimize latency.
|
|
Application.targetFrameRate = -1;
|
|
Application.runInBackground = true; // don't require companion window focus
|
|
QualitySettings.maxQueuedFrames = -1;
|
|
QualitySettings.vSyncCount = 0; // this applies to the companion window
|
|
|
|
if (lockPhysicsUpdateRateToRenderFrequency && Time.timeScale > 0.0f)
|
|
{
|
|
var vr = SteamVR.instance;
|
|
if (vr != null)
|
|
{
|
|
var timing = new Compositor_FrameTiming();
|
|
timing.m_nSize = (uint)System.Runtime.InteropServices.Marshal.SizeOf(typeof(Compositor_FrameTiming));
|
|
vr.compositor.GetFrameTiming(ref timing, 0);
|
|
|
|
Time.fixedDeltaTime = Time.timeScale / vr.hmd_DisplayFrequency;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|