Skip to main content
Version : 3.5.x

04. Bonjour Floor

Votre premier effet haptique : un sol virtuel horizontal qui offre une résistance lorsque le curseur s'y appuie. La force est simulée par un simple ressort de contrainte — stiffness × penetration_depth — appliqué le long de Z via set_cursor_force.

Ce que vous apprendrez :

  • Utilisation de set_cursor_force pour appliquer une force de rappel
  • Lecture cursor_position et la puissance de calcul en temps réel
  • (Python) Définition d'un réglage prédéfini de l'espace de travail (arm_front_centered) de sorte que l'origine se trouve au centre de l'espace de travail
  • (Python) Réglage interactif par le clavier de la hauteur et de la rigidité du plancher via le keyboard colis

Flux de travail

  1. Ouvrir un WebSocket vers ws://localhost:10001 et attendre la première trame d'état.
  2. À la première image : enregistrer le profil de session. La variante Python envoie en outre configure.preset: arm_front_centered ainsi, le point d'origine se trouve au centre de l'espace de travail ; les variantes C++ utilisent la configuration déjà active sur l'appareil.
  3. À chaque image : lire cursor_position.z, calculer force_z = max(0, (floor_pos - z) * stiffness), puis le renvoyer sous forme de set_cursor_force commande.
  4. Les requêtes suivantes n'envoient que la commande « force » : le profil de session ne nécessite qu'un seul échange initial.
  5. (Python) À chaque itération, le programme interroge également les touches fléchées du clavier et met à jour floor_pos / stiffness en direct.

Paramètres

NomPar défautObjectif
floor_pos0.10 mCoordonnée Z du plan du sol virtuel
stiffness1000 N/mConstante de ressort (pénétration de 1 mm → 1 N)
PRINT_EVERY_MS100–200Manette de télémétrie
Nom du profil de sessionco.haply.inverse.tutorials:hello-floorIdentifie cette simulation dans Haply
Interactif (Python uniquement)

La variante Python utilise le keyboard paquet (privilèges élevés requis sous Linux) :

  • / — relever / abaisser le plancher
  • / — réduire / augmenter la rigidité
  • R — rétablir les paramètres par défaut
Renforcer les effectifs

Les forces sont cumulées au niveau du tick du service : elles sont additionnées entre toutes les sources avant d'être envoyées à l'appareil. Un tutoriel comme celui-ci peut coexister avec d'autres générateurs de forces sans que l'un ou l'autre ne bloque l'autre.

Champs d'état lus

De data.inverse3[i].state:

  • cursor_position.zvec3, utilisé pour calculer la profondeur de pénétration
  • current_cursor_force — transmis par télémétrie

Envoyer / recevoir

À chaque itération : lire la coordonnée Z du curseur, calculer force_z = max(0, (floor_pos - z) * stiffness), et envoyer un set_cursor_force. Le premier message sortant contient également le profil de session (toutes variantes) et, pour Python, configure.preset: arm_front_centered.

Boucle asynchrone unique. La négociation lors de la première trame inclut le profil + configure.preset donc floor_pos = 0.1 s'aligne sur les coordonnées du centre de l'espace de travail.

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))

Source : Python · C++ · C++ Glaze

À lire également : Commandes de contrôle (set_cursor_force) · Montage et espace de travail (préréglages) · Types (vec3) · Séances · Tutoriel 07 (Base et support)