-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathREADME.md
192 lines (147 loc) · 6.49 KB
/
README.md
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
<img src="./docs/playwright-logo.svg" height="75px" align="right">
# ⚛️ react-declarative-e2e
> Playwright end-to-end testbed for [react-declarative](https://github.com/react-declarative/react-declarative)
![Build Status Passing](https://raw.githubusercontent.com/dwyl/repo-badges/main/svg/build-passing.svg)
![Code Coverage 100%](https://raw.githubusercontent.com/dwyl/repo-badges/main/svg/coverage-100.svg)
![screenshot](./docs/screenshot.png)
## Purpose
This will help you to tweak `react-declarative` with your own UI Kit. Check this [article for guide](https://github.com/react-declarative/react-declarative/blob/master/docs/other/how-to-implement-uikit.md)
## Contribute
> [!IMPORTANT]
> Made especially for newbies as an advanced documentation for `react-declarative` to solve their problems. **⭐Star** and **💻Fork** It [on github](https://github.com/react-declarative/react-declarative) will be appreciated
## Usage
1. Build the project
```bash
npm run build
```
2. Run tests
```bash
npm run test:ui
```
or
```bash
npm run test
```
## Test Cases
**Fields:**
1. [TypographyField](./e2e/spec/Fields/Typography.spec.ts)
2. [TextField](./e2e/spec/Fields/TextField.spec.ts)
3. [RadioField](./e2e/spec/Fields/Radio.spec.ts)
4. [CheckboxField](./e2e/spec/Fields/Checkbox.spec.ts)
5. [ComboField](./e2e/spec/Fields/Combo.spec.ts)
6. [ItemsField](./e2e/spec/Fields/Items.spec.ts)
7. [TreeField](./e2e/spec/Fields/Tree.spec.ts)
8. [YesNoField](./e2e/spec/Fields/YesNo.spec.ts)
9. [SwitchField](./e2e/spec/Fields/Switch.spec.ts)
10. [ProgressField](./e2e/spec/Fields/Progress.spec.ts)
11. [SliderField](./e2e/spec/Fields/Slider.spec.ts)
12. [ChooseField](./e2e/spec/Fields/Choose.spec.ts)
13. [CompleteField](./e2e/spec/Fields/Complete.spec.ts)
14. [DateField](./e2e/spec/Fields/Date.spec.ts)
15. [TimeField](./e2e/spec/Fields/Time.spec.ts)
**Layouts:**
1. [FragmentLayout](./e2e/spec/Fields/Fragment.spec.ts)
2. [GroupLayout](./e2e/spec/Fields/Group.spec.ts)
3. [PaperLayout](./e2e/spec/Fields/Paper.spec.ts)
4. [OutlineLayout](./e2e/spec/Fields/Outline.spec.ts)
4. [ExpansionLayout](./e2e/spec/Fields/Expansion.spec.ts)
**Actions:**
1. [ButtonField](./e2e/spec/Fields/Button.spec.ts)
2. [IconField](./e2e/spec/Fields/Icon.spec.ts)
**Forms**
1. [Form1](./e2e/spec/Forms/Form1.spec.ts)
2. [Form2](./e2e/spec/Forms/Form2.spec.ts)
## Test coverage for business logic
When you using classic jest + enzyme testbed for React SPA you are writing unit tests only. It is possible to mock the API but [If you do it you will not coverage the API itself](https://github.com/react-declarative/react-declarative/blob/master/docs/other/code-sideeffect.md). It's Ok if you want to loot your time in Jira but pointless for busines cause the business earns on inobvious connections in data flow.
[If this is a button then It is clickable](https://en.wikipedia.org/wiki/Duck_test). Any UI Kit is the sealed technology, you don't need to research the GUI like it was in the good old times of [seventees Xerox](https://en.wikipedia.org/wiki/Xerox_Alto)
The important part is the reaction of app after data mutation when button just clicked. Business need to document their requirements in the code, so the only way to solve that problem is to keep the field validation and button visibility rules in the UI schema.
Check [the Form1 example](./e2e/spec/Forms/Form1.spec.ts), this is a real integrational test, not the mocked one which a-priory coverage the problem partially. **P.S.** The [Form2 example](./e2e/spec/Forms/Form2.spec.ts) coverage the mobile-first approach too
## Code Sample
```tsx
import { expect, test } from "@playwright/test";
import { waitForReady } from "../helpers/wait-for-ready";
import { renderFields } from "../helpers/render-fields";
import TypedField from "../model/TypedField";
import FieldType from "../model/FieldType";
test.beforeEach(async ({ page }) => {
await waitForReady(page);
});
const fields: TypedField[] = [
{
type: FieldType.YesNo,
name: 'yesno',
testId: 'yesno-field',
dirty: true,
isReadonly: ({ readonly }) => readonly,
isDisabled: ({ disabled }) => disabled,
isInvalid: ({ invalid }) => invalid || null,
},
];
test("Will show intermediate state", async ({ page }) => {
const textField = await renderFields(page, fields, {
data: {
yesno: null,
},
});
const inputValue = await textField.getByRole('combobox').inputValue();
await expect(inputValue).toEqual("");
});
test("Will show true state", async ({ page }) => {
const textField = await renderFields(page, fields, {
data: {
yesno: true,
},
});
const inputValue = await textField.getByRole('combobox').inputValue();
await expect(inputValue).toEqual("Yes");
});
test("Will show false state", async ({ page }) => {
const textField = await renderFields(page, fields, {
data: {
yesno: false,
},
});
const inputValue = await textField.getByRole('combobox').inputValue();
await expect(inputValue).toEqual("No");
});
test("Will set true state", async ({ page }) => {
const textField = await renderFields(page, fields);
await textField.click();
await page.getByText("Yes", { exact: true }).first().click();
const inputValue = await textField.getByRole('combobox').inputValue();
await expect(inputValue).toEqual("Yes");
});
test("Will set false state", async ({ page }) => {
const textField = await renderFields(page, fields);
await textField.click();
await page.getByText("No", { exact: true }).first().click();
const inputValue = await textField.getByRole('combobox').inputValue();
await expect(inputValue).toEqual("No");
});
test("Will show invalid message", async ({ page }) => {
const componentGroup = await renderFields(page, fields, {
data: {
invalid: "Invalid",
}
});
await expect(componentGroup).toContainText('Invalid');
});
test("Will show disabled state", async ({ page }) => {
const componentGroup = await renderFields(page, fields, {
data: {
disabled: true,
}
});
const isDisabled = await componentGroup.getByLabel('Yesno').isDisabled();
await expect(isDisabled).toBeTruthy();
});
test("Will show readonly state", async ({ page }) => {
const componentGroup = await renderFields(page, fields, {
data: {
readonly: true,
},
});
const isEditable = await componentGroup.getByLabel('Yesno').isEditable();
await expect(isEditable).toBeFalsy();
});
```