在 Vaadin 中将字符串设置为 LocalDate 转换器 Table

Setting a String to LocalDate converter in Vaadin Table

继 "Learning Vaadin 7, Second Edition" 这本书之后,我现在正尝试在 table 中显示简单的 beans。然而,本书只涵盖了旧java.util.Dateclass的用法。我正在尝试使用转换器显示 LocalDate 属性。

我要显示的 bean(人):

public class Person {

private long id;
private String firstName;
private String lastName;
private LocalDate birthdate;
private Gender gender;

// .. GETTERS & SETTERS 

我写了一个 LocalDateToStringConverter,实现了 com.vaadin.data.util.converter.Converter。

package be.kapture.converters;

import com.vaadin.data.util.converter.Converter;

import java.time.LocalDate;
import java.time.format.DateTimeFormatter;
import java.time.format.DateTimeParseException;
import java.util.Locale;

public class LocalDateToStringConverter implements Converter<String, LocalDate> {

    @Override
    public LocalDate convertToModel(String value, Class<? extends LocalDate> targetType, Locale locale) throws ConversionException {
        DateTimeFormatter formatter = DateTimeFormatter.ofPattern("dd-MM-yyyy");
        try {
            return LocalDate.parse(value, formatter);

        } catch (DateTimeParseException ex) {
            return null;
        }
    }

    @Override
    public String convertToPresentation(LocalDate value, Class<? extends String> targetType, Locale locale) throws ConversionException {
        return value.toString();
    }

    @Override
    public Class<LocalDate> getModelType() {
        return LocalDate.class;
    }

    @Override
    public Class<String> getPresentationType() {
        return String.class;
    }
}

在 UI 中,这是我设置转换器的代码片段:

...
Table table = new Table("", container);
table.setConverter("birthdate", new LocalDateToStringConverter());

verticalLayout.addComponent(table);

'container' 是一个 BeanItemContainer,我在其中放置了一些示例 Person 对象。 在我的浏览器中访问 Vaadin 应用程序时,出现以下异常:

jun 27, 2016 1:56:45 PM org.apache.catalina.core.StandardWrapperValve invoke
SEVERE: Servlet.service() for servlet [HelloVaadinServlet] in context with path [] threw exception [com.vaadin.server.ServiceException: com.vaadin.ui.Table$CacheUpdateException: Error during Table cache update. Additional causes not shown.] with root cause
com.vaadin.data.util.converter.Converter$ConversionException: Unable to convert value of type java.time.LocalDate to presentation type class java.lang.String. No converter is set and the types are not compatible.
    at com.vaadin.data.util.converter.ConverterUtil.convertFromModel(ConverterUtil.java:116)
    at com.vaadin.ui.AbstractField.convertFromModel(AbstractField.java:736)
    at com.vaadin.ui.AbstractField.convertFromModel(AbstractField.java:721)
    at com.vaadin.ui.AbstractField.setPropertyDataSource(AbstractField.java:657)
    at com.vaadin.ui.Table.bindPropertyToField(Table.java:4140)
    at com.vaadin.ui.Table.getPropertyValue(Table.java:4109)
    at com.vaadin.ui.Table.parseItemIdToCells(Table.java:2386)
    at com.vaadin.ui.Table.getVisibleCellsNoCache(Table.java:2225)
    at com.vaadin.ui.Table.refreshRenderedCells(Table.java:1745)
    at com.vaadin.ui.Table.refreshRowCache(Table.java:2691)
    at com.vaadin.ui.Table.containerItemSetChange(Table.java:4587)
    at com.vaadin.data.util.AbstractContainer.fireItemSetChange(AbstractContainer.java:242)
    at com.vaadin.data.util.AbstractInMemoryContainer.fireItemsAdded(AbstractInMemoryContainer.java:1012)
    at com.vaadin.data.util.AbstractInMemoryContainer.fireItemAdded(AbstractInMemoryContainer.java:994)
    at com.vaadin.data.util.AbstractInMemoryContainer.internalAddItemAtEnd(AbstractInMemoryContainer.java:884)
    at com.vaadin.data.util.AbstractBeanContainer.addItem(AbstractBeanContainer.java:533)
    at com.vaadin.data.util.AbstractBeanContainer.addBean(AbstractBeanContainer.java:598)
    at com.vaadin.data.util.BeanItemContainer.addItem(BeanItemContainer.java:227)
    at be.kapture.MyUI.init(MyUI.java:88)
    at com.vaadin.ui.UI.doInit(UI.java:682)
    at com.vaadin.server.communication.UIInitHandler.getBrowserDetailsUI(UIInitHandler.java:214)
    at com.vaadin.server.communication.UIInitHandler.synchronizedHandleRequest(UIInitHandler.java:74)
    at com.vaadin.server.SynchronizedRequestHandler.handleRequest(SynchronizedRequestHandler.java:41)
    at com.vaadin.server.VaadinService.handleRequest(VaadinService.java:1409)
    at com.vaadin.server.VaadinServlet.service(VaadinServlet.java:364)
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:729)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:292)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:207)
    at org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:52)
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:240)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:207)
    at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:212)
    at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:106)
    at org.apache.catalina.authenticator.AuthenticatorBase.invoke(AuthenticatorBase.java:502)
    at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:141)
    at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:79)
    at org.apache.catalina.valves.AbstractAccessLogValve.invoke(AbstractAccessLogValve.java:616)
    at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:88)
    at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:528)
    at org.apache.coyote.http11.AbstractHttp11Processor.process(AbstractHttp11Processor.java:1099)
    at org.apache.coyote.AbstractProtocol$AbstractConnectionHandler.process(AbstractProtocol.java:672)
    at org.apache.tomcat.util.net.AprEndpoint$SocketProcessor.doRun(AprEndpoint.java:2508)
    at org.apache.tomcat.util.net.AprEndpoint$SocketProcessor.run(AprEndpoint.java:2497)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
    at org.apache.tomcat.util.threads.TaskThread$WrappingRunnable.run(TaskThread.java:61)
    at java.lang.Thread.run(Thread.java:745)

