How to click x number of times in Cypress - testing

I have a list of objects on my site that all have 'Add' buttons next to them. When the first 'Add' button is clicked, that object is added and the row disappears and is replaced by the next one down. The object name is the same. I want to .click() three times to add the first three objects in the list, before saving. How can I do this?
I'm aware of .click() to click a single object. I'm also aware of .click ({ multiple: true}) to click all the objects on the page. However, I want it to stop clicking after the third time.
Currently set to click multiple times to add all the objects in the list (which is incorrect):
cy.get('#FieldListDialog > div > table > tr > td > button.button.add-button')
.should('exist')
.click({ multiple: true });

To hammer click a button you can use this:
for(let n = 0; n < 10; n ++){
cy.get('#FieldListDialog > div > table > tr > td > button.button.add-button')
.click()
}
The multiple: true is used to click several elements, for example to click all buttons on the page.

If you need just a few clicks you could also go for this great solution:
cy.get("button").click().click().click()
.click().click().click().click().click()
.click().click()
In my opinion this syntax also gives a great sensation of what the user would be doing in this scenario.

The problem is with your selector. If you have n number of buttons, and you want to click all of them, then you need to match all the buttons, not just a single one. So look for a selector (e.g. a class unique to your Add button) that matches all of them.
You can then use .each() to iterate through them and break out of the loop when you reach a certain index:
cy.get('#your_selector_to_your_add_buttons')
.each(($el, $index) => {
if($index == 3){
return false;
}
cy.wrap($el).click()
} )

