--- title: GUIs (Formspecs) layout: default root: ../.. idx: 4.5 description: Lerne, wie man GUIs mit formspecs anzeigt redirect_from: /de/chapters/formspecs.html submit_vuln: level: warning title: Malicious clients can submit anything at anytime message: Sie sollten niemals einer formspec-Übermittlung vertrauen. Ein böswilliger Client kann jederzeit alles übermitteln, was er will - auch wenn Sie ihm nie die den formspec gezeigt haben. Das bedeutet, dass Sie die Berechtigungen prüfen sollten und sicherstellen, dass sie die Aktion durchführen dürfen. --- ## Einleitung
Ofen-Inventar
Screenshot eines formspec für Öfen, beschriftet.
In diesem Kapitel werden wir lernen, wie man einen Formspec erstellt und ihn dem Benutzer anzeigt. Ein Formspec ist der Spezifikationscode für ein Form. In Minetest sind Forms Fenster wie das Spielerinventar und können eine eine Vielzahl von Elementen wie Beschriftungen, Schaltflächen und Felder enthalten. Beachten Sie, dass Sie, wenn Sie keine Benutzereingaben benötigen, zum Beispiel wenn Sie nur Informationen für den Spieler bereitstellen möchten, Sie die Verwendung von [Heads Up Display (HUD)](hud.html)-Elementen anstelle von Formspecs verwenden sollten, da unerwartete Fenster das Spielgeschehen stören können. - [Reale oder Legacy-Koordinaten](#reale-oder-legacy-koordinaten) - [Anatomie eines formspecs](#anatomie-eines-a-formspecs) - [Elemente](#elemente) - [Header](#header) - [Ratespiel](#ratespiel) - [Padding und Abstände](#padding-und-abstände) - [Empfang von Formspec-Übermittlungen](#empfang-von-formspec-übermittlungen) - [Contexts](#contexts) - [Formspec-Quellen](#formspec-quellen) - [Node Meta Formspecs](#node-meta-formspecs) - [Spieler Inventar Formspecs](#spieler-inventar-formspecs) - [Sie sind dran](#sie-sind-dran) ## Reale oder Legacy-Koordinaten In älteren Versionen von Minetest waren die Formspecs inkonsistent. Wegen der Art und Weise, wie verschiedene Elemente auf unerwartete Art und Weise positioniert wurden war es schwierig, die Platzierung der Elemente vorherzusagen und auszurichten. Minetest 5.1.0 enthält eine Funktion namens Koordinaten, die dieses Problem durch die Einführung eines konsistenten Koordinatensystem beheben. Die Verwendung von realen Koordinaten wird dringend empfohlen, und deshalb dieses Kapitel ausschließlich diese verwenden. Die Verwendung einer formspec_version von 2 oder höher aktiviert reale Koordinaten. ## Anatomie eines formspecs ### Elemente Formspec ist eine domänenspezifische Sprache mit einem ungewöhnlichen Format. Sie besteht aus einer Reihe von Elementen mit der folgenden Form: type[param1;param2] Der Elementtyp wird deklariert und dann werden alle Parameter in eckigen Klammern angegeben. Mehrere Elemente können miteinander verbunden werden, oder auf mehrere Zeilen verteilt werden, etwa so: foo[param1]bar[param1] bo[param1] Elemente sind Elemente wie Textfelder oder Schaltflächen oder können Metadaten wie Größe oder Hintergrund sein. Sie sollten nachschlagen in der [lua_api.txt 🇬🇧](https://minetest.gitlab.io/minetest/formspec/) für eine Liste aller möglichen Elemente. ### Header Der Header eines Formspec enthält Informationen, die zuerst erscheinen müssen. Diese umfasst die Größe des Formspec, die Position, den Anker und ob das spielweite Thema angewendet werden soll. Die Elemente im Header müssen in einer bestimmten Reihenfolge definiert werden, sonst wird ein Fehler angezeigt. Diese Reihenfolge ist im obigen Absatz angegeben und - wie immer - in der Lua-API-Referenz dokumentiert. Die Größe wird in Formspec-Slots angegeben - eine Maßeinheit, die etwa 64 Pixeln entspricht, jedoch abhängig von der Bildschirmdichte und den Skalierungs Einstellungen des Clients ist. Hier ist ein formspec mit der Größe "2,2": formspec_version[4] size[2,2] Beachten Sie, dass wir die Formspec-Sprachversion ausdrücklich definiert haben müssen. Ohne dies wird stattdessen das Altsystem verwendet - was die Verwendung der konsistenten Elementpositionierung und anderer neuer Funktionen verhindert. Die Elemente position und anchor werden verwendet, um das Formspec auf dem Bildschirm zu platzieren. Die Position legt fest, wo auf dem Bildschirm das formspec sein wird, und ist standardmäßig auf die Mitte (`0,5,0,5`). Der Anker legt fest, wo auf dem formspec die Position ist, so dass Sie das Formspec mit dem Rand des Bildschirms ausrichten können. Das Formspec kann auf diese Weise links vom Bildschirm platziert werden: formspec_version[4] size[2,2] position[0,0.5] anchor[0,0.5] Dadurch wird der Anker an den linken mittleren Rand des Formspec-Feldes gesetzt, und die Position dieses Ankers auf der linken Seite des Bildschirms. ## Ratespiel
Rate-Formspec
Das Ratespiel formspec.
Der beste Weg, etwas zu lernen, ist, etwas zu machen, also lasst uns ein Ratespiel machen. Das Prinzip ist einfach: Die Mod entscheidet sich für eine Zahl, und der Spieler errät die Zahl. Die Mod sagt dann, ob die erratene Zahl höher oder niedriger ist als die tatsächliche Zahl. Zunächst erstellen wir eine Funktion, die den Formspec-Code erzeugt. Es ist gute Praxis, dies zu tun, da es die Wiederverwendung an anderer Stelle erleichtert.
```lua guessing = {} function guessing.get_formspec(name) -- TODO: Anzeige, ob die letzte Schätzung höher oder niedriger war local text = "Ich denke an eine Zahl... Raten Sie mal!" local formspec = { "formspec_version[4]", "size[6,3.476]", "label[0.375,0.5;", minetest.formspec_escape(text), "]", "field[0.375,1.25;5.25,0.8;nummer;Nummer;]", "button[1.5,2.3;3,0.8;raten;Raten]" } -- table.concat ist schneller als String-Verkettung - `..` return table.concat(formspec, "") end ``` Im obigen Code platzieren wir ein Feld, eine Beschriftung und eine Schaltfläche. Ein Feld erlaubt die Eingabe von Text und eine Schaltfläche dient zum Absenden des Forms. Sie werden feststellen, dass die Elemente sorgfältig positioniert sind, um Padding und Abstände hinzuzufügen, was später erklärt wird. Als Nächstes wollen wir dem Spieler erlauben, den Formspec anzuzeigen. Der beste Weg, dies zu tun ist die Verwendung von `show_formspec`: ```lua function guessing.show_to(name) minetest.show_formspec(name, "guessing:game", guessing.get_formspec(name)) end minetest.register_chatcommand("game", { func = function(name) guessing.show_to(name) end, }) ``` Die Funktion `show_formspec` akzeptiert einen Spielernamen, den Namen der Formspec und den Formspec selbst. Der Formspec-Name sollte ein gültiger Itemname sein, d.h. im Format `Modname:Gegenstandsname`. ### Padding und Abstände
Padding und Abstände
The guessing game formspec.
Padding ist der Abstand zwischen dem Rand des Formspec und seinem Inhalt oder zwischen nicht verwandten Elementen, dargestellt in Rot. Abstand ist der Abstand zwischen zusammenhängenden Elementen, der blau dargestellt wird. Ein Padding von `0,375` und ein Abstand von `0,25` sind üblich.
### Empfang von Formspec-Übermittlungen Wenn `show_formspec` aufgerufen wird, wird der formspec an den Client gesendet, um angezeigt zu werden. Damit Formspecs nützlich sind, müssen Informationen vom Client zum Server zurückgeschickt werden. Die Methode dafür heißt formspec field submission, und für `show_formspec` wird diese Übermittlung über einen globalen Callback empfangen: ```lua minetest.register_on_player_receive_fields(function(player, formname, fields) if formname ~= "guessing:game" then return end if fields.guess then local pname = player:get_player_name() minetest.chat_send_all(pname .. " riet " .. fields.number) end end) ``` Die in `minetest.register_on_player_receive_fields` angegebene Funktion wird jedes Mal aufgerufen, wenn ein Benutzer ein Formular absendet. Die meisten Callbacks müssen den an die Funktion übergebenen Formularnamen überprüfen und beenden, wenn es sich nicht um das richtige Form handelt; einige müssen jedoch möglicherweise für mehrere Formulare oder für alle Formulare funktionieren. Der Parameter `fields` der Funktion ist eine Tabelle mit den vom Benutzer übermittelten Werten Benutzer übermittelten Werte, die durch Zeichenketten indiziert sind. Benannte Elemente erscheinen in dem Feld unter ihrem eigenen Namen, aber nur, wenn sie für das Ereignis, das die Übermittlung verursacht hat, relevant sind. Ein Schaltflächenelement erscheint beispielsweise nur dann in Feldern, wenn die betreffende Schaltfläche gedrückt wurde. {% include notice.html notice=page.submit_vuln %} Der formspec wird also an den Client gesendet, und der Client sendet Informationen zurück. Der nächste Schritt besteht darin, den Zielwert irgendwie zu generieren und zu speichern, und die die formspec auf der Grundlage von Schätzungen zu aktualisieren. Dies geschieht mit Hilfe eines Konzepts namens "contexts". ### Contexts In vielen Fällen möchten Sie, dass minetest.show_formspec Informationen an den Callback weitergeben, die nicht an den Client gesendet werden sollen. Dies könnte beinhalten dass ein Chat-Befehl aufgerufen wurde, oder worum es in dem Dialog geht. In diesem Fall, der Zielwert, der gespeichert werden muss. Ein Context ist eine pro-Spieler-Tabelle zum Speichern von Informationen, und die Contexts für alle Online-Spieler werden in einer dateilokalen Variablen gespeichert: ```lua local _contexts = {} local function get_context(name) local context = _contexts[name] or {} _contexts[name] = context return context end minetest.register_on_leaveplayer(function(spieler) _contexts[spieler:get_player_name()] = nil end) ``` Als nächstes müssen wir den Show-Code ändern, um den Context zu aktualisieren zu aktualisieren, bevor der Formspec angezeigt wird: ```lua function guessing.show_to(name) local context = get_context(name) context.target = context.target or math.random(1, 10) local fs = guessing.get_formspec(name, context) minetest.show_formspec(name, "guessing:game", fs) end ``` Wir müssen auch den Code für die Generierung von Formularen ändern, um den Context zu verwenden: ```lua function guessing.get_formspec(name, context) local text if not context.guess then text = "I'm thinking of a number... Make a guess!" elseif context.guess == context.target then text = "Hurray, you got it!" elseif context.guess > context.target then text = "Too high!" else text = "Too low!" end ``` Beachten Sie, dass es gute Praxis ist, wenn `get_formspec` den Context nur liest und überhaupt nicht zu aktualisiert. Dies kann die Funktion einfacher machen, und auch leichter zu testen. Und schließlich müssen wir den Handler aktualisieren, um den Context mit der Vermutung zu aktualisieren: ```lua if fields.guess then local name = spieler:get_player_name() local context = get_context(name) context.guess = tonumber(fields.number) guessing.show_to(name) end ``` ## Formspec-Quellen Es gibt drei verschiedene Möglichkeiten, wie ein Formspec an den Client übermittelt werden kann: * [show_formspec](#ratespiel): Bei der oben beschriebenen Methode werden die Felder durch `register_on_player_receive_fields` empfangen. * [Node Meta Formspecs](#node-meta-formspecs): der Node enthält in seinen Metadaten eine Formularvorgabe, und der Client zeigt es *sofort* an, wenn der Spieler mit der rechten Maustaste klickt. Felder werden durch eine Methode in der Node-Definition namens `on_receive_fields` empfangen. * [Player Inventory Formspecs](#spieler-inventar-formspecs): der Formspec wird irgendwann an den Client gesendet und dann sofort angezeigt, wenn der Spieler auf "i" drückt. Felder werden durch `register_on_player_receive_fields` empfangen. ### Node Meta Formspecs `minetest.show_formspec` ist nicht die einzige Möglichkeit, einen Formspec anzuzeigen; Sie können auch Formspecs zu den Metadaten eines [Nodes](node_metadata.html) hinzufügen. Zum Beispiel, wird dieses bei Truhen verwendet, um ein schnelleres Öffnen zu ermöglichen - man muss nicht darauf warten, dass der Server dem Spieler den Formspec für die Truhe schickt. ```lua minetest.register_node("meinemod:rechtsclick", { description = "Rechtsclicke me!", tiles = {"mymod_rechtsclick.png"}, groups = {cracky = 1}, after_place_node = function(pos, placer) -- Diese Funktion wird ausgeführt, wenn das Kisten-Node platziert wird. -- Der folgende Code setzt den formspec für Kiste. -- Meta ist eine Möglichkeit, Daten in einem Node zu speichern. local meta = minetest.get_meta(pos) meta:set_string("formspec", "formspec_version[4]" .. "size[5,5]" .. "label[1,1;Dies wird beim Rechtsklick angezeigt]" .. "field[1,2;2,1;x;x;]") end, on_receive_fields = function(pos, formname, fields, spieler) if fields.quit then return end print(fields.x) end }) ``` Auf diese Weise eingestellte Formspecs lösen nicht denselben Callback aus. Um Formulareingaben für meta formspecs zu erhalten, müssen Sie einen `on_receive_fields`-Eintrag bei der Registrierung des Nodes enthalten. Diese Art von Callback wird ausgelöst, wenn Sie die Eingabetaste in einem Feld drücken, was mit `minetest.show_formspec` unmöglich ist; diese Art von Formular kann jedoch nur durch Rechtsklick auf einen Knoten angezeigt werden. Sie kann nicht programmatisch ausgelöst werden. ### Spieler Inventar Formspecs Der Formspec für das Spielerinventar wird angezeigt, wenn der Spieler auf i drückt. Der globale Callback wird verwendet, um Ereignisse von diesem Formspec zu empfangen, und der formname ist `""`. Es gibt eine Reihe von verschiedenen Mods, die es ermöglichen, das das Spielerinventar anzupassen. Die offiziell empfohlene Mod ist [Simple Fast Inventory (sfinv)](https://github.com/rubenwardy/sfinv/blob/master/Tutorial.md), und ist in Minetest Game enthalten. Ich als Übersetzer empfehle jedoch eher [i3](https://github.com/minetest-mods/i3) oder [unified inventory](https://github.com/minetest-mods/unified_inventory) ### Sie sind dran * Erweitern Sie das Ratespiel, um die höchste Punktzahl jedes Spielers zu ermitteln, wobei die höchste Punktzahl angibt, wie viele Ratschläge nötig waren. * Erstellen Sie einen Node namens "Inbox", in dem Benutzer ein Formspec öffnen und Nachrichten hinterlassen können. Dieser Node sollte den Namen des Placers als `owner` in der Meta speichern, und sollte `show_formspec` verwenden, um verschiedene Formspecs für verschiedene Spieler anzuzeigen.