OpenXrInputPlugin: User-friendly input actions

Replaces the raw controller button inputs with named OpenXR actions.
There's a lot of engine components that expect raw controller inputs
like the VR teleport script. Those will have to be refactored later,
but for now this works well enough and is perfectly usable. A small
issue I've hit is the LY input working for the teleport script,
but not for smooth locomotion. I've hacked around this by having
the "walk" action bound both to LX/LY and to the Translate actions.
It's a bit janky for teleports, but it's functional.

TODO: Feedback on intuitive bindings for other controller types besides
just the HTC Vive controllers.
This commit is contained in:
Ada 2024-12-12 06:15:27 +10:00 committed by Ada
parent dc3a508051
commit cedc5be526
7 changed files with 374 additions and 260 deletions

View file

@ -0,0 +1,29 @@
{
"name": "OpenXR Actions",
"channels": [
{ "from": "OpenXR.LeftHand", "to": "Standard.LeftHand" },
{ "from": "OpenXR.RightHand", "to": "Standard.RightHand" },
{ "from": "OpenXR.Head", "to" : "Standard.Head", "when" : [ "Application.InHMD"] },
{ "from": "OpenXR.LeftInteract", "to": "Standard.LT", "filters": [{"type": "deadZone", "min": 0.05}]},
{ "from": "OpenXR.RightInteract", "to": "Standard.RT", "filters": [{"type": "deadZone", "min": 0.05}]},
{ "from": "OpenXR.LeftInteractClick", "to": "Standard.LTClick" },
{ "from": "OpenXR.RightInteractClick", "to": "Standard.RTClick" },
{ "from": "OpenXR.LeftGrip", "to": "Standard.LeftGrip" },
{ "from": "OpenXR.RightGrip", "to": "Standard.RightGrip" },
{ "from": "OpenXR.WalkX", "to": "Actions.TranslateX" },
{ "from": "OpenXR.WalkY", "to": "Actions.TranslateZ" },
{ "from": "OpenXR.WalkX", "peek": true, "to": "Standard.LX" },
{ "from": "OpenXR.WalkY", "peek": true, "to": "Standard.LY" },
{ "from": "OpenXR.Turn", "to": "Standard.RX"},
{ "from": "OpenXR.Teleport", "to": "Standard.RY" },
{ "from": "OpenXR.CycleCamera", "to": "Actions.CycleCamera" },
{ "from": "OpenXR.Sprint", "to": "Actions.Sprint" },
{ "from": "OpenXR.ToggleTablet", "to": "Standard.LeftSecondaryThumb" },
{ "from": "OpenXR.Jump", "to": "Standard.RightSecondaryThumb" }
]
}

View file

@ -2373,12 +2373,6 @@ void Application::update(float deltaTime) {
AnimDebugDraw::getInstance().update(); AnimDebugDraw::getInstance().update();
} }
// a hack to prevent the engine from trying
// to pump out hundreds and hundreds of simulation
// ticks per second that can't be displayed
std::this_thread::sleep_for(5ms);
{ // Game loop is done, mark the end of the frame for the scene transactions and the render loop to take over { // Game loop is done, mark the end of the frame for the scene transactions and the render loop to take over
PerformanceTimer perfTimer("enqueueFrame"); PerformanceTimer perfTimer("enqueueFrame");
getMain3DScene()->enqueueFrame(); getMain3DScene()->enqueueFrame();
@ -2393,6 +2387,9 @@ void Application::update(float deltaTime) {
if (getActiveDisplayPlugin()->isHmd()) { if (getActiveDisplayPlugin()->isHmd()) {
PerformanceTimer perfTimer("squeezeVision"); PerformanceTimer perfTimer("squeezeVision");
_visionSqueeze.updateVisionSqueeze(myAvatar->getSensorToWorldMatrix(), deltaTime); _visionSqueeze.updateVisionSqueeze(myAvatar->getSensorToWorldMatrix(), deltaTime);
// FIXME HACK: OpenXR doesn't limit the game rate for some reason and wastes cpu time
std::this_thread::sleep_for(5ms);
} }
} }

View file

@ -469,6 +469,7 @@ bool OpenVrDisplayPlugin::internalActivate() {
vr::VRCompositor()->ForceInterleavedReprojectionOn(true); vr::VRCompositor()->ForceInterleavedReprojectionOn(true);
} }
#if 0
// set up default sensor space such that the UI overlay will align with the front of the room. // set up default sensor space such that the UI overlay will align with the front of the room.
auto chaperone = vr::VRChaperone(); auto chaperone = vr::VRChaperone();
if (chaperone) { if (chaperone) {
@ -485,6 +486,7 @@ bool OpenVrDisplayPlugin::internalActivate() {
qDebug() << "OpenVR: error could not get chaperone pointer"; qDebug() << "OpenVR: error could not get chaperone pointer";
#endif #endif
} }
#endif
if (_threadedSubmit) { if (_threadedSubmit) {
_submitThread = std::make_shared<OpenVrSubmitThread>(*this); _submitThread = std::make_shared<OpenVrSubmitThread>(*this);
@ -775,6 +777,9 @@ QString OpenVrDisplayPlugin::getPreferredAudioOutDevice() const {
} }
QRectF OpenVrDisplayPlugin::getPlayAreaRect() { QRectF OpenVrDisplayPlugin::getPlayAreaRect() {
#if 1
return QRectF();
#else
auto chaperone = vr::VRChaperone(); auto chaperone = vr::VRChaperone();
if (!chaperone) { if (!chaperone) {
qWarning() << "No chaperone"; qWarning() << "No chaperone";
@ -806,6 +811,7 @@ QRectF OpenVrDisplayPlugin::getPlayAreaRect() {
glm::vec2 dimensions = glm::vec2(maxXZ.x - minXZ.x, maxXZ.z - minXZ.z); glm::vec2 dimensions = glm::vec2(maxXZ.x - minXZ.x, maxXZ.z - minXZ.z);
return QRectF(center.x, center.y, dimensions.x, dimensions.y); return QRectF(center.x, center.y, dimensions.x, dimensions.y);
#endif
} }
DisplayPlugin::StencilMaskMeshOperator OpenVrDisplayPlugin::getStencilMaskMeshOperator() { DisplayPlugin::StencilMaskMeshOperator OpenVrDisplayPlugin::getStencilMaskMeshOperator() {

View file

@ -82,6 +82,7 @@ bool OpenXrContext::initInstance() {
return false; return false;
bool openglSupported = false; bool openglSupported = false;
bool bindingModificationSupported = false;
qCInfo(xr_context_cat, "Runtime supports %d extensions:", count); qCInfo(xr_context_cat, "Runtime supports %d extensions:", count);
for (uint32_t i = 0; i < count; i++) { for (uint32_t i = 0; i < count; i++) {
@ -89,6 +90,18 @@ bool OpenXrContext::initInstance() {
if (strcmp(XR_KHR_OPENGL_ENABLE_EXTENSION_NAME, properties[i].extensionName) == 0) { if (strcmp(XR_KHR_OPENGL_ENABLE_EXTENSION_NAME, properties[i].extensionName) == 0) {
openglSupported = true; openglSupported = true;
} }
if (strcmp(XR_KHR_BINDING_MODIFICATION_EXTENSION_NAME, properties[i].extensionName) == 0) {
bindingModificationSupported = true;
}
if (strcmp(XR_EXT_DPAD_BINDING_EXTENSION_NAME, properties[i].extensionName) == 0) {
_dpadBindingSupported = true;
}
if (strcmp(XR_EXT_PALM_POSE_EXTENSION_NAME, properties[i].extensionName) == 0) {
_palmPoseSupported = true;
}
} }
if (!openglSupported) { if (!openglSupported) {
@ -96,14 +109,21 @@ bool OpenXrContext::initInstance() {
return false; return false;
} }
std::vector<const char*> enabled = { XR_KHR_OPENGL_ENABLE_EXTENSION_NAME }; std::vector<const char*> enabled = {XR_KHR_OPENGL_ENABLE_EXTENSION_NAME};
if (bindingModificationSupported && _dpadBindingSupported) {
enabled.emplace(enabled.end(), XR_KHR_BINDING_MODIFICATION_EXTENSION_NAME);
enabled.emplace(enabled.end(), XR_EXT_DPAD_BINDING_EXTENSION_NAME);
}
if (_palmPoseSupported) {
enabled.emplace(enabled.end(), XR_EXT_PALM_POSE_EXTENSION_NAME);
}
XrInstanceCreateInfo info = { XrInstanceCreateInfo info = {
.type = XR_TYPE_INSTANCE_CREATE_INFO, .type = XR_TYPE_INSTANCE_CREATE_INFO,
.applicationInfo = { .applicationInfo = {
.applicationName = "overte", .applicationName = "Overte",
.applicationVersion = 1, .applicationVersion = 1,
.engineName = "overte", .engineName = "Overte",
.engineVersion = 0, .engineVersion = 0,
.apiVersion = XR_CURRENT_API_VERSION, .apiVersion = XR_CURRENT_API_VERSION,
}, },
@ -126,6 +146,8 @@ bool OpenXrContext::initInstance() {
xrStringToPath(_instance, "/user/hand/left", &_handPaths[0]); xrStringToPath(_instance, "/user/hand/left", &_handPaths[0]);
xrStringToPath(_instance, "/user/hand/right", &_handPaths[1]); xrStringToPath(_instance, "/user/hand/right", &_handPaths[1]);
xrStringToPath(_instance, "/interaction_profiles/htc/vive_controller", &_viveControllerPath);
return true; return true;
} }
@ -342,6 +364,11 @@ bool OpenXrContext::pollEvents() {
if (!xrCheck(_instance, res, "Failed to get interaction profile")) if (!xrCheck(_instance, res, "Failed to get interaction profile"))
continue; continue;
_dpadNeedsClick = false;
if (state.interactionProfile == _viveControllerPath) {
_dpadNeedsClick = true;
}
uint32_t bufferCountOutput; uint32_t bufferCountOutput;
char profilePath[XR_MAX_PATH_LENGTH]; char profilePath[XR_MAX_PATH_LENGTH];
res = xrPathToString(_instance, state.interactionProfile, XR_MAX_PATH_LENGTH, &bufferCountOutput, res = xrPathToString(_instance, state.interactionProfile, XR_MAX_PATH_LENGTH, &bufferCountOutput,

View file

@ -69,9 +69,15 @@ public:
QString _systemName; QString _systemName;
bool _isSessionRunning = false; bool _isSessionRunning = false;
bool _dpadBindingSupported = false;
bool _palmPoseSupported = false;
bool _dpadNeedsClick = false;
private: private:
XrSessionState _lastSessionState = XR_SESSION_STATE_UNKNOWN; XrSessionState _lastSessionState = XR_SESSION_STATE_UNKNOWN;
XrPath _viveControllerPath = XR_NULL_PATH;
public: public:
OpenXrContext(); OpenXrContext();
~OpenXrContext(); ~OpenXrContext();
@ -101,4 +107,4 @@ inline static glm::quat xrQuatToGlm(const XrQuaternionf& q) {
return glm::quat(q.w, q.x, q.y, q.z); return glm::quat(q.w, q.x, q.y, q.z);
} }
bool xrCheck(XrInstance instance, XrResult result, const char* message); bool xrCheck(XrInstance instance, XrResult result, const char* message);

View file

@ -102,7 +102,7 @@ void OpenXrInputPlugin::loadSettings() {
void OpenXrInputPlugin::saveSettings() const { void OpenXrInputPlugin::saveSettings() const {
} }
OpenXrInputPlugin::InputDevice::InputDevice(std::shared_ptr<OpenXrContext> c) : controller::InputDevice("Vive") { OpenXrInputPlugin::InputDevice::InputDevice(std::shared_ptr<OpenXrContext> c) : controller::InputDevice("OpenXR") {
_context = c; _context = c;
} }
@ -126,8 +126,10 @@ bool OpenXrInputPlugin::InputDevice::triggerHapticPulse(float strength, float du
nanoseconds durationNs = duration_cast<nanoseconds>(milliseconds(static_cast<int>(duration * 10.0f))); nanoseconds durationNs = duration_cast<nanoseconds>(milliseconds(static_cast<int>(duration * 10.0f)));
XrDuration xrDuration = durationNs.count(); XrDuration xrDuration = durationNs.count();
if (!_actions.at("/output/haptic")->applyHaptic(index, xrDuration, XR_FREQUENCY_UNSPECIFIED, 0.5f * strength)) { auto path = (index == 0) ? "hand_haptic_left" : "hand_haptic_right";
qCCritical(xr_input_cat, "Failed to apply haptic feedback!");
if (!_actions.at(path)->applyHaptic(xrDuration, XR_FREQUENCY_UNSPECIFIED, 0.5f * strength)) {
qCCritical(xr_input_cat) << "Failed to apply haptic feedback!";
} }
return true; return true;
@ -142,12 +144,8 @@ bool OpenXrInputPlugin::Action::init(XrActionSet actionSet) {
.subactionPaths = _context->_handPaths, .subactionPaths = _context->_handPaths,
}; };
QString name = QString::fromStdString(_path); strncpy(info.actionName, _id.c_str(), XR_MAX_ACTION_NAME_SIZE - 1);
name.replace("/input/", ""); strncpy(info.localizedActionName, _friendlyName.c_str(), XR_MAX_LOCALIZED_ACTION_NAME_SIZE - 1);
name.replace("/", "-");
strcpy(info.actionName, name.toUtf8().data());
name.replace("-", " ");
strcpy(info.localizedActionName, name.toUtf8().data());
XrResult result = xrCreateAction(actionSet, &info, &_action); XrResult result = xrCreateAction(actionSet, &info, &_action);
if (!xrCheck(instance, result, "Failed to create action")) if (!xrCheck(instance, result, "Failed to create action"))
@ -163,23 +161,20 @@ bool OpenXrInputPlugin::Action::init(XrActionSet actionSet) {
return true; return true;
} }
const std::vector<std::string> HAND_PATHS = { "left", "right" };
std::vector<XrActionSuggestedBinding> OpenXrInputPlugin::Action::getBindings() { std::vector<XrActionSuggestedBinding> OpenXrInputPlugin::Action::getBindings() {
assert(_action != XR_NULL_HANDLE); assert(_action != XR_NULL_HANDLE);
std::vector<XrActionSuggestedBinding> bindings; std::vector<XrActionSuggestedBinding> bindings;
for (uint32_t i = 0; i < HAND_COUNT; i++) { for (uint32_t i = 0; i < HAND_COUNT; i++) {
XrPath path; XrPath path;
std::string pathString = "/user/hand/" + HAND_PATHS[i] + _path; xrStringToPath(_context->_instance, _id.c_str(), &path);
xrStringToPath(_context->_instance, pathString.c_str(), &path);
XrActionSuggestedBinding binding = { .action = _action, .binding = path }; XrActionSuggestedBinding binding = { .action = _action, .binding = path };
bindings.push_back(binding); bindings.push_back(binding);
} }
return bindings; return bindings;
} }
XrActionStateFloat OpenXrInputPlugin::Action::getFloat(uint32_t handId) { XrActionStateFloat OpenXrInputPlugin::Action::getFloat() {
XrActionStateFloat state = { XrActionStateFloat state = {
.type = XR_TYPE_ACTION_STATE_FLOAT, .type = XR_TYPE_ACTION_STATE_FLOAT,
}; };
@ -187,7 +182,6 @@ XrActionStateFloat OpenXrInputPlugin::Action::getFloat(uint32_t handId) {
XrActionStateGetInfo info = { XrActionStateGetInfo info = {
.type = XR_TYPE_ACTION_STATE_GET_INFO, .type = XR_TYPE_ACTION_STATE_GET_INFO,
.action = _action, .action = _action,
.subactionPath = _context->_handPaths[handId],
}; };
XrResult result = xrGetActionStateFloat(_context->_session, &info, &state); XrResult result = xrGetActionStateFloat(_context->_session, &info, &state);
@ -196,7 +190,23 @@ XrActionStateFloat OpenXrInputPlugin::Action::getFloat(uint32_t handId) {
return state; return state;
} }
XrActionStateBoolean OpenXrInputPlugin::Action::getBool(uint32_t handId) { XrActionStateVector2f OpenXrInputPlugin::Action::getVector2f() {
XrActionStateVector2f state = {
.type = XR_TYPE_ACTION_STATE_VECTOR2F,
};
XrActionStateGetInfo info = {
.type = XR_TYPE_ACTION_STATE_GET_INFO,
.action = _action,
};
XrResult result = xrGetActionStateVector2f(_context->_session, &info, &state);
xrCheck(_context->_instance, result, "Failed to get vector2 state!");
return state;
}
XrActionStateBoolean OpenXrInputPlugin::Action::getBool() {
XrActionStateBoolean state = { XrActionStateBoolean state = {
.type = XR_TYPE_ACTION_STATE_BOOLEAN, .type = XR_TYPE_ACTION_STATE_BOOLEAN,
}; };
@ -204,7 +214,6 @@ XrActionStateBoolean OpenXrInputPlugin::Action::getBool(uint32_t handId) {
XrActionStateGetInfo info = { XrActionStateGetInfo info = {
.type = XR_TYPE_ACTION_STATE_GET_INFO, .type = XR_TYPE_ACTION_STATE_GET_INFO,
.action = _action, .action = _action,
.subactionPath = _context->_handPaths[handId],
}; };
XrResult result = xrGetActionStateBoolean(_context->_session, &info, &state); XrResult result = xrGetActionStateBoolean(_context->_session, &info, &state);
@ -213,14 +222,13 @@ XrActionStateBoolean OpenXrInputPlugin::Action::getBool(uint32_t handId) {
return state; return state;
} }
XrSpaceLocation OpenXrInputPlugin::Action::getPose(uint32_t handId) { XrSpaceLocation OpenXrInputPlugin::Action::getPose() {
XrActionStatePose state = { XrActionStatePose state = {
.type = XR_TYPE_ACTION_STATE_POSE, .type = XR_TYPE_ACTION_STATE_POSE,
}; };
XrActionStateGetInfo info = { XrActionStateGetInfo info = {
.type = XR_TYPE_ACTION_STATE_GET_INFO, .type = XR_TYPE_ACTION_STATE_GET_INFO,
.action = _action, .action = _action,
.subactionPath = _context->_handPaths[handId],
}; };
XrResult result = xrGetActionStatePose(_context->_session, &info, &state); XrResult result = xrGetActionStatePose(_context->_session, &info, &state);
@ -231,14 +239,14 @@ XrSpaceLocation OpenXrInputPlugin::Action::getPose(uint32_t handId) {
}; };
if (_context->_lastPredictedDisplayTime.has_value()) { if (_context->_lastPredictedDisplayTime.has_value()) {
result = xrLocateSpace(_poseSpaces[handId], _context->_stageSpace, _context->_lastPredictedDisplayTime.value(), &location); result = xrLocateSpace(_poseSpace, _context->_stageSpace, _context->_lastPredictedDisplayTime.value(), &location);
xrCheck(_context->_instance, result, "Failed to locate hand space!"); xrCheck(_context->_instance, result, "Failed to locate hand space!");
} }
return location; return location;
} }
bool OpenXrInputPlugin::Action::applyHaptic(uint32_t handId, XrDuration duration, float frequency, float amplitude) { bool OpenXrInputPlugin::Action::applyHaptic(XrDuration duration, float frequency, float amplitude) {
XrHapticVibration vibration = { XrHapticVibration vibration = {
.type = XR_TYPE_HAPTIC_VIBRATION, .type = XR_TYPE_HAPTIC_VIBRATION,
.duration = duration, .duration = duration,
@ -249,7 +257,6 @@ bool OpenXrInputPlugin::Action::applyHaptic(uint32_t handId, XrDuration duration
XrHapticActionInfo haptic_action_info = { XrHapticActionInfo haptic_action_info = {
.type = XR_TYPE_HAPTIC_ACTION_INFO, .type = XR_TYPE_HAPTIC_ACTION_INFO,
.action = _action, .action = _action,
.subactionPath = _context->_handPaths[handId],
}; };
XrResult result = xrApplyHapticFeedback(_context->_session, &haptic_action_info, (const XrHapticBaseHeader*)&vibration); XrResult result = xrApplyHapticFeedback(_context->_session, &haptic_action_info, (const XrHapticBaseHeader*)&vibration);
@ -259,44 +266,40 @@ bool OpenXrInputPlugin::Action::applyHaptic(uint32_t handId, XrDuration duration
bool OpenXrInputPlugin::Action::createPoseSpaces() { bool OpenXrInputPlugin::Action::createPoseSpaces() {
assert(_action != XR_NULL_HANDLE); assert(_action != XR_NULL_HANDLE);
for (int hand = 0; hand < HAND_COUNT; hand++) { XrActionSpaceCreateInfo info = {
XrActionSpaceCreateInfo info = { .type = XR_TYPE_ACTION_SPACE_CREATE_INFO,
.type = XR_TYPE_ACTION_SPACE_CREATE_INFO, .action = _action,
.action = _action, .poseInActionSpace = XR_INDENTITY_POSE,
.subactionPath = _context->_handPaths[hand], };
.poseInActionSpace = XR_INDENTITY_POSE,
};
XrResult result = xrCreateActionSpace(_context->_session, &info, &_poseSpaces[hand]); XrResult result = xrCreateActionSpace(_context->_session, &info, &_poseSpace);
if (!xrCheck(_context->_instance, result, "Failed to create hand pose space")) if (!xrCheck(_context->_instance, result, "Failed to create hand pose space"))
return false; return false;
}
return true; return true;
} }
bool OpenXrInputPlugin::InputDevice::initBindings(const std::string& profileName, bool OpenXrInputPlugin::InputDevice::initBindings(const std::string& profileName,
const std::vector<std::string>& actionsToBind) { const std::map<std::string, std::string>& actionsToBind) {
XrPath profilePath; XrPath profilePath;
XrResult result = xrStringToPath(_context->_instance, profileName.c_str(), &profilePath); XrResult result = xrStringToPath(_context->_instance, profileName.c_str(), &profilePath);
if (!xrCheck(_context->_instance, result, "Failed to get interaction profile")) if (!xrCheck(_context->_instance, result, "Failed to get interaction profile"))
return false; return false;
std::vector<XrActionSuggestedBinding> bindings; std::vector<XrActionSuggestedBinding> suggestions;
for (const std::string& path : actionsToBind) { for (const auto& [actionName, inputPathRaw] : actionsToBind) {
if (!_actions.contains(path)) { XrActionSuggestedBinding bind = {
qCWarning(xr_input_cat, "%s has unbound input %s", profileName.c_str(), path.c_str()); .action = _actions[actionName]->_action,
continue; };
} xrStringToPath(_context->_instance, inputPathRaw.c_str(), &bind.binding);
std::vector<XrActionSuggestedBinding> actionBindings = _actions.at(path)->getBindings(); suggestions.emplace(suggestions.end(), bind);
bindings.insert(std::end(bindings), std::begin(actionBindings), std::end(actionBindings));
} }
const XrInteractionProfileSuggestedBinding suggestedBinding = { const XrInteractionProfileSuggestedBinding suggestedBinding = {
.type = XR_TYPE_INTERACTION_PROFILE_SUGGESTED_BINDING, .type = XR_TYPE_INTERACTION_PROFILE_SUGGESTED_BINDING,
.interactionProfile = profilePath, .interactionProfile = profilePath,
.countSuggestedBindings = (uint32_t)bindings.size(), .countSuggestedBindings = (uint32_t)suggestions.size(),
.suggestedBindings = bindings.data(), .suggestedBindings = suggestions.data(),
}; };
result = xrSuggestInteractionProfileBindings(_context->_instance, &suggestedBinding); result = xrSuggestInteractionProfileBindings(_context->_instance, &suggestedBinding);
@ -307,60 +310,41 @@ bool OpenXrInputPlugin::InputDevice::initBindings(const std::string& profileName
controller::Input::NamedVector OpenXrInputPlugin::InputDevice::getAvailableInputs() const { controller::Input::NamedVector OpenXrInputPlugin::InputDevice::getAvailableInputs() const {
using namespace controller; using namespace controller;
// clang-format off
QVector<Input::NamedPair> availableInputs{ QVector<Input::NamedPair> availableInputs{
makePair(HEAD, "Head"), makePair(HEAD, "Head"),
makePair(LEFT_HAND, "LeftHand"), makePair(LEFT_HAND, "LeftHand"),
makePair(RIGHT_HAND, "RightHand"), makePair(RIGHT_HAND, "RightHand"),
// Trackpad analogs // INPUT FIXME: Actions.Translate.{X,Z} work
makePair(LX, "LX"), // perfectly but Standard.LY is unreliable
makePair(LY, "LY"), makePair(LX, "WalkX"),
makePair(RX, "RX"), makePair(LY, "WalkY"),
makePair(RY, "RY"),
// capacitive touch on the touch pad makePair(LT, "LeftInteract"),
makePair(LS_TOUCH, "LSTouch"), makePair(RT, "RightInteract"),
makePair(RS_TOUCH, "RSTouch"), makePair(LT_CLICK, "LeftInteractClick"),
makePair(RT_CLICK, "RightInteractClick"),
// touch pad press
makePair(LS, "LS"),
makePair(RS, "RS"),
// Differentiate where we are in the touch pad click
makePair(LS_CENTER, "LSCenter"),
makePair(LS_X, "LSX"),
makePair(LS_Y, "LSY"),
makePair(RS_CENTER, "RSCenter"),
makePair(RS_X, "RSX"),
makePair(RS_Y, "RSY"),
// triggers
makePair(LT, "LT"),
makePair(RT, "RT"),
// Trigger clicks
makePair(LT_CLICK, "LTClick"),
makePair(RT_CLICK, "RTClick"),
// low profile side grip button.
makePair(LEFT_GRIP, "LeftGrip"), makePair(LEFT_GRIP, "LeftGrip"),
makePair(RIGHT_GRIP, "RightGrip"), makePair(RIGHT_GRIP, "RightGrip"),
makePair(LEFT_PRIMARY_THUMB, "LeftPrimaryThumb"), // INPUT TODO: horrific hack that breaks depending on handedness
makePair(RIGHT_PRIMARY_THUMB, "RightPrimaryThumb"), // because the input system is in dire need of a refactor,
makePair(LEFT_SECONDARY_THUMB, "LeftSecondaryThumb"), // it was (mostly) designed with raw inputs in mind which makes
makePair(RIGHT_SECONDARY_THUMB, "RightSecondaryThumb"), // it extremely difficult to map onto openxr actions
makePair(LEFT_PRIMARY_THUMB, "ToggleTablet"),
makePair(LEFT_SECONDARY_THUMB, "LeftApplicationMenu"), makePair(RIGHT_PRIMARY_THUMB, "Jump"),
makePair(RIGHT_SECONDARY_THUMB, "RightApplicationMenu"), makePair(DD, "Sprint"),
makePair(RX, "Turn"),
makePair(RY, "Teleport"),
makePair(LS_TOUCH, "CycleCamera"),
}; };
// clang-format on
return availableInputs; return availableInputs;
} }
QString OpenXrInputPlugin::InputDevice::getDefaultMappingConfig() const { QString OpenXrInputPlugin::InputDevice::getDefaultMappingConfig() const {
// FIXME: for some reason vive works but openxr_generic breaks the vertical trackpad? return PathUtils::resourcesPath() + "/controllers/openxr.json";
return PathUtils::resourcesPath() + "/controllers/vive.json";
} }
bool OpenXrInputPlugin::InputDevice::initActions() { bool OpenXrInputPlugin::InputDevice::initActions() {
@ -373,107 +357,96 @@ bool OpenXrInputPlugin::InputDevice::initActions() {
XrActionSetCreateInfo actionSetInfo = { XrActionSetCreateInfo actionSetInfo = {
.type = XR_TYPE_ACTION_SET_CREATE_INFO, .type = XR_TYPE_ACTION_SET_CREATE_INFO,
.actionSetName = "action_set", .actionSetName = "overte",
.localizedActionSetName = "Action Set", .localizedActionSetName = "Overte",
.priority = 0, .priority = 0,
}; };
XrResult result = xrCreateActionSet(instance, &actionSetInfo, &_actionSet); XrResult result = xrCreateActionSet(instance, &actionSetInfo, &_actionSet);
if (!xrCheck(instance, result, "Failed to create action set.")) if (!xrCheck(instance, result, "Failed to create action set."))
return false; return false;
// clang-format off std::map<std::string, std::pair<std::string, XrActionType>> actionTypes = {
std::map<std::string, XrActionType> actionsToInit = { {"tablet", {"Toggle Tablet", XR_ACTION_TYPE_BOOLEAN_INPUT}},
{ "/output/haptic", XR_ACTION_TYPE_VIBRATION_OUTPUT }, {"teleport", {"Teleport", XR_ACTION_TYPE_BOOLEAN_INPUT}},
{ "/input/grip/pose", XR_ACTION_TYPE_POSE_INPUT }, {"cycle_camera", {"Cycle Camera", XR_ACTION_TYPE_BOOLEAN_INPUT}},
// click but pretend it's a float for the trigger actions {"interact_left", {"Left Interact", XR_ACTION_TYPE_FLOAT_INPUT}},
{ "/input/select/click", XR_ACTION_TYPE_FLOAT_INPUT }, {"interact_right", {"Right Interact", XR_ACTION_TYPE_FLOAT_INPUT}},
{ "/input/menu/click", XR_ACTION_TYPE_BOOLEAN_INPUT },
{ "/input/system/click", XR_ACTION_TYPE_BOOLEAN_INPUT },
{ "/input/trackpad/x", XR_ACTION_TYPE_FLOAT_INPUT }, {"grip_left", {"Left Grip", XR_ACTION_TYPE_FLOAT_INPUT}},
{ "/input/trackpad/y", XR_ACTION_TYPE_FLOAT_INPUT }, {"grip_right", {"Right Grip", XR_ACTION_TYPE_FLOAT_INPUT}},
{ "/input/trackpad/touch", XR_ACTION_TYPE_BOOLEAN_INPUT },
{ "/input/trackpad/click", XR_ACTION_TYPE_BOOLEAN_INPUT },
{ "/input/thumbstick/x", XR_ACTION_TYPE_FLOAT_INPUT }, {"turn_left", {"Turn Left", XR_ACTION_TYPE_BOOLEAN_INPUT}},
{ "/input/thumbstick/y", XR_ACTION_TYPE_FLOAT_INPUT }, {"turn_right", {"Turn Right", XR_ACTION_TYPE_BOOLEAN_INPUT}},
{ "/input/thumbstick/touch", XR_ACTION_TYPE_BOOLEAN_INPUT }, {"walk", {"Walk", XR_ACTION_TYPE_VECTOR2F_INPUT}},
{ "/input/thumbstick/click", XR_ACTION_TYPE_BOOLEAN_INPUT }, {"sprint", {"Sprint", XR_ACTION_TYPE_BOOLEAN_INPUT}},
{ "/input/a/click", XR_ACTION_TYPE_BOOLEAN_INPUT }, {"jump", {"Jump", XR_ACTION_TYPE_BOOLEAN_INPUT}},
{ "/input/b/click", XR_ACTION_TYPE_BOOLEAN_INPUT },
{ "/input/a/touch", XR_ACTION_TYPE_BOOLEAN_INPUT },
{ "/input/b/touch", XR_ACTION_TYPE_BOOLEAN_INPUT },
{ "/input/squeeze/click", XR_ACTION_TYPE_FLOAT_INPUT }, // in case the runtime doesn't support dpad emulation
{ "/input/trigger/value", XR_ACTION_TYPE_FLOAT_INPUT }, {"stick_left", {"Left Stick (Fallback)", XR_ACTION_TYPE_VECTOR2F_INPUT}},
{ "/input/trigger/click", XR_ACTION_TYPE_BOOLEAN_INPUT }, {"stick_right", {"Right Stick (Fallback)", XR_ACTION_TYPE_VECTOR2F_INPUT}},
{ "/input/trigger/touch", XR_ACTION_TYPE_BOOLEAN_INPUT }, {"stick_click_left", {"Left Stick Click (Fallback)", XR_ACTION_TYPE_BOOLEAN_INPUT}},
{"stick_click_right", {"Right Stick Click (Fallback)", XR_ACTION_TYPE_BOOLEAN_INPUT}},
{"hand_pose_left", {"Left Hand Pose", XR_ACTION_TYPE_POSE_INPUT}},
{"hand_pose_right", {"Right Hand Pose", XR_ACTION_TYPE_POSE_INPUT}},
{"hand_haptic_left", {"Left Hand Haptic", XR_ACTION_TYPE_VIBRATION_OUTPUT}},
{"hand_haptic_right", {"Right Hand Haptic", XR_ACTION_TYPE_VIBRATION_OUTPUT}},
}; };
// clang-format on
for (const auto& [path, type] : actionsToInit) { // palm pose is nice but monado doesn't support it yet
std::shared_ptr<Action> action = std::make_shared<Action>(_context, type, path); auto hand_pose_name = (_context->_palmPoseSupported) ? "/palm_ext/pose" : "/grip/pose";
// TODO: set up the openxr dpad bindings modifier (looks complicated)
std::map<std::string, std::map<std::string, std::string>> actionSuggestions = {
{"/interaction_profiles/khr/simple_controller", {
{"tablet", "/user/hand/left/input/menu/click"},
{"teleport", "/user/hand/right/input/menu/click"},
{"interact_left", "/user/hand/left/input/select/click"},
{"interact_right", "/user/hand/right/input/select/click"},
{"hand_pose_left", std::string("/user/hand/left/input") + hand_pose_name},
{"hand_pose_right", std::string("/user/hand/right/input") + hand_pose_name},
{"hand_haptic_left", "/user/hand/left/output/haptic"},
{"hand_haptic_right", "/user/hand/right/output/haptic"},
}},
{"/interaction_profiles/htc/vive_controller", {
{"tablet", "/user/hand/left/input/menu/click"},
//{"teleport", "/user/hand/right/input/trackpad/dpad_up"},
//{"cycle_camera", "/user/hand/right/input/trackpad/dpad_down"},
{"interact_left", "/user/hand/left/input/trigger/value"},
{"interact_right", "/user/hand/right/input/trigger/value"},
{"grip_left", "/user/hand/left/input/squeeze/click"},
{"grip_right", "/user/hand/right/input/squeeze/click"},
{"jump", "/user/hand/right/input/menu/click"},
{"walk", "/user/hand/left/input/trackpad"},
//{"turn_left", "/user/hand/right/input/trackpad/dpad_left"},
//{"turn_right", "/user/hand/right/input/trackpad/dpad_right"},
{"stick_left", "/user/hand/left/input/trackpad"},
{"stick_right", "/user/hand/right/input/trackpad"},
{"stick_click_left", "/user/hand/left/input/trackpad/click"},
{"stick_click_right", "/user/hand/right/input/trackpad/click"},
{"hand_pose_left", std::string("/user/hand/left/input") + hand_pose_name},
{"hand_pose_right", std::string("/user/hand/right/input") + hand_pose_name},
{"hand_haptic_left", "/user/hand/left/output/haptic"},
{"hand_haptic_right", "/user/hand/right/output/haptic"},
}},
};
for (const auto& [id, args] : actionTypes) {
auto friendlyName = args.first;
auto xr_type = args.second;
std::shared_ptr<Action> action = std::make_shared<Action>(_context, id, friendlyName, xr_type);
if (!action->init(_actionSet)) { if (!action->init(_actionSet)) {
qCCritical(xr_input_cat, "Creating action %s failed!", path.c_str()); qCCritical(xr_input_cat) << "Creating action " << id.c_str() << " failed!";
} else { } else {
_actions.emplace(path, action); _actions.emplace(id, action);
} }
} }
// Khronos Simple Controller for (const auto& [profile, input] : actionSuggestions) {
std::vector<std::string> simpleBindings = { if (!initBindings(profile, input)) {
"/input/select/click", qCWarning(xr_input_cat) << "Failed to suggest actions for " << profile.c_str();
"/input/menu/click", }
"/input/grip/pose",
"/output/haptic",
};
if (!initBindings("/interaction_profiles/khr/simple_controller", simpleBindings)) {
qCCritical(xr_input_cat, "Failed to init bindings for khr/simple_controller");
}
// HTC Vive
std::vector<std::string> viveBindings = {
"/input/system/click",
"/input/squeeze/click",
"/input/menu/click",
"/input/trigger/click",
"/input/trigger/value",
"/input/trackpad/x",
"/input/trackpad/y",
"/input/trackpad/click",
"/input/trackpad/touch",
"/input/grip/pose",
"/output/haptic",
};
if (!initBindings("/interaction_profiles/htc/vive_controller", viveBindings)) {
qCCritical(xr_input_cat, "Failed to init bindings for htc/vive_controller");
}
// Valve Index Controller
// clang-format off
std::vector<std::string> indexBindings = {
"/input/grip/pose",
"/input/thumbstick/x",
"/input/thumbstick/y",
"/input/thumbstick/touch",
"/input/thumbstick/click",
"/input/a/click",
"/input/a/touch",
"/input/b/click",
"/input/b/touch",
"/input/trigger/value",
"/input/trigger/click",
"/input/trigger/touch",
"/output/haptic",
"/input/system/click",
};
// clang-format on
if (!initBindings("/interaction_profiles/valve/index_controller", indexBindings)) {
qCCritical(xr_input_cat, "Failed to init bindings for valve/index_controller");
} }
XrSessionActionSetsAttachInfo attachInfo = { XrSessionActionSetsAttachInfo attachInfo = {
@ -499,8 +472,8 @@ void OpenXrInputPlugin::InputDevice::update(float deltaTime, const controller::I
return; return;
} }
if (!initActions()) { if (!_actionsInitialized && !initActions()) {
qCCritical(xr_input_cat, "Could not initialize actions!"); qCCritical(xr_input_cat) << "Could not initialize actions!";
return; return;
} }
@ -534,7 +507,8 @@ void OpenXrInputPlugin::InputDevice::update(float deltaTime, const controller::I
static const glm::quat rightRotationOffset = glm::inverse(rightQuarterZ * eighthX) * touchToHand; static const glm::quat rightRotationOffset = glm::inverse(rightQuarterZ * eighthX) * touchToHand;
for (int i = 0; i < HAND_COUNT; i++) { for (int i = 0; i < HAND_COUNT; i++) {
XrSpaceLocation handLocation = _actions.at("/input/grip/pose")->getPose(i); auto hand_path = (i == 0) ? "hand_pose_left" : "hand_pose_right";
XrSpaceLocation handLocation = _actions.at(hand_path)->getPose();
bool locationValid = (handLocation.locationFlags & XR_SPACE_LOCATION_ORIENTATION_VALID_BIT) != 0; bool locationValid = (handLocation.locationFlags & XR_SPACE_LOCATION_ORIENTATION_VALID_BIT) != 0;
if (locationValid) { if (locationValid) {
vec3 translation = xrVecToGlm(handLocation.pose.position); vec3 translation = xrVecToGlm(handLocation.pose.position);
@ -556,87 +530,160 @@ void OpenXrInputPlugin::InputDevice::update(float deltaTime, const controller::I
glm::mat4 defaultHeadOffset = createMatFromQuatAndPos(-DEFAULT_AVATAR_HEAD_ROT, -DEFAULT_AVATAR_HEAD_TO_MIDDLE_EYE_OFFSET); glm::mat4 defaultHeadOffset = createMatFromQuatAndPos(-DEFAULT_AVATAR_HEAD_ROT, -DEFAULT_AVATAR_HEAD_TO_MIDDLE_EYE_OFFSET);
_poseStateMap[controller::HEAD] = _context->_lastHeadPose.postTransform(defaultHeadOffset).transform(sensorToAvatar); _poseStateMap[controller::HEAD] = _context->_lastHeadPose.postTransform(defaultHeadOffset).transform(sensorToAvatar);
std::vector<std::pair<controller::StandardAxisChannel, std::string>> axesToUpdate[2] = { std::vector<std::pair<std::string, controller::StandardAxisChannel>> floatsToUpdate = {
{ {"interact_left", controller::LT},
{ controller::LT, "/input/trigger/value" }, {"interact_right", controller::RT},
{ controller::LT, "/input/select/click" }, {"grip_left", controller::LEFT_GRIP},
{ controller::LEFT_GRIP, "/input/squeeze/click" }, {"grip_right", controller::RIGHT_GRIP},
{ controller::LX, "/input/trackpad/x" },
{ controller::LY, "/input/trackpad/y" },
{ controller::LX, "/input/thumbstick/x" },
{ controller::LY, "/input/thumbstick/y" },
},
{
{ controller::RT, "/input/trigger/value" },
{ controller::RT, "/input/select/click" },
{ controller::RIGHT_GRIP, "/input/squeeze/click" },
{ controller::RX, "/input/trackpad/x" },
{ controller::RY, "/input/trackpad/y" },
{ controller::RX, "/input/thumbstick/x" },
{ controller::RY, "/input/thumbstick/y" },
},
}; };
for (uint32_t i = 0; i < HAND_COUNT; i++) { for (const auto& [name, channel] : floatsToUpdate) {
for (const auto& [channel, path] : axesToUpdate[i]) { auto action = _actions.at(name)->getFloat();
auto action = _actions.at(path)->getFloat(i); if (action.isActive) {
if (action.isActive) { _axisStateMap[channel].value = action.currentState;
_axisStateMap[channel].value = action.currentState;
}
} }
} }
std::vector<std::pair<controller::StandardButtonChannel, std::string>> buttonsToUpdate[2] = { std::vector<std::tuple<std::string, controller::StandardAxisChannel, controller::StandardAxisChannel>> axesToUpdate = {
{ //{"stick_left", controller::LX, controller::LY},
{ controller::LS, "/input/trackpad/click" }, //{"stick_right", controller::RX, controller::RY},
{ controller::LS_TOUCH, "/input/trackpad/touch" }, {"walk", controller::LX, controller::LY},
{ controller::LS, "/input/thumbstick/click" },
{ controller::LS_TOUCH, "/input/thumbstick/touch" },
{ controller::LT_CLICK, "/input/trigger/click" },
{ controller::LEFT_PRIMARY_THUMB, "/input/a/click" },
{ controller::LEFT_PRIMARY_THUMB, "/input/system/click" },
{ controller::LEFT_SECONDARY_THUMB, "/input/b/click" },
{ controller::LEFT_SECONDARY_THUMB, "/input/menu/click" },
},
{
{ controller::RS, "/input/trackpad/click" },
{ controller::RS_TOUCH, "/input/trackpad/touch" },
{ controller::RS, "/input/thumbstick/click" },
{ controller::RS_TOUCH, "/input/thumbstick/touch" },
{ controller::RT_CLICK, "/input/trigger/click" },
{ controller::RIGHT_PRIMARY_THUMB, "/input/a/click" },
{ controller::RIGHT_PRIMARY_THUMB, "/input/system/click" },
{ controller::RIGHT_SECONDARY_THUMB, "/input/b/click" },
{ controller::RIGHT_SECONDARY_THUMB, "/input/menu/click" },
},
}; };
for (uint32_t i = 0; i < HAND_COUNT; i++) { for (const auto& [name, x_channel, y_channel] : axesToUpdate) {
for (const auto& [channel, path] : buttonsToUpdate[i]) { auto action = _actions.at(name)->getVector2f();
auto action = _actions.at(path)->getBool(i); if (action.isActive) {
if (action.isActive && action.currentState) { _axisStateMap[x_channel].value = action.currentState.x;
_buttonPressedMap.insert(channel); _axisStateMap[y_channel].value = -action.currentState.y;
}
} }
} }
// TODO: better alternative to having every controller emulate a vive one // INPUT TODO: more hacks
partitionTouchpad(controller::LS, controller::LX, controller::LY, controller::LS_CENTER, controller::LS_X, controller::LS_Y); {
partitionTouchpad(controller::RS, controller::RX, controller::RY, controller::RS_CENTER, controller::RS_X, controller::RS_Y); auto turn_left = _actions.at("turn_left")->getBool();
if (turn_left.isActive && turn_left.currentState) {
_axisStateMap[controller::RX].value -= 1.0f;
}
auto turn_right = _actions.at("turn_right")->getBool();
if (turn_right.isActive && turn_right.currentState) {
_axisStateMap[controller::RX].value += 1.0f;
}
// INPUT TODO: the teleport script is hardcoded to use the LY/RY axes
auto teleport = _actions.at("teleport")->getBool();
if (teleport.isActive && teleport.currentState) {
_axisStateMap[controller::RY].value += 1.0f;
}
}
// don't double up on the stick values between the proper actions and fallback sticks
_axisStateMap[controller::LX].value = std::clamp(_axisStateMap[controller::LX].value, -1.0f, 1.0f);
_axisStateMap[controller::LY].value = std::clamp(_axisStateMap[controller::LY].value, -1.0f, 1.0f);
_axisStateMap[controller::RX].value = std::clamp(_axisStateMap[controller::RX].value, -1.0f, 1.0f);
_axisStateMap[controller::RY].value = std::clamp(_axisStateMap[controller::RY].value, -1.0f, 1.0f);
std::vector<std::pair<std::string, controller::StandardButtonChannel>> buttonsToUpdate = {
{"tablet", controller::LEFT_PRIMARY_THUMB},
{"jump", controller::RIGHT_PRIMARY_THUMB},
{"cycle_camera", controller::LS_TOUCH},
{"sprint", controller::DD},
};
for (const auto& [name, channel] : buttonsToUpdate) {
auto action = _actions.at(name)->getBool();
if (action.isActive && action.currentState) {
_buttonPressedMap.insert(channel);
}
}
// INPUT TODO: it's really not necessary to expose "interact click" bindings,
// but the engine expects there to be click buttons to work properly
{
auto left = _actions.at("interact_left")->getFloat();
if (left.isActive && left.currentState == 1.0f) {
_buttonPressedMap.insert(controller::LT_CLICK);
}
auto right = _actions.at("interact_right")->getFloat();
if (right.isActive && right.currentState == 1.0f) {
_buttonPressedMap.insert(controller::RT_CLICK);
}
}
// emulate dpad if the dpad extension isn't available
emulateStickDPad();
} }
// copied from openvr/ViveControllerManager void OpenXrInputPlugin::InputDevice::emulateStickDPad() {
void OpenXrInputPlugin::InputDevice::partitionTouchpad(int sButton, int xAxis, int yAxis, int centerPseudoButton, int xPseudoButton, int yPseudoButton) { auto right_stick = _actions.at("stick_right")->getVector2f();
// Populate the L/RS_CENTER/OUTER pseudo buttons, corresponding to a partition of the L/RS space based on the X/Y values.
const float CENTER_DEADBAND = 0.6f; auto left_stick_click = _actions.at("stick_click_left")->getBool();
const float DIAGONAL_DIVIDE_IN_RADIANS = PI / 4.0f; auto right_stick_click = _actions.at("stick_click_right")->getBool();
if (_buttonPressedMap.find(sButton) != _buttonPressedMap.end()) {
float absX = abs(_axisStateMap[xAxis].value); auto left_needs_click = _context->_dpadNeedsClick && left_stick_click.isActive;
float absY = abs(_axisStateMap[yAxis].value); auto right_needs_click = _context->_dpadNeedsClick && left_stick_click.isActive;
glm::vec2 cartesianQuadrantI(absX, absY);
float angle = glm::atan(cartesianQuadrantI.y / cartesianQuadrantI.x); auto left_clicked = left_needs_click ? left_stick_click.currentState : true;
float radius = glm::length(cartesianQuadrantI); auto right_clicked = right_needs_click ? right_stick_click.currentState : true;
bool isCenter = radius < CENTER_DEADBAND;
_buttonPressedMap.insert(isCenter ? centerPseudoButton : ((angle < DIAGONAL_DIVIDE_IN_RADIANS) ? xPseudoButton :yPseudoButton)); if (right_stick.isActive && right_clicked) {
// camera switch (right stick down)
if (
right_stick.currentState.y < -0.6f
&& right_stick.currentState.x > -0.4f
&& right_stick.currentState.x < 0.4f
) {
_buttonPressedMap.insert(controller::LS_TOUCH);
}
// snap turn left
if (
right_stick.currentState.x < -0.6f
&& right_stick.currentState.y > -0.4f
&& right_stick.currentState.y < 0.4f
) {
_axisStateMap[controller::RX].value = -1.0f;
_axisStateMap[controller::RY].value = 0.0f;
}
// snap turn right
if (
right_stick.currentState.x > 0.6f
&& right_stick.currentState.y > -0.4f
&& right_stick.currentState.y < 0.4f
) {
_axisStateMap[controller::RX].value = 1.0f;
_axisStateMap[controller::RY].value = 0.0f;
}
// teleport
if (
right_stick.currentState.y > 0.6f
&& right_stick.currentState.x > -0.4f
&& right_stick.currentState.x < 0.4f
) {
_axisStateMap[controller::RX].value = 0.0f;
_axisStateMap[controller::RY].value = -1.0f;
}
}
// set stick inputs to zero if they're not
// clicked in or too close to the center
if (
!right_clicked
|| (right_stick.currentState.x > -0.6f
&& right_stick.currentState.x < 0.6f
&& right_stick.currentState.y > -0.6f
&& right_stick.currentState.y < 0.6f)
) {
_axisStateMap[controller::RX].value = 0.0f;
_axisStateMap[controller::RY].value = 0.0f;
}
// don't walk unless the trackpad is clicked in
if (!left_clicked) {
_axisStateMap[controller::LX].value = 0.0f;
_axisStateMap[controller::LY].value = 0.0f;
} }
} }

View file

@ -47,26 +47,28 @@ public:
private: private:
class Action { class Action {
public: public:
Action(std::shared_ptr<OpenXrContext> c, XrActionType type, const std::string& path) { Action(std::shared_ptr<OpenXrContext> c, const std::string& id, const std::string &friendlyName, XrActionType type) {
_context = c; _context = c;
_path = path; _id = id;
_friendlyName = friendlyName;
_type = type; _type = type;
} }
bool init(XrActionSet actionSet); bool init(XrActionSet actionSet);
std::vector<XrActionSuggestedBinding> getBindings(); std::vector<XrActionSuggestedBinding> getBindings();
XrActionStateFloat getFloat(uint32_t handId); XrActionStateFloat getFloat();
XrActionStateBoolean getBool(uint32_t handId); XrActionStateVector2f getVector2f();
XrSpaceLocation getPose(uint32_t handId); XrActionStateBoolean getBool();
bool applyHaptic(uint32_t handId, XrDuration duration, float frequency, float amplitude); XrSpaceLocation getPose();
bool applyHaptic(XrDuration duration, float frequency, float amplitude);
XrAction _action = XR_NULL_HANDLE;
private: private:
bool createPoseSpaces(); bool createPoseSpaces();
XrAction _action = XR_NULL_HANDLE;
std::shared_ptr<OpenXrContext> _context; std::shared_ptr<OpenXrContext> _context;
std::string _path; std::string _id, _friendlyName;
XrActionType _type; XrActionType _type;
XrSpace _poseSpaces[HAND_COUNT] = { XR_NULL_HANDLE, XR_NULL_HANDLE }; XrSpace _poseSpace = XR_NULL_HANDLE;
}; };
class InputDevice : public controller::InputDevice { class InputDevice : public controller::InputDevice {
@ -80,7 +82,7 @@ private:
void focusOutEvent() override; void focusOutEvent() override;
bool triggerHapticPulse(float strength, float duration, uint16_t index) override; bool triggerHapticPulse(float strength, float duration, uint16_t index) override;
void partitionTouchpad(int sButton, int xAxis, int yAxis, int centerPseudoButton, int xPseudoButton, int yPseudoButton); void emulateStickDPad();
mutable std::recursive_mutex _lock; mutable std::recursive_mutex _lock;
template <typename F> template <typename F>
@ -98,7 +100,7 @@ private:
bool _actionsInitialized = false; bool _actionsInitialized = false;
bool initActions(); bool initActions();
bool initBindings(const std::string& profileName, const std::vector<std::string>& actionsToBind); bool initBindings(const std::string& profileName, const std::map<std::string, std::string>& actionsToBind);
}; };
bool _registeredWithInputMapper = false; bool _registeredWithInputMapper = false;