Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Test for hidden value workarounds for checkbox inputs so they can sen… #609

Open
wants to merge 2 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 4 additions & 1 deletion README.md
Original file line number Diff line number Diff line change
Expand Up @@ -850,8 +850,11 @@ of all cases covered:
unchecked.
- if there's more than one checkbox with the same `name` attribute, they are
all treated collectively as a single form control, which returns the value
as an **array** containing all the values of the selected checkboxes in the
as an **array** containing all the values of the checkboxes in the
collection.
- a hidden input with same name before the checkbox is allowed which is a
common workaround to allow browsers to send `false` for unchecked
checkboxes.
- `<input type="radio">` elements are all grouped by the `name` attribute, and
such a group treated as a single form control. This form control returns the
value as a **string** corresponding to the `value` attribute of the selected
Expand Down
15 changes: 15 additions & 0 deletions src/__tests__/to-have-form-values.js
Original file line number Diff line number Diff line change
Expand Up @@ -189,6 +189,21 @@ describe('.toHaveFormValues', () => {
}).toThrowError(/must be of the same type/)
})

it('allows a checkbox and a hidden input which is a common workaround so forms can send "false" for a checkbox', () => {
const {container} = render(`
<form>
<input type="hidden" name="checkbox-with-hidden-false" value=0>
<input type="checkbox" name="checkbox-with-hidden-false" value=1>
</form>
`)
const form = container.querySelector('form')
expect(() => {
expect(form).toHaveFormValues({
'checkbox-with-hidden-false': ['0', '1'],
})
}).not.toThrowError()
})

it('detects multiple elements with the same type and name', () => {
const {container} = render(`
<form>
Expand Down
15 changes: 12 additions & 3 deletions src/to-have-form-values.js
Original file line number Diff line number Diff line change
Expand Up @@ -11,9 +11,18 @@ import {
function getMultiElementValue(elements) {
const types = [...new Set(elements.map(element => element.type))]
if (types.length !== 1) {
throw new Error(
'Multiple form elements with the same name must be of the same type',
)
if (
types.length === 2 &&
types[0] === 'hidden' &&
types[1] === 'checkbox'
) {
// Allow the special case where there's a 'checkbox' input, and a matching 'hidden' input
// before it, which works around browser forms so a 'false' value is submitted.
} else {
throw new Error(
'Multiple form elements with the same name must be of the same type',
)
}
}
switch (types[0]) {
case 'radio': {
Expand Down