cypress-map
Extra Cypress query commands for v12+
Install
Add this package as a dev dependency
$ npm i -D cypress-map
# or using Yarn
$ yarn add -D cypress-map
Include this package in your spec or support file to use all custom query commands
import 'cypress-map'
Alternative: import only the query commands you need:
import 'cypress-map/commands/map'
import 'cypress-map/commands/tap'
API
apply
const double = (n) => n * 2
cy.wrap(100).apply(double).should('equal', 200)
partial
Sometimes you have the callback to apply, and you know the first argument(s), and just need to put the subject at the last position. This is where you can partially apply the known arguments to the given callback.
cy.wrap(100).partial(Cypress._.add, 5).should('equal', 105)
cy.wrap(100)
.apply((subject) => Cypress._.add(5, subject))
.should('equal', 105)
map
cy.get('.matching')
.map('innerText')
.should('deep.equal', ['first', 'third', 'fourth'])
mapInvoke
cy.get('#items li')
.find('.price')
.map('innerText')
.mapInvoke('replace', '$', '')
.mapInvoke('trim')
reduce
cy.get('#items li')
.find('.price')
.map('innerText')
.mapInvoke('replace', '$', '')
.map(parseFloat)
.reduce((max, n) => (n > max ? n : max))
You can provide the initial accumulator value
cy.wrap([1, 2, 3])
.reduce((sum, n) => sum + n, 10)
.should('equal', 16)
See reduce.cy.js
tap
cy.get('#items li')
.find('.price')
.map('innerText')
.tap()
.mapInvoke('replace', '$', '')
.mapInvoke('trim')
.tap(console.info, 'trimmed strings')
Notice: if the label is provided, the callback function is called with label and the subject.
print
A better cy.log
: yields the value, intelligently stringifies values using %
and string-format notation.
cy.wrap(42)
.print()
.should('equal', 42)
cy.wrap(42).print('the answer is %d')
cy.wrap({ name: 'Joe' }).print('person %o')
cy.wrap({ name: 'Joe' }).print('person name {0.name}')
cy.wrap(arr).print('array length {0.length}')
cy.wrap(arr).print((a) => `array with ${a.length} items`)
cy.wrap(arr).print((list) => list[2])
See print.cy.js for more examples
primo
cy.get('.matching')
.map('innerText')
.primo()
.invoke('toUpperCase')
.should('equal', 'FIRST')
See primo.cy.js
prop
Works like cy.its
for objects, but gets the property for jQuery objects, which cy.its
does not
cy.get('#items li.matching').last().prop('ariaLabel').should('equal', 'four')
See prop.cy.js
update
Changes a single property inside the subject by running it through the given callback function. Useful to do type conversions, for example, let's convert the "age" property to a Number
cy.wrap({ age: '20' }).update('age', Number).should('deep.equal', { age: 20 })
at
Returns a DOM element from jQuery object at position k
. Returns an item from array at position k
. For negative index, counts the items from the end.
cy.get('#items li').at(-1).its('innerText').should('equal', 'fifth')
See at.cy.js
table
📝 to learn more about cy.table
command, read the blog post Test HTML Tables Using cy.table Query Command.
Extracts all cells from the current subject table. Yields a 2D array of strings.
cy.get('table').table()
You can slice the table to yield just a region .table(x, y, w, h)
For example, you can get 2 by 2 subregion
cy.get('table')
.table(0, 2, 2, 2)
.should('deep.equal', [
['Cary', '30'],
['Joe', '28'],
])
See the spec table.cy.js for more examples.
Tip: you can combine cy.table
with cy.map
, cy.mapInvoke
to get the parts of the table. For example, the same 2x2 part of the table could be extracted with:
cy.get('table')
.table()
.invoke('slice', 2, 4)
.mapInvoke('slice', 0, 2)
.should('deep.equal', [
['Cary', '30'],
['Joe', '28'],
])
Tip 2: to get just the headings row, combine .table
and .its
queries
cy.get('table')
.table(0, 0, 3, 1)
.its(0)
.should('deep.equal', ['Name', 'Age', 'Date (YYYY-MM-DD)'])
To get the last row, you could do:
cy.get('table').table().invoke('slice', -1).its(0)
To get the first column joined into a single array (instead of array of 1x1 arrays)
cy.get('table')
.table(0, 1, 1)
.invoke('flatMap', Cypress._.identity)
.should('deep.equal', ['Dave', 'Cary', 'Joe', 'Anna'])
invokeOnce
In Cypress v12 cy.invoke
became a query, which made working with asynchronous methods really unwieldy. The cy.invokeOnce
is a return the old way of calling the method and yielding the resolved value.
cy.wrap(app)
.invokeOnce('fetchName')
.should('equal', 'My App')
See the spec invoke-once.cy.js for more examples.
cy.invoke vs cy.map vs cy.mapInvoke
Here are a few examples to clarify the different between the cy.invoke
, cy.map
, and cy.mapInvoke
query commands, see diff.cy.js
const list = ['apples', 'plums', 'bananas']
cy.wrap(list)
.invoke('sort')
.should('deep.equal', ['apples', 'bananas', 'plums'])
cy.wrap(list)
.mapInvoke('toUpperCase')
.should('deep.equal', ['APPLES', 'PLUMS', 'BANANAS'])
const reverse = (s) => s.split('').reverse().join('')
cy.wrap(list)
.map(reverse)
.should('deep.equal', ['selppa', 'smulp', 'sananab'])
.map('length')
.should('deep.equal', [6, 5, 7])
Types
This package includes TypeScript command definitions for its custom commands in the file commands/index.d.ts. To use it from your JavaScript specs:
If you are using TypeScript, include this module in your types list
{
"compilerOptions": {
"types": ["cypress", "cypress-map"]
}
}
See also
- cypress-should-really has similar functional helpers for constructing the
should(callback)
function on the fly.
Small print
Author: Gleb Bahmutov <gleb.bahmutov@gmail.com> © 2022
License: MIT - do anything with the code, but don't blame me if it does not work.
Support: if you find any problems with this module, email / tweet /
open issue on Github