Skip to content

Commit 1e8e98c

Browse files
authored
test: Test with motion enabled (#1823)
1 parent 8b2d787 commit 1e8e98c

13 files changed

+143
-16
lines changed

.eslintrc.js

+3-2
Original file line numberDiff line numberDiff line change
@@ -117,7 +117,7 @@ module.exports = {
117117
},
118118
},
119119
{
120-
files: ['build-tools/integ/**', '**/__integ__/**', '**/__a11y__/**'],
120+
files: ['build-tools/integ/**', '**/__integ__/**', '**/__a11y__/**', '**/__motion__/**'],
121121
rules: {
122122
// useBrowser is not a React hook
123123
'react-hooks/rules-of-hooks': 'off',
@@ -157,7 +157,7 @@ module.exports = {
157157
},
158158
{
159159
files: ['src/**'],
160-
excludedFiles: ['src/**/__tests__/**', 'src/**/__integ__/**', 'src/**/__a11y__/**'],
160+
excludedFiles: ['src/**/__tests__/**', 'src/**/__integ__/**', 'src/**/__a11y__/**', 'src/**/__motion__/**'],
161161
rules: {
162162
'@cloudscape-design/no-implicit-svg-focusable': 'error',
163163
'@cloudscape-design/prefer-live-region': 'warn',
@@ -258,6 +258,7 @@ module.exports = {
258258
'src/**/__tests__/**',
259259
'src/**/__integ__/**',
260260
'src/**/__a11y__/**',
261+
'src/**/__motion__/**',
261262
'src/test-utils/**',
262263
'src/internal/vendor/**',
263264
],

CONTRIBUTING.md

+21-1
Original file line numberDiff line numberDiff line change
@@ -96,7 +96,8 @@ The package contains three types of tests:
9696

9797
- **Build-tool tests** test the build-tools code in a NodeJS context.
9898
- **Unit tests** emulate a browser environment using JSDOM.
99-
- **Integration tests** test against real browser behavior on Chrome.
99+
- **Integration tests** test against real browser behavior on Chrome, with motion disabled.
100+
- **Motion tests** run a specific set of tests on Chrome, with motion enabled.
100101

101102
#### Run all tests:
102103

@@ -128,6 +129,21 @@ npm start
128129
npm run test:integ
129130
```
130131

132+
#### Run motion tests:
133+
134+
As in integration tests, make sure you have ChromeDriver installed and start the dev server:
135+
136+
```
137+
npm i -g chromedriver
138+
npm start
139+
```
140+
141+
Then, run the motion tests in a separate terminal:
142+
143+
```
144+
npm test:motion
145+
```
146+
131147
#### Run a single test suite
132148

133149
To run a single test suite, you can call Jest directly using the appropriate config:
@@ -139,6 +155,9 @@ npx jest -c jest.unit.config.js src/button/__tests__/button.test.tsx
139155
# Run all input integration tests
140156
npx jest -c jest.integ.config.js src/input
141157
158+
# Run motion tests for the flashbar component
159+
npx jest -c jest.motion.config.js src/flashbar
160+
142161
# Test all stylelint rules
143162
npx jest -c jest.build-tools.config.js build-tools/stylelint
144163
```
@@ -202,6 +221,7 @@ After the test is done, you can find a report in `backstop/html_report/index.htm
202221
│ ├── <component-dir>
203222
│ │ ├── __tests__ - jest unit tests
204223
│ │ ├── __integ__ - jest integration tests
224+
│ │ ├── __motion__ - jest motion tests
205225
│ │ ├── index.tsx - main component file (imports other files and styles)
206226
│ │ └── styles.scss - main SCSS file
207227
| │

build-tools/integ/setup.motion.js

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
2+
// SPDX-License-Identifier: Apache-2.0
3+
const { configure } = require('@cloudscape-design/browser-test-tools/use-browser');
4+
5+
configure({
6+
browserName: 'ChromeHeadless',
7+
browserCreatorOptions: {
8+
seleniumUrl: 'http://localhost:9515',
9+
},
10+
webdriverOptions: {
11+
baseUrl: 'http://localhost:8080',
12+
},
13+
});
14+
15+
jest.retryTimes(3, { logErrorsBeforeRetry: true });

build-tools/tasks/index.js

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ module.exports = {
1616
a11y: require('./a11y'),
1717
generateI18nMessages: require('./generate-i18n-messages'),
1818
integ: require('./integ'),
19+
motion: require('./motion'),
1920
licenses: require('./licenses'),
2021
themeableSource: require('./themeable-source'),
2122
bundleVendorFiles: require('./bundle-vendor-files'),

build-tools/tasks/motion.js

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
2+
// SPDX-License-Identifier: Apache-2.0
3+
const execa = require('execa');
4+
const glob = require('glob');
5+
const waitOn = require('wait-on');
6+
const { task } = require('../utils/gulp-utils.js');
7+
8+
module.exports = task('test:motion', async () => {
9+
const devServer = execa('webpack', ['serve', '--config', 'pages/webpack.config.integ.js'], {
10+
env: {
11+
NODE_ENV: 'development',
12+
},
13+
});
14+
await waitOn({ resources: ['http://localhost:8080'] });
15+
16+
const files = glob.sync('src/**/__motion__/**/*.test.ts');
17+
await execa('jest', ['-c', 'jest.motion.config.js', ...files], { stdio: 'inherit' });
18+
19+
devServer.cancel();
20+
});

gulpfile.js

+3
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@ const {
1818
a11y,
1919
generateI18nMessages,
2020
integ,
21+
motion,
2122
licenses,
2223
themeableSource,
2324
bundleVendorFiles,
@@ -38,6 +39,7 @@ exports.test = series(unit, integ, a11y);
3839
exports['test:unit'] = unit;
3940
exports['test:integ'] = integ;
4041
exports['test:a11y'] = a11y;
42+
exports['test:motion'] = motion;
4143

4244
exports.watch = () => {
4345
watch(
@@ -47,6 +49,7 @@ exports.watch = () => {
4749
'!**/__tests__/**',
4850
'!**/__integ__/**',
4951
'!**/__a11y__/**',
52+
'!**/__motion__/**',
5053
'!src/internal/vendor/**/*.ts',
5154
],
5255
typescript

jest.motion.config.js

+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
2+
// SPDX-License-Identifier: Apache-2.0
3+
const path = require('path');
4+
const os = require('os');
5+
6+
module.exports = {
7+
verbose: true,
8+
testEnvironment: 'node',
9+
globals: {
10+
'ts-jest': {
11+
tsconfig: 'tsconfig.integ.json',
12+
},
13+
},
14+
reporters: ['default', 'github-actions'],
15+
testTimeout: 60_000, // 1min
16+
maxWorkers: os.cpus().length * (process.env.GITHUB_ACTION ? 3 : 1),
17+
globalSetup: '<rootDir>/build-tools/integ/global-setup.js',
18+
globalTeardown: '<rootDir>/build-tools/integ/global-teardown.js',
19+
setupFilesAfterEnv: [path.join(__dirname, 'build-tools', 'integ', 'setup.motion.js')],
20+
moduleFileExtensions: ['js', 'ts'],
21+
testRegex: '(/(__motion__)/.*(\\.|/)test)\\.[jt]sx?$',
22+
preset: 'ts-jest',
23+
};

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@
1414
"test:unit": "gulp test:unit",
1515
"test:a11y": "gulp test:a11y",
1616
"test:integ": "gulp test:integ",
17+
"test:motion": "gulp test:motion",
1718
"test:visual": "./backstop/node_modules/.bin/backstop --config backstop.config.js",
1819
"lint": "eslint --ignore-path .gitignore --ext ts,tsx,js . && stylelint --ignore-path .gitignore '{src,pages}/**/*.{css,scss}'",
1920
"start": "gulp watch & NODE_ENV=development webpack serve --config pages/webpack.config.js",

src/flashbar/__integ__/pages/base.ts

-11
Original file line numberDiff line numberDiff line change
@@ -3,8 +3,6 @@
33
import { BasePageObject } from '@cloudscape-design/browser-test-tools/page-objects';
44
import selectors from '../../../../lib/components/flashbar/styles.selectors.js';
55
import createWrapper from '../../../../lib/components/test-utils/selectors';
6-
import useBrowser from '@cloudscape-design/browser-test-tools/use-browser';
7-
import { FlashbarInteractivePage } from './interactive-page';
86

97
export const flashbar = createWrapper().findFlashbar();
108

@@ -28,12 +26,3 @@ export class FlashbarBasePage extends BasePageObject {
2826
);
2927
}
3028
}
31-
32-
export const setupTest = (path: string, testFn: (page: FlashbarInteractivePage) => Promise<void>) => {
33-
return useBrowser(async browser => {
34-
const page = new FlashbarInteractivePage(browser);
35-
await browser.url(`#/light/flashbar/${path}`);
36-
await page.waitForVisible(flashbar.toSelector());
37-
await testFn(page);
38-
});
39-
};

src/flashbar/__integ__/pages/interactive-page.ts

+9
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
// SPDX-License-Identifier: Apache-2.0
33
import { FlashbarBasePage, flashbar } from './base';
44
import useBrowser from '@cloudscape-design/browser-test-tools/use-browser';
5+
import createWrapper from '../../../../lib/components/test-utils/selectors';
56

67
export class FlashbarInteractivePage extends FlashbarBasePage {
78
async addInfoFlash() {
@@ -24,6 +25,14 @@ export class FlashbarInteractivePage extends FlashbarBasePage {
2425
await this.click('[data-id="stack-items"]');
2526
}
2627

28+
async dismissFirstItem() {
29+
await this.click(createWrapper().findFlashbar().findItems().get(1).findDismissButton().toSelector());
30+
}
31+
32+
getItem(index: number) {
33+
return createWrapper().findFlashbar().findItems().get(index);
34+
}
35+
2736
async removeAll() {
2837
await this.click('[data-id="remove-all"]');
2938
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
// Copyright Amazon.com, Inc. or its affiliates. All Rights Reserved.
2+
// SPDX-License-Identifier: Apache-2.0
3+
import useBrowser from '@cloudscape-design/browser-test-tools/use-browser';
4+
import { flashbar } from '../__integ__/pages/base';
5+
import { FlashbarInteractivePage } from '../__integ__/pages/interactive-page';
6+
7+
const setupTest = (options: { visualRefresh?: boolean }, testFn: (page: FlashbarInteractivePage) => Promise<void>) => {
8+
return useBrowser(async browser => {
9+
const page = new FlashbarInteractivePage(browser);
10+
const baseUrl = '#/light/flashbar/interactive';
11+
const url =
12+
options.visualRefresh === undefined
13+
? baseUrl
14+
: `${baseUrl}?${new URLSearchParams({
15+
visualRefresh: `${options.visualRefresh}`,
16+
}).toString()}`;
17+
await browser.url(url);
18+
await page.waitForVisible(flashbar.toSelector());
19+
await testFn(page);
20+
});
21+
};
22+
23+
describe('Collapsible flashbar', () => {
24+
describe('can dismiss all items', () => {
25+
it.each([true, false])('visualRefresh=%s', visualRefresh =>
26+
setupTest({ visualRefresh }, async page => {
27+
await page.removeAll();
28+
await page.toggleStackingFeature();
29+
await expect(page.countFlashes()).resolves.toBe(0);
30+
await page.addInfoFlash();
31+
await expect(page.countFlashes()).resolves.toBe(1);
32+
await page.waitForVisible(page.getItem(1).findDismissButton().toSelector());
33+
await page.dismissFirstItem();
34+
await page.waitForAssertion(() => expect(page.countFlashes()).resolves.toBe(0));
35+
})()
36+
);
37+
});
38+
});

tsconfig.integ.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,5 +11,5 @@
1111
"resolveJsonModule": true,
1212
"moduleResolution": "node"
1313
},
14-
"include": ["**/__integ__/**/*.ts", "**/__a11y__/**/*.ts", "types"]
14+
"include": ["**/__integ__/**/*.ts", "**/__a11y__/**/*.ts", "**/__motion__/**/*.ts", "types"]
1515
}

tsconfig.json

+8-1
Original file line numberDiff line numberDiff line change
@@ -15,5 +15,12 @@
1515
"incremental": true
1616
},
1717
"include": ["src", "types"],
18-
"exclude": ["**/__tests__/**", "src/test-utils/**", "**/__integ__/**", "**/__a11y__/**", "src/internal/vendor/**"]
18+
"exclude": [
19+
"**/__tests__/**",
20+
"src/test-utils/**",
21+
"**/__integ__/**",
22+
"**/__a11y__/**",
23+
"**/__motion__/**",
24+
"src/internal/vendor/**"
25+
]
1926
}

0 commit comments

Comments
 (0)