Несколько FXML с контроллерами, общий объект

Добыча всех,

Я нашел кучу сообщений уже по этой теме, но я до сих пор не могу передать объект из Controller1 в Controller2. Есть ли где-нибудь полный учебник или какой-то пример проекта, который делает это?

Я добрался настолько далеко, пока не застрял:

Класс страны

public class Country {
private SimpleStringProperty country = new SimpleStringProperty("");

//Constructor
public Country() {
}

//GETTERS
public String getCountry() {
    return country.get();
}

//SETTERS
public void setCountry(String value) {
    country.set(value);
}

@Override
public String toString() {
    return getCountry();
}
}

При запуске программы загружается основной FXML (Sample.fxml). Он содержит панель с панелью меню на верхней панели и панель содержимого в центре. При инициализации я создаю новый объект Country и сохраняю его в глобальной переменной. У меня есть метод, который загружает другой FXML в область содержимого при нажатии элемента меню:

SampleController.java

public class SampleController implements Initializable {

@FXML
private Pane pContent;

private Country c;

@FXML
private void handleButtonAction(ActionEvent event) throws IOException {
    System.out.println(c); //this prints Belgium, which is correct

    URL url = getClass().getResource("Sub1.fxml");

    FXMLLoader fxmlloader = new FXMLLoader();
    fxmlloader.setLocation(url);
    fxmlloader.setBuilderFactory(new JavaFXBuilderFactory());

    pContent.getChildren().clear();
    pContent.getChildren().add((Node) fxmlloader.load(url.openStream()));
}

@Override
public void initialize(URL url, ResourceBundle rb) {
    c = new Country();
    c.setCountry("Belgium");
}

public Country getCountryFromSampleController(){
    return c;
}
}

Теперь я хочу захватить объект Country, когда Sub1.fxml загружается, что означает, что мне нужно получить объект country при инициализации():

Sub1Controller.java

public class Sub1Controller implements Initializable {

/**
 * Initializes the controller class.
 */
@Override
public void initialize(URL url, ResourceBundle rb) {
    SampleController sp = new SampleController(); //I don't know how to fetch the original SampleController object
    System.out.println(sp.getCountryFromSampleController()); 
    //this prints null, which is ofcourse logical because I make a new SampleController object.         
}    
}

Вопрос, который у меня есть, как я могу получить исходный объект SampleController, чтобы использовать метод getCountryFromRoot() для получения объекта Country со значением Belgium? Я искал эту проблему часами и часами и читал каждое сообщение в StackOverflow об этом, но, похоже, я не нашел недостающую ссылку... любая помощь (желательно с этим кодом) ценится!

Извините за длинный пост, я старался быть как можно более полным, иначе я никогда не пойму...

Ответ 1

FXML - это простая форма шаблона MVC. Файл FXML - это представление, контроллер явно, что пропущено? Модель - место, где вы храните все данные относительно вашего текущего вида и, таким образом, вы можете использовать для совместного использования данных страны между контроллерами.


1. Одним из возможных подходов к внедрению модели является "контекст". Давайте рассмотрим случай, тогда у вас есть только одна модель для всего проекта, чтобы вы могли иметь глобальный контекст в форме Singleton

public class Context {
    private final static Context instance = new Context();

    public static Context getInstance() {
        return instance;
    }

    private Country country = new Country();

    public Country currentCountry() {
        return country;
    }
}

Ваш SampleController будет иметь следующие изменения:

@Override
public void initialize(URL url, ResourceBundle rb) {
    Context.getInstance().currentCountry().setCountry("Belgium");
}

И SubController1 может получить к нему доступ таким же образом:

@Override
public void initialize(URL url, ResourceBundle rb) {
    System.out.println(Context.getInstance().currentCountry().getCountry());
}

2. Другой способ - передать контекст в SubController1, после чего вы загрузите его xml. Он будет работать лучше, если вы не хотите иметь глобальную модель приложения. Поэтому создайте аналогичный класс Context, но без полей экземпляра, и:

public class Sub1Controller implements Initializable {
    private Context context;
    public void setContext(Context context) {
        this.context = context;
        // initialize country dependent data here rather then in initialize()
    }
}

Настройка контекста в SampleController:

Context currentContext = new Context();

@Override
public void initialize(URL url, ResourceBundle rb) {
    currentContext.currentCountry().setCountry("Belgium");
}

@FXML
private void handleButtonAction(ActionEvent event) throws IOException {
    URL url = getClass().getResource("Sub1.fxml");

    FXMLLoader fxmlloader = new FXMLLoader();
    fxmlloader.setLocation(url);
    fxmlloader.setBuilderFactory(new JavaFXBuilderFactory());

    pContent.getChildren().clear();
    pContent.getChildren().add((Node) fxmlloader.load(url.openStream()));
            // here we go
    ((Sub1Controller)fxmlloader.getController()).setContext(currentContext);
}