Заполнение табличного представления строками

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

Допустим, у меня есть массив строк из двух столбцов (String[][]) с парами «имя, значение». Теперь я просто хочу создать табличное представление, которое показывает данные в двух столбцах, отображая имя в первом и значение во втором столбце для всех строк в исходном массиве.

Что я пробовал? Ничего, но кажется, что вам нужно создать наблюдаемые списки, по одному для каждого столбца, привязать его к соответствующему столбцу, а затем добавить столбец в табличное представление. Но это включает в себя «фабрики» во всех примерах, которые я видел, что является чуждым мне понятием.

Я предполагаю, что это просто, но я не могу обдумать это. Пожалуйста помоги.


person The Unfun Cat    schedule 31.08.2012    source источник


Ответы (3)


Фабрика значений ячеек

Для каждого столбца в таблице вы устанавливаете/создаете Фабрику значений ячеек. Каждая строка имеет соответствующий объект в tableview.getItems(). Чтобы определить, как этот объект отображается в столбцах, в каждом столбце используется собственная фабрика значений ячеек. Фабрики принимают объект и возвращают отображаемое значение.

Поскольку String[][] представляет собой массив String[], мы хотим, чтобы фабрики принимали String[] и возвращали String, соответствующие его столбцу.

Пример

Вот пример создания фабрик значений ячеек таким образом. Это немного многословно, но это можно убрать с помощью лямбда-выражений! (см. раздел Лямбда-выражения).

// ---------------------------------------Initialize the data
String[][] data = ...; // Get name/value pairs from somewhere

// ---------------------------------------Setup a TableView with two TableColumns 
    /* ... Code ... */

// ---------------------------------------Add Cell Value Factories
nameColumn.setCellValueFactory(new Callback<TableColumn.CellDataFeatures<String[], String>, ObservableValue<String>>() {
    @Override
    public ObservableValue<String> call(TableColumn.CellDataFeatures<String[], String> p) {
        String[] x = p.getValue();
        if (x != null && x.length>0) {
            return new SimpleStringProperty(x[0]);
        } else {
            return new SimpleStringProperty("<no name>");
        }
    }
});

valueColumn.setCellValueFactory(new Callback<TableColumn.CellDataFeatures<String[], String>, ObservableValue<String>>() {
    @Override
    public ObservableValue<String> call(TableColumn.CellDataFeatures<String[], String> p) {
        String[] x = p.getValue();
        if (x != null && x.length>1) {
            return new SimpleStringProperty(x[1]);
        } else {
            return new SimpleStringProperty("<no value>");
        }
    }
});

// ---------------------------------------Add Data to TableView
tableView.getItems().addAll(Arrays.asList(data));

Результат

Если вы бросите этот пример (через исходный код) на сцену, вы получите вот что. это работает!

лямбды

Java немного многословен, особенно при работе с анонимными классами. К счастью, есть лямбда-выражения, которые возвращают Java немного удобочитаемости. Вот те же фабрики значений ячеек из примера, переписанные с помощью лямбда-выражений.

nameColumn.setCellValueFactory((p)->{
        String[] x = p.getValue();
        return new SimpleStringProperty(x != null && x.length>0 ? x[0] : "<no name>");
});

valueColumn.setCellValueFactory((p)->{
        String[] x = p.getValue();
        return new SimpleStringProperty(x != null && x.length>1 ? x[1] : "<no value>");
});

Исходный код

Вот автономный класс JavaFX, который таким образом использует фабрики значений ячеек.

import java.util.Arrays;
import java.util.List;
import javafx.application.Application;
import javafx.beans.property.SimpleStringProperty;
import javafx.beans.value.ObservableValue;
import javafx.scene.Scene;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.layout.StackPane;
import javafx.stage.Stage;
import javafx.util.Callback;

/**
 *
 * @author nonfrt
 */
public class TableStuff extends Application {