You just had to move your assertion after the click and said click until not exists
.click({ multiple: true })
.should('not.exist');```

you could use lodash
https://lodash.com/docs/4.17.15#times
import { times } from 'lodash'
times(2, () => {
cy.get('#FieldListDialog > div > table > tr > td > button.button.add-button')
.click()
})

You could also do like this:
Cypress._.times(10, () => {
cy.get('#FieldListDialog > div > table > tr > td > button.button.add-button').click()
})

The best way I could find to do this was by using the Wait for a number of milliseconds or wait for an aliased resource to resolve before moving on to the next command.
cy.get('#FieldListDialog > div > table > tr > td > button.button.add-button')
.should('exist')
.click();
cy.wait(500)
cy.get('#FieldListDialog > div > table > tr > td > button.button.add-button')
.should('exist')
.click();
cy.wait(500)
cy.get('#FieldListDialog > div > table > tr > td > button.button.add-button')
.should('exist')
.click();

Following the comments by Kamran Eyyubov and monikapatelIT, I was able to come up with the following to simulate a button mash, using the Lodash library that cypress exposes.
// Fetch a reference outside the loop so it can be referenced inside the
// loop.
cy.get('#submit-button')
.as('submitButton');
// Simulate a button mash. Click the button 10 times.
Cypress._.times(10, function () {
cy.get('#submitButton')
// The submit button is disabled on the first click, so
// force is set to TRUE to prevent the test from failing
// due to clicking a disabled element.
.click({force: true});
});

// option 1
cy.get('button')
.dblclick()
// option 2
cy.get('button')
.click({ multiple: true })

I did and command
Cypress.Commands.add('MultiClick',(element:string,times: number) => {
for(let n = 0; n < times; n ++){
cy.get(element).click({ force: true })
}
})
and then use it
cy.MultiClick('[data-cy=qty-product]',2)

Related

Cypress Variable undefined

I am trying to set a variable to then use later on in an assert. I want to verify a date on one page is the same on another page. I have the below;
cy.get("#due > div > div > div > input").then(($dueDatePicker) => {
const dueDate = $dueDatePicker.text();
cy.log(dueDate);
});
I then have the below assert later on in the test;
cy.get("#content > div > section.content.mtm10 > div.row > div > div > div.box-body").contains(dueDate);
However Cypress tells me dueDate is undefined in the Assert.
What am I missing? Thanks
There is a scoping issue with where you are defining the variable. There are a few ways to get around this.
The first would be to declare the variable outside of the Cypress chain.
let dueDate = ''
cy.get('foo')
.then(($el) => {
dueDate = $el.text()
})
.get('bar').contains(dueDate);
You could also return the value from the .then().
cy.get('foo')
.then(($el) => {
return $el.text();
})
.then((dueDate) => {
cy.get('bar').contains(dueDate);
})
I'd recommend Filip Hric's blogpost on Cypress variables for more information on understanding how the variables/scopes/timing all work together.
As long as you are using the value later in the same test, an alias is the preferred way to store a result for a later assertion
cy.get("#due > div > div > div > input")
.invoke('text')
.as('dueDate')
...
cy.get('#dueDate').then(dueDate => {
cy.get("#content > div > section.content.mtm10 > div.row > div > div > div.box-body")
.contains(dueDate)
})

While opening links in reverse order in new tabs, I want to exclude "sports" and "IR" links

I want to open links from contents from bottom to top in new tab:
driver.get("https://en.wikipedia.org/wiki/Sports_video_game");
WebElement contents = driver.findElement(By.xpath("//*[#id='toc']"));
List<WebElement> links = contents.findElements(By.tagName("a"));
String clickonlinkTab=Keys.chord(Keys.CONTROL, Keys.ENTER);
for(int i = links.size() - 1; i >= 0; i--)
{
links.get(i).sendKeys(clickonlinkTab);
links.findElement(By.partialLinkText("sports")).notclick();
links.findElement(By.partialLinkText("IR")).notclick();
}
I want Selenium to skip links which has "sports" and "IR" any where on it.
As "notclick" is not a valid method, the simplest solution is to check if the link text contains "Sports" or "IR" before performing the click action. E.g:
for (int i = links.size() - 1; i >= 0; i--) {
if (!(links.get(i).getText().toLowerCase().contains("sports")
|| links.get(i).getText().toLowerCase().contains("ir"))) {
links.get(i).sendKeys(clickonlinkTab);
}
}
Here is a link to a runnable test case where you can see the results, including printing to verify that it works. If you want to edit, making an account can be done for free via the edit button.

What are the steps involved for svg button click and copy the url

Writing steps for UI testing using karate framework
so basically svg button, have to click then the URL in the new tab for further functionality processing
how to handle that ?
i tried below code
def secure_url = mouse("#root > section > section > main > div > div > div.ant-table-wrapper > div > div > div > div > div > table > tbody > tr:nth-child(1) > td:nth-child(7) > span > button > svg").click()
and verified the secure_url value using
print secure_url
it prints com.intuit.karate.driver.DriverMouse#664e5dee
but Expected is URL
Most Karate methods will return an Element object on which you have to call more methods. Please spend some time to read the docs: https://github.com/intuit/karate/tree/master/karate-core#chaining
* def elem = locate('#some-id')
* def url = elem.attribute('href')

Link not working when clicked by Testcafe inside iframe

I am trying to sett upp some simple verification in our demoshop.
It works until clicking the "genomför köp" (make purchases) link is clicked.
The link markes as clicked (color change) but the proper actions are not performedand you stay on the same side.
if you add a debugline after the "await t.click('#purchase-button');" line you can click the link manualy, you can hover over the link so testcafe finds it. But for some reason tha click() dont work.
EDIT:1: the OS i am using is WIn10, running testcafe 1.8.2 under phpstorm 2019.3.3 and calling it with "testcafe chrome *.js"
Any help would be appreciated.
/Christoffer
Relevant code is below and the demoshop is open to use.
fixture`demoshop.resurs.com`
.page(https://demoshop.resurs.com);
test(
'Run: demoshop.resurs.com', async t => {
await t
.click('.home-shop-now')
.click('#root > div > main > div > div.container > div.products-container.mb-60 > div:nth-child(1) > a > img')
.click('#root > div > main > div > div > div.product-detail__container > div.product-detail__info-container > div.product-detail__add-to-cart > div > button')
.click('#root > div > main > div > span > header > div > div > div > span.shopping-bag-icon.clickable > div')
.click('#root > div > main > div > span > div > div > div > div.shopping-cart-container > a');
await t.switchToIframe('iframe');
await t.click('#purchase-button'); //This does not work properly
await t.expect(Selector('h2').innerText).contains('Tack för din order!');
}
);
I was able to reproduce the issue with TestCafe#1.8.2 but the test works as expected with TestCafe#1.8.4.
I recommend you update to the latest TestCafe version and apply changes provided by #DmitryOstashev:
import { Selector } from 'testcafe';
fixture`demoshop.resurs.com`
.page('https://demoshop.resurs.com');
test(
'Run: demoshop.resurs.com', async t => {
await t
.click('.home-shop-now')
.click('#root > div > main > div > div.container > div.products-container.mb-60 > div:nth-child(1) > a > img')
.click('#root > div > main > div > div > div.product-detail__container > div.product-detail__info-container > div.product-detail__add-to-cart > div > button')
.click('#root > div > main > div > span > header > div > div > div > span.shopping-bag-icon.clickable > div')
.click('#root > div > main > div > span > div > div > div > div.shopping-cart-container > a');
await t.switchToIframe('iframe');
await t.click('#purchase-button');
await t.switchToMainWindow();
await t.expect(Selector('h2').innerText).contains('Tack för din order!');
}
);

network - disableEditMode on click

When double-clicking on an edge or a node I enter the edit or add mode, then I can do the modification. This works fine:
network.on("doubleClick",function(params) {
console.log(params);
if ( (params.edges.length > 0) && (params.nodes.length == 0) ) {
network.editEdgeMode();
} else if (params.nodes.length > 0) {
network.addEdgeMode();
}
});
I would like to get out of the mode when the user clicks elsewhere, for that I tried to use the click event in combination with disableEditMode() but this does not seem to work:
network.on("click",function(params) {
console.log(params);
network.disableEditMode();
});
Is this actually possible? Is this the purpose the disableEditMode() is made for?