Skip to content

Inefficient sample code for getting screen contents in Unity on Meta Quest to workaround lack of 'camera access'

License

Notifications You must be signed in to change notification settings

trev3d/QuestDisplayAccessDemo

Repository files navigation

Quest Display Access Demo

Developers want camera access on the Meta Quest. Meta hasn't let us have it yet. The next best thing is display access. Thanks to Android's MediaProjector API, you can copy the display image to a texture in your Unity project in 'near realtime' (several frames of latency) as demonstrated here. No PC, embedded browser, or dev mode required

scrcpy_VmJvDrjcQL

Special thanks

@t-34400's QuestMediaProjection repo demonstrated using Google's ML barcode reader.

@Gustorvo's pull request replaced a texture copy over the JNI with a pointer

Setup

  • Add the 'DisplayCapture' and 'DepthKit' folders to your project.

Screenshot_1

  • Open your player settings and set your Android Target API level to Android 14.0 (API level 34)

image

  • Make sure you're using custom Main Manifest and Main Gradle Template files

Screenshot_2

  • Update your AndroidManifest.xml file with these lines:
<!--ADD THESE LINES TO YOUR MANIFEST <MANIFEST> SECTION!!!-->
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_MEDIA_PROJECTION" />
<!--ADD THESE LINES TO YOUR MANIFEST <MANIFEST> SECTION!!!-->
<!--ADD THESE LINES TO YOUR MANIFEST <APPLICATION> SECTION!!!-->
<activity android:name="com.trev3d.DisplayCapture.DisplayCaptureRequestActivity" android:exported="false" />
<service android:name="com.trev3d.DisplayCapture.DisplayCaptureNotificationService" android:exported="false" android:foregroundServiceType="mediaProjection" />
<!--ADD THESE LINES TO YOUR MANIFEST <APPLICATION> SECTION!!!-->

image

  • Update your mainTemplate.gradle file with these lines:
/* ADD THESE LINES TO YOUR GRADLE DEPENDENCIES SECTION */
implementation 'androidx.appcompat:appcompat:1.6.1'
implementation 'com.google.mlkit:barcode-scanning:17.3.0'
implementation 'com.google.code.gson:gson:2.11.0'
/* ADD THESE LINES TO YOUR GRADLE DEPENDENCIES SECTION */

image

  • Refer to Demo.scene on setting up the necessary components. I'll update and document these in a future commit (sorry)

image

⚠️ Issues (please read)!

To fix

⚠️ MediaProjection stop callback doesn't seem to work correctly

Gotchas

⚠️⚠️⚠️ Google's ML barcode scanner annoyingly reorders corner point order so that codes always face 'up' relative to the viewer. This makes it near impossible to properly track the orientation of flat-facing codes as the orientation will always face 'toward' you. You can get around this by using two codes and the vector between them as your orientation

⚠️ While display capture and QR code reading will work on any headset, QR code tracking will only work on Quest 3 / Quest 3S due to other headsets lacking depth estimation features.

⚠️ Display capture is expensive, as is QR code tracking

⚠️ You may need to be on Quest system software v68 or higher

⚠️ This only works on-headset. This will not work through QuestLink

⚠️ You cannot video record the display 'normally' while this app's MediaProjector session is running. You can instead use scrcpy to record any prototypes or demos you make with this.

⚠️ This still isn't proper camera access. Any virtual elements will obscure physical objects in the image. If you need to track something, you must not render anything on top of it!

Other info

  • The captured view is ~82 degrees in horizontal and vertical FOV on Quest 3
  • The capture texture is 1024x1024
  • MediaProjection currently captures frames from the left eye buffer
  • Quest system camera settings do not affect the capture resolution, framerate, or eye

Reference

About

Inefficient sample code for getting screen contents in Unity on Meta Quest to workaround lack of 'camera access'

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published