@pautena/react-design-system 0.1.2 → 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (191) hide show
  1. package/README.md +4 -0
  2. package/dist/cjs/index.js +4 -259
  3. package/dist/cjs/index.js.map +1 -1
  4. package/dist/cjs/types/generators/model-router/screens/add-screen.d.ts +1 -1
  5. package/dist/cjs/types/generators/model-router/screens/list-screen.d.ts +1 -1
  6. package/dist/cjs/types/generators/model-router/screens/screens.types.d.ts +20 -0
  7. package/dist/cjs/types/generators/model-router/screens/update-screen.d.ts +1 -1
  8. package/dist/cjs/types/index.d.ts +1 -0
  9. package/dist/esm/index.js +4 -259
  10. package/dist/esm/index.js.map +1 -1
  11. package/dist/esm/types/generators/model-router/screens/add-screen.d.ts +1 -1
  12. package/dist/esm/types/generators/model-router/screens/list-screen.d.ts +1 -1
  13. package/dist/esm/types/generators/model-router/screens/screens.types.d.ts +20 -0
  14. package/dist/esm/types/generators/model-router/screens/update-screen.d.ts +1 -1
  15. package/dist/esm/types/index.d.ts +1 -0
  16. package/dist/index.d.ts +52 -3
  17. package/package.json +13 -2
  18. package/src/components/app-bar/app-bar.stories.tsx +54 -0
  19. package/src/components/app-bar/app-bar.test.tsx +142 -0
  20. package/src/components/app-bar/app-bar.tsx +150 -0
  21. package/src/components/app-bar/app-bar.types.ts +17 -0
  22. package/src/components/app-bar/index.ts +3 -0
  23. package/src/components/app-bar/mini-app-bar/index.ts +1 -0
  24. package/src/components/app-bar/mini-app-bar/mini-app-bar.tsx +31 -0
  25. package/src/components/bullet/bullet.stories.tsx +43 -0
  26. package/src/components/bullet/bullet.test.tsx +24 -0
  27. package/src/components/bullet/bullet.tsx +30 -0
  28. package/src/components/bullet/index.ts +1 -0
  29. package/src/components/center-container/center-container.stories.tsx +50 -0
  30. package/src/components/center-container/center-container.test.tsx +16 -0
  31. package/src/components/center-container/center-container.tsx +32 -0
  32. package/src/components/center-container/index.ts +1 -0
  33. package/src/components/content/content.stories.tsx +23 -0
  34. package/src/components/content/content.test.tsx +26 -0
  35. package/src/components/content/content.tsx +11 -0
  36. package/src/components/content/content.types.ts +5 -0
  37. package/src/components/content/index.ts +2 -0
  38. package/src/components/drawer/__snapshots__/drawer.test.tsx.snap +20 -0
  39. package/src/components/drawer/drawer.context.ts +20 -0
  40. package/src/components/drawer/drawer.mixins.ts +24 -0
  41. package/src/components/drawer/drawer.mock.tsx +100 -0
  42. package/src/components/drawer/drawer.provider.tsx +23 -0
  43. package/src/components/drawer/drawer.test.tsx +97 -0
  44. package/src/components/drawer/drawer.tsx +30 -0
  45. package/src/components/drawer/drawer.types.ts +53 -0
  46. package/src/components/drawer/index.ts +5 -0
  47. package/src/components/drawer/mini-drawer/index.ts +1 -0
  48. package/src/components/drawer/mini-drawer/mini-drawer.stories.tsx +34 -0
  49. package/src/components/drawer/mini-drawer/mini-drawer.tsx +67 -0
  50. package/src/components/drawer-content/drawer-content.stories.tsx +29 -0
  51. package/src/components/drawer-content/drawer-content.test.tsx +34 -0
  52. package/src/components/drawer-content/drawer-content.tsx +18 -0
  53. package/src/components/drawer-content/index.ts +1 -0
  54. package/src/components/drawer-item/drawer-item.stories.tsx +62 -0
  55. package/src/components/drawer-item/drawer-item.test.tsx +119 -0
  56. package/src/components/drawer-item/drawer-item.tsx +71 -0
  57. package/src/components/drawer-item/index.ts +1 -0
  58. package/src/components/drawer-section/drawer-section.mock.tsx +39 -0
  59. package/src/components/drawer-section/drawer-section.stories.tsx +28 -0
  60. package/src/components/drawer-section/drawer-section.test.tsx +44 -0
  61. package/src/components/drawer-section/drawer-section.tsx +40 -0
  62. package/src/components/drawer-section/index.ts +1 -0
  63. package/src/components/header/header.dummy.ts +55 -0
  64. package/src/components/header/header.stories.tsx +116 -0
  65. package/src/components/header/header.test.tsx +159 -0
  66. package/src/components/header/header.tsx +121 -0
  67. package/src/components/header/header.types.ts +61 -0
  68. package/src/components/header/index.ts +2 -0
  69. package/src/components/index.ts +18 -0
  70. package/src/components/label/index.ts +1 -0
  71. package/src/components/label/label.stories.tsx +49 -0
  72. package/src/components/label/label.test.tsx +30 -0
  73. package/src/components/label/label.tsx +60 -0
  74. package/src/components/link/index.ts +1 -0
  75. package/src/components/link/link.tsx +17 -0
  76. package/src/components/loading-area/index.ts +1 -0
  77. package/src/components/loading-area/loading-area.stories.tsx +17 -0
  78. package/src/components/loading-area/loading-area.test.tsx +11 -0
  79. package/src/components/loading-area/loading-area.tsx +13 -0
  80. package/src/components/placeholder/index.ts +1 -0
  81. package/src/components/placeholder/placeholder.mock.ts +15 -0
  82. package/src/components/placeholder/placeholder.stories.tsx +44 -0
  83. package/src/components/placeholder/placeholder.test.tsx +76 -0
  84. package/src/components/placeholder/placeholder.tsx +75 -0
  85. package/src/components/query-container/index.ts +1 -0
  86. package/src/components/query-container/query-container.stories.tsx +68 -0
  87. package/src/components/query-container/query-container.test.tsx +95 -0
  88. package/src/components/query-container/query-container.tsx +71 -0
  89. package/src/components/sign-in/index.ts +1 -0
  90. package/src/components/sign-in/sign-in.stories.tsx +36 -0
  91. package/src/components/sign-in/sign-in.test.tsx +95 -0
  92. package/src/components/sign-in/sign-in.tsx +97 -0
  93. package/src/components/tab/index.ts +2 -0
  94. package/src/components/tab/tab-card/index.ts +1 -0
  95. package/src/components/tab/tab-card/tab-card.dummy.tsx +30 -0
  96. package/src/components/tab/tab-card/tab-card.stories.tsx +22 -0
  97. package/src/components/tab/tab-card/tab-card.test.tsx +53 -0
  98. package/src/components/tab/tab-card/tab-card.tsx +27 -0
  99. package/src/components/tab/tab-panel/index.ts +1 -0
  100. package/src/components/tab/tab-panel/tab-panel.test.tsx +26 -0
  101. package/src/components/tab/tab-panel/tab-panel.tsx +27 -0
  102. package/src/components/table/enhanced-remote-table/enhanced-remote-table.mock.tsx +27 -0
  103. package/src/components/table/enhanced-remote-table/enhanced-remote-table.stories.tsx +24 -0
  104. package/src/components/table/enhanced-remote-table/enhanced-remote-table.test.tsx +77 -0
  105. package/src/components/table/enhanced-remote-table/enhanced-remote-table.tsx +74 -0
  106. package/src/components/table/enhanced-remote-table/index.ts +1 -0
  107. package/src/components/table/enhanced-table/enhanced-table-head.tsx +58 -0
  108. package/src/components/table/enhanced-table/enhanced-table.mock.tsx +93 -0
  109. package/src/components/table/enhanced-table/enhanced-table.stories.tsx +21 -0
  110. package/src/components/table/enhanced-table/enhanced-table.test.tsx +107 -0
  111. package/src/components/table/enhanced-table/enhanced-table.tsx +136 -0
  112. package/src/components/table/enhanced-table/index.ts +2 -0
  113. package/src/components/table/index.ts +2 -0
  114. package/src/components/table-list/index.ts +1 -0
  115. package/src/components/table-list/table-list.stories.tsx +75 -0
  116. package/src/components/table-list/table-list.test.tsx +291 -0
  117. package/src/components/table-list/table-list.tsx +127 -0
  118. package/src/components/value-displays/group-value-card/group-value-card.mock.tsx +35 -0
  119. package/src/components/value-displays/group-value-card/group-value-card.stories.tsx +26 -0
  120. package/src/components/value-displays/group-value-card/group-value-card.test.tsx +58 -0
  121. package/src/components/value-displays/group-value-card/group-value-card.tsx +63 -0
  122. package/src/components/value-displays/group-value-card/index.ts +1 -0
  123. package/src/components/value-displays/index.ts +4 -0
  124. package/src/components/value-displays/value-boolean/index.ts +1 -0
  125. package/src/components/value-displays/value-boolean/value-boolean.stories.tsx +25 -0
  126. package/src/components/value-displays/value-boolean/value-boolean.test.tsx +27 -0
  127. package/src/components/value-displays/value-boolean/value-boolean.tsx +33 -0
  128. package/src/components/value-displays/value-card/index.ts +1 -0
  129. package/src/components/value-displays/value-card/value-card.stories.tsx +22 -0
  130. package/src/components/value-displays/value-card/value-card.test.tsx +18 -0
  131. package/src/components/value-displays/value-card/value-card.tsx +12 -0
  132. package/src/components/value-displays/value-text/index.ts +1 -0
  133. package/src/components/value-displays/value-text/value-test.test.tsx +21 -0
  134. package/src/components/value-displays/value-text/value-text.stories.tsx +26 -0
  135. package/src/components/value-displays/value-text/value-text.tsx +32 -0
  136. package/src/generators/generators.mock.ts +238 -0
  137. package/src/generators/generators.model.ts +46 -0
  138. package/src/generators/index.ts +4 -0
  139. package/src/generators/model-form/index.ts +1 -0
  140. package/src/generators/model-form/model-form.stories.tsx +30 -0
  141. package/src/generators/model-form/model-form.test.tsx +100 -0
  142. package/src/generators/model-form/model-form.tsx +97 -0
  143. package/src/generators/model-router/index.ts +1 -0
  144. package/src/generators/model-router/model-router.test.tsx +831 -0
  145. package/src/generators/model-router/model-router.tsx +30 -0
  146. package/src/generators/model-router/model-router.types.ts +14 -0
  147. package/src/generators/model-router/screens/add-screen.tsx +70 -0
  148. package/src/generators/model-router/screens/details-screen.tsx +62 -0
  149. package/src/generators/model-router/screens/index.ts +4 -0
  150. package/src/generators/model-router/screens/list-screen.tsx +125 -0
  151. package/src/generators/model-router/screens/screens.types.ts +38 -0
  152. package/src/generators/model-router/screens/update-screen.tsx +97 -0
  153. package/src/generators/model-router/stories/details-screen.stories.tsx +38 -0
  154. package/src/generators/model-router/stories/list-screen.stories.tsx +96 -0
  155. package/src/generators/model-router/stories/model-router.stories.tsx +176 -0
  156. package/src/generators/model-router/stories/templates.tsx +39 -0
  157. package/src/generators/object-details/index.ts +1 -0
  158. package/src/generators/object-details/object-details.stories.tsx +20 -0
  159. package/src/generators/object-details/object-details.test.tsx +21 -0
  160. package/src/generators/object-details/object-details.tsx +76 -0
  161. package/src/index.ts +5 -0
  162. package/src/layouts/app-bar-with-drawer-layout/app-bar-with-drawer-layout.stories.tsx +28 -0
  163. package/src/layouts/app-bar-with-drawer-layout/app-bar-with-drawer-layout.test.tsx +30 -0
  164. package/src/layouts/app-bar-with-drawer-layout/app-bar-with-drawer-layout.tsx +37 -0
  165. package/src/layouts/app-bar-with-drawer-layout/index.ts +1 -0
  166. package/src/layouts/header-layout/header-layout.stories.tsx +204 -0
  167. package/src/layouts/header-layout/header-layout.test.tsx +37 -0
  168. package/src/layouts/header-layout/header-layout.tsx +23 -0
  169. package/src/layouts/header-layout/index.ts +1 -0
  170. package/src/layouts/index.ts +2 -0
  171. package/src/providers/index.ts +2 -0
  172. package/src/providers/notification-center/index.ts +2 -0
  173. package/src/providers/notification-center/notification-center.context.ts +37 -0
  174. package/src/providers/notification-center/notification-center.provider.tsx +51 -0
  175. package/src/providers/notification-center/notification-center.stories.tsx +52 -0
  176. package/src/providers/notification-center/notification-center.test.tsx +112 -0
  177. package/src/providers/tab-provider/index.ts +2 -0
  178. package/src/providers/tab-provider/tab-provider.context.ts +8 -0
  179. package/src/providers/tab-provider/tab-provider.provider.tsx +13 -0
  180. package/src/storybook.tsx +90 -0
  181. package/src/tests/assertions.ts +76 -0
  182. package/src/tests/components.tsx +60 -0
  183. package/src/tests/content-placeholder.stories.tsx +16 -0
  184. package/src/tests/index.ts +3 -0
  185. package/src/tests/skeleton-card.stories.tsx +18 -0
  186. package/src/tests/testing-library.tsx +65 -0
  187. package/src/utils/arrays.test.ts +9 -0
  188. package/src/utils/arrays.ts +7 -0
  189. package/src/utils/index.ts +2 -0
  190. package/src/utils/theme.ts +11 -0
  191. package/.prettierrc.js +0 -5
