Scala - Holen Sie sich FXML UI-Elemente

Ich arbeite an einem Scala / JavaFX-Projekt mit IntelliJ und einem Plugin für Scala.
Der typische Weg in Java, auf Elemente Ihrer fxml-Datei zuzugreifen, besteht darin, für jedes Element, auf das Sie zugreifen möchten, eine ID festzulegen und dann in Ihrer Controller-Klasse eine Variable wie "@FXML private Label mylabelid;" zu deklarieren.
In Scala funktioniert es fast genauso wie in meinem Quellcode. ABER ich erhalte eine NullPointerException in Zeile 73 (markiert mit einem PROBLEM-Kommentar). Dies ist die einzige Bezeichnung, die nur auf null gesetzt ist. Ich habe verschiedene Dinge ausprobiert. Jedes einzelne Element wird wie erwartet mit Ausnahme von missingInputLabel gesetzt.
LoginView.fxml:

<?xml version="1.0" encoding="UTF-8"?>
<?import java.net.*?>
<?import javafx.scene.control.*?>
<?import javafx.scene.layout.*?>
<GridPane xmlns:fx="http://javafx.com/fxml/1" fx:controller="logingui.LoginController">
    <stylesheets>
        <URL value="@/data/gtevStyle.css" />
    </stylesheets>

    <Label text="SSH-Nutzername:" GridPane.columnIndex="0" GridPane.rowIndex="0" />
    <TextField fx:id="sshNameField" promptText="SSH-Nutzername" onKeyReleased="#onKeyReleased_textField" GridPane.columnIndex="1" GridPane.rowIndex="0" />
    <Label text="SSH-Passwort:" GridPane.columnIndex="0" GridPane.rowIndex="1" />
    <PasswordField fx:id="sshPasswdField" promptText="SSH-Passwort" onKeyReleased="#onKeyReleased_textField" GridPane.columnIndex="1" GridPane.rowIndex="1" />
    <Label text="Datenbank-Nutzername:" GridPane.columnIndex="0" GridPane.rowIndex="2" />
    <TextField fx:id="databaseNameField" promptText="Datenbank-Nutzername" onKeyReleased="#onKeyReleased_textField" GridPane.columnIndex="1" GridPane.rowIndex="2" />
    <Label text="Datenbank-Passwort:" GridPane.columnIndex="0" GridPane.rowIndex="3" />
    <PasswordField fx:id="databasePasswdField" promptText="Datenbank-Passwort" onKeyReleased="#onKeyReleased_textField" GridPane.columnIndex="1" GridPane.rowIndex="3" />

    <Label text="Name oder Passwort falsch" fx:id="nameOrPasswdWrong" styleClass="warningLabel" GridPane.columnIndex="1" GridPane.rowIndex="4" />
    <Label text="Fehlende Angaben" fx:id="missingInputLabel" styleClass="warningLabel" GridPane.columnIndex="1" GridPane.rowIndex="5" />
    <Button text="Login" onAction="#login" GridPane.columnIndex="0" GridPane.rowIndex="5" />
</GridPane>

LoginController.scala:

package logingui

import java.net.URL
import java.sql.SQLException
import java.util.ResourceBundle
import java.util.logging.Level
import java.util.logging.Logger
import javafx.fxml.FXML
import javafx.fxml.Initializable
import javafx.scene.control.Label
import javafx.scene.control.PasswordField
import javafx.scene.control.TextField
import javafx.scene.input.KeyEvent
import javafx.scene.input.KeyCode
import connection.GTEVConnection
import javafx.stage.Stage

class LoginController extends Initializable {
  @FXML
  private var nameOrPasswdWrong: Label = _
  @FXML
  private var missingInputLabel: Label = _ //TODO missingInputLabel is set to null?!
  @FXML
  private var sshNameField: TextField = _
  @FXML
  private var databaseNameField: TextField = _
  @FXML
  private var sshPasswdField: PasswordField = _
  @FXML
  private var databasePasswdField: PasswordField = _
  //Um einfach auf alle Text-/Passwortfelder referenzieren zu können
  private var textFields: Array[TextField] = Array[TextField](sshNameField, databaseNameField, sshPasswdField, databasePasswdField)
  var callbackWhenFinished: () => Unit = () => GTEVConnection.close() //Default: Simply close GTEVConnection

  def onKeyReleased_textField(ev: KeyEvent) { //TODO declare def as private and mark with @FXML?
    if (ev.getCode == KeyCode.ENTER) {
      login()
    } else {
      nameOrPasswdWrong.setVisible(false)
    }
  }

  def login() {
    checkInputOfTextFields()
    try {
      if (!missingInputLabel.isVisible) {
        GTEVConnection.createConnection(sshNameField.getText, sshPasswdField.getText, databaseNameField.getText, databasePasswdField.getText)
        callbackWhenFinished()
        sshNameField.getScene.getWindow.asInstanceOf[Stage].close()
      }
    } catch {
      case ex: SQLException =>
        Logger.getLogger("LoginContollerLogger").log(Level.SEVERE, null, ex) //TODO Loggername durch Klassennamen ersetzen
        nameOrPasswdWrong.setVisible(true)
    }
  }

  private def checkInputOfTextFields() {
    var anyMissingInput: Boolean = false
    for (t: TextField <- textFields) {
      if (t.getText.isEmpty) {
        anyMissingInput = true
        t.getStyleClass.add("missingInput")
      } else {
        t.getStyleClass.remove("missingInput")
      }
      missingInputLabel.setVisible(anyMissingInput)
    }
  }

  override def initialize(location: URL, resources: ResourceBundle) {
    nameOrPasswdWrong.setVisible(false)
    missingInputLabel.setVisible(false) //PROBLEM: NullPointerException
  }
}

Sag mir, wenn ich etwas verpasst habe oder du etwas anderes wissen willst und danke für deine Hilfe.
Greetings Tracker

Antworten auf die Frage(2)

Ihre Antwort auf die Frage