JavaFX Как правильно обрабатывать события при использовании пользовательских фабрик ячеек?

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

У меня есть A.fxml, AController.java (класс контроллера для A.fxml). У меня есть TableView с определенными фабриками ячеек. Я предпочитаю определять свои фабрики ячеек в отдельном классе, чтобы я мог использовать их в случае необходимости. Я предпочитаю писать весь мой код обработки событий в моем классе контроллера. Но если я использую собственные фабрики ячеек, то я вынужден написать обработку событий в самом классе фабрики ячеек.

Есть ли способ обработки пользовательских событий фабрики ячеек в самом классе моего контроллера? или хотя бы просто выбросить событие из пользовательского класса фабрики ячеек в мой класс контроллера и обработать?

Заранее спасибо.

2 ответа

Вы можете передать объект на фабрику, которая определяет, когда следует открыть контекстное меню, и который готовит меню.

Пример:

public interface CellContextMenuProvider<S, T> {

    /**
     * Prepares the context menu for opening.
     * @param cell the cell the menu was requested for
     * @param menu the menu to prepare
     */
    public void prepareContextMenu(TableCell<S, T> cell, ContextMenu menu);

    /**
     * Checks, if a cell continaing a certain item should have an active context
     * menu.
     * @param empty if the cell is empty
     * @param item the item of the cell
     * @return {@literal true} iff the context menu should be enabled.
     */
    public boolean enableContextMenu(boolean empty, T item);

    /**
     * Prepares the intial menu. This menu must not be empty, otherwise it won't
     * be shown when it's requested for the first time.
     * @param menu the menu to prepare
     */
    public void prepareInitialContextMenu(ContextMenu menu);

}
public class CellFactory<S, T> implements Callback<TableColumn<S, T>, TableCell<S, T>> {

    private final CellContextMenuProvider<S, T> menuProvider;
    private final ContextMenu contextMenu;

    public CellFactory(@NamedArg("menuProvider") CellContextMenuProvider<S, T> menuProvider) {
        this.menuProvider = menuProvider;

        if (menuProvider == null) {
            this.contextMenu = null;
        } else {
            this.contextMenu = new ContextMenu();
            menuProvider.prepareInitialContextMenu(contextMenu);
        }

        this.menuEventHandler = evt -> {
            if (this.contextMenu != null) {
                TableCell<S, T> source = (TableCell<S, T>) evt.getSource();
                this.menuProvider.prepareContextMenu(source, this.contextMenu);
            }
        };
    }

    public CellFactory() {
        this(null);
    }

    private final EventHandler<ContextMenuEvent> menuEventHandler;

    @Override
    public TableCell<S, T> call(TableColumn<S, T> param) {
        TableCell<S, T> result = new TableCell<S, T>() {

            @Override
            protected void updateItem(T item, boolean empty) {
                super.updateItem(item, empty);
                setText(Objects.toString(item, ""));
                setContextMenu(menuProvider != null && menuProvider.enableContextMenu(empty, item) ? contextMenu : null);
            }

        };
        result.setOnContextMenuRequested(menuEventHandler);
        if (menuProvider != null && menuProvider.enableContextMenu(true, null)) {
            result.setContextMenu(contextMenu);
        }
        return result;
    }

}
public class AController {

    @FXML
    private TableView<Item<Integer>> table;

    public void initialize() {
        for (int i = 0; i < 100; i++) {
            table.getItems().add(new Item<>(i));
        }
    }    

    public CellContextMenuProvider<Item<Integer>, Integer> getMenuProvider() {
        return new CellContextMenuProvider<Item<Integer>, Integer>() {

            private final MenuItem item = new MenuItem("Say Hello World");

            {
                item.setOnAction(evt -> System.out.println("Hello World"));
            }

            @Override
            public void prepareContextMenu(TableCell<Item<Integer>, Integer> cell, ContextMenu menu) {
            }

            @Override
            public void prepareInitialContextMenu(ContextMenu menu) {
                menu.getItems().setAll(item);
            }

            @Override
            public boolean enableContextMenu(boolean empty, Integer item) {
                // only for odd items
                return !empty && (item % 2) != 0;
            }

        };
    }

}

A.fxml

<TableView fx:id="table" prefHeight="400.0" prefWidth="600.0" xmlns="http://javafx.com/javafx/8.0.60" xmlns:fx="http://javafx.com/fxml/1" fx:controller="fxml.table.AController">
  <columns>
      <TableColumn prefWidth="159.0" text="C1">
          <cellValueFactory>
              <PropertyValueFactory property="value" />
          </cellValueFactory>
          <cellFactory>
              <CellFactory menuProvider="$controller.menuProvider"/>
          </cellFactory>
      </TableColumn>
  </columns>
</TableView>

Примечание: если контекстное меню всегда одинаково, вы также можете добавить EventHandler свойства к фабрике и использовать их как, например, onAction атрибуты для кнопок, которые позволят вам передавать обработчики событий контроллера, что приведет к сокращению / упрощению кода.

Не было бы возможно дать вашей фабрике клеток FunctionalInterface в качестве параметра при обработке вашего события?(не уверен, что это хорошая идея)

Я представляю ваш код следующим образом:

контроллер:

myTableView.setCellFactory(new MyOwnCellFactory<>(() -> { 
    // event handling 
}));

MyOwnCellFactory:

public MyOwnCellFactory(MyFunctionalInterface myInterface) {
    functionalInterface = myInterface;
}

// something something

functionalInterface.handleEvent();

FunctionalInterface:

@FunctionalInterface
public interface MyFunctionalInterface {
    public void handleEvent();
}

Не уверен, правильно ли я понимаю вашу идею. Не проверял код, просто выписал его из головы.

Другие вопросы по тегам