@@ -0,0 +1,136 @@
1
+ /* eslint-disable @typescript-eslint/no-explicit-any */
2
+ import React, { ChangeEvent } from "react";
3
+ import { ReactNode, useState } from "react";
4
+ import {
5
+ Box,
6
+ TableContainer,
7
+ TextField,
8
+ TableBody,
9
+ InputAdornment,
10
+ Table,
11
+ CircularProgress,
12
+ TableCell,
13
+ TableRow,
14
+ } from "@mui/material";
15
+ import Search from "@mui/icons-material/Search";
16
+ import { EnhancedTableHead, HeadCell, Order } from "./enhanced-table-head";
17
+
18
+ function getFilter<T>(columns: HeadCell[], search: string) {
19
+ return (d: T) => {
20
+ return (
21
+ !search ||
22
+ columns.some((col) => {
23
+ let value = (d as any)[col.id];
24
+ if (value?.toLowerCase) {
25
+ value = value.toLowerCase();
26
+ }
27
+ return value?.toString().includes(search.toLowerCase());
28
+ })
29
+ );
30
+ };
31
+ }
32
+
33
+ function getComparator(order: Order, orderBy: any): (a: any, b: any) => number {
34
+ return order === "desc"
35
+ ? (a, b) => descendingComparator(a, b, orderBy)
36
+ : (a, b) => -descendingComparator(a, b, orderBy);
37
+ }
38
+ function descendingComparator<T>(a: T, b: T, orderBy: keyof T) {
39
+ if (b[orderBy] < a[orderBy]) {
40
+ return -1;
41
+ }
42
+ if (b[orderBy] > a[orderBy]) {
43
+ return 1;
44
+ }
45
+ return 0;
46
+ }
47
+
48
+ interface Props<T> {
49
+ readonly data: T[];
50
+ search?: boolean;
51
+ defaultSort: string;
52
+ defaultOrder?: Order;
53
+ loading?: boolean;
54
+ columns: HeadCell[];
55
+ children: (data: T[]) => ReactNode;
56
+ }
57
+
58
+ export const EnhancedTable = <T,>({
59
+ children,
60
+ data,
61
+ search,
62
+ columns,
63
+ defaultSort,
64
+ defaultOrder = "asc",
65
+ loading = false,
66
+ }: Props<T>) => {
67
+ const [searchFilter, setSearchFilter] = useState<string>("");
68
+ const [order, setOrder] = useState<Order>(defaultOrder);
69
+ const [orderBy, setOrderBy] = useState(defaultSort);
70
+
71
+ const handleRequestSort = (property: string) => {
72
+ const isAsc = orderBy === property && order === "asc";
73
+ setOrder(isAsc ? "desc" : "asc");
74
+ setOrderBy(property);
75
+ };
76
+
77
+ const filteredData = data
78
+ .slice()
79
+ .filter(getFilter<T>(columns, searchFilter))
80
+ .sort(getComparator(order, orderBy));
81
+
82
+ return (
83
+ <>
84
+ <Box sx={{ paddingX: 1, paddingBottom: 2 }}>
85
+ {search && (
86
+ <Box paddingY={2}>
87
+ <TextField
88
+ fullWidth
89
+ placeholder="Search"
90
+ InputProps={{
91
+ role: "search",
92
+ startAdornment: (
93
+ <InputAdornment position="start">
94
+ <Search />
95
+ </InputAdornment>
96
+ ),
97
+ }}
98
+ onChange={(e: ChangeEvent<any>) => setSearchFilter(e.target.value)}
99
+ />
100
+ </Box>
101
+ )}
102
+ <TableContainer>
103
+ <Table>
104
+ <EnhancedTableHead
105
+ order={order}
106
+ orderBy={orderBy}
107
+ headCells={columns}
108
+ onRequestSort={handleRequestSort}
109
+ />
110
+ <TableBody>
111
+ {loading ? (
112
+ <TableRow>
113
+ <TableCell colSpan={columns.length} sx={{ textAlign: "center" }}>
114
+ <CircularProgress />
115
+ </TableCell>
116
+ </TableRow>
117
+ ) : filteredData.length === 0 ? (
118
+ <TableRow>
119
+ <TableCell colSpan={columns.length} sx={{ textAlign: "center" }}>
120
+ No data
121
+ </TableCell>
122
+ </TableRow>
123
+ ) : (
124
+ children(filteredData)
125
+ )}
126
+ </TableBody>
127
+ </Table>
128
+ </TableContainer>
129
+ </Box>
130
+ </>
131
+ );
132
+ };
133
+
134
+ EnhancedTable.defaultProps = {
135
+ defaultOrder: "asc",
136
+ };
@@ -0,0 +1,2 @@
1
+ export * from "./enhanced-table";
2
+ export * from "./enhanced-table-head";
@@ -0,0 +1,2 @@
1
+ export * from "./enhanced-table";
2
+ export * from "./enhanced-remote-table";
@@ -0,0 +1 @@
1
+ export * from "./table-list";
@@ -0,0 +1,75 @@
1
+ import React from "react";
2
+ import { useDemoData } from "@mui/x-data-grid-generator";
3
+ import { ComponentMeta } from "@storybook/react";
4
+ import { createTemplate } from "../../storybook";
5
+ import { TableList } from "./table-list";
6
+ import { action } from "@storybook/addon-actions";
7
+
8
+ const maxColumns = 3;
9
+
10
+ export default {
11
+ title: "Tables/TableList",
12
+ component: TableList,
13
+ parameters: {
14
+ layout: "fullscreen",
15
+ },
16
+ } as ComponentMeta<typeof TableList>;
17
+
18
+ const Template = createTemplate(({ dataSetType, size, ...rest }) => {
19
+ const { data } = useDemoData({
20
+ dataSet: dataSetType,
21
+ rowLength: size,
22
+ maxColumns,
23
+ });
24
+ const { rows } = data;
25
+ const columns = data.columns.map(({ field, headerName }) => ({
26
+ id: field,
27
+ label: headerName,
28
+ sort: true,
29
+ disablePadding: false,
30
+ numeric: false,
31
+ }));
32
+
33
+ return <TableList {...rest} columns={columns} data={rows} onClick={action("onClick row")} />;
34
+ });
35
+
36
+ export const Default = Template.bind({});
37
+ Default.args = {
38
+ size: 40,
39
+ dataSetType: "Commodity",
40
+ search: true,
41
+ };
42
+
43
+ export const Loading = Template.bind({});
44
+ Loading.args = {
45
+ size: 40,
46
+ dataSetType: "Commodity",
47
+ search: true,
48
+ loading: true,
49
+ };
50
+
51
+ export const WihtoutSearch = Template.bind({});
52
+ WihtoutSearch.args = {
53
+ size: 40,
54
+ dataSetType: "Commodity",
55
+ search: false,
56
+ };
57
+
58
+ export const WithOptions = Template.bind({});
59
+ WithOptions.args = {
60
+ size: 40,
61
+ dataSetType: "Commodity",
62
+ search: false,
63
+ options: [
64
+ {
65
+ id: "edit",
66
+ label: "Edit",
67
+ onClick: action("Option edit"),
68
+ },
69
+ {
70
+ id: "remove",
71
+ label: "Remove",
72
+ onClick: action("Option remove"),
73
+ },
74
+ ],
75
+ };
@@ -0,0 +1,291 @@
1
+ import React from "react";
2
+ import { expectProgressIndicator, render } from "../../tests";
3
+ import { screen } from "@testing-library/react";
4
+ import userEvents from "@testing-library/user-event";
5
+ import { TableRowOption, TableList } from "./table-list";
6
+ import { HeadCell } from "../table/enhanced-table";
7
+ import { BasicModelInstance } from "../../generators";
8
+
9
+ const columns: HeadCell[] = [
10
+ {
11
+ id: "id",
12
+ numeric: true,
13
+ disablePadding: false,
14
+ label: "Item ID",
15
+ sort: false,
16
+ },
17
+ {
18
+ id: "name",
19
+ numeric: false,
20
+ disablePadding: false,
21
+ label: "Name",
22
+ sort: true,
23
+ },
24
+ {
25
+ id: "value",
26
+ numeric: true,
27
+ disablePadding: false,
28
+ label: "Value",
29
+ sort: true,
30
+ },
31
+ ];
32
+ const data: BasicModelInstance[] = [
33
+ {
34
+ id: "item-c",
35
+ name: "fg Item 1",
36
+ value: 1,
37
+ },
38
+ {
39
+ id: "item-a",
40
+ name: "aa Item 2",
41
+ value: 2,
42
+ },
43
+ {
44
+ id: "item-l",
45
+ name: "ba Item 3",
46
+ value: 3,
47
+ },
48
+ {
49
+ id: "item-j",
50
+ name: "aa Item 4",
51
+ value: 4,
52
+ },
53
+ {
54
+ id: "item-w",
55
+ name: "cd Item 5",
56
+ value: 5,
57
+ },
58
+ ];
59
+
60
+ const options = [
61
+ {
62
+ id: "edit",
63
+ label: "Edit",
64
+ onClick: jest.fn(),
65
+ },
66
+ {
67
+ id: "remove",
68
+ label: "Remove",
69
+ onClick: jest.fn(),
70
+ },
71
+ ];
72
+
73
+ describe("TableList", () => {
74
+ const renderInstance = ({
75
+ search = false,
76
+ defaultSort = "value",
77
+ loading = false,
78
+ options = undefined,
79
+ onClick = jest.fn(),
80
+ }: {
81
+ defaultSort?: string;
82
+ loading?: boolean;
83
+ search?: boolean;
84
+ options?: TableRowOption<any>[];
85
+ onClick?: jest.Mock;
86
+ } = {}) => {
87
+ const instance = render(
88
+ <TableList
89
+ search={search}
90
+ data={data}
91
+ columns={columns}
92
+ defaultSort={defaultSort}
93
+ defaultOrder="asc"
94
+ loading={loading}
95
+ options={options}
96
+ onClick={onClick}
97
+ />,
98
+ );
99
+
100
+ return { ...instance, onClick };
101
+ };
102
+
103
+ const openOptionsMenu = async (index = 0) => {
104
+ await userEvents.click(screen.getAllByTestId("MoreVertIcon")[index]);
105
+ };
106
+
107
+ it("would render all items and its values", () => {
108
+ renderInstance();
109
+
110
+ data.forEach(({ id, name, value }) => {
111
+ expect(screen.getByRole("cell", { name: id })).toBeInTheDocument();
112
+ expect(screen.getByRole("cell", { name: name })).toBeInTheDocument();
113
+ expect(screen.getByRole("cell", { name: value })).toBeInTheDocument();
114
+ });
115
+ });
116
+
117
+ it("all items are rendered in the correct default order", async () => {
118
+ renderInstance();
119
+
120
+ [/item 1/i, /item 2/i, /item 3/i, /item 4/i, /item 5/i].forEach((item, index) => {
121
+ expect(screen.getByRole("cell", { name: item })).toHaveAttribute(
122
+ "aria-rowindex",
123
+ index.toString(),
124
+ );
125
+ });
126
+ });
127
+
128
+ it("loading is true a loading indicator is displayed", () => {
129
+ renderInstance({ loading: true });
130
+
131
+ expectProgressIndicator();
132
+ });
133
+
134
+ describe("search", () => {
135
+ it("would render a search input if search is true", () => {
136
+ renderInstance({ search: true });
137
+
138
+ expect(screen.getByRole("search")).toBeInTheDocument();
139
+ });
140
+
141
+ it("wouldn't render a search input if search is false", () => {
142
+ renderInstance({ search: false });
143
+
144
+ expect(screen.queryByRole("search")).not.toBeInTheDocument();
145
+ });
146
+ });
147
+
148
+ describe("if I search an 'a'", () => {
149
+ it("would render the items 2,3 and 4", async () => {
150
+ renderInstance({ search: true });
151
+
152
+ await userEvents.type(screen.getByPlaceholderText(/search/i), "a");
153
+
154
+ expect(screen.getByText(/item 2/i)).toBeInTheDocument();
155
+ expect(screen.getByText(/item 3/i)).toBeInTheDocument();
156
+ expect(screen.getByText(/item 4/i)).toBeInTheDocument();
157
+
158
+ expect(screen.queryByText(/item 1/i)).not.toBeInTheDocument();
159
+ expect(screen.queryByText(/item 5/i)).not.toBeInTheDocument();
160
+ });
161
+
162
+ it("would render all the items if I remove the search", async () => {
163
+ renderInstance({ search: true });
164
+
165
+ const search = screen.getByPlaceholderText(/search/i);
166
+ await userEvents.type(search, "a");
167
+ await userEvents.clear(search);
168
+
169
+ expect(screen.getByText(/item 1/i)).toBeInTheDocument();
170
+ expect(screen.getByText(/item 2/i)).toBeInTheDocument();
171
+ expect(screen.getByText(/item 3/i)).toBeInTheDocument();
172
+ expect(screen.getByText(/item 4/i)).toBeInTheDocument();
173
+ expect(screen.getByText(/item 5/i)).toBeInTheDocument();
174
+ });
175
+ });
176
+
177
+ describe("sort by a sortable column", () => {
178
+ it("once it will be asc sorted", async () => {
179
+ renderInstance();
180
+
181
+ await userEvents.click(screen.getByText(/name/i));
182
+
183
+ expect(screen.getByRole("cell", { name: /item 2/i })).toHaveAttribute("aria-rowindex", "0");
184
+ expect(screen.getByRole("cell", { name: /item 4/i })).toHaveAttribute("aria-rowindex", "1");
185
+ expect(screen.getByRole("cell", { name: /item 3/i })).toHaveAttribute("aria-rowindex", "2");
186
+ expect(screen.getByRole("cell", { name: /item 5/i })).toHaveAttribute("aria-rowindex", "3");
187
+ expect(screen.getByRole("cell", { name: /item 1/i })).toHaveAttribute("aria-rowindex", "4");
188
+ });
189
+
190
+ it("two times it will be desc sorted", async () => {
191
+ renderInstance();
192
+
193
+ await userEvents.click(screen.getByText(/name/i));
194
+ await userEvents.click(screen.getByText(/name/i));
195
+
196
+ expect(screen.getByRole("cell", { name: /item 1/i })).toHaveAttribute("aria-rowindex", "0");
197
+ expect(screen.getByRole("cell", { name: /item 5/i })).toHaveAttribute("aria-rowindex", "1");
198
+ expect(screen.getByRole("cell", { name: /item 3/i })).toHaveAttribute("aria-rowindex", "2");
199
+ expect(screen.getByRole("cell", { name: /item 4/i })).toHaveAttribute("aria-rowindex", "3");
200
+ expect(screen.getByRole("cell", { name: /item 2/i })).toHaveAttribute("aria-rowindex", "4");
201
+ });
202
+ });
203
+
204
+ it("sort a non sortable column", async () => {
205
+ renderInstance();
206
+
207
+ await userEvents.click(screen.getByText(/id/i));
208
+
209
+ expect(screen.getByRole("cell", { name: /item 1/i })).toHaveAttribute("aria-rowindex", "0");
210
+ expect(screen.getByRole("cell", { name: /item 2/i })).toHaveAttribute("aria-rowindex", "1");
211
+ expect(screen.getByRole("cell", { name: /item 3/i })).toHaveAttribute("aria-rowindex", "2");
212
+ expect(screen.getByRole("cell", { name: /item 4/i })).toHaveAttribute("aria-rowindex", "3");
213
+ expect(screen.getByRole("cell", { name: /item 5/i })).toHaveAttribute("aria-rowindex", "4");
214
+ });
215
+
216
+ it("would call onClick if a row is clicked", async () => {
217
+ const { onClick } = renderInstance();
218
+
219
+ await userEvents.click(screen.getByRole("row", { name: /item 3/i }));
220
+
221
+ expect(onClick).toHaveBeenCalledTimes(1);
222
+ expect(onClick).toHaveBeenCalledWith(data[2]);
223
+ });
224
+
225
+ it("would continue working if onClick is not defined and we click a row", async () => {
226
+ renderInstance({ onClick: undefined });
227
+
228
+ await userEvents.click(screen.getByRole("row", { name: /item 3/i }));
229
+ });
230
+
231
+ describe("options menu", () => {
232
+ it("would render a button to open the menu if doesn't have options", () => {
233
+ renderInstance({ options: undefined });
234
+
235
+ expect(screen.queryByTestId("MoreVertIcon")).not.toBeInTheDocument();
236
+ });
237
+
238
+ it("would render a button to open the menu if has options", () => {
239
+ renderInstance({ options });
240
+
241
+ screen.getAllByTestId("MoreVertIcon");
242
+ });
243
+
244
+ it("would render a menu if has options", async () => {
245
+ renderInstance({ options });
246
+
247
+ await openOptionsMenu();
248
+
249
+ expect(await screen.findByRole("menu")).toBeInTheDocument();
250
+ });
251
+
252
+ it.each([
253
+ [/edit/i, 0],
254
+ [/remove/i, 1],
255
+ ])("would call the option onClick with the row if %s option is clicked", async (name, i) => {
256
+ const index = 1;
257
+ const item = data[index];
258
+ const option = options[i];
259
+
260
+ renderInstance({ options });
261
+
262
+ await openOptionsMenu(index);
263
+
264
+ await userEvents.click(screen.getByRole("menuitem", { name }));
265
+
266
+ expect(option.onClick).toHaveBeenCalledTimes(1);
267
+ expect(option.onClick).toHaveBeenCalledWith(item);
268
+ });
269
+
270
+ it("would close the menu when an option is clicked", async () => {
271
+ const index = 1;
272
+
273
+ renderInstance({ options });
274
+
275
+ await openOptionsMenu(index);
276
+ await userEvents.click(screen.getByRole("menuitem", { name: /edit/i }));
277
+
278
+ expect(screen.queryByRole("menu")).not.toBeInTheDocument();
279
+ });
280
+
281
+ it("wouldn't call the component onClick if an option if clicked", async () => {
282
+ const { onClick } = renderInstance({ options });
283
+
284
+ await openOptionsMenu();
285
+
286
+ await userEvents.click(screen.getByRole("menuitem", { name: /edit/i }));
287
+
288
+ expect(onClick).not.toHaveBeenCalled();
289
+ });
290
+ });
291
+ });
@@ -0,0 +1,127 @@
1
+ import { TableRow, TableCell, IconButton, MenuItem, Menu } from "@mui/material";
2
+ import React from "react";
3
+ import { EnhancedTable, HeadCell, Order } from "../table/enhanced-table";
4
+ import MoreVertIcon from "@mui/icons-material/MoreVert";
5
+ import { BasicModelInstance } from "~/generators";
6
+
7
+ const OptionsId = "__options";
8
+
9
+ export interface TableRowOption<T extends BasicModelInstance> {
10
+ id: string;
11
+ label: string;
12
+ onClick: (item: T) => void;
13
+ }
14
+
15
+ export interface TableListProps<T extends BasicModelInstance> {
16
+ columns: HeadCell[];
17
+ data: T[];
18
+ search?: boolean;
19
+ defaultSort: string;
20
+ defaultOrder?: Order;
21
+ loading?: boolean;
22
+ options?: TableRowOption<T>[];
23
+ onClick?: (d: T) => void;
24
+ }
25
+
26
+ export const TableList = <T extends BasicModelInstance>({
27
+ columns: columnsProp,
28
+ options,
29
+ data,
30
+ onClick,
31
+ search,
32
+ defaultSort,
33
+ defaultOrder,
34
+ loading,
35
+ }: TableListProps<T>) => {
36
+ const columns = columnsProp;
37
+ const [anchorMenuEl, setAnchorMenuEl] = React.useState<null | { item: T; anchor: HTMLElement }>(
38
+ null,
39
+ );
40
+ if (options && !columns.some((c) => c.id === OptionsId)) {
41
+ columns.push({
42
+ id: OptionsId,
43
+ label: "",
44
+ disablePadding: false,
45
+ numeric: false,
46
+ sort: false,
47
+ });
48
+ }
49
+
50
+ return (
51
+ <>
52
+ <EnhancedTable<T>
53
+ columns={columns}
54
+ data={data}
55
+ search={search}
56
+ defaultSort={defaultSort}
57
+ defaultOrder={defaultOrder}
58
+ loading={loading}
59
+ >
60
+ {(filteredData) =>
61
+ filteredData.map((row, i) => {
62
+ return (
63
+ <TableRow
64
+ key={row.id}
65
+ onClick={() => onClick && onClick(row)}
66
+ role="row"
67
+ aria-rowindex={i}
68
+ sx={{ cursor: onClick && "pointer" }}
69
+ >
70
+ {columns.map(({ id }, j) => (
71
+ <TableCell role="cell" scope="row" key={id} aria-rowindex={i} aria-colindex={j}>
72
+ {row[id]}
73
+ </TableCell>
74
+ ))}
75
+ {options && (
76
+ <TableCell>
77
+ <IconButton
78
+ data-testid={`options-${row.id}`}
79
+ onClick={(event) => {
80
+ event.stopPropagation();
81
+ setAnchorMenuEl({
82
+ item: row,
83
+ anchor: event.currentTarget,
84
+ });
85
+ }}
86
+ >
87
+ <MoreVertIcon />
88
+ </IconButton>
89
+ </TableCell>
90
+ )}
91
+ </TableRow>
92
+ );
93
+ })
94
+ }
95
+ </EnhancedTable>
96
+ {options && (
97
+ <Menu
98
+ anchorEl={anchorMenuEl?.anchor}
99
+ open={!!anchorMenuEl}
100
+ onClose={() => setAnchorMenuEl(null)}
101
+ anchorOrigin={{
102
+ vertical: "top",
103
+ horizontal: "left",
104
+ }}
105
+ transformOrigin={{
106
+ vertical: "top",
107
+ horizontal: "left",
108
+ }}
109
+ >
110
+ {options.map(({ id, label, onClick }) => (
111
+ <MenuItem
112
+ key={id}
113
+ onClick={() => {
114
+ if (anchorMenuEl) {
115
+ onClick(anchorMenuEl?.item);
116
+ }
117
+ setAnchorMenuEl(null);
118
+ }}
119
+ >
120
+ {label}
121
+ </MenuItem>
122
+ ))}
123
+ </Menu>
124
+ )}
125
+ </>
126
+ );
127
+ };
@@ -0,0 +1,35 @@
1
+ import React from "react";
2
+ import { ValueBoolean } from "../value-boolean/value-boolean";
3
+ import { ValueText } from "../value-text";
4
+ import { GroupValueCard, GroupValueItem } from "./group-value-card";
5
+
6
+ export const GroupValueCardDummy = (props) => {
7
+ return (
8
+ <GroupValueCard {...props}>
9
+ <GroupValueItem xs={12} sm={6} md={4}>
10
+ <ValueText label="Hello world" value="Lorem ipsum sit amet" />
11
+ </GroupValueItem>
12
+ <GroupValueItem xs={12} sm={6} md={2}>
13
+ <ValueBoolean label="Enabled" value />
14
+ </GroupValueItem>
15
+ <GroupValueItem xs={12} sm={6} md={3}>
16
+ <ValueText label="Quantity" value="1200" />
17
+ </GroupValueItem>
18
+ <GroupValueItem xs={12} sm={6} md={3}>
19
+ <ValueText label="Currency" value="EUR" />
20
+ </GroupValueItem>
21
+ <GroupValueItem xs={12} sm={6} md={6}>
22
+ <ValueText
23
+ label="I am Batman"
24
+ value=" Does it come in black? It's ends here. Hero can be anyone"
25
+ />
26
+ </GroupValueItem>
27
+ <GroupValueItem xs={12} sm={6} md={3}>
28
+ <ValueText label="Status" value="Open" />
29
+ </GroupValueItem>
30
+ <GroupValueItem xs={12} sm={6} md={3}>
31
+ <ValueText label="Level" value="2144" />
32
+ </GroupValueItem>
33
+ </GroupValueCard>
34
+ );
35
+ };
@@ -0,0 +1,26 @@
1
+ import { ComponentMeta } from "@storybook/react";
2
+ import { GroupValueCard } from "./group-value-card";
3
+ import { createTemplate, withPadding } from "../../../storybook";
4
+ import { GroupValueCardDummy } from "./group-value-card.mock";
5
+
6
+ export default {
7
+ title: "Value displays/GroupValueCard",
8
+ component: GroupValueCard,
9
+ decorators: [withPadding(2)],
10
+ parameters: {
11
+ layout: "fullscreen",
12
+ },
13
+ } as ComponentMeta<typeof GroupValueCard>;
14
+
15
+ const Template = createTemplate(GroupValueCardDummy);
16
+
17
+ export const Default = Template.bind({});
18
+ Default.args = {
19
+ title: "Hello world",
20
+ subtitle: "Lorem ipsum sit amet",
21
+ };
22
+
23
+ export const WihtoutSubtitle = Template.bind({});
24
+ WihtoutSubtitle.args = {
25
+ title: "Hello world",
26
+ };