@addev-be/ui 0.1.18

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 (98) hide show
  1. package/assets/fonts/montserrat-200.woff2 +0 -0
  2. package/assets/fonts/montserrat-400.woff2 +0 -0
  3. package/assets/fonts/montserrat-500.woff2 +0 -0
  4. package/assets/fonts/montserrat-700.woff2 +0 -0
  5. package/assets/icons/arrow-down-a-z.svg +1 -0
  6. package/assets/icons/arrow-up-z-a.svg +1 -0
  7. package/assets/icons/arrows-rotate.svg +1 -0
  8. package/assets/icons/arrows-up-down.svg +1 -0
  9. package/assets/icons/check.svg +1 -0
  10. package/assets/icons/copy.svg +1 -0
  11. package/assets/icons/down.svg +1 -0
  12. package/assets/icons/filter-full.svg +1 -0
  13. package/assets/icons/filter-slash.svg +1 -0
  14. package/assets/icons/filter.svg +1 -0
  15. package/assets/icons/hashtag.svg +1 -0
  16. package/assets/icons/image-slash.svg +1 -0
  17. package/assets/icons/left.svg +1 -0
  18. package/assets/icons/magnifier.svg +1 -0
  19. package/assets/icons/phone.svg +1 -0
  20. package/assets/icons/plus.svg +1 -0
  21. package/assets/icons/right.svg +1 -0
  22. package/assets/icons/spinner-third.svg +1 -0
  23. package/assets/icons/table-columns.svg +1 -0
  24. package/assets/icons/up.svg +1 -0
  25. package/assets/icons/user-tie.svg +1 -0
  26. package/eslint.config.js +28 -0
  27. package/package.json +49 -0
  28. package/src/Icons.tsx +80 -0
  29. package/src/components/data/DataGrid/AdvancedRequestDataGrid.tsx +236 -0
  30. package/src/components/data/DataGrid/DataGridCell.tsx +78 -0
  31. package/src/components/data/DataGrid/DataGridColumnsModal/helpers.ts +14 -0
  32. package/src/components/data/DataGrid/DataGridColumnsModal/hooks.tsx +58 -0
  33. package/src/components/data/DataGrid/DataGridColumnsModal/index.tsx +181 -0
  34. package/src/components/data/DataGrid/DataGridColumnsModal/styles.ts +104 -0
  35. package/src/components/data/DataGrid/DataGridEditableCell.tsx +54 -0
  36. package/src/components/data/DataGrid/DataGridFilterMenu/hooks.tsx +75 -0
  37. package/src/components/data/DataGrid/DataGridFilterMenu/index.tsx +190 -0
  38. package/src/components/data/DataGrid/DataGridFilterMenu/styles.ts +100 -0
  39. package/src/components/data/DataGrid/DataGridFooter.tsx +64 -0
  40. package/src/components/data/DataGrid/DataGridHeader.tsx +129 -0
  41. package/src/components/data/DataGrid/DataGridHeaderCell.tsx +166 -0
  42. package/src/components/data/DataGrid/FilterModalContent/index.tsx +125 -0
  43. package/src/components/data/DataGrid/FilterModalContent/styles.ts +22 -0
  44. package/src/components/data/DataGrid/FilterValuesScroller.tsx +131 -0
  45. package/src/components/data/DataGrid/VirtualScroller.tsx +51 -0
  46. package/src/components/data/DataGrid/helpers/advancedRequests.tsx +61 -0
  47. package/src/components/data/DataGrid/helpers/columns.tsx +259 -0
  48. package/src/components/data/DataGrid/helpers/filters.ts +219 -0
  49. package/src/components/data/DataGrid/helpers/index.ts +3 -0
  50. package/src/components/data/DataGrid/hooks/index.ts +30 -0
  51. package/src/components/data/DataGrid/hooks/useDataGrid.tsx +225 -0
  52. package/src/components/data/DataGrid/hooks/useDataGridCopy.ts +166 -0
  53. package/src/components/data/DataGrid/hooks/useDataGridSettings.ts +49 -0
  54. package/src/components/data/DataGrid/index.tsx +145 -0
  55. package/src/components/data/DataGrid/styles.ts +284 -0
  56. package/src/components/data/DataGrid/types.ts +232 -0
  57. package/src/components/data/index.ts +3 -0
  58. package/src/components/forms/Button.tsx +99 -0
  59. package/src/components/forms/IconButton.tsx +57 -0
  60. package/src/components/forms/IndeterminateCheckbox.tsx +46 -0
  61. package/src/components/forms/Select.tsx +40 -0
  62. package/src/components/forms/index.ts +5 -0
  63. package/src/components/forms/styles.ts +20 -0
  64. package/src/components/index.ts +3 -0
  65. package/src/components/layout/Dropdown/index.tsx +79 -0
  66. package/src/components/layout/Dropdown/styles.ts +44 -0
  67. package/src/components/layout/Loading/index.tsx +28 -0
  68. package/src/components/layout/Loading/styles.ts +29 -0
  69. package/src/components/layout/Modal/index.tsx +51 -0
  70. package/src/components/layout/Modal/styles.ts +110 -0
  71. package/src/components/layout/index.ts +3 -0
  72. package/src/config/index.ts +14 -0
  73. package/src/helpers/getScrollbarSize.ts +14 -0
  74. package/src/helpers/numbers.ts +20 -0
  75. package/src/hooks/index.ts +2 -0
  76. package/src/hooks/useElementSize.ts +24 -0
  77. package/src/hooks/useWindowSize.ts +20 -0
  78. package/src/index.ts +7 -0
  79. package/src/providers/PortalsProvider/index.tsx +54 -0
  80. package/src/providers/PortalsProvider/styles.ts +27 -0
  81. package/src/providers/SettingsProvider/index.tsx +70 -0
  82. package/src/providers/ThemeProvider/ThemeProvider.ts +55 -0
  83. package/src/providers/ThemeProvider/defaultTheme.ts +444 -0
  84. package/src/providers/ThemeProvider/index.ts +3 -0
  85. package/src/providers/ThemeProvider/types.ts +123 -0
  86. package/src/providers/UiProviders/index.tsx +65 -0
  87. package/src/providers/UiProviders/styles.ts +10 -0
  88. package/src/providers/hooks.ts +8 -0
  89. package/src/providers/index.ts +5 -0
  90. package/src/services/WebSocketService.ts +147 -0
  91. package/src/services/advancedRequests.ts +100 -0
  92. package/src/services/base.ts +31 -0
  93. package/src/services/hooks.ts +13 -0
  94. package/src/services/index.ts +2 -0
  95. package/src/styles/animations.scss +30 -0
  96. package/src/styles/index.scss +42 -0
  97. package/src/typings.d.ts +6 -0
  98. package/tsconfig.json +18 -0
@@ -0,0 +1,46 @@
1
+ import {
2
+ ChangeEvent,
3
+ FC,
4
+ InputHTMLAttributes,
5
+ useLayoutEffect,
6
+ useRef,
7
+ } from 'react';
8
+
9
+ import { join } from 'lodash';
10
+
11
+ type IndeterminateCheckboxProps = InputHTMLAttributes<HTMLInputElement> & {
12
+ className?: string;
13
+ checked: boolean | undefined;
14
+ onChange?: (e: ChangeEvent<HTMLInputElement>) => void;
15
+ };
16
+
17
+ export const IndeterminateCheckbox: FC<IndeterminateCheckboxProps> = ({
18
+ className,
19
+ checked,
20
+ onChange,
21
+ ...props
22
+ }) => {
23
+ const checkbox = useRef<HTMLInputElement>(null);
24
+
25
+ useLayoutEffect(() => {
26
+ if (checkbox.current)
27
+ checkbox.current.indeterminate = checked === undefined;
28
+ }, [checked]);
29
+
30
+ return (
31
+ <input
32
+ type="checkbox"
33
+ className={join(
34
+ [
35
+ 'h-4 w-4 rounded border-gray-300 text-sky-600 focus:ring-sky-600',
36
+ className,
37
+ ],
38
+ ' '
39
+ )}
40
+ ref={checkbox}
41
+ checked={checked}
42
+ onChange={onChange}
43
+ {...props}
44
+ />
45
+ );
46
+ };
@@ -0,0 +1,40 @@
1
+ import { SelectHTMLAttributes } from 'react';
2
+ import { inputStyle } from './styles';
3
+ import styled from 'styled-components';
4
+
5
+ type SelectProps<T> = {
6
+ items: T[];
7
+ itemKey: keyof T | ((item: T) => string);
8
+ itemLabel: keyof T | ((item: T) => string);
9
+ };
10
+
11
+ const StyledSelect = styled.select`
12
+ ${inputStyle}
13
+ `;
14
+
15
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
16
+ export const Select = <T extends { [k: string]: any }>({
17
+ items,
18
+ itemKey,
19
+ itemLabel,
20
+ ...props
21
+ }: SelectProps<T> & SelectHTMLAttributes<HTMLSelectElement>) => {
22
+ const keyGetter =
23
+ typeof itemKey === 'function' ? itemKey : (item: T) => item[itemKey];
24
+ const labelGetter =
25
+ typeof itemLabel === 'function' ? itemLabel : (item: T) => item[itemLabel];
26
+
27
+ return (
28
+ <StyledSelect {...props}>
29
+ {items.map((item) => {
30
+ const key = keyGetter(item);
31
+ const label = labelGetter(item);
32
+ return (
33
+ <option key={key} value={key}>
34
+ {label}
35
+ </option>
36
+ );
37
+ })}
38
+ </StyledSelect>
39
+ );
40
+ };
@@ -0,0 +1,5 @@
1
+ export * from './Button';
2
+ export * from './Select';
3
+ export * from './IconButton';
4
+ export * from './IndeterminateCheckbox';
5
+ export { Input } from './styles';
@@ -0,0 +1,20 @@
1
+ import styled, { css } from 'styled-components';
2
+
3
+ export const inputStyle = css`
4
+ font-family: inherit;
5
+ font-size: inherit;
6
+ color: inherit;
7
+ border: 1px solid var(--color-gray-300);
8
+ border-radius: var(--rounded-md);
9
+ padding: var(--space-1);
10
+ width: 100%;
11
+ box-sizing: border-box;
12
+
13
+ &:focus {
14
+ outline: 2px solid var(--color-primary-500);
15
+ }
16
+ `;
17
+
18
+ export const Input = styled.input`
19
+ ${inputStyle}
20
+ `;
@@ -0,0 +1,3 @@
1
+ export * from './data';
2
+ export * from './forms';
3
+ export * from './layout';
@@ -0,0 +1,79 @@
1
+ import * as styles from './styles';
2
+
3
+ import { CSSProperties, FC, PropsWithChildren, useMemo } from 'react';
4
+
5
+ import { usePortals } from '../../../providers';
6
+
7
+ type DropdownProps = PropsWithChildren<
8
+ styles.DropdownContainerProps & {
9
+ $sourceRect: DOMRect;
10
+ $autoPositionX?: boolean;
11
+ $autoPositionY?: boolean;
12
+ onClose?: () => void;
13
+ style?: CSSProperties;
14
+ }
15
+ >;
16
+
17
+ const getDropdownStyle = (dropdown: DropdownProps): CSSProperties => {
18
+ const minHeight = Array.isArray(dropdown.$height)
19
+ ? dropdown.$height[0]
20
+ : dropdown.$height;
21
+ const maxHeight = Array.isArray(dropdown.$height)
22
+ ? dropdown.$height[1]
23
+ : dropdown.$height;
24
+ const bottomSpace = document.body.scrollHeight - dropdown.$sourceRect.bottom;
25
+ const topSpace = dropdown.$sourceRect.top;
26
+ const isDropdownBelow =
27
+ !dropdown.$autoPositionY ||
28
+ bottomSpace > topSpace ||
29
+ bottomSpace >= minHeight;
30
+ const height = Math.max(
31
+ Math.min(maxHeight, isDropdownBelow ? bottomSpace : topSpace),
32
+ minHeight
33
+ );
34
+
35
+ const rightSpace = document.body.scrollWidth - dropdown.$sourceRect.right;
36
+ const leftSpace = dropdown.$sourceRect.left;
37
+ const isDropdownRight =
38
+ !dropdown.$autoPositionX ||
39
+ rightSpace > leftSpace ||
40
+ rightSpace >= dropdown.$width;
41
+
42
+ const dropdownStyle: CSSProperties = {
43
+ ...(isDropdownBelow
44
+ ? { top: topSpace + dropdown.$sourceRect.height }
45
+ : { bottom: bottomSpace + dropdown.$sourceRect.height }),
46
+ ...(isDropdownRight ? { left: leftSpace } : { right: rightSpace }),
47
+ width: dropdown.$width,
48
+ height,
49
+ minHeight,
50
+ maxHeight,
51
+ boxSizing: 'border-box',
52
+ ...dropdown.style,
53
+ };
54
+ return dropdownStyle;
55
+ };
56
+
57
+ export const Dropdown: FC<DropdownProps> = ({
58
+ children,
59
+ onClose,
60
+ ...props
61
+ }) => {
62
+ const { createPortal } = usePortals();
63
+
64
+ const style = useMemo(() => getDropdownStyle(props), [props]);
65
+
66
+ const modalPortal = useMemo(
67
+ () =>
68
+ createPortal(
69
+ <styles.DropdownBackdrop onClick={onClose}>
70
+ <styles.DropdownContainer {...props} style={style}>
71
+ {children}
72
+ </styles.DropdownContainer>
73
+ </styles.DropdownBackdrop>
74
+ ),
75
+ [children, createPortal, onClose, props, style]
76
+ );
77
+
78
+ return modalPortal;
79
+ };
@@ -0,0 +1,44 @@
1
+ import styled, { css } from 'styled-components';
2
+
3
+ export const DropdownBackdrop = styled.div.attrs({
4
+ className: 'DropdownBackdrop',
5
+ })`
6
+ position: absolute;
7
+ z-index: 1000;
8
+ top: 0;
9
+ left: 0;
10
+ right: 0;
11
+ bottom: 0;
12
+ background-color: rgba(0, 0, 0, 0.5);
13
+ display: flex;
14
+ justify-content: center;
15
+ align-items: center;
16
+ `;
17
+
18
+ export type DropdownContainerProps = {
19
+ $width: number;
20
+ $height: number | number[];
21
+ $zIndex?: number;
22
+ };
23
+
24
+ export const DropdownContainer = styled.div.attrs({
25
+ className: 'DropdownContainer',
26
+ })<DropdownContainerProps>`
27
+ background: var(--color-neutral-0);
28
+ border-radius: var(--rounded-md);
29
+ box-shadow: var(--shadow-lg);
30
+ outline: 1px solid var(--color-neutral-200);
31
+
32
+ position: absolute;
33
+ z-index: 1001;
34
+ ${({ $width }) => ($width ? `${$width}px` : 'auto')};
35
+ ${({ $height }) =>
36
+ Array.isArray($height)
37
+ ? css`
38
+ min-height: ${$height[0]}px;
39
+ max-height: ${$height[1]}px;
40
+ `
41
+ : css`
42
+ height: ${$height ? `${$height}px` : 'auto'};
43
+ `}
44
+ `;
@@ -0,0 +1,28 @@
1
+ import * as styles from './styles';
2
+
3
+ import { FC, PropsWithChildren, useMemo } from 'react';
4
+
5
+ import { LoadingIcon } from '../../../Icons';
6
+ import { usePortals } from '../../../providers';
7
+
8
+ type LoadingProps = PropsWithChildren<{
9
+ visible?: boolean;
10
+ }>;
11
+
12
+ export const Loading: FC<LoadingProps> = ({ visible, children }) => {
13
+ const { createPortal } = usePortals();
14
+ return useMemo(
15
+ () =>
16
+ visible
17
+ ? createPortal(
18
+ <styles.LoadingBackdrop>
19
+ <LoadingIcon />
20
+ <styles.LoadingContainer>
21
+ {children || 'Loading ...'}
22
+ </styles.LoadingContainer>
23
+ </styles.LoadingBackdrop>
24
+ )
25
+ : null,
26
+ [children, createPortal, visible]
27
+ );
28
+ };
@@ -0,0 +1,29 @@
1
+ import styled from 'styled-components';
2
+
3
+ export const LoadingContainer = styled.div.attrs({
4
+ className: 'LoadingContent',
5
+ })`
6
+ padding: var(--space-4);
7
+ `;
8
+
9
+ export const LoadingBackdrop = styled.div.attrs({
10
+ className: 'LoadingBackdrop',
11
+ })`
12
+ position: fixed;
13
+ z-index: 1100;
14
+ top: 0;
15
+ left: 0;
16
+ right: 0;
17
+ bottom: 0;
18
+ background-color: rgba(0, 0, 0, 0.5);
19
+ display: flex;
20
+ flex-direction: column;
21
+ justify-content: center;
22
+ align-items: center;
23
+ color: var(--color-gray-50);
24
+ fill: var(--color-gray-50);
25
+ & > svg {
26
+ width: var(--space-8);
27
+ height: var(--space-8);
28
+ }
29
+ `;
@@ -0,0 +1,51 @@
1
+ import * as styles from './styles';
2
+
3
+ import { FC, PropsWithChildren, useMemo } from 'react';
4
+
5
+ import { usePortals } from '../../../providers';
6
+
7
+ type ModalProps = PropsWithChildren<
8
+ styles.ModalContainerProps & {
9
+ closeOnBackdropClick?: boolean;
10
+ onClose?: () => void;
11
+ }
12
+ >;
13
+
14
+ type ModalFC = FC<ModalProps> & {
15
+ Header: typeof styles.ModalHeader;
16
+ Message: typeof styles.ModalMessage;
17
+ Content: typeof styles.ModalContent;
18
+ ContentWithIcon: typeof styles.ModalContentWithIcon;
19
+ Footer: typeof styles.ModalFooter;
20
+ Buttons: typeof styles.ModalButtons;
21
+ };
22
+
23
+ export const Modal: ModalFC = ({
24
+ children,
25
+ closeOnBackdropClick,
26
+ onClose,
27
+ ...props
28
+ }) => {
29
+ const { createPortal } = usePortals();
30
+
31
+ const modalPortal = useMemo(
32
+ () =>
33
+ createPortal(
34
+ <styles.ModalBackdrop
35
+ onClick={closeOnBackdropClick ? onClose : undefined}
36
+ >
37
+ <styles.ModalContainer {...props}>{children}</styles.ModalContainer>
38
+ </styles.ModalBackdrop>
39
+ ),
40
+ [children, closeOnBackdropClick, createPortal, onClose, props]
41
+ );
42
+
43
+ return modalPortal;
44
+ };
45
+
46
+ Modal.Header = styles.ModalHeader;
47
+ Modal.Message = styles.ModalMessage;
48
+ Modal.Content = styles.ModalContent;
49
+ Modal.ContentWithIcon = styles.ModalContentWithIcon;
50
+ Modal.Footer = styles.ModalFooter;
51
+ Modal.Buttons = styles.ModalButtons;
@@ -0,0 +1,110 @@
1
+ import styled, { css } from 'styled-components';
2
+
3
+ export const ModalBackdrop = styled.div.attrs({
4
+ className: 'ModalBackdrop',
5
+ })`
6
+ position: fixed;
7
+ z-index: 1000;
8
+ top: 0;
9
+ left: 0;
10
+ right: 0;
11
+ bottom: 0;
12
+ background-color: rgba(0, 0, 0, 0.5);
13
+ display: flex;
14
+ justify-content: center;
15
+ align-items: center;
16
+ `;
17
+
18
+ export type ModalContainerProps = {
19
+ $width?: number;
20
+ $height?: number;
21
+ $zIndex?: number;
22
+ $fullscreen?: boolean;
23
+ };
24
+
25
+ export const ModalContainer = styled.div.attrs({
26
+ className: 'ModalContainer',
27
+ })<ModalContainerProps>`
28
+ background: var(--color-neutral-100);
29
+ border-radius: var(--rounded-lg);
30
+ box-shadow: var(--shadow-lg);
31
+ display: flex;
32
+ flex-direction: column;
33
+ max-height: 100vh;
34
+ overflow: hidden;
35
+
36
+ position: fixed;
37
+ z-index: 1001;
38
+ ${({ $fullscreen, $width, $height }) =>
39
+ $fullscreen
40
+ ? css`
41
+ top: 0;
42
+ left: 0;
43
+ right: 0;
44
+ bottom: 0;
45
+ margin: var(--space-4);
46
+ `
47
+ : css`
48
+ top: 50%;
49
+ left: 50%;
50
+ transform: translate(-50%, -50%);
51
+ width: ${$width ? `${$width}px` : 'auto'};
52
+ height: ${$height ? `${$height}px` : 'auto'};
53
+ `}
54
+ `;
55
+
56
+ export const ModalHeader = styled.div.attrs({ className: 'ModalHeader' })`
57
+ font-weight: bold;
58
+ font-size: var(--text-lg);
59
+ padding: var(--space-2);
60
+ background-color: var(--color-neutral-200);
61
+ padding: var(--space-4);
62
+ `;
63
+
64
+ export const ModalMessage = styled.p.attrs({ className: 'ModalMessage' })`
65
+ margin-bottom: 1rem;
66
+ `;
67
+
68
+ export const ModalContent = styled.div.attrs({ className: 'ModalContent' })`
69
+ display: flex;
70
+ flex-direction: column;
71
+ flex: 1;
72
+ overflow: auto;
73
+ padding: var(--space-4);
74
+ `;
75
+
76
+ export const ModalContentWithIcon = styled.div.attrs({
77
+ className: 'ModalContentWithIcon',
78
+ })`
79
+ display: flex;
80
+ flex-direction: row;
81
+ padding: var(--space-4);
82
+ gap: var(--space-2);
83
+
84
+ & > svg {
85
+ margin: var(--space-2);
86
+ width: var(--space-8);
87
+ height: var(--space-8);
88
+ }
89
+ `;
90
+
91
+ export const ModalFooter = styled.div.attrs({ className: 'ModalFooter' })`
92
+ display: flex;
93
+ flex-direction: row;
94
+ justify-content: space-between;
95
+ padding: var(--space-2);
96
+ background-color: var(--color-neutral-200);
97
+ padding: var(--space-4);
98
+ `;
99
+
100
+ export const ModalButtons = styled(ModalFooter).attrs({
101
+ className: 'ModalButtons',
102
+ })`
103
+ display: flex;
104
+ flex-direction: row;
105
+ gap: var(--space-2);
106
+
107
+ button.right:first-of-type {
108
+ margin-left: auto;
109
+ }
110
+ `;
@@ -0,0 +1,3 @@
1
+ export * from './Dropdown';
2
+ export * from './Modal';
3
+ export * from './Loading';
@@ -0,0 +1,14 @@
1
+ export type EnvType = 'dev' | 'test' | 'staging' | 'prod';
2
+
3
+ export type Config = {
4
+ webSocketUrl: string;
5
+ httpServerUrl: `${string}/`;
6
+ };
7
+
8
+ type DeepPartial<T> = T extends object
9
+ ? {
10
+ [P in keyof T]?: DeepPartial<T[P]>;
11
+ }
12
+ : T;
13
+
14
+ export type PartialConfig = DeepPartial<Config>;
@@ -0,0 +1,14 @@
1
+ let scrollBarSize: number[] | undefined = undefined;
2
+
3
+ export const getScrollBarSize = () => {
4
+ if (!scrollBarSize) {
5
+ const el = document.createElement('div');
6
+ el.style.cssText = 'overflow:scroll; visibility:hidden; position:absolute;';
7
+ document.body.appendChild(el);
8
+ const width = el.offsetWidth - el.clientWidth;
9
+ const height = el.offsetHeight - el.clientHeight;
10
+ el.remove();
11
+ scrollBarSize = [width, height];
12
+ }
13
+ return scrollBarSize;
14
+ };
@@ -0,0 +1,20 @@
1
+ export const formatMoney = (number: number, decimals = 2, currency = 'EUR') =>
2
+ new Intl.NumberFormat(navigator.language, {
3
+ style: 'currency',
4
+ currency,
5
+ minimumFractionDigits: decimals,
6
+ maximumFractionDigits: decimals,
7
+ }).format(number);
8
+
9
+ export const formatPercentage = (number: number, decimals = 2) =>
10
+ new Intl.NumberFormat(navigator.language, {
11
+ style: 'percent',
12
+ minimumFractionDigits: decimals,
13
+ maximumFractionDigits: decimals,
14
+ }).format(number);
15
+
16
+ export const formatNumber = (number: number, decimals = 2) =>
17
+ new Intl.NumberFormat(navigator.language, {
18
+ minimumFractionDigits: decimals,
19
+ maximumFractionDigits: decimals,
20
+ }).format(number);
@@ -0,0 +1,2 @@
1
+ export * from './useElementSize';
2
+ export * from './useWindowSize';
@@ -0,0 +1,24 @@
1
+ import { useEffect, useState } from 'react';
2
+
3
+ export const useElementSize = (element: HTMLElement | null) => {
4
+ const [size, setSize] = useState({ width: 0, height: 0 });
5
+
6
+ useEffect(() => {
7
+ if (!element) {
8
+ return;
9
+ }
10
+
11
+ const observer = new ResizeObserver((entries) => {
12
+ const entry = entries[0];
13
+ const { width, height } = entry.contentRect;
14
+ setSize({ width, height });
15
+ });
16
+
17
+ observer.observe(element);
18
+ return () => {
19
+ observer.disconnect();
20
+ };
21
+ }, [element]);
22
+
23
+ return size;
24
+ };
@@ -0,0 +1,20 @@
1
+ import { useEffect, useState } from 'react';
2
+
3
+ export const useWindowSize = () => {
4
+ const [size, setSize] = useState({ width: 0, height: 0 });
5
+
6
+ useEffect(() => {
7
+ const handleResize = () => {
8
+ setSize({ width: window.outerWidth, height: window.outerHeight });
9
+ };
10
+
11
+ window.addEventListener('resize', handleResize);
12
+ handleResize();
13
+
14
+ return () => {
15
+ window.removeEventListener('resize', handleResize);
16
+ };
17
+ }, []);
18
+
19
+ return size;
20
+ };
package/src/index.ts ADDED
@@ -0,0 +1,7 @@
1
+ export * from './components';
2
+ export * from './providers';
3
+
4
+ export * from './Icons';
5
+
6
+ export * from './config';
7
+ export * from './services';
@@ -0,0 +1,54 @@
1
+ import * as styles from './styles';
2
+
3
+ import {
4
+ FC,
5
+ PropsWithChildren,
6
+ ReactNode,
7
+ ReactPortal,
8
+ createContext,
9
+ useCallback,
10
+ useMemo,
11
+ useRef,
12
+ } from 'react';
13
+
14
+ import { createPortal } from 'react-dom';
15
+
16
+ export type PortalProps = {
17
+ key: string;
18
+ children: ReactNode;
19
+ };
20
+
21
+ export type PortalsContextType = {
22
+ createPortal: (children: ReactNode) => ReactPortal | undefined;
23
+ };
24
+
25
+ export const PortalsContext = createContext<PortalsContextType>({
26
+ createPortal: () => undefined,
27
+ });
28
+
29
+ export const PortalsProvider: FC<PropsWithChildren> = ({ children }) => {
30
+ const containerRef = useRef<HTMLDivElement | null>(null);
31
+
32
+ const createLocalPortal = useCallback(
33
+ (children: ReactNode) =>
34
+ createPortal(
35
+ <styles.PortalContainer>{children}</styles.PortalContainer>,
36
+ containerRef.current!
37
+ ),
38
+ []
39
+ );
40
+
41
+ const value = useMemo(
42
+ () => ({
43
+ createPortal: createLocalPortal,
44
+ }),
45
+ [createLocalPortal]
46
+ );
47
+
48
+ return (
49
+ <PortalsContext.Provider value={value}>
50
+ {children}
51
+ <styles.PortalsWrapper ref={containerRef} />
52
+ </PortalsContext.Provider>
53
+ );
54
+ };
@@ -0,0 +1,27 @@
1
+ import styled from 'styled-components';
2
+
3
+ export const BackdropContainer = styled.div.attrs({
4
+ className: 'BackdropContainer',
5
+ })`
6
+ position: fixed;
7
+ top: 0;
8
+ left: 0;
9
+ right: 0;
10
+ bottom: 0;
11
+ z-index: 1000;
12
+ background-color: rgba(0, 0, 0, 0.5);
13
+ display: flex;
14
+ justify-content: center;
15
+ align-items: center;
16
+ `;
17
+
18
+ export const PortalsWrapper = styled.div.attrs({
19
+ className: 'PortalsWrapper',
20
+ })`
21
+ width: 0;
22
+ height: 0;
23
+ `;
24
+
25
+ export const PortalContainer = styled.div.attrs({
26
+ className: 'PortalContainer',
27
+ })``;