@stack-spot/portal-components 2.9.0 → 2.10.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.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,12 @@
1
1
  # Changelog
2
2
 
3
+ ## [2.10.0](https://github.com/stack-spot/portal-commons/compare/portal-components@v2.9.0...portal-components@v2.10.0) (2024-11-21)
4
+
5
+
6
+ ### Features
7
+
8
+ * FileTree View component ([#496](https://github.com/stack-spot/portal-commons/issues/496)) ([d66e12d](https://github.com/stack-spot/portal-commons/commit/d66e12dd34e8314f7fb49e3a9cf7b8efc3e93fb2))
9
+
3
10
  ## [2.9.0](https://github.com/stack-spot/portal-commons/compare/portal-components@v2.8.1...portal-components@v2.9.0) (2024-10-30)
4
11
 
5
12
 
@@ -0,0 +1,23 @@
1
+ import * as icons from '@citric/icons';
2
+ export type IconName = keyof typeof icons;
3
+ export type MenuOption = {
4
+ /**
5
+ * The text that will be displayed for the option.
6
+ */
7
+ label: string;
8
+ /**
9
+ * The icon that will be displayed alongside the text.
10
+ */
11
+ icon: IconName;
12
+ /**
13
+ * The function that will be executed when the option is clicked.
14
+ */
15
+ onClick: VoidFunction;
16
+ };
17
+ interface Props {
18
+ loading?: boolean;
19
+ menu: MenuOption[];
20
+ }
21
+ export declare function More({ loading, menu }: Props): import("react/jsx-runtime").JSX.Element;
22
+ export {};
23
+ //# sourceMappingURL=More.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"More.d.ts","sourceRoot":"","sources":["../../../src/components/FileTreeView/More.tsx"],"names":[],"mappings":"AAEA,OAAO,KAAK,KAAK,MAAM,eAAe,CAAA;AAMtC,MAAM,MAAM,QAAQ,GAAG,MAAM,OAAO,KAAK,CAAA;AAEzC,MAAM,MAAM,UAAU,GAAG;IACzB;;OAEG;IACH,KAAK,EAAE,MAAM,CAAC;IACd;;OAEG;IACH,IAAI,EAAE,QAAQ,CAAC;IACf;;OAEG;IACH,OAAO,EAAE,YAAY,CAAC;CACrB,CAAA;AAED,UAAU,KAAK;IACb,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,IAAI,EAAE,UAAU,EAAE,CAAC;CACpB;AA8CD,wBAAgB,IAAI,CAAC,EAAE,OAAO,EAAE,IAAI,EAAE,EAAE,KAAK,2CAuC5C"}
@@ -0,0 +1,55 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { Box, IconBox, Text } from '@citric/core';
3
+ import * as icons from '@citric/icons';
4
+ import { EllipsisVertical } from '@citric/icons';
5
+ import { IconButton } from '@citric/ui';
6
+ import { theme } from '@stack-spot/portal-theme';
7
+ import { createElement, useState } from 'react';
8
+ const styles = {
9
+ MoreWrapper: {
10
+ position: 'relative',
11
+ 'button': {
12
+ width: '2rem',
13
+ height: '2rem',
14
+ },
15
+ '.drop': {
16
+ zIndex: 1,
17
+ position: 'absolute',
18
+ listStyle: 'none',
19
+ display: 'flex',
20
+ margin: '0',
21
+ padding: '0.25rem',
22
+ flexDirection: 'column',
23
+ justifyContent: 'center',
24
+ alignItems: 'flex-start',
25
+ gap: '0.5rem',
26
+ borderRadius: '0.25rem',
27
+ border: `0.063rem solid ${theme.color.light[400]}`,
28
+ backgroundColor: '${theme.color.light[500]}',
29
+ boxShadow: '0 0 0 0.125rem ${theme.color.gray[600]}',
30
+ right: '2px',
31
+ width: 'max-content',
32
+ 'ul': {
33
+ padding: 0,
34
+ margin: 0,
35
+ },
36
+ '.item-list-float': {
37
+ minWidth: '230px',
38
+ listStyleType: 'none',
39
+ padding: '8px',
40
+ paddingRight: '12px',
41
+ display: 'flex',
42
+ alignItems: 'center',
43
+ cursor: 'pointer',
44
+ '&:hover': {
45
+ backgroundColor: `${theme.color.light[600]}`,
46
+ },
47
+ },
48
+ },
49
+ },
50
+ };
51
+ export function More({ loading, menu }) {
52
+ const [opened, setOpened] = useState(false);
53
+ return (_jsxs(Box, { sx: styles.MoreWrapper, children: [_jsx(IconButton, { color: "light", size: "sm", appearance: "circle", "aria-label": "More options", disabled: loading, onClick: () => setOpened((old) => !old), onBlur: () => setTimeout(() => setOpened(false), 200), children: _jsx(IconBox, { size: "xs", children: _jsx(EllipsisVertical, {}) }) }), opened && (_jsx("div", { className: "drop", children: _jsx("ul", { className: "drop-wrapper", children: menu.map((menuItem, key) => (_jsxs("li", { role: "button", className: "item-list-float", onClick: menuItem.onClick, children: [_jsx(IconBox, { size: "sm", children: createElement(icons[menuItem.icon]) }), _jsx(Text, { appearance: "body2", children: menuItem.label })] }, key))) }) }))] }));
54
+ }
55
+ //# sourceMappingURL=More.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"More.js","sourceRoot":"","sources":["../../../src/components/FileTreeView/More.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,32 @@
1
+ import { MenuOption } from './More.js';
2
+ export interface FileDir {
3
+ /**
4
+ * The name of the directory or file.
5
+ * */
6
+ name: string;
7
+ /**
8
+ * dirs - An array containing the directories present in the file tree.
9
+ */
10
+ dirs: FileDir[];
11
+ /**
12
+ * isFile - A flag indicating whether the item is a file (`true`) or a directory (`false`).
13
+ */
14
+ isFile?: boolean;
15
+ /**
16
+ * itemLink - The URL or path link associated with the item in the file tree.
17
+ */
18
+ itemLink: string;
19
+ /**
20
+ * The menu options that will be rendered at the end of the line for this item.
21
+ */
22
+ menuOptions?: MenuOption[];
23
+ }
24
+ interface Props {
25
+ /**
26
+ * Represents the hierarchical structure of directories and files, providing the component with an organized view of the file system.
27
+ */
28
+ fileTree: FileDir;
29
+ }
30
+ export declare const FileTreeView: ({ fileTree }: Props) => import("react/jsx-runtime").JSX.Element;
31
+ export {};
32
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/components/FileTreeView/index.tsx"],"names":[],"mappings":"AAOA,OAAO,EAAE,UAAU,EAAQ,MAAM,QAAQ,CAAA;AAiEzC,MAAM,WAAW,OAAO;IACtB;;SAEK;IACL,IAAI,EAAE,MAAM,CAAC;IACb;;OAEG;IACH,IAAI,EAAE,OAAO,EAAE,CAAC;IAChB;;OAEG;IACH,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB;;OAEG;IACH,QAAQ,EAAE,MAAM,CAAC;IACjB;;OAEG;IACH,WAAW,CAAC,EAAE,UAAU,EAAE,CAAC;CAC5B;AAED,UAAU,KAAK;IACb;;OAEG;IACH,QAAQ,EAAE,OAAO,CAAC;CACnB;AAED,eAAO,MAAM,YAAY,iBAAkB,KAAK,4CAwE/C,CAAA"}
@@ -0,0 +1,80 @@
1
+ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
2
+ import { Box, IconBox, Text } from '@citric/core';
3
+ import { ChevronDown, Code, Folder } from '@citric/icons';
4
+ import { theme } from '@stack-spot/portal-theme';
5
+ import { useState } from 'react';
6
+ import { useAnchorTag } from '../../context/anchor.js';
7
+ import { More } from './More.js';
8
+ const styles = {
9
+ FileTreeViewWrapper: {
10
+ height: '100%',
11
+ display: 'flex',
12
+ flexDirection: 'column',
13
+ margin: '16px',
14
+ },
15
+ AccordionItemWrapper: {
16
+ width: '100%',
17
+ 'ul': {
18
+ listStyleType: 'none',
19
+ margin: 0,
20
+ padding: 0,
21
+ flexDirection: 'column',
22
+ marginLeft: '20px',
23
+ justifyContent: 'center',
24
+ },
25
+ 'li': {
26
+ display: 'flex',
27
+ listStyleType: 'none',
28
+ margin: '2px 0',
29
+ padding: 0,
30
+ flexDirection: 'row',
31
+ alignItems: 'center',
32
+ gap: '4px',
33
+ },
34
+ '.accordion-node-group': {
35
+ justifyContent: 'space-between',
36
+ width: '100%',
37
+ '&:hover': {
38
+ backgroundColor: `${theme.color.light[500]}`,
39
+ borderRadius: '4px',
40
+ },
41
+ 'a': {
42
+ display: 'flex',
43
+ alignItems: 'center',
44
+ textDecoration: 'underline',
45
+ },
46
+ },
47
+ '.accordion-node': {
48
+ cursor: 'pointer',
49
+ flexDirection: 'row',
50
+ display: 'flex',
51
+ gap: '4px',
52
+ alignItems: 'center',
53
+ background: 'transparent',
54
+ color: 'inherit',
55
+ border: 'none',
56
+ },
57
+ 'actions': {
58
+ display: 'flex',
59
+ flexDirection: 'row',
60
+ gap: '4px',
61
+ alignItems: 'center',
62
+ },
63
+ },
64
+ };
65
+ export const FileTreeView = ({ fileTree }) => {
66
+ const AccordionItem = ({ name, isFile, dirs, defaultOpened = false, menuOptions, itemLink }) => {
67
+ const Link = useAnchorTag();
68
+ const [opened, setOpened] = useState(defaultOpened);
69
+ const onExpand = () => {
70
+ setOpened(old => !old);
71
+ };
72
+ const FolderItem = () => (_jsxs("li", { className: "accordion-node-group", children: [_jsxs("button", { className: "accordion-node", onClick: onExpand, children: [_jsx(IconBox, { className: "icon", children: _jsx(ChevronDown, {}) }), _jsx(IconBox, { children: _jsx(Folder, {}) }), _jsx(Text, { children: name !== '' ? name : '/' })] }), menuOptions &&
73
+ _jsx("div", { className: "actions", children: _jsx(More, { menu: menuOptions }) })] }));
74
+ const FileItem = () => (_jsxs("li", { className: "accordion-node-group", children: [_jsxs(Link, { href: itemLink, children: [_jsx(IconBox, { children: _jsx(Code, {}) }), _jsx(Text, { as: "label", children: name })] }), menuOptions &&
75
+ _jsx("div", { className: "actions", children: _jsx(More, { menu: menuOptions }) })] }));
76
+ return (_jsxs(Box, { sx: styles.AccordionItemWrapper, children: [isFile ? _jsx(FileItem, {}) : _jsx(FolderItem, {}), _jsx("ul", { className: "accordion-children", children: opened && !isFile && dirs && dirs.length > 0 && (_jsx(_Fragment, { children: dirs?.map((subDir, index) => (_jsx("li", { children: _jsx(AccordionItem, { ...subDir }) }, index))) })) })] }));
77
+ };
78
+ return (_jsx(Box, { sx: styles.FileTreeViewWrapper, children: _jsx(AccordionItem, { ...fileTree, defaultOpened: true }) }));
79
+ };
80
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/components/FileTreeView/index.tsx"],"names":[],"mappings":";AACA,OAAO,EAAE,GAAG,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,cAAc,CAAA;AAEjD,OAAO,EAAE,WAAW,EAAE,IAAI,EAAE,MAAM,EAAE,MAAM,eAAe,CAAA;AACzD,OAAO,EAAE,KAAK,EAAE,MAAM,0BAA0B,CAAA;AAChD,OAAO,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAA;AAChC,OAAO,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAA;AACnD,OAAO,EAAc,IAAI,EAAE,MAAM,QAAQ,CAAA;AAEzC,MAAM,MAAM,GAAG;IACb,mBAAmB,EAAE;QACnB,MAAM,EAAE,MAAM;QACd,OAAO,EAAE,MAAM;QACf,aAAa,EAAE,QAAQ;QACvB,MAAM,EAAE,MAAM;KACf;IACD,oBAAoB,EAAE;QACpB,KAAK,EAAE,MAAM;QACb,IAAI,EAAE;YACJ,aAAa,EAAE,MAAM;YACrB,MAAM,EAAE,CAAC;YACT,OAAO,EAAE,CAAC;YACV,aAAa,EAAE,QAAQ;YACvB,UAAU,EAAE,MAAM;YAClB,cAAc,EAAE,QAAQ;SACzB;QACD,IAAI,EAAE;YACJ,OAAO,EAAE,MAAM;YACf,aAAa,EAAE,MAAM;YACrB,MAAM,EAAE,OAAO;YACf,OAAO,EAAE,CAAC;YACV,aAAa,EAAE,KAAK;YACpB,UAAU,EAAE,QAAQ;YACpB,GAAG,EAAE,KAAK;SACX;QAED,uBAAuB,EAAE;YACvB,cAAc,EAAE,eAAe;YAC/B,KAAK,EAAE,MAAM;YACb,SAAS,EAAE;gBACT,eAAe,EAAE,GAAG,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE;gBAC5C,YAAY,EAAE,KAAK;aACpB;YAED,GAAG,EAAE;gBACH,OAAO,EAAE,MAAM;gBACf,UAAU,EAAE,QAAQ;gBACpB,cAAc,EAAE,WAAW;aAC5B;SACF;QAED,iBAAiB,EAAE;YACjB,MAAM,EAAE,SAAS;YACjB,aAAa,EAAE,KAAK;YACpB,OAAO,EAAE,MAAM;YACf,GAAG,EAAE,KAAK;YACV,UAAU,EAAE,QAAQ;YACpB,UAAU,EAAE,aAAa;YACzB,KAAK,EAAE,SAAS;YAChB,MAAM,EAAE,MAAM;SACf;QAED,SAAS,EAAE;YACT,OAAO,EAAE,MAAM;YACf,aAAa,EAAE,KAAK;YACpB,GAAG,EAAE,KAAK;YACV,UAAU,EAAE,QAAQ;SACrB;KACF;CACF,CAAA;AAiCD,MAAM,CAAC,MAAM,YAAY,GAAG,CAAC,EAAE,QAAQ,EAAS,EAAE,EAAE;IAElD,MAAM,aAAa,GAAG,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,aAAa,GAAG,KAAK,EAAE,WAAW,EAAE,QAAQ,EAC1D,EAAE,EAAE;QAEjC,MAAM,IAAI,GAAG,YAAY,EAAE,CAAA;QAC3B,MAAM,CAAC,MAAM,EAAE,SAAS,CAAC,GAAG,QAAQ,CAAU,aAAa,CAAC,CAAA;QAE5D,MAAM,QAAQ,GAAG,GAAG,EAAE;YACpB,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC,CAAA;QACxB,CAAC,CAAA;QAED,MAAM,UAAU,GAAG,GAAG,EAAE,CAAC,CACvB,cAAI,SAAS,EAAC,sBAAsB,aAClC,kBAAQ,SAAS,EAAC,gBAAgB,EAAC,OAAO,EAAE,QAAQ,aAClD,KAAC,OAAO,IAAC,SAAS,EAAC,MAAM,YACvB,KAAC,WAAW,KAAG,GACP,EACV,KAAC,OAAO,cACN,KAAC,MAAM,KAAG,GACF,EACV,KAAC,IAAI,cAAE,IAAI,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,GAAG,GAAQ,IAChC,EACP,WAAW;oBACX,cAAK,SAAS,EAAC,SAAS,YACtB,KAAC,IAAI,IAAC,IAAI,EAAE,WAAW,GAAI,GACvB,IAEL,CACN,CAAA;QAED,MAAM,QAAQ,GAAG,GAAG,EAAE,CAAC,CACrB,cAAI,SAAS,EAAC,sBAAsB,aAClC,MAAC,IAAI,IAAC,IAAI,EAAE,QAAQ,aAClB,KAAC,OAAO,cACN,KAAC,IAAI,KAAG,GACA,EACV,KAAC,IAAI,IAAC,EAAE,EAAC,OAAO,YAAE,IAAI,GAAQ,IACzB,EACL,WAAW;oBACX,cAAK,SAAS,EAAC,SAAS,YACtB,KAAC,IAAI,IAAC,IAAI,EAAE,WAAW,GAAI,GACvB,IAEJ,CACP,CAAA;QAED,OAAO,CACL,MAAC,GAAG,IAAC,EAAE,EAAE,MAAM,CAAC,oBAAoB,aAEhC,MAAM,CAAC,CAAC,CAAC,KAAC,QAAQ,KAAG,CAAC,CAAC,CAAC,KAAC,UAAU,KAAG,EAExC,aAAI,SAAS,EAAC,oBAAoB,YAC/B,MAAM,IAAI,CAAC,MAAM,IAAI,IAAI,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,IAAI,CAC/C,4BACG,IAAI,EAAE,GAAG,CAAC,CAAC,MAAM,EAAE,KAAK,EAAE,EAAE,CAAC,CAC5B,uBACE,KAAC,aAAa,OAAK,MAAM,GAAI,IADtB,KAAK,CAER,CACP,CAAC,GACD,CACJ,GACE,IACD,CACP,CAAA;IACH,CAAC,CAAA;IAED,OAAO,CACL,KAAC,GAAG,IAAC,EAAE,EAAE,MAAM,CAAC,mBAAmC,YACjD,KAAC,aAAa,OAAK,QAAQ,EAAE,aAAa,EAAE,IAAI,GAAI,GAChD,CACP,CAAA;AACH,CAAC,CAAA"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@stack-spot/portal-components",
3
- "version": "2.9.0",
3
+ "version": "2.10.0",
4
4
  "type": "module",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
@@ -22,7 +22,8 @@
22
22
  "./AnimatedHeight": "./dist/components/AnimatedHeight.js",
23
23
  "./SearchInput": "./dist/components/form/SearchInput.js",
24
24
  "./Select": "./dist/components/form/Select/index.js",
25
- "./Stepper": "./dist/components/Stepper/index.js"
25
+ "./Stepper": "./dist/components/Stepper/index.js",
26
+ "./FileTreeView": "./dist/components/FileTreeView/index.js"
26
27
  },
27
28
  "scripts": {
28
29
  "build": "rimraf dist && tsc && tsc-esm-fix --target='dist'",
@@ -0,0 +1,114 @@
1
+ import { Box, IconBox, Text } from '@citric/core'
2
+ import { SxProperties } from '@citric/core/dist/sx'
3
+ import * as icons from '@citric/icons'
4
+ import { EllipsisVertical } from '@citric/icons'
5
+ import { IconButton } from '@citric/ui'
6
+ import { theme } from '@stack-spot/portal-theme'
7
+ import { createElement, useState } from 'react'
8
+
9
+ export type IconName = keyof typeof icons
10
+
11
+ export type MenuOption = {
12
+ /**
13
+ * The text that will be displayed for the option.
14
+ */
15
+ label: string,
16
+ /**
17
+ * The icon that will be displayed alongside the text.
18
+ */
19
+ icon: IconName,
20
+ /**
21
+ * The function that will be executed when the option is clicked.
22
+ */
23
+ onClick: VoidFunction,
24
+ }
25
+
26
+ interface Props {
27
+ loading?: boolean,
28
+ menu: MenuOption[],
29
+ }
30
+
31
+ const styles = {
32
+ MoreWrapper: {
33
+ position: 'relative',
34
+ 'button': {
35
+ width: '2rem',
36
+ height: '2rem',
37
+ },
38
+ '.drop': {
39
+ zIndex: 1,
40
+ position: 'absolute',
41
+ listStyle: 'none',
42
+ display: 'flex',
43
+ margin: '0',
44
+ padding: '0.25rem',
45
+ flexDirection: 'column',
46
+ justifyContent: 'center',
47
+ alignItems: 'flex-start',
48
+ gap: '0.5rem',
49
+ borderRadius: '0.25rem',
50
+ border: `0.063rem solid ${theme.color.light[400]}`,
51
+ backgroundColor: '${theme.color.light[500]}',
52
+ boxShadow: '0 0 0 0.125rem ${theme.color.gray[600]}',
53
+ right: '2px',
54
+ width: 'max-content',
55
+ 'ul': {
56
+ padding: 0,
57
+ margin: 0,
58
+ },
59
+ '.item-list-float': {
60
+ minWidth: '230px',
61
+ listStyleType: 'none',
62
+ padding: '8px',
63
+ paddingRight: '12px',
64
+ display: 'flex',
65
+ alignItems: 'center',
66
+ cursor: 'pointer',
67
+ '&:hover': {
68
+ backgroundColor: `${theme.color.light[600]}`,
69
+ },
70
+ },
71
+ },
72
+ },
73
+ }
74
+
75
+ export function More({ loading, menu }: Props) {
76
+
77
+ const [opened, setOpened] = useState(false)
78
+
79
+ return (
80
+ <Box sx={styles.MoreWrapper as SxProperties}>
81
+ <IconButton
82
+ color="light"
83
+ size="sm"
84
+ appearance="circle"
85
+ aria-label="More options"
86
+ disabled={loading}
87
+ onClick={() => setOpened((old) => !old)}
88
+ onBlur={() => setTimeout(() => setOpened(false), 200)}
89
+ >
90
+ <IconBox size="xs">
91
+ <EllipsisVertical />
92
+ </IconBox>
93
+ </IconButton>
94
+ {
95
+ opened && (
96
+ <div className="drop">
97
+ <ul className="drop-wrapper">
98
+ {
99
+ menu.map((menuItem, key) => (
100
+ <li role="button" className="item-list-float" key={key} onClick={menuItem.onClick}>
101
+ <IconBox size="sm">
102
+ {createElement(icons[menuItem.icon])}
103
+ </IconBox>
104
+ <Text appearance="body2">{menuItem.label}</Text>
105
+ </li>
106
+ ))
107
+ }
108
+ </ul>
109
+ </div>
110
+ )
111
+ }
112
+ </Box>
113
+ )
114
+ }
@@ -0,0 +1,175 @@
1
+
2
+ import { Box, IconBox, Text } from '@citric/core'
3
+ import { SxProperties } from '@citric/core/dist/sx'
4
+ import { ChevronDown, Code, Folder } from '@citric/icons'
5
+ import { theme } from '@stack-spot/portal-theme'
6
+ import { useState } from 'react'
7
+ import { useAnchorTag } from '../../context/anchor'
8
+ import { MenuOption, More } from './More'
9
+
10
+ const styles = {
11
+ FileTreeViewWrapper: {
12
+ height: '100%',
13
+ display: 'flex',
14
+ flexDirection: 'column',
15
+ margin: '16px',
16
+ },
17
+ AccordionItemWrapper: {
18
+ width: '100%',
19
+ 'ul': {
20
+ listStyleType: 'none',
21
+ margin: 0,
22
+ padding: 0,
23
+ flexDirection: 'column',
24
+ marginLeft: '20px',
25
+ justifyContent: 'center',
26
+ },
27
+ 'li': {
28
+ display: 'flex',
29
+ listStyleType: 'none',
30
+ margin: '2px 0',
31
+ padding: 0,
32
+ flexDirection: 'row',
33
+ alignItems: 'center',
34
+ gap: '4px',
35
+ },
36
+
37
+ '.accordion-node-group': {
38
+ justifyContent: 'space-between',
39
+ width: '100%',
40
+ '&:hover': {
41
+ backgroundColor: `${theme.color.light[500]}`,
42
+ borderRadius: '4px',
43
+ },
44
+
45
+ 'a': {
46
+ display: 'flex',
47
+ alignItems: 'center',
48
+ textDecoration: 'underline',
49
+ },
50
+ },
51
+
52
+ '.accordion-node': {
53
+ cursor: 'pointer',
54
+ flexDirection: 'row',
55
+ display: 'flex',
56
+ gap: '4px',
57
+ alignItems: 'center',
58
+ background: 'transparent',
59
+ color: 'inherit',
60
+ border: 'none',
61
+ },
62
+
63
+ 'actions': {
64
+ display: 'flex',
65
+ flexDirection: 'row',
66
+ gap: '4px',
67
+ alignItems: 'center',
68
+ },
69
+ },
70
+ }
71
+
72
+
73
+ export interface FileDir {
74
+ /**
75
+ * The name of the directory or file.
76
+ * */
77
+ name: string,
78
+ /**
79
+ * dirs - An array containing the directories present in the file tree.
80
+ */
81
+ dirs: FileDir[],
82
+ /**
83
+ * isFile - A flag indicating whether the item is a file (`true`) or a directory (`false`).
84
+ */
85
+ isFile?: boolean,
86
+ /**
87
+ * itemLink - The URL or path link associated with the item in the file tree.
88
+ */
89
+ itemLink: string,
90
+ /**
91
+ * The menu options that will be rendered at the end of the line for this item.
92
+ */
93
+ menuOptions?: MenuOption[],
94
+ }
95
+
96
+ interface Props {
97
+ /**
98
+ * Represents the hierarchical structure of directories and files, providing the component with an organized view of the file system.
99
+ */
100
+ fileTree: FileDir,
101
+ }
102
+
103
+ export const FileTreeView = ({ fileTree }: Props) => {
104
+
105
+ const AccordionItem = ({ name, isFile, dirs, defaultOpened = false, menuOptions, itemLink }: FileDir
106
+ & { defaultOpened?: boolean }) => {
107
+
108
+ const Link = useAnchorTag()
109
+ const [opened, setOpened] = useState<boolean>(defaultOpened)
110
+
111
+ const onExpand = () => {
112
+ setOpened(old => !old)
113
+ }
114
+
115
+ const FolderItem = () => (
116
+ <li className="accordion-node-group">
117
+ <button className="accordion-node" onClick={onExpand}>
118
+ <IconBox className="icon">
119
+ <ChevronDown />
120
+ </IconBox>
121
+ <IconBox>
122
+ <Folder />
123
+ </IconBox>
124
+ <Text>{name !== '' ? name : '/'}</Text>
125
+ </button>
126
+ { menuOptions &&
127
+ <div className="actions">
128
+ <More menu={menuOptions} />
129
+ </div>
130
+ }
131
+ </li>
132
+ )
133
+
134
+ const FileItem = () => (
135
+ <li className="accordion-node-group">
136
+ <Link href={itemLink}>
137
+ <IconBox>
138
+ <Code />
139
+ </IconBox>
140
+ <Text as="label">{name}</Text>
141
+ </Link>
142
+ { menuOptions &&
143
+ <div className="actions">
144
+ <More menu={menuOptions} />
145
+ </div>
146
+ }
147
+ </li >
148
+ )
149
+
150
+ return (
151
+ <Box sx={styles.AccordionItemWrapper}>
152
+ {
153
+ isFile ? <FileItem /> : <FolderItem />
154
+ }
155
+ <ul className="accordion-children">
156
+ {opened && !isFile && dirs && dirs.length > 0 && (
157
+ <>
158
+ {dirs?.map((subDir, index) => (
159
+ <li key={index}>{
160
+ <AccordionItem {...subDir} />
161
+ }</li>
162
+ ))}
163
+ </>
164
+ )}
165
+ </ul>
166
+ </Box>
167
+ )
168
+ }
169
+
170
+ return (
171
+ <Box sx={styles.FileTreeViewWrapper as SxProperties}>
172
+ <AccordionItem {...fileTree} defaultOpened={true} />
173
+ </Box>
174
+ )
175
+ }