JavaFX8 TableView не обновляется с помощью List из вызова RestTemplate в запланированной службе

Я достаточно новичок в Java и JavaFX, и я разрабатываю приложение JavaFX и использую REST api внутри задачи javafx.concurrent.ScheduledService для обновления TableView. Я могу получить ответ от моей службы SpringBoot REST и вижу, что данные устанавливаются в объекте модели, ObservableList, а также в самом TableView, но таблица все еще остается пустой.

Конечная точка SpringBoot Api:

 @RequestMapping(value = "/getAllActive", method = RequestMethod.GET)
public List<IssuedTicket> getAllActive () {
    List<IssuedTicket> issuedTicketList = issuedTicketService.findAll();
    return issuedTicketList;
}

Ответ JSON от конечной точки выше:

[{"id":2,"ticketId":1230717013545,"dateArrived":"23-07-17","timeArrived":"01:35:45","deviceId":1},{"id":3,"ticketId":1230717013552,"dateArrived":"23-07-17","timeArrived":"01:35:52","deviceId":1},{"id":4,"ticketId":1230717013556,"dateArrived":"23-07-17","timeArrived":"01:35:56","deviceId":1}]

В моем приложении JavaFX:

HomeSceneController.java

package com.ronintech.bayTrans.ui.main;

import com.jfoenix.controls.JFXButton;
import com.jfoenix.controls.JFXTextField;
import com.ronintech.bayTrans.model.ActiveTickets;
import com.ronintech.bayTrans.utils.RestErrorHandler;
import javafx.collections.FXCollections;
import javafx.collections.ObservableList;
import javafx.concurrent.ScheduledService;
import javafx.concurrent.Task;
import javafx.event.ActionEvent;
import javafx.fxml.FXML;
import javafx.fxml.Initializable;
import javafx.scene.control.TableColumn;
import javafx.scene.control.TableView;
import javafx.scene.control.cell.PropertyValueFactory;
import javafx.scene.layout.AnchorPane;
import javafx.util.Duration;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.http.*;
import org.springframework.web.client.RestTemplate;

import java.net.URL;
import java.util.List;
import java.util.ResourceBundle;

public class HomeSceneController implements Initializable{

@FXML
private AnchorPane homeAnchorPane;

@FXML
private JFXTextField ticketTxt;

@FXML
private JFXButton scanTicketBtn;

@FXML
private TableView<ActiveTickets> activeTicketsTable;

@FXML
private TableColumn<ActiveTickets, Long > ticketIdCol;

@FXML
private TableColumn<ActiveTickets, String > dateArrivedCol;

@FXML
private TableColumn<ActiveTickets, String> timeArrivedCol;

private ObservableList<ActiveTickets> activeTicketsList;

private static final Logger LOGGER = LoggerFactory.getLogger(HomeSceneController.class);
private static final String ACTIVE_TICKETS_URL = "http://localhost:9090/api/ticket/getAllActive";

@Override
public void initialize(URL location, ResourceBundle resources) {
    getActiveTickets.reset();
    getActiveTickets.setPeriod(Duration.seconds(20));
    getActiveTickets.start();

    getActiveTickets.setOnSucceeded(event -> {
        LOGGER.info("onSuccess");
        List<ActiveTickets> activeList = getActiveTickets.getValue();
        activeTicketsList = FXCollections.observableArrayList(activeList);

        ticketIdCol.setCellValueFactory(new PropertyValueFactory<ActiveTickets, Long>("ticketId"));
        dateArrivedCol.setCellValueFactory(new PropertyValueFactory<ActiveTickets, String>("dateArrived"));
        timeArrivedCol.setCellValueFactory(new PropertyValueFactory<ActiveTickets, String>("timeArrived"));
        activeTicketsTable.setItems(activeTicketsList);

        LOGGER.info("Items in Table");
        LOGGER.info(activeTicketsTable.getItems().toString());
    });

    getActiveTickets.setOnFailed(event -> {
        LOGGER.error("service task FAILED");
    });
}

@FXML
void openTicketModal(ActionEvent event) {
}

private ScheduledService<List<ActiveTickets>> getActiveTickets = new ScheduledService<List<ActiveTickets>>() {
    @Override
    protected Task<List<ActiveTickets>> createTask() {
        return new Task<List<ActiveTickets>>() {
            @Override
            protected List<ActiveTickets> call() throws Exception {
                LOGGER.info("Scheduled Service getActiveTickets STARTED");
                RestTemplate restTemplate = new RestTemplate();
                restTemplate.setErrorHandler(new RestErrorHandler());

                List<ActiveTickets> response = restTemplate.getForObject(ACTIVE_TICKETS_URL, List.class);

                LOGGER.info(response.toString);
                return response;
            }
        };
    }
};

}

