通常のXR機器の開発では、特にVR機器に適している場合は、ハンドル操作を使用します。
Oculus SDKがOVRInputを提供し、コントローラーの重要な情報を取得できることはわかっています。
// public variable that can be set to LTouch or RTouch in the Unity Inspector
public Controller controller;
// returns a float of the Hand Trigger’s current state on the Oculus Touch controller
// specified by the controller variable.
OVRInput.Get(OVRInput.Axis1D.PrimaryHandTrigger, controller);
// returns true if the primary button (“A” or “X”) is pressed on the Oculus Touch controller
// specified by the controller variable.
OVRInput.Get(OVRInput.Button.One, controller);
私たちが開発するアプリケーションはOculusで実行するだけでなく、他のVRデバイスで実行する場合、各VRデバイスのSDKに接続する必要がありますか?
慌てる必要はありません。実際、Unityはすでにこれを行っています。Unityは、さまざまなVRデバイスで使用できる抽象化であるXR入力インターフェイスをカプセル化します。
それの使い方?まず、Unityによってパッケージ化された各ボタンの対応を見てみましょう。すでに市場に出回っている多くのVRデバイスをサポートしています。主にOculus。公式ウェブサイトにアクセスして参照できます:Unity-マニュアル:Unity XR Input
使用法をカプセル化したので、GameObjectをマウントすることで使用できます。自分で使用できます。
using System.Collections.Generic;
using UnityEngine;
using UnityEngine.Events;
using UnityEngine.XR;
[System.Serializable]
public enum InputFeature
{
//
// ժҪ:
// The primary face button being pressed on a device, or sole button if only one
// is available.
primaryButton,
//
// ժҪ:
// The primary face button being touched on a device.
primaryTouch,
//
// ժҪ:
// The secondary face button being pressed on a device.
secondaryButton,
//
// ժҪ:
// The secondary face button being touched on a device.
secondaryTouch,
//
// ժҪ:
// A binary measure of whether the device is being gripped.
gripButton,
//
// ժҪ:
// A binary measure of whether the index finger is activating the trigger.
triggerButton,
//
// ժҪ:
// Represents a menu button, used to pause, go back, or otherwise exit gameplay.
menuButton,
//
// ժҪ:
// Represents the primary 2D axis being clicked or otherwise depressed.
primary2DAxisClick,
//
// ժҪ:
// Represents the primary 2D axis being touched.
primary2DAxisTouch,
//
// ժҪ:
// Represents the secondary 2D axis being clicked or otherwise depressed.
secondary2DAxisClick,
//
// ժҪ:
// Represents the secondary 2D axis being touched.
secondary2DAxisTouch,
//
// ժҪ:
// Use this property to test whether the user is currently wearing and/or interacting
// with the XR device. The exact behavior of this property varies with each type
// of device: some devices have a sensor specifically to detect user proximity,
// however you can reasonably infer that a user is present with the device when
// the property is UserPresenceState.Present.
userPresence
}
public class XRControllerButtonWatcher : MonoBehaviour
{
private Dictionary<InputFeature, InputFeatureUsage<bool>> inputFeatureUsageMap = new Dictionary<InputFeature, InputFeatureUsage<bool>>() {
{ InputFeature.primaryButton, CommonUsages.primaryButton },
{ InputFeature.primaryTouch, CommonUsages.primaryTouch },
{ InputFeature.secondaryButton, CommonUsages.secondaryButton },
{ InputFeature.secondaryTouch, CommonUsages.secondaryTouch },
{ InputFeature.gripButton, CommonUsages.gripButton },
{ InputFeature.triggerButton, CommonUsages.triggerButton },
{ InputFeature.menuButton, CommonUsages.menuButton },
{ InputFeature.primary2DAxisClick, CommonUsages.primary2DAxisClick },
{ InputFeature.primary2DAxisTouch, CommonUsages.primary2DAxisTouch },
{ InputFeature.secondary2DAxisClick, CommonUsages.secondary2DAxisClick },
{ InputFeature.secondary2DAxisTouch, CommonUsages.secondary2DAxisTouch },
{ InputFeature.userPresence, CommonUsages.userPresence }
};
[SerializeField]
public InputFeature ButtonInputFeature;
[SerializeField]
public InputDeviceCharacteristics inputDeviceType = InputDeviceCharacteristics.None;
[SerializeField]
public UnityEvent OnInputDown;
[SerializeField]
public UnityEvent OnInputUp;
private InputFeatureUsage<bool> buttonInputFeatureUsage;
private bool lastButtonState = false;
private List<InputDevice> devicesButton;
private void Awake()
{
if (OnInputDown == null)
OnInputDown = new UnityEvent();
if (OnInputUp == null)
OnInputUp = new UnityEvent();
devicesButton = new List<InputDevice>();
if (!inputFeatureUsageMap.TryGetValue(ButtonInputFeature, out buttonInputFeatureUsage))
{
Debug.LogError("not found inputFeature: " + ButtonInputFeature);
}
}
private void OnEnable()
{
Init();
}
private void OnDisable()
{
InputDevices.deviceConnected -= InputDevices_deviceConnected;
InputDevices.deviceDisconnected -= InputDevices_deviceDisconnected;
devicesButton.Clear();
}
private void InputDevices_deviceConnected(InputDevice device)
{
bool discardedValue;
if (device.TryGetFeatureValue(buttonInputFeatureUsage, out discardedValue))
{
Debug.Log($"add device: {device.name}");
devicesButton.Add(device); // Add any devices that have a primary button.
}
}
private void InputDevices_deviceDisconnected(InputDevice device)
{
if (devicesButton.Contains(device))
devicesButton.Remove(device);
}
private void Update()
{
bool tempState = false;
foreach (var device in devicesButton)
{
bool primaryButtonState = false;
tempState = device.TryGetFeatureValue(buttonInputFeatureUsage, out primaryButtonState) // did get a value
&& primaryButtonState // the value we got
|| tempState; // cumulative result from other controllers
}
if (tempState != lastButtonState) // Button state changed since last frame
{
if (tempState)
OnInputDown?.Invoke();
else
OnInputUp?.Invoke();
lastButtonState = tempState;
}
}
public void Init()
{
List<InputDevice> allDevices = new List<InputDevice>();
if (inputDeviceType == InputDeviceCharacteristics.None)
InputDevices.GetDevices(allDevices);
else
InputDevices.GetDevicesWithCharacteristics(inputDeviceType, allDevices);
foreach (InputDevice device in allDevices)
InputDevices_deviceConnected(device);
InputDevices.deviceConnected += InputDevices_deviceConnected;
InputDevices.deviceDisconnected += InputDevices_deviceDisconnected;
}
}
Unityに付属のXR入力を使用すると、さまざまなデバイスを十分にサポートできます。各VRデバイスのSDKの入力を1つずつ接続する必要はありません。
参照ドキュメント: