@campxdev/react-blueprint 1.2.6 → 1.2.8
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.
- package/export.ts +4 -2
- package/package.json +4 -1
- package/src/App.tsx +2 -14
- package/src/components/Assets/Icons/IconComponents/EmailIcon.tsx +22 -0
- package/src/components/Assets/Icons/IconComponents/SmsIcon.tsx +22 -0
- package/src/components/Assets/Icons/IconComponents/TextLocalIcon.tsx +8 -0
- package/src/components/Assets/Icons/IconComponents/TextlocalLogo.svg +9 -0
- package/src/components/Assets/Icons/IconComponents/WhatsappIcon.tsx +30 -0
- package/src/components/Assets/Icons/Icons.tsx +8 -0
- package/src/components/DataDisplay/Card/Card.tsx +3 -5
- package/src/components/Input/Button/Button.tsx +17 -2
- package/src/components/Navigation/DropDownMenu/DropDownMenu.tsx +3 -1
- package/src/components/Navigation/TableColumnFilters/TableColumnFilters.tsx +34 -11
- package/src/components/Navigation/TableDensityFilter/TableDensityFilter.tsx +16 -5
- package/src/contexts/Providers.tsx +8 -1
- package/src/hooks/export.ts +1 -0
- package/src/hooks/usePageHeader.ts +27 -0
- package/src/index.tsx +5 -2
- package/src/state/export.ts +2 -0
- package/src/state/filters/filtersSlice.ts +41 -0
- package/src/state/store.ts +13 -0
- package/src/stories/Navigation/ColumnFilters.stories.tsx +5 -15
- package/src/stories/Navigation/TableDensityFilter.stories.tsx +2 -8
package/export.ts
CHANGED
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@campxdev/react-blueprint",
|
|
3
|
-
"version": "1.2.
|
|
3
|
+
"version": "1.2.8",
|
|
4
4
|
"main": "./export.ts",
|
|
5
5
|
"private": false,
|
|
6
6
|
"dependencies": {
|
|
@@ -11,6 +11,7 @@
|
|
|
11
11
|
"@mui/material": "^5.15.20",
|
|
12
12
|
"@mui/x-data-grid": "^7.5.1",
|
|
13
13
|
"@mui/x-date-pickers": "^7.12.1",
|
|
14
|
+
"@reduxjs/toolkit": "^2.2.7",
|
|
14
15
|
"@storybook/addon-backgrounds": "^8.2.6",
|
|
15
16
|
"@storybook/addon-designs": "^8.0.3",
|
|
16
17
|
"@testing-library/jest-dom": "^5.14.1",
|
|
@@ -35,9 +36,11 @@
|
|
|
35
36
|
"react-hook-form": "^7.52.0",
|
|
36
37
|
"react-joyride": "^2.8.2",
|
|
37
38
|
"react-query": "^3.39.3",
|
|
39
|
+
"react-redux": "^9.1.2",
|
|
38
40
|
"react-router-dom": "^6.24.0",
|
|
39
41
|
"react-scripts": "^5.0.1",
|
|
40
42
|
"recharts": "^2.12.7",
|
|
43
|
+
"redux": "^5.0.1",
|
|
41
44
|
"storybook-dark-mode": "^4.0.1",
|
|
42
45
|
"typescript": "^5.5.2",
|
|
43
46
|
"web-vitals": "^2.1.0",
|
package/src/App.tsx
CHANGED
|
@@ -1,12 +1,8 @@
|
|
|
1
|
-
import { Stack } from '@mui/material';
|
|
2
|
-
import { ReactNode, useState } from 'react';
|
|
3
1
|
import './App.css';
|
|
4
|
-
import {
|
|
5
|
-
import { AppHeader, SearchBar } from './components/export';
|
|
2
|
+
import { AppHeader, Icons } from './components/export';
|
|
6
3
|
import Providers from './contexts/Providers';
|
|
7
4
|
|
|
8
5
|
function App() {
|
|
9
|
-
const [selected, setSelected] = useState<ReactNode[]>([]);
|
|
10
6
|
const sampleChips = [
|
|
11
7
|
{ label: 'Chip1', value: 1 },
|
|
12
8
|
{ label: 'Chip2', value: 2 },
|
|
@@ -22,15 +18,7 @@ function App() {
|
|
|
22
18
|
collapsed={false}
|
|
23
19
|
institutionsData={[]}
|
|
24
20
|
/>
|
|
25
|
-
<
|
|
26
|
-
|
|
27
|
-
<Tags tags={sampleChips} />
|
|
28
|
-
|
|
29
|
-
<Stack>
|
|
30
|
-
{selected.map((item) => {
|
|
31
|
-
return <>{item}</>;
|
|
32
|
-
})}
|
|
33
|
-
</Stack>
|
|
21
|
+
<Icons.WhatsappIcon />
|
|
34
22
|
</Providers>
|
|
35
23
|
);
|
|
36
24
|
}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
export const EmailIcon = () => {
|
|
2
|
+
return (
|
|
3
|
+
<div>
|
|
4
|
+
<svg
|
|
5
|
+
xmlns="http://www.w3.org/2000/svg"
|
|
6
|
+
width="24"
|
|
7
|
+
height="25"
|
|
8
|
+
viewBox="0 0 24 25"
|
|
9
|
+
fill="none"
|
|
10
|
+
>
|
|
11
|
+
<path
|
|
12
|
+
d="M19 20.5H5C4.20459 20.4992 3.44199 20.1829 2.87956 19.6204C2.31712 19.058 2.00079 18.2954 2 17.5V7.5C2.00079 6.70459 2.31712 5.94199 2.87956 5.37956C3.44199 4.81712 4.20459 4.50079 5 4.5H19C19.7954 4.50079 20.558 4.81712 21.1204 5.37956C21.6829 5.94199 21.9992 6.70459 22 7.5V17.5C21.9992 18.2954 21.6829 19.058 21.1204 19.6204C20.558 20.1829 19.7954 20.4992 19 20.5Z"
|
|
13
|
+
fill="#B2B1FF"
|
|
14
|
+
/>
|
|
15
|
+
<path
|
|
16
|
+
d="M22 7.5C21.9992 6.70459 21.6829 5.94199 21.1204 5.37956C20.558 4.81712 19.7954 4.50079 19 4.5H5C4.20459 4.50079 3.44199 4.81712 2.87956 5.37956C2.31712 5.94199 2.00079 6.70459 2 7.5V8.561L11.479 14.353C11.6361 14.4489 11.8165 14.4996 12.0005 14.4996C12.1845 14.4996 12.3649 14.4489 12.522 14.353L22 8.561V7.5Z"
|
|
17
|
+
fill="#6563FF"
|
|
18
|
+
/>
|
|
19
|
+
</svg>
|
|
20
|
+
</div>
|
|
21
|
+
);
|
|
22
|
+
};
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
export const SmsIcon = () => {
|
|
2
|
+
return (
|
|
3
|
+
<div>
|
|
4
|
+
<svg
|
|
5
|
+
xmlns="http://www.w3.org/2000/svg"
|
|
6
|
+
width="24"
|
|
7
|
+
height="25"
|
|
8
|
+
viewBox="0 0 24 25"
|
|
9
|
+
fill="none"
|
|
10
|
+
>
|
|
11
|
+
<path
|
|
12
|
+
d="M12 2.75C6.624 2.75 2.25 7.124 2.25 12.5C2.25 14.13 2.666 15.74 3.455 17.178L2.279 21.294C2.204 21.556 2.277 21.838 2.47 22.03C2.613 22.173 2.804 22.25 3 22.25C3.069 22.25 3.138 22.241 3.206 22.221L7.322 21.045C8.76 21.835 10.37 22.25 12 22.25C17.376 22.25 21.75 17.876 21.75 12.5C21.75 7.124 17.376 2.75 12 2.75Z"
|
|
13
|
+
fill="#BAD7FF"
|
|
14
|
+
/>
|
|
15
|
+
<path
|
|
16
|
+
d="M13 12.5C13 13.052 12.552 13.5 12 13.5C11.448 13.5 11 13.052 11 12.5C11 11.948 11.448 11.5 12 11.5C12.552 11.5 13 11.948 13 12.5ZM8 11.5C7.448 11.5 7 11.948 7 12.5C7 13.052 7.448 13.5 8 13.5C8.552 13.5 9 13.052 9 12.5C9 11.948 8.552 11.5 8 11.5ZM16 11.5C15.448 11.5 15 11.948 15 12.5C15 13.052 15.448 13.5 16 13.5C16.552 13.5 17 13.052 17 12.5C17 11.948 16.552 11.5 16 11.5Z"
|
|
17
|
+
fill="#549BFF"
|
|
18
|
+
/>
|
|
19
|
+
</svg>
|
|
20
|
+
</div>
|
|
21
|
+
);
|
|
22
|
+
};
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
<svg width="316" height="316" viewBox="0 0 316 316" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
|
|
2
|
+
<path d="M0 0H316V316H0V0Z" fill="url(#pattern0_4854_62318)"/>
|
|
3
|
+
<defs>
|
|
4
|
+
<pattern id="pattern0_4854_62318" patternContentUnits="objectBoundingBox" width="1" height="1">
|
|
5
|
+
<use xlink:href="#image0_4854_62318" transform="scale(0.00316456)"/>
|
|
6
|
+
</pattern>
|
|
7
|
+
<image id="image0_4854_62318" width="316" height="316" xlink:href=""/>
|
|
8
|
+
</defs>
|
|
9
|
+
</svg>
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
|
|
3
|
+
interface WhatsappIconProps {
|
|
4
|
+
width?: string | number;
|
|
5
|
+
height?: string | number;
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
export const WhatsappIcon: React.FC<WhatsappIconProps> = ({
|
|
9
|
+
width = 24,
|
|
10
|
+
height = 25,
|
|
11
|
+
}) => {
|
|
12
|
+
return (
|
|
13
|
+
<div>
|
|
14
|
+
<svg
|
|
15
|
+
xmlns="http://www.w3.org/2000/svg"
|
|
16
|
+
width={width}
|
|
17
|
+
height={height}
|
|
18
|
+
viewBox="0 0 24 25"
|
|
19
|
+
fill="none"
|
|
20
|
+
>
|
|
21
|
+
<path
|
|
22
|
+
fillRule="evenodd"
|
|
23
|
+
clipRule="evenodd"
|
|
24
|
+
d="M11.9971 3C6.48459 3 2.00001 7.48541 2.00001 13C1.99682 15.1063 2.66369 17.159 3.90418 18.8612L2.65834 22.5766L6.50168 21.3483C8.13244 22.4293 10.0464 23.0039 12.0029 23C17.5154 23 22 18.5142 22 13C22 7.48583 17.5158 3 12.0029 3H11.9971ZM9.20542 8.07916C9.01126 7.615 8.86459 7.5975 8.57084 7.58541C8.45931 7.57835 8.34759 7.5746 8.23584 7.57416C7.85334 7.57416 7.45376 7.68583 7.21293 7.9325C6.91918 8.23249 6.19043 8.93166 6.19043 10.3662C6.19043 11.8008 7.23626 13.1883 7.37751 13.3825C7.52459 13.5762 9.41709 16.5625 12.3554 17.7796C14.6533 18.7321 15.3354 18.6437 15.8583 18.5321C16.6225 18.3675 17.5804 17.8029 17.8213 17.1212C18.0625 16.4392 18.0625 15.8571 17.9921 15.7337C17.9213 15.6104 17.7271 15.54 17.4338 15.3929C17.1396 15.2458 15.7113 14.5404 15.4408 14.4462C15.1767 14.3462 14.9242 14.3817 14.7242 14.6637C14.4421 15.0579 14.1658 15.4575 13.9425 15.6983C13.7663 15.8867 13.4779 15.9104 13.2371 15.81C12.9138 15.675 12.0088 15.3575 10.8921 14.3642C10.0279 13.5937 9.44042 12.6358 9.27001 12.3475C9.09917 12.0537 9.25251 11.8833 9.38751 11.7246C9.53459 11.5421 9.67542 11.4129 9.82251 11.2425C9.96917 11.0717 10.0517 10.9837 10.1458 10.7842C10.2458 10.59 10.175 10.39 10.1042 10.2425C10.0342 10.0958 9.44626 8.66124 9.20542 8.07958V8.07916Z"
|
|
25
|
+
fill="#67C15E"
|
|
26
|
+
/>
|
|
27
|
+
</svg>
|
|
28
|
+
</div>
|
|
29
|
+
);
|
|
30
|
+
};
|
|
@@ -28,6 +28,7 @@ import { DeviceIcon } from './IconComponents/DeviceIcon';
|
|
|
28
28
|
import { DocumentIcon } from './IconComponents/DocumentIcon';
|
|
29
29
|
import { DownloadIcon } from './IconComponents/DownloadIcon';
|
|
30
30
|
import { EditIcon } from './IconComponents/EditIcon';
|
|
31
|
+
import { EmailIcon } from './IconComponents/EmailIcon';
|
|
31
32
|
import { EnrollxIcon } from './IconComponents/EnrollxIcon';
|
|
32
33
|
import { ExamResultIcon } from './IconComponents/ExamResultIcon';
|
|
33
34
|
import { ExamxIcon } from './IconComponents/ExamxIcon';
|
|
@@ -60,8 +61,10 @@ import { RightIcon } from './IconComponents/RightIcon';
|
|
|
60
61
|
import { SaveIcon } from './IconComponents/SaveIcon';
|
|
61
62
|
import { SearchIcon } from './IconComponents/SearchIcon';
|
|
62
63
|
import { ShareIcon } from './IconComponents/ShareIcon';
|
|
64
|
+
import { SmsIcon } from './IconComponents/SmsIcon';
|
|
63
65
|
import { StandardIcon } from './IconComponents/StandardIcon';
|
|
64
66
|
import { SuccessFilledIcon } from './IconComponents/SuccessFilledIcon';
|
|
67
|
+
import { TextLocalIcon } from './IconComponents/TextLocalIcon';
|
|
65
68
|
import { TicketsIcon } from './IconComponents/TicketsIcon';
|
|
66
69
|
import { UmsIcon } from './IconComponents/UmsIcon';
|
|
67
70
|
import { UnCheckedCheckboxIcon } from './IconComponents/UncheckCheckBoxIcon';
|
|
@@ -70,10 +73,15 @@ import { ViewIcon } from './IconComponents/ViewIcon';
|
|
|
70
73
|
import { VisibilityOffIcon } from './IconComponents/VisibiityOffIcon';
|
|
71
74
|
import { VisibilityIcon } from './IconComponents/VisibilityIcon';
|
|
72
75
|
import { WarningFilledIcon } from './IconComponents/WarningFilledIcon';
|
|
76
|
+
import { WhatsappIcon } from './IconComponents/WhatsappIcon';
|
|
73
77
|
|
|
74
78
|
export const Icons = {
|
|
75
79
|
NoteIcon,
|
|
76
80
|
RedoIcon,
|
|
81
|
+
WhatsappIcon,
|
|
82
|
+
SmsIcon,
|
|
83
|
+
EmailIcon,
|
|
84
|
+
TextLocalIcon,
|
|
77
85
|
ArrowBackIcon,
|
|
78
86
|
CollapseIcon,
|
|
79
87
|
ExpandIcon,
|
|
@@ -44,7 +44,7 @@ export interface CardProps {
|
|
|
44
44
|
};
|
|
45
45
|
status?: string;
|
|
46
46
|
imageSrc?: string;
|
|
47
|
-
|
|
47
|
+
titleImgIcon?: ReactNode;
|
|
48
48
|
checkBox?: boolean;
|
|
49
49
|
moreOptions?: boolean;
|
|
50
50
|
menu?: Array<ReactNode>;
|
|
@@ -56,7 +56,7 @@ export interface CardProps {
|
|
|
56
56
|
|
|
57
57
|
export const Card = ({
|
|
58
58
|
title,
|
|
59
|
-
|
|
59
|
+
titleImgIcon,
|
|
60
60
|
caption,
|
|
61
61
|
status,
|
|
62
62
|
footer,
|
|
@@ -94,9 +94,7 @@ export const Card = ({
|
|
|
94
94
|
</StyledStack>
|
|
95
95
|
)} */}
|
|
96
96
|
|
|
97
|
-
{
|
|
98
|
-
<img src={titleImgUrl} alt={title} height={'50px'} width={'50px'} />
|
|
99
|
-
)}
|
|
97
|
+
{titleImgIcon && titleImgIcon}
|
|
100
98
|
|
|
101
99
|
<Stack justifyContent={'center'}>
|
|
102
100
|
<Typography variant={titleProps.variant} sx={titleProps.sx}>
|
|
@@ -7,13 +7,28 @@ import './ButtonLoader.css';
|
|
|
7
7
|
export type ButtonProps = {
|
|
8
8
|
loading?: boolean;
|
|
9
9
|
color?: string;
|
|
10
|
+
background?: string;
|
|
11
|
+
hoverBackground?: string;
|
|
10
12
|
} & Omit<MuiButtonProps, 'color'>;
|
|
11
13
|
|
|
12
|
-
export const Button = ({
|
|
14
|
+
export const Button = ({
|
|
15
|
+
loading = false,
|
|
16
|
+
color,
|
|
17
|
+
background,
|
|
18
|
+
hoverBackground,
|
|
19
|
+
...props
|
|
20
|
+
}: ButtonProps) => {
|
|
13
21
|
return (
|
|
14
22
|
<MuiButton
|
|
15
|
-
sx={{ color, ...props.sx }}
|
|
16
23
|
{...props}
|
|
24
|
+
sx={{
|
|
25
|
+
color,
|
|
26
|
+
background,
|
|
27
|
+
':hover': {
|
|
28
|
+
background: hoverBackground,
|
|
29
|
+
},
|
|
30
|
+
...props.sx,
|
|
31
|
+
}}
|
|
17
32
|
endIcon={loading ? <div className="buttonLoader"></div> : props.endIcon}
|
|
18
33
|
disabled={props.disabled || loading}
|
|
19
34
|
>
|
|
@@ -87,7 +87,9 @@ export const DropdownMenu = ({
|
|
|
87
87
|
{menuItems.map((item, index) => (
|
|
88
88
|
<Fragment key={index}>
|
|
89
89
|
{item}
|
|
90
|
-
<
|
|
90
|
+
{index < menuItems.length - 1 && (
|
|
91
|
+
<Divider flexItem sx={{ margin: '0 !important' }} />
|
|
92
|
+
)}
|
|
91
93
|
</Fragment>
|
|
92
94
|
))}
|
|
93
95
|
</Stack>
|
|
@@ -1,7 +1,9 @@
|
|
|
1
1
|
import { KeyboardArrowDown } from '@mui/icons-material';
|
|
2
2
|
import { MenuListProps, MenuProps, Typography, useTheme } from '@mui/material';
|
|
3
|
-
import { GridColDef } from '@mui/x-data-grid';
|
|
3
|
+
import { GridColDef, GridColumnVisibilityModel } from '@mui/x-data-grid';
|
|
4
4
|
import { useState } from 'react';
|
|
5
|
+
import { useDispatch } from 'react-redux';
|
|
6
|
+
import { setColumnVisibilityModel } from '../../../state/export';
|
|
5
7
|
import { Button, Icons, SearchBar, SingleCheckBox } from '../../export';
|
|
6
8
|
import { DropdownMenu } from '../export';
|
|
7
9
|
|
|
@@ -9,16 +11,26 @@ export type TableColumnFiltersProps = {
|
|
|
9
11
|
columns: GridColDef[];
|
|
10
12
|
menuProps?: Omit<MenuProps, 'open'>;
|
|
11
13
|
menuListProps?: MenuListProps;
|
|
12
|
-
|
|
14
|
+
uniqueId: string;
|
|
13
15
|
};
|
|
14
16
|
|
|
15
17
|
export const TableColumnFilters = ({
|
|
16
18
|
columns,
|
|
17
|
-
|
|
19
|
+
uniqueId,
|
|
18
20
|
...props
|
|
19
21
|
}: TableColumnFiltersProps) => {
|
|
20
|
-
const
|
|
22
|
+
const dispatch = useDispatch();
|
|
23
|
+
|
|
24
|
+
const [columnVisibilityModel, setGridColumnVisibilityModels] =
|
|
25
|
+
useState<GridColumnVisibilityModel>(
|
|
26
|
+
columns.reduce((acc, column) => {
|
|
27
|
+
acc[column.field] = true;
|
|
28
|
+
return acc;
|
|
29
|
+
}, {} as GridColumnVisibilityModel),
|
|
30
|
+
);
|
|
31
|
+
|
|
21
32
|
const [search, setSearch] = useState<string>('');
|
|
33
|
+
|
|
22
34
|
const theme = useTheme();
|
|
23
35
|
|
|
24
36
|
const filteredOptions = columns.filter((option) =>
|
|
@@ -26,11 +38,19 @@ export const TableColumnFilters = ({
|
|
|
26
38
|
);
|
|
27
39
|
|
|
28
40
|
const handleCheckboxChange = (column: GridColDef, checked: boolean) => {
|
|
29
|
-
const
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
41
|
+
const newColumnVisibilityModel = {
|
|
42
|
+
...columnVisibilityModel,
|
|
43
|
+
[column.field]: !checked,
|
|
44
|
+
};
|
|
45
|
+
setGridColumnVisibilityModels(newColumnVisibilityModel);
|
|
46
|
+
if (uniqueId) {
|
|
47
|
+
dispatch(
|
|
48
|
+
setColumnVisibilityModel({
|
|
49
|
+
uniqueId: uniqueId,
|
|
50
|
+
columnVisibilityModel: newColumnVisibilityModel,
|
|
51
|
+
}),
|
|
52
|
+
);
|
|
53
|
+
}
|
|
34
54
|
};
|
|
35
55
|
return (
|
|
36
56
|
<>
|
|
@@ -50,6 +70,9 @@ export const TableColumnFilters = ({
|
|
|
50
70
|
borderRadius: '20px',
|
|
51
71
|
}}
|
|
52
72
|
variant="contained"
|
|
73
|
+
background={theme.palette.surface.grey}
|
|
74
|
+
hoverBackground={theme.palette.surface.grey}
|
|
75
|
+
color={theme.palette.text.primary}
|
|
53
76
|
endIcon={<KeyboardArrowDown />}
|
|
54
77
|
>
|
|
55
78
|
Columns
|
|
@@ -84,11 +107,11 @@ export const TableColumnFilters = ({
|
|
|
84
107
|
}}
|
|
85
108
|
icon={<Icons.VisibilityIcon />}
|
|
86
109
|
checkedIcon={<Icons.VisibilityOffIcon />}
|
|
87
|
-
checked={
|
|
110
|
+
checked={!columnVisibilityModel[item.field]}
|
|
88
111
|
label={
|
|
89
112
|
<Typography
|
|
90
113
|
color={
|
|
91
|
-
|
|
114
|
+
!columnVisibilityModel[item.field]
|
|
92
115
|
? theme.palette.text.tertiary
|
|
93
116
|
: theme.palette.text.primary
|
|
94
117
|
}
|
|
@@ -1,6 +1,8 @@
|
|
|
1
1
|
import { Box, MenuListProps, MenuProps, styled, useTheme } from '@mui/material';
|
|
2
2
|
import { GridDensity } from '@mui/x-data-grid';
|
|
3
3
|
import { useState } from 'react';
|
|
4
|
+
import { useDispatch } from 'react-redux';
|
|
5
|
+
import { setDensity } from '../../../state/export';
|
|
4
6
|
import {
|
|
5
7
|
Button,
|
|
6
8
|
DropDownIcon,
|
|
@@ -12,14 +14,16 @@ import {
|
|
|
12
14
|
export type TableDensityFilterProps = {
|
|
13
15
|
menuProps?: Omit<MenuProps, 'open'>;
|
|
14
16
|
menuListProps?: MenuListProps;
|
|
15
|
-
|
|
17
|
+
uniqueId: string;
|
|
16
18
|
};
|
|
17
19
|
|
|
18
20
|
export const TableDensityFilter = ({
|
|
19
|
-
|
|
21
|
+
uniqueId,
|
|
20
22
|
...props
|
|
21
23
|
}: TableDensityFilterProps) => {
|
|
22
|
-
const
|
|
24
|
+
const dispatch = useDispatch();
|
|
25
|
+
|
|
26
|
+
const [density, setGridDensity] = useState<GridDensity>('standard');
|
|
23
27
|
|
|
24
28
|
const gridDensity: GridDensity[] = ['compact', 'standard', 'comfortable'];
|
|
25
29
|
|
|
@@ -37,8 +41,15 @@ export const TableDensityFilter = ({
|
|
|
37
41
|
};
|
|
38
42
|
|
|
39
43
|
const handleClick = (value: GridDensity, close: () => void) => {
|
|
40
|
-
|
|
41
|
-
|
|
44
|
+
setGridDensity(value);
|
|
45
|
+
if (uniqueId) {
|
|
46
|
+
dispatch(
|
|
47
|
+
setDensity({
|
|
48
|
+
uniqueId: uniqueId,
|
|
49
|
+
density: value,
|
|
50
|
+
}),
|
|
51
|
+
);
|
|
52
|
+
}
|
|
42
53
|
close();
|
|
43
54
|
};
|
|
44
55
|
|
|
@@ -1,6 +1,13 @@
|
|
|
1
1
|
import { ReactNode } from 'react';
|
|
2
|
+
import { Provider } from 'react-redux';
|
|
3
|
+
|
|
4
|
+
import store from '../state/store';
|
|
2
5
|
import { MuiThemeProvider } from '../themes/MuiThemeProvider';
|
|
3
6
|
|
|
4
7
|
export default function Providers({ children }: { children: ReactNode }) {
|
|
5
|
-
return
|
|
8
|
+
return (
|
|
9
|
+
<MuiThemeProvider>
|
|
10
|
+
<Provider store={store}>{children}</Provider>
|
|
11
|
+
</MuiThemeProvider>
|
|
12
|
+
);
|
|
6
13
|
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from './usePageHeader';
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { useRef } from 'react';
|
|
2
|
+
import { useDispatch, useSelector } from 'react-redux';
|
|
3
|
+
import { v4 } from 'uuid';
|
|
4
|
+
import {
|
|
5
|
+
FilterState,
|
|
6
|
+
resetStateForUniqueId,
|
|
7
|
+
} from '../state/filters/filtersSlice';
|
|
8
|
+
import { RootState } from '../state/store';
|
|
9
|
+
|
|
10
|
+
export const usePageHeader = () => {
|
|
11
|
+
const uuidRef = useRef(v4());
|
|
12
|
+
const dispatch = useDispatch();
|
|
13
|
+
const filterState = useSelector(
|
|
14
|
+
(state: RootState) => state.filters[uuidRef.current] || ({} as FilterState),
|
|
15
|
+
);
|
|
16
|
+
|
|
17
|
+
const resetState = () => {
|
|
18
|
+
dispatch(resetStateForUniqueId({ uniqueId: uuidRef.current }));
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
return {
|
|
22
|
+
columnVisibilityModel: filterState.columnVisibilityModel,
|
|
23
|
+
density: filterState.density,
|
|
24
|
+
uniqueId: uuidRef.current,
|
|
25
|
+
resetState,
|
|
26
|
+
};
|
|
27
|
+
};
|
package/src/index.tsx
CHANGED
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
import React from 'react';
|
|
2
2
|
import ReactDOM from 'react-dom/client';
|
|
3
|
-
import './index.css';
|
|
4
3
|
import App from './App';
|
|
4
|
+
import Providers from './contexts/Providers';
|
|
5
|
+
import './index.css';
|
|
5
6
|
import reportWebVitals from './reportWebVitals';
|
|
6
7
|
|
|
7
8
|
const root = ReactDOM.createRoot(
|
|
@@ -9,7 +10,9 @@ const root = ReactDOM.createRoot(
|
|
|
9
10
|
);
|
|
10
11
|
root.render(
|
|
11
12
|
<React.StrictMode>
|
|
12
|
-
<
|
|
13
|
+
<Providers>
|
|
14
|
+
<App />
|
|
15
|
+
</Providers>
|
|
13
16
|
</React.StrictMode>,
|
|
14
17
|
);
|
|
15
18
|
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
import { GridColumnVisibilityModel, GridDensity } from '@mui/x-data-grid';
|
|
2
|
+
import { createSlice } from '@reduxjs/toolkit';
|
|
3
|
+
|
|
4
|
+
export type FilterState = {
|
|
5
|
+
[uniqueId: string]: {
|
|
6
|
+
density: GridDensity;
|
|
7
|
+
columnVisibilityModel: GridColumnVisibilityModel;
|
|
8
|
+
};
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
const initialState: FilterState = {};
|
|
12
|
+
|
|
13
|
+
const filtersSlice = createSlice({
|
|
14
|
+
name: 'filters',
|
|
15
|
+
initialState,
|
|
16
|
+
reducers: {
|
|
17
|
+
setColumnVisibilityModel: (state, action) => {
|
|
18
|
+
const { uniqueId, columnVisibilityModel } = action.payload;
|
|
19
|
+
if (!state[uniqueId]) {
|
|
20
|
+
state[uniqueId] = { density: 'standard', columnVisibilityModel: {} };
|
|
21
|
+
}
|
|
22
|
+
state[uniqueId].columnVisibilityModel = columnVisibilityModel;
|
|
23
|
+
},
|
|
24
|
+
setDensity: (state, action) => {
|
|
25
|
+
const { uniqueId, density } = action.payload;
|
|
26
|
+
if (!state[uniqueId]) {
|
|
27
|
+
state[uniqueId] = { density: 'standard', columnVisibilityModel: {} };
|
|
28
|
+
}
|
|
29
|
+
state[uniqueId].density = density;
|
|
30
|
+
},
|
|
31
|
+
resetStateForUniqueId: (state, action) => {
|
|
32
|
+
const { uniqueId } = action.payload;
|
|
33
|
+
state[uniqueId] = { density: 'standard', columnVisibilityModel: {} };
|
|
34
|
+
},
|
|
35
|
+
},
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
export const { setColumnVisibilityModel, setDensity, resetStateForUniqueId } =
|
|
39
|
+
filtersSlice.actions;
|
|
40
|
+
|
|
41
|
+
export default filtersSlice.reducer;
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { configureStore } from '@reduxjs/toolkit';
|
|
2
|
+
import filtersSlice from './filters/filtersSlice';
|
|
3
|
+
|
|
4
|
+
const store = configureStore({
|
|
5
|
+
reducer: {
|
|
6
|
+
filters: filtersSlice,
|
|
7
|
+
},
|
|
8
|
+
});
|
|
9
|
+
|
|
10
|
+
export type RootState = ReturnType<typeof store.getState>;
|
|
11
|
+
export type AppDispatch = typeof store.dispatch;
|
|
12
|
+
|
|
13
|
+
export default store;
|
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import { GridColumnVisibilityModel } from '@mui/x-data-grid';
|
|
1
2
|
import { Meta, StoryFn } from '@storybook/react';
|
|
2
3
|
import { useState } from 'react';
|
|
3
4
|
import {
|
|
@@ -13,10 +14,6 @@ const meta: Meta<typeof TableColumnFilters> = {
|
|
|
13
14
|
control: { type: 'object' },
|
|
14
15
|
description: 'Array of filter options with labels and values.',
|
|
15
16
|
},
|
|
16
|
-
onChange: {
|
|
17
|
-
action: 'changed',
|
|
18
|
-
description: 'Callback when the selected values change.',
|
|
19
|
-
},
|
|
20
17
|
menuProps: {
|
|
21
18
|
control: { type: 'object' },
|
|
22
19
|
description: 'Props to customize the MUI Menu component.',
|
|
@@ -34,23 +31,16 @@ const meta: Meta<typeof TableColumnFilters> = {
|
|
|
34
31
|
export default meta;
|
|
35
32
|
|
|
36
33
|
const Template: StoryFn<TableColumnFiltersProps> = (args) => {
|
|
37
|
-
const [selectedValues, setSelectedValues] =
|
|
34
|
+
const [selectedValues, setSelectedValues] =
|
|
35
|
+
useState<GridColumnVisibilityModel>();
|
|
38
36
|
|
|
39
|
-
const handleOnChange = (values:
|
|
37
|
+
const handleOnChange = (values: GridColumnVisibilityModel) => {
|
|
40
38
|
setSelectedValues(values);
|
|
41
|
-
args.onChange(values);
|
|
42
39
|
};
|
|
43
40
|
|
|
44
41
|
return (
|
|
45
42
|
<>
|
|
46
|
-
<TableColumnFilters {...args}
|
|
47
|
-
<div>
|
|
48
|
-
{selectedValues.length > 0 ? (
|
|
49
|
-
<ul>{selectedValues.map((value) => value)}</ul>
|
|
50
|
-
) : (
|
|
51
|
-
<p>No values selected.</p>
|
|
52
|
-
)}
|
|
53
|
-
</div>
|
|
43
|
+
<TableColumnFilters {...args} />
|
|
54
44
|
</>
|
|
55
45
|
);
|
|
56
46
|
};
|
|
@@ -9,10 +9,6 @@ const meta: Meta<typeof TableDensityFilter> = {
|
|
|
9
9
|
title: 'Navigation/TableDensityFilter',
|
|
10
10
|
component: TableDensityFilter,
|
|
11
11
|
argTypes: {
|
|
12
|
-
onChange: {
|
|
13
|
-
action: 'changed',
|
|
14
|
-
description: 'Callback when the selected values change.',
|
|
15
|
-
},
|
|
16
12
|
menuProps: {
|
|
17
13
|
control: { type: 'object' },
|
|
18
14
|
description: 'Props to customize the MUI Menu component.',
|
|
@@ -30,13 +26,11 @@ const meta: Meta<typeof TableDensityFilter> = {
|
|
|
30
26
|
export default meta;
|
|
31
27
|
|
|
32
28
|
const Template: StoryFn<TableDensityFilterProps> = (args) => {
|
|
33
|
-
const handleOnChange = (value: GridDensity) => {
|
|
34
|
-
args.onChange(value);
|
|
35
|
-
};
|
|
29
|
+
const handleOnChange = (value: GridDensity) => {};
|
|
36
30
|
|
|
37
31
|
return (
|
|
38
32
|
<>
|
|
39
|
-
<TableDensityFilter {...args}
|
|
33
|
+
<TableDensityFilter {...args} />
|
|
40
34
|
</>
|
|
41
35
|
);
|
|
42
36
|
};
|