基本力反馈教程
本教程将指导您创建一个包含刚度和阻尼的基本触觉模拟,模拟与球体等静态物体接触的物理特性。 本教程结束时,您将获得一个模拟,可以感受球体的存在,并调整其硬度和阻尼属性,获得不同的触觉体验。
导言
本教程的核心挑战是开发一个函数,能够计算与具有刚度和阻尼的球体接触时产生的力。 在这种情况下,刚度的作用就像弹簧一样,越压缩产生的力越大。 而阻尼则代表物体的运动阻力,移动速度越快,阻力越大。
场景设置
首先,按照以下步骤设置触觉装备 快速入门指南确保 触觉起源的位置、旋转和缩放设置为 (0, 0, 0) 和 (1, 1, 1) 分别是
然后,创建一个名为 Sphere 具有以下特性
- 职位: (0, 0, -0.1)(设备前方约 10 厘米处)。
- 规模: (0.2, 0.2, 0.2)(相当于一个直径 20 厘米的球体)
力反馈脚本
添加一个新的 C# 脚本到 球体 名为 SphereForceFeedback.cs.
该脚本将计算Inverse3 光标与球体接触时所受的力,同时考虑到刚度和阻尼。
用以下属性初始化脚本:
[SerializeField]
private Inverse3 inverse3;
[Range(0, 800)]
public float stiffness = 300f;
[Range(0, 3)]
public float damping = 1f;
private Vector3 _ballPosition;
private float _ballRadius;
private float _cursorRadius;
只有当光标穿透球体时,力的计算才会发生,从而模拟触摸具有确定刚度和阻尼的实物的感觉。
力的计算 ForceCalculation 方法将负责这一工作,同时考虑光标的位置和速度:
private Vector3 ForceCalculation(Vector3 cursorPosition, Vector3 cursorVelocity, float cursorRadius,
    Vector3 otherPosition, float otherRadius)
{
    var force = Vector3.zero;
    var distanceVector = cursorPosition - otherPosition;
    var distance = distanceVector.magnitude;
    var penetration = otherRadius + cursorRadius - distance;
    if (penetration > 0)
    {
        // Normalize the distance vector to get the direction of the force
        var normal = distanceVector.normalized;
        // Calculate the force based on penetration
        force = normal * penetration * stiffness;
        // Apply damping based on the cursor velocity
        force -= cursorVelocity * damping;
    }
    return force;
}
private void OnDeviceStateChanged(Inverse3 device)
{
    // Calculate the ball force
    var force = ForceCalculation(device.LocalPosition, device.LocalVelocity,
        _cursorRadius, _ballPosition, _ballRadius);
    device.CursorSetLocalForce(force);
}
在 Awake 方法,初始化 _ballPosition, _ballRadius和 _cursorRadius 来设置场景数据:
private void SaveSceneData()
{
    var t = transform;
    _ballPosition = t.position;
    _ballRadius = t.lossyScale.x / 2f;
    _cursorRadius = inverse3.Cursor.Model.transform.lossyScale.x / 2f;
}
private void Awake()
{
    SaveSceneData();
}
确保注册和取消注册 OnDeviceStateChanged 中的回调 OnEnable 和 OnDisable 方法,以正确处理交互过程中的力反馈:
protected void OnEnable()
{
    inverse3.DeviceStateChanged += OnDeviceStateChanged;
}
protected void OnDisable()
{
    inverse3.DeviceStateChanged -= OnDeviceStateChanged;
}
游戏体验
按住Inverse3 光标,进入 "播放 "模式并尝试触摸球体。 您应该能够感受到球体的存在,并通过 Unity 检查器操作其刚度和阻尼属性,从而获得与虚拟对象进行交互的真实感。

源文件
本示例的完整场景和所有相关文件均可从 Unity 软件包管理器中的教程示例中导入。
SphereForceFeedback.cs
/*
 * Copyright 2024 Haply Robotics Inc. All rights reserved.
 */
using Haply.Inverse.Unity;
using UnityEngine;
namespace Haply.Samples.Tutorials._2_BasicForceFeedback
{
    public class SphereForceFeedback : MonoBehaviour
    {
        // must assign in inspector
        public Inverse3 inverse3;
        [Range(0, 800)]
        // Stiffness of the force feedback.
        public float stiffness = 300f;
        [Range(0, 3)]
        public float damping = 1f;
        private Vector3 _ballPosition;
        private float _ballRadius;
        private float _cursorRadius;
        /// <summary>
        /// Stores the cursor and sphere transform data for access by the haptic thread.
        /// </summary>
        private void SaveSceneData()
        {
            var t = transform;
            _ballPosition = t.position;
            _ballRadius = t.lossyScale.x / 2f;
            _cursorRadius = inverse3.Cursor.Model.transform.lossyScale.x / 2f;
        }
        /// <summary>
        /// Saves the initial scene data cache.
        /// </summary>
        private void Awake()
        {
            SaveSceneData();
        }
        /// <summary>
        /// Subscribes to the DeviceStateChanged event.
        /// </summary>
        private void OnEnable()
        {
            inverse3.DeviceStateChanged += OnDeviceStateChanged;
        }
        /// <summary>
        /// Unsubscribes from the DeviceStateChanged event.
        /// </summary>
        private void OnDisable()
        {
            inverse3.DeviceStateChanged -= OnDeviceStateChanged;
        }
        /// <summary>
        /// Calculates the force based on the cursor's position and another sphere position.
        /// </summary>
        /// <param name="cursorPosition">The position of the cursor.</param>
        /// <param name="cursorVelocity">The velocity of the cursor.</param>
        /// <param name="cursorRadius">The radius of the cursor.</param>
        /// <param name="otherPosition">The position of the other sphere (e.g., ball).</param>
        /// <param name="otherRadius">The radius of the other sphere.</param>
        /// <returns>The calculated force vector.</returns>
        private Vector3 ForceCalculation(Vector3 cursorPosition, Vector3 cursorVelocity, float cursorRadius,
            Vector3 otherPosition, float otherRadius)
        {
            var force = Vector3.zero;
            var distanceVector = cursorPosition - otherPosition;
            var distance = distanceVector.magnitude;
            var penetration = otherRadius + cursorRadius - distance;
            if (penetration > 0)
            {
                // Normalize the distance vector to get the direction of the force
                var normal = distanceVector.normalized;
                // Calculate the force based on penetration
                force = normal * penetration * stiffness;
                // Apply damping based on the cursor velocity
                force -= cursorVelocity * damping;
            }
            return force;
        }
        /// <summary>
        /// Event handler that calculates and send the force to the device when the cursor's position changes.
        /// </summary>
        /// <param name="device">The Inverse3 device instance.</param>
        private void OnDeviceStateChanged(Inverse3 device)
        {
            // Calculate the ball force
            var force = ForceCalculation(device.LocalPosition, device.LocalVelocity,
                _cursorRadius, _ballPosition, _ballRadius);
            device.CursorSetLocalForce(force);
        }
    }
}