mirror of
https://github.com/overte-org/overte.git
synced 2025-04-07 00:52:58 +02:00
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:
parent
dc3a508051
commit
cedc5be526
7 changed files with 374 additions and 260 deletions
29
interface/resources/controllers/openxr.json
Normal file
29
interface/resources/controllers/openxr.json
Normal 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" }
|
||||||
|
]
|
||||||
|
}
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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() {
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
Loading…
Reference in a new issue