datajunction-ui 0.0.1-rc.17 → 0.0.1-rc.19

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 (40) hide show
  1. package/.env +1 -1
  2. package/dj-logo.svg +10 -0
  3. package/package.json +28 -6
  4. package/src/app/__tests__/__snapshots__/index.test.tsx.snap +5 -84
  5. package/src/app/components/djgraph/DJNode.jsx +1 -1
  6. package/src/app/components/djgraph/LayoutFlow.jsx +1 -1
  7. package/src/app/constants.js +2 -0
  8. package/src/app/icons/AlertIcon.jsx +32 -0
  9. package/src/app/icons/DJLogo.jsx +36 -0
  10. package/src/app/icons/DeleteIcon.jsx +21 -0
  11. package/src/app/icons/EditIcon.jsx +18 -0
  12. package/src/app/index.tsx +70 -36
  13. package/src/app/pages/AddEditNodePage/FormikSelect.jsx +33 -0
  14. package/src/app/pages/AddEditNodePage/FullNameField.jsx +36 -0
  15. package/src/app/pages/AddEditNodePage/Loadable.jsx +16 -0
  16. package/src/app/pages/AddEditNodePage/NodeQueryField.jsx +89 -0
  17. package/src/app/pages/AddEditNodePage/__tests__/FormikSelect.test.jsx +44 -0
  18. package/src/app/pages/AddEditNodePage/__tests__/FullNameField.test.jsx +29 -0
  19. package/src/app/pages/AddEditNodePage/__tests__/NodeQueryField.test.jsx +30 -0
  20. package/src/app/pages/AddEditNodePage/__tests__/__snapshots__/index.test.jsx.snap +84 -0
  21. package/src/app/pages/AddEditNodePage/__tests__/index.test.jsx +353 -0
  22. package/src/app/pages/AddEditNodePage/index.jsx +349 -0
  23. package/src/app/pages/LoginPage/assets/sign-in-with-github.png +0 -0
  24. package/src/app/pages/LoginPage/assets/sign-in-with-google.png +0 -0
  25. package/src/app/pages/LoginPage/index.jsx +90 -0
  26. package/src/app/pages/NamespacePage/__tests__/__snapshots__/index.test.tsx.snap +50 -2
  27. package/src/app/pages/NamespacePage/index.jsx +38 -9
  28. package/src/app/pages/NodePage/NodeGraphTab.jsx +1 -2
  29. package/src/app/pages/NodePage/NodeHistory.jsx +0 -1
  30. package/src/app/pages/NodePage/index.jsx +43 -26
  31. package/src/app/pages/Root/index.tsx +20 -3
  32. package/src/app/pages/SQLBuilderPage/index.jsx +54 -8
  33. package/src/app/services/DJService.js +180 -32
  34. package/src/setupTests.ts +1 -1
  35. package/src/styles/index.css +82 -5
  36. package/src/styles/login.css +67 -0
  37. package/src/styles/node-creation.scss +190 -0
  38. package/src/styles/styles.scss +44 -0
  39. package/src/styles/styles.scss.d.ts +9 -0
  40. package/webpack.config.js +11 -1
package/.env CHANGED
@@ -1,2 +1,2 @@
1
1
  REACT_APP_DJ_URL=http://localhost:8000
2
- REACT_USE_SSE=true
2
+ REACT_USE_SSE=true
package/dj-logo.svg ADDED
@@ -0,0 +1,10 @@
1
+ <svg width="83" height="83" viewBox="0 0 83 83" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
2
+ <rect width="83" height="83" fill="url(#pattern0)"/>
3
+ <rect width="83" height="83" stroke="black"/>
4
+ <defs>
5
+ <pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1">
6
+ <use xlink:href="#image0_1_2" transform="scale(0.00364964)"/>
7
+ </pattern>
8
+ <image id="image0_1_2" width="274" height="274" xlink:href=""/>
9
+ </defs>
10
+ </svg>
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "datajunction-ui",
3
- "version": "0.0.1-rc.17",
3
+ "version": "0.0.1-rc.19",
4
4
  "description": "DataJunction Metrics Platform UI",
5
5
  "module": "src/index.tsx",
6
6
  "repository": {
@@ -23,9 +23,10 @@
23
23
  "@babel/core": "^7.18.2",
24
24
  "@babel/preset-env": "^7.18.2",
25
25
  "@babel/preset-react": "7.18.6",
26
+ "@codemirror/lang-sql": "^6.4.0",
26
27
  "@reduxjs/toolkit": "1.8.5",
27
- "@testing-library/jest-dom": "5.16.5",
28
- "@testing-library/react": "13.4.0",
28
+ "@testing-library/jest-dom": "6.1.2",
29
+ "@testing-library/react": "14.0.0",
29
30
  "@types/fontfaceobserver": "^2.1.0",
30
31
  "@types/jest": "^27.5.2",
31
32
  "@types/node": "^14.18.27",
@@ -39,31 +40,39 @@
39
40
  "@types/testing-library__jest-dom": "^5.14.5",
40
41
  "@types/webpack": "^5.28.0",
41
42
  "@types/webpack-env": "^1.18.0",
43
+ "@uiw/codemirror-extensions-basic-setup": "4.21.12",
44
+ "@uiw/codemirror-extensions-langs": "4.21.12",
45
+ "@uiw/react-codemirror": "4.21.12",
42
46
  "babel-loader": "9.1.2",
43
47
  "chalk": "4.1.2",
48
+ "codemirror": "^6.0.0",
44
49
  "cronstrue": "2.27.0",
45
50
  "cross-env": "7.0.3",
46
- "css-loader": "6.7.3",
51
+ "css-loader": "6.8.1",
47
52
  "dagre": "^0.8.5",
48
53
  "datajunction": "0.0.1-rc.0",
49
54
  "file-loader": "6.2.0",
50
55
  "fontfaceobserver": "2.3.0",
56
+ "formik": "2.4.3",
51
57
  "husky": "8.0.1",
52
58
  "i18next": "21.9.2",
53
59
  "i18next-browser-languagedetector": "6.1.5",
54
60
  "i18next-scanner": "4.0.0",
55
61
  "inquirer": "7.3.3",
56
62
  "inquirer-directory": "2.2.0",
63
+ "js-cookie": "3.0.5",
57
64
  "lint-staged": "13.0.3",
58
65
  "node-plop": "0.26.3",
59
66
  "plop": "2.7.6",
60
67
  "prettier": "2.7.1",
61
68
  "react": "18.2.0",
62
69
  "react-app-polyfill": "3.0.0",
70
+ "react-cookie": "4.1.1",
63
71
  "react-dom": "18.2.0",
64
72
  "react-helmet-async": "1.3.0",
65
73
  "react-i18next": "11.18.6",
66
74
  "react-is": "18.2.0",
75
+ "react-querybuilder": "6.5.1",
67
76
  "react-redux": "7.2.8",
68
77
  "react-router-dom": "6.3.0",
69
78
  "react-scripts": "5.0.1",
@@ -75,10 +84,12 @@
75
84
  "redux-saga": "1.2.1",
76
85
  "rimraf": "3.0.2",
77
86
  "sanitize.css": "13.0.0",
87
+ "sass": "1.66.1",
88
+ "sass-loader": "13.3.2",
78
89
  "serve": "14.0.1",
79
90
  "shelljs": "0.8.5",
80
91
  "sql-formatter": "^12.2.0",
81
- "style-loader": "3.3.2",
92
+ "style-loader": "3.3.3",
82
93
  "stylelint": "14.12.0",
83
94
  "stylelint-config-recommended": "9.0.0",
84
95
  "ts-loader": "9.4.2",
@@ -136,6 +147,9 @@
136
147
  ]
137
148
  },
138
149
  "jest": {
150
+ "transformIgnorePatterns": [
151
+ "!node_modules/"
152
+ ],
139
153
  "collectCoverageFrom": [
140
154
  "src/**/*.{js,jsx,ts,tsx}",
141
155
  "!src/**/*/*.d.ts",
@@ -153,13 +167,21 @@
153
167
  }
154
168
  }
155
169
  },
170
+ "resolutions": {
171
+ "@codemirror/state": "6.2.0",
172
+ "@codemirror/view": "6.2.0",
173
+ "@lezer/common": "^1.0.0"
174
+ },
156
175
  "devDependencies": {
157
176
  "@babel/plugin-proposal-class-properties": "7.18.6",
177
+ "@babel/plugin-proposal-private-property-in-object": "7.21.11",
178
+ "@testing-library/user-event": "14.4.3",
158
179
  "eslint-config-prettier": "8.8.0",
159
180
  "eslint-plugin-prettier": "4.2.1",
160
181
  "eslint-plugin-react-hooks": "4.6.0",
161
182
  "html-webpack-plugin": "5.5.1",
162
183
  "jest": "^29.5.0",
163
- "mini-css-extract-plugin": "2.7.5"
184
+ "jest-fetch-mock": "3.0.3",
185
+ "mini-css-extract-plugin": "2.7.6"
164
186
  }
