05. Positionssteuerung
Bewegt den Inverse3 über set_cursor_position. Das Interaktionsmodell unterscheidet sich je nach Sprache — C++ verwendet einmalige Zufallsziele, Python nutzt eine kontinuierliche, über die Tastatur gesteuerte Bewegung.
Was Sie lernen werden:
- Verwendung von
set_cursor_positionfür die Positionsregelung - Zwei verschiedene Interaktionsmodelle für denselben zugrunde liegenden Befehl
- Das Ziel an eine Arbeitsbereichskugel binden – Minverse einen kleineren Radius als Inverse3
- Festlegen einer Arbeitsbereichsvoreinstellung, sodass der Ursprung in der Mitte des Arbeitsbereichs liegt
Arbeitsablauf
C++ (Random-Target-Modell)
- Starte einen Hintergrund-Eingabethread, der zeilenweise gepufferte Tastenanschläge liest (
n,+,-,q) aus der Standardeingabe. - Öffne den WebSocket. Registriere im ersten Status-Frame den Sitzungsprofil und festlegen
configure.preset: arm_front_centered. Erzeuge den ersten Zufallswert innerhalb einer Kugel (Rejection Sampling, Radius 0,08 m). - Bei jedem Tick sende eine
set_cursor_positionBefehl an das aktuelle Ziel. Der Cursor folgt ihm flüssig – der Dienst begrenzt die Rate und interpoliert. - Wenn der Benutzer etwas eingibt
n+ ENTER: Der Eingabe-Thread generiert ein neues zufälliges Ziel.+/-Geschwindigkeit anpassen;qbeendet.
Python (Hold-to-Move-Modell)
- Öffne den WebSocket. Überprüfe im ersten Status-Frame
status.calibrated— den Benutzer darauf hinweisen, falls das Gerät noch nicht kalibriert ist. - Lesen
config.typeum den Arbeitsbereichsradius auszuwählen (minverse= 0,04 m, alle anderen Werte = 0,10 m). - Registrieren Sie die Sitzungsprofil und festlegen
configure.preset: arm_front_centered. - Bei jedem Tick: Tastaturstatus abfragen (
W/A/S/D/Q/E), aktualisiere die Zielposition umSPEEDentlang jeder gedrückten Achse, an die Arbeitsbereichskugel andocken und sendenset_cursor_position.Rsetzt das Ziel auf den Ursprung zurück.
Parameter
| Name | Standard (C++) | Standard (Python) | Zweck |
|---|---|---|---|
workspace_radius / RADIUS_INVERSE3 | 0.08 m | 0.10 m (Inverse3) / 0.04 m (Minverse) | Radius der Zielkugel |
speed_step / SPEED | 0.01 / Presse | 0.00005 m / Tick | Schritt pro Interaktion |
PRINT_EVERY_MS | — | 100 | Telemetrie-Drossel (Python) |
| Sitzungsprofil | co.haply.inverse.tutorials:position-control | dasselbe | Identifiziert in Haply |
Die Python-Variante prüft status.calibrated aus dem ersten Status-Frame und fragt den Benutzer, ob das Gerät noch nicht kalibriert ist. Die C++-Variante geht davon aus, dass die Kalibrierung bereits abgeschlossen ist.
Statusfelder gelesen
data.inverse3[0].device_id— zum Erstellen des Befehlsdata.inverse3[0].state.cursor_position— Telemetrie- (Python, nur das erste Bild)
data.inverse3[0].config.type— wählt Minverse von Inverse3 Minverse aus - (Python, nur das erste Bild)
data.inverse3[0].status.calibrated— fordert den Benutzer auf, wenn der Wert „false“ ist
Senden / Empfangen
Kommunikationsablauf
- C++ führt einen Stdin-Hintergrund-Thread aus, der schreibt
std::atomic<float>Ziele; der WebSocket-Thread liest sie bei jedem Tick. Aufn+ ENTER: Der Eingabe-Thread generiert ein neues zufälliges Ziel; beiq… beide Threads wurden geschlossen. - Python ist ein Single-Thread-Asynchronverfahren – die WebSocket-Schleife fragt bei jedem Tick den Tastaturstatus ab und aktualisiert
positiondirekt.
Die Nutzdaten der Inverse-API sind identisch: Der erste Tick enthält das Sitzungsprofil + configure.preset, nachfolgende Ticks enthalten nur set_cursor_position.
- Python
- C++ (nlohmann)
- C++ (Glaze)
Einzelne asynchrone Schleife. Tastaturabfrage (handle_keys) wird bei jedem Tick inline ausgeführt – ohne Threads. config.type und status.calibrated werden einmalig aus dem ersten Zustandsrahmen gelesen.
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"]
radius = get_workspace_radius(data["inverse3"][0].get("config", {}))
# Handshake: profile + preset (one-shot)
request_msg = {
"session": {"configure": {"profile": {"name": SLUG}}},
"inverse3": [{
"device_id": device_id,
"configure": {"preset": {"preset": "arm_front_centered"}},
}],
}
else:
# Per tick: update position from keyboard (classic polling, not shown), send command
position = handle_keys(position, radius)
request_msg = {
"inverse3": [{
"device_id": device_id,
"commands": {"set_cursor_position": {"position": position}},
}],
}
await websocket.send(json.dumps(request_msg))
Zwei-Thread-Modell: Ein Hintergrund-Thread liest aus der Standardeingabe und schreibt std::atomic<float> Ziele; der libhv-E/A-Thread läuft ws.onmessage bei jedem Takt und liest die Atomics aus.
// Shared state written by the stdin thread, read by the ws thread
static std::atomic<float> target_x{0.0f}, target_y{0.0f}, target_z{0.0f};
ws.onmessage = [&](const std::string &message) {
const json data = json::parse(message);
if (data["inverse3"].empty()) return;
json request = {};
const bool do_handshake = first_message.exchange(false);
if (do_handshake) {
request["session"] = {{"configure", {{"profile",
{{"name", "co.haply.inverse.tutorials:position-control"}}}}}};
generate_random_target();
}
request["inverse3"] = json::array();
for (auto &el : data["inverse3"].items()) {
json dev = {
{"device_id", el.value()["device_id"]},
{"commands", {{"set_cursor_position",
{{"position", {{"x", target_x.load()},
{"y", target_y.load()},
{"z", target_z.load()}}}}}}},
};
if (do_handshake)
dev["configure"] = {{"preset", {{"preset", "arm_front_centered"}}}};
request["inverse3"].push_back(dev);
}
ws.send(request.dump());
};
ws.open("ws://localhost:10001");
std::thread input_thr(input_thread_func); // stdin reader — writes target atomics
while (running) std::this_thread::sleep_for(100ms);
Dasselbe Zwei-Thread-Modell. Typisierte Strukturen für den Befehl — std::optional<device_configure> enthält die einmalige Voreinstellung pro Gerät; wird bei nachfolgenden Ticks aus der JSON-Datei weggelassen.
// Struct models
struct vec3 { float x{}, y{}, z{}; };
struct set_cursor_position_cmd { vec3 position; };
struct preset_cfg { std::string preset; };
struct device_configure { std::optional<preset_cfg> preset; };
struct device_commands {
std::string device_id;
std::optional<device_configure> configure; // one-shot
struct commands_t {
std::optional<set_cursor_position_cmd> set_cursor_position;
} commands;
};
struct commands_message {
std::optional<session_cmd> session; // one-shot
std::vector<device_commands> inverse3;
};
// Send / receive
ws.onmessage = [&](const std::string &msg) {
devices_message data{};
if (glz::read<glz_settings>(data, msg)) return;
commands_message request;
const bool do_handshake = first_message.exchange(false);
if (do_handshake) {
request.session = session_cmd{ /* profile = position-control */ };
}
for (const auto &dev : data.inverse3) {
device_commands dc{ .device_id = dev.device_id };
dc.commands.set_cursor_position = set_cursor_position_cmd{
.position = {target_x.load(), target_y.load(), target_z.load()}};
if (do_handshake)
dc.configure = device_configure{ .preset = preset_cfg{"arm_front_centered"} };
request.inverse3.push_back(std::move(dc));
}
std::string out;
(void)glz::write_json(request, out);
ws.send(out);
};
ws.open("ws://localhost:10001");
std::thread input_thr(input_thread_func);
while (running) std::this_thread::sleep_for(100ms);
Quelle: Python · C++ · C++ Glaze
Siehe auch: Steuerbefehle (set_cursor_position) · Mount & Arbeitsbereich (Voreinstellungen) · Typen (vec3) · Tutorial 06 (Kombiniert)