Ocean
Loading...
Searching...
No Matches
NativeApplication.h
Go to the documentation of this file.
1/*
2 * Copyright (c) Meta Platforms, Inc. and affiliates.
3 *
4 * This source code is licensed under the MIT license found in the
5 * LICENSE file in the root directory of this source tree.
6 */
7
8#ifndef META_OCEAN_PLATFORM_META_QUEST_OPENXR_APPLICATION_NATIVE_APPLICATION_H
9#define META_OCEAN_PLATFORM_META_QUEST_OPENXR_APPLICATION_NATIVE_APPLICATION_H
10
12
14
16
18
19#include <openxr/openxr.h>
20
21#ifdef OCEAN_PLATFORM_BUILD_ANDROID
22 #include <android/native_window_jni.h>
23 #include <android_native_app_glue.h>
24 #include <android/window.h>
25#endif
26
27namespace Ocean
28{
29
30namespace Platform
31{
32
33namespace Meta
34{
35
36namespace Quest
37{
38
39namespace OpenXR
40{
41
42namespace Application
43{
44
45/**
46 * This class implements a basic OpenXR application for Quest devices based on an Android NativeActivity.
47 * @ingroup platformmetaquestopenxrapplication
48 */
49class OCEAN_PLATFORM_META_QUEST_OPENXR_APPLICATION_EXPORT NativeApplication
50{
51 protected:
52
53 /**
54 * Definition of an unordered map mapping from Android permissions to a counter which is used to delay the announcement of Android permissions;
55 * Positive counter values indicate that the permission has been granted and the counter is used to delay the announcement of the granted permission, cf. `handlePendingAndroidPermissions()` for details.
56 * Negative counter values are used to store the state of a permission.
57 */
58 typedef std::unordered_map<std::string, int> AndroidPermissionMap;
59
60 /// The counter value for permissions not yet requested.
61 static constexpr int permissionNotYetRequested_ = -2;
62
63 //// The counter value for permissions not yet granted.
64 static constexpr int permissionNotYetGranted_ = -1;
65
66 /**
67 * Definition of an unordered set holding strings.
68 */
69 typedef std::unordered_set<std::string> StringSet;
70
71 /**
72 * Definition of a vector holding XrViewConfigurationView objects.
73 */
74 typedef std::vector<XrViewConfigurationView> XrViewConfigurationViews;
75
76 public:
77
78 /**
79 * Destructs this object.
80 */
82
83 /**
84 * The run method of this application.
85 * Calling this function will start the application logic and will return before the application closes.
86 * True, if the application closes after a successful execution; False, if the application did not start successfully
87 */
88 virtual bool run();
89
90 /**
91 * Returns the device type that this application is configured for
92 * @return The device type
93 */
94 inline Device::DeviceType deviceType() const;
95
96 /**
97 * Returns the device name that this application is configured for
98 * @return The device name
99 */
100 inline std::string deviceName() const;
101
102 /**
103 * Requests an Android permission that needs to be granted by the user.
104 * In case the user grants the permission, the callback function `onPermissionGranted()` will be invoked (when requesting the permission for the first time).
105 * @param permission The name of the Android permission to request, must be valid
106 * @see onPermissionGranted().
107 */
108 void requestAndroidPermission(std::string&& permission);
109
110 /**
111 * Requests several Android permissions that needs to be granted by the user.
112 * In case the user grants the permission, the callback function `onPermissionGranted()` will be invoked (when requesting the permission for the first time).
113 * @param permissions The names of the Android permissions to request, must be valid
114 * @see onPermissionGranted().
115 */
116 inline void requestAndroidPermissions(std::vector<std::string>&& permissions);
117
118 /**
119 * Returns the Android permissions which have been granted by the user.
120 * @return The currently granted Android permissions
121 */
123
124 protected:
125
126#ifdef OCEAN_PLATFORM_BUILD_ANDROID
127
128 /**
129 * Creates a new application object.
130 * @param androidApp The android app object as provided in the main function of the native activity, must be valid
131 */
132 explicit NativeApplication(struct android_app* androidApp);
133
134#endif // OCEAN_PLATFORM_BUILD_ANDROID
135
136 /**
137 * Disabled copy constructor.
138 * @param nativeApplication Application object which would have been copied
139 */
140 NativeApplication(const NativeApplication& nativeApplication) = delete;
141
142 /**
143 * Returns the names of the necessary OpenXR extensions the application needs.
144 * @return The application's necessary OpenXR extensions
145 */
147
148 /**
149 * Main loop of the application.
150 */
151 virtual void applicationLoop();
152
153 /**
154 * Processes all pending Android events via ALooper_pollOnce.
155 */
156 virtual void processAndroidEvents();
157
158 /**
159 * Processes all pending OpenXR events.
160 */
161 virtual void processOpenXREvents();
162
163 /**
164 * Event function for OpenXR events which are not handled in processOpenXREvents().
165 * @param xrEventDataBaseHeader The base header of the OpenXR event
166 */
167 virtual void onOpenXREvent(const XrEventDataBaseHeader& xrEventDataBaseHeader);
168
169 /**
170 * Event functions for changed reference spaces.
171 * @param xrReferenceSpaceType The type of the reference which has changed
172 * @param previous_T_changed The transformation between the changed space and the previous space, invalid if unknown
173 * @param changeTime The time after which xrLocateSpace or xrLocateViews will return values that respect the change
174 */
175 virtual void onChangedReferenceSpace(const XrReferenceSpaceType xrReferenceSpaceType, const HomogenousMatrix4& previous_T_changed, const XrTime& changeTime);
176
177 /**
178 * Returns the current state of the OpenXR session.
179 * @return The session's state
180 */
181 inline XrSessionState xrSessionState() const;
182
183 /**
184 * Creates the OpenXR session.
185 * @param xrViewConfigurationViews The view configuration(s) for the frame buffer(s)
186 * @return True, if succeeded
187 */
188 virtual bool createOpenXRSession(const XrViewConfigurationViews& xrViewConfigurationViews) = 0;
189
190 /**
191 * Releases the OpenXR session.
192 */
193 virtual void releaseOpenXRSession() = 0;
194
195 /**
196 * Event function called after OpenXR has been initialized.
197 */
199
200 /**
201 * Event function called whenever the state of the OpenXR session changed.
202 * @param xrEventDataSessionStateChanged The new OpenXR session state
203 */
204 virtual void onOpenXRSessionChanged(const XrEventDataSessionStateChanged& xrEventDataSessionStateChanged);
205
206 /**
207 * Event function called whenever the session is ready, when the session state changed to XR_SESSION_STATE_READY.
208 */
209 virtual void onOpenXRSessionReady();
210
211 /**
212 * Event function called whenever the session is stopping, when the session state changed to XR_SESSION_STATE_STOPPING.
213 */
215
216 /**
217 * Event function call when an Android (or Oculus) permission is granted
218 * @param permission The name of the permission that has been granted
219 */
220 virtual void onAndroidPermissionGranted(const std::string& permission);
221
222 /**
223 * Event function call when the Android Activity is started.
224 */
225 virtual void onActivityStart();
226
227 /**
228 * Event function call when the Android Activity is resumed.
229 */
230 virtual void onActivityResume();
231
232 /**
233 * Event function call when the Android Activity is paused.
234 */
235 virtual void onActivityPause();
236
237 /**
238 * Event function call when the Android Activity is stopped.
239 */
240 virtual void onActivityStop();
241
242 /**
243 * Event function call when the Android Activity is destroyed.
244 */
245 virtual void onActivityDestroy();
246
247 /**
248 * Event function call when the window of the Android Activity is initialized.
249 */
250 virtual void onActivityInitWindow();
251
252 /**
253 * Event function call when the window of the Android Activity is terminated.
254 */
255 virtual void onActivityTermWindow();
256
257 /**
258 * Event function called at startup to set e.g., windows flags.
259 */
260 virtual void onStartup();
261
262 /**
263 * Idle event function called within the main loop whenever all Android related events have been processed.
264 */
265 virtual void onIdle() = 0;
266
267 /**
268 * Checks if a specific permission has been granted.
269 * @param permission The string identifier of a permission, must be valid
270 * @param translate Optionally translate permissions internally (short to long name), must be `false` in conjunction with Oculus permission such as `com.oculus.permission.USE_SCENE`
271 * @return True if the specified permission has been granted, otherwise false
272 */
273 bool isAndroidPermissionGranted(const std::string& permission, const bool translate = false) const;
274
275 /**
276 * Checks if any item from a set of pending permissions has been granted to the app
277 * If a pending permission has been found to be granted, it will be removed from the set of pending permissions and added to the set of granted permission.
278 * This will also call `onPermissionGranted()` in order to notify the application about the permission changes.
279 *
280 * Important note:
281 *
282 * In Android, when permissions are granted by the user for the firs time, there seems to be a delay
283 * for some permissions between when (A) they are marked as granted, i.e., `hasPermission()` returns
284 * `true` and (B) when the permission is actually granted and can be used. Any action that requires
285 * such an affected permission that is run in the time between (A) and (B) is still likely to fail (as
286 * if the permission had not been granted in the first place). The permission will be usable after
287 * a few moments (i.e., after a few milliseconds or after a few iterations of the main loop of the app).
288 *
289 * On consecutive starts of the app and assuming all requested permissions have been granted before,
290 * this is no longer a problem; `hasPermission()` returns `true` and the permission is immediately
291 * usable.
292 *
293 * The function applies a short delay before announcing them via `onPermissionGranted()` to all
294 * permissions.
295 *
296 * @param firstCheck True when this function is called for the first time since the app started, otherwise false; cf. the note above.
297 */
298 void handlePendingAndroidPermissions(const bool firstCheck);
299
300 /**
301 * Disabled copy operator.
302 * @param nativeApplication Application object which would have been copied
303 * @return Reference to this object
304 */
305 NativeApplication& operator=(const NativeApplication& nativeApplication) = delete;
306
307 /**
308 * Android's command event function.
309 * @param androidApp The android app object
310 * @param cmd The command
311 */
312 static void onAndroidCommand(struct android_app* androidApp, int32_t cmd);
313
314 protected:
315
316 /// The OpenXR instance used by this application.
318
319 /// The OpenXR view configuration type to be used.
320 XrViewConfigurationType xrViewConfigurationType_ = XR_VIEW_CONFIGURATION_TYPE_PRIMARY_STEREO;
321
322#ifdef OCEAN_PLATFORM_BUILD_ANDROID
323
324 /// The Java native interface envrionment.
325 JNIEnv* jniEnv_ = nullptr;
326
327 // The android app object as provided in the main function of the native activity.
328 struct android_app* androidApp_ = nullptr;
329
330#endif
331
332 /// True, if the application has been resumed; False, if the application has e.g., been paused or stopped.
333 bool applicationResumed_ = false;
334
335 /// The native Android window, if any.
336 ANativeWindow* androidNativeWindow_ = nullptr;
337
338 /// The device type that this application is configured for.
339 Device::DeviceType deviceType_ = Meta::Quest::Device::DT_UNKNOWN;
340
341 /// The map mapping the names of pending Android permissions to a delay counter.
343
344 /// The set holding the names of granted Android permissions.
346
347 /// The lock for Android permissions.
349
350 private:
351
352 /// The current OpenXR session state.
353 XrSessionState xrSessionState_ = XR_SESSION_STATE_UNKNOWN;
354
355 /// True, if the OpenXR session is ready to be used.
356 bool xrSessionIsReady_ = false;
357};
358
363
364inline std::string NativeApplication::deviceName() const
365{
367}
368
369inline void NativeApplication::requestAndroidPermissions(std::vector<std::string>&& permissions)
370{
371 const ScopedLock scopedLock(androidPermissionLock_);
372
373 for (std::string& permission : permissions)
374 {
375 requestAndroidPermission(std::move(permission));
376 }
377}
378
379inline XrSessionState NativeApplication::xrSessionState() const
380{
381 return xrSessionState_;
382}
383
384}
385
386}
387
388}
389
390}
391
392}
393
394}
395
396#endif // META_OCEAN_PLATFORM_META_QUEST_OPENXR_APPLICATION_NATIVE_APPLICATION_H
This class implements a recursive lock object.
Definition Lock.h:31
DeviceType
Definition of individual device types.
Definition platform/meta/quest/Device.h:37
static std::string deviceName()
Returns the name of the device.
Definition platform/meta/quest/Device.h:82
This class implements a basic OpenXR application for Quest devices based on an Android NativeActivity...
Definition NativeApplication.h:50
virtual void onActivityInitWindow()
Event function call when the window of the Android Activity is initialized.
Device::DeviceType deviceType() const
Returns the device type that this application is configured for.
Definition NativeApplication.h:359
void requestAndroidPermission(std::string &&permission)
Requests an Android permission that needs to be granted by the user.
virtual void onActivityStart()
Event function call when the Android Activity is started.
void handlePendingAndroidPermissions(const bool firstCheck)
Checks if any item from a set of pending permissions has been granted to the app If a pending permiss...
virtual void onAndroidPermissionGranted(const std::string &permission)
Event function call when an Android (or Oculus) permission is granted.
virtual void applicationLoop()
Main loop of the application.
NativeApplication & operator=(const NativeApplication &nativeApplication)=delete
Disabled copy operator.
std::unordered_map< std::string, int > AndroidPermissionMap
Definition of an unordered map mapping from Android permissions to a counter which is used to delay t...
Definition NativeApplication.h:58
NativeApplication(struct android_app *androidApp)
Creates a new application object.
virtual void onOpenXRSessionStopping()
Event function called whenever the session is stopping, when the session state changed to XR_SESSION_...
bool isAndroidPermissionGranted(const std::string &permission, const bool translate=false) const
Checks if a specific permission has been granted.
virtual void processOpenXREvents()
Processes all pending OpenXR events.
virtual void onOpenXRInstanceInitialized()
Event function called after OpenXR has been initialized.
virtual void onStartup()
Event function called at startup to set e.g., windows flags.
StringSet grantedAndroidPermissions() const
Returns the Android permissions which have been granted by the user.
virtual void onActivityStop()
Event function call when the Android Activity is stopped.
std::unordered_set< std::string > StringSet
Definition of an unordered set holding strings.
Definition NativeApplication.h:69
std::vector< XrViewConfigurationView > XrViewConfigurationViews
Definition of a vector holding XrViewConfigurationView objects.
Definition NativeApplication.h:74
virtual void releaseOpenXRSession()=0
Releases the OpenXR session.
virtual void onIdle()=0
Idle event function called within the main loop whenever all Android related events have been process...
AndroidPermissionMap pendingAndroidPermissionMap_
The map mapping the names of pending Android permissions to a delay counter.
Definition NativeApplication.h:342
XrSessionState xrSessionState() const
Returns the current state of the OpenXR session.
Definition NativeApplication.h:379
virtual void onChangedReferenceSpace(const XrReferenceSpaceType xrReferenceSpaceType, const HomogenousMatrix4 &previous_T_changed, const XrTime &changeTime)
Event functions for changed reference spaces.
std::string deviceName() const
Returns the device name that this application is configured for.
Definition NativeApplication.h:364
virtual void processAndroidEvents()
Processes all pending Android events via ALooper_pollOnce.
virtual void onActivityDestroy()
Event function call when the Android Activity is destroyed.
Platform::OpenXR::Instance xrInstance_
The OpenXR instance used by this application.
Definition NativeApplication.h:317
virtual bool createOpenXRSession(const XrViewConfigurationViews &xrViewConfigurationViews)=0
Creates the OpenXR session.
virtual void onActivityResume()
Event function call when the Android Activity is resumed.
StringSet grantedAndroidPermissionSet_
The set holding the names of granted Android permissions.
Definition NativeApplication.h:345
virtual void onActivityTermWindow()
Event function call when the window of the Android Activity is terminated.
virtual StringSet necessaryOpenXRExtensionNames() const
Returns the names of the necessary OpenXR extensions the application needs.
Device::DeviceType deviceType_
The device type that this application is configured for.
Definition NativeApplication.h:339
virtual void onOpenXRSessionReady()
Event function called whenever the session is ready, when the session state changed to XR_SESSION_STA...
Lock androidPermissionLock_
The lock for Android permissions.
Definition NativeApplication.h:348
virtual void onOpenXREvent(const XrEventDataBaseHeader &xrEventDataBaseHeader)
Event function for OpenXR events which are not handled in processOpenXREvents().
virtual bool run()
The run method of this application.
virtual void onActivityPause()
Event function call when the Android Activity is paused.
virtual void onOpenXRSessionChanged(const XrEventDataSessionStateChanged &xrEventDataSessionStateChanged)
Event function called whenever the state of the OpenXR session changed.
void requestAndroidPermissions(std::vector< std::string > &&permissions)
Requests several Android permissions that needs to be granted by the user.
Definition NativeApplication.h:369
static void onAndroidCommand(struct android_app *androidApp, int32_t cmd)
Android's command event function.
XrSessionState xrSessionState_
The current OpenXR session state.
Definition NativeApplication.h:353
NativeApplication(const NativeApplication &nativeApplication)=delete
Disabled copy constructor.
This class wraps an OpenXR instance.
Definition Instance.h:30
This class implements a scoped lock object for recursive lock objects.
Definition Lock.h:135
The namespace covering the entire Ocean framework.
Definition Accessor.h:15