06. Combiné (Inverse3 Wireless VerseGrip)
Tutoriel pour deux appareils : pointez la poignée et maintenez un bouton enfoncé pour déplacer le Inverse3 dans cette direction. Le curseur est confiné à l'intérieur d'un espace de travail sphérique.
Ce que vous apprendrez :
- Lecture de deux types d'appareils dans le même cadre d'état (
inverse3etwireless_verse_grip) - Déterminer la direction vers laquelle pointe la poignée à partir de sa quaternion (local
+Yaxe) - Utilisation de
set_cursor_positionpour diriger le curseur vers une cible calculée - Fixation de la cible dans un espace de travail sécurisé — Minverse un rayon plus petit qu'Inverse3
- Définir un réglage prédéfini de l'espace de travail (
arm_front_centered) de sorte que le point d'origine se trouve au milieu de la portée
Flux de travail
- Découvrez ces deux appareils :
- C++ requête sur les variantes
GET /devicesvia HTTP au démarrage, puis afficher une invite de calibrage et attendre que l'on appuie sur ENTRÉE. - Python lit les deux identifiants de périphérique à partir de la première trame d'état WebSocket.
- C++ requête sur les variantes
- Enregistrer le profil de session et définir
configure.preset: arm_front_centereddès le premier message (poignée de main en une seule étape). - À chaque coup de manivelle : lisez les indications de la poignée
orientationetbuttons.{a, b}État. - Si un bouton de mouvement est maintenu enfoncé, calculez la direction de la poignée dans l'espace mondial (
R(q) · ĵ— l'unité pivotée autour de l'axe +Y) et l'accumuler jusqu'à la position cible mise à l'échelle parSPEED. - Placez la cible à l'intérieur de la sphère de travail et envoyez-la via
set_cursor_position. - (Python uniquement) Adaptez le rayon de la sphère en fonction de l'appareil
config.type—minverse= 0,04 m, tout le reste = 0,10 m.
Paramètres
| Nom | Par défaut | Objectif |
|---|---|---|
SPEED | 0.01 m/tick | Déplacement par pas pendant qu'un bouton est maintenu enfoncé |
RADIUS_INVERSE3 | 0.10 m | Rayon de serrage de l'espace de travail pour Inverse3 Inverse3x |
RADIUS_MINVERSE | 0.04 m | Rayon de la pince de l'espace de travail pour Minverse Python uniquement — le C++ utilise des valeurs fixes) 0.10) |
PRINT_EVERY_MS | 200 | Manette de télémétrie |
| Nom du profil de session | co.haply.inverse.tutorials:combined | Identifie cette simulation dans Haply |
- Laissez Inverse3 (ou placez la poignée sur l'encrier et attendez que la LED reste allumée en continu).
- Retirez le manche de l'encrier.
- Maintenez la touche A ou B enfoncée et faites pivoter la poignée : le curseur se déplace dans la direction indiquée par la poignée.
Champs d'état lus
À partir du cadre d'état par tick :
data.inverse3[0].state.cursor_position—vec3data.wireless_verse_grip[0].state.orientation—quaterniondata.wireless_verse_grip[0].state.buttons.{a, b, c}— booléens- (Python, première image uniquement)
data.inverse3[0].config.type— sélectionne Minverse « Inverse3 Minverse - (Python, première image uniquement)
data.inverse3[0].status.calibrated— demande à l'utilisateur si la valeur est fausse
Envoyer / recevoir
Le calcul de conversion entre quaternions et directions (rotation +Y par R(q)) et la pince sphérique relèvent de l'algèbre linéaire classique — voir les fichiers source. Du côté de l'API inverse, il s'agit de la négociation de connexion + par tick set_cursor_position.
- Python
- C++ (nlohmann)
- C++ (Glaze)
Boucle asynchrone unique. Python lit les deux identifiants de périphérique à partir de la première trame d'état ; la négociation établit le profil + configure.preset: arm_front_centered au premier set_cursor_position.
async with websockets.connect(URI) as websocket:
while True:
msg = await websocket.recv()
data = json.loads(msg)
if first_message:
first_message = False
inverse3_id = data["inverse3"][0]["device_id"]
grip_id = data["wireless_verse_grip"][0]["device_id"]
radius = get_workspace_radius(data["inverse3"][0].get("config", {}))
# Handshake: profile + preset + first position command
request_msg = {
"session": {"configure": {"profile": {"name": SLUG}}},
"inverse3": [{
"device_id": inverse3_id,
"configure": {"preset": {"preset": "arm_front_centered"}},
"commands": {"set_cursor_position": {"position": position}},
}],
}
else:
# Per tick: update position from grip pointing direction (classic math, not shown), send
request_msg = {
"inverse3": [{
"device_id": inverse3_id,
"commands": {"set_cursor_position": {"position": position}},
}],
}
await websocket.send(json.dumps(request_msg))
Le C++ détecte les deux identifiants de périphérique via GET /devices au démarrage (HTTP), puis ouvre le WebSocket. onmessage s'exécute sur le thread d'E/S de libhv ; le thread principal se bloque sur ENTER.
// Startup (synchronous):
const std::string inv3_device_id = get_first_device_id("inverse3");
const std::string grip_device_id = get_first_device_id("wireless_verse_grip");
// Per tick:
ws.onmessage = [&](const std::string &msg) {
json data = json::parse(msg);
// ... classic math (not shown): update local Inverse3State + WirelessVerseGripState,
// compute new position from grip orientation, clamp to sphere ...
json command;
command["inverse3"] = json::array();
command["inverse3"].push_back({
{"device_id", inv3_device_id},
{"commands", {{"set_cursor_position",
{{"position", {{"x", pos.x}, {"y", pos.y}, {"z", pos.z}}}}}}}
});
if (first_message) {
first_message = false;
command["session"] = {{"configure", {{"profile",
{{"name", "co.haply.inverse.tutorials:combined"}}}}}};
command["inverse3"][0]["configure"] = {
{"preset", {{"preset", "arm_front_centered"}}}};
}
ws.send(command.dump());
};
ws.open("ws://localhost:10001");
while (std::cin.get() != '\n') {} // block main thread
Même modèle de rappel libhv. Des structures typées modélisent à la fois les trames d'état et la commande sortante — deux std::vector<> dans devices_message laissons un seul glz::read fournir les deux types d'appareils.
// Struct models
struct quat { float w{1.0f}, x{}, y{}, z{}; };
struct button_state { bool a{}, b{}, c{}; };
struct wvg_state { quat orientation{}; uint8_t hall{}; button_state buttons{}; };
struct wvg_device { std::string device_id; wvg_state state; };
struct inverse_state { vec3 cursor_position{}, cursor_velocity{}; };
struct inverse_device { std::string device_id; inverse_state state; };
struct devices_message {
std::vector<inverse_device> inverse3;
std::vector<wvg_device> wireless_verse_grip;
};
struct set_cursor_position_cmd { vec3 position; };
struct commands_message {
std::optional<session_cmd> session;
std::vector<device_commands> inverse3;
};
// Send / receive
ws.onmessage = [&](const std::string &msg) {
devices_message data{};
if (glz::read<glz_settings>(data, msg)) return;
const auto &wvg = data.wireless_verse_grip[0].state;
cursor_pos = data.inverse3[0].state.cursor_position;
// ... classic math (not shown): if (wvg.buttons.a || wvg.buttons.b)
// move in pointing dir; clamp to sphere ...
commands_message out_cmds{};
device_commands dc{ .device_id = inv3_device_id };
dc.commands.set_cursor_position = set_cursor_position_cmd{cursor_pos};
if (first_message) {
first_message = false;
out_cmds.session = session_cmd{ /* profile = combined */ };
dc.configure = device_configure{ .preset = preset_cfg{"arm_front_centered"} };
}
out_cmds.inverse3.push_back(std::move(dc));
std::string out_json;
(void)glz::write_json(out_cmds, out_json);
ws.send(out_json);
};
ws.open("ws://localhost:10001");
while (std::cin.get() != '\n') {} // block main thread
Source : Python · C++ · C++ Glaze
À lire également : Commandes de contrôle (set_cursor_position) · Types (quaternion, vec3) · Montage et espace de travail (préréglages) · Tutoriel 03 (VG sans fil) · Tutoriel 05 (Contrôle de position)