为什么不注册 Converter 来进行 String 到 LocalDate 的转换(如果需要,反之亦然)?我试过在 Table.setConverter 方法中使用匿名内部 class 作为参数,结果相同。我在这里做错了什么?

更新(根据要求):

这是完整的 UI 代码,其中使用了 Table、容器和转换器。注意:这是一个没有实际目标的 "HelloVaadin" 沙盒项目。它是专门为像这个问题这样的目的而设置的,试图将 Java 8 的 LocalDate 集成到 Vaadin 项目中。

package be.kapture;

import be.kapture.converters.LocalDateToDateConverter;
import be.kapture.converters.LocalDateToStringConverter;
import be.kapture.entities.Person;
import be.kapture.util.CustomFieldGroupFieldFactory;
import com.vaadin.annotations.*;
import com.vaadin.data.fieldgroup.FieldGroup;
import com.vaadin.data.util.BeanItem;
import com.vaadin.data.util.BeanItemContainer;
import com.vaadin.server.VaadinRequest;
import com.vaadin.server.VaadinServlet;
import com.vaadin.ui.*;

import javax.servlet.annotation.WebServlet;
import java.time.LocalDate;
import java.util.Arrays;

import static com.vaadin.data.Property.ValueChangeListener;


@Theme("mytheme")
@Widgetset("be.kapture.MyAppWidgetset")
@PreserveOnRefresh
@Title("Hello Vaadin!")

public class MyUI extends UI implements Window.CloseListener {

    private static final Person person1 = new Person(1L, "John", "DOE", LocalDate.of(70, 1, 1));
    private static final Person person2 = new Person(2L, "Jane", "doe", LocalDate.of(70, 1, 1));
    private static final Person person3 = new Person(3L, "jules", "winnf", LocalDate.of(48, 11, 21));
    private static final Person person4 = new Person(4L, "vincent", "Vega", LocalDate.of(54, 2, 17));

    private static final BeanItemContainer<Person> container = new BeanItemContainer<>(Person.class);

    static {
        container.addAll(Arrays.asList(person1, person2, person3, person4));
    }

    private final VerticalLayout verticalLayout = new VerticalLayout();

    @Override
    protected void init(VaadinRequest vaadinRequest) {

        Person person = new Person(1L);
        person.setFirstName("John");
        person.setLastName("Doe");
        person.setBirthdate(LocalDate.now());

        BeanItem<Person> beanItem = new BeanItem<>(person);

        FieldGroup group = new FieldGroup(beanItem);
        group.setFieldFactory(new CustomFieldGroupFieldFactory());

        Field<?> id = group.buildAndBind("id");
        Field<?> firstName = group.buildAndBind("firstName");
        Field<?> lastName = group.buildAndBind("lastName");

        Field<?> birthdate = group.buildAndBind("birthdate");

        Field<?> gender = group.buildAndBind("gender");

//        birthdate.setConverter(new LocalDateToDateConverter());
//        birthdate.setPropertyDataSource(item.getItemProperty("birthdate"));
//        FormLayout layout = new FormLayout(id, firstName, lastName,
//                birthdate);
//        layout.setMargin(true);
//        setContent(layout);

        verticalLayout.setMargin(true);
        verticalLayout.setSpacing(true);

        verticalLayout.addComponents(id, firstName, lastName, birthdate, gender);

         // Define a person which cannot exist
        Person nullPerson = new Person(-1L);
        nullPerson.setFirstName("Test");
        container.addItem(nullPerson);
        final ListSelect select = new ListSelect("", container);

        // Send events on directly when clicked
        select.setImmediate(true);

        // Handle the value of the person as null
        select.setNullSelectionItemId(nullPerson);
        select.setItemCaptionPropertyId("firstName");
        select.addValueChangeListener((ValueChangeListener) event -> System.out.println(select.getValue()));

        verticalLayout.addComponent(select);

        Table table = new Table("");
        table.setEditable(true);
        table.setConverter(LocalDateToDateConverter.class);
        table.setContainerDataSource(container);

        verticalLayout.addComponent(table);

        setContent( verticalLayout);
    }

    @Override
    public void windowClose(Window.CloseEvent e) {
        Notification.show("Window closed.");
    }

    @WebServlet(urlPatterns = "/*", name = "MyUIServlet", asyncSupported = true)
    @VaadinServletConfiguration(ui = MyUI.class, productionMode = false)
    public static class MyUIServlet extends VaadinServlet {
    }
}

我很好奇并想出了你的问题。 setConverter(Object, Converter) 方法只允许更改该列中的值向用户显示的方式(文本表示)。这就是为什么第二个参数的类型是 Converter<java.lang.String,?>.

在您的示例中,您在 table 中启用了编辑功能。这需要 Vaadin 知道它将如何为您的 LocalDate 列提供 table 单元格编辑器。默认情况下,table 不知道 LocalDate。我知道您有 2 个选项:

  1. 实施 converter factory,returns Converter LocalDateString。然后 table 组件能够显示一个文本字段,您可以在其中根据转换器中的格式输入日期。我没有尝试用户输入无效字符串时发生的情况。
  2. 在 table 组件上实施 table field factory and call setTableFieldFactory。现场工厂会显示 PopupDateField 或类似的东西。这样你是类型安全的并且可以使用内置的日期字段。

IMO 后者会是一个更好的用户体验,但当然它在开发中需要更多的努力。