VR Focus Management
When you submit your application, you provide the application and metadata necessary to list it in the Rift Store and launch it from Oculus Home.
Once launched from Oculus Home, you need to write a loop that polls for session status. ovr_GetSessionStatus
returns a struct with the following Booleans:
ShouldQuit
- True if the application should initiate shutdown.HmdPresent
- True if an HMD is present.DisplayLost
- True if the HMD was unplugged or the display driver was manually disabled or encountered a TDR.HmdMounted
- True if the HMD is on the user’s head.IsVisible
- True if the game or experience has VR focus and is visible in the HMD.ShouldRecenter
- True if the application should call ovr_RecenterTrackingOrigin
. This is triggered when the user initiates recentering through the Universal Menu.OverlayPresent
- True if a system overlay is present, such as a dashboard.DepthRequested
- True if runtime is requesting that the application provide depth buffers with projection layers.
Managing When a User Quits
If ShouldQuit
is true
, save the application state and shut down or shut down without saving the application state. The user will automatically return to Oculus Home.
Depending on the type of application, you can prompt the user to start where he or she left off the next time it is opened (e.g., a multi-level game) or you can just start from the beginning of the experience (e.g., a passive video). If this is a multiplayer game, you might want to quit locally without ending the game.
Managing When a User Requests Re-Centering
If ShouldRecenter
is true
, the application should call ovr_RecenterTrackingOrigin
or ovr_SpecifyTrackingOrigin
and be prepared for future tracking positions to be based on a different origin.
Some applications may have reason to ignore the request or to implement it via an internal mechanism other than via ovr_RecenterTrackingOrigin
. In such cases the application can call ovr_ClearShouldRecenterFlag
to cause the re-center request to be cleared.
Managing an Unplugged Headset
If DisplayLost
is true
:
- Pause the application, including audio.
- Display a prompt on the monitor that says that the headset was unplugged.
- Destroy any TextureSwapChains or mirror textures.
- Call
ovrDestroy
. - Poll
ovrSessionStatus::HmdPresent
until true
. - Call
ovrCreate
to recreate the session. - Recreate any
TextureSwapChain
s or mirror textures. - Resume the application.
If ovrDetect
doesn’t isn’t returned as true
after a specified amount of time, act as though ShouldQuit
returned true
. If the user takes no action after a specified amount of time, choose a default action (save the session or close without saving) and close the application.
Note: For multiplayer games, you might want to follow the same process without pausing the game.
Managing an Unavailable Headset
When a user removes the headset or if your application does not have VR focus, HmdMounted
or IsVisible
returns false
. Pause the application until they return true
.
When your application loses VR focus, it automatically stops receiving input. If your application does not use the Oculus input API, it will need to ignore any received input.
Note: For multiplayer games, you might want the game to continue without pausing.
Managing Loss of Windows Focus
When your application loses Windows focus, the Oculus Remote, Xbox controller, and Touch controllers will continue to work normally. However, the application will lose control of the mouse and keyboard.
If your application loses Windows focus and maintains VR focus (IsVisible
), continue to process input and allow the application to run normally. If the keyboard or mouse is needed to continue, prompt the user to remove the headset and use Alt-Tab to regain Windows focus.
Managing Dashboards and Application Focus
With the introduction of Dash, your application should now indicate whether or not it is it is prepared to respond to ovrSessionStatus focus states, including
ovrSessionStatus::HasInputFocus
. For more information, please see
Oculus Dash.
bool shouldQuit = false;
void RunApplication()
{
ovrInitParams initParams = { ovrInit_RequestVersion | ovrInit_FocusAware, OVR_MINOR_VERSION, NULL, 0, 0 };
ovrResult result = ovr_Initialize(&initParams);
VALIDATE(OVR_SUCCESS(result), "Failed to initialize libOVR.");
if (OVR_SUCCESS(result))
{
ovrSession session;
ovrGraphicsLuid luid;
result = ovr_Create(&session, &luid);
result = ovr_WaitToBeginFrame(session, 0);
result = ovr_BeginFrame(session, 0);
if (OVR_SUCCESS(result))
{
ovrSessionStatus ss;
<create graphics device with luid>
<create render target via ovr_CreateTextureSwapChain>
while (!shouldQuit)
{
<get next frame pose, e.g. via ovr_GetEyePoses>
<render frame>
result = ovr_EndFrame(...);
if (result == ovrSuccess_NotVisible)
{
<turn off audio output>
do { // Wait until we regain visibility or should quit
<sleep>
result = ovr_GetSessionStatus(session, &ss);
if (ss.ShouldQuit)
shouldQuit = true;
} while (OVR_SUCCESS(result) && !ss.IsVisible && !shouldQuit);
<possibly re-enable audio>
}
else if (result == ovrError_DisplayLost)
{
// We can either immediately quit or do the following:
<destroy render target and graphics device>
ovr_Destroy(session);
do { // Spin while trying to recreate session.
result = ovr_Create(&session, &luid);
} while (OVR_FAILURE(result) && !shouldQuit);
if (OVR_SUCCESS(result))
{
<recreate graphics device with luid>
<recreate render target via ovr_CreateTextureSwapChain>
}
}
else if (OVR_FAILURE(result))
{
shouldQuit = true;
}
ovr_GetSessionStatus(session, &ss);
if (ss.ShouldQuit)
shouldQuit = true;
if (ss.ShouldRecenter)
{
ovr_RecenterTrackingOrigin(session); // or ovr_ClearShouldRecenterFlag(session) to ignore the request.
<do anything else needed to handle this>
}
}
<destroy render target via ovr_DestroyTextureSwapChain>
<destroy graphics device>
ovr_Destroy(session);
}
ovr_Shutdown();
}
}
Application behavior expectations under OpenXR are the same as they are under the LibOVR CAPI, except that the OpenXR API presents focus status changes mostly via session state change events. OpenXR session state analogues to the LibOVR CAPI ovrSessionStatus are thus:
ShouldQuit
- XR_TYPE_EVENT_DATA_SESSION_STATE_CHANGED / XR_SESSION_STATE_EXITINGHmdPresent
- No OpenXR equivalent exists. The ovr_Detect may be used by OpenXR applications.DisplayLost
- XR_TYPE_EVENT_DATA_SESSION_STATE_CHANGED / XR_SESSION_STATE_LOSS_PENDINGHmdMounted
- XR_TYPE_EVENT_DATA_SESSION_STATE_CHANGED / XR_SESSION_STATE_READY,XR_SESSION_STATE_STOPPINGIsVisible
- XR_TYPE_EVENT_DATA_SESSION_STATE_CHANGED / XR_SESSION_STATE_VISIBLE,XR_SESSION_STATE_SYNCHRONIZEDShouldRecenter
- XR_TYPE_EVENT_DATA_RECENTER_REQUESTED_OCULUS, enabled by the XR_OCULUS_recenter_event extension.HasInputFocus
- XR_TYPE_EVENT_DATA_SESSION_STATE_CHANGED / XR_SESSION_STATE_FOCUSED,XR_SESSION_STATE_VISIBLEOverlayPresent
- No OpenXR equivalent exists nor is necessary.DepthRequested
- No OpenXR equivalent exists. Depth buffers should always be provided when the XR_KHR_composition_layer_depth extension is enabled.