Skip to content

Commit cc8721e

Browse files
SergiCLgnapse
andauthoredMar 26, 2020
feat: add exact mode option for toHaveClass (#176) (#217)
* feat: add exact mode option for toHaveClass (#176) * Update src/to-have-class.js Co-authored-by: Ernesto García <gnapse@gmail.com>
1 parent eb51c17 commit cc8721e

File tree

3 files changed

+143
-16
lines changed

3 files changed

+143
-16
lines changed
 

‎README.md

+6-1
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@ clear to read and to maintain.
4646
<!-- START doctoc generated TOC please keep comment here to allow auto update -->
4747
<!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
4848

49+
4950
- [Installation](#installation)
5051
- [Usage](#usage)
5152
- [Custom matchers](#custom-matchers)
@@ -498,7 +499,7 @@ expect(button).toHaveAttribute('type', expect.not.stringContaining('but'))
498499
### `toHaveClass`
499500

500501
```typescript
501-
toHaveClass(...classNames: string[])
502+
toHaveClass(...classNames: string[], options?: {exact: boolean})
502503
```
503504

504505
This allows you to check whether the given element has certain classes within
@@ -525,6 +526,9 @@ expect(deleteButton).toHaveClass('btn-danger btn')
525526
expect(deleteButton).toHaveClass('btn-danger', 'btn')
526527
expect(deleteButton).not.toHaveClass('btn-link')
527528

529+
expect(deleteButton).toHaveClass('btn-danger extra btn', {exact: true}) // to check if the element has EXACTLY a set of classes
530+
expect(deleteButton).not.toHaveClass('btn-danger extra', {exact: true}) // if it has more than expected it is going to fail
531+
528532
expect(noClasses).not.toHaveClass()
529533
```
530534

@@ -940,6 +944,7 @@ Thanks goes to these people ([emoji key][emojis]):
940944

941945
<!-- markdownlint-enable -->
942946
<!-- prettier-ignore-end -->
947+
943948
<!-- ALL-CONTRIBUTORS-LIST:END -->
944949

945950
This project follows the [all-contributors][all-contributors] specification.

‎src/__tests__/to-have-class.js

+104-14
Original file line numberDiff line numberDiff line change
@@ -2,21 +2,25 @@
22

33
import {render} from './helpers/test-utils'
44

5+
const renderElementWithClasses = () =>
6+
render(`
7+
<div>
8+
<button data-testid="delete-button" class="btn extra btn-danger">
9+
Delete item
10+
</button>
11+
<button data-testid="cancel-button">
12+
Cancel
13+
</button>
14+
<svg data-testid="svg-spinner" class="spinner clockwise">
15+
<path />
16+
</svg>
17+
<div data-testid="only-one-class" class="alone"></div>
18+
<div data-testid="no-classes"></div>
19+
</div>
20+
`)
21+
522
test('.toHaveClass', () => {
6-
const {queryByTestId} = render(`
7-
<div>
8-
<button data-testid="delete-button" class="btn extra btn-danger">
9-
Delete item
10-
</button>
11-
<button data-testid="cancel-button">
12-
Cancel
13-
</button>
14-
<svg data-testid="svg-spinner" class="spinner clockwise">
15-
<path />
16-
</svg>
17-
<div data-testid="no-classes"></div>
18-
</div>
19-
`)
23+
const {queryByTestId} = renderElementWithClasses()
2024

2125
expect(queryByTestId('delete-button')).toHaveClass('btn')
2226
expect(queryByTestId('delete-button')).toHaveClass('btn-danger')
@@ -91,3 +95,89 @@ test('.toHaveClass', () => {
9195
expect(queryByTestId('delete-button')).not.toHaveClass(' '),
9296
).toThrowError(/(none)/)
9397
})
98+
99+
test('.toHaveClass with exact mode option', () => {
100+
const {queryByTestId} = renderElementWithClasses()
101+
102+
expect(queryByTestId('delete-button')).toHaveClass('btn extra btn-danger', {
103+
exact: true,
104+
})
105+
expect(queryByTestId('delete-button')).not.toHaveClass('btn extra', {
106+
exact: true,
107+
})
108+
expect(
109+
queryByTestId('delete-button'),
110+
).not.toHaveClass('btn extra btn-danger foo', {exact: true})
111+
112+
expect(queryByTestId('delete-button')).toHaveClass('btn extra btn-danger', {
113+
exact: false,
114+
})
115+
expect(queryByTestId('delete-button')).toHaveClass('btn extra', {
116+
exact: false,
117+
})
118+
expect(
119+
queryByTestId('delete-button'),
120+
).not.toHaveClass('btn extra btn-danger foo', {exact: false})
121+
122+
expect(queryByTestId('delete-button')).toHaveClass(
123+
'btn',
124+
'extra',
125+
'btn-danger',
126+
{exact: true},
127+
)
128+
expect(queryByTestId('delete-button')).not.toHaveClass('btn', 'extra', {
129+
exact: true,
130+
})
131+
expect(queryByTestId('delete-button')).not.toHaveClass(
132+
'btn',
133+
'extra',
134+
'btn-danger',
135+
'foo',
136+
{exact: true},
137+
)
138+
139+
expect(queryByTestId('delete-button')).toHaveClass(
140+
'btn',
141+
'extra',
142+
'btn-danger',
143+
{exact: false},
144+
)
145+
expect(queryByTestId('delete-button')).toHaveClass('btn', 'extra', {
146+
exact: false,
147+
})
148+
expect(queryByTestId('delete-button')).not.toHaveClass(
149+
'btn',
150+
'extra',
151+
'btn-danger',
152+
'foo',
153+
{exact: false},
154+
)
155+
156+
expect(queryByTestId('only-one-class')).toHaveClass('alone', {exact: true})
157+
expect(queryByTestId('only-one-class')).not.toHaveClass('alone foo', {
158+
exact: true,
159+
})
160+
expect(queryByTestId('only-one-class')).not.toHaveClass('alone', 'foo', {
161+
exact: true,
162+
})
163+
164+
expect(queryByTestId('only-one-class')).toHaveClass('alone', {exact: false})
165+
expect(queryByTestId('only-one-class')).not.toHaveClass('alone foo', {
166+
exact: false,
167+
})
168+
expect(queryByTestId('only-one-class')).not.toHaveClass('alone', 'foo', {
169+
exact: false,
170+
})
171+
172+
expect(() =>
173+
expect(queryByTestId('only-one-class')).not.toHaveClass('alone', {
174+
exact: true,
175+
}),
176+
).toThrowError(/Expected the element not to have EXACTLY defined classes/)
177+
178+
expect(() =>
179+
expect(queryByTestId('only-one-class')).toHaveClass('alone', 'foo', {
180+
exact: true,
181+
}),
182+
).toThrowError(/Expected the element to have EXACTLY defined classes/)
183+
})

‎src/to-have-class.js

+33-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,20 @@
11
import {matcherHint, printExpected} from 'jest-matcher-utils'
22
import {checkHtmlElement, getMessage} from './utils'
33

4+
function getExpectedClassNamesAndOptions(params) {
5+
const lastParam = params.pop()
6+
let expectedClassNames, options
7+
8+
if (typeof lastParam === 'object') {
9+
expectedClassNames = params
10+
options = lastParam
11+
} else {
12+
expectedClassNames = params.concat(lastParam)
13+
options = { exact: false }
14+
}
15+
return {expectedClassNames, options}
16+
}
17+
418
function splitClassNames(str) {
519
if (!str) {
620
return []
@@ -12,13 +26,31 @@ function isSubset(subset, superset) {
1226
return subset.every(item => superset.includes(item))
1327
}
1428

15-
export function toHaveClass(htmlElement, ...expectedClassNames) {
29+
export function toHaveClass(htmlElement, ...params) {
1630
checkHtmlElement(htmlElement, toHaveClass, this)
31+
const {expectedClassNames, options} = getExpectedClassNamesAndOptions(params)
32+
1733
const received = splitClassNames(htmlElement.getAttribute('class'))
1834
const expected = expectedClassNames.reduce(
1935
(acc, className) => acc.concat(splitClassNames(className)),
2036
[],
2137
)
38+
39+
if (options.exact) {
40+
return {
41+
pass: isSubset(expected, received) && expected.length === received.length,
42+
message: () => {
43+
const to = this.isNot ? 'not to' : 'to'
44+
return getMessage(
45+
`Expected the element ${to} have EXACTLY defined classes`,
46+
expected.join(' '),
47+
'Received',
48+
received.join(' '),
49+
)
50+
},
51+
}
52+
}
53+
2254
return expected.length > 0
2355
? {
2456
pass: isSubset(expected, received),

0 commit comments

Comments
 (0)
Please sign in to comment.