diff options
4 files changed, 133 insertions, 29 deletions
diff --git a/server/src/main/java/com/vaadin/ui/AbstractDateField.java b/server/src/main/java/com/vaadin/ui/AbstractDateField.java index bfe3119f26..67067bd070 100644 --- a/server/src/main/java/com/vaadin/ui/AbstractDateField.java +++ b/server/src/main/java/com/vaadin/ui/AbstractDateField.java @@ -48,6 +48,7 @@ import com.vaadin.event.FieldEvents.BlurNotifier; import com.vaadin.event.FieldEvents.FocusEvent; import com.vaadin.event.FieldEvents.FocusListener; import com.vaadin.event.FieldEvents.FocusNotifier; +import com.vaadin.server.ErrorMessage; import com.vaadin.server.UserError; import com.vaadin.shared.Registration; import com.vaadin.shared.ui.datefield.AbstractDateFieldServerRpc; @@ -95,10 +96,6 @@ public abstract class AbstractDateField<T extends Temporal & TemporalAdjuster & if ("".equals(newDateString)) { newDate = null; - // TODO check if the following 3 lines are necessary - hasChanges = !getState(false).parsable; - getState().parsable = true; - currentParseErrorMessage = null; } else { newDate = reconstructDateFromFields(resolutions, oldDate); } @@ -108,32 +105,26 @@ public abstract class AbstractDateField<T extends Temporal & TemporalAdjuster & if (hasChanges) { dateString = newDateString; + currentParseErrorMessage = null; if (newDateString == null || newDateString.isEmpty()) { - getState().parsable = true; - currentParseErrorMessage = null; - setComponentError(null); setValue(newDate, true); } else { if (invalidDateString) { Result<T> parsedDate = handleUnparsableDateString( dateString); - parsedDate.ifOk(v -> { - getState().parsable = true; - currentParseErrorMessage = null; - setValue(v, true); - }); + parsedDate.ifOk(v -> setValue(v, true)); if (parsedDate.isError()) { dateString = null; - getState().parsable = false; currentParseErrorMessage = parsedDate .getMessage().orElse("Parsing error"); - setComponentError( - new UserError(getParseErrorMessage())); - setValue(null, true); + + if (!isDifferentValue(null)) { + doSetValue(null); + } else { + setValue(null, true); + } } } else { - getState().parsable = true; - currentParseErrorMessage = null; setValue(newDate, true); } } @@ -526,6 +517,7 @@ public abstract class AbstractDateField<T extends Temporal & TemporalAdjuster & */ @Override public void setValue(T value) { + currentParseErrorMessage = null; /* * First handle special case when the client side component have a date * string but value is null (e.g. unparsable date string typed in by the @@ -702,14 +694,21 @@ public abstract class AbstractDateField<T extends Temporal & TemporalAdjuster & // validator? ValidationResult result = validator.apply(value, new ValueContext(this, this)); + if (result.isError()) { currentParseErrorMessage = getDateOutOfRangeMessage(); } + + getState().parsable = currentParseErrorMessage == null; + + ErrorMessage errorMessage; if (currentParseErrorMessage == null) { - setComponentError(null); + errorMessage = null; } else { - setComponentError(new UserError(currentParseErrorMessage)); + errorMessage = new UserError(currentParseErrorMessage); } + setComponentError(errorMessage); + updateResolutions(); } diff --git a/uitest/src/main/java/com/vaadin/tests/components/datefield/DateFieldSetAfterInvalid.java b/uitest/src/main/java/com/vaadin/tests/components/datefield/DateFieldSetAfterInvalid.java new file mode 100644 index 0000000000..6d7baefb86 --- /dev/null +++ b/uitest/src/main/java/com/vaadin/tests/components/datefield/DateFieldSetAfterInvalid.java @@ -0,0 +1,36 @@ +package com.vaadin.tests.components.datefield; + +import java.time.LocalDate; + +import com.vaadin.server.VaadinRequest; +import com.vaadin.tests.components.AbstractTestUI; +import com.vaadin.ui.Button; +import com.vaadin.ui.DateField; + +public class DateFieldSetAfterInvalid extends AbstractTestUI { + + @Override + protected String getTestDescription() { + return "DateField to programatically change the value after having invalid text"; + } + + @Override + protected Integer getTicketNumber() { + return 9763; + } + + @Override + protected void setup(VaadinRequest request) { + DateField dateField = new DateField(); + addComponent(dateField); + + Button nowButton = new Button("Today"); + nowButton.addClickListener(e -> dateField.setValue(LocalDate.now())); + addComponent(nowButton); + + Button clearButton = new Button("Clear"); + clearButton.addClickListener(e -> dateField.clear()); + addComponent(clearButton); + } + +} diff --git a/uitest/src/main/java/com/vaadin/tests/components/datefield/DisabledParentLayout.java b/uitest/src/main/java/com/vaadin/tests/components/datefield/DisabledParentLayout.java index 847a25174a..914c7b490f 100644 --- a/uitest/src/main/java/com/vaadin/tests/components/datefield/DisabledParentLayout.java +++ b/uitest/src/main/java/com/vaadin/tests/components/datefield/DisabledParentLayout.java @@ -30,22 +30,15 @@ public class DisabledParentLayout extends AbstractReindeerTestUI { @Override protected void setup(VaadinRequest request) { - VerticalLayout content = new VerticalLayout(); - - content.setSpacing(true); - content.setMargin(true); final VerticalLayout pane = new VerticalLayout(); AbstractLocalDateField dateField = new TestDateField(); pane.addComponent(dateField); - content.addComponent(pane); - Button button = new Button("Test"); button.addClickListener(event -> pane.setEnabled(!pane.isEnabled())); - content.addComponent(button); - addComponent(content); + addComponents(pane, button); } @Override @@ -55,7 +48,7 @@ public class DisabledParentLayout extends AbstractReindeerTestUI { @Override protected Integer getTicketNumber() { - return 13124; + return 4773; } } diff --git a/uitest/src/test/java/com/vaadin/tests/components/datefield/DateFieldSetAfterInvalidTest.java b/uitest/src/test/java/com/vaadin/tests/components/datefield/DateFieldSetAfterInvalidTest.java new file mode 100644 index 0000000000..5e25c55cc6 --- /dev/null +++ b/uitest/src/test/java/com/vaadin/tests/components/datefield/DateFieldSetAfterInvalidTest.java @@ -0,0 +1,76 @@ +package com.vaadin.tests.components.datefield; + +import static java.time.temporal.ChronoUnit.DAYS; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertTrue; + +import java.time.LocalDate; + +import org.junit.Test; + +import com.vaadin.testbench.By; +import com.vaadin.testbench.elements.ButtonElement; +import com.vaadin.testbench.elements.DateFieldElement; +import com.vaadin.tests.tb3.MultiBrowserTest; + +public class DateFieldSetAfterInvalidTest extends MultiBrowserTest { + + private static final org.openqa.selenium.By ERROR_INDICATOR_BY = By + .className("v-errorindicator"); + + private static String INVALID_TEXT = "abc"; + + @Test + public void setValueAfterBeingInvalid() { + openTestURL(); + + DateFieldElement dateField = $(DateFieldElement.class).first(); + dateField.setDate(LocalDate.now().minus(5, DAYS)); + assertNoErrorIndicator(); + + dateField.setValue(dateField.getValue() + INVALID_TEXT); + assertErrorIndicator(); + + $(ButtonElement.class).caption("Today").first().click(); + + assertFalse(dateField.getValue().endsWith(INVALID_TEXT)); + assertNoErrorIndicator(); + } + + @Test + public void clearAfterBeingInvalid() { + openTestURL(); + + DateFieldElement dateField = $(DateFieldElement.class).first(); + dateField.setDate(LocalDate.now().minus(5, DAYS)); + assertNoErrorIndicator(); + + dateField.setValue(dateField.getValue() + INVALID_TEXT); + assertErrorIndicator(); + + $(ButtonElement.class).caption("Clear").first().click(); + + assertTrue(dateField.getValue().isEmpty()); + assertNoErrorIndicator(); + + dateField.setValue(INVALID_TEXT); + assertErrorIndicator(); + } + + @Test + public void invalidTypedText() { + openTestURL(); + + DateFieldElement dateField = $(DateFieldElement.class).first(); + dateField.setValue(INVALID_TEXT); + assertErrorIndicator(); + } + + private void assertErrorIndicator() { + assertElementPresent(ERROR_INDICATOR_BY); + } + + private void assertNoErrorIndicator() { + assertElementNotPresent(ERROR_INDICATOR_BY); + } +} |