openxr_dpad_binding_extension.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274
  1. /**************************************************************************/
  2. /* openxr_dpad_binding_extension.cpp */
  3. /**************************************************************************/
  4. /* This file is part of: */
  5. /* GODOT ENGINE */
  6. /* https://godotengine.org */
  7. /**************************************************************************/
  8. /* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */
  9. /* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur. */
  10. /* */
  11. /* Permission is hereby granted, free of charge, to any person obtaining */
  12. /* a copy of this software and associated documentation files (the */
  13. /* "Software"), to deal in the Software without restriction, including */
  14. /* without limitation the rights to use, copy, modify, merge, publish, */
  15. /* distribute, sublicense, and/or sell copies of the Software, and to */
  16. /* permit persons to whom the Software is furnished to do so, subject to */
  17. /* the following conditions: */
  18. /* */
  19. /* The above copyright notice and this permission notice shall be */
  20. /* included in all copies or substantial portions of the Software. */
  21. /* */
  22. /* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
  23. /* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
  24. /* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. */
  25. /* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
  26. /* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
  27. /* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
  28. /* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
  29. /**************************************************************************/
  30. #include "openxr_dpad_binding_extension.h"
  31. #include "../openxr_api.h"
  32. #include "core/math/math_funcs.h"
  33. // Implementation for:
  34. // https://registry.khronos.org/OpenXR/specs/1.1/html/xrspec.html#XR_EXT_dpad_binding
  35. ///////////////////////////////////////////////////////////////////////////////////////////////////
  36. // OpenXRDPadBindingExtension
  37. OpenXRDPadBindingExtension *OpenXRDPadBindingExtension::singleton = nullptr;
  38. OpenXRDPadBindingExtension *OpenXRDPadBindingExtension::get_singleton() {
  39. return singleton;
  40. }
  41. OpenXRDPadBindingExtension::OpenXRDPadBindingExtension() {
  42. singleton = this;
  43. }
  44. OpenXRDPadBindingExtension::~OpenXRDPadBindingExtension() {
  45. singleton = nullptr;
  46. }
  47. HashMap<String, bool *> OpenXRDPadBindingExtension::get_requested_extensions() {
  48. HashMap<String, bool *> request_extensions;
  49. // Note, we're dependent on the binding modifier extension, this may be requested by multiple extension wrappers.
  50. request_extensions[XR_KHR_BINDING_MODIFICATION_EXTENSION_NAME] = &binding_modifier_ext;
  51. request_extensions[XR_EXT_DPAD_BINDING_EXTENSION_NAME] = &dpad_binding_ext;
  52. return request_extensions;
  53. }
  54. bool OpenXRDPadBindingExtension::is_available() {
  55. return binding_modifier_ext && dpad_binding_ext;
  56. }
  57. ///////////////////////////////////////////////////////////////////////////////////////////////////
  58. // OpenXRDpadBindingModifier
  59. void OpenXRDpadBindingModifier::_bind_methods() {
  60. ClassDB::bind_method(D_METHOD("set_action_set", "action_set"), &OpenXRDpadBindingModifier::set_action_set);
  61. ClassDB::bind_method(D_METHOD("get_action_set"), &OpenXRDpadBindingModifier::get_action_set);
  62. ADD_PROPERTY(PropertyInfo(Variant::OBJECT, "action_set", PROPERTY_HINT_RESOURCE_TYPE, "OpenXRActionSet"), "set_action_set", "get_action_set");
  63. ClassDB::bind_method(D_METHOD("set_input_path", "input_path"), &OpenXRDpadBindingModifier::set_input_path);
  64. ClassDB::bind_method(D_METHOD("get_input_path"), &OpenXRDpadBindingModifier::get_input_path);
  65. ADD_PROPERTY(PropertyInfo(Variant::STRING, "input_path", PROPERTY_HINT_TYPE_STRING, "binding_path"), "set_input_path", "get_input_path");
  66. ClassDB::bind_method(D_METHOD("set_threshold", "threshold"), &OpenXRDpadBindingModifier::set_threshold);
  67. ClassDB::bind_method(D_METHOD("get_threshold"), &OpenXRDpadBindingModifier::get_threshold);
  68. ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "threshold", PROPERTY_HINT_RANGE, "0.0,1.0,0.01"), "set_threshold", "get_threshold");
  69. ClassDB::bind_method(D_METHOD("set_threshold_released", "threshold_released"), &OpenXRDpadBindingModifier::set_threshold_released);
  70. ClassDB::bind_method(D_METHOD("get_threshold_released"), &OpenXRDpadBindingModifier::get_threshold_released);
  71. ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "threshold_released", PROPERTY_HINT_RANGE, "0.0,1.0,0.01"), "set_threshold_released", "get_threshold_released");
  72. ClassDB::bind_method(D_METHOD("set_center_region", "center_region"), &OpenXRDpadBindingModifier::set_center_region);
  73. ClassDB::bind_method(D_METHOD("get_center_region"), &OpenXRDpadBindingModifier::get_center_region);
  74. ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "center_region", PROPERTY_HINT_RANGE, "0.0,1.0,0.01"), "set_center_region", "get_center_region");
  75. ClassDB::bind_method(D_METHOD("set_wedge_angle", "wedge_angle"), &OpenXRDpadBindingModifier::set_wedge_angle);
  76. ClassDB::bind_method(D_METHOD("get_wedge_angle"), &OpenXRDpadBindingModifier::get_wedge_angle);
  77. ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "wedge_angle", PROPERTY_HINT_RANGE, "1.0,180.0,0.1,radians_as_degrees"), "set_wedge_angle", "get_wedge_angle");
  78. ClassDB::bind_method(D_METHOD("set_is_sticky", "is_sticky"), &OpenXRDpadBindingModifier::set_is_sticky);
  79. ClassDB::bind_method(D_METHOD("get_is_sticky"), &OpenXRDpadBindingModifier::get_is_sticky);
  80. ADD_PROPERTY(PropertyInfo(Variant::BOOL, "is_sticky"), "set_is_sticky", "get_is_sticky");
  81. ClassDB::bind_method(D_METHOD("set_on_haptic", "haptic"), &OpenXRDpadBindingModifier::set_on_haptic);
  82. ClassDB::bind_method(D_METHOD("get_on_haptic"), &OpenXRDpadBindingModifier::get_on_haptic);
  83. ADD_PROPERTY(PropertyInfo(Variant::OBJECT, "on_haptic", PROPERTY_HINT_RESOURCE_TYPE, "OpenXRHapticBase"), "set_on_haptic", "get_on_haptic");
  84. ClassDB::bind_method(D_METHOD("set_off_haptic", "haptic"), &OpenXRDpadBindingModifier::set_off_haptic);
  85. ClassDB::bind_method(D_METHOD("get_off_haptic"), &OpenXRDpadBindingModifier::get_off_haptic);
  86. ADD_PROPERTY(PropertyInfo(Variant::OBJECT, "off_haptic", PROPERTY_HINT_RESOURCE_TYPE, "OpenXRHapticBase"), "set_off_haptic", "get_off_haptic");
  87. }
  88. OpenXRDpadBindingModifier::OpenXRDpadBindingModifier() {
  89. ERR_FAIL_COND(dpad_bindings_data.resize_initialized(sizeof(XrInteractionProfileDpadBindingEXT)) != OK);
  90. dpad_bindings = (XrInteractionProfileDpadBindingEXT *)dpad_bindings_data.ptrw();
  91. dpad_bindings->type = XR_TYPE_INTERACTION_PROFILE_DPAD_BINDING_EXT;
  92. dpad_bindings->next = nullptr;
  93. dpad_bindings->forceThreshold = 0.6;
  94. dpad_bindings->forceThresholdReleased = 0.4;
  95. dpad_bindings->centerRegion = 0.1;
  96. dpad_bindings->wedgeAngle = Math::deg_to_rad(90.0);
  97. dpad_bindings->isSticky = false;
  98. }
  99. void OpenXRDpadBindingModifier::set_action_set(const Ref<OpenXRActionSet> p_action_set) {
  100. action_set = p_action_set;
  101. }
  102. Ref<OpenXRActionSet> OpenXRDpadBindingModifier::get_action_set() const {
  103. return action_set;
  104. }
  105. void OpenXRDpadBindingModifier::set_input_path(const String &p_input_path) {
  106. input_path = p_input_path;
  107. emit_changed();
  108. }
  109. String OpenXRDpadBindingModifier::get_input_path() const {
  110. return input_path;
  111. }
  112. void OpenXRDpadBindingModifier::set_threshold(float p_threshold) {
  113. ERR_FAIL_NULL(dpad_bindings);
  114. ERR_FAIL_COND(p_threshold < 0.0 || p_threshold > 1.0);
  115. dpad_bindings->forceThreshold = p_threshold;
  116. emit_changed();
  117. }
  118. float OpenXRDpadBindingModifier::get_threshold() const {
  119. ERR_FAIL_NULL_V(dpad_bindings, 0.0);
  120. return dpad_bindings->forceThreshold;
  121. }
  122. void OpenXRDpadBindingModifier::set_threshold_released(float p_threshold) {
  123. ERR_FAIL_NULL(dpad_bindings);
  124. ERR_FAIL_COND(p_threshold < 0.0 || p_threshold > 1.0);
  125. dpad_bindings->forceThresholdReleased = p_threshold;
  126. emit_changed();
  127. }
  128. float OpenXRDpadBindingModifier::get_threshold_released() const {
  129. ERR_FAIL_NULL_V(dpad_bindings, 0.0);
  130. return dpad_bindings->forceThresholdReleased;
  131. }
  132. void OpenXRDpadBindingModifier::set_center_region(float p_center_region) {
  133. ERR_FAIL_NULL(dpad_bindings);
  134. ERR_FAIL_COND(p_center_region < 0.0 || p_center_region > 1.0);
  135. dpad_bindings->centerRegion = p_center_region;
  136. emit_changed();
  137. }
  138. float OpenXRDpadBindingModifier::get_center_region() const {
  139. ERR_FAIL_NULL_V(dpad_bindings, 0.0);
  140. return dpad_bindings->centerRegion;
  141. }
  142. void OpenXRDpadBindingModifier::set_wedge_angle(float p_wedge_angle) {
  143. ERR_FAIL_NULL(dpad_bindings);
  144. dpad_bindings->wedgeAngle = p_wedge_angle;
  145. emit_changed();
  146. }
  147. float OpenXRDpadBindingModifier::get_wedge_angle() const {
  148. ERR_FAIL_NULL_V(dpad_bindings, 0.0);
  149. return dpad_bindings->wedgeAngle;
  150. }
  151. void OpenXRDpadBindingModifier::set_wedge_angle_deg(float p_wedge_angle) {
  152. ERR_FAIL_NULL(dpad_bindings);
  153. dpad_bindings->wedgeAngle = Math::deg_to_rad(p_wedge_angle);
  154. emit_changed();
  155. }
  156. float OpenXRDpadBindingModifier::get_wedge_angle_deg() const {
  157. ERR_FAIL_NULL_V(dpad_bindings, 0.0);
  158. return Math::rad_to_deg(dpad_bindings->wedgeAngle);
  159. }
  160. void OpenXRDpadBindingModifier::set_is_sticky(bool p_sticky) {
  161. ERR_FAIL_NULL(dpad_bindings);
  162. dpad_bindings->isSticky = p_sticky;
  163. emit_changed();
  164. }
  165. bool OpenXRDpadBindingModifier::get_is_sticky() const {
  166. ERR_FAIL_NULL_V(dpad_bindings, false);
  167. return dpad_bindings->isSticky;
  168. }
  169. void OpenXRDpadBindingModifier::set_on_haptic(const Ref<OpenXRHapticBase> &p_haptic) {
  170. on_haptic = p_haptic;
  171. emit_changed();
  172. }
  173. Ref<OpenXRHapticBase> OpenXRDpadBindingModifier::get_on_haptic() const {
  174. return on_haptic;
  175. }
  176. void OpenXRDpadBindingModifier::set_off_haptic(const Ref<OpenXRHapticBase> &p_haptic) {
  177. off_haptic = p_haptic;
  178. emit_changed();
  179. }
  180. Ref<OpenXRHapticBase> OpenXRDpadBindingModifier::get_off_haptic() const {
  181. return off_haptic;
  182. }
  183. PackedByteArray OpenXRDpadBindingModifier::get_ip_modification() {
  184. ERR_FAIL_NULL_V(dpad_bindings, PackedByteArray());
  185. OpenXRAPI *openxr_api = OpenXRAPI::get_singleton();
  186. ERR_FAIL_NULL_V(openxr_api, PackedByteArray());
  187. OpenXRDPadBindingExtension *dpad_binding_ext = OpenXRDPadBindingExtension::get_singleton();
  188. if (!dpad_binding_ext || !dpad_binding_ext->is_available()) {
  189. // Extension not enabled!
  190. WARN_PRINT("DPad binding extension is not enabled or available.");
  191. return PackedByteArray();
  192. }
  193. dpad_bindings->binding = openxr_api->get_xr_path(input_path);
  194. ERR_FAIL_COND_V(dpad_bindings->binding == XR_NULL_PATH, PackedByteArray());
  195. // Get our action set
  196. ERR_FAIL_COND_V(action_set.is_null(), PackedByteArray());
  197. RID action_set_rid = openxr_api->find_action_set(action_set->get_name());
  198. ERR_FAIL_COND_V(!action_set_rid.is_valid(), PackedByteArray());
  199. dpad_bindings->actionSet = openxr_api->action_set_get_handle(action_set_rid);
  200. // These are set already:
  201. // - forceThreshold
  202. // - forceThresholdReleased
  203. // - centerRegion
  204. // - wedgeAngle
  205. // - isSticky
  206. if (on_haptic.is_valid()) {
  207. dpad_bindings->onHaptic = on_haptic->get_xr_structure();
  208. } else {
  209. dpad_bindings->onHaptic = nullptr;
  210. }
  211. if (off_haptic.is_valid()) {
  212. dpad_bindings->offHaptic = off_haptic->get_xr_structure();
  213. } else {
  214. dpad_bindings->offHaptic = nullptr;
  215. }
  216. return dpad_bindings_data;
  217. }