    @Override
    public void start(Stage primaryStage) {

        // Create the data structure
        String[][] data = new String[5][2];
        data[0] = new String[]{"Jon Skeet","876k"};
        data[1] = new String[]{"Darin Dimitrov","670k"};
        data[2] = new String[]{"BalusC","660k"};
        data[3] = new String[]{"Hans Passant","635k"};
        data[4] = new String[]{"Marc Gravell","610k"};

        // Create the table and columns
        TableView<String[]> tv = new TableView();
            TableColumn<String[],String> nameColumn = new TableColumn();
                nameColumn.setText("Name Column");

            TableColumn<String[],String> valueColumn = new TableColumn();
                valueColumn.setText("Value Column");
        tv.getColumns().addAll(nameColumn,valueColumn);

        // Add cell value factories
//        nameColumn.setCellValueFactory((p)->{
//                String[] x = p.getValue();
//                return new SimpleStringProperty(x != null && x.length>0 ? x[0] : "<no name>");
//        });
//
//        valueColumn.setCellValueFactory((p)->{
//                String[] x = p.getValue();
//                return new SimpleStringProperty(x != null && x.length>1 ? x[1] : "<no value>");
//        });
        nameColumn.setCellValueFactory(new Callback<TableColumn.CellDataFeatures<String[], String>, ObservableValue<String>>() {
            @Override
            public ObservableValue<String> call(TableColumn.CellDataFeatures<String[], String> p) {
                String[] x = p.getValue();
                if (x != null && x.length>0) {
                    return new SimpleStringProperty(x[0]);
                } else {
                    return new SimpleStringProperty("<no name>");
                }
            }
        });

        valueColumn.setCellValueFactory(new Callback<TableColumn.CellDataFeatures<String[], String>, ObservableValue<String>>() {
            @Override
            public ObservableValue<String> call(TableColumn.CellDataFeatures<String[], String> p) {
                String[] x = p.getValue();
                if (x != null && x.length>1) {
                    return new SimpleStringProperty(x[1]);
                } else {
                    return new SimpleStringProperty("<no value>");
                }
            }
        });

        // Add Data
        tv.getItems().addAll(Arrays.asList(data));

        // Finish setting the stage
        StackPane root = new StackPane();
        root.getChildren().add(tv);

        Scene scene = new Scene(root, 300, 250);

        primaryStage.setTitle("Cell Value Factory Example");
        primaryStage.setScene(scene);
        primaryStage.show();
    }

    /**
     * @param args the command line arguments
     */
    public static void main(String[] args) {
        launch(args);
    }
}

[ПРИМЕЧАНИЕ: ответ на этот другой вопрос StackOverFlow является еще одним примером этого. ]

person NonlinearFruit    schedule 29.06.2016

Да, это было просто, ты, болван. Создайте класс Row:

Row(String fName, String fValue){
    this.fieldName = new SimpleStringProperty(fName);
    this.fieldValue = new SimpleStringProperty(fValue);
}

Для каждой строки в массиве строк вы создаете объект-строку, который добавляете в наблюдаемый список.

private ObservableList<Row> observableList = FXCollections.observableArrayList(          
              new Row("The Unfun Cat", "Is a terrible programmer"), 
              new Row("Stack Overflow", "Rules!");

Затем вы создаете tablecolumn для обоих столбцов в массиве.

TableColumn columnName = new TableColumn("Name");
columnName.setCellValueFactory(
        new PropertyValueFactory<Row,String>("fieldName"));

(идентично значению)

Затем вы добавляете observableList в tableView с помощью tableView.setItems(observableList) и, наконец, вызываете метод tableView.getColumns.addAll(fieldName,fieldValue);

(Это заставляет меня задаться вопросом, как сделать это для общего случая, когда вы не знаете, сколько столбцов находится в исходном массиве String[][]? Может ли объект строки иметь ArrayList для представления произвольного количества SimpleStringProperties? Как бы вы подключились это к ValueFactory?)

Пс. если кто-то создаст более педагогический пример, я поставлю его сообщению отметку «решено».

person The Unfun Cat    schedule 31.08.2012
comment
Я не понимаю, почему JavaFX TableView должен быть таким сложным. Разве они не могут предложить простые методы? Это действительно раздражает. Есть ли другие решения? - person SomethingSomething; 15.03.2015

Вы также можете попробовать это с картой, не зная, сколько столбцов вы добавите в таблицу! поэтому вы можете использовать ключ карты в качестве заголовка таблицы

TableView<Map<String, String>> tableView = new TableView<>();

ArrayList<Map<String, String>> valuesArray = new ArrayList<>();
for(Object object : filter.getEntities()) {
  Map<String, String> values = getValuesAsStringsForObject(object);
  valuesArray.add(values);
}

for(Object o : filter.getEntities()){
  List<String> headers = getHeaders(o);
  for (String header :headers){
    TableColumn<Map<String, String>, String> tableColumn = new TableColumn<>(header);
    tableColumn.setCellValueFactory(param -> new SimpleStringProperty(param.getValue().get(header)));

    tableView.getColumns().add(tableColumn);
  }
  break;
}

tableView.getItems().addAll(valuesArray);
return tableView;
person Yago Rodriguez    schedule 05.10.2016