Несколько 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);
}
Ответ 2
с помощью Flow API DataFX вы можете вводить данные в экземпляры вашего контроллера с помощью CDI: