@genesislcap/blank-app-seed 3.30.0-prerelease.2 → 3.30.0-prerelease.21

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 (92) hide show
  1. package/.genx/package.json +1 -1
  2. package/.genx/prompts/api.js +2 -9
  3. package/.genx/static.js +0 -1
  4. package/.genx/templates/react/chart.hbs +7 -3
  5. package/.genx/templates/react/component/component.column.defs.hbs +4 -2
  6. package/.genx/templates/react/component/component.create.form.hbs +2 -2
  7. package/.genx/templates/react/component/component.gridOptions.hbs +4 -2
  8. package/.genx/templates/react/component/component.hbs +64 -27
  9. package/.genx/templates/react/component/component.index.hbs +1 -1
  10. package/.genx/templates/react/component/component.update.form.hbs +1 -1
  11. package/.genx/templates/react/entityManager.hbs +62 -49
  12. package/.genx/templates/react/form.hbs +7 -3
  13. package/.genx/templates/react/grid.hbs +16 -10
  14. package/.genx/templates/react/route.hbs +1 -2
  15. package/.genx/templates/react/tabsLayout.hbs +5 -5
  16. package/.genx/utils/generateRoute.js +8 -7
  17. package/.genx/utils/generateTile.js +31 -16
  18. package/.genx/versions.json +3 -3
  19. package/.idea/modules.xml +6 -6
  20. package/CHANGELOG.md +133 -0
  21. package/client-tmp/angular/__mocks__/fileMock.js +1 -0
  22. package/client-tmp/angular/angular.json +0 -29
  23. package/client-tmp/angular/jest.config.js +20 -0
  24. package/client-tmp/angular/package.json +6 -6
  25. package/client-tmp/angular/src/app/layouts/default/default.layout.spec.ts +2 -2
  26. package/client-tmp/angular/src/jest.setup.ts +46 -0
  27. package/client-tmp/angular/tsconfig.app.json +1 -2
  28. package/client-tmp/angular/tsconfig.json +1 -2
  29. package/client-tmp/angular/tsconfig.spec.json +13 -7
  30. package/client-tmp/react/.babelrc +10 -0
  31. package/client-tmp/react/.eslintrc.cjs +1 -0
  32. package/client-tmp/react/jest.config.ts +15 -0
  33. package/client-tmp/react/jest.setup.ts +1 -0
  34. package/client-tmp/react/lint-css.ts +19 -0
  35. package/client-tmp/react/package.json +40 -15
  36. package/client-tmp/react/public/favicon.ico +0 -0
  37. package/client-tmp/react/public/index.html +13 -0
  38. package/client-tmp/react/src/App.tsx +95 -68
  39. package/client-tmp/react/src/components/{ErrorMessage.test.js → ErrorMessage/ErrorMessage.test.tsx} +31 -16
  40. package/client-tmp/react/src/components/ErrorMessage/{ErrorMessage.jsx → ErrorMessage.tsx} +9 -4
  41. package/client-tmp/react/src/config.ts +2 -16
  42. package/client-tmp/react/src/custom-elements.d.ts +21 -9
  43. package/client-tmp/react/src/environments/environment.prod.ts +8 -0
  44. package/client-tmp/react/src/environments/environment.ts +9 -0
  45. package/client-tmp/react/src/guards/ProtectedGuard.tsx +64 -0
  46. package/client-tmp/react/src/helpers/hasPermissionHelper.tsx +10 -0
  47. package/client-tmp/react/src/helpers/isAuthenticatedHelper.tsx +14 -0
  48. package/client-tmp/react/src/helpers/isConnectedHelper.tsx +16 -0
  49. package/client-tmp/react/src/layouts/blank/BlankLayout.tsx +1 -1
  50. package/client-tmp/react/src/layouts/default/DefaultLayout.tsx +26 -15
  51. package/client-tmp/react/src/main.tsx +21 -5
  52. package/client-tmp/react/src/pages/AuthPage/AuthPage.test.tsx +10 -0
  53. package/client-tmp/react/src/pages/AuthPage/{AuthPage.jsx → AuthPage.tsx} +3 -2
  54. package/client-tmp/react/src/pages/NotFoundPage/NotFoundPage.css +0 -0
  55. package/client-tmp/react/src/pages/NotFoundPage/NotFoundPage.tsx +14 -0
  56. package/client-tmp/react/src/pages/NotPermittedPage/NotPermittedPage.test.tsx +17 -0
  57. package/client-tmp/react/src/pages/NotPermittedPage/NotPermittedPage.tsx +14 -0
  58. package/client-tmp/react/src/pbc/README.md +7 -0
  59. package/client-tmp/react/src/pbc/container.tsx +44 -0
  60. package/client-tmp/react/src/pbc/elementsRenderer.tsx +40 -0
  61. package/client-tmp/react/src/pbc/utils.ts +121 -0
  62. package/client-tmp/react/src/services/store.service.ts +48 -0
  63. package/client-tmp/react/src/share/foundation-login.ts +13 -6
  64. package/client-tmp/react/src/share/genesis-components.ts +49 -23
  65. package/client-tmp/react/src/store/AuthContext.tsx +0 -2
  66. package/client-tmp/react/src/store/RoutesContext.tsx +74 -0
  67. package/client-tmp/react/src/types/LayoutName.ts +3 -0
  68. package/client-tmp/react/src/types/RouteLayouts.ts +3 -1
  69. package/client-tmp/react/src/utils/getLayoutNameByRoute.ts +5 -0
  70. package/client-tmp/react/src/utils/history.ts +4 -2
  71. package/client-tmp/react/src/utils/index.ts +2 -1
  72. package/client-tmp/react/test/e2e/fixture.ts +1 -1
  73. package/client-tmp/react/tsconfig.app.json +2 -26
  74. package/client-tmp/react/tsconfig.json +31 -8
  75. package/client-tmp/react/tsconfig.node.json +1 -1
  76. package/client-tmp/react/vite.config.ts +70 -0
  77. package/client-tmp/react/webpack.config.js +103 -0
  78. package/client-tmp/web-components/src/main/main.styles.ts +1 -1
  79. package/package.json +1 -1
  80. package/server/settings.gradle.kts +2 -2
  81. package/client-tmp/angular/karma.conf.js +0 -9
  82. package/client-tmp/angular/src/app/app.component.spec.ts +0 -23
  83. package/client-tmp/react/env.development.json +0 -3
  84. package/client-tmp/react/lint-css.js +0 -18
  85. package/client-tmp/react/src/guards/AuthGuard.tsx +0 -32
  86. package/client-tmp/react/src/guards/PermissionsGuard.tsx +0 -37
  87. package/client-tmp/react/src/index.js +0 -17
  88. package/client-tmp/react/src/pages/NotPermittedPage/NotPermittedPage.jsx +0 -13
  89. package/client-tmp/react/src/reportWebVitals.js +0 -13
  90. package/client-tmp/react/vite.config.js +0 -59
  91. /package/client-tmp/react/src/{setupTests.js → setupTests.ts} +0 -0
  92. /package/server/{{appName}}-app/src/main/genesis/scripts/{{{appName}}-data-pipelines.kts → {{appName}}-pipelines.kts} +0 -0