Мой объект модели: ActiveTickets.java

package com.ronintech.bayTrans.model;

import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import javafx.beans.property.SimpleIntegerProperty;
import javafx.beans.property.SimpleLongProperty;
import javafx.beans.property.SimpleStringProperty;

@JsonIgnoreProperties(ignoreUnknown = true)
public class ActiveTickets {

private SimpleLongProperty id;
private SimpleLongProperty ticketId;
private SimpleStringProperty dateArrived;
private SimpleStringProperty timeArrived;
private SimpleIntegerProperty deviceId;

public ActiveTickets(long id,
                     long ticketId,
                     String dateArrived,
                     String timeArrived,
                     int deviceId) {
    this.id = new SimpleLongProperty(id);
    this.ticketId = new SimpleLongProperty(ticketId);
    this.dateArrived = new SimpleStringProperty(dateArrived);
    this.timeArrived = new SimpleStringProperty(timeArrived);
    this.deviceId = new SimpleIntegerProperty(deviceId);
}

public long getId() {
    return id.get();
}

public SimpleLongProperty idProperty() {
    return id;
}

public long getTicketId() {
    return ticketId.get();
}

public SimpleLongProperty ticketIdProperty() {
    return ticketId;
}

public String getDateArrived() {
    return dateArrived.get();
}

public SimpleStringProperty dateArrivedProperty() {
    return dateArrived;
}

public String getTimeArrived() {
    return timeArrived.get();
}

public SimpleStringProperty timeArrivedProperty() {
    return timeArrived;
}

public int getDeviceId() {
    return deviceId.get();
}

public SimpleIntegerProperty deviceIdProperty() {
    return deviceId;
}

public void setId(long id) {
    this.id.set(id);
}

public void setTicketId(long ticketId) {
    this.ticketId.set(ticketId);
}

public void setDateArrived(String dateArrived) {
    this.dateArrived.set(dateArrived);
}

public void setTimeArrived(String timeArrived) {
    this.timeArrived.set(timeArrived);
}

public void setDeviceId(int deviceId) {
    this.deviceId.set(deviceId);
}

@Override
public String toString() {
    return "ActiveTickets{" +
            "ticketId=" + ticketId +
            ", dateArrived=" + dateArrived +
            ", timeArrived=" + timeArrived +
            '}';
}
}

Ответ на вызов RestTemplate:

00:43:06.901 [Thread-9] INFO com.ronintech.bayTrans.ui.main.HomeSceneController - [{id=2, ticketId=1230717013545, dateArrived=23-07-17, timeArrived=01:35:45, deviceId=1}, {id=3, ticketId=1230717013552, dateArrived=23-07-17, timeArrived=01:35:52, deviceId=1}, {id=4, ticketId=1230717013556, dateArrived=23-07-17, timeArrived=01:35:56, deviceId=1}]

Журнал после установки ObservableList в TableView:

activeTicketsTable.setItems(activeTicketsList);

00:43:07.159 [JavaFX Application Thread] INFO com.ronintech.bayTrans.ui.main.HomeSceneController - [{id=2, ticketId=1230717013545, dateArrived=23-07-17, timeArrived=01:35:45, deviceId=1}, {id=3, ticketId=1230717013552, dateArrived=23-07-17, timeArrived=01:35:52, deviceId=1}, {id=4, ticketId=1230717013556, dateArrived=23-07-17, timeArrived=01:35:56, deviceId=1}]

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

