#927 Adding support for cypress and Percy
| @@ -1,8 +1,9 @@ | ||||
| dist: trusty | ||||
| language: node_js | ||||
| node_js: | ||||
|   - "8" | ||||
|   - "10" | ||||
| script: | ||||
|   - yarn test --coverage | ||||
|   - yarn e2e | ||||
| after_success: | ||||
|   - cat ./coverage/lcov.info | ./node_modules/.bin/coveralls | ||||
|   | ||||
							
								
								
									
										1
									
								
								cypress.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1 @@ | ||||
| { "video": false } | ||||
							
								
								
									
										272
									
								
								cypress/examples/actions.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,272 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Actions', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/actions') | ||||
|   }) | ||||
|  | ||||
|   // https://on.cypress.io/interacting-with-elements | ||||
|  | ||||
|   it('.type() - type into a DOM element', () => { | ||||
|     // https://on.cypress.io/type | ||||
|     cy.get('.action-email') | ||||
|       .type('fake@email.com').should('have.value', 'fake@email.com') | ||||
|  | ||||
|       // .type() with special character sequences | ||||
|       .type('{leftarrow}{rightarrow}{uparrow}{downarrow}') | ||||
|       .type('{del}{selectall}{backspace}') | ||||
|  | ||||
|       // .type() with key modifiers | ||||
|       .type('{alt}{option}') //these are equivalent | ||||
|       .type('{ctrl}{control}') //these are equivalent | ||||
|       .type('{meta}{command}{cmd}') //these are equivalent | ||||
|       .type('{shift}') | ||||
|  | ||||
|       // Delay each keypress by 0.1 sec | ||||
|       .type('slow.typing@email.com', { delay: 100 }) | ||||
|       .should('have.value', 'slow.typing@email.com') | ||||
|  | ||||
|     cy.get('.action-disabled') | ||||
|       // Ignore error checking prior to type | ||||
|       // like whether the input is visible or disabled | ||||
|       .type('disabled error checking', { force: true }) | ||||
|       .should('have.value', 'disabled error checking') | ||||
|   }) | ||||
|  | ||||
|   it('.focus() - focus on a DOM element', () => { | ||||
|     // https://on.cypress.io/focus | ||||
|     cy.get('.action-focus').focus() | ||||
|       .should('have.class', 'focus') | ||||
|       .prev().should('have.attr', 'style', 'color: orange;') | ||||
|   }) | ||||
|  | ||||
|   it('.blur() - blur off a DOM element', () => { | ||||
|     // https://on.cypress.io/blur | ||||
|     cy.get('.action-blur').type('About to blur').blur() | ||||
|       .should('have.class', 'error') | ||||
|       .prev().should('have.attr', 'style', 'color: red;') | ||||
|   }) | ||||
|  | ||||
|   it('.clear() - clears an input or textarea element', () => { | ||||
|     // https://on.cypress.io/clear | ||||
|     cy.get('.action-clear').type('Clear this text') | ||||
|       .should('have.value', 'Clear this text') | ||||
|       .clear() | ||||
|       .should('have.value', '') | ||||
|   }) | ||||
|  | ||||
|   it('.submit() - submit a form', () => { | ||||
|     // https://on.cypress.io/submit | ||||
|     cy.get('.action-form') | ||||
|       .find('[type="text"]').type('HALFOFF') | ||||
|     cy.get('.action-form').submit() | ||||
|       .next().should('contain', 'Your form has been submitted!') | ||||
|   }) | ||||
|  | ||||
|   it('.click() - click on a DOM element', () => { | ||||
|     // https://on.cypress.io/click | ||||
|     cy.get('.action-btn').click() | ||||
|  | ||||
|     // You can click on 9 specific positions of an element: | ||||
|     //  ----------------------------------- | ||||
|     // | topLeft        top       topRight | | ||||
|     // |                                   | | ||||
|     // |                                   | | ||||
|     // |                                   | | ||||
|     // | left          center        right | | ||||
|     // |                                   | | ||||
|     // |                                   | | ||||
|     // |                                   | | ||||
|     // | bottomLeft   bottom   bottomRight | | ||||
|     //  ----------------------------------- | ||||
|  | ||||
|     // clicking in the center of the element is the default | ||||
|     cy.get('#action-canvas').click() | ||||
|  | ||||
|     cy.get('#action-canvas').click('topLeft') | ||||
|     cy.get('#action-canvas').click('top') | ||||
|     cy.get('#action-canvas').click('topRight') | ||||
|     cy.get('#action-canvas').click('left') | ||||
|     cy.get('#action-canvas').click('right') | ||||
|     cy.get('#action-canvas').click('bottomLeft') | ||||
|     cy.get('#action-canvas').click('bottom') | ||||
|     cy.get('#action-canvas').click('bottomRight') | ||||
|  | ||||
|     // .click() accepts an x and y coordinate | ||||
|     // that controls where the click occurs :) | ||||
|  | ||||
|     cy.get('#action-canvas') | ||||
|       .click(80, 75) // click 80px on x coord and 75px on y coord | ||||
|       .click(170, 75) | ||||
|       .click(80, 165) | ||||
|       .click(100, 185) | ||||
|       .click(125, 190) | ||||
|       .click(150, 185) | ||||
|       .click(170, 165) | ||||
|  | ||||
|     // click multiple elements by passing multiple: true | ||||
|     cy.get('.action-labels>.label').click({ multiple: true }) | ||||
|  | ||||
|     // Ignore error checking prior to clicking | ||||
|     cy.get('.action-opacity>.btn').click({ force: true }) | ||||
|   }) | ||||
|  | ||||
|   it('.dblclick() - double click on a DOM element', () => { | ||||
|     // https://on.cypress.io/dblclick | ||||
|  | ||||
|     // Our app has a listener on 'dblclick' event in our 'scripts.js' | ||||
|     // that hides the div and shows an input on double click | ||||
|     cy.get('.action-div').dblclick().should('not.be.visible') | ||||
|     cy.get('.action-input-hidden').should('be.visible') | ||||
|   }) | ||||
|  | ||||
|   it('.check() - check a checkbox or radio element', () => { | ||||
|     // https://on.cypress.io/check | ||||
|  | ||||
|     // By default, .check() will check all | ||||
|     // matching checkbox or radio elements in succession, one after another | ||||
|     cy.get('.action-checkboxes [type="checkbox"]').not('[disabled]') | ||||
|       .check().should('be.checked') | ||||
|  | ||||
|     cy.get('.action-radios [type="radio"]').not('[disabled]') | ||||
|       .check().should('be.checked') | ||||
|  | ||||
|     // .check() accepts a value argument | ||||
|     cy.get('.action-radios [type="radio"]') | ||||
|       .check('radio1').should('be.checked') | ||||
|  | ||||
|     // .check() accepts an array of values | ||||
|     cy.get('.action-multiple-checkboxes [type="checkbox"]') | ||||
|       .check(['checkbox1', 'checkbox2']).should('be.checked') | ||||
|  | ||||
|     // Ignore error checking prior to checking | ||||
|     cy.get('.action-checkboxes [disabled]') | ||||
|       .check({ force: true }).should('be.checked') | ||||
|  | ||||
|     cy.get('.action-radios [type="radio"]') | ||||
|       .check('radio3', { force: true }).should('be.checked') | ||||
|   }) | ||||
|  | ||||
|   it('.uncheck() - uncheck a checkbox element', () => { | ||||
|     // https://on.cypress.io/uncheck | ||||
|  | ||||
|     // By default, .uncheck() will uncheck all matching | ||||
|     // checkbox elements in succession, one after another | ||||
|     cy.get('.action-check [type="checkbox"]') | ||||
|       .not('[disabled]') | ||||
|       .uncheck().should('not.be.checked') | ||||
|  | ||||
|     // .uncheck() accepts a value argument | ||||
|     cy.get('.action-check [type="checkbox"]') | ||||
|       .check('checkbox1') | ||||
|       .uncheck('checkbox1').should('not.be.checked') | ||||
|  | ||||
|     // .uncheck() accepts an array of values | ||||
|     cy.get('.action-check [type="checkbox"]') | ||||
|       .check(['checkbox1', 'checkbox3']) | ||||
|       .uncheck(['checkbox1', 'checkbox3']).should('not.be.checked') | ||||
|  | ||||
|     // Ignore error checking prior to unchecking | ||||
|     cy.get('.action-check [disabled]') | ||||
|       .uncheck({ force: true }).should('not.be.checked') | ||||
|   }) | ||||
|  | ||||
|   it('.select() - select an option in a <select> element', () => { | ||||
|     // https://on.cypress.io/select | ||||
|  | ||||
|     // Select option(s) with matching text content | ||||
|     cy.get('.action-select').select('apples') | ||||
|  | ||||
|     cy.get('.action-select-multiple') | ||||
|     .select(['apples', 'oranges', 'bananas']) | ||||
|  | ||||
|     // Select option(s) with matching value | ||||
|     cy.get('.action-select').select('fr-bananas') | ||||
|  | ||||
|     cy.get('.action-select-multiple') | ||||
|       .select(['fr-apples', 'fr-oranges', 'fr-bananas']) | ||||
|   }) | ||||
|  | ||||
|   it('.scrollIntoView() - scroll an element into view', () => { | ||||
|     // https://on.cypress.io/scrollintoview | ||||
|  | ||||
|     // normally all of these buttons are hidden, | ||||
|     // because they're not within | ||||
|     // the viewable area of their parent | ||||
|     // (we need to scroll to see them) | ||||
|     cy.get('#scroll-horizontal button') | ||||
|       .should('not.be.visible') | ||||
|  | ||||
|     // scroll the button into view, as if the user had scrolled | ||||
|     cy.get('#scroll-horizontal button').scrollIntoView() | ||||
|       .should('be.visible') | ||||
|  | ||||
|     cy.get('#scroll-vertical button') | ||||
|       .should('not.be.visible') | ||||
|  | ||||
|     // Cypress handles the scroll direction needed | ||||
|     cy.get('#scroll-vertical button').scrollIntoView() | ||||
|       .should('be.visible') | ||||
|  | ||||
|     cy.get('#scroll-both button') | ||||
|       .should('not.be.visible') | ||||
|  | ||||
|     // Cypress knows to scroll to the right and down | ||||
|     cy.get('#scroll-both button').scrollIntoView() | ||||
|       .should('be.visible') | ||||
|   }) | ||||
|  | ||||
|   it('.trigger() - trigger an event on a DOM element', () => { | ||||
|     // https://on.cypress.io/trigger | ||||
|  | ||||
|     // To interact with a range input (slider) | ||||
|     // we need to set its value & trigger the | ||||
|     // event to signal it changed | ||||
|  | ||||
|     // Here, we invoke jQuery's val() method to set | ||||
|     // the value and trigger the 'change' event | ||||
|     cy.get('.trigger-input-range') | ||||
|       .invoke('val', 25) | ||||
|       .trigger('change') | ||||
|       .get('input[type=range]').siblings('p') | ||||
|       .should('have.text', '25') | ||||
|   }) | ||||
|  | ||||
|   it('cy.scrollTo() - scroll the window or element to a position', () => { | ||||
|  | ||||
|     // https://on.cypress.io/scrollTo | ||||
|  | ||||
|     // You can scroll to 9 specific positions of an element: | ||||
|     //  ----------------------------------- | ||||
|     // | topLeft        top       topRight | | ||||
|     // |                                   | | ||||
|     // |                                   | | ||||
|     // |                                   | | ||||
|     // | left          center        right | | ||||
|     // |                                   | | ||||
|     // |                                   | | ||||
|     // |                                   | | ||||
|     // | bottomLeft   bottom   bottomRight | | ||||
|     //  ----------------------------------- | ||||
|  | ||||
|     // if you chain .scrollTo() off of cy, we will | ||||
|     // scroll the entire window | ||||
|     cy.scrollTo('bottom') | ||||
|  | ||||
|     cy.get('#scrollable-horizontal').scrollTo('right') | ||||
|  | ||||
|     // or you can scroll to a specific coordinate: | ||||
|     // (x axis, y axis) in pixels | ||||
|     cy.get('#scrollable-vertical').scrollTo(250, 250) | ||||
|  | ||||
|     // or you can scroll to a specific percentage | ||||
|     // of the (width, height) of the element | ||||
|     cy.get('#scrollable-both').scrollTo('75%', '25%') | ||||
|  | ||||
|     // control the easing of the scroll (default is 'swing') | ||||
|     cy.get('#scrollable-vertical').scrollTo('center', { easing: 'linear' }) | ||||
|  | ||||
|     // control the duration of the scroll (in ms) | ||||
|     cy.get('#scrollable-both').scrollTo('center', { duration: 2000 }) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										42
									
								
								cypress/examples/aliasing.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,42 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Aliasing', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/aliasing') | ||||
|   }) | ||||
|  | ||||
|   it('.as() - alias a DOM element for later use', () => { | ||||
|     // https://on.cypress.io/as | ||||
|  | ||||
|     // Alias a DOM element for use later | ||||
|     // We don't have to traverse to the element | ||||
|     // later in our code, we reference it with @ | ||||
|  | ||||
|     cy.get('.as-table').find('tbody>tr') | ||||
|       .first().find('td').first() | ||||
|       .find('button').as('firstBtn') | ||||
|  | ||||
|     // when we reference the alias, we place an | ||||
|     // @ in front of its name | ||||
|     cy.get('@firstBtn').click() | ||||
|  | ||||
|     cy.get('@firstBtn') | ||||
|       .should('have.class', 'btn-success') | ||||
|       .and('contain', 'Changed') | ||||
|   }) | ||||
|  | ||||
|   it('.as() - alias a route for later use', () => { | ||||
|  | ||||
|     // Alias the route to wait for its response | ||||
|     cy.server() | ||||
|     cy.route('GET', 'comments/*').as('getComment') | ||||
|  | ||||
|     // we have code that gets a comment when | ||||
|     // the button is clicked in scripts.js | ||||
|     cy.get('.network-btn').click() | ||||
|  | ||||
|     // https://on.cypress.io/wait | ||||
|     cy.wait('@getComment').its('status').should('eq', 200) | ||||
|  | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										168
									
								
								cypress/examples/assertions.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,168 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Assertions', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/assertions') | ||||
|   }) | ||||
|  | ||||
|   describe('Implicit Assertions', () => { | ||||
|     it('.should() - make an assertion about the current subject', () => { | ||||
|       // https://on.cypress.io/should | ||||
|       cy.get('.assertion-table') | ||||
|         .find('tbody tr:last') | ||||
|         .should('have.class', 'success') | ||||
|         .find('td') | ||||
|         .first() | ||||
|         // checking the text of the <td> element in various ways | ||||
|         .should('have.text', 'Column content') | ||||
|         .should('contain', 'Column content') | ||||
|         .should('have.html', 'Column content') | ||||
|         // chai-jquery uses "is()" to check if element matches selector | ||||
|         .should('match', 'td') | ||||
|         // to match text content against a regular expression | ||||
|         // first need to invoke jQuery method text() | ||||
|         // and then match using regular expression | ||||
|         .invoke('text') | ||||
|         .should('match', /column content/i) | ||||
|  | ||||
|       // a better way to check element's text content against a regular expression | ||||
|       // is to use "cy.contains" | ||||
|       // https://on.cypress.io/contains | ||||
|       cy.get('.assertion-table') | ||||
|         .find('tbody tr:last') | ||||
|         // finds first <td> element with text content matching regular expression | ||||
|         .contains('td', /column content/i) | ||||
|         .should('be.visible') | ||||
|  | ||||
|       // for more information about asserting element's text | ||||
|       // see https://on.cypress.io/using-cypress-faq#How-do-I-get-an-element’s-text-contents | ||||
|     }) | ||||
|  | ||||
|     it('.and() - chain multiple assertions together', () => { | ||||
|       // https://on.cypress.io/and | ||||
|       cy.get('.assertions-link') | ||||
|         .should('have.class', 'active') | ||||
|         .and('have.attr', 'href') | ||||
|         .and('include', 'cypress.io') | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   describe('Explicit Assertions', () => { | ||||
|     // https://on.cypress.io/assertions | ||||
|     it('expect - make an assertion about a specified subject', () => { | ||||
|       // We can use Chai's BDD style assertions | ||||
|       expect(true).to.be.true | ||||
|       const o = { foo: 'bar' } | ||||
|  | ||||
|       expect(o).to.equal(o) | ||||
|       expect(o).to.deep.equal({ foo: 'bar' }) | ||||
|       // matching text using regular expression | ||||
|       expect('FooBar').to.match(/bar$/i) | ||||
|     }) | ||||
|  | ||||
|     it('pass your own callback function to should()', () => { | ||||
|       // Pass a function to should that can have any number | ||||
|       // of explicit assertions within it. | ||||
|       // The ".should(cb)" function will be retried | ||||
|       // automatically until it passes all your explicit assertions or times out. | ||||
|       cy.get('.assertions-p') | ||||
|         .find('p') | ||||
|         .should(($p) => { | ||||
|           // https://on.cypress.io/$ | ||||
|           // return an array of texts from all of the p's | ||||
|           // @ts-ignore TS6133 unused variable | ||||
|           const texts = $p.map((i, el) => Cypress.$(el).text()) | ||||
|  | ||||
|           // jquery map returns jquery object | ||||
|           // and .get() convert this to simple array | ||||
|           const paragraphs = texts.get() | ||||
|  | ||||
|           // array should have length of 3 | ||||
|           expect(paragraphs, 'has 3 paragraphs').to.have.length(3) | ||||
|  | ||||
|           // use second argument to expect(...) to provide clear | ||||
|           // message with each assertion | ||||
|           expect(paragraphs, 'has expected text in each paragraph').to.deep.eq([ | ||||
|             'Some text from first p', | ||||
|             'More text from second p', | ||||
|             'And even more text from third p', | ||||
|           ]) | ||||
|         }) | ||||
|     }) | ||||
|  | ||||
|     it('finds element by class name regex', () => { | ||||
|       cy.get('.docs-header') | ||||
|         .find('div') | ||||
|         // .should(cb) callback function will be retried | ||||
|         .should(($div) => { | ||||
|           expect($div).to.have.length(1) | ||||
|  | ||||
|           const className = $div[0].className | ||||
|  | ||||
|           expect(className).to.match(/heading-/) | ||||
|         }) | ||||
|         // .then(cb) callback is not retried, | ||||
|         // it either passes or fails | ||||
|         .then(($div) => { | ||||
|           expect($div, 'text content').to.have.text('Introduction') | ||||
|         }) | ||||
|     }) | ||||
|  | ||||
|     it('can throw any error', () => { | ||||
|       cy.get('.docs-header') | ||||
|         .find('div') | ||||
|         .should(($div) => { | ||||
|           if ($div.length !== 1) { | ||||
|             // you can throw your own errors | ||||
|             throw new Error('Did not find 1 element') | ||||
|           } | ||||
|  | ||||
|           const className = $div[0].className | ||||
|  | ||||
|           if (!className.match(/heading-/)) { | ||||
|             throw new Error(`Could not find class "heading-" in ${className}`) | ||||
|           } | ||||
|         }) | ||||
|     }) | ||||
|  | ||||
|     it('matches unknown text between two elements', () => { | ||||
|       /** | ||||
|        * Text from the first element. | ||||
|        * @type {string} | ||||
|       */ | ||||
|       let text | ||||
|  | ||||
|       /** | ||||
|        * Normalizes passed text, | ||||
|        * useful before comparing text with spaces and different capitalization. | ||||
|        * @param {string} s Text to normalize | ||||
|       */ | ||||
|       const normalizeText = (s) => s.replace(/\s/g, '').toLowerCase() | ||||
|  | ||||
|       cy.get('.two-elements') | ||||
|         .find('.first') | ||||
|         .then(($first) => { | ||||
|           // save text from the first element | ||||
|           text = normalizeText($first.text()) | ||||
|         }) | ||||
|  | ||||
|       cy.get('.two-elements') | ||||
|         .find('.second') | ||||
|         .should(($div) => { | ||||
|           // we can massage text before comparing | ||||
|           const secondText = normalizeText($div.text()) | ||||
|  | ||||
|           expect(secondText, 'second text').to.equal(text) | ||||
|         }) | ||||
|     }) | ||||
|  | ||||
|     it('assert - assert shape of an object', () => { | ||||
|       const person = { | ||||
|         name: 'Joe', | ||||
|         age: 20, | ||||
|       } | ||||
|  | ||||
|       assert.isObject(person, 'value is object') | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										56
									
								
								cypress/examples/connectors.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,56 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Connectors', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/connectors') | ||||
|   }) | ||||
|  | ||||
|   it('.each() - iterate over an array of elements', () => { | ||||
|     // https://on.cypress.io/each | ||||
|     cy.get('.connectors-each-ul>li') | ||||
|       .each(($el, index, $list) => { | ||||
|         console.log($el, index, $list) | ||||
|       }) | ||||
|   }) | ||||
|  | ||||
|   it('.its() - get properties on the current subject', () => { | ||||
|     // https://on.cypress.io/its | ||||
|     cy.get('.connectors-its-ul>li') | ||||
|       // calls the 'length' property yielding that value | ||||
|       .its('length') | ||||
|       .should('be.gt', 2) | ||||
|   }) | ||||
|  | ||||
|   it('.invoke() - invoke a function on the current subject', () => { | ||||
|     // our div is hidden in our script.js | ||||
|     // $('.connectors-div').hide() | ||||
|  | ||||
|     // https://on.cypress.io/invoke | ||||
|     cy.get('.connectors-div').should('be.hidden') | ||||
|       // call the jquery method 'show' on the 'div.container' | ||||
|       .invoke('show') | ||||
|       .should('be.visible') | ||||
|   }) | ||||
|  | ||||
|   it('.spread() - spread an array as individual args to callback function', () => { | ||||
|     // https://on.cypress.io/spread | ||||
|     const arr = ['foo', 'bar', 'baz'] | ||||
|  | ||||
|     cy.wrap(arr).spread((foo, bar, baz) => { | ||||
|       expect(foo).to.eq('foo') | ||||
|       expect(bar).to.eq('bar') | ||||
|       expect(baz).to.eq('baz') | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   it('.then() - invoke a callback function with the current subject', () => { | ||||
|     // https://on.cypress.io/then | ||||
|     cy.get('.connectors-list > li') | ||||
|       .then(($lis) => { | ||||
|         expect($lis, '3 items').to.have.length(3) | ||||
|         expect($lis.eq(0), 'first item').to.contain('Walk the dog') | ||||
|         expect($lis.eq(1), 'second item').to.contain('Feed the cat') | ||||
|         expect($lis.eq(2), 'third item').to.contain('Write JavaScript') | ||||
|       }) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										78
									
								
								cypress/examples/cookies.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,78 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Cookies', () => { | ||||
|   beforeEach(() => { | ||||
|     Cypress.Cookies.debug(true) | ||||
|  | ||||
|     cy.visit('https://example.cypress.io/commands/cookies') | ||||
|  | ||||
|     // clear cookies again after visiting to remove | ||||
|     // any 3rd party cookies picked up such as cloudflare | ||||
|     cy.clearCookies() | ||||
|   }) | ||||
|  | ||||
|   it('cy.getCookie() - get a browser cookie', () => { | ||||
|     // https://on.cypress.io/getcookie | ||||
|     cy.get('#getCookie .set-a-cookie').click() | ||||
|  | ||||
|     // cy.getCookie() yields a cookie object | ||||
|     cy.getCookie('token').should('have.property', 'value', '123ABC') | ||||
|   }) | ||||
|  | ||||
|   it('cy.getCookies() - get browser cookies', () => { | ||||
|     // https://on.cypress.io/getcookies | ||||
|     cy.getCookies().should('be.empty') | ||||
|  | ||||
|     cy.get('#getCookies .set-a-cookie').click() | ||||
|  | ||||
|     // cy.getCookies() yields an array of cookies | ||||
|     cy.getCookies().should('have.length', 1).should((cookies) => { | ||||
|  | ||||
|       // each cookie has these properties | ||||
|       expect(cookies[0]).to.have.property('name', 'token') | ||||
|       expect(cookies[0]).to.have.property('value', '123ABC') | ||||
|       expect(cookies[0]).to.have.property('httpOnly', false) | ||||
|       expect(cookies[0]).to.have.property('secure', false) | ||||
|       expect(cookies[0]).to.have.property('domain') | ||||
|       expect(cookies[0]).to.have.property('path') | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   it('cy.setCookie() - set a browser cookie', () => { | ||||
|     // https://on.cypress.io/setcookie | ||||
|     cy.getCookies().should('be.empty') | ||||
|  | ||||
|     cy.setCookie('foo', 'bar') | ||||
|  | ||||
|     // cy.getCookie() yields a cookie object | ||||
|     cy.getCookie('foo').should('have.property', 'value', 'bar') | ||||
|   }) | ||||
|  | ||||
|   it('cy.clearCookie() - clear a browser cookie', () => { | ||||
|     // https://on.cypress.io/clearcookie | ||||
|     cy.getCookie('token').should('be.null') | ||||
|  | ||||
|     cy.get('#clearCookie .set-a-cookie').click() | ||||
|  | ||||
|     cy.getCookie('token').should('have.property', 'value', '123ABC') | ||||
|  | ||||
|     // cy.clearCookies() yields null | ||||
|     cy.clearCookie('token').should('be.null') | ||||
|  | ||||
|     cy.getCookie('token').should('be.null') | ||||
|   }) | ||||
|  | ||||
|   it('cy.clearCookies() - clear browser cookies', () => { | ||||
|     // https://on.cypress.io/clearcookies | ||||
|     cy.getCookies().should('be.empty') | ||||
|  | ||||
|     cy.get('#clearCookies .set-a-cookie').click() | ||||
|  | ||||
|     cy.getCookies().should('have.length', 1) | ||||
|  | ||||
|     // cy.clearCookies() yields null | ||||
|     cy.clearCookies() | ||||
|  | ||||
|     cy.getCookies().should('be.empty') | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										222
									
								
								cypress/examples/cypress_api.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,222 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Cypress.Commands', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/cypress-api') | ||||
|   }) | ||||
|  | ||||
|   // https://on.cypress.io/custom-commands | ||||
|  | ||||
|   it('.add() - create a custom command', () => { | ||||
|     Cypress.Commands.add('console', { | ||||
|       prevSubject: true, | ||||
|     }, (subject, method) => { | ||||
|       // the previous subject is automatically received | ||||
|       // and the commands arguments are shifted | ||||
|  | ||||
|       // allow us to change the console method used | ||||
|       method = method || 'log' | ||||
|  | ||||
|       // log the subject to the console | ||||
|       // @ts-ignore TS7017 | ||||
|       console[method]('The subject is', subject) | ||||
|  | ||||
|       // whatever we return becomes the new subject | ||||
|       // we don't want to change the subject so | ||||
|       // we return whatever was passed in | ||||
|       return subject | ||||
|     }) | ||||
|  | ||||
|     // @ts-ignore TS2339 | ||||
|     cy.get('button').console('info').then(($button) => { | ||||
|       // subject is still $button | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
|  | ||||
| context('Cypress.Cookies', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/cypress-api') | ||||
|   }) | ||||
|  | ||||
|   // https://on.cypress.io/cookies | ||||
|   it('.debug() - enable or disable debugging', () => { | ||||
|     Cypress.Cookies.debug(true) | ||||
|  | ||||
|     // Cypress will now log in the console when | ||||
|     // cookies are set or cleared | ||||
|     cy.setCookie('fakeCookie', '123ABC') | ||||
|     cy.clearCookie('fakeCookie') | ||||
|     cy.setCookie('fakeCookie', '123ABC') | ||||
|     cy.clearCookie('fakeCookie') | ||||
|     cy.setCookie('fakeCookie', '123ABC') | ||||
|   }) | ||||
|  | ||||
|   it('.preserveOnce() - preserve cookies by key', () => { | ||||
|     // normally cookies are reset after each test | ||||
|     cy.getCookie('fakeCookie').should('not.be.ok') | ||||
|  | ||||
|     // preserving a cookie will not clear it when | ||||
|     // the next test starts | ||||
|     cy.setCookie('lastCookie', '789XYZ') | ||||
|     Cypress.Cookies.preserveOnce('lastCookie') | ||||
|   }) | ||||
|  | ||||
|   it('.defaults() - set defaults for all cookies', () => { | ||||
|     // now any cookie with the name 'session_id' will | ||||
|     // not be cleared before each new test runs | ||||
|     Cypress.Cookies.defaults({ | ||||
|       whitelist: 'session_id', | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
|  | ||||
| context('Cypress.Server', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/cypress-api') | ||||
|   }) | ||||
|  | ||||
|   // Permanently override server options for | ||||
|   // all instances of cy.server() | ||||
|  | ||||
|   // https://on.cypress.io/cypress-server | ||||
|   it('.defaults() - change default config of server', () => { | ||||
|     Cypress.Server.defaults({ | ||||
|       delay: 0, | ||||
|       force404: false, | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| context('Cypress.arch', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/cypress-api') | ||||
|   }) | ||||
|  | ||||
|   it('Get CPU architecture name of underlying OS', () => { | ||||
|     // https://on.cypress.io/arch | ||||
|     expect(Cypress.arch).to.exist | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| context('Cypress.config()', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/cypress-api') | ||||
|   }) | ||||
|  | ||||
|   it('Get and set configuration options', () => { | ||||
|     // https://on.cypress.io/config | ||||
|     let myConfig = Cypress.config() | ||||
|  | ||||
|     expect(myConfig).to.have.property('animationDistanceThreshold', 5) | ||||
|     expect(myConfig).to.have.property('baseUrl', null) | ||||
|     expect(myConfig).to.have.property('defaultCommandTimeout', 4000) | ||||
|     expect(myConfig).to.have.property('requestTimeout', 5000) | ||||
|     expect(myConfig).to.have.property('responseTimeout', 30000) | ||||
|     expect(myConfig).to.have.property('viewportHeight', 660) | ||||
|     expect(myConfig).to.have.property('viewportWidth', 1000) | ||||
|     expect(myConfig).to.have.property('pageLoadTimeout', 60000) | ||||
|     expect(myConfig).to.have.property('waitForAnimations', true) | ||||
|  | ||||
|     expect(Cypress.config('pageLoadTimeout')).to.eq(60000) | ||||
|  | ||||
|     // this will change the config for the rest of your tests! | ||||
|     Cypress.config('pageLoadTimeout', 20000) | ||||
|  | ||||
|     expect(Cypress.config('pageLoadTimeout')).to.eq(20000) | ||||
|  | ||||
|     Cypress.config('pageLoadTimeout', 60000) | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| context('Cypress.dom', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/cypress-api') | ||||
|   }) | ||||
|  | ||||
|   // https://on.cypress.io/dom | ||||
|   it('.isHidden() - determine if a DOM element is hidden', () => { | ||||
|     let hiddenP = Cypress.$('.dom-p p.hidden').get(0) | ||||
|     let visibleP = Cypress.$('.dom-p p.visible').get(0) | ||||
|  | ||||
|     // our first paragraph has css class 'hidden' | ||||
|     expect(Cypress.dom.isHidden(hiddenP)).to.be.true | ||||
|     expect(Cypress.dom.isHidden(visibleP)).to.be.false | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| context('Cypress.env()', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/cypress-api') | ||||
|   }) | ||||
|  | ||||
|   // We can set environment variables for highly dynamic values | ||||
|  | ||||
|   // https://on.cypress.io/environment-variables | ||||
|   it('Get environment variables', () => { | ||||
|     // https://on.cypress.io/env | ||||
|     // set multiple environment variables | ||||
|     Cypress.env({ | ||||
|       host: 'veronica.dev.local', | ||||
|       api_server: 'http://localhost:8888/v1/', | ||||
|     }) | ||||
|  | ||||
|     // get environment variable | ||||
|     expect(Cypress.env('host')).to.eq('veronica.dev.local') | ||||
|  | ||||
|     // set environment variable | ||||
|     Cypress.env('api_server', 'http://localhost:8888/v2/') | ||||
|     expect(Cypress.env('api_server')).to.eq('http://localhost:8888/v2/') | ||||
|  | ||||
|     // get all environment variable | ||||
|     expect(Cypress.env()).to.have.property('host', 'veronica.dev.local') | ||||
|     expect(Cypress.env()).to.have.property('api_server', 'http://localhost:8888/v2/') | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| context('Cypress.log', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/cypress-api') | ||||
|   }) | ||||
|  | ||||
|   it('Control what is printed to the Command Log', () => { | ||||
|     // https://on.cypress.io/cypress-log | ||||
|   }) | ||||
| }) | ||||
|  | ||||
|  | ||||
| context('Cypress.platform', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/cypress-api') | ||||
|   }) | ||||
|  | ||||
|   it('Get underlying OS name', () => { | ||||
|     // https://on.cypress.io/platform | ||||
|     expect(Cypress.platform).to.be.exist | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| context('Cypress.version', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/cypress-api') | ||||
|   }) | ||||
|  | ||||
|   it('Get current version of Cypress being run', () => { | ||||
|     // https://on.cypress.io/version | ||||
|     expect(Cypress.version).to.be.exist | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| context('Cypress.spec', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/cypress-api') | ||||
|   }) | ||||
|  | ||||
|   it('Get current spec information', () => { | ||||
|     // https://on.cypress.io/spec | ||||
|     // wrap the object so we can inspect it easily by clicking in the command log | ||||
|     cy.wrap(Cypress.spec).should('have.keys', ['name', 'relative', 'absolute']) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										114
									
								
								cypress/examples/files.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,114 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| /// JSON fixture file can be loaded directly using | ||||
| // the built-in JavaScript bundler | ||||
| // @ts-ignore | ||||
| const requiredExample = require('../../fixtures/example') | ||||
|  | ||||
| context('Files', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/files') | ||||
|   }) | ||||
|  | ||||
|   beforeEach(() => { | ||||
|     // load example.json fixture file and store | ||||
|     // in the test context object | ||||
|     cy.fixture('example.json').as('example') | ||||
|   }) | ||||
|  | ||||
|   it('cy.fixture() - load a fixture', () => { | ||||
|     // https://on.cypress.io/fixture | ||||
|  | ||||
|     // Instead of writing a response inline you can | ||||
|     // use a fixture file's content. | ||||
|  | ||||
|     cy.server() | ||||
|     cy.fixture('example.json').as('comment') | ||||
|     // when application makes an Ajax request matching "GET comments/*" | ||||
|     // Cypress will intercept it and reply with object | ||||
|     // from the "comment" alias | ||||
|     cy.route('GET', 'comments/*', '@comment').as('getComment') | ||||
|  | ||||
|     // we have code that gets a comment when | ||||
|     // the button is clicked in scripts.js | ||||
|     cy.get('.fixture-btn').click() | ||||
|  | ||||
|     cy.wait('@getComment').its('responseBody') | ||||
|       .should('have.property', 'name') | ||||
|       .and('include', 'Using fixtures to represent data') | ||||
|  | ||||
|     // you can also just write the fixture in the route | ||||
|     cy.route('GET', 'comments/*', 'fixture:example.json').as('getComment') | ||||
|  | ||||
|     // we have code that gets a comment when | ||||
|     // the button is clicked in scripts.js | ||||
|     cy.get('.fixture-btn').click() | ||||
|  | ||||
|     cy.wait('@getComment').its('responseBody') | ||||
|       .should('have.property', 'name') | ||||
|       .and('include', 'Using fixtures to represent data') | ||||
|  | ||||
|     // or write fx to represent fixture | ||||
|     // by default it assumes it's .json | ||||
|     cy.route('GET', 'comments/*', 'fx:example').as('getComment') | ||||
|  | ||||
|     // we have code that gets a comment when | ||||
|     // the button is clicked in scripts.js | ||||
|     cy.get('.fixture-btn').click() | ||||
|  | ||||
|     cy.wait('@getComment').its('responseBody') | ||||
|       .should('have.property', 'name') | ||||
|       .and('include', 'Using fixtures to represent data') | ||||
|   }) | ||||
|  | ||||
|   it('cy.fixture() or require - load a fixture', function () { | ||||
|     // we are inside the "function () { ... }" | ||||
|     // callback and can use test context object "this" | ||||
|     // "this.example" was loaded in "beforeEach" function callback | ||||
|     expect(this.example, 'fixture in the test context') | ||||
|       .to.deep.equal(requiredExample) | ||||
|  | ||||
|     // or use "cy.wrap" and "should('deep.equal', ...)" assertion | ||||
|     // @ts-ignore | ||||
|     cy.wrap(this.example, 'fixture vs require') | ||||
|       .should('deep.equal', requiredExample) | ||||
|   }) | ||||
|  | ||||
|   it('cy.readFile() - read a files contents', () => { | ||||
|     // https://on.cypress.io/readfile | ||||
|  | ||||
|     // You can read a file and yield its contents | ||||
|     // The filePath is relative to your project's root. | ||||
|     cy.readFile('cypress.json').then((json) => { | ||||
|       expect(json).to.be.an('object') | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   it('cy.writeFile() - write to a file', () => { | ||||
|     // https://on.cypress.io/writefile | ||||
|  | ||||
|     // You can write to a file | ||||
|  | ||||
|     // Use a response from a request to automatically | ||||
|     // generate a fixture file for use later | ||||
|     cy.request('https://jsonplaceholder.cypress.io/users') | ||||
|       .then((response) => { | ||||
|         cy.writeFile('cypress/fixtures/users.json', response.body) | ||||
|       }) | ||||
|     cy.fixture('users').should((users) => { | ||||
|       expect(users[0].name).to.exist | ||||
|     }) | ||||
|  | ||||
|     // JavaScript arrays and objects are stringified | ||||
|     // and formatted into text. | ||||
|     cy.writeFile('cypress/fixtures/profile.json', { | ||||
|       id: 8739, | ||||
|       name: 'Jane', | ||||
|       email: 'jane@example.com', | ||||
|     }) | ||||
|  | ||||
|     cy.fixture('profile').should((profile) => { | ||||
|       expect(profile.name).to.eq('Jane') | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										52
									
								
								cypress/examples/local_storage.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,52 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Local Storage', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/local-storage') | ||||
|   }) | ||||
|   // Although local storage is automatically cleared | ||||
|   // in between tests to maintain a clean state | ||||
|   // sometimes we need to clear the local storage manually | ||||
|  | ||||
|   it('cy.clearLocalStorage() - clear all data in local storage', () => { | ||||
|     // https://on.cypress.io/clearlocalstorage | ||||
|     cy.get('.ls-btn').click().should(() => { | ||||
|       expect(localStorage.getItem('prop1')).to.eq('red') | ||||
|       expect(localStorage.getItem('prop2')).to.eq('blue') | ||||
|       expect(localStorage.getItem('prop3')).to.eq('magenta') | ||||
|     }) | ||||
|  | ||||
|     // clearLocalStorage() yields the localStorage object | ||||
|     cy.clearLocalStorage().should((ls) => { | ||||
|       expect(ls.getItem('prop1')).to.be.null | ||||
|       expect(ls.getItem('prop2')).to.be.null | ||||
|       expect(ls.getItem('prop3')).to.be.null | ||||
|     }) | ||||
|  | ||||
|     // Clear key matching string in Local Storage | ||||
|     cy.get('.ls-btn').click().should(() => { | ||||
|       expect(localStorage.getItem('prop1')).to.eq('red') | ||||
|       expect(localStorage.getItem('prop2')).to.eq('blue') | ||||
|       expect(localStorage.getItem('prop3')).to.eq('magenta') | ||||
|     }) | ||||
|  | ||||
|     cy.clearLocalStorage('prop1').should((ls) => { | ||||
|       expect(ls.getItem('prop1')).to.be.null | ||||
|       expect(ls.getItem('prop2')).to.eq('blue') | ||||
|       expect(ls.getItem('prop3')).to.eq('magenta') | ||||
|     }) | ||||
|  | ||||
|     // Clear keys matching regex in Local Storage | ||||
|     cy.get('.ls-btn').click().should(() => { | ||||
|       expect(localStorage.getItem('prop1')).to.eq('red') | ||||
|       expect(localStorage.getItem('prop2')).to.eq('blue') | ||||
|       expect(localStorage.getItem('prop3')).to.eq('magenta') | ||||
|     }) | ||||
|  | ||||
|     cy.clearLocalStorage(/prop1|2/).should((ls) => { | ||||
|       expect(ls.getItem('prop1')).to.be.null | ||||
|       expect(ls.getItem('prop2')).to.be.null | ||||
|       expect(ls.getItem('prop3')).to.eq('magenta') | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										32
									
								
								cypress/examples/location.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,32 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Location', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/location') | ||||
|   }) | ||||
|  | ||||
|   it('cy.hash() - get the current URL hash', () => { | ||||
|     // https://on.cypress.io/hash | ||||
|     cy.hash().should('be.empty') | ||||
|   }) | ||||
|  | ||||
|   it('cy.location() - get window.location', () => { | ||||
|     // https://on.cypress.io/location | ||||
|     cy.location().should((location) => { | ||||
|       expect(location.hash).to.be.empty | ||||
|       expect(location.href).to.eq('https://example.cypress.io/commands/location') | ||||
|       expect(location.host).to.eq('example.cypress.io') | ||||
|       expect(location.hostname).to.eq('example.cypress.io') | ||||
|       expect(location.origin).to.eq('https://example.cypress.io') | ||||
|       expect(location.pathname).to.eq('/commands/location') | ||||
|       expect(location.port).to.eq('') | ||||
|       expect(location.protocol).to.eq('https:') | ||||
|       expect(location.search).to.be.empty | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   it('cy.url() - get the current URL', () => { | ||||
|     // https://on.cypress.io/url | ||||
|     cy.url().should('eq', 'https://example.cypress.io/commands/location') | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										83
									
								
								cypress/examples/misc.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,83 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Misc', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/misc') | ||||
|   }) | ||||
|  | ||||
|   it('.end() - end the command chain', () => { | ||||
|     // https://on.cypress.io/end | ||||
|  | ||||
|     // cy.end is useful when you want to end a chain of commands | ||||
|     // and force Cypress to re-query from the root element | ||||
|     cy.get('.misc-table').within(() => { | ||||
|       // ends the current chain and yields null | ||||
|       cy.contains('Cheryl').click().end() | ||||
|  | ||||
|       // queries the entire table again | ||||
|       cy.contains('Charles').click() | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   it('cy.exec() - execute a system command', () => { | ||||
|     // https://on.cypress.io/exec | ||||
|  | ||||
|     // execute a system command. | ||||
|     // so you can take actions necessary for | ||||
|     // your test outside the scope of Cypress. | ||||
|     cy.exec('echo Jane Lane') | ||||
|       .its('stdout').should('contain', 'Jane Lane') | ||||
|  | ||||
|     // we can use Cypress.platform string to | ||||
|     // select appropriate command | ||||
|     // https://on.cypress/io/platform | ||||
|     cy.log(`Platform ${Cypress.platform} architecture ${Cypress.arch}`) | ||||
|  | ||||
|     if (Cypress.platform === 'win32') { | ||||
|       cy.exec('print cypress.json') | ||||
|         .its('stderr').should('be.empty') | ||||
|     } else { | ||||
|       cy.exec('cat cypress.json') | ||||
|         .its('stderr').should('be.empty') | ||||
|  | ||||
|       cy.exec('pwd') | ||||
|         .its('code').should('eq', 0) | ||||
|     } | ||||
|   }) | ||||
|  | ||||
|   it('cy.focused() - get the DOM element that has focus', () => { | ||||
|     // https://on.cypress.io/focused | ||||
|     cy.get('.misc-form').find('#name').click() | ||||
|     cy.focused().should('have.id', 'name') | ||||
|  | ||||
|     cy.get('.misc-form').find('#description').click() | ||||
|     cy.focused().should('have.id', 'description') | ||||
|   }) | ||||
|  | ||||
|   context('Cypress.Screenshot', function () { | ||||
|     it('cy.screenshot() - take a screenshot', () => { | ||||
|       // https://on.cypress.io/screenshot | ||||
|       cy.screenshot('my-image') | ||||
|     }) | ||||
|  | ||||
|     it('Cypress.Screenshot.defaults() - change default config of screenshots', function () { | ||||
|       Cypress.Screenshot.defaults({ | ||||
|         blackout: ['.foo'], | ||||
|         capture: 'viewport', | ||||
|         clip: { x: 0, y: 0, width: 200, height: 200 }, | ||||
|         scale: false, | ||||
|         disableTimersAndAnimations: true, | ||||
|         screenshotOnRunFailure: true, | ||||
|         beforeScreenshot () { }, | ||||
|         afterScreenshot () { }, | ||||
|       }) | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   it('cy.wrap() - wrap an object', () => { | ||||
|     // https://on.cypress.io/wrap | ||||
|     cy.wrap({ foo: 'bar' }) | ||||
|       .should('have.property', 'foo') | ||||
|       .and('include', 'bar') | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										56
									
								
								cypress/examples/navigation.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,56 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Navigation', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io') | ||||
|     cy.get('.navbar-nav').contains('Commands').click() | ||||
|     cy.get('.dropdown-menu').contains('Navigation').click() | ||||
|   }) | ||||
|  | ||||
|   it('cy.go() - go back or forward in the browser\'s history', () => { | ||||
|     // https://on.cypress.io/go | ||||
|  | ||||
|     cy.location('pathname').should('include', 'navigation') | ||||
|  | ||||
|     cy.go('back') | ||||
|     cy.location('pathname').should('not.include', 'navigation') | ||||
|  | ||||
|     cy.go('forward') | ||||
|     cy.location('pathname').should('include', 'navigation') | ||||
|  | ||||
|     // clicking back | ||||
|     cy.go(-1) | ||||
|     cy.location('pathname').should('not.include', 'navigation') | ||||
|  | ||||
|     // clicking forward | ||||
|     cy.go(1) | ||||
|     cy.location('pathname').should('include', 'navigation') | ||||
|   }) | ||||
|  | ||||
|   it('cy.reload() - reload the page', () => { | ||||
|     // https://on.cypress.io/reload | ||||
|     cy.reload() | ||||
|  | ||||
|     // reload the page without using the cache | ||||
|     cy.reload(true) | ||||
|   }) | ||||
|  | ||||
|   it('cy.visit() - visit a remote url', () => { | ||||
|     // https://on.cypress.io/visit | ||||
|  | ||||
|     // Visit any sub-domain of your current domain | ||||
|  | ||||
|     // Pass options to the visit | ||||
|     cy.visit('https://example.cypress.io/commands/navigation', { | ||||
|       timeout: 50000, // increase total time for the visit to resolve | ||||
|       onBeforeLoad (contentWindow) { | ||||
|         // contentWindow is the remote page's window object | ||||
|         expect(typeof contentWindow === 'object').to.be.true | ||||
|       }, | ||||
|       onLoad (contentWindow) { | ||||
|         // contentWindow is the remote page's window object | ||||
|         expect(typeof contentWindow === 'object').to.be.true | ||||
|       }, | ||||
|     }) | ||||
|     }) | ||||
| }) | ||||
							
								
								
									
										194
									
								
								cypress/examples/network_requests.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,194 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Network Requests', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/network-requests') | ||||
|   }) | ||||
|  | ||||
|   // Manage AJAX / XHR requests in your app | ||||
|  | ||||
|   it('cy.server() - control behavior of network requests and responses', () => { | ||||
|     // https://on.cypress.io/server | ||||
|  | ||||
|     cy.server().should((server) => { | ||||
|       // the default options on server | ||||
|       // you can override any of these options | ||||
|       expect(server.delay).to.eq(0) | ||||
|       expect(server.method).to.eq('GET') | ||||
|       expect(server.status).to.eq(200) | ||||
|       expect(server.headers).to.be.null | ||||
|       expect(server.response).to.be.null | ||||
|       expect(server.onRequest).to.be.undefined | ||||
|       expect(server.onResponse).to.be.undefined | ||||
|       expect(server.onAbort).to.be.undefined | ||||
|  | ||||
|       // These options control the server behavior | ||||
|       // affecting all requests | ||||
|  | ||||
|       // pass false to disable existing route stubs | ||||
|       expect(server.enable).to.be.true | ||||
|       // forces requests that don't match your routes to 404 | ||||
|       expect(server.force404).to.be.false | ||||
|       // whitelists requests from ever being logged or stubbed | ||||
|       expect(server.whitelist).to.be.a('function') | ||||
|     }) | ||||
|  | ||||
|     cy.server({ | ||||
|       method: 'POST', | ||||
|       delay: 1000, | ||||
|       status: 422, | ||||
|       response: {}, | ||||
|     }) | ||||
|  | ||||
|     // any route commands will now inherit the above options | ||||
|     // from the server. anything we pass specifically | ||||
|     // to route will override the defaults though. | ||||
|   }) | ||||
|  | ||||
|   it('cy.request() - make an XHR request', () => { | ||||
|     // https://on.cypress.io/request | ||||
|     cy.request('https://jsonplaceholder.cypress.io/comments') | ||||
|       .should((response) => { | ||||
|         expect(response.status).to.eq(200) | ||||
|         expect(response.body).to.have.length(500) | ||||
|         expect(response).to.have.property('headers') | ||||
|         expect(response).to.have.property('duration') | ||||
|       }) | ||||
|   }) | ||||
|  | ||||
|  | ||||
|   it('cy.request() - verify response using BDD syntax', () => { | ||||
|     cy.request('https://jsonplaceholder.cypress.io/comments') | ||||
|     .then((response) => { | ||||
|       // https://on.cypress.io/assertions | ||||
|       expect(response).property('status').to.equal(200) | ||||
|       expect(response).property('body').to.have.length(500) | ||||
|       expect(response).to.include.keys('headers', 'duration') | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   it('cy.request() with query parameters', () => { | ||||
|     // will execute request | ||||
|     // https://jsonplaceholder.cypress.io/comments?postId=1&id=3 | ||||
|     cy.request({ | ||||
|       url: 'https://jsonplaceholder.cypress.io/comments', | ||||
|       qs: { | ||||
|         postId: 1, | ||||
|         id: 3, | ||||
|       }, | ||||
|     }) | ||||
|     .its('body') | ||||
|     .should('be.an', 'array') | ||||
|     .and('have.length', 1) | ||||
|     .its('0') // yields first element of the array | ||||
|     .should('contain', { | ||||
|       postId: 1, | ||||
|       id: 3, | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   it('cy.request() - pass result to the second request', () => { | ||||
|     // first, let's find out the userId of the first user we have | ||||
|     cy.request('https://jsonplaceholder.cypress.io/users?_limit=1') | ||||
|       .its('body.0') // yields the first element of the returned list | ||||
|       .then((user) => { | ||||
|         expect(user).property('id').to.be.a('number') | ||||
|         // make a new post on behalf of the user | ||||
|         cy.request('POST', 'https://jsonplaceholder.cypress.io/posts', { | ||||
|           userId: user.id, | ||||
|           title: 'Cypress Test Runner', | ||||
|           body: 'Fast, easy and reliable testing for anything that runs in a browser.', | ||||
|         }) | ||||
|       }) | ||||
|       // note that the value here is the returned value of the 2nd request | ||||
|       // which is the new post object | ||||
|       .then((response) => { | ||||
|         expect(response).property('status').to.equal(201) // new entity created | ||||
|         expect(response).property('body').to.contain({ | ||||
|           id: 101, // there are already 100 posts, so new entity gets id 101 | ||||
|           title: 'Cypress Test Runner', | ||||
|         }) | ||||
|         // we don't know the user id here - since it was in above closure | ||||
|         // so in this test just confirm that the property is there | ||||
|         expect(response.body).property('userId').to.be.a('number') | ||||
|       }) | ||||
|   }) | ||||
|  | ||||
|   it('cy.request() - save response in the shared test context', () => { | ||||
|     // https://on.cypress.io/variables-and-aliases | ||||
|     cy.request('https://jsonplaceholder.cypress.io/users?_limit=1') | ||||
|       .its('body.0') // yields the first element of the returned list | ||||
|       .as('user') // saves the object in the test context | ||||
|       .then(function () { | ||||
|         // NOTE 👀 | ||||
|         //  By the time this callback runs the "as('user')" command | ||||
|         //  has saved the user object in the test context. | ||||
|         //  To access the test context we need to use | ||||
|         //  the "function () { ... }" callback form, | ||||
|         //  otherwise "this" points at a wrong or undefined object! | ||||
|         cy.request('POST', 'https://jsonplaceholder.cypress.io/posts', { | ||||
|           userId: this.user.id, | ||||
|           title: 'Cypress Test Runner', | ||||
|           body: 'Fast, easy and reliable testing for anything that runs in a browser.', | ||||
|         }) | ||||
|         .its('body').as('post') // save the new post from the response | ||||
|       }) | ||||
|       .then(function () { | ||||
|         // When this callback runs, both "cy.request" API commands have finished | ||||
|         // and the test context has "user" and "post" objects set. | ||||
|         // Let's verify them. | ||||
|         expect(this.post, 'post has the right user id').property('userId').to.equal(this.user.id) | ||||
|       }) | ||||
|   }) | ||||
|  | ||||
|   it('cy.route() - route responses to matching requests', () => { | ||||
|     // https://on.cypress.io/route | ||||
|  | ||||
|     let message = 'whoa, this comment does not exist' | ||||
|  | ||||
|     cy.server() | ||||
|  | ||||
|     // Listen to GET to comments/1 | ||||
|     cy.route('GET', 'comments/*').as('getComment') | ||||
|  | ||||
|     // we have code that gets a comment when | ||||
|     // the button is clicked in scripts.js | ||||
|     cy.get('.network-btn').click() | ||||
|  | ||||
|     // https://on.cypress.io/wait | ||||
|     cy.wait('@getComment').its('status').should('eq', 200) | ||||
|  | ||||
|     // Listen to POST to comments | ||||
|     cy.route('POST', '/comments').as('postComment') | ||||
|  | ||||
|     // we have code that posts a comment when | ||||
|     // the button is clicked in scripts.js | ||||
|     cy.get('.network-post').click() | ||||
|     cy.wait('@postComment') | ||||
|  | ||||
|     // get the route | ||||
|     cy.get('@postComment').should((xhr) => { | ||||
|       expect(xhr.requestBody).to.include('email') | ||||
|       expect(xhr.requestHeaders).to.have.property('Content-Type') | ||||
|       expect(xhr.responseBody).to.have.property('name', 'Using POST in cy.route()') | ||||
|     }) | ||||
|  | ||||
|     // Stub a response to PUT comments/ **** | ||||
|     cy.route({ | ||||
|       method: 'PUT', | ||||
|       url: 'comments/*', | ||||
|       status: 404, | ||||
|       response: { error: message }, | ||||
|       delay: 500, | ||||
|     }).as('putComment') | ||||
|  | ||||
|     // we have code that puts a comment when | ||||
|     // the button is clicked in scripts.js | ||||
|     cy.get('.network-put').click() | ||||
|  | ||||
|     cy.wait('@putComment') | ||||
|  | ||||
|     // our 404 statusCode logic in scripts.js executed | ||||
|     cy.get('.network-put-comment').should('contain', message) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										87
									
								
								cypress/examples/querying.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,87 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Querying', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/querying') | ||||
|   }) | ||||
|  | ||||
|   // The most commonly used query is 'cy.get()', you can | ||||
|   // think of this like the '$' in jQuery | ||||
|  | ||||
|   it('cy.get() - query DOM elements', () => { | ||||
|     // https://on.cypress.io/get | ||||
|  | ||||
|     cy.get('#query-btn').should('contain', 'Button') | ||||
|  | ||||
|     cy.get('.query-btn').should('contain', 'Button') | ||||
|  | ||||
|     cy.get('#querying .well>button:first').should('contain', 'Button') | ||||
|     //              ↲ | ||||
|     // Use CSS selectors just like jQuery | ||||
|  | ||||
|     cy.get('[data-test-id="test-example"]').should('have.class', 'example') | ||||
|  | ||||
|     // 'cy.get()' yields jQuery object, you can get its attribute | ||||
|     // by invoking `.attr()` method | ||||
|     cy.get('[data-test-id="test-example"]') | ||||
|       .invoke('attr', 'data-test-id') | ||||
|       .should('equal', 'test-example') | ||||
|  | ||||
|     // or you can get element's CSS property | ||||
|     cy.get('[data-test-id="test-example"]') | ||||
|       .invoke('css', 'position') | ||||
|       .should('equal', 'static') | ||||
|  | ||||
|     // or use assertions directly during 'cy.get()' | ||||
|     // https://on.cypress.io/assertions | ||||
|     cy.get('[data-test-id="test-example"]') | ||||
|       .should('have.attr', 'data-test-id', 'test-example') | ||||
|       .and('have.css', 'position', 'static') | ||||
|   }) | ||||
|  | ||||
|   it('cy.contains() - query DOM elements with matching content', () => { | ||||
|     // https://on.cypress.io/contains | ||||
|     cy.get('.query-list') | ||||
|       .contains('bananas') | ||||
|       .should('have.class', 'third') | ||||
|  | ||||
|     // we can pass a regexp to `.contains()` | ||||
|     cy.get('.query-list') | ||||
|       .contains(/^b\w+/) | ||||
|       .should('have.class', 'third') | ||||
|  | ||||
|     cy.get('.query-list') | ||||
|       .contains('apples') | ||||
|       .should('have.class', 'first') | ||||
|  | ||||
|     // passing a selector to contains will | ||||
|     // yield the selector containing the text | ||||
|     cy.get('#querying') | ||||
|       .contains('ul', 'oranges') | ||||
|       .should('have.class', 'query-list') | ||||
|  | ||||
|     cy.get('.query-button') | ||||
|       .contains('Save Form') | ||||
|       .should('have.class', 'btn') | ||||
|   }) | ||||
|  | ||||
|   it('.within() - query DOM elements within a specific element', () => { | ||||
|     // https://on.cypress.io/within | ||||
|     cy.get('.query-form').within(() => { | ||||
|       cy.get('input:first').should('have.attr', 'placeholder', 'Email') | ||||
|       cy.get('input:last').should('have.attr', 'placeholder', 'Password') | ||||
|     }) | ||||
|   }) | ||||
|  | ||||
|   it('cy.root() - query the root DOM element', () => { | ||||
|     // https://on.cypress.io/root | ||||
|  | ||||
|     // By default, root is the document | ||||
|     cy.root().should('match', 'html') | ||||
|  | ||||
|     cy.get('.query-ul').within(() => { | ||||
|       // In this within, the root is now the ul DOM element | ||||
|       cy.root().should('have.class', 'query-ul') | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										95
									
								
								cypress/examples/spies_stubs_clocks.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,95 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Spies, Stubs, and Clock', () => { | ||||
|   it('cy.spy() - wrap a method in a spy', () => { | ||||
|     // https://on.cypress.io/spy | ||||
|     cy.visit('https://example.cypress.io/commands/spies-stubs-clocks') | ||||
|  | ||||
|     const obj = { | ||||
|       foo () {}, | ||||
|     } | ||||
|  | ||||
|     const spy = cy.spy(obj, 'foo').as('anyArgs') | ||||
|  | ||||
|     obj.foo() | ||||
|  | ||||
|     expect(spy).to.be.called | ||||
|   }) | ||||
|  | ||||
|   it('cy.spy() retries until assertions pass', () => { | ||||
|     cy.visit('https://example.cypress.io/commands/spies-stubs-clocks') | ||||
|  | ||||
|     const obj = { | ||||
|       /** | ||||
|        * Prints the argument passed | ||||
|        * @param x {any} | ||||
|       */ | ||||
|       foo (x) { | ||||
|         console.log('obj.foo called with', x) | ||||
|       }, | ||||
|     } | ||||
|  | ||||
|     cy.spy(obj, 'foo').as('foo') | ||||
|  | ||||
|     setTimeout(() => { | ||||
|       obj.foo('first') | ||||
|     }, 500) | ||||
|  | ||||
|     setTimeout(() => { | ||||
|       obj.foo('second') | ||||
|     }, 2500) | ||||
|  | ||||
|     cy.get('@foo').should('have.been.calledTwice') | ||||
|   }) | ||||
|  | ||||
|   it('cy.stub() - create a stub and/or replace a function with stub', () => { | ||||
|     // https://on.cypress.io/stub | ||||
|     cy.visit('https://example.cypress.io/commands/spies-stubs-clocks') | ||||
|  | ||||
|     const obj = { | ||||
|       /** | ||||
|        * prints both arguments to the console | ||||
|        * @param a {string} | ||||
|        * @param b {string} | ||||
|       */ | ||||
|       foo (a, b) { | ||||
|         console.log('a', a, 'b', b) | ||||
|       }, | ||||
|     } | ||||
|  | ||||
|     const stub = cy.stub(obj, 'foo').as('foo') | ||||
|  | ||||
|     obj.foo('foo', 'bar') | ||||
|  | ||||
|     expect(stub).to.be.called | ||||
|   }) | ||||
|  | ||||
|   it('cy.clock() - control time in the browser', () => { | ||||
|     // https://on.cypress.io/clock | ||||
|  | ||||
|     // create the date in UTC so its always the same | ||||
|     // no matter what local timezone the browser is running in | ||||
|     const now = new Date(Date.UTC(2017, 2, 14)).getTime() | ||||
|  | ||||
|     cy.clock(now) | ||||
|     cy.visit('https://example.cypress.io/commands/spies-stubs-clocks') | ||||
|     cy.get('#clock-div').click() | ||||
|       .should('have.text', '1489449600') | ||||
|   }) | ||||
|  | ||||
|   it('cy.tick() - move time in the browser', () => { | ||||
|     // https://on.cypress.io/tick | ||||
|  | ||||
|     // create the date in UTC so its always the same | ||||
|     // no matter what local timezone the browser is running in | ||||
|     const now = new Date(Date.UTC(2017, 2, 14)).getTime() | ||||
|  | ||||
|     cy.clock(now) | ||||
|     cy.visit('https://example.cypress.io/commands/spies-stubs-clocks') | ||||
|     cy.get('#tick-div').click() | ||||
|       .should('have.text', '1489449600') | ||||
|     cy.tick(10000) // 10 seconds passed | ||||
|     cy.get('#tick-div').click() | ||||
|       .should('have.text', '1489449610') | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										121
									
								
								cypress/examples/traversal.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,121 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Traversal', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/traversal') | ||||
|   }) | ||||
|  | ||||
|   it('.children() - get child DOM elements', () => { | ||||
|     // https://on.cypress.io/children | ||||
|     cy.get('.traversal-breadcrumb') | ||||
|       .children('.active') | ||||
|       .should('contain', 'Data') | ||||
|   }) | ||||
|  | ||||
|   it('.closest() - get closest ancestor DOM element', () => { | ||||
|     // https://on.cypress.io/closest | ||||
|     cy.get('.traversal-badge') | ||||
|       .closest('ul') | ||||
|       .should('have.class', 'list-group') | ||||
|   }) | ||||
|  | ||||
|   it('.eq() - get a DOM element at a specific index', () => { | ||||
|     // https://on.cypress.io/eq | ||||
|     cy.get('.traversal-list>li') | ||||
|       .eq(1).should('contain', 'siamese') | ||||
|   }) | ||||
|  | ||||
|   it('.filter() - get DOM elements that match the selector', () => { | ||||
|     // https://on.cypress.io/filter | ||||
|     cy.get('.traversal-nav>li') | ||||
|       .filter('.active').should('contain', 'About') | ||||
|   }) | ||||
|  | ||||
|   it('.find() - get descendant DOM elements of the selector', () => { | ||||
|     // https://on.cypress.io/find | ||||
|     cy.get('.traversal-pagination') | ||||
|       .find('li').find('a') | ||||
|       .should('have.length', 7) | ||||
|   }) | ||||
|  | ||||
|   it('.first() - get first DOM element', () => { | ||||
|     // https://on.cypress.io/first | ||||
|     cy.get('.traversal-table td') | ||||
|       .first().should('contain', '1') | ||||
|   }) | ||||
|  | ||||
|   it('.last() - get last DOM element', () => { | ||||
|     // https://on.cypress.io/last | ||||
|     cy.get('.traversal-buttons .btn') | ||||
|       .last().should('contain', 'Submit') | ||||
|   }) | ||||
|  | ||||
|   it('.next() - get next sibling DOM element', () => { | ||||
|     // https://on.cypress.io/next | ||||
|     cy.get('.traversal-ul') | ||||
|       .contains('apples').next().should('contain', 'oranges') | ||||
|   }) | ||||
|  | ||||
|   it('.nextAll() - get all next sibling DOM elements', () => { | ||||
|     // https://on.cypress.io/nextall | ||||
|     cy.get('.traversal-next-all') | ||||
|       .contains('oranges') | ||||
|       .nextAll().should('have.length', 3) | ||||
|   }) | ||||
|  | ||||
|   it('.nextUntil() - get next sibling DOM elements until next el', () => { | ||||
|     // https://on.cypress.io/nextuntil | ||||
|     cy.get('#veggies') | ||||
|       .nextUntil('#nuts').should('have.length', 3) | ||||
|   }) | ||||
|  | ||||
|   it('.not() - remove DOM elements from set of DOM elements', () => { | ||||
|     // https://on.cypress.io/not | ||||
|     cy.get('.traversal-disabled .btn') | ||||
|       .not('[disabled]').should('not.contain', 'Disabled') | ||||
|   }) | ||||
|  | ||||
|   it('.parent() - get parent DOM element from DOM elements', () => { | ||||
|     // https://on.cypress.io/parent | ||||
|     cy.get('.traversal-mark') | ||||
|       .parent().should('contain', 'Morbi leo risus') | ||||
|   }) | ||||
|  | ||||
|   it('.parents() - get parent DOM elements from DOM elements', () => { | ||||
|     // https://on.cypress.io/parents | ||||
|     cy.get('.traversal-cite') | ||||
|       .parents().should('match', 'blockquote') | ||||
|   }) | ||||
|  | ||||
|   it('.parentsUntil() - get parent DOM elements from DOM elements until el', () => { | ||||
|     // https://on.cypress.io/parentsuntil | ||||
|     cy.get('.clothes-nav') | ||||
|       .find('.active') | ||||
|       .parentsUntil('.clothes-nav') | ||||
|       .should('have.length', 2) | ||||
|   }) | ||||
|  | ||||
|   it('.prev() - get previous sibling DOM element', () => { | ||||
|     // https://on.cypress.io/prev | ||||
|     cy.get('.birds').find('.active') | ||||
|       .prev().should('contain', 'Lorikeets') | ||||
|   }) | ||||
|  | ||||
|   it('.prevAll() - get all previous sibling DOM elements', () => { | ||||
|     // https://on.cypress.io/prevAll | ||||
|     cy.get('.fruits-list').find('.third') | ||||
|       .prevAll().should('have.length', 2) | ||||
|   }) | ||||
|  | ||||
|   it('.prevUntil() - get all previous sibling DOM elements until el', () => { | ||||
|     // https://on.cypress.io/prevUntil | ||||
|     cy.get('.foods-list').find('#nuts') | ||||
|       .prevUntil('#veggies').should('have.length', 3) | ||||
|   }) | ||||
|  | ||||
|   it('.siblings() - get all sibling DOM elements', () => { | ||||
|     // https://on.cypress.io/siblings | ||||
|     cy.get('.traversal-pills .active') | ||||
|       .siblings().should('have.length', 2) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										133
									
								
								cypress/examples/utilities.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,133 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Utilities', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/utilities') | ||||
|   }) | ||||
|  | ||||
|   it('Cypress._ - call a lodash method', () => { | ||||
|     // https://on.cypress.io/_ | ||||
|     cy.request('https://jsonplaceholder.cypress.io/users') | ||||
|       .then((response) => { | ||||
|         let ids = Cypress._.chain(response.body).map('id').take(3).value() | ||||
|  | ||||
|         expect(ids).to.deep.eq([1, 2, 3]) | ||||
|       }) | ||||
|   }) | ||||
|  | ||||
|   it('Cypress.$ - call a jQuery method', () => { | ||||
|     // https://on.cypress.io/$ | ||||
|     let $li = Cypress.$('.utility-jquery li:first') | ||||
|  | ||||
|     cy.wrap($li) | ||||
|       .should('not.have.class', 'active') | ||||
|       .click() | ||||
|       .should('have.class', 'active') | ||||
|   }) | ||||
|  | ||||
|   it('Cypress.Blob - blob utilities and base64 string conversion', () => { | ||||
|     // https://on.cypress.io/blob | ||||
|     cy.get('.utility-blob').then(($div) => | ||||
|     // https://github.com/nolanlawson/blob-util#imgSrcToDataURL | ||||
|     // get the dataUrl string for the javascript-logo | ||||
|       Cypress.Blob.imgSrcToDataURL('https://example.cypress.io/assets/img/javascript-logo.png', undefined, 'anonymous') | ||||
|       .then((dataUrl) => { | ||||
|         // create an <img> element and set its src to the dataUrl | ||||
|         let img = Cypress.$('<img />', { src: dataUrl }) | ||||
|  | ||||
|         // need to explicitly return cy here since we are initially returning | ||||
|         // the Cypress.Blob.imgSrcToDataURL promise to our test | ||||
|         // append the image | ||||
|         $div.append(img) | ||||
|  | ||||
|         cy.get('.utility-blob img').click() | ||||
|           .should('have.attr', 'src', dataUrl) | ||||
|       })) | ||||
|   }) | ||||
|  | ||||
|   it('Cypress.minimatch - test out glob patterns against strings', () => { | ||||
|     // https://on.cypress.io/minimatch | ||||
|     let matching = Cypress.minimatch('/users/1/comments', '/users/*/comments', { | ||||
|       matchBase: true, | ||||
|     }) | ||||
|  | ||||
|     expect(matching, 'matching wildcard').to.be.true | ||||
|  | ||||
|     matching = Cypress.minimatch('/users/1/comments/2', '/users/*/comments', { | ||||
|       matchBase: true, | ||||
|     }) | ||||
|     expect(matching, 'comments').to.be.false | ||||
|  | ||||
|     // ** matches against all downstream path segments | ||||
|     matching = Cypress.minimatch('/foo/bar/baz/123/quux?a=b&c=2', '/foo/**', { | ||||
|       matchBase: true, | ||||
|     }) | ||||
|     expect(matching, 'comments').to.be.true | ||||
|  | ||||
|     // whereas * matches only the next path segment | ||||
|  | ||||
|     matching = Cypress.minimatch('/foo/bar/baz/123/quux?a=b&c=2', '/foo/*', { | ||||
|       matchBase: false, | ||||
|     }) | ||||
|     expect(matching, 'comments').to.be.false | ||||
|   }) | ||||
|  | ||||
|  | ||||
|   it('Cypress.moment() - format or parse dates using a moment method', () => { | ||||
|     // https://on.cypress.io/moment | ||||
|     const time = Cypress.moment().utc('2014-04-25T19:38:53.196Z').format('h:mm A') | ||||
|  | ||||
|     expect(time).to.be.a('string') | ||||
|  | ||||
|     cy.get('.utility-moment').contains('3:38 PM') | ||||
|       .should('have.class', 'badge') | ||||
|  | ||||
|     // the time in the element should be between 3pm and 5pm | ||||
|     const start = Cypress.moment('3:00 PM', 'LT') | ||||
|     const end = Cypress.moment('5:00 PM', 'LT') | ||||
|  | ||||
|     cy.get('.utility-moment .badge') | ||||
|       .should(($el) => { | ||||
|         // parse American time like "3:38 PM" | ||||
|         const m = Cypress.moment($el.text().trim(), 'LT') | ||||
|  | ||||
|         // display hours + minutes + AM|PM | ||||
|         const f = 'h:mm A' | ||||
|  | ||||
|         expect(m.isBetween(start, end), | ||||
|           `${m.format(f)} should be between ${start.format(f)} and ${end.format(f)}`).to.be.true | ||||
|       }) | ||||
|   }) | ||||
|  | ||||
|  | ||||
|   it('Cypress.Promise - instantiate a bluebird promise', () => { | ||||
|     // https://on.cypress.io/promise | ||||
|     let waited = false | ||||
|  | ||||
|     /** | ||||
|      * @return Bluebird<string> | ||||
|      */ | ||||
|     function waitOneSecond () { | ||||
|       // return a promise that resolves after 1 second | ||||
|       // @ts-ignore TS2351 (new Cypress.Promise) | ||||
|       return new Cypress.Promise((resolve, reject) => { | ||||
|         setTimeout(() => { | ||||
|           // set waited to true | ||||
|           waited = true | ||||
|  | ||||
|           // resolve with 'foo' string | ||||
|           resolve('foo') | ||||
|         }, 1000) | ||||
|       }) | ||||
|     } | ||||
|  | ||||
|     cy.then(() => | ||||
|     // return a promise to cy.then() that | ||||
|     // is awaited until it resolves | ||||
|       // @ts-ignore TS7006 | ||||
|       waitOneSecond().then((str) => { | ||||
|         expect(str).to.eq('foo') | ||||
|         expect(waited).to.be.true | ||||
|       })) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										59
									
								
								cypress/examples/viewport.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,59 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Viewport', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/viewport') | ||||
|   }) | ||||
|  | ||||
|   it('cy.viewport() - set the viewport size and dimension', () => { | ||||
|     // https://on.cypress.io/viewport | ||||
|  | ||||
|     cy.get('#navbar').should('be.visible') | ||||
|     cy.viewport(320, 480) | ||||
|  | ||||
|     // the navbar should have collapse since our screen is smaller | ||||
|     cy.get('#navbar').should('not.be.visible') | ||||
|     cy.get('.navbar-toggle').should('be.visible').click() | ||||
|     cy.get('.nav').find('a').should('be.visible') | ||||
|  | ||||
|     // lets see what our app looks like on a super large screen | ||||
|     cy.viewport(2999, 2999) | ||||
|  | ||||
|     // cy.viewport() accepts a set of preset sizes | ||||
|     // to easily set the screen to a device's width and height | ||||
|  | ||||
|     // We added a cy.wait() between each viewport change so you can see | ||||
|     // the change otherwise it is a little too fast to see :) | ||||
|  | ||||
|     cy.viewport('macbook-15') | ||||
|     cy.wait(200) | ||||
|     cy.viewport('macbook-13') | ||||
|     cy.wait(200) | ||||
|     cy.viewport('macbook-11') | ||||
|     cy.wait(200) | ||||
|     cy.viewport('ipad-2') | ||||
|     cy.wait(200) | ||||
|     cy.viewport('ipad-mini') | ||||
|     cy.wait(200) | ||||
|     cy.viewport('iphone-6+') | ||||
|     cy.wait(200) | ||||
|     cy.viewport('iphone-6') | ||||
|     cy.wait(200) | ||||
|     cy.viewport('iphone-5') | ||||
|     cy.wait(200) | ||||
|     cy.viewport('iphone-4') | ||||
|     cy.wait(200) | ||||
|     cy.viewport('iphone-3') | ||||
|     cy.wait(200) | ||||
|  | ||||
|     // cy.viewport() accepts an orientation for all presets | ||||
|     // the default orientation is 'portrait' | ||||
|     cy.viewport('ipad-2', 'portrait') | ||||
|     cy.wait(200) | ||||
|     cy.viewport('iphone-4', 'landscape') | ||||
|     cy.wait(200) | ||||
|  | ||||
|     // The viewport will be reset back to the default dimensions | ||||
|     // in between tests (the  default can be set in cypress.json) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										34
									
								
								cypress/examples/waiting.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,34 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Waiting', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/waiting') | ||||
|   }) | ||||
|   // BE CAREFUL of adding unnecessary wait times. | ||||
|   // https://on.cypress.io/best-practices#Unnecessary-Waiting | ||||
|  | ||||
|   // https://on.cypress.io/wait | ||||
|   it('cy.wait() - wait for a specific amount of time', () => { | ||||
|     cy.get('.wait-input1').type('Wait 1000ms after typing') | ||||
|     cy.wait(1000) | ||||
|     cy.get('.wait-input2').type('Wait 1000ms after typing') | ||||
|     cy.wait(1000) | ||||
|     cy.get('.wait-input3').type('Wait 1000ms after typing') | ||||
|     cy.wait(1000) | ||||
|   }) | ||||
|  | ||||
|   it('cy.wait() - wait for a specific route', () => { | ||||
|     cy.server() | ||||
|  | ||||
|     // Listen to GET to comments/1 | ||||
|     cy.route('GET', 'comments/*').as('getComment') | ||||
|  | ||||
|     // we have code that gets a comment when | ||||
|     // the button is clicked in scripts.js | ||||
|     cy.get('.network-btn').click() | ||||
|  | ||||
|     // wait for GET comments/1 | ||||
|     cy.wait('@getComment').its('status').should('eq', 200) | ||||
|   }) | ||||
|  | ||||
| }) | ||||
							
								
								
									
										22
									
								
								cypress/examples/window.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,22 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| context('Window', () => { | ||||
|   beforeEach(() => { | ||||
|     cy.visit('https://example.cypress.io/commands/window') | ||||
|   }) | ||||
|  | ||||
|   it('cy.window() - get the global window object', () => { | ||||
|     // https://on.cypress.io/window | ||||
|     cy.window().should('have.property', 'top') | ||||
|   }) | ||||
|  | ||||
|   it('cy.document() - get the document object', () => { | ||||
|     // https://on.cypress.io/document | ||||
|     cy.document().should('have.property', 'charset').and('eq', 'UTF-8') | ||||
|   }) | ||||
|  | ||||
|   it('cy.title() - get the title', () => { | ||||
|     // https://on.cypress.io/title | ||||
|     cy.title().should('include', 'Kitchen Sink') | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										5
									
								
								cypress/fixtures/example.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,5 @@ | ||||
| { | ||||
|   "name": "Using fixtures to represent data", | ||||
|   "email": "hello@cypress.io", | ||||
|   "body": "Fixtures are a great way to mock data for responses to routes" | ||||
| } | ||||
							
								
								
									
										28
									
								
								cypress/helpers/util.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,28 @@ | ||||
| /* eslint-env jest */ | ||||
| import { Base64 } from 'js-base64' | ||||
|  | ||||
| export const mermaidUrl = (graphStr, options, api) => { | ||||
|   const obj = { | ||||
|     code: graphStr, | ||||
|     mermaid: options | ||||
|   } | ||||
|   const objStr = JSON.stringify(obj) | ||||
|   let url = 'http://localhost:9000/e2e.html?graph=' + Base64.encodeURI(objStr) | ||||
|   if (api) { | ||||
|     url = 'http://localhost:9000/xss.html?graph=' + graphStr | ||||
|   } | ||||
|  | ||||
|   if (options.listUrl) { | ||||
|     cy.log(options.listId, ' ', url) | ||||
|   } | ||||
|  | ||||
|   return url | ||||
| } | ||||
|  | ||||
| export const imgSnapshotTest =  (graphStr, options, api) => { | ||||
|   const url = mermaidUrl(graphStr, options, api) | ||||
|  | ||||
|   cy.visit(url); | ||||
|   cy.get('svg') | ||||
|   cy.percySnapshot() | ||||
| } | ||||
							
								
								
									
										25
									
								
								cypress/integration/e2e/classDiagram.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest} from '../../helpers/util' | ||||
|  | ||||
| describe('Sequencediagram', () => { | ||||
|   it('should render a simple class diagrams', () => { | ||||
|     imgSnapshotTest(` | ||||
|     classDiagram | ||||
|       Class01 <|-- AveryLongClass : Cool | ||||
|       Class03 *-- Class04 | ||||
|       Class05 o-- Class06 | ||||
|       Class07 .. Class08 | ||||
|       Class09 --> C2 : Where am i? | ||||
|       Class09 --* C3 | ||||
|       Class09 --|> Class07 | ||||
|       Class07 : equals() | ||||
|       Class07 : Object[] elementData | ||||
|       Class01 : size() | ||||
|       Class01 : int chimp | ||||
|       Class01 : int gorilla | ||||
|       Class08 <--> C2: Cool label | ||||
|       `, | ||||
|     {}) | ||||
|     cy.get('svg') | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										324
									
								
								cypress/integration/e2e/flowchart.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,324 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest} from '../../helpers/util' | ||||
|  | ||||
|  | ||||
| describe('Flowcart', () => { | ||||
|   it('should render a simple flowchart', () => { | ||||
|     imgSnapshotTest(`graph TD | ||||
|       A[Christmas] -->|Get money| B(Go shopping) | ||||
|       B --> C{Let me think} | ||||
|       C -->|One| D[Laptop] | ||||
|       C -->|Two| E[iPhone] | ||||
|       C -->|Three| F[fa:fa-car Car] | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|   it('should render a simple flowchart with line breaks', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TD | ||||
|       A[Christmas] -->|Get money| B(Go shopping) | ||||
|       B --> C{Let me thinksssss<br/>ssssssssssssssssssssss<br/>sssssssssssssssssssssssssss} | ||||
|       C -->|One| D[Laptop] | ||||
|       C -->|Two| E[iPhone] | ||||
|       C -->|Three| F[Car] | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render a simple flowchart with trapezoid and inverse trapezoid vertex options.', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TD | ||||
|       A[/Christmas\\] | ||||
|       A -->|Get money| B[\\Go shopping/] | ||||
|       B --> C{Let me thinksssss<br/>ssssssssssssssssssssss<br/>sssssssssssssssssssssssssss} | ||||
|       C -->|One| D[/Laptop/] | ||||
|       C -->|Two| E[\\iPhone\\] | ||||
|       C -->|Three| F[Car] | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should style nodes via a class.', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TD | ||||
|       1A --> 1B | ||||
|       1B --> 1C | ||||
|       1C --> D | ||||
|       1C --> E | ||||
|  | ||||
|       classDef processHead fill:#888888,color:white,font-weight:bold,stroke-width:3px,stroke:#001f3f | ||||
|       class 1A,1B,D,E processHead | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render a flowchart full of circles', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph LR | ||||
|       47(SAM.CommonFA.FMESummary)-->48(SAM.CommonFA.CommonFAFinanceBudget) | ||||
|       37(SAM.CommonFA.BudgetSubserviceLineVolume)-->48(SAM.CommonFA.CommonFAFinanceBudget) | ||||
|       35(SAM.CommonFA.PopulationFME)-->47(SAM.CommonFA.FMESummary) | ||||
|       41(SAM.CommonFA.MetricCost)-->47(SAM.CommonFA.FMESummary) | ||||
|       44(SAM.CommonFA.MetricOutliers)-->47(SAM.CommonFA.FMESummary) | ||||
|       46(SAM.CommonFA.MetricOpportunity)-->47(SAM.CommonFA.FMESummary) | ||||
|       40(SAM.CommonFA.OPVisits)-->47(SAM.CommonFA.FMESummary) | ||||
|       38(SAM.CommonFA.CommonFAFinanceRefund)-->47(SAM.CommonFA.FMESummary) | ||||
|       43(SAM.CommonFA.CommonFAFinancePicuDays)-->47(SAM.CommonFA.FMESummary) | ||||
|       42(SAM.CommonFA.CommonFAFinanceNurseryDays)-->47(SAM.CommonFA.FMESummary) | ||||
|       45(SAM.CommonFA.MetricPreOpportunity)-->46(SAM.CommonFA.MetricOpportunity) | ||||
|       35(SAM.CommonFA.PopulationFME)-->45(SAM.CommonFA.MetricPreOpportunity) | ||||
|       41(SAM.CommonFA.MetricCost)-->45(SAM.CommonFA.MetricPreOpportunity) | ||||
|       41(SAM.CommonFA.MetricCost)-->44(SAM.CommonFA.MetricOutliers) | ||||
|       39(SAM.CommonFA.ChargeDetails)-->43(SAM.CommonFA.CommonFAFinancePicuDays) | ||||
|       39(SAM.CommonFA.ChargeDetails)-->42(SAM.CommonFA.CommonFAFinanceNurseryDays) | ||||
|       39(SAM.CommonFA.ChargeDetails)-->41(SAM.CommonFA.MetricCost) | ||||
|       39(SAM.CommonFA.ChargeDetails)-->40(SAM.CommonFA.OPVisits) | ||||
|       35(SAM.CommonFA.PopulationFME)-->39(SAM.CommonFA.ChargeDetails) | ||||
|       36(SAM.CommonFA.PremetricCost)-->39(SAM.CommonFA.ChargeDetails) | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|   it('should render a flowchart full of icons', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TD | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002("fa:fa-creative-commons My System | Test Environment") | ||||
|       82072290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Shared Business Logic Server:Service 1") | ||||
|       db052290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Shared Business Logic Server:Service 2") | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Shared Report Server:Service 1") | ||||
|       30122290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Shared Report Server:Service 2") | ||||
|       5e112290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Dedicated Test Business Logic Server:Service 1") | ||||
|       c1112290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Dedicated Test Business Logic Server:Service 2") | ||||
|       b7042290_1ec3_e711_8c5a_005056ad0002("fa:fa-circle [DBServer\\SharedDbInstance].[SupportDb]") | ||||
|       8f102290_1ec3_e711_8c5a_005056ad0002("fa:fa-circle [DBServer\\SharedDbInstance].[DevelopmentDb]") | ||||
|       0e102290_1ec3_e711_8c5a_005056ad0002("fa:fa-circle [DBServer\\SharedDbInstance].[TestDb]") | ||||
|       07132290_1ec3_e711_8c5a_005056ad0002("fa:fa-circle [DBServer\\SharedDbInstance].[SharedReportingDb]") | ||||
|       c7072290_1ec3_e711_8c5a_005056ad0002("fa:fa-server Shared Business Logic Server") | ||||
|       ca122290_1ec3_e711_8c5a_005056ad0002("fa:fa-server Shared Report Server") | ||||
|       68102290_1ec3_e711_8c5a_005056ad0002("fa:fa-server Dedicated Test Business Logic Server") | ||||
|       f4112290_1ec3_e711_8c5a_005056ad0002("fa:fa-database [DBServer\\SharedDbInstance]") | ||||
|       d6072290_1ec3_e711_8c5a_005056ad0002("fa:fa-server DBServer") | ||||
|       71082290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs DBServer\\:MSSQLSERVER") | ||||
|       c0102290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs DBServer\\:SQLAgent") | ||||
|       9a072290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs DBServer\\:SQLBrowser") | ||||
|       1d0a2290_1ec3_e711_8c5a_005056ad0002("fa:fa-server VmHost1") | ||||
|       200a2290_1ec3_e711_8c5a_005056ad0002("fa:fa-server VmHost2") | ||||
|       1c0a2290_1ec3_e711_8c5a_005056ad0002("fa:fa-server VmHost3") | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->82072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->db052290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->4e112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->30122290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->5e112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->c1112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       82072290_1ec3_e711_8c5a_005056ad0002-->b7042290_1ec3_e711_8c5a_005056ad0002 | ||||
|       82072290_1ec3_e711_8c5a_005056ad0002-->8f102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       82072290_1ec3_e711_8c5a_005056ad0002-->0e102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       82072290_1ec3_e711_8c5a_005056ad0002-->c7072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       db052290_1ec3_e711_8c5a_005056ad0002-->c7072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       db052290_1ec3_e711_8c5a_005056ad0002-->82072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002-->b7042290_1ec3_e711_8c5a_005056ad0002 | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002-->8f102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002-->0e102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002-->07132290_1ec3_e711_8c5a_005056ad0002 | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002-->ca122290_1ec3_e711_8c5a_005056ad0002 | ||||
|       30122290_1ec3_e711_8c5a_005056ad0002-->ca122290_1ec3_e711_8c5a_005056ad0002 | ||||
|       30122290_1ec3_e711_8c5a_005056ad0002-->4e112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       5e112290_1ec3_e711_8c5a_005056ad0002-->8f102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       5e112290_1ec3_e711_8c5a_005056ad0002-->68102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       c1112290_1ec3_e711_8c5a_005056ad0002-->68102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       c1112290_1ec3_e711_8c5a_005056ad0002-->5e112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       b7042290_1ec3_e711_8c5a_005056ad0002-->f4112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       8f102290_1ec3_e711_8c5a_005056ad0002-->f4112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       0e102290_1ec3_e711_8c5a_005056ad0002-->f4112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       07132290_1ec3_e711_8c5a_005056ad0002-->f4112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       c7072290_1ec3_e711_8c5a_005056ad0002-->1d0a2290_1ec3_e711_8c5a_005056ad0002 | ||||
|       ca122290_1ec3_e711_8c5a_005056ad0002-->200a2290_1ec3_e711_8c5a_005056ad0002 | ||||
|       68102290_1ec3_e711_8c5a_005056ad0002-->1c0a2290_1ec3_e711_8c5a_005056ad0002 | ||||
|       f4112290_1ec3_e711_8c5a_005056ad0002-->d6072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       f4112290_1ec3_e711_8c5a_005056ad0002-->71082290_1ec3_e711_8c5a_005056ad0002 | ||||
|       f4112290_1ec3_e711_8c5a_005056ad0002-->c0102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       f4112290_1ec3_e711_8c5a_005056ad0002-->9a072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       d6072290_1ec3_e711_8c5a_005056ad0002-->1c0a2290_1ec3_e711_8c5a_005056ad0002 | ||||
|       71082290_1ec3_e711_8c5a_005056ad0002-->d6072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       c0102290_1ec3_e711_8c5a_005056ad0002-->d6072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       c0102290_1ec3_e711_8c5a_005056ad0002-->71082290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9a072290_1ec3_e711_8c5a_005056ad0002-->d6072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9a072290_1ec3_e711_8c5a_005056ad0002-->71082290_1ec3_e711_8c5a_005056ad0002 | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render labels with numbers at the start', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TB;subgraph "number as labels";1;end; | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|   it('should render subgraphs', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TB | ||||
|       subgraph One | ||||
|         a1-->a2 | ||||
|       end | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render subgraphs with a title startign with a digit', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TB | ||||
|       subgraph 2Two | ||||
|         a1-->a2 | ||||
|       end | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render styled subgraphs', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TB | ||||
|       A | ||||
|       B | ||||
|       subgraph foo[Foo SubGraph] | ||||
|         C | ||||
|         D | ||||
|       end | ||||
|       subgraph bar[Bar SubGraph] | ||||
|         E | ||||
|         F | ||||
|       end | ||||
|       G | ||||
|  | ||||
|       A-->B | ||||
|       B-->C | ||||
|       C-->D | ||||
|       B-->D | ||||
|       D-->E | ||||
|       E-->A | ||||
|       E-->F | ||||
|       F-->D | ||||
|       F-->G | ||||
|       B-->G | ||||
|       G-->D | ||||
|  | ||||
|       style foo fill:#F99,stroke-width:2px,stroke:#F0F | ||||
|       style bar fill:#999,stroke-width:10px,stroke:#0F0 | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render a flowchart with ling sames and class definitoins', () => { | ||||
|     imgSnapshotTest(`graph LR | ||||
|       sid-B3655226-6C29-4D00-B685-3D5C734DC7E1[" | ||||
|  | ||||
|       提交申请 | ||||
|       熊大 | ||||
|       "]; | ||||
|       class sid-B3655226-6C29-4D00-B685-3D5C734DC7E1 node-executed; | ||||
|       sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A[" | ||||
|       负责人审批 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A node-executed; | ||||
|       sid-E27C0367-E6D6-497F-9736-3CDC21FDE221[" | ||||
|       DBA审批 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-E27C0367-E6D6-497F-9736-3CDC21FDE221 node-executed; | ||||
|       sid-BED98281-9585-4D1B-934E-BD1AC6AC0EFD[" | ||||
|       SA审批 | ||||
|       阿美 | ||||
|       "]; | ||||
|       class sid-BED98281-9585-4D1B-934E-BD1AC6AC0EFD node-executed; | ||||
|       sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7[" | ||||
|       主管审批 | ||||
|       光头强 | ||||
|       "]; | ||||
|       class sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7 node-executed; | ||||
|       sid-A1B3CD96-7697-4D7C-BEAA-73D187B1BE89[" | ||||
|       DBA确认 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-A1B3CD96-7697-4D7C-BEAA-73D187B1BE89 node-executed; | ||||
|       sid-3E35A7FF-A2F4-4E07-9247-DBF884C81937[" | ||||
|       SA确认 | ||||
|       阿美 | ||||
|       "]; | ||||
|       class sid-3E35A7FF-A2F4-4E07-9247-DBF884C81937 node-executed; | ||||
|       sid-4FC27B48-A6F9-460A-A675-021F5854FE22[" | ||||
|       结束 | ||||
|       "]; | ||||
|       class sid-4FC27B48-A6F9-460A-A675-021F5854FE22 node-executed; | ||||
|       sid-19DD9E9F-98C1-44EE-B604-842AFEE76F1E[" | ||||
|       SA执行1 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-19DD9E9F-98C1-44EE-B604-842AFEE76F1E node-executed; | ||||
|       sid-6C2120F3-D940-4958-A067-0903DCE879C4[" | ||||
|       SA执行2 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-6C2120F3-D940-4958-A067-0903DCE879C4 node-executed; | ||||
|       sid-9180E2A0-5C4B-435F-B42F-0D152470A338[" | ||||
|       DBA执行1 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-9180E2A0-5C4B-435F-B42F-0D152470A338 node-executed; | ||||
|       sid-03A2C3AC-5337-48A5-B154-BB3FD0EC8DAD[" | ||||
|       DBA执行3 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-03A2C3AC-5337-48A5-B154-BB3FD0EC8DAD node-executed; | ||||
|       sid-D5E1F2F4-306C-47A2-BF74-F66E3D769756[" | ||||
|       DBA执行2 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-D5E1F2F4-306C-47A2-BF74-F66E3D769756 node-executed; | ||||
|       sid-8C3F2F1D-F014-4F99-B966-095DC1A2BD93[" | ||||
|       DBA执行4 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-8C3F2F1D-F014-4F99-B966-095DC1A2BD93 node-executed; | ||||
|       sid-1897B30A-9C5C-4D5B-B80B-76A038785070[" | ||||
|       负责人确认 | ||||
|       梁静茹 | ||||
|       "]; | ||||
|       class sid-1897B30A-9C5C-4D5B-B80B-76A038785070 node-executed; | ||||
|       sid-B3655226-6C29-4D00-B685-3D5C734DC7E1-->sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7; | ||||
|       sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A-->sid-1897B30A-9C5C-4D5B-B80B-76A038785070; | ||||
|       sid-E27C0367-E6D6-497F-9736-3CDC21FDE221-->sid-A1B3CD96-7697-4D7C-BEAA-73D187B1BE89; | ||||
|       sid-BED98281-9585-4D1B-934E-BD1AC6AC0EFD-->sid-3E35A7FF-A2F4-4E07-9247-DBF884C81937; | ||||
|       sid-19DD9E9F-98C1-44EE-B604-842AFEE76F1E-->sid-6C2120F3-D940-4958-A067-0903DCE879C4; | ||||
|       sid-9180E2A0-5C4B-435F-B42F-0D152470A338-->sid-D5E1F2F4-306C-47A2-BF74-F66E3D769756; | ||||
|       sid-03A2C3AC-5337-48A5-B154-BB3FD0EC8DAD-->sid-8C3F2F1D-F014-4F99-B966-095DC1A2BD93; | ||||
|       sid-6C2120F3-D940-4958-A067-0903DCE879C4-->sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A; | ||||
|       sid-1897B30A-9C5C-4D5B-B80B-76A038785070-->sid-4FC27B48-A6F9-460A-A675-021F5854FE22; | ||||
|       sid-3E35A7FF-A2F4-4E07-9247-DBF884C81937-->sid-19DD9E9F-98C1-44EE-B604-842AFEE76F1E; | ||||
|       sid-A1B3CD96-7697-4D7C-BEAA-73D187B1BE89-->sid-9180E2A0-5C4B-435F-B42F-0D152470A338; | ||||
|       sid-A1B3CD96-7697-4D7C-BEAA-73D187B1BE89-->sid-03A2C3AC-5337-48A5-B154-BB3FD0EC8DAD; | ||||
|       sid-D5E1F2F4-306C-47A2-BF74-F66E3D769756-->sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A; | ||||
|       sid-8C3F2F1D-F014-4F99-B966-095DC1A2BD93-->sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A; | ||||
|       sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7-->sid-BED98281-9585-4D1B-934E-BD1AC6AC0EFD; | ||||
|       sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7-->sid-E27C0367-E6D6-497F-9736-3CDC21FDE221; | ||||
|       sid-3E35A7FF-A2F4-4E07-9247-DBF884C81937-->sid-6C2120F3-D940-4958-A067-0903DCE879C4; | ||||
|       sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7-->sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A; | ||||
|       sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7-->sid-4FC27B48-A6F9-460A-A675-021F5854FE22; | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render color of styled nodes', () => { | ||||
|     imgSnapshotTest(` | ||||
|       graph LR | ||||
|         foo-->bar | ||||
|  | ||||
|         classDef foo fill:lightblue,color:green,stroke:#FF9E2C,font-weight:bold | ||||
|         style foo fill:#F99,stroke-width:2px,stroke:#F0F | ||||
|         style bar fill:#999,color: #00ff00, stroke-width:10px,stroke:#0F0 | ||||
|       `, | ||||
|     { | ||||
|       listUrl: false, | ||||
|       listId: 'color styling', | ||||
|       logLevel: 0 | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										39
									
								
								cypress/integration/e2e/gantt.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,39 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest } from '../../helpers/util.js' | ||||
|  | ||||
| describe('Sequencediagram', () => { | ||||
|   it('should render a gantt chart', () => { | ||||
|     imgSnapshotTest(` | ||||
|     gantt | ||||
|       dateFormat  YYYY-MM-DD | ||||
|       axisFormat  %d/%m | ||||
|       title Adding GANTT diagram to mermaid | ||||
|       excludes weekdays 2014-01-10 | ||||
|  | ||||
|       section A section | ||||
|       Completed task            :done,    des1, 2014-01-06,2014-01-08 | ||||
|       Active task               :active,  des2, 2014-01-09, 3d | ||||
|       Future task               :         des3, after des2, 5d | ||||
|       Future task2               :         des4, after des3, 5d | ||||
|  | ||||
|       section Critical tasks | ||||
|       Completed task in the critical line :crit, done, 2014-01-06,24h | ||||
|       Implement parser and jison          :crit, done, after des1, 2d | ||||
|       Create tests for parser             :crit, active, 3d | ||||
|       Future task in critical line        :crit, 5d | ||||
|       Create tests for renderer           :2d | ||||
|       Add to mermaid                      :1d | ||||
|  | ||||
|       section Documentation | ||||
|       Describe gantt syntax               :active, a1, after des1, 3d | ||||
|       Add gantt diagram to demo page      :after a1  , 20h | ||||
|       Add another diagram to demo page    :doc1, after a1  , 48h | ||||
|  | ||||
|       section Last section | ||||
|       Describe gantt syntax               :after doc1, 3d | ||||
|       Add gantt diagram to demo page      : 20h | ||||
|       Add another diagram to demo page    : 48h | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										26
									
								
								cypress/integration/e2e/gitGraph.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest } from '../../helpers/util.js' | ||||
|  | ||||
| describe('Sequencediagram', () => { | ||||
|   it('should render a simple git graph', () => { | ||||
|     imgSnapshotTest(` | ||||
|     gitGraph: | ||||
|       options | ||||
|       { | ||||
|           "nodeSpacing": 150, | ||||
|           "nodeRadius": 10 | ||||
|       } | ||||
|       end | ||||
|       commit | ||||
|       branch newbranch | ||||
|       checkout newbranch | ||||
|       commit | ||||
|       commit | ||||
|       checkout master | ||||
|       commit | ||||
|       commit | ||||
|       merge newbranch | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										12
									
								
								cypress/integration/e2e/info.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest } from '../../helpers/util.js' | ||||
|  | ||||
| describe('Sequencediagram', () => { | ||||
|   it('should render a simple info diagrams', () => { | ||||
|     imgSnapshotTest(` | ||||
|     info | ||||
|        showInfo | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										118
									
								
								cypress/integration/e2e/sequencediagram.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,118 @@ | ||||
| /// <reference types="Cypress" /> | ||||
|  | ||||
| import { imgSnapshotTest} from '../../helpers/util'; | ||||
|  | ||||
| context('Aliasing', () => { | ||||
|   it('should render a simple sequence diagrams', () => { | ||||
|     imgSnapshotTest(` | ||||
|       sequenceDiagram | ||||
|         participant Alice | ||||
|         participant Bob | ||||
|         participant John as John<br/>Second Line | ||||
|         Alice ->> Bob: Hello Bob, how are you? | ||||
|         Bob-->>John: How about you John? | ||||
|         Bob--x Alice: I am good thanks! | ||||
|         Bob-x John: I am good thanks! | ||||
|         Note right of John: Bob thinks a long<br/>long time, so long<br/>that the text does<br/>not fit on a row. | ||||
|         Bob-->Alice: Checking with John... | ||||
|         alt either this | ||||
|           Alice->>John: Yes | ||||
|           else or this | ||||
|           Alice->>John: No | ||||
|           else or this will happen | ||||
|           Alice->John: Maybe | ||||
|         end | ||||
|         par this happens in parallel | ||||
|         Alice -->> Bob: Parallel message 1 | ||||
|         and | ||||
|         Alice -->> John: Parallel message 2 | ||||
|         end | ||||
|       `, {}) | ||||
|   }) | ||||
|   context('background rects', () => { | ||||
|     it('should render a single and nested rects', () => { | ||||
|       imgSnapshotTest(` | ||||
|         sequenceDiagram | ||||
|           participant A | ||||
|           participant B | ||||
|           participant C | ||||
|           participant D | ||||
|           participant E | ||||
|           participant G | ||||
|  | ||||
|           A ->>+ B: Task 1 | ||||
|           rect rgb(178, 102, 255) | ||||
|             B ->>+ C: Task 2 | ||||
|             C -->>- B: Return | ||||
|           end | ||||
|  | ||||
|           A ->> D: Task 3 | ||||
|           rect rgb(0, 128, 255) | ||||
|             D ->>+ E: Task 4 | ||||
|             rect rgb(0, 204, 0) | ||||
|             E ->>+ G: Task 5 | ||||
|             G -->>- E: Return | ||||
|             end | ||||
|             E ->> E: Task 6 | ||||
|           end | ||||
|           D -->> A: Complete | ||||
|       `, {}) | ||||
|     }) | ||||
|     it('should render rect around and inside loops', () => { | ||||
|       imgSnapshotTest(` | ||||
|         sequenceDiagram | ||||
|           A ->> B: 1 | ||||
|           rect rgb(204, 0, 102) | ||||
|             loop check C | ||||
|               C ->> C: Every 10 seconds | ||||
|             end | ||||
|           end | ||||
|           A ->> B: 2 | ||||
|           loop check D | ||||
|             C ->> D: 3 | ||||
|             rect rgb(153, 153, 255) | ||||
|             D -->> D: 5 | ||||
|             D --> C: 4 | ||||
|             end | ||||
|           end | ||||
|       `, {}) | ||||
|     }) | ||||
|     it('should render rect around and inside alts', () => { | ||||
|       imgSnapshotTest(` | ||||
|         sequenceDiagram | ||||
|           A ->> B: 1 | ||||
|           rect rgb(204, 0, 102) | ||||
|             alt yes | ||||
|               C ->> C: 1 | ||||
|             else no | ||||
|               rect rgb(0, 204, 204) | ||||
|                 C ->> C: 0 | ||||
|               end | ||||
|             end | ||||
|           end | ||||
|           B ->> A: Return | ||||
|       `, {}) | ||||
|     }) | ||||
|     it('should render rect around and inside opts', () => { | ||||
|       imgSnapshotTest(` | ||||
|         sequenceDiagram | ||||
|           A ->> B: 1 | ||||
|           rect rgb(204, 0, 102) | ||||
|             opt maybe | ||||
|               C -->> D: Do something | ||||
|               rect rgb(0, 204, 204) | ||||
|                 C ->> C: 0 | ||||
|               end | ||||
|             end | ||||
|           end | ||||
|  | ||||
|           opt possibly | ||||
|             rect rgb(0, 204, 204) | ||||
|               C ->> C: 0 | ||||
|             end | ||||
|           end | ||||
|           B ->> A: Return | ||||
|       `, {}) | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										10
									
								
								cypress/platform/bundle-test.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,10 @@ | ||||
| import mermaid from '../../dist/mermaid.core' | ||||
|  | ||||
| mermaid.initialize({ | ||||
|   theme: 'forest', | ||||
|   gantt: { axisFormatter: [ | ||||
|     ['%Y-%m-%d', (d) => { | ||||
|       return d.getDay() === 1 | ||||
|     }] | ||||
|   ] } | ||||
| }) | ||||
							
								
								
									
										83
									
								
								cypress/platform/click.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,83 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <head> | ||||
|   <meta charset="utf-8"> | ||||
|   <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
|   <title>Mermaid Quick Test Page</title> | ||||
|   <link rel="icon" type="image/png" href="data:image/png;base64,iVBORw0KGgo="> | ||||
| </head> | ||||
| <body> | ||||
|   <div id="FirstLine" class="mermaid"> | ||||
|     graph TB | ||||
|       Function-->URL | ||||
|       click Function clickByFlow "Add a div" | ||||
|       click URL "https://mermaidjs.github.io/" "Visit <strong>mermaid docs</strong>" | ||||
| </div> | ||||
| <div id="FirstLine" class="mermaid"> | ||||
|   graph TB | ||||
|     1Function-->2URL | ||||
|     click 1Function clickByFlow "Add a div" | ||||
|     click 2URL "https://mermaidjs.github.io/" "Visit <strong>mermaid docs</strong>" | ||||
| </div> | ||||
|  | ||||
| <div class="mermaid"> | ||||
|     gantt | ||||
|     dateFormat  YYYY-MM-DD | ||||
|     axisFormat  %d/%m | ||||
|     title Adding GANTT diagram to mermaid | ||||
|     excludes weekdays 2014-01-10 | ||||
|  | ||||
|     section A section | ||||
|     Completed task            :done,    des1, 2014-01-06,2014-01-08 | ||||
|     Active task               :active,  des2, 2014-01-09, 3d | ||||
|     Future task               :         des3, after des2, 5d | ||||
|     Future task2               :         des4, after des3, 5d | ||||
|  | ||||
|     section Critical tasks | ||||
|     Completed task in the critical line :crit, done, 2014-01-06,24h | ||||
|     Implement parser and jison          :crit, done, after des1, 2d | ||||
|     Create tests for parser             :crit, active, 3d | ||||
|     Future task in critical line        :crit, 5d | ||||
|     Create tests for renderer           :2d | ||||
|     Add to mermaid                      :1d | ||||
|  | ||||
|     section Documentation | ||||
|     Describe gantt syntax               :active, a1, after des1, 3d | ||||
|     Add gantt diagram to demo page      :after a1  , 20h | ||||
|     Add another diagram to demo page    :doc1, after a1  , 48h | ||||
|  | ||||
|     section Clickable | ||||
|     Visit mermaidjs               :active, cl1, 2014-01-07,2014-01-10 | ||||
|     Calling a Callback (look at the console log) :cl2, after cl1, 3d | ||||
|  | ||||
|     click cl1 href "https://mermaidjs.github.io/" | ||||
|     click cl2 call clickByGantt("test", test, test) | ||||
|  | ||||
|     section Last section | ||||
|     Describe gantt syntax               :after doc1, 3d | ||||
|     Add gantt diagram to demo page      : 20h | ||||
|     Add another diagram to demo page    : 48h | ||||
|       </div> | ||||
|  | ||||
|   <script src="./mermaid.js"></script> | ||||
|   <script> | ||||
|   function clickByFlow(elemName) { | ||||
|     const div = document.createElement('div') | ||||
|     div.className = 'created-by-click' | ||||
|     div.style = 'padding: 20px; background: green; color: white;' | ||||
|     div.innerText = 'Clicked By Flow' | ||||
|  | ||||
|     document.getElementsByTagName('body')[0].appendChild(div) | ||||
|   } | ||||
|   function clickByGantt(elemName) { | ||||
|     const div = document.createElement('div') | ||||
|     div.className = 'created-by-gant-click' | ||||
|     div.style = 'padding: 20px; background: green; color: white;' | ||||
|     div.innerText = 'Clicked By Gant' | ||||
|  | ||||
|     document.getElementsByTagName('body')[0].appendChild(div) | ||||
|   } | ||||
|   mermaid.initialize({ startOnLoad: true, securityLevel: 'loose', logLevel: 1 }); | ||||
|   </script> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										21
									
								
								cypress/platform/e2e.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,21 @@ | ||||
| <html> | ||||
|   <head> | ||||
|     <script src="/e2e.js"></script> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css?family=Montserrat&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <style></style> | ||||
|   </head> | ||||
|   <body> | ||||
|     <script src="./mermaid.js"></script> | ||||
|     <script> | ||||
|       // Notice startOnLoad=false | ||||
|       // This prevents default handling in mermaid from render before the e2e logic is applied | ||||
|       mermaid.initialize({ | ||||
|         startOnLoad: false, | ||||
|         useMaxWidth: true, | ||||
|       }); | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										46
									
								
								cypress/platform/flow.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,46 @@ | ||||
| <html> | ||||
|   <head> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css?family=Montserrat&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <style>body { | ||||
|       font-family: 'trebuchet ms', verdana, arial; | ||||
|     }</style> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div class="mermaid"> | ||||
|       graph TB | ||||
|       subgraph One | ||||
|         a1-->a2-->a3 | ||||
|       end | ||||
|     </div> | ||||
|     <div class="mermaid"> | ||||
|       graph TB | ||||
|         a_a --> b_b:::apa --> c_c:::apa | ||||
|         classDef apa fill:#f9f,stroke:#333,stroke-width:4px; | ||||
|         class a_a apa; | ||||
|     </div> | ||||
|     <div class="mermaid"> | ||||
|       graph TB | ||||
|         a_a(Aftonbladet) --> b_b[gorilla]:::apa --> c_c{chimp}:::apa -->a_a | ||||
|         a_a --> c --> d_d --> c_c | ||||
|         classDef apa fill:#f9f,stroke:#333,stroke-width:4px; | ||||
|         class a_a apa; | ||||
|         click a_a "http://www.aftonbladet.se" "apa" | ||||
|     </div> | ||||
|     <script src="./mermaid.js"></script> | ||||
|     <script> | ||||
|       mermaid.initialize({ | ||||
|         theme: 'forest', | ||||
|         // themeCSS: '.node rect { fill: red; }', | ||||
|         logLevel: 3, | ||||
|         flowchart: { curve: 'linear' }, | ||||
|         gantt: { axisFormat: '%m/%d/%Y' }, | ||||
|         sequence: { actorMargin: 50 }, | ||||
|         // sequenceDiagram: { actorMargin: 300 } // deprecated | ||||
|       }); | ||||
|     </script> | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										26
									
								
								cypress/platform/info.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,26 @@ | ||||
| <html> | ||||
|   <head> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css?family=Montserrat&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|  | ||||
|   </head> | ||||
|   <body> | ||||
|     <h1>info below</h1> | ||||
|     <div class="mermaid">info</div> | ||||
|     <script src="./mermaid.js"></script> | ||||
|     <script> | ||||
|       mermaid.initialize({ | ||||
|         theme: 'forest', | ||||
|         // themeCSS: '.node rect { fill: red; }', | ||||
|         logLevel: 3, | ||||
|         flowchart: { curve: 'linear' }, | ||||
|         gantt: { axisFormat: '%m/%d/%Y' }, | ||||
|         sequence: { actorMargin: 50 }, | ||||
|         // sequenceDiagram: { actorMargin: 300 } // deprecated | ||||
|       }); | ||||
|     </script> | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										45
									
								
								cypress/platform/subgraph.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,45 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <head> | ||||
|   <meta charset="utf-8"> | ||||
|   <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
|   <title>Mermaid Quick Test Page</title> | ||||
|   <link rel="icon" type="image/png" href="data:image/png;base64,iVBORw0KGgo="> | ||||
|   <style> | ||||
|     body { | ||||
|       font-family: 'trebuchet ms', verdana, arial; | ||||
|     } | ||||
|   </style> | ||||
| </head> | ||||
| <body> | ||||
|   <div class="mermaid"> | ||||
|     graph TD | ||||
|       A[Christmas] -->|Get money| B(Go shopping) | ||||
|       subgraph 1test["id starting with number"] | ||||
|       A | ||||
|       end | ||||
|       style 1test fill:#F99,stroke-width:2px,stroke:#F0F | ||||
|   </div> | ||||
|   <div class="mermaid"> | ||||
|     graph TD | ||||
|       A.a[Christmas]:::someclass -->|Get money| B(Go shopping):::someclass | ||||
|       subgraph test["id starting with number"] | ||||
|       A.a | ||||
|       end | ||||
|       style test fill:#F99,stroke-width:2px,stroke:#F0F | ||||
|       classDef someclass fill:#f96; | ||||
|   </div> | ||||
|   <div class="mermaid"> | ||||
|       graph TD | ||||
|       9e122290-->82072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       style 9e122290 fill:#F99,stroke-width:2px,stroke:#F0F | ||||
|   </div> | ||||
|   <script src="./mermaid.js"></script> | ||||
|   <script> | ||||
|      function showFullFirstSquad(elemName) { | ||||
|             console.log('show ' + elemName); | ||||
|         } | ||||
|         mermaid.initialize({ startOnLoad: true, securityLevel: 'loose', logLevel: 1 }); | ||||
|   </script> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										37
									
								
								cypress/platform/vertices.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,37 @@ | ||||
| <!DOCTYPE html> | ||||
| <html> | ||||
| <head> | ||||
|   <meta charset="utf-8"> | ||||
|   <meta http-equiv="X-UA-Compatible" content="IE=edge"> | ||||
|   <title>Mermaid Quick Test Page</title> | ||||
|   <link rel="icon" type="image/png" href="data:image/png;base64,iVBORw0KGgo="> | ||||
|   <style> | ||||
|     body { | ||||
|       font-family: 'trebuchet ms', verdana, arial; | ||||
|     } | ||||
|   </style> | ||||
| </head> | ||||
| <body> | ||||
|   <div class="mermaid"> | ||||
|     info | ||||
|   </div> | ||||
|   <div class="mermaid"> | ||||
|     graph TD | ||||
|       subgraph one | ||||
|         1 | ||||
|       end | ||||
|   </div> | ||||
|   <!-- <div class="mermaid"> | ||||
|     graph TD | ||||
|       A --> B --> C | ||||
|   </div> --> | ||||
|  | ||||
|   <script src="./mermaid.js"></script> | ||||
|   <script> | ||||
|      function showFullFirstSquad(elemName) { | ||||
|             console.log('show ' + elemName); | ||||
|         } | ||||
|         mermaid.initialize({ startOnLoad: true, securityLevel: 'loose', logLevel: 1 }); | ||||
|   </script> | ||||
| </body> | ||||
| </html> | ||||
							
								
								
									
										66
									
								
								cypress/platform/viewer.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,66 @@ | ||||
| import { Base64 } from 'js-base64' | ||||
| import mermaid2 from '../../src/mermaid' | ||||
|  | ||||
| /** | ||||
|  * ##contentLoaded | ||||
|  * Callback function that is called when page is loaded. This functions fetches configuration for mermaid rendering and | ||||
|  * calls init for rendering the mermaid diagrams on the page. | ||||
|  */ | ||||
| const contentLoaded = function () { | ||||
|   let pos = document.location.href.indexOf('?graph=') | ||||
|   if (pos > 0) { | ||||
|     pos = pos + 7 | ||||
|     const graphBase64 = document.location.href.substr(pos) | ||||
|     const graphObj = JSON.parse(Base64.decode(graphBase64)) | ||||
|     // const graph = 'hello' | ||||
|     console.log(graphObj) | ||||
|     const div = document.createElement('div') | ||||
|     div.id = 'block' | ||||
|     div.className = 'mermaid' | ||||
|     div.innerHTML = graphObj.code | ||||
|     document.getElementsByTagName('body')[0].appendChild(div) | ||||
|     global.mermaid.initialize(graphObj.mermaid) | ||||
|     // console.log('graphObj.mermaid', graphObj.mermaid) | ||||
|     global.mermaid.init() | ||||
|   } | ||||
| } | ||||
| const contentLoadedApi = function () { | ||||
|   let pos = document.location.href.indexOf('?graph=') | ||||
|   if (pos > 0) { | ||||
|     pos = pos + 7 | ||||
|     const graphBase64 = document.location.href.substr(pos) | ||||
|     const graphObj = JSON.parse(Base64.decode(graphBase64)) | ||||
|     // const graph = 'hello' | ||||
|     const div = document.createElement('div') | ||||
|     div.id = 'block' | ||||
|     div.className = 'mermaid' | ||||
|     // div.innerHTML = graphObj.code | ||||
|     document.getElementsByTagName('body')[0].appendChild(div) | ||||
|     global.mermaid.initialize(graphObj.mermaid) | ||||
|  | ||||
|     mermaid2.render('newid', graphObj.code, (svgCode, bindFunctions) => { | ||||
|       div.innerHTML = svgCode | ||||
|  | ||||
|       bindFunctions(div) | ||||
|     }, div) | ||||
|   } | ||||
| } | ||||
|  | ||||
| if (typeof document !== 'undefined') { | ||||
|   /*! | ||||
|    * Wait for document loaded before starting the execution | ||||
|    */ | ||||
|   window.addEventListener( | ||||
|     'load', | ||||
|     function () { | ||||
|       if (this.location.href.match('xss.html')) { | ||||
|         this.console.log('Using api') | ||||
|         contentLoadedApi() | ||||
|       } else { | ||||
|         this.console.log('Not using api') | ||||
|         contentLoaded() | ||||
|       } | ||||
|     }, | ||||
|     false | ||||
|   ) | ||||
| } | ||||
							
								
								
									
										23
									
								
								cypress/platform/webpackUsage.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,23 @@ | ||||
| <!doctype html> | ||||
| <html> | ||||
|  | ||||
| <body> | ||||
|   <div class="mermaid"> | ||||
| graph LR | ||||
| A-->B | ||||
|   </div> | ||||
|   <div class="mermaid"> | ||||
| gantt | ||||
| title A Gantt Diagram | ||||
| dateFormat  YYYY-MM-DD | ||||
| section Section | ||||
| A task           :a1, 2014-01-01, 30d | ||||
| Another task     :after a1  , 20d | ||||
| section Another | ||||
| Task in sec      :2014-01-12  , 12d | ||||
| another task      : 24d | ||||
|   </div> | ||||
|   <script src="./bundle-test.js" charset="utf-8"></script> | ||||
| </body> | ||||
|  | ||||
| </html> | ||||
							
								
								
									
										44
									
								
								cypress/platform/xss.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,44 @@ | ||||
| <html> | ||||
|   <head> | ||||
|     <script src="/e2e.js"></script> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css?family=Montserrat&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <style> | ||||
|       .malware { | ||||
|         position: fixed; | ||||
|         bottom:0; | ||||
|         left:0; | ||||
|         right:0; | ||||
|         height: 150px; | ||||
|         background: red; | ||||
|         color: black; | ||||
|         display: flex; | ||||
|         display: flex; | ||||
|         justify-content: center; | ||||
|         align-items: center; | ||||
|         font-family: monospace; | ||||
|         font-size: 72px; | ||||
|       } | ||||
|     </style> | ||||
|     <script> | ||||
|       function xssAttack(){ | ||||
|         const div = document.createElement('div') | ||||
|         div.id = 'the-malware' | ||||
|         div.className = 'malware' | ||||
|         div.innerHTML = 'XSS Succeeded' | ||||
|         document.getElementsByTagName('body')[0].appendChild(div) | ||||
|       } | ||||
|     </script> | ||||
|   </head> | ||||
|   <body> | ||||
|     <script src="./mermaid.js"></script> | ||||
|     <script> | ||||
|       mermaid.initialize({ | ||||
|         startOnLoad: false, | ||||
|         useMaxWidth: true, | ||||
|       }); | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
							
								
								
									
										
											BIN
										
									
								
								cypress/platform/xss.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						| After Width: | Height: | Size: 6.3 KiB | 
							
								
								
									
										25
									
								
								cypress/plugins/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,25 @@ | ||||
| // *********************************************************** | ||||
| // This example plugins/index.js can be used to load plugins | ||||
| // | ||||
| // You can change the location of this file or turn off loading | ||||
| // the plugins file with the 'pluginsFile' configuration option. | ||||
| // | ||||
| // You can read more here: | ||||
| // https://on.cypress.io/plugins-guide | ||||
| // *********************************************************** | ||||
|  | ||||
| // This function is called when a project is opened or re-opened (e.g. due to | ||||
| // the project's config changing) | ||||
|  | ||||
| // module.exports = (on, config) => { | ||||
| //   // `on` is used to hook into various events Cypress emits | ||||
| //   // `config` is the resolved Cypress config | ||||
| // } | ||||
|  | ||||
| let percyHealthCheck = require("@percy/cypress/task"); | ||||
|  | ||||
| module.exports = (on, config) => { | ||||
|   // `on` is used to hook into various events Cypress emits | ||||
|   // `config` is the resolved Cypress config | ||||
|   on("task", percyHealthCheck); | ||||
| }; | ||||
| After Width: | Height: | Size: 39 KiB | 
| After Width: | Height: | Size: 32 KiB | 
| After Width: | Height: | Size: 38 KiB | 
| After Width: | Height: | Size: 27 KiB | 
| After Width: | Height: | Size: 21 KiB | 
| After Width: | Height: | Size: 25 KiB | 
| After Width: | Height: | Size: 27 KiB | 
| After Width: | Height: | Size: 4.9 KiB | 
| After Width: | Height: | Size: 6.3 KiB | 
| After Width: | Height: | Size: 24 KiB | 
| After Width: | Height: | Size: 5.8 KiB | 
| After Width: | Height: | Size: 6.0 KiB | 
| After Width: | Height: | Size: 6.0 KiB | 
| After Width: | Height: | Size: 7.3 KiB | 
| After Width: | Height: | Size: 47 KiB | 
| After Width: | Height: | Size: 5.8 KiB | 
| After Width: | Height: | Size: 18 KiB | 
| After Width: | Height: | Size: 46 KiB | 
| After Width: | Height: | Size: 16 KiB | 
| After Width: | Height: | Size: 16 KiB | 
							
								
								
									
										27
									
								
								cypress/spec/classDiagram.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,27 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest } from '../helpers/util.js' | ||||
| const { toMatchImageSnapshot } = require('jest-image-snapshot') | ||||
|  | ||||
| expect.extend({ toMatchImageSnapshot }) | ||||
|  | ||||
| describe('Sequencediagram', () => { | ||||
|   it('should render a simple class diagrams', () => { | ||||
|     imgSnapshotTest(` | ||||
|     classDiagram | ||||
|       Class01 <|-- AveryLongClass : Cool | ||||
|       Class03 *-- Class04 | ||||
|       Class05 o-- Class06 | ||||
|       Class07 .. Class08 | ||||
|       Class09 --> C2 : Where am i? | ||||
|       Class09 --* C3 | ||||
|       Class09 --|> Class07 | ||||
|       Class07 : equals() | ||||
|       Class07 : Object[] elementData | ||||
|       Class01 : size() | ||||
|       Class01 : int chimp | ||||
|       Class01 : int gorilla | ||||
|       Class08 <--> C2: Cool label | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										326
									
								
								cypress/spec/flowchart.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,326 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest } from '../helpers/util.js' | ||||
| const { toMatchImageSnapshot } = require('jest-image-snapshot') | ||||
|  | ||||
| expect.extend({ toMatchImageSnapshot }) | ||||
|  | ||||
| describe('Flowcart', () => { | ||||
|   it('should render a simple flowchart', () => { | ||||
|     imgSnapshotTest(`graph TD | ||||
|       A[Christmas] -->|Get money| B(Go shopping) | ||||
|       B --> C{Let me think} | ||||
|       C -->|One| D[Laptop] | ||||
|       C -->|Two| E[iPhone] | ||||
|       C -->|Three| F[fa:fa-car Car] | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|   it('should render a simple flowchart with line breaks', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TD | ||||
|       A[Christmas] -->|Get money| B(Go shopping) | ||||
|       B --> C{Let me thinksssss<br/>ssssssssssssssssssssss<br/>sssssssssssssssssssssssssss} | ||||
|       C -->|One| D[Laptop] | ||||
|       C -->|Two| E[iPhone] | ||||
|       C -->|Three| F[Car] | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render a simple flowchart with trapezoid and inverse trapezoid vertex options.', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TD | ||||
|       A[/Christmas\\] | ||||
|       A -->|Get money| B[\\Go shopping/] | ||||
|       B --> C{Let me thinksssss<br/>ssssssssssssssssssssss<br/>sssssssssssssssssssssssssss} | ||||
|       C -->|One| D[/Laptop/] | ||||
|       C -->|Two| E[\\iPhone\\] | ||||
|       C -->|Three| F[Car] | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should style nodes via a class.', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TD | ||||
|       1A --> 1B | ||||
|       1B --> 1C | ||||
|       1C --> D | ||||
|       1C --> E | ||||
|  | ||||
|       classDef processHead fill:#888888,color:white,font-weight:bold,stroke-width:3px,stroke:#001f3f | ||||
|       class 1A,1B,D,E processHead | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render a flowchart full of circles', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph LR | ||||
|       47(SAM.CommonFA.FMESummary)-->48(SAM.CommonFA.CommonFAFinanceBudget) | ||||
|       37(SAM.CommonFA.BudgetSubserviceLineVolume)-->48(SAM.CommonFA.CommonFAFinanceBudget) | ||||
|       35(SAM.CommonFA.PopulationFME)-->47(SAM.CommonFA.FMESummary) | ||||
|       41(SAM.CommonFA.MetricCost)-->47(SAM.CommonFA.FMESummary) | ||||
|       44(SAM.CommonFA.MetricOutliers)-->47(SAM.CommonFA.FMESummary) | ||||
|       46(SAM.CommonFA.MetricOpportunity)-->47(SAM.CommonFA.FMESummary) | ||||
|       40(SAM.CommonFA.OPVisits)-->47(SAM.CommonFA.FMESummary) | ||||
|       38(SAM.CommonFA.CommonFAFinanceRefund)-->47(SAM.CommonFA.FMESummary) | ||||
|       43(SAM.CommonFA.CommonFAFinancePicuDays)-->47(SAM.CommonFA.FMESummary) | ||||
|       42(SAM.CommonFA.CommonFAFinanceNurseryDays)-->47(SAM.CommonFA.FMESummary) | ||||
|       45(SAM.CommonFA.MetricPreOpportunity)-->46(SAM.CommonFA.MetricOpportunity) | ||||
|       35(SAM.CommonFA.PopulationFME)-->45(SAM.CommonFA.MetricPreOpportunity) | ||||
|       41(SAM.CommonFA.MetricCost)-->45(SAM.CommonFA.MetricPreOpportunity) | ||||
|       41(SAM.CommonFA.MetricCost)-->44(SAM.CommonFA.MetricOutliers) | ||||
|       39(SAM.CommonFA.ChargeDetails)-->43(SAM.CommonFA.CommonFAFinancePicuDays) | ||||
|       39(SAM.CommonFA.ChargeDetails)-->42(SAM.CommonFA.CommonFAFinanceNurseryDays) | ||||
|       39(SAM.CommonFA.ChargeDetails)-->41(SAM.CommonFA.MetricCost) | ||||
|       39(SAM.CommonFA.ChargeDetails)-->40(SAM.CommonFA.OPVisits) | ||||
|       35(SAM.CommonFA.PopulationFME)-->39(SAM.CommonFA.ChargeDetails) | ||||
|       36(SAM.CommonFA.PremetricCost)-->39(SAM.CommonFA.ChargeDetails) | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|   it('should render a flowchart full of icons', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TD | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002("fa:fa-creative-commons My System | Test Environment") | ||||
|       82072290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Shared Business Logic Server:Service 1") | ||||
|       db052290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Shared Business Logic Server:Service 2") | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Shared Report Server:Service 1") | ||||
|       30122290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Shared Report Server:Service 2") | ||||
|       5e112290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Dedicated Test Business Logic Server:Service 1") | ||||
|       c1112290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs Dedicated Test Business Logic Server:Service 2") | ||||
|       b7042290_1ec3_e711_8c5a_005056ad0002("fa:fa-circle [DBServer\\SharedDbInstance].[SupportDb]") | ||||
|       8f102290_1ec3_e711_8c5a_005056ad0002("fa:fa-circle [DBServer\\SharedDbInstance].[DevelopmentDb]") | ||||
|       0e102290_1ec3_e711_8c5a_005056ad0002("fa:fa-circle [DBServer\\SharedDbInstance].[TestDb]") | ||||
|       07132290_1ec3_e711_8c5a_005056ad0002("fa:fa-circle [DBServer\\SharedDbInstance].[SharedReportingDb]") | ||||
|       c7072290_1ec3_e711_8c5a_005056ad0002("fa:fa-server Shared Business Logic Server") | ||||
|       ca122290_1ec3_e711_8c5a_005056ad0002("fa:fa-server Shared Report Server") | ||||
|       68102290_1ec3_e711_8c5a_005056ad0002("fa:fa-server Dedicated Test Business Logic Server") | ||||
|       f4112290_1ec3_e711_8c5a_005056ad0002("fa:fa-database [DBServer\\SharedDbInstance]") | ||||
|       d6072290_1ec3_e711_8c5a_005056ad0002("fa:fa-server DBServer") | ||||
|       71082290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs DBServer\\:MSSQLSERVER") | ||||
|       c0102290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs DBServer\\:SQLAgent") | ||||
|       9a072290_1ec3_e711_8c5a_005056ad0002("fa:fa-cogs DBServer\\:SQLBrowser") | ||||
|       1d0a2290_1ec3_e711_8c5a_005056ad0002("fa:fa-server VmHost1") | ||||
|       200a2290_1ec3_e711_8c5a_005056ad0002("fa:fa-server VmHost2") | ||||
|       1c0a2290_1ec3_e711_8c5a_005056ad0002("fa:fa-server VmHost3") | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->82072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->db052290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->4e112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->30122290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->5e112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9e122290_1ec3_e711_8c5a_005056ad0002-->c1112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       82072290_1ec3_e711_8c5a_005056ad0002-->b7042290_1ec3_e711_8c5a_005056ad0002 | ||||
|       82072290_1ec3_e711_8c5a_005056ad0002-->8f102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       82072290_1ec3_e711_8c5a_005056ad0002-->0e102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       82072290_1ec3_e711_8c5a_005056ad0002-->c7072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       db052290_1ec3_e711_8c5a_005056ad0002-->c7072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       db052290_1ec3_e711_8c5a_005056ad0002-->82072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002-->b7042290_1ec3_e711_8c5a_005056ad0002 | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002-->8f102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002-->0e102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002-->07132290_1ec3_e711_8c5a_005056ad0002 | ||||
|       4e112290_1ec3_e711_8c5a_005056ad0002-->ca122290_1ec3_e711_8c5a_005056ad0002 | ||||
|       30122290_1ec3_e711_8c5a_005056ad0002-->ca122290_1ec3_e711_8c5a_005056ad0002 | ||||
|       30122290_1ec3_e711_8c5a_005056ad0002-->4e112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       5e112290_1ec3_e711_8c5a_005056ad0002-->8f102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       5e112290_1ec3_e711_8c5a_005056ad0002-->68102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       c1112290_1ec3_e711_8c5a_005056ad0002-->68102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       c1112290_1ec3_e711_8c5a_005056ad0002-->5e112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       b7042290_1ec3_e711_8c5a_005056ad0002-->f4112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       8f102290_1ec3_e711_8c5a_005056ad0002-->f4112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       0e102290_1ec3_e711_8c5a_005056ad0002-->f4112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       07132290_1ec3_e711_8c5a_005056ad0002-->f4112290_1ec3_e711_8c5a_005056ad0002 | ||||
|       c7072290_1ec3_e711_8c5a_005056ad0002-->1d0a2290_1ec3_e711_8c5a_005056ad0002 | ||||
|       ca122290_1ec3_e711_8c5a_005056ad0002-->200a2290_1ec3_e711_8c5a_005056ad0002 | ||||
|       68102290_1ec3_e711_8c5a_005056ad0002-->1c0a2290_1ec3_e711_8c5a_005056ad0002 | ||||
|       f4112290_1ec3_e711_8c5a_005056ad0002-->d6072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       f4112290_1ec3_e711_8c5a_005056ad0002-->71082290_1ec3_e711_8c5a_005056ad0002 | ||||
|       f4112290_1ec3_e711_8c5a_005056ad0002-->c0102290_1ec3_e711_8c5a_005056ad0002 | ||||
|       f4112290_1ec3_e711_8c5a_005056ad0002-->9a072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       d6072290_1ec3_e711_8c5a_005056ad0002-->1c0a2290_1ec3_e711_8c5a_005056ad0002 | ||||
|       71082290_1ec3_e711_8c5a_005056ad0002-->d6072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       c0102290_1ec3_e711_8c5a_005056ad0002-->d6072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       c0102290_1ec3_e711_8c5a_005056ad0002-->71082290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9a072290_1ec3_e711_8c5a_005056ad0002-->d6072290_1ec3_e711_8c5a_005056ad0002 | ||||
|       9a072290_1ec3_e711_8c5a_005056ad0002-->71082290_1ec3_e711_8c5a_005056ad0002 | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render labels with numbers at the start', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TB;subgraph "number as labels";1;end; | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|   it('should render subgraphs', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TB | ||||
|       subgraph One | ||||
|         a1-->a2 | ||||
|       end | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render subgraphs with a title startign with a digit', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TB | ||||
|       subgraph 2Two | ||||
|         a1-->a2 | ||||
|       end | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render styled subgraphs', () => { | ||||
|     imgSnapshotTest(` | ||||
|     graph TB | ||||
|       A | ||||
|       B | ||||
|       subgraph foo[Foo SubGraph] | ||||
|         C | ||||
|         D | ||||
|       end | ||||
|       subgraph bar[Bar SubGraph] | ||||
|         E | ||||
|         F | ||||
|       end | ||||
|       G | ||||
|  | ||||
|       A-->B | ||||
|       B-->C | ||||
|       C-->D | ||||
|       B-->D | ||||
|       D-->E | ||||
|       E-->A | ||||
|       E-->F | ||||
|       F-->D | ||||
|       F-->G | ||||
|       B-->G | ||||
|       G-->D | ||||
|  | ||||
|       style foo fill:#F99,stroke-width:2px,stroke:#F0F | ||||
|       style bar fill:#999,stroke-width:10px,stroke:#0F0 | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render a flowchart with ling sames and class definitoins', () => { | ||||
|     imgSnapshotTest(`graph LR | ||||
|       sid-B3655226-6C29-4D00-B685-3D5C734DC7E1[" | ||||
|  | ||||
|       提交申请 | ||||
|       熊大 | ||||
|       "]; | ||||
|       class sid-B3655226-6C29-4D00-B685-3D5C734DC7E1 node-executed; | ||||
|       sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A[" | ||||
|       负责人审批 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A node-executed; | ||||
|       sid-E27C0367-E6D6-497F-9736-3CDC21FDE221[" | ||||
|       DBA审批 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-E27C0367-E6D6-497F-9736-3CDC21FDE221 node-executed; | ||||
|       sid-BED98281-9585-4D1B-934E-BD1AC6AC0EFD[" | ||||
|       SA审批 | ||||
|       阿美 | ||||
|       "]; | ||||
|       class sid-BED98281-9585-4D1B-934E-BD1AC6AC0EFD node-executed; | ||||
|       sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7[" | ||||
|       主管审批 | ||||
|       光头强 | ||||
|       "]; | ||||
|       class sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7 node-executed; | ||||
|       sid-A1B3CD96-7697-4D7C-BEAA-73D187B1BE89[" | ||||
|       DBA确认 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-A1B3CD96-7697-4D7C-BEAA-73D187B1BE89 node-executed; | ||||
|       sid-3E35A7FF-A2F4-4E07-9247-DBF884C81937[" | ||||
|       SA确认 | ||||
|       阿美 | ||||
|       "]; | ||||
|       class sid-3E35A7FF-A2F4-4E07-9247-DBF884C81937 node-executed; | ||||
|       sid-4FC27B48-A6F9-460A-A675-021F5854FE22[" | ||||
|       结束 | ||||
|       "]; | ||||
|       class sid-4FC27B48-A6F9-460A-A675-021F5854FE22 node-executed; | ||||
|       sid-19DD9E9F-98C1-44EE-B604-842AFEE76F1E[" | ||||
|       SA执行1 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-19DD9E9F-98C1-44EE-B604-842AFEE76F1E node-executed; | ||||
|       sid-6C2120F3-D940-4958-A067-0903DCE879C4[" | ||||
|       SA执行2 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-6C2120F3-D940-4958-A067-0903DCE879C4 node-executed; | ||||
|       sid-9180E2A0-5C4B-435F-B42F-0D152470A338[" | ||||
|       DBA执行1 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-9180E2A0-5C4B-435F-B42F-0D152470A338 node-executed; | ||||
|       sid-03A2C3AC-5337-48A5-B154-BB3FD0EC8DAD[" | ||||
|       DBA执行3 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-03A2C3AC-5337-48A5-B154-BB3FD0EC8DAD node-executed; | ||||
|       sid-D5E1F2F4-306C-47A2-BF74-F66E3D769756[" | ||||
|       DBA执行2 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-D5E1F2F4-306C-47A2-BF74-F66E3D769756 node-executed; | ||||
|       sid-8C3F2F1D-F014-4F99-B966-095DC1A2BD93[" | ||||
|       DBA执行4 | ||||
|       强子 | ||||
|       "]; | ||||
|       class sid-8C3F2F1D-F014-4F99-B966-095DC1A2BD93 node-executed; | ||||
|       sid-1897B30A-9C5C-4D5B-B80B-76A038785070[" | ||||
|       负责人确认 | ||||
|       梁静茹 | ||||
|       "]; | ||||
|       class sid-1897B30A-9C5C-4D5B-B80B-76A038785070 node-executed; | ||||
|       sid-B3655226-6C29-4D00-B685-3D5C734DC7E1-->sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7; | ||||
|       sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A-->sid-1897B30A-9C5C-4D5B-B80B-76A038785070; | ||||
|       sid-E27C0367-E6D6-497F-9736-3CDC21FDE221-->sid-A1B3CD96-7697-4D7C-BEAA-73D187B1BE89; | ||||
|       sid-BED98281-9585-4D1B-934E-BD1AC6AC0EFD-->sid-3E35A7FF-A2F4-4E07-9247-DBF884C81937; | ||||
|       sid-19DD9E9F-98C1-44EE-B604-842AFEE76F1E-->sid-6C2120F3-D940-4958-A067-0903DCE879C4; | ||||
|       sid-9180E2A0-5C4B-435F-B42F-0D152470A338-->sid-D5E1F2F4-306C-47A2-BF74-F66E3D769756; | ||||
|       sid-03A2C3AC-5337-48A5-B154-BB3FD0EC8DAD-->sid-8C3F2F1D-F014-4F99-B966-095DC1A2BD93; | ||||
|       sid-6C2120F3-D940-4958-A067-0903DCE879C4-->sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A; | ||||
|       sid-1897B30A-9C5C-4D5B-B80B-76A038785070-->sid-4FC27B48-A6F9-460A-A675-021F5854FE22; | ||||
|       sid-3E35A7FF-A2F4-4E07-9247-DBF884C81937-->sid-19DD9E9F-98C1-44EE-B604-842AFEE76F1E; | ||||
|       sid-A1B3CD96-7697-4D7C-BEAA-73D187B1BE89-->sid-9180E2A0-5C4B-435F-B42F-0D152470A338; | ||||
|       sid-A1B3CD96-7697-4D7C-BEAA-73D187B1BE89-->sid-03A2C3AC-5337-48A5-B154-BB3FD0EC8DAD; | ||||
|       sid-D5E1F2F4-306C-47A2-BF74-F66E3D769756-->sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A; | ||||
|       sid-8C3F2F1D-F014-4F99-B966-095DC1A2BD93-->sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A; | ||||
|       sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7-->sid-BED98281-9585-4D1B-934E-BD1AC6AC0EFD; | ||||
|       sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7-->sid-E27C0367-E6D6-497F-9736-3CDC21FDE221; | ||||
|       sid-3E35A7FF-A2F4-4E07-9247-DBF884C81937-->sid-6C2120F3-D940-4958-A067-0903DCE879C4; | ||||
|       sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7-->sid-4DA958A0-26D9-4D47-93A7-70F39FD7D51A; | ||||
|       sid-7CE72B24-E0C1-46D3-8132-8BA66BE05AA7-->sid-4FC27B48-A6F9-460A-A675-021F5854FE22; | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|  | ||||
|   it('should render color of styled nodes', () => { | ||||
|     imgSnapshotTest(` | ||||
|       graph LR | ||||
|         foo-->bar | ||||
|  | ||||
|         classDef foo fill:lightblue,color:green,stroke:#FF9E2C,font-weight:bold | ||||
|         style foo fill:#F99,stroke-width:2px,stroke:#F0F | ||||
|         style bar fill:#999,color: #00ff00, stroke-width:10px,stroke:#0F0 | ||||
|       `, | ||||
|     { | ||||
|       listUrl: false, | ||||
|       listId: 'color styling', | ||||
|       logLevel: 0 | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										42
									
								
								cypress/spec/gantt.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,42 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest } from '../helpers/util.js' | ||||
| const { toMatchImageSnapshot } = require('jest-image-snapshot') | ||||
|  | ||||
| expect.extend({ toMatchImageSnapshot }) | ||||
|  | ||||
| describe('Sequencediagram', () => { | ||||
|   it('should render a gantt chart', () => { | ||||
|     imgSnapshotTest(` | ||||
|     gantt | ||||
|       dateFormat  YYYY-MM-DD | ||||
|       axisFormat  %d/%m | ||||
|       title Adding GANTT diagram to mermaid | ||||
|       excludes weekdays 2014-01-10 | ||||
|  | ||||
|       section A section | ||||
|       Completed task            :done,    des1, 2014-01-06,2014-01-08 | ||||
|       Active task               :active,  des2, 2014-01-09, 3d | ||||
|       Future task               :         des3, after des2, 5d | ||||
|       Future task2               :         des4, after des3, 5d | ||||
|  | ||||
|       section Critical tasks | ||||
|       Completed task in the critical line :crit, done, 2014-01-06,24h | ||||
|       Implement parser and jison          :crit, done, after des1, 2d | ||||
|       Create tests for parser             :crit, active, 3d | ||||
|       Future task in critical line        :crit, 5d | ||||
|       Create tests for renderer           :2d | ||||
|       Add to mermaid                      :1d | ||||
|  | ||||
|       section Documentation | ||||
|       Describe gantt syntax               :active, a1, after des1, 3d | ||||
|       Add gantt diagram to demo page      :after a1  , 20h | ||||
|       Add another diagram to demo page    :doc1, after a1  , 48h | ||||
|  | ||||
|       section Last section | ||||
|       Describe gantt syntax               :after doc1, 3d | ||||
|       Add gantt diagram to demo page      : 20h | ||||
|       Add another diagram to demo page    : 48h | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										29
									
								
								cypress/spec/gitGraph.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,29 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest } from '../helpers/util.js' | ||||
| const { toMatchImageSnapshot } = require('jest-image-snapshot') | ||||
|  | ||||
| expect.extend({ toMatchImageSnapshot }) | ||||
|  | ||||
| describe('Sequencediagram', () => { | ||||
|   it('should render a simple git graph', () => { | ||||
|     imgSnapshotTest(` | ||||
|     gitGraph: | ||||
|       options | ||||
|       { | ||||
|           "nodeSpacing": 150, | ||||
|           "nodeRadius": 10 | ||||
|       } | ||||
|       end | ||||
|       commit | ||||
|       branch newbranch | ||||
|       checkout newbranch | ||||
|       commit | ||||
|       commit | ||||
|       checkout master | ||||
|       commit | ||||
|       commit | ||||
|       merge newbranch | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										15
									
								
								cypress/spec/info.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest } from '../helpers/util.js' | ||||
| const { toMatchImageSnapshot } = require('jest-image-snapshot') | ||||
|  | ||||
| expect.extend({ toMatchImageSnapshot }) | ||||
|  | ||||
| describe('Sequencediagram', () => { | ||||
|   it('should render a simple info diagrams', () => { | ||||
|     imgSnapshotTest(` | ||||
|     info | ||||
|        showInfo | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										121
									
								
								cypress/spec/sequencediagram.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,121 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest } from '../helpers/util.js' | ||||
| const { toMatchImageSnapshot } = require('jest-image-snapshot') | ||||
|  | ||||
| expect.extend({ toMatchImageSnapshot }) | ||||
|  | ||||
| describe('Sequencediagram', () => { | ||||
|   it('should render a simple sequence diagrams', () => { | ||||
|     imgSnapshotTest(` | ||||
|       sequenceDiagram | ||||
|         participant Alice | ||||
|         participant Bob | ||||
|         participant John as John<br/>Second Line | ||||
|         Alice ->> Bob: Hello Bob, how are you? | ||||
|         Bob-->>John: How about you John? | ||||
|         Bob--x Alice: I am good thanks! | ||||
|         Bob-x John: I am good thanks! | ||||
|         Note right of John: Bob thinks a long<br/>long time, so long<br/>that the text does<br/>not fit on a row. | ||||
|         Bob-->Alice: Checking with John... | ||||
|         alt either this | ||||
|           Alice->>John: Yes | ||||
|           else or this | ||||
|           Alice->>John: No | ||||
|           else or this will happen | ||||
|           Alice->John: Maybe | ||||
|         end | ||||
|         par this happens in parallel | ||||
|         Alice -->> Bob: Parallel message 1 | ||||
|         and | ||||
|         Alice -->> John: Parallel message 2 | ||||
|         end | ||||
|       `, | ||||
|     {}) | ||||
|   }) | ||||
|   describe('background rects', () => { | ||||
|     it('should render a single and nested rects', () => { | ||||
|       imgSnapshotTest(` | ||||
|         sequenceDiagram | ||||
|           participant A | ||||
|           participant B | ||||
|           participant C | ||||
|           participant D | ||||
|           participant E | ||||
|           participant G | ||||
|  | ||||
|           A ->>+ B: Task 1 | ||||
|           rect rgb(178, 102, 255) | ||||
|             B ->>+ C: Task 2 | ||||
|             C -->>- B: Return | ||||
|           end | ||||
|  | ||||
|           A ->> D: Task 3 | ||||
|           rect rgb(0, 128, 255) | ||||
|             D ->>+ E: Task 4 | ||||
|             rect rgb(0, 204, 0) | ||||
|             E ->>+ G: Task 5 | ||||
|             G -->>- E: Return | ||||
|             end | ||||
|             E ->> E: Task 6 | ||||
|           end | ||||
|           D -->> A: Complete | ||||
|       `, {}) | ||||
|     }) | ||||
|     it('should render rect around and inside loops', () => { | ||||
|       imgSnapshotTest(` | ||||
|         sequenceDiagram | ||||
|           A ->> B: 1 | ||||
|           rect rgb(204, 0, 102) | ||||
|             loop check C | ||||
|               C ->> C: Every 10 seconds | ||||
|             end | ||||
|           end | ||||
|           A ->> B: 2 | ||||
|           loop check D | ||||
|             C ->> D: 3 | ||||
|             rect rgb(153, 153, 255) | ||||
|             D -->> D: 5 | ||||
|             D --> C: 4 | ||||
|             end | ||||
|           end | ||||
|       `, {}) | ||||
|     }) | ||||
|     it('should render rect around and inside alts', () => { | ||||
|       imgSnapshotTest(` | ||||
|         sequenceDiagram | ||||
|           A ->> B: 1 | ||||
|           rect rgb(204, 0, 102) | ||||
|             alt yes | ||||
|               C ->> C: 1 | ||||
|             else no | ||||
|               rect rgb(0, 204, 204) | ||||
|                 C ->> C: 0 | ||||
|               end | ||||
|             end | ||||
|           end | ||||
|           B ->> A: Return | ||||
|       `, {}) | ||||
|     }) | ||||
|     it('should render rect around and inside opts', () => { | ||||
|       imgSnapshotTest(` | ||||
|         sequenceDiagram | ||||
|           A ->> B: 1 | ||||
|           rect rgb(204, 0, 102) | ||||
|             opt maybe | ||||
|               C -->> D: Do something | ||||
|               rect rgb(0, 204, 204) | ||||
|                 C ->> C: 0 | ||||
|               end | ||||
|             end | ||||
|           end | ||||
|  | ||||
|           opt possibly | ||||
|             rect rgb(0, 204, 204) | ||||
|               C ->> C: 0 | ||||
|             end | ||||
|           end | ||||
|           B ->> A: Return | ||||
|       `, {}) | ||||
|     }) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										16
									
								
								cypress/spec/webpackUsage.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,16 @@ | ||||
| /* eslint-env jest */ | ||||
| const { toMatchImageSnapshot } = require('jest-image-snapshot') | ||||
|  | ||||
| expect.extend({ toMatchImageSnapshot }) | ||||
|  | ||||
| describe('Sequencediagram', () => { | ||||
|   it('should render a simple sequence diagrams', () => { | ||||
|     const url = 'http://localhost:9000/webpackUsage.html' | ||||
|  | ||||
|     await page.goto(url) | ||||
|  | ||||
|     const image = await page.screenshot() | ||||
|  | ||||
|     expect(image).toMatchImageSnapshot() | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										15
									
								
								cypress/spec/xss.spec.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| /* eslint-env jest */ | ||||
| import { imgSnapshotTest } from '../helpers/util.js' | ||||
| const { toMatchImageSnapshot } = require('jest-image-snapshot') | ||||
|  | ||||
| expect.extend({ toMatchImageSnapshot }) | ||||
|  | ||||
| /* eslint-disable */ | ||||
| describe('XSS', () => { | ||||
|   it('should handle xss in tags', () => { | ||||
|     // const str = 'graph LR;\nB-->D(<img onerror=location=`javascript\u003aalert\u0028document.domain\u0029` src=x>);' | ||||
|     const str = 'eyJjb2RlIjoiXG5ncmFwaCBMUlxuICAgICAgQi0tPkQoPGltZyBvbmVycm9yPWxvY2F0aW9uPWBqYXZhc2NyaXB0XFx1MDAzYXhzc0F0dGFja1xcdTAwMjhkb2N1bWVudC5kb21haW5cXHUwMDI5YCBzcmM9eD4pOyIsIm1lcm1haWQiOnsidGhlbWUiOiJkZWZhdWx0In19'; | ||||
|     imgSnapshotTest(str, | ||||
|     {}, true) | ||||
|   }) | ||||
| }) | ||||
							
								
								
									
										27
									
								
								cypress/support/commands.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,27 @@ | ||||
| // *********************************************** | ||||
| // This example commands.js shows you how to | ||||
| // create various custom commands and overwrite | ||||
| // existing commands. | ||||
| // | ||||
| // For more comprehensive examples of custom | ||||
| // commands please read more here: | ||||
| // https://on.cypress.io/custom-commands | ||||
| // *********************************************** | ||||
| // | ||||
| // | ||||
| // -- This is a parent command -- | ||||
| // Cypress.Commands.add("login", (email, password) => { ... }) | ||||
| // | ||||
| // | ||||
| // -- This is a child command -- | ||||
| // Cypress.Commands.add("drag", { prevSubject: 'element'}, (subject, options) => { ... }) | ||||
| // | ||||
| // | ||||
| // -- This is a dual command -- | ||||
| // Cypress.Commands.add("dismiss", { prevSubject: 'optional'}, (subject, options) => { ... }) | ||||
| // | ||||
| // | ||||
| // -- This is will overwrite an existing command -- | ||||
| // Cypress.Commands.overwrite("visit", (originalFn, url, options) => { ... }) | ||||
|  | ||||
| import '@percy/cypress' | ||||
							
								
								
									
										20
									
								
								cypress/support/index.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,20 @@ | ||||
| // *********************************************************** | ||||
| // This example support/index.js is processed and | ||||
| // loaded automatically before your test files. | ||||
| // | ||||
| // This is a great place to put global configuration and | ||||
| // behavior that modifies Cypress. | ||||
| // | ||||
| // You can change the location of this file or turn off | ||||
| // automatically serving support files with the | ||||
| // 'supportFile' configuration option. | ||||
| // | ||||
| // You can read more here: | ||||
| // https://on.cypress.io/configuration | ||||
| // *********************************************************** | ||||
|  | ||||
| // Import commands.js using ES2015 syntax: | ||||
| import './commands' | ||||
|  | ||||
| // Alternatively you can use CommonJS syntax: | ||||
| // require('./commands') | ||||
							
								
								
									
										46
									
								
								e2e/platform/flow.html
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,46 @@ | ||||
| <html> | ||||
|   <head> | ||||
|     <link | ||||
|       href="https://fonts.googleapis.com/css?family=Montserrat&display=swap" | ||||
|       rel="stylesheet" | ||||
|     /> | ||||
|     <style>body { | ||||
|       font-family: 'trebuchet ms', verdana, arial; | ||||
|     }</style> | ||||
|   </head> | ||||
|   <body> | ||||
|     <div class="mermaid"> | ||||
|       graph TB | ||||
|       subgraph One | ||||
|         a1-->a2-->a3 | ||||
|       end | ||||
|     </div> | ||||
|     <div class="mermaid"> | ||||
|       graph TB | ||||
|         a_a --> b_b:::apa --> c_c:::apa | ||||
|         classDef apa fill:#f9f,stroke:#333,stroke-width:4px; | ||||
|         class a_a apa; | ||||
|     </div> | ||||
|     <div class="mermaid"> | ||||
|       graph TB | ||||
|         a_a(Aftonbladet) --> b_b[gorilla]:::apa --> c_c{chimp}:::apa -->a_a | ||||
|         a_a --> c --> d_d --> c_c | ||||
|         classDef apa fill:#f9f,stroke:#333,stroke-width:4px; | ||||
|         class a_a apa; | ||||
|         click a_a "http://www.aftonbladet.se" "apa" | ||||
|     </div> | ||||
|     <script src="./mermaid.js"></script> | ||||
|     <script> | ||||
|       mermaid.initialize({ | ||||
|         theme: 'forest', | ||||
|         // themeCSS: '.node rect { fill: red; }', | ||||
|         logLevel: 3, | ||||
|         flowchart: { curve: 'linear' }, | ||||
|         gantt: { axisFormat: '%m/%d/%Y' }, | ||||
|         sequence: { actorMargin: 50 }, | ||||
|         // sequenceDiagram: { actorMargin: 300 } // deprecated | ||||
|       }); | ||||
|     </script> | ||||
|     </script> | ||||
|   </body> | ||||
| </html> | ||||
| Before Width: | Height: | Size: 30 KiB | 
| Before Width: | Height: | Size: 21 KiB | 
| Before Width: | Height: | Size: 24 KiB | 
| Before Width: | Height: | Size: 21 KiB | 
| Before Width: | Height: | Size: 3.2 KiB | 
| Before Width: | Height: | Size: 18 KiB | 
| Before Width: | Height: | Size: 12 KiB | 
| Before Width: | Height: | Size: 18 KiB | 
| Before Width: | Height: | Size: 18 KiB | 
| Before Width: | Height: | Size: 31 KiB | 
							
								
								
									
										12
									
								
								package.json
									
									
									
									
									
								
							
							
						
						| @@ -19,9 +19,11 @@ | ||||
|     "minify": "minify ./dist/mermaid.js > ./dist/mermaid.min.js", | ||||
|     "release": "yarn build -p --config webpack.config.prod.babel.js", | ||||
|     "lint": "standard", | ||||
|     "e2e": "yarn lint && jest e2e --config e2e/jest.config.js", | ||||
|     "e2e:depr": "yarn lint && jest e2e --config e2e/jest.config.js", | ||||
|     "cypress": "percy exec -- cypress run", | ||||
|     "e2e": "start-server-and-test dev http://localhost:9000/ cypress", | ||||
|     "e2e-upd": "yarn lint && jest e2e -u --config e2e/jest.config.js", | ||||
|     "dev": "yarn lint && webpack-dev-server --config webpack.config.e2e.js", | ||||
|     "dev": "webpack-dev-server --config webpack.config.e2e.js", | ||||
|     "test": "yarn lint && jest src", | ||||
|     "test:watch": "jest --watch src", | ||||
|     "prepublishOnly": "yarn build && yarn release && yarn test", | ||||
| @@ -36,7 +38,8 @@ | ||||
|   "standard": { | ||||
|     "ignore": [ | ||||
|       "**/parser/*.js", | ||||
|       "dist/**/*.js" | ||||
|       "dist/**/*.js", | ||||
|       "cypress/**/*.js" | ||||
|     ], | ||||
|     "globals": [ | ||||
|       "page" | ||||
| @@ -59,12 +62,14 @@ | ||||
|     "@babel/core": "^7.2.2", | ||||
|     "@babel/preset-env": "^7.2.0", | ||||
|     "@babel/register": "^7.0.0", | ||||
|     "@percy/cypress": "^2.0.1", | ||||
|     "babel-core": "7.0.0-bridge.0", | ||||
|     "babel-jest": "^23.6.0", | ||||
|     "babel-loader": "^8.0.4", | ||||
|     "coveralls": "^3.0.2", | ||||
|     "css-loader": "^2.0.1", | ||||
|     "css-to-string-loader": "^0.1.3", | ||||
|     "cypress": "^3.4.1", | ||||
|     "husky": "^1.2.1", | ||||
|     "identity-obj-proxy": "^3.0.0", | ||||
|     "jest": "^23.6.0", | ||||
| @@ -77,6 +82,7 @@ | ||||
|     "puppeteer": "^1.17.0", | ||||
|     "sass-loader": "^7.1.0", | ||||
|     "standard": "^12.0.1", | ||||
|     "start-server-and-test": "^1.10.0", | ||||
|     "webpack": "^4.27.1", | ||||
|     "webpack-cli": "^3.1.2", | ||||
|     "webpack-dev-server": "^3.4.1", | ||||
|   | ||||
 Knut Sveidqvist
					Knut Sveidqvist