165
187
  }
@@ -1,88 +1,9 @@
1
1
  // Jest Snapshot v1, https://goo.gl/fbAQLP
2
2
 
3
3
  exports[`<App /> should render and match the snapshot 1`] = `
4
- <BrowserRouter>
5
- <Helmet
6
- defaultTitle="DataJunction: A Metrics Platform"
7
- defer={true}
8
- encodeSpecialCharacters={true}
9
- prioritizeSeoTags={false}
10
- titleTemplate="DataJunction: %s"
11
- >
12
- <meta
13
- content="DataJunction serves as a semantic layer to help manage metrics"
14
- name="description"
15
- />
16
- </Helmet>
17
- <Context.Provider
18
- value={
19
- Object {
20
- "DataJunctionAPI": Object {
21
- "clientCode": [Function],
22
- "columns": [Function],
23
- "commonDimensions": [Function],
24
- "compiledSql": [Function],
25
- "cube": [Function],
26
- "dag": [Function],
27
- "data": [Function],
28
- "downstreams": [Function],
29
- "history": [Function],
30
- "lineage": [Function],
31
- "materializations": [Function],
32
- "metric": [Function],
33
- "metrics": [Function],
34
- "namespace": [Function],
35
- "namespaces": [Function],
36
- "node": [Function],
37
- "node_dag": [Function],
38
- "node_lineage": [Function],
39
- "nodesWithDimension": [Function],
40
- "revisions": [Function],
41
- "sql": [Function],
42
- "sqls": [Function],
43
- "stream": [Function],
44
- "upstreams": [Function],
45
- },
46
- }
47
- }
48
- >
49
- <Routes>
50
- <Route
51
- element={<Unknown />}
52
- path="/"
53
- >
54
- <React.Fragment>
55
- <Route
56
- path="nodes"
57
- >
58
- <Route
59
- element={<NodePage />}
60
- path=":name"
61
- />
62
- </Route>
63
- <Route
64
- element={<NamespacePage />}
65
- path="/"
66
- />
67
- <Route
68
- path="namespaces"
69
- >
70
- <Route
71
- element={<NamespacePage />}
72
- path=":namespace"
73
- />
74
- </Route>
75
- <Route
76
- element={<SQLBuilderPage />}
77
- path="sql"
78
- />
79
- </React.Fragment>
80
- </Route>
81
- <Route
82
- element={<Unknown />}
83
- path="*"
84
- />
85
- </Routes>
86
- </Context.Provider>
87
- </BrowserRouter>
4
+ <CookiesProvider>
5
+ <BrowserRouter>
6
+ <LoginPage />
7
+ </BrowserRouter>
8
+ </CookiesProvider>
88
9
  `;
@@ -1,4 +1,4 @@
1
- import { memo, useLayoutEffect, useRef, useState } from 'react';
1
+ import { memo } from 'react';
2
2
  import { Handle, Position } from 'reactflow';
3
3
  import Collapse from './Collapse';
4
4
 
