Как правильно обновить узлы пользовательского интерфейса при изменении состояния объекта в JavaFX? - PullRequest
3 голосов
/ 03 апреля 2019

Наличие java-класса под названием Player, например:

public class Player {
    private String playerName;
    private int score;

    public Player(String playerName) {
        this.playerName = playerName;
        this.score = 0;
    }

    public void incrementScore(int by) {
        this.score+=by;
    }

    public String getPlayerName() {
        return this.playerName;
    }

}

Я просто хотел узнать, какой правильный способ обновить пользовательский интерфейс, подобный предоставленному, после того, как экземпляр Player изменит свой счет:

public class PlayerUI extends Application {

    public static void main(String[] args) {
        launch();
    }

    @Override
    public void start(Stage primaryStage) throws Exception {
        Player player = new Player("Mario");

        VBox box = new VBox();
        box.setSpacing(20.);
        box.setPadding(new Insets(20.));

        Text playerName = new Text("Player: " + player.getPlayerName());
        Text score = new Text("Score: " + player.getPlayerScore());

        Button incrementScore = new Button("Score up");
        incrementScore.setOnAction(new IncrementScoreHandler(player, score));

        box.getChildren().add(playerName);
        box.getChildren().add(score);
        box.getChildren().add(incrementScore);

        Scene mainScene = new Scene(box);
        primaryStage.setScene(mainScene);
        primaryStage.setResizable(false);

        primaryStage.show();
    }

}

Я сделал это с помощью EventHandler, который получает игрока и вовлеченный узел в качестве параметра в конструкторе.

public class IncrementScoreHandler implements EventHandler<ActionEvent> {
    private Player player;
    private Text scoreText;

    public IncrementScoreHandler(Player player, Text score) {
        this.player = player;
        this.scoreText = score;
    }

    @Override
    public void handle(ActionEvent event) {
        player.incrementScore(1);
        this.scoreText.setText("Score: " + player.getPlayerScore());

    }

}

Это правильно?Лучше ли провести рефакторинг Player и позволить playerName и получить оценку StringProperty и IntegerProperty?

1 Ответ

4 голосов
/ 04 апреля 2019

Да, вы хотели бы реорганизовать класс Player для использования наблюдаемых свойств.

Тогда вам просто нужно связать ваши Text узлы (или Label узлы в приведенном ниже примере)к этим свойствам.

import javafx.application.Application;
import javafx.beans.property.IntegerProperty;
import javafx.beans.property.SimpleIntegerProperty;
import javafx.beans.property.SimpleStringProperty;
import javafx.beans.property.StringProperty;
import javafx.geometry.Insets;
import javafx.geometry.Pos;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.control.Label;
import javafx.scene.layout.GridPane;
import javafx.scene.layout.VBox;
import javafx.stage.Stage;

public class PropertiesExample extends Application {

    public static void main(String[] args) {
        launch(args);
    }

    @Override
    public void start(Stage primaryStage) {

        // Simple interface
        VBox root = new VBox(5);
        root.setPadding(new Insets(10));
        root.setAlignment(Pos.CENTER);

        // Create our Player
        Player player = new Player("Mario");

        // For this example, a simple GridPane will hold our nodes
        GridPane gridPane = new GridPane();

        // Add our data headers
        gridPane.add(new Label("Player:"), 0, 0);
        gridPane.add(new Label("Score:"), 0, 1);

        // Our labels to hold the changeable data
        Label lblPlayerName = new Label();
        Label lblPlayerScore = new Label();
        gridPane.add(lblPlayerName, 1, 0);
        gridPane.add(lblPlayerScore, 1, 1);

        // Use the Player properties to bind our displayed values to the Labels
        lblPlayerName.textProperty().bind(player.playerNameProperty());
        lblPlayerScore.textProperty().bind(player.scoreProperty().asString());

        Button btnIncrementScore = new Button("Score Up");
        btnIncrementScore.setOnAction(event -> player.incrementScore());

        root.getChildren().addAll(gridPane, btnIncrementScore);

        // Show the Stage
        primaryStage.setWidth(300);
        primaryStage.setHeight(300);
        primaryStage.setScene(new Scene(root));
        primaryStage.show();
    }
}

class Player {
    private StringProperty playerName = new SimpleStringProperty();
    private IntegerProperty score = new SimpleIntegerProperty();

    public Player(String playerName) {
        this.playerName.set(playerName);
        this.score.set(0);
    }

    public void incrementScore() {
        // Add 1 point to our score IntegerProperty
        this.score.set(score.get() + 1);
        System.out.println(score.get());
    }

    public String getPlayerName() {
        return playerName.get();
    }

    public StringProperty playerNameProperty() {
        return playerName;
    }

    public void setPlayerName(String playerName) {
        this.playerName.set(playerName);
    }

    public int getScore() {
        return score.get();
    }

    public IntegerProperty scoreProperty() {
        return score;
    }

    public void setScore(int score) {
        this.score.set(score);
    }
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...