Skip to content

Commit 5060fa4

Browse files
committed
fix(pat validation): With depending date validation also validate the dependend date.
Fixes: #970
1 parent c4e00aa commit 5060fa4

File tree

2 files changed

+43
-30
lines changed

2 files changed

+43
-30
lines changed

src/pat/validation/validation.js

+19-12
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ export default Base.extend({
5757
for (const [cnt, input] of this.inputs.entries()) {
5858
// Cancelable debouncer.
5959
const debouncer = utils.debounce((e) => {
60-
this.check_input(input, e);
60+
this.check_input({ input: input, event: e });
6161
}, this.options.delay);
6262

6363
events.add_event_listener(
@@ -82,12 +82,12 @@ export default Base.extend({
8282
this.el,
8383
"submit",
8484
`pat-validation--blur-${input.name}--${cnt}--validator`,
85-
(e) => this.check_input(input, e) // immediate check with submit. Otherwise submit is not cancelable.
85+
(e) => this.check_input({ input: input, event: e }) // immediate check with submit. Otherwise submit is not cancelable.
8686
);
8787
}
8888
},
8989

90-
check_input(input, e) {
90+
check_input({ input, event, stop = false }) {
9191
if (input.disabled) {
9292
// No need to check disabled inputs.
9393
return;
@@ -97,7 +97,7 @@ export default Base.extend({
9797
this.set_validity({ input: input, msg: "" });
9898
const validity_state = input.validity;
9999

100-
if (e.submitter?.hasAttribute("formnovalidate")) {
100+
if (event?.submitter?.hasAttribute("formnovalidate")) {
101101
// Do not submit when a button with ``formnovalidate`` was used.
102102
return;
103103
}
@@ -139,6 +139,8 @@ export default Base.extend({
139139

140140
let not_after;
141141
let not_before;
142+
let not_after_el;
143+
let not_before_el;
142144
const date = new Date(input.value);
143145
if (isNaN(date)) {
144146
// Should not happen or input only partially typed in.
@@ -149,9 +151,8 @@ export default Base.extend({
149151
not_after = new Date(input_options.not.after);
150152
if (isNaN(not_after)) {
151153
// Handle value as selector
152-
not_after = document.querySelector(
153-
input_options.not.after
154-
)?.value;
154+
not_after_el = document.querySelector(input_options.not.after);
155+
not_after = not_after_el?.value;
155156
not_after =
156157
not_after &&
157158
new Date(
@@ -167,9 +168,8 @@ export default Base.extend({
167168
not_before = new Date(input_options.not.before);
168169
if (isNaN(not_before)) {
169170
// Handle value as selector
170-
not_before = document.querySelector(
171-
input_options.not.before
172-
)?.value;
171+
not_before_el = document.querySelector(input_options.not.before);
172+
not_before = not_before_el?.value;
173173
not_before =
174174
not_before &&
175175
new Date(
@@ -185,6 +185,13 @@ export default Base.extend({
185185
} else if (not_before && date < not_before) {
186186
this.set_validity({ input: input, msg: msg });
187187
}
188+
// always check the other input to clear/set errors
189+
!stop && // do not re-check when stop is set to avoid infinite loops
190+
not_after_el &&
191+
this.check_input({ input: not_after_el, stop: true });
192+
!stop &&
193+
not_before_el &&
194+
this.check_input({ input: not_before_el, stop: true });
188195
}
189196

190197
if (!validity_state.customError) {
@@ -248,9 +255,9 @@ export default Base.extend({
248255
}
249256
}
250257

251-
if (e.type === "submit") {
258+
if (event?.type === "submit") {
252259
// Do not submit in error case.
253-
e.preventDefault();
260+
event.preventDefault();
254261
}
255262
this.set_error_message(input, input_options);
256263
},

src/pat/validation/validation.test.js

+24-18
Original file line numberDiff line numberDiff line change
@@ -824,15 +824,6 @@ describe("pat-validation", function () {
824824
inp_end.value = "2020-10-05";
825825
inp_end.dispatchEvent(events.change_event());
826826
await utils.timeout(1); // wait a tick for async to settle.
827-
expect(el.querySelectorAll("em.warning").length).toBe(1);
828-
expect(el.querySelectorAll("em.warning")[0].textContent).toBe(
829-
"The end date must on or before the start date."
830-
);
831-
832-
// Violate the before/after constraint
833-
inp_start.value = "2020-10-06";
834-
inp_start.dispatchEvent(events.change_event());
835-
await utils.timeout(1); // wait a tick for async to settle.
836827
expect(el.querySelectorAll("em.warning").length).toBe(2);
837828
expect(el.querySelectorAll("em.warning")[0].textContent).toBe(
838829
"The start date must on or before the end date."
@@ -849,6 +840,18 @@ describe("pat-validation", function () {
849840
await utils.timeout(1); // wait a tick for async to settle.
850841
expect(el.querySelectorAll("em.warning").length).toBe(0);
851842

843+
// Violate the before/after constraint
844+
inp_start.value = "2020-10-11";
845+
inp_start.dispatchEvent(events.change_event());
846+
await utils.timeout(1); // wait a tick for async to settle.
847+
expect(el.querySelectorAll("em.warning").length).toBe(2);
848+
expect(el.querySelectorAll("em.warning")[0].textContent).toBe(
849+
"The start date must on or before the end date."
850+
);
851+
expect(el.querySelectorAll("em.warning")[1].textContent).toBe(
852+
"The end date must on or before the start date."
853+
);
854+
852855
// Fulfill the before/after constraint - start before end
853856
inp_start.value = "2020-10-01";
854857
inp_start.dispatchEvent(events.change_event());
@@ -947,15 +950,6 @@ describe("pat-validation", function () {
947950
inp_end.value = "2022-01-05T09:00";
948951
inp_end.dispatchEvent(events.change_event());
949952
await utils.timeout(1); // wait a tick for async to settle.
950-
expect(el.querySelectorAll("em.warning").length).toBe(1);
951-
expect(el.querySelectorAll("em.warning")[0].textContent).toBe(
952-
"The end date/time must on or before the start date/time."
953-
);
954-
955-
// Violate the before/after constraint
956-
inp_start.value = "2022-01-05T11:00";
957-
inp_start.dispatchEvent(events.change_event());
958-
await utils.timeout(1); // wait a tick for async to settle.
959953
expect(el.querySelectorAll("em.warning").length).toBe(2);
960954
expect(el.querySelectorAll("em.warning")[0].textContent).toBe(
961955
"The start date/time must on or before the end date/time."
@@ -972,6 +966,18 @@ describe("pat-validation", function () {
972966
await utils.timeout(1); // wait a tick for async to settle.
973967
expect(el.querySelectorAll("em.warning").length).toBe(0);
974968

969+
// Violate the before/after constraint
970+
inp_start.value = "2022-01-05T11:00";
971+
inp_start.dispatchEvent(events.change_event());
972+
await utils.timeout(1); // wait a tick for async to settle.
973+
expect(el.querySelectorAll("em.warning").length).toBe(2);
974+
expect(el.querySelectorAll("em.warning")[0].textContent).toBe(
975+
"The start date/time must on or before the end date/time."
976+
);
977+
expect(el.querySelectorAll("em.warning")[1].textContent).toBe(
978+
"The end date/time must on or before the start date/time."
979+
);
980+
975981
// Fulfill the before/after constraint - start before end
976982
inp_start.value = "2022-01-04T10:00";
977983
inp_start.dispatchEvent(events.change_event());

0 commit comments

Comments
 (0)