@@ -1,31 +1,75 @@
1
+ import { useEffect, useState } from 'react';
2
+ import { unstable_HistoryRouter as HistoryRouter, Routes, Route, useLocation } from 'react-router-dom';
1
3
  import {
2
- unstable_HistoryRouter as HistoryRouter,
3
- Routes,
4
- Route,
5
- useLocation,
6
- } from 'react-router-dom';
7
- import { useEffect } from 'react';
8
- import { history, setApiHost{{#if FDC3.channels.length}}, listenToChannel{{/if}} } from './utils';
4
+ history,
5
+ setApiHost,
6
+ getLayoutNameByRoute,
7
+ {{#if FDC3.channels.length~}}
8
+ listenToChannel,
9
+ onFDC3Ready,
10
+ {{/if}}
11
+ } from './utils';
12
+ import { customEventFactory, registerStylesTarget } from '@/pbc/utils';
9
13
  import LayoutWrapper from './layouts/LayoutWrapper';
10
- import { AUTH_PATH, NOT_PERMITTED_PATH, routeLayouts } from './config';
11
- import AuthGuard from './guards/AuthGuard';
12
- import PermissionsGuard from './guards/PermissionsGuard';
14
+ import LayoutName from '@/types/LayoutName';
15
+ import { AUTH_PATH, routeLayouts } from './config';
13
16
  import { AuthProvider } from './store/AuthContext';
14
- // Pages Components
17
+ import { RoutesProvider, useRoutesContext } from './store/RoutesContext';
15
18
  import AuthPage from './pages/AuthPage/AuthPage';
16
- import NotPermittedPage from './pages/NotPermittedPage/NotPermittedPage';
17
- {{#each routes}}
18
- import {{pascalCase this.name}} from './pages/{{kebabCase this.name}}/{{pascalCase this.name}}';
19
- {{/each}}
20
-
21
- // Genesis Components
22
- import './share/genesis-components';
19
+ import { registerComponents as genesisRegisterComponents } from './share/genesis-components';
20
+ import { configureFoundationLogin } from './share/foundation-login';
21
+ import ProtectedGuard from './guards/ProtectedGuard';
22
+ import { storeService } from '@/services/store.service';
23
23
 
24
- const LayoutWithLocation = () => {
24
+ const DynamicLayout = () => {
25
25
  const location = useLocation();
26
- const layout = routeLayouts[location.pathname] || 'default';
27
- {{#if FDC3.channels.length}}
26
+ const [layoutName, setLayoutName] = useState<LayoutName>(routeLayouts[location.pathname] || 'default');
27
+ const handleRouteChange = (location: any) => {
28
+ setLayoutName(getLayoutNameByRoute(location.pathname));
29
+ };
30
+ const route = useRoutesContext().find((r) => r.path === location.pathname);
31
+ let pageComponent;
32
+ let content;
33
+
28
34
  useEffect(() => {
35
+ handleRouteChange(location);
36
+ const unlisten = history.listen(handleRouteChange);
37
+
38
+ return () => {
39
+ unlisten();
40
+ }
41
+ }, [location]);
42
+
43
+ if (route) {
44
+ pageComponent = route.element;
45
+ } else {
46
+ pageComponent = <AuthPage />;
47
+ }
48
+
49
+ if (location.pathname === `/${AUTH_PATH}` || location.pathname === '/') {
50
+ content = pageComponent;
51
+ } else {
52
+ content = <ProtectedGuard>{pageComponent}</ProtectedGuard>
53
+ }
54
+
55
+ return <LayoutWrapper layout={layoutName}>{content}</LayoutWrapper>
56
+
57
+ };
58
+
59
+ interface AppProps {
60
+ rootElement: HTMLElement;
61
+ }
62
+
63
+ const App: React.FC<AppProps> = ({ rootElement }) => {
64
+ const [isStoreConnected, setIsStoreConnected] = useState(false);
65
+ const dispatchCustomEvent = (type: string, detail?: any) => {
66
+ rootElement.dispatchEvent(customEventFactory(type, detail));
67
+ };
68
+ const handleStoreConnected = (event: CustomEvent) => {
69
+ storeService.onConnected(event);
70
+ };
71
+ {{#if FDC3.channels.length~}}
72
+ const FDC3ReadyHandler = () => {
29
73
  {{#each FDC3.channels}}
30
74
  listenToChannel('{{this.name}}', '{{this.type}}', (result) => {
31
75
  console.log('Received FDC3 channel message on: {{this.name}} channel, type: {{this.type}}', result);
@@ -33,61 +77,44 @@ const LayoutWithLocation = () => {
33
77
  // E.g. open a modal or route to specific page: Route.path.push(`[Route name]`);
34
78
  });
35
79
  {{/each}}
36
-
37
- return () => {
38
- console.log('Component is being unmounted');
39
- };
40
- }, []);
80
+ };
41
81
  {{/if}}
42
82
 
43
- let pageComponent;
44
- let permissionCode = '{{this.permissions.viewRight}}';
45
-
46
- switch (location.pathname) {
47
- case `/${AUTH_PATH}`:
48
- pageComponent = <AuthPage />;
49
- break;
50
- case `/${NOT_PERMITTED_PATH}`:
51
- pageComponent = <NotPermittedPage />;
52
- break;
53
- {{#each routes}}
54
- case '/{{kebabCase this.name}}':
55
- pageComponent = <{{pascalCase this.name}} />;
56
- permissionCode = '{{this.permissions.viewRight}}';
57
- break;
58
- {{/each}}
59
- default:
60
- pageComponent = <AuthPage />;
61
- }
83
+ setApiHost();
84
+ genesisRegisterComponents();
85
+ configureFoundationLogin({ router: history });
62
86
 
63
- if (
64
- location.pathname === '/auth' ||
65
- location.pathname === '/'
66
- ) {
67
- return <LayoutWrapper layout={layout}>{pageComponent}</LayoutWrapper>;
68
- } else {
69
- return (
70
- <AuthGuard>
71
- <PermissionsGuard permissionCode={permissionCode}>
72
- <LayoutWrapper layout={layout}>{pageComponent}</LayoutWrapper>
73
- </PermissionsGuard>
74
- </AuthGuard>
75
- );
76
- }
77
- };
87
+ useEffect(() => {
88
+ registerStylesTarget(document.body, 'main');
89
+ {{#if FDC3.channels.length~}}
90
+ onFDC3Ready(FDC3ReadyHandler);
91
+ {{/if}}
92
+ if (!isStoreConnected) {
93
+ rootElement.addEventListener('store-connected', handleStoreConnected);
94
+ dispatchCustomEvent('store-connected', rootElement);
95
+ dispatchCustomEvent('store-ready', true);
96
+ setIsStoreConnected(true);
97
+ }
78
98
 
79
- const App: React.FC = () => {
80
- setApiHost();
99
+ return () => {
100
+ if (isStoreConnected) {
101
+ rootElement.removeEventListener('store-connected', handleStoreConnected);
102
+ dispatchCustomEvent('store-disconnected');
103
+ }
104
+ };
105
+ }, [isStoreConnected]);
81
106
 
82
107
  return (
83
108
  <AuthProvider>
84
- <HistoryRouter history={history}>
85
- <Routes>
86
- <Route path="*" element={<LayoutWithLocation />} />
87
- </Routes>
88
- </HistoryRouter>
109
+ <RoutesProvider>
110
+ <HistoryRouter history={history as any}>
111
+ <Routes>
112
+ <Route path="*" element={<DynamicLayout />} />
113
+ </Routes>
114
+ </HistoryRouter>
115
+ </RoutesProvider>
89
116
  </AuthProvider>
90
117
  );
91
118
  };
92
119
 
93
- export default App;
120
+ export default App;
@@ -1,35 +1,33 @@
1
- import React from 'react';
2
1
  import { render, screen } from '@testing-library/react';
3
- import '@testing-library/jest-dom';
4
- import ErrorMessage from './ErrorMessage';
2
+ import { default as ErrorMessage, ErrorMessageProps } from './ErrorMessage';
5
3
 
6
4
  describe('ErrorMessage Component', () => {
7
- const message = 'Test Error Message';
5
+ const message: string = 'Test Error Message';
8
6
 
9
7
  test('renders the message as a div by default', () => {
10
8
  render(<ErrorMessage message={message} />);
11
- const displayedMessage = screen.getByText(message);
9
+ const displayedMessage: HTMLElement = screen.getByText(message);
12
10
  expect(displayedMessage.tagName).toBe('DIV');
13
11
  expect(displayedMessage).toBeInTheDocument();
14
12
  });
15
13
 
16
14
  test('renders the message as an h1 element when elementType is "h1"', () => {
17
15
  render(<ErrorMessage message={message} elementType="h1" />);
18
- const displayedMessage = screen.getByRole('heading', { level: 1 });
16
+ const displayedMessage: HTMLElement = screen.getByRole('heading', { level: 1 });
19
17
  expect(displayedMessage.tagName).toBe('H1');
20
18
  expect(displayedMessage).toBeInTheDocument();
21
19
  });
22
20
 
23
21
  test('renders the message as a p element when elementType is "p"', () => {
24
22
  render(<ErrorMessage message={message} elementType="p" />);
25
- const displayedMessage = screen.getByText(message);
23
+ const displayedMessage: HTMLElement = screen.getByText(message);
26
24
  expect(displayedMessage.tagName).toBe('P');
27
25
  expect(displayedMessage).toBeInTheDocument();
28
26
  });
29
27
 
30
28
  test('applies the correct styles to the error message wrapper', () => {
31
29
  render(<ErrorMessage message={message} />);
32
- const wrapper = screen.getByText(message).closest('section');
30
+ const wrapper: HTMLElement | null = screen.getByText(message).closest('section');
33
31
  expect(wrapper).toHaveStyle(`
34
32
  display: flex;
35
33
  flex-direction: column;
@@ -42,8 +40,8 @@ describe('ErrorMessage Component', () => {
42
40
 
43
41
  test('applies the correct styles to the error message itself', () => {
44
42
  render(<ErrorMessage message={message} />);
45
- const displayedMessage = screen.getByText(message);
46
- const messageWrapper = displayedMessage.parentElement;
43
+ const displayedMessage: HTMLElement = screen.getByText(message);
44
+ const messageWrapper: HTMLElement | null = displayedMessage.parentElement;
47
45
 
48
46
  expect(messageWrapper).toBeInTheDocument();
49
47
  expect(messageWrapper).toHaveStyle(`
@@ -64,17 +62,34 @@ describe('ErrorMessage Component', () => {
64
62
  });
65
63
 
66
64
  test('renders nothing if the message is an empty string', () => {
67
- const { container } = render(<ErrorMessage message="" />);
65
+ const { container }: { container: HTMLElement } = render(<ErrorMessage message="" />);
68
66
  expect(container.firstChild).toBeNull();
69
67
  });
70
68
 
71
69
  test('renders the correct element type for various elementType props', () => {
72
- const elementTypes = ['h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'p', 'span', 'div'];
73
- elementTypes.forEach((elementType) => {
74
- const { container } = render(<ErrorMessage message={message} elementType={elementType} />);
75
- const displayedMessage = container.querySelector(elementType);
70
+ const elementTypes: ErrorMessageProps['elementType'][] = ['h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'p', 'span', 'div'];
71
+ elementTypes.forEach((elementType: ErrorMessageProps['elementType']) => {
72
+ const { container }: { container: HTMLElement } = render(<ErrorMessage message={message} elementType={elementType} />);
73
+ const displayedMessage: HTMLElement | null = container.querySelector(elementType as string);
76
74
  expect(displayedMessage).toBeInTheDocument();
77
- expect(displayedMessage.tagName).toBe(elementType.toUpperCase());
75
+ expect(displayedMessage?.tagName).toBe((elementType as string).toUpperCase());
78
76
  });
79
77
  });
78
+
79
+ test('renders the message as a div element when elementType is unknown', () => {
80
+ render(<ErrorMessage message={message} />);
81
+ const displayedMessage: HTMLElement = screen.getByText(message);
82
+ expect(displayedMessage.tagName).toBe('DIV');
83
+ expect(displayedMessage).toBeInTheDocument();
84
+ });
85
+
86
+ test('renders nothing if the message is null', () => {
87
+ const { container }: { container: HTMLElement } = render(<ErrorMessage />);
88
+ expect(container.firstChild).toBeNull();
89
+ });
90
+
91
+ test('renders nothing if the message is undefined', () => {
92
+ const { container }: { container: HTMLElement } = render(<ErrorMessage message={undefined} />);
93
+ expect(container.firstChild).toBeNull();
94
+ });
80
95
  });
@@ -3,7 +3,7 @@ import React from 'react';
3
3
  const styles = {
4
4
  errorMessageWrapper: {
5
5
  display: 'flex',
6
- flexDirection: 'column',
6
+ flexDirection: 'column' as const,
7
7
  justifyContent: 'center',
8
8
  alignItems: 'center',
9
9
  height: '100%',
@@ -14,11 +14,11 @@ const styles = {
14
14
  backgroundColor: 'var(--neutral-layer-4)',
15
15
  borderColor: 'var(--error-color)',
16
16
  borderRadius: '7px',
17
- borderStyle: 'solid',
17
+ borderStyle: 'solid' as const,
18
18
  borderWidth: '4px',
19
19
  padding: '5px',
20
20
  margin: '15px',
21
- textAlign: 'center',
21
+ textAlign: 'center' as const,
22
22
  width: 'fit-content',
23
23
  alignSelf: 'center',
24
24
  height: 'auto',
@@ -26,7 +26,12 @@ const styles = {
26
26
  },
27
27
  };
28
28
 
29
- const ErrorMessage = ({ elementType = 'div', message = '' }) => {
29
+ export interface ErrorMessageProps {
30
+ elementType?: 'div' | 'h1' | 'h2' | 'h3' | 'h4' | 'h5' | 'h6' | 'p' | 'span';
31
+ message?: string;
32
+ }
33
+
34
+ const ErrorMessage: React.FC<ErrorMessageProps> = ({ elementType = 'div', message = '' }) => {
30
35
  const ElementType = elementType;
31
36
 
32
37
  return message && message !== '' && (
@@ -1,32 +1,18 @@
1
1
  import { RouteLayouts } from './types/RouteLayouts';
2
+ import { environment } from '@environment';
2
3
 
3
4
  export const routeLayouts: RouteLayouts = {
4
- '/auth-mock': 'blank',
5
5
  '/auth': 'blank',
6
6
  '/': 'blank',
7
7
  };
8
8
 
9
- import type { MainMenu } from './types/menu';
10
-
11
9
  export const AUTH_PATH = 'auth';
12
10
  export const NOT_PERMITTED_PATH = 'not-permitted';
13
11
 
14
12
  export const API_DATA = {
15
- URL: import.meta.env.VITE_API_HOST,
13
+ URL: environment.API_HOST,
16
14
  AUTH: {
17
15
  username: '', // provide login to a user in given environment
18
16
  password: '', // provide password to a user in given environment
19
17
  },
20
18
  };
21
-
22
- export const mainMenu: MainMenu = [
23
- {{#each routes}}
24
- {
25
- index: {{@index}},
26
- path: '{{kebabCase this.name}}',
27
- title: '{{#if this.title}}{{this.title}}{{else}}{{this.name}}{{/if}}',
28
- icon: '{{this.icon}}',
29
- variant: 'solid'
30
- }{{#unless @last}},{{/unless}}
31
- {{/each}}
32
- ];
@@ -1,11 +1,23 @@
1
- declare namespace JSX {
2
- interface IntrinsicElements {
3
- // Wildcard for all webcomponents:
4
- [elemName: string]: unknown;
5
- }
6
- }
1
+ import React, { useState, DOMAttributes } from 'react';
7
2
 
8
- declare module '*.module.css' {
9
- const classes: { [key: string]: string };
10
- export default classes;
3
+ type CustomElement<T = React.HTMLAttributes<HTMLElement>> = Partial<T & DOMAttributes<T> & { [key: string]: any }>;
4
+
5
+ declare module "react/jsx-runtime" {
6
+ namespace JSX {
7
+ interface IntrinsicElements {
8
+ 'rapid-design-system-provider': CustomElement;
9
+ 'entity-management': CustomElement;
10
+ 'foundation-form': CustomElement;
11
+ 'rapid-grid-pro': CustomElement;
12
+ 'grid-pro-genesis-datasource': CustomElement;
13
+ 'grid-pro-column': CustomElement;
14
+ 'rapid-g2plot-chart': CustomElement;
15
+ 'chart-datasource': CustomElement;
16
+ 'client-app-login': CustomElement;
17
+ 'rapid-layout': CustomElement;
18
+ 'rapid-layout-region': CustomElement;
19
+ 'rapid-layout-item': CustomElement;
20
+ 'foundation-header': CustomElement;
21
+ }
22
+ }
11
23
  }
@@ -0,0 +1,8 @@
1
+ export const environment = {
2
+ production: true,
3
+ PORT: 4200,
4
+ ENABLE_SSO: {{enableSSO}},
5
+ HOST: 'localhost',
6
+ PROTOCOL: 'http',
7
+ BUILDER: 'webpack',
8
+ };
@@ -0,0 +1,9 @@
1
+ export const environment = {
2
+ production: false,
3
+ API_HOST: '{{apiHost}}',
4
+ PORT: 4200,
5
+ ENABLE_SSO: {{enableSSO}},
6
+ HOST: 'localhost',
7
+ PROTOCOL: 'http',
8
+ BUILDER: 'webpack'
9
+ };
@@ -0,0 +1,64 @@
1
+ import { useState, useEffect, ReactNode } from 'react';
2
+ import { RouteObject } from 'react-router';
3
+ import isConnectedHelper from '@/helpers/isConnectedHelper';
4
+ import isAuthenticatedHelper from '@/helpers/isAuthenticatedHelper';
5
+ import hasPermissionHelper from '@/helpers/hasPermissionHelper';
6
+ import { useRoutesContext } from '@/store/RoutesContext';
7
+ import { NOT_PERMITTED_PATH, AUTH_PATH } from '@/config';
8
+
9
+ enum PermissionState {
10
+ ALLOWED = 'allowed',
11
+ DENIED = 'denied',
12
+ UNKNOWN = 'unknown',
13
+ }
14
+
15
+ const redirectUrlByPermissionState: { [key in Partial<PermissionState>]?: string } = {
16
+ [PermissionState.DENIED]: NOT_PERMITTED_PATH,
17
+ [PermissionState.UNKNOWN]: AUTH_PATH,
18
+ };
19
+
20
+ type ExtendedRouteObject = RouteObject & {
21
+ data?: {
22
+ permissionCode?: string;
23
+ };
24
+ path: string;
25
+ }
26
+
27
+ const ProtectedGuard: React.FC<{ children: ReactNode }> = ({ children }: { children: ReactNode }) => {
28
+ const routes = useRoutesContext() as ExtendedRouteObject[];
29
+ const [isConnected, setIsConnected] = useState<boolean | null>(null);
30
+ const isAuthenticated: boolean | null = null;
31
+ const route = routes.find(({ path }) => path === location.pathname);
32
+ const hasPermission = route?.data?.permissionCode ? hasPermissionHelper(route.data?.permissionCode) : true;
33
+
34
+ useEffect(() => {
35
+ isConnectedHelper().then((connectedState: boolean): void => {
36
+ setIsConnected(connectedState);
37
+ });
38
+ }, []);
39
+
40
+ useEffect((): void => {
41
+ if (isConnected === null) {
42
+ return;
43
+ }
44
+
45
+ const isAuthenticated = isAuthenticatedHelper();
46
+
47
+
48
+ let permissionState;
49
+
50
+ if (!isConnected || !isAuthenticated) {
51
+ permissionState = PermissionState.UNKNOWN;
52
+ } else if (hasPermission === false) {
53
+ permissionState = PermissionState.DENIED;
54
+ }
55
+
56
+ if (permissionState) {
57
+ window.location.href = `/${redirectUrlByPermissionState[permissionState]}`;
58
+ }
59
+ }, [routes, isConnected, isAuthenticated, hasPermission]);
60
+
61
+ return children;
62
+ };
63
+
64
+ export default ProtectedGuard;
@@ -0,0 +1,10 @@
1
+ import { getUser } from '@genesislcap/foundation-user';
2
+
3
+ const hasPermissionHelper = (permissionCode: string | undefined): boolean => {
4
+ const user = getUser();
5
+
6
+ return !!(permissionCode && !user.hasPermission(permissionCode))
7
+
8
+ };
9
+
10
+ export default hasPermissionHelper;
@@ -0,0 +1,14 @@
1
+ import { getUser } from '@genesislcap/foundation-user';
2
+
3
+ const isAuthenticatedHelper = (): boolean => {
4
+ const user = getUser();
5
+
6
+ if (!user.isAuthenticated) {
7
+ user.trackPath();
8
+ return false;
9
+ }
10
+
11
+ return true;
12
+ };
13
+
14
+ export default isAuthenticatedHelper;
@@ -0,0 +1,16 @@
1
+ import { getConnect } from '@genesislcap/foundation-comms';
2
+
3
+ const isConnectedHelper = (): Promise<boolean> => {
4
+ const connect = getConnect();
5
+ const hostUrl = sessionStorage.getItem('hostUrl');
6
+
7
+ if (connect.isConnected) {
8
+ return new Promise((resolve) => resolve(true));
9
+ }
10
+ if (!hostUrl) {
11
+ return new Promise((resolve) => resolve(false));
12
+ }
13
+ return connect.connect(hostUrl);
14
+ };
15
+
16
+ export default isConnectedHelper;
@@ -1,7 +1,7 @@
1
1
  import React, { ReactNode, useEffect, useRef } from 'react';
2
2
  import { configureDesignSystem } from '@genesislcap/foundation-ui';
3
3
  import styles from './BlankLayout.module.css';
4
- import * as designTokens from '../../styles/design-tokens.json';
4
+ import * as designTokens from '@/styles/design-tokens.json';
5
5
 
6
6
  interface BlankLayoutProps {
7
7
  children: ReactNode;
@@ -1,24 +1,36 @@
1
1
  import React, { ReactNode, useEffect, useRef } from 'react';
2
- import { configureDesignSystem } from '@genesislcap/foundation-ui';
2
+ import { RouteObject } from 'react-router';
3
+ import { configureDesignSystem, getNavItems } from '@genesislcap/foundation-ui';
3
4
  import { useNavigate } from 'react-router-dom';
4
5
  import {
5
6
  baseLayerLuminance,
6
7
  StandardLuminance,
7
8
  } from '@microsoft/fast-components';
8
9
  import styles from './DefaultLayout.module.css';
9
- import { mainMenu } from '../../config';
10
- import * as designTokens from '../../styles/design-tokens.json';
10
+ import PBCElementsRenderer from '@/pbc/elementsRenderer';
11
+ import * as designTokens from '@/styles/design-tokens.json';
12
+ import { useRoutesContext } from '@/store/RoutesContext';
11
13
 
12
14
  interface DefaultLayoutProps {
13
15
  children: ReactNode;
14
16
  }
15
17
 
18
+ type ExtendedRouteObject = RouteObject & {
19
+ data?: {
20
+ navItems?: any;
21
+ };
22
+ path: string;
23
+ }
24
+
16
25
  const DefaultLayout: React.FC<DefaultLayoutProps> = ({ children }) => {
17
26
  const navigate = useNavigate();
18
-
19
27
  const designSystemProviderRef = useRef<HTMLElement>(null);
20
28
  const foundationHeaderRef = useRef<HTMLElement>(null);
21
- const allRoutes = mainMenu;
29
+ const routes = useRoutesContext() as ExtendedRouteObject[];
30
+ const navItems = getNavItems(routes.flatMap((route) => ({
31
+ path: route.path || '',
32
+ navItems: route.data?.navItems,
33
+ })));
22
34
 
23
35
  const onLuminanceToggle = (): void => {
24
36
  if (designSystemProviderRef.current) {
@@ -57,22 +69,21 @@ const DefaultLayout: React.FC<DefaultLayoutProps> = ({ children }) => {
57
69
 
58
70
  return (
59
71
  <rapid-design-system-provider ref={designSystemProviderRef} class={className}>
72
+ <PBCElementsRenderer target={['layout-start']} />
60
73
  <foundation-header
61
74
  ref={foundationHeaderRef}
62
75
  show-luminance-toggle-button
63
76
  show-misc-toggle-button
64
- navigateTo={(path) => navigate(path)}
77
+ routeNavItems={navItems}
78
+ navigateTo={(path: string) => navigate(path)}
65
79
  >
66
- <section className={styles['routes-wrapper']} slot="routes">
67
- {allRoutes.map((route, index) => (
68
- <rapid-button key={index} onClick={() => navigate(route.path)}>
69
- <rapid-icon name={route.icon}></rapid-icon>
70
- {route.title}
71
- </rapid-button>
72
- ))}
73
- </section>
74
80
  </foundation-header>
75
- <section className={styles['content']}>{children}</section>
81
+ <section className={styles['content']}>
82
+ <PBCElementsRenderer target={['content-start']} />
83
+ {children}
84
+ <PBCElementsRenderer target={['content', 'content-end']} />
85
+ </section>
86
+ <PBCElementsRenderer target={['layout', 'layout-end']} />
76
87
  </rapid-design-system-provider>
77
88
  );
78
89
  };
@@ -1,10 +1,26 @@
1
1
  import React from 'react'
2
2
  import ReactDOM from 'react-dom/client'
3
3
  import App from './App.tsx'
4
+
5
+ import { registerPBCs } from '@/pbc/utils';
6
+ import { createLogger } from '@genesislcap/foundation-logger';
7
+
4
8
  import './styles/styles.css'
5
9
 
6
- ReactDOM.createRoot(document.getElementById('root')!).render(
7
- <React.StrictMode>
8
- <App />
9
- </React.StrictMode>,
10
- )
10
+ const logger = createLogger('main');
11
+
12
+ function bootstrapApp() {
13
+ const rootEelement = document.getElementById('root');
14
+ if (rootEelement) {
15
+ ReactDOM.createRoot(rootEelement!).render(
16
+ <React.StrictMode>
17
+ <App rootElement={rootEelement} />
18
+ </React.StrictMode>,
19
+ )
20
+ }
21
+ }
22
+
23
+ registerPBCs()
24
+ .then(hasAssets => logger.debug(hasAssets ? 'PBCs registered' : 'No PBCs detected'))
25
+ .catch((err) => logger.error(err))
26
+ .finally(bootstrapApp)
@@ -0,0 +1,10 @@
1
+ import { render } from '@testing-library/react';
2
+ import AuthPage from './AuthPage';
3
+
4
+ test('renders AuthPage component', () => {
5
+ const { container } = render(<AuthPage />);
6
+ const authPageElement: Element | null = container.querySelector('.auth-page');
7
+ expect(authPageElement).toBeInTheDocument();
8
+ const clientAppLoginElement: Element | null = container.querySelector('client-app-login');
9
+ expect(clientAppLoginElement).toBeInTheDocument();
10
+ });
@@ -1,6 +1,7 @@
1
+ import React from 'react';
1
2
  import './AuthPage.css';
2
3
 
3
- const AuthPage = () => {
4
+ const AuthPage: React.FC = () => {
4
5
  return (
5
6
  <section className="auth-page">
6
7
  <client-app-login></client-app-login>
@@ -8,4 +9,4 @@ const AuthPage = () => {
8
9
  );
9
10
  };
10
11
 
11
- export default AuthPage;
12
+ export default AuthPage;