跳至主要内容
版本:3.5.x

04. 你好,地板

你的第一个触觉效果:一个虚拟的水平地面,当光标按压上去时会产生反作用力。这种力由一个简单的负值弹簧模型实现—— stiffness × penetration_depth — 沿 Z 轴施加 set_cursor_force.

您将学到:

工作流程

  1. 打开一个 WebSocket 连接至 ws://localhost:10001 并等待第一个状态帧。
  2. 在第一个帧中:注册 会话配置文件. Python 版本还会发送 configure.preset: arm_front_centered 因此,原点位于工作区的中间;C++ 版本则使用设备上当前已启用的配置。
  3. 在每一帧上:读取 cursor_position.z, 计算 force_z = max(0, (floor_pos - z) * stiffness),并将其作为 set_cursor_force 命令。
  4. 后续的TICK消息仅发送force命令——会话配置文件采用一次性握手机制。
  5. (Python) 每个时间步长还会轮询键盘方向键并进行更新 floor_pos / stiffness 直播。

参数

名称默认目的
floor_pos0.10 m虚拟地板平面的Z坐标
stiffness1000 不适用弹簧刚度(1 毫米压入深度 → 1 牛顿)
PRINT_EVERY_MS100–200遥测节流阀
会话配置文件名称co.haply.inverse.tutorials:hello-floor在Haply 中标识此模拟
交互式(仅限 Python)

Python 变体使用 keyboard 软件包(在 Linux 上需要提升权限):

  • / — 升高/降低地板平面
  • / — 降低 / 提高刚度
  • R — 恢复默认设置
扩展力量

在服务 tick 中,力是可累加的——在发送至设备之前,会先对所有来源的力进行求和。此类教程可与其他力生成器共存,彼此之间不会相互阻塞。

读取状态字段

来自 data.inverse3[i].state:

  • cursor_position.zvec3,用于计算穿透深度
  • current_cursor_force — 用于遥测报告

发送/接收

每次更新:读取光标的Z值,计算 force_z = max(0, (floor_pos - z) * stiffness),并发送一封 set_cursor_force. 第一条发送消息还携带会话配置文件(所有变体),对于 Python 而言, configure.preset: arm_front_centered.

单个异步循环。首帧握手携带配置文件 + configure.preset 所以 floor_pos = 0.1 与工作区中心坐标系对齐。

async with websockets.connect(URI) as websocket:
while True:
msg = await websocket.recv()
data = json.loads(msg)

if first_message:
first_message = False
device_id = data["inverse3"][0]["device_id"]
# Handshake: profile + preset (one-shot)
request_msg = {
"session": {"configure": {"profile": {"name": "co.haply.inverse.tutorials:hello-floor"}}},
"inverse3": [{
"device_id": device_id,
"configure": {"preset": {"preset": "arm_front_centered"}}
}]
}
else:
# Per tick: compute force along Z, send set_cursor_force
z = data["inverse3"][0]["state"]["cursor_position"]["z"]
force_z = 0.0 if z > floor_pos else (floor_pos - z) * stiffness
request_msg = {
"inverse3": [{
"device_id": device_id,
"commands": {"set_cursor_force":
{"vector": {"x": 0.0, "y": 0.0, "z": force_z}}}
}]
}

await websocket.send(json.dumps(request_msg))

来源: Python·C++·C++ Glaze

相关: 控制命令 (set_cursor_force) · 安装与工作区(预设) · 类型 (vec3) · 会话 · 教程 07(基座与安装)