ComboBox不会观察SimpleObjectProperty中所做的更改

时间:2016-10-27 22:18:17

标签: java javafx javafx-8

假设我有这样简单的域POJO类:

package sample;

    public class Item {

        private String name;
        private String state;

        public Item() {}

        public Item(String name, String state) {
            this.name = name;
            this.state = state;
        }

        public String getName() {
            return name;
        }

        public void setName(String name) {
            this.name = name;
        }

        public String getState() {
            return state;
        }

        public void setState(String state) {
            this.state = state;
        }
  }

并假设我有一个限制:我不允许对此课程进行任何更改。所以我需要写一个包装器,对吧?

package sample;

    import javafx.beans.property.ObjectProperty;
    import javafx.beans.property.SimpleObjectProperty;

    public class ItemWrapper {

        private ObjectProperty<Item> itemProperty = new SimpleObjectProperty<>();

        private Item item;


        public ItemWrapper(Item item) {
            setItem(item);
        }

        public Item getItem() {
            return itemProperty.get();
        }

        public void setItem(Item item) {
            this.item = item;
            itemProperty.set(item);
        }

        public ObjectProperty<Item> itemProperty(){
            return itemProperty;
        }

    }

然后假设我有一个小的JavaFX应用程序,可以显示项目列表,并有可能更改项目的状态。我们来看看吧。

Main.java

package sample;

    import javafx.application.Application;
    import javafx.fxml.FXMLLoader;
    import javafx.scene.Parent;
    import javafx.scene.Scene;
    import javafx.stage.Stage;

    public class Main extends Application {

        @Override
        public void start(Stage primaryStage) throws Exception{
            Parent root = FXMLLoader.load(getClass().getResource("sample.fxml"));
            primaryStage.setTitle("Item View");
            primaryStage.setScene(new Scene(root, 200, 100));
            primaryStage.show();
        }


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

Sample.fxml

<?xml version="1.0" encoding="UTF-8"?>

  <?import javafx.scene.control.Button?>
  <?import javafx.scene.control.ComboBox?>
  <?import javafx.scene.control.Label?>
  <?import javafx.scene.layout.AnchorPane?>

  <AnchorPane maxHeight="-Infinity" maxWidth="-Infinity" minHeight="-Infinity" minWidth="-Infinity" prefHeight="108.0" prefWidth="225.0" xmlns="http://javafx.com/javafx/8.0.60" xmlns:fx="http://javafx.com/fxml/1" fx:controller="sample.Controller">
     <children>
        <Label layoutX="14.0" layoutY="14.0" text="show item" />
        <ComboBox fx:id="cbxItemsSelector" layoutX="14.0" layoutY="31.0" prefWidth="150.0" />
        <Button fx:id="btnChangeState" layoutX="14.0" layoutY="65.0" mnemonicParsing="false" onAction="#changeItemState" prefHeight="25.0" prefWidth="85.0" text="change state" />
     </children>
  </AnchorPane>

Controller.java

package sample;

    import javafx.beans.Observable;
    import javafx.collections.FXCollections;
    import javafx.collections.ObservableList;
    import javafx.fxml.FXML;
    import javafx.scene.control.ComboBox;
    import javafx.scene.control.ListCell;
    import javafx.scene.control.ListView;
    import javafx.util.Callback;

    import java.util.ArrayList;
    import java.util.List;
    import java.util.stream.Collectors;

    public class Controller {

        @FXML
        private ComboBox<ItemWrapper> cbxItemsSelector;

        private ObservableList<ItemWrapper> items;

        public void initialize(){
            loadItems();
            customizeSelectorCellView();
        }

        private void loadItems() {
            List<Item> itemsList = new ArrayList<>();
            itemsList.add(new Item("first item","initialized"));
            itemsList.add(new Item("second item","initialized"));

            List<ItemWrapper> itemWrappers = itemsList.stream().map(ItemWrapper::new).collect(Collectors.toList());
            items = FXCollections.observableList(itemWrappers,wrapper -> new Observable[]{wrapper.itemProperty()});

            cbxItemsSelector.setItems(items);
        }

        private void customizeSelectorCellView(){


            cbxItemsSelector.setButtonCell(new ListCell<ItemWrapper>() {
                @Override
                protected void updateItem(ItemWrapper itemWrapper, boolean empty) {
                    super.updateItem(itemWrapper, empty);
                    if (empty) {
                        setText("");
                    } else {
                        setText(itemWrapper.getItem().getName());
                    }
                }
            });
            cbxItemsSelector.setCellFactory(
                    new Callback<ListView<ItemWrapper>, ListCell<ItemWrapper>>() {
                        @Override
                        public ListCell<ItemWrapper> call(ListView<ItemWrapper> p) {
                            ListCell cell = new ListCell<ItemWrapper>() {
                                @Override
                                protected void updateItem(ItemWrapper itemWrapper, boolean empty) {
                                    //System.out.println("update item");
                                    super.updateItem(itemWrapper, empty);
                                    if (empty) {
                                        setText("");
                                    } else {
                                        Item item = itemWrapper.getItem();
                                        setText(String.format("name: %s\n state: %s\n",item.getName(),item.getState()));
                                    }
                                }
                            };return cell;
                        }
                    }
            );
        }

        @FXML
        public void changeItemState(){
            ItemWrapper selectedItemWrapper =  cbxItemsSelector.getSelectionModel().getSelectedItem();
            if (selectedItemWrapper == null) return;

            Item item = selectedItemWrapper.getItem();
            item.setState("started");

        }
    }

问题是comboBox在某些项目状态更新后没有看到应用的更改。问题的原因是我只更改了对象(即项目)字段,而我没有通过setItem方法设置新项目。 所以看起来字段的变化是不可观察的。 我也知道如果我在SimpleObjectProperty类中将StringProperty替换为两个ItemWrapper(对于名称和状态字段),它将起作用。但我不想进行现场复制。如果我在Item课程中有100个字段怎么办?

我有两个问题:
1)如何让ComboBox在Item类中的任何字段中看到由其他外部方做出的更改而不进行字段重复(不创建其他字段属性)?也许我应该使用一些特殊的装订? 2)有没有办法摆脱Wrapper类而保持域Item类未经修改?

1 个答案:

答案 0 :(得分:0)

据我所知, $('input[type=text]').each(function () { $(this).change(function () { alert($(this).val()); $(this).focus(); }); }); 需要创建和注册。拿一个厕所:https://docs.oracle.com/javase/tutorial/uiswing/events/propertychangelistener.html

我希望我有所帮助。

祝你有个愉快的一天。 :)