TableView

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

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

ИЗМЕНИТЬ

Моя структура TableView в FXML:

<TableView fx:id="activeTicketsTable" prefHeight="800.0" prefWidth="800.0" tableMenuButtonVisible="true">
    <columns>
        <TableColumn fx:id="ticketIdCol" prefWidth="75.0" text="Ticket ID"/>
        <TableColumn fx:id="dateArrivedCol" prefWidth="75.0" text="Date Arrived"/>
        <TableColumn fx:id="timeArrivedCol" prefWidth="75.0" text="Time Arrived"/>
    </columns>
    <columnResizePolicy>
        <TableView fx:constant="CONSTRAINED_RESIZE_POLICY"/>
    </columnResizePolicy>
</TableView>

ОБНОВЛЕНИЕ

Похоже, что маршалинг происходит неправильно.

Это след отладки ObservableList activeTicketsList:

activeTicketsList

Список был задан как ArrayList для LinkedHashMaps '

Тогда как когда я вручную создаю ObservableList:

private final ObservableList<ActiveTickets> data = FXCollections.observableArrayList(
        new ActiveTickets(1,1234,"date1","time1",1),
        new ActiveTickets(2,5678,"date2","time2",2)
);

"данные" правильно настроены как ArrayList объектов класса ActiveTickets.

данные

Как можно убедиться, что маршалинг JSON -> Object выполняется правильно в RestTemplate?


person VRN.dev    schedule 23.07.2017    source источник
comment
На снимке экрана показано, что в таблице действительно есть элементы (пустая таблица имеет заполнитель вместо строк). Таким образом, все журналы и снимки экрана согласуются с тем, что фабрика значений ячеек не задана для столбцов. Те, которые вы показываете в коде, выглядят правильно; я предполагаю, что каким-то образом экземпляры столбцов таблицы, для которых вы вызываете setCellValueFactory(...), каким-то образом не являются правильными экземплярами столбцов таблицы.   -  person James_D    schedule 24.07.2017
comment
Да, таблица устанавливается со списком, предоставленным Сервисом, но похоже, что маршалинг объектов JSON- ›идет не так. Есть какие-нибудь советы, как это исправить?   -  person VRN.dev    schedule 24.07.2017
comment
Посмотрите, помогает ли stackoverflow.com/questions/8108887/. Вашему классу ActiveTickets почти наверняка нужен конструктор без аргументов.   -  person James_D    schedule 24.07.2017
comment
Это похоже на то, что сказал Джеймс_Д, добавьте конструктор без аргументов и используйте либо ParameterizedTypeReference, либо просто маршалинг в массив вместо списка. Чтобы это сработало, вам необходимо выполнить оба шага.   -  person Jai    schedule 25.07.2017


Ответы (1)


Я не уверен, что это действительно проблема, но обычно мои клиентские проекты REST не маршалируются напрямую в список (это может быть проблема Джексона). Вот что бы я сделал:

ActiveTickets[] response = restTemplate.getForObject(ACTIVE_TICKETS_URL, ActiveTickets[].class);

return Arrays.asList(response);

Обратите внимание, что вы все равно вернете список в конечной точке; вам не нужно это менять.

Обновлять

Прочтите это, чтобы узнать другой способ сделать это.

person Jai    schedule 24.07.2017
comment
Пробовал, но служба переходит в состояние FAILED. 11: 45: 51.658 [Поток приложения JavaFX] ОШИБКА com.ronintech.bayTrans.ui.main.HomeSceneController - служебная задача НЕ ВЫПОЛНЕНА. Но я согласен с предположением, что при маршалинге из JSON должна быть какая-то проблема - ›Объект - person VRN.dev; 24.07.2017
comment
@ VRN.dev говорит ли stacktrace о том, где и что пошло не так? - person Jai; 24.07.2017
comment
@ VRN.dev, кстати, вашему ActiveTickets, вероятно, понадобится пустой конструктор по умолчанию. - person Jai; 24.07.2017