@@ -1,4 +1,4 @@
1
- import React, { useCallback, useEffect, useMemo, useState } from 'react';
1
+ import React, { useCallback, useEffect, useMemo } from 'react';
2
2
  import ReactFlow, {
3
3
  addEdge,
4
4
  MiniMap,
@@ -0,0 +1,2 @@
1
+ export const DJ_AUTH_COOKIE = '__dj';
2
+ export const DJ_LOGGED_IN_FLAG_COOKIE = '__djlif';
@@ -0,0 +1,32 @@
1
+ const AlertIcon = props => (
2
+ <svg
3
+ width="2em"
4
+ height="2em"
5
+ viewBox="0 0 24 24"
6
+ version="1.1"
7
+ xmlns="http://www.w3.org/2000/svg"
8
+ >
9
+ <title>alert_fill</title>
10
+ <g id="page-1" stroke="none" strokeWidth="1" fill="none" fillRule="evenodd">
11
+ <g
12
+ id="System"
13
+ transform="translate(-48.000000, -48.000000)"
14
+ fillRule="nonzero"
15
+ >
16
+ <g id="alert_fill" transform="translate(48.000000, 48.000000)">
17
+ <path
18
+ d="M24,0 L24,24 L0,24 L0,0 L24,0 Z M12.5934901,23.257841 L12.5819402,23.2595131 L12.5108777,23.2950439 L12.4918791,23.2987469 L12.4918791,23.2987469 L12.4767152,23.2950439 L12.4056548,23.2595131 C12.3958229,23.2563662 12.3870493,23.2590235 12.3821421,23.2649074 L12.3780323,23.275831 L12.360941,23.7031097 L12.3658947,23.7234994 L12.3769048,23.7357139 L12.4804777,23.8096931 L12.4953491,23.8136134 L12.4953491,23.8136134 L12.5071152,23.8096931 L12.6106902,23.7357139 L12.6232938,23.7196733 L12.6232938,23.7196733 L12.6266527,23.7031097 L12.609561,23.275831 C12.6075724,23.2657013 12.6010112,23.2592993 12.5934901,23.257841 L12.5934901,23.257841 Z M12.8583906,23.1452862 L12.8445485,23.1473072 L12.6598443,23.2396597 L12.6498822,23.2499052 L12.6498822,23.2499052 L12.6471943,23.2611114 L12.6650943,23.6906389 L12.6699349,23.7034178 L12.6699349,23.7034178 L12.678386,23.7104931 L12.8793402,23.8032389 C12.8914285,23.8068999 12.9022333,23.8029875 12.9078286,23.7952264 L12.9118235,23.7811639 L12.8776777,23.1665331 C12.8752882,23.1545897 12.8674102,23.1470016 12.8583906,23.1452862 L12.8583906,23.1452862 Z M12.1430473,23.1473072 C12.1332178,23.1423925 12.1221763,23.1452606 12.1156365,23.1525954 L12.1099173,23.1665331 L12.0757714,23.7811639 C12.0751323,23.7926639 12.0828099,23.8018602 12.0926481,23.8045676 L12.108256,23.8032389 L12.3092106,23.7104931 L12.3186497,23.7024347 L12.3186497,23.7024347 L12.3225043,23.6906389 L12.340401,23.2611114 L12.337245,23.2485176 L12.337245,23.2485176 L12.3277531,23.2396597 L12.1430473,23.1473072 Z"
19
+ id="MingCute"
20
+ fillRule="nonzero"
21
+ ></path>
22
+ <path
23
+ d="M13.299,3.1477 L21.933,18.1022 C22.5103,19.1022 21.7887,20.3522 20.634,20.3522 L3.36601,20.3522 C2.21131,20.3522 1.48962,19.1022 2.06697,18.1022 L10.7009,3.14771 C11.2783,2.14771 12.7217,2.1477 13.299,3.1477 Z M12,15 C11.4477,15 11,15.4477 11,16 C11,16.5523 11.4477,17 12,17 C12.5523,17 13,16.5523 13,16 C13,15.4477 12.5523,15 12,15 Z M12,8 C11.48715,8 11.0644908,8.38604429 11.0067275,8.88337975 L11,9 L11,13 C11,13.5523 11.4477,14 12,14 C12.51285,14 12.9355092,13.613973 12.9932725,13.1166239 L13,13 L13,9 C13,8.44772 12.5523,8 12,8 Z"
24
+ id="shape"
25
+ fill="#09244B"
26
+ ></path>
27
+ </g>
28
+ </g>
29
+ </g>
30
+ </svg>
31
+ );
32
+ export default AlertIcon;
@@ -0,0 +1,36 @@
1
+ const DJLogo = props => (
2
+ <svg
3
+ width="30"
4
+ height="30"
5
+ style={{
6
+ marginRight: '10px',
7
+ marginBottom: '2px',
8
+ stroke: 'transparent',
9
+ strokeWidth: '0px',
10
+ }}
11
+ viewBox="0 0 83 83"
12
+ fill="none"
13
+ xmlns="http://www.w3.org/2000/svg"
14
+ xmlnsXlink="http://www.w3.org/1999/xlink"
15
+ >
16
+ <rect width="83" height="83" fill="url(#pattern0)" />
17
+ <rect width="83" height="83" stroke="black" />
18
+ <defs>
19
+ <pattern
20
+ id="pattern0"
21
+ patternContentUnits="objectBoundingBox"
22
+ width="1"
23
+ height="1"
24
+ >
25
+ <use xlinkHref="#image0_1_2" transform="scale(0.00364964)" />
26
+ </pattern>
27
+ <image
28
+ id="image0_1_2"
29
+ width="274"
30
+ height="274"
31
+ xlinkHref=""
32
+ />
33
+ </defs>
34
+ </svg>
35
+ );
36
+ export default DJLogo;
@@ -0,0 +1,21 @@
1
+ const DeleteIcon = props => (
2
+ <svg
3
+ className="feather feather-trash-2"
4
+ fill="none"
5
+ height="24"
6
+ stroke="currentColor"
7
+ strokeLinecap="round"
8
+ strokeLinejoin="round"
9
+ strokeWidth="2"
10
+ viewBox="0 0 24 24"
11
+ width="24"
12
+ xmlns="http://www.w3.org/2000/svg"
13
+ >
14
+ <polyline points="3 6 5 6 21 6" />
15
+ <path d="M19 6v14a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V6m3 0V4a2 2 0 0 1 2-2h4a2 2 0 0 1 2 2v2" />
16
+ <line x1="10" x2="10" y1="11" y2="17" />
17
+ <line x1="14" x2="14" y1="11" y2="17" />
18
+ </svg>
19
+ );
20
+
21
+ export default DeleteIcon;
@@ -0,0 +1,18 @@
1
+ const EditIcon = props => (
2
+ <svg
3
+ className="feather feather-edit"
4
+ fill="none"
5
+ height="24"
6
+ stroke="currentColor"
7
+ strokeLinecap="round"
8
+ strokeLinejoin="round"
9
+ strokeWidth="2"
10
+ viewBox="0 0 24 24"
11
+ width="24"
12
+ xmlns="http://www.w3.org/2000/svg"
13
+ >
14
+ <path d="M11 4H4a2 2 0 0 0-2 2v14a2 2 0 0 0 2 2h14a2 2 0 0 0 2-2v-7" />
15
+ <path d="M18.5 2.5a2.121 2.121 0 0 1 3 3L12 15l-4 1 1-4 9.5-9.5z" />
16
+ </svg>
17
+ );
18
+ export default EditIcon;
package/src/app/index.tsx CHANGED
@@ -10,49 +10,83 @@ import { BrowserRouter, Routes, Route } from 'react-router-dom';
10
10
  import { NamespacePage } from './pages/NamespacePage/Loadable';
11
11
  import { NodePage } from './pages/NodePage/Loadable';
12
12
  import { SQLBuilderPage } from './pages/SQLBuilderPage/Loadable';
13
+ import { AddEditNodePage } from './pages/AddEditNodePage/Loadable';
13
14
  import { NotFoundPage } from './pages/NotFoundPage/Loadable';
15
+ import { LoginPage } from './pages/LoginPage';
14
16
  import { Root } from './pages/Root/Loadable';
15
17
  import DJClientContext from './providers/djclient';
16
18
  import { DataJunctionAPI } from './services/DJService';
19
+ import { CookiesProvider, useCookies } from 'react-cookie';
20
+ import * as Constants from './constants';
17
21
 
18
22
  export function App() {
23
+ const [cookies] = useCookies([Constants.DJ_LOGGED_IN_FLAG_COOKIE]);
19
24
  return (
20
- <BrowserRouter>
21
- <Helmet
22
- titleTemplate="DataJunction: %s"
23
- defaultTitle="DataJunction: A Metrics Platform"
24
- >
25
- <meta
26
- name="description"
27
- content="DataJunction serves as a semantic layer to help manage metrics"
28
- />
29
- </Helmet>
30
- <DJClientContext.Provider value={{ DataJunctionAPI }}>
31
- <Routes>
32
- <Route
33
- path="/"
34
- element={<Root />}
35
- children={
36
- <>
37
- <Route path="nodes" key="nodes">
38
- <Route path=":name" element={<NodePage />} />
39
- </Route>
25
+ <CookiesProvider>
26
+ <BrowserRouter>
27
+ {cookies.__djlif || process.env.REACT_DISABLE_AUTH === 'true' ? (
28
+ <>
29
+ <Helmet
30
+ titleTemplate="DataJunction: %s"
31
+ defaultTitle="DataJunction: A Metrics Platform"
32
+ >
33
+ <meta
34
+ name="description"
35
+ content="DataJunction serves as a semantic layer to help manage metrics"
36
+ />
37
+ </Helmet>
38
+ <DJClientContext.Provider value={{ DataJunctionAPI }}>
39
+ <Routes>
40
+ <Route
41
+ path="/"
42
+ element={<Root />}
43
+ children={
44
+ <>
45
+ <Route path="nodes" key="nodes">
46
+ <Route path=":name" element={<NodePage />} />
47
+ <Route
48
+ path=":name/edit"
49
+ key="edit"
50
+ element={<AddEditNodePage />}
51
+ />
52
+ </Route>
40
53
 
41
- <Route path="/" element={<NamespacePage />} key="index" />
42
- <Route path="namespaces">
43
- <Route
44
- path=":namespace"
45
- element={<NamespacePage />}
46
- key="namespaces"
47
- />
48
- </Route>
49
- <Route path="sql" key="sql" element={<SQLBuilderPage />} />
50
- </>
51
- }
52
- />
53
- <Route path="*" element={<NotFoundPage />} />
54
- </Routes>
55
- </DJClientContext.Provider>
56
- </BrowserRouter>
54
+ <Route path="/" element={<NamespacePage />} key="index" />
55
+ <Route path="namespaces">
56
+ <Route
57
+ path=":namespace"
58
+ element={<NamespacePage />}
59
+ key="namespaces"
60
+ />
61
+ </Route>
62
+ <Route path="create/:nodeType">
63
+ <Route
64
+ path=":initialNamespace"
65
+ key="create"
66
+ element={<AddEditNodePage />}
67
+ />
68
+ <Route
69
+ path=""
70
+ key="create"
71
+ element={<AddEditNodePage />}
72
+ />
73
+ </Route>
74
+ <Route
75
+ path="sql"
76
+ key="sql"
77
+ element={<SQLBuilderPage />}
78
+ />
79
+ </>
80
+ }
81
+ />
82
+ <Route path="*" element={<NotFoundPage />} />
83
+ </Routes>
84
+ </DJClientContext.Provider>
85
+ </>
86
+ ) : (
87
+ <LoginPage />
88
+ )}
89
+ </BrowserRouter>
90
+ </CookiesProvider>
57
91
  );
58
92
  }
@@ -0,0 +1,33 @@
1
+ /**
2
+ * A React Select component for use in Formik forms.
3
+ */
4
+ import { useField } from 'formik';
5
+ import Select from 'react-select';
6
+
7
+ export const FormikSelect = ({
8
+ selectOptions,
9
+ formikFieldName,
10
+ placeholder,
11
+ defaultValue,
12
+ style,
13
+ }) => {
14
+ // eslint-disable-next-line no-unused-vars
15
+ const [field, _, helpers] = useField(formikFieldName);
16
+ const { setValue } = helpers;
17
+
18
+ return (
19
+ <Select
20
+ className="SelectInput"
21
+ defaultValue={defaultValue}
22
+ options={selectOptions}
23
+ placeholder={placeholder}
24
+ onBlur={field.onBlur}
25
+ onChange={option => setValue(option.value)}
26
+ styles={style}
27
+ />
28
+ );
29
+ };
30
+
31
+ FormikSelect.defaultProps = {
32
+ placeholder: '',
33
+ };
@@ -0,0 +1,36 @@
1
+ /**
2
+ * A field for the full node name, which is generated based on the node's input
3
+ * namespace and display name.
4
+ */
5
+ import { useField, useFormikContext } from 'formik';
6
+ import { useEffect } from 'react';
7
+
8
+ export const FullNameField = props => {
9
+ const { values, setFieldValue } = useFormikContext();
10
+ const [field, meta] = useField(props);
11
+
12
+ useEffect(() => {
13
+ // Set the value of the node's full name based on its namespace and display name
14
+ if (values.namespace && values.display_name) {
15
+ setFieldValue(
16
+ props.name,
17
+ `${values.namespace}.${values.display_name
18
+ .toLowerCase()
19
+ .replace(/ /g, '_')}`,
20
+ );
21
+ }
22
+ }, [setFieldValue, props.name, values]);
23
+
24
+ return (
25
+ <>
26
+ <input
27
+ {...props}
28
+ {...field}
29
+ className="FullNameField"
30
+ disabled="disabled"
31
+ id="FullName"
32
+ />
33
+ {!!meta.touched && !!meta.error && <div>{meta.error}</div>}
34
+ </>
35
+ );
36
+ };
@@ -0,0 +1,16 @@
1
+ /**
2
+ * Asynchronously loads the component for the Node page
3
+ */
4
+
5
+ import * as React from 'react';
6
+ import { lazyLoad } from '../../../utils/loadable';
7
+
8
+ export const AddEditNodePage = () => {
9
+ return lazyLoad(
10
+ () => import('./index'),
11
+ module => module.AddEditNodePage,
12
+ {
13
+ fallback: <div></div>,
14
+ },
15
+ )();
16
+ };