diff --git a/.dockerignore b/.dockerignore
new file mode 100644
index 00000000..d71e6f1c
--- /dev/null
+++ b/.dockerignore
@@ -0,0 +1,7 @@
+.git
+node_modules
+build
+test-coverage
+reference
+
+.DS_Store
diff --git a/build.Dockerfile b/build.Dockerfile
new file mode 100644
index 00000000..7d4e9724
--- /dev/null
+++ b/build.Dockerfile
@@ -0,0 +1,14 @@
+FROM node:22.5-alpine
+
+EXPOSE 3000
+EXPOSE 3010
+
+WORKDIR /app
+COPY . ./build-temp
+
+WORKDIR /app/build-temp
+RUN npm ci
+
+WORKDIR /app
+RUN mv ./build-temp/node_modules ./
+RUN rm -rf ./build-temp
diff --git a/index.js b/index.js
deleted file mode 100644
index ca9e8d6f..00000000
--- a/index.js
+++ /dev/null
@@ -1 +0,0 @@
-module.exports = require("./src/");
diff --git a/jest.config.js b/jest.config.js
index c9d36ea6..af3de0db 100644
--- a/jest.config.js
+++ b/jest.config.js
@@ -7,10 +7,13 @@ module.exports = {
moduleNameMapper: {
'@stomp/stompjs': '/src/__mocks__/fileMock.js',
'@stomp/rx-stomp': '/src/__mocks__/fileMock.js',
- '(pdfjs-dist/legacy/build/pdf.worker)': '/src/__mocks__/constructorMock.js',
+ '(pdfjs-dist/build/pdf.worker.mjs)': '/src/__mocks__/constructorMock.js',
'\\.(css|less)$': '/src/__mocks__/styleMock.js',
'\\.(svg)$': '/src/__mocks__/svgMock.js',
- 'react-markdown': '/node_modules/react-markdown/react-markdown.min.js',
+ 'react-markdown': [
+ '/node_modules/react-markdown/react-markdown.min.js',
+ '/../node_modules/react-markdown/react-markdown.min.js',
+ ],
},
transform: {
'\\.(js|jsx|ts|tsx)$': 'babel-jest',
@@ -21,6 +24,7 @@ module.exports = {
transformIgnorePatterns: [
"node_modules/(?!(dateformat|remark-gfm|micromark-.+|decode-named-character-reference|character-entities|mdast-.*|escape-string-regexp|unist-util-.*|markdown-.*|ccount|d3-[a-z]+)/)",
],
+ globalSetup: '/scripts/jest-global-setup.js',
setupFiles: [
'jest-canvas-mock',
],
diff --git a/lib/components/NeonPage/NeonPage.d.ts b/lib/components/NeonPage/NeonPage.d.ts
index 22f53e77..25c6d948 100644
--- a/lib/components/NeonPage/NeonPage.d.ts
+++ b/lib/components/NeonPage/NeonPage.d.ts
@@ -30,6 +30,7 @@ declare namespace NeonPage {
export const progress: PropTypes.Requireable;
export const resetStateAfterRuntimeError: PropTypes.Requireable<(...args: any[]) => any>;
export { children as sidebarContent };
+ export const sidebarContentResponsive: PropTypes.Requireable;
export const sidebarContainerClassName: PropTypes.Requireable;
export const sidebarLinks: PropTypes.Requireable<(PropTypes.InferProps<{
name: PropTypes.Validator;
@@ -83,6 +84,8 @@ declare namespace NeonPage {
export function resetStateAfterRuntimeError_1(): void;
export { resetStateAfterRuntimeError_1 as resetStateAfterRuntimeError };
export const sidebarContent: null;
+ const sidebarContentResponsive_1: boolean;
+ export { sidebarContentResponsive_1 as sidebarContentResponsive };
const sidebarContainerClassName_1: null;
export { sidebarContainerClassName_1 as sidebarContainerClassName };
const sidebarLinks_1: null;
diff --git a/lib/components/NeonPage/NeonPage.js b/lib/components/NeonPage/NeonPage.js
index e8f043d5..c49c8e6d 100644
--- a/lib/components/NeonPage/NeonPage.js
+++ b/lib/components/NeonPage/NeonPage.js
@@ -354,6 +354,7 @@ const NeonPage = props => {
progress,
resetStateAfterRuntimeError,
sidebarContent,
+ sidebarContentResponsive,
sidebarContainerClassName: sidebarContainerClassNameProp,
sidebarLinks,
sidebarLinksAdditionalContent,
@@ -685,7 +686,7 @@ const NeonPage = props => {
} : {};
const sidebarClassName = sidebarContainerClassNameProp ? "".concat(classes.sidebarContainer, " ").concat(sidebarContainerClassNameProp) : classes.sidebarContainer;
// Arbitrary Content Sidebar (no automatic skeleton)
- if (hasSidebarContent) {
+ if (hasSidebarContent && !sidebarContentResponsive) {
return /*#__PURE__*/_react.default.createElement("div", {
ref: sidebarRef,
className: sidebarClassName,
@@ -721,6 +722,28 @@ const NeonPage = props => {
className: classes.sidebarSubtitle
}, sidebarSubtitle)));
};
+ // Arbitrary Content Sidebar (no automatic skeleton)
+ if (hasSidebarContent) {
+ return /*#__PURE__*/_react.default.createElement("div", {
+ ref: sidebarRef,
+ className: sidebarClassName,
+ style: sidebarContainerStyle
+ }, /*#__PURE__*/_react.default.createElement("div", {
+ className: !sidebarUnsticky && !belowMd ? "".concat(classes.sidebarInnerStickyContainer, " neon__sidebar-sticky") : null
+ }, /*#__PURE__*/_react.default.createElement("div", {
+ className: classes.sidebarTitleContainer
+ }, renderSidebarTitle(), !belowMd ? null : /*#__PURE__*/_react.default.createElement(_IconButton.default, {
+ size: "small",
+ onClick: () => setSidebarExpanded(!sidebarExpanded)
+ }, sidebarExpanded ? /*#__PURE__*/_react.default.createElement(_ExpandLess.default, {
+ fontSize: "large"
+ }) : /*#__PURE__*/_react.default.createElement(_ExpandMore.default, {
+ fontSize: "large"
+ }))), !belowMd || sidebarExpanded ? /*#__PURE__*/_react.default.createElement(_react.default.Fragment, null, /*#__PURE__*/_react.default.createElement(_Divider.default, {
+ className: classes.sidebarDivider,
+ style: _extends({}, dividerStyle)
+ }), sidebarContent) : null));
+ }
// Render Single Sidebar Link
const renderLink = function (link) {
let standalone = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false;
@@ -861,6 +884,7 @@ NeonPage.propTypes = {
progress: _propTypes.default.number,
resetStateAfterRuntimeError: _propTypes.default.func,
sidebarContent: children,
+ sidebarContentResponsive: _propTypes.default.bool,
sidebarContainerClassName: _propTypes.default.string,
sidebarLinks: _propTypes.default.arrayOf(_propTypes.default.shape({
name: _propTypes.default.string.isRequired,
@@ -895,6 +919,7 @@ NeonPage.defaultProps = {
progress: null,
resetStateAfterRuntimeError: () => {},
sidebarContent: null,
+ sidebarContentResponsive: false,
sidebarContainerClassName: null,
sidebarLinks: null,
sidebarLinksAdditionalContent: null,
diff --git a/package-lock.json b/package-lock.json
index da3ff36e..0b019f9c 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -1,12 +1,12 @@
{
"name": "portal-core-components",
- "version": "2.9.2",
+ "version": "2.10.0",
"lockfileVersion": 3,
"requires": true,
"packages": {
"": {
"name": "portal-core-components",
- "version": "2.9.2",
+ "version": "2.10.0",
"dependencies": {
"@date-io/moment": "^1.3.13",
"@fortawesome/fontawesome-svg-core": "^6.6.0",
@@ -83,7 +83,7 @@
"@babel/preset-typescript": "^7.24.7",
"@jest/globals": "^29.7.0",
"@testing-library/react-hooks": "^8.0.1",
- "@typescript-eslint/parser": "^7.16.1",
+ "@typescript-eslint/parser": "^7.17.0",
"commander": "^12.1.0",
"cssnano": "^7.0.4",
"eslint": "^8.57.0",
@@ -92,11 +92,11 @@
"eslint-plugin-jsx-a11y": "^6.9.0",
"eslint-plugin-react": "^7.35.0",
"express": "^4.19.2",
- "html-loader": "^5.0.0",
+ "html-loader": "^5.1.0",
"jest-canvas-mock": "^2.5.2",
"jsdom": "^24.1.1",
"node-fetch": "^3.3.2",
- "postcss": "^8.4.39",
+ "postcss": "^8.4.40",
"prettier": "^3.3.3",
"react-scripts": "^5.0.1",
"react-test-renderer": "^17.0.2",
@@ -108,7 +108,7 @@
"cssnano": "^7.0.4",
"jsdom": "^24.1.1",
"node-fetch": "^3.3.2",
- "postcss": "^8.4.39",
+ "postcss": "^8.4.40",
"prettier": "^3.3.3",
"worker-loader": "^3.0.8"
},
@@ -2822,9 +2822,9 @@
}
},
"node_modules/@emotion/cache": {
- "version": "11.13.0",
- "resolved": "https://registry.npmjs.org/@emotion/cache/-/cache-11.13.0.tgz",
- "integrity": "sha512-hPV345J/tH0Cwk2wnU/3PBzORQ9HeX+kQSbwI+jslzpRCHE6fSGTohswksA/Ensr8znPzwfzKZCmAM9Lmlhp7g==",
+ "version": "11.13.1",
+ "resolved": "https://registry.npmjs.org/@emotion/cache/-/cache-11.13.1.tgz",
+ "integrity": "sha512-iqouYkuEblRcXmylXIwwOodiEK5Ifl7JcX7o6V4jI3iW4mLXX3dmt5xwBtIkJiQEXFAI+pC8X0i67yiPkH9Ucw==",
"license": "MIT",
"dependencies": {
"@emotion/memoize": "^0.9.0",
@@ -11365,9 +11365,9 @@
}
},
"node_modules/electron-to-chromium": {
- "version": "1.5.0",
- "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.0.tgz",
- "integrity": "sha512-Vb3xHHYnLseK8vlMJQKJYXJ++t4u1/qJ3vykuVrVjvdiOEhYyT1AuP4x03G8EnPmYvYOhe9T+dADTmthjRQMkA==",
+ "version": "1.5.1",
+ "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.1.tgz",
+ "integrity": "sha512-FKbOCOQ5QRB3VlIbl1LZQefWIYwszlBloaXcY2rbfpu9ioJnNh3TK03YtIDKDo3WKBi8u+YV4+Fn2CkEozgf4w==",
"dev": true,
"license": "ISC"
},
@@ -14272,9 +14272,9 @@
"license": "MIT"
},
"node_modules/html-loader": {
- "version": "5.0.0",
- "resolved": "https://registry.npmjs.org/html-loader/-/html-loader-5.0.0.tgz",
- "integrity": "sha512-puaGKdjdVVIFRtgIC2n5dt5bt0N5j6heXlAQZ4Do1MLjHmOT1gCE1Ogg7XZNeJlnOVHHsrZKGs5dfh+XwZ3XPw==",
+ "version": "5.1.0",
+ "resolved": "https://registry.npmjs.org/html-loader/-/html-loader-5.1.0.tgz",
+ "integrity": "sha512-Jb3xwDbsm0W3qlXrCZwcYqYGnYz55hb6aoKQTlzyZPXsPpi6tHXzAfqalecglMQgNvtEfxrCQPaKT90Irt5XDA==",
"dev": true,
"license": "MIT",
"dependencies": {
@@ -23911,9 +23911,9 @@
}
},
"node_modules/postcss": {
- "version": "8.4.39",
- "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.39.tgz",
- "integrity": "sha512-0vzE+lAiG7hZl1/9I8yzKLx3aR9Xbof3fBHKunvMfOCYAtMhrsnccJY2iTURb9EZd5+pLuiNV9/c/GZJOHsgIw==",
+ "version": "8.4.40",
+ "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.40.tgz",
+ "integrity": "sha512-YF2kKIUzAofPMpfH6hOi2cGnv/HrUlfucspc7pDyvv7kGdqXrfj8SCl/t8owkEgKEuu8ZcRjSOxFxVLqwChZ2Q==",
"dev": true,
"funding": [
{
@@ -24469,9 +24469,9 @@
}
},
"node_modules/postcss-load-config/node_modules/yaml": {
- "version": "2.4.5",
- "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.4.5.tgz",
- "integrity": "sha512-aBx2bnqDzVOyNKfsysjA2ms5ZlnjSAW2eG3/L5G/CSujfjLJTJsEw1bGw8kCf04KodQWk1pxlGnZ56CRxiawmg==",
+ "version": "2.5.0",
+ "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.5.0.tgz",
+ "integrity": "sha512-2wWLbGbYDiSqqIKoPjar3MPgB94ErzCtrNE1FdqGuaO0pi2JGjmE8aW8TDZwzU7vuxcGRdL/4gPQwQ7hD5AMSw==",
"dev": true,
"license": "ISC",
"bin": {
@@ -28109,9 +28109,9 @@
"license": "MIT"
},
"node_modules/tailwindcss": {
- "version": "3.4.6",
- "resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-3.4.6.tgz",
- "integrity": "sha512-1uRHzPB+Vzu57ocybfZ4jh5Q3SdlH7XW23J5sQoM9LhE9eIOlzxer/3XPSsycvih3rboRsvt0QCmzSrqyOYUIA==",
+ "version": "3.4.7",
+ "resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-3.4.7.tgz",
+ "integrity": "sha512-rxWZbe87YJb4OcSopb7up2Ba4U82BoiSGUdoDr3Ydrg9ckxFS/YWsvhN323GMcddgU65QRy7JndC7ahhInhvlQ==",
"dev": true,
"license": "MIT",
"dependencies": {
diff --git a/package.json b/package.json
index b0f9126c..3e61683f 100644
--- a/package.json
+++ b/package.json
@@ -1,6 +1,6 @@
{
"name": "portal-core-components",
- "version": "2.9.2",
+ "version": "2.10.0",
"main": "./lib/index.js",
"private": true,
"homepage": "http://localhost:3010/core-components",
@@ -73,7 +73,7 @@
"cssnano": "^7.0.4",
"jsdom": "^24.1.1",
"node-fetch": "^3.3.2",
- "postcss": "^8.4.39",
+ "postcss": "^8.4.40",
"prettier": "^3.3.3",
"worker-loader": "^3.0.8"
},
@@ -112,7 +112,7 @@
"@babel/preset-typescript": "^7.24.7",
"@jest/globals": "^29.7.0",
"@testing-library/react-hooks": "^8.0.1",
- "@typescript-eslint/parser": "^7.16.1",
+ "@typescript-eslint/parser": "^7.17.0",
"commander": "^12.1.0",
"cssnano": "^7.0.4",
"eslint": "^8.57.0",
@@ -121,11 +121,11 @@
"eslint-plugin-jsx-a11y": "^6.9.0",
"eslint-plugin-react": "^7.35.0",
"express": "^4.19.2",
- "html-loader": "^5.0.0",
+ "html-loader": "^5.1.0",
"jest-canvas-mock": "^2.5.2",
"jsdom": "^24.1.1",
"node-fetch": "^3.3.2",
- "postcss": "^8.4.39",
+ "postcss": "^8.4.40",
"prettier": "^3.3.3",
"react-scripts": "^5.0.1",
"react-test-renderer": "^17.0.2",
@@ -151,7 +151,19 @@
"lib:post-clean-styleGuides": "rm ./lib/components/**/StyleGuide.*",
"lib:post-clean-img": "rm -r ./lib/components/SiteMap/png && rm -r ./lib/components/SiteMap/svg_source && rm -r ./lib/components/DataThemeIcon/svg_source",
"lib:post-clean-tests": "rm -r ./lib/*/__tests__ && rm -r ./lib/*/*/__tests__",
- "checks": "npm run lint && npm run test && npm run lib && npm run build"
+ "checks": "npm run lib && npm run lint && npm run test && npm run build",
+ "checks:docker": "npm run build:docker-image && npm run lib:docker && npm run test:docker && npm run build:docker",
+ "start:docker": "docker run --rm --tty --interactive --volume=/$(pwd):/app/src --volume=/app/src/node_modules --workdir=/app/src --publish=3010:3010 portal-core-components:latest-builder npm run start",
+ "start:docker-prod": "docker run --rm --tty --interactive --volume=/$(pwd):/app/src --volume=/app/src/node_modules --workdir=/app/src --publish=3000:3000 portal-core-components:latest-builder npm run start:prod",
+ "build:docker-clean": "docker image rm portal-core-components:latest-builder",
+ "build:docker-build-image": "docker buildx build --no-cache --tag portal-core-components:latest-builder --file ./build.Dockerfile .",
+ "build:docker-image": "(npm run build:docker-clean || true) && npm run build:docker-build-image",
+ "build:docker": "docker run --rm --tty --interactive --env GENERATE_SOURCEMAP=false --env NODE_OPTIONS=--max_old_space_size=4096 --volume=/$(pwd):/app/src --volume=/app/src/node_modules --workdir=/app/src portal-core-components:latest-builder npm run build",
+ "lib:docker": "docker run --rm --tty --interactive --volume=/$(pwd):/app/src --volume=/app/src/node_modules --workdir=/app/src portal-core-components:latest-builder npm run lib",
+ "test:docker": "docker run --rm --tty --interactive --volume=/$(pwd):/app/src --volume=/app/src/node_modules --workdir=/app/src portal-core-components:latest-builder npm run test:docker-command",
+ "test:docker-update-snapshots": "docker run --rm --tty --interactive --volume=/$(pwd):/app/src --volume=/app/src/node_modules --workdir=/app/src portal-core-components:latest-builder npm run test:docker-command-update-snapshots",
+ "test:docker-command": "npm run lint && npm run test",
+ "test:docker-command-update-snapshots": "npm run lint && npm run test:updateSnapshots"
},
"browserslist": {
"production": [
diff --git a/scripts/jest-global-setup.js b/scripts/jest-global-setup.js
new file mode 100644
index 00000000..05de598c
--- /dev/null
+++ b/scripts/jest-global-setup.js
@@ -0,0 +1,5 @@
+const jestGlobalSetup = async () => {
+ process.env.TZ = 'UTC';
+};
+
+export default jestGlobalSetup;
diff --git a/src/lib_components/components/DownloadStepForm/__tests__/__snapshots__/DownloadStepForm.jsx.snap b/src/lib_components/components/DownloadStepForm/__tests__/__snapshots__/DownloadStepForm.jsx.snap
index 6ae50553..4822033c 100644
--- a/src/lib_components/components/DownloadStepForm/__tests__/__snapshots__/DownloadStepForm.jsx.snap
+++ b/src/lib_components/components/DownloadStepForm/__tests__/__snapshots__/DownloadStepForm.jsx.snap
@@ -861,7 +861,7 @@ exports[`DownloadStepForm Step summary renders correctly 1`] = `
- 3 sites — Dec 2019 - Dec 2019
+ 3 sites — Jan 2020 - Jan 2020
diff --git a/src/lib_components/components/NeonAuth/__tests__/__snapshots__/UserCard.jsx.snap b/src/lib_components/components/NeonAuth/__tests__/__snapshots__/UserCard.jsx.snap
index 0dfdfc15..bf9fb494 100644
--- a/src/lib_components/components/NeonAuth/__tests__/__snapshots__/UserCard.jsx.snap
+++ b/src/lib_components/components/NeonAuth/__tests__/__snapshots__/UserCard.jsx.snap
@@ -89,7 +89,7 @@ exports[`NeonAuth - UserCard Renders with a lastLogin prop 1`] = `
- May 4th, 2018 6:00:00 pm
+ May 5th, 2018 12:00:00 am
diff --git a/src/lib_components/components/NeonPage/NeonPage.jsx b/src/lib_components/components/NeonPage/NeonPage.jsx
index 64537a96..af8400c5 100644
--- a/src/lib_components/components/NeonPage/NeonPage.jsx
+++ b/src/lib_components/components/NeonPage/NeonPage.jsx
@@ -371,6 +371,7 @@ const NeonPage = (props) => {
progress,
resetStateAfterRuntimeError,
sidebarContent,
+ sidebarContentResponsive,
sidebarContainerClassName: sidebarContainerClassNameProp,
sidebarLinks,
sidebarLinksAdditionalContent,
@@ -721,7 +722,7 @@ const NeonPage = (props) => {
? `${classes.sidebarContainer} ${sidebarContainerClassNameProp}`
: classes.sidebarContainer;
// Arbitrary Content Sidebar (no automatic skeleton)
- if (hasSidebarContent) {
+ if (hasSidebarContent && !sidebarContentResponsive) {
return (
{sidebarContent}
@@ -755,6 +756,37 @@ const NeonPage = (props) => {
);
};
+ // Arbitrary Content Sidebar (no automatic skeleton)
+ if (hasSidebarContent) {
+ return (
+
+
+
+ {renderSidebarTitle()}
+ {!belowMd ? null : (
+ setSidebarExpanded(!sidebarExpanded)}>
+ {(
+ sidebarExpanded
+ ?
+ :
+ )}
+
+ )}
+
+ {(!belowMd || sidebarExpanded) ? (
+ <>
+
+ {sidebarContent}
+ >
+ ) : null}
+
+
+ );
+ }
// Render Single Sidebar Link
const renderLink = (link, standalone = false) => {
if (!link) { return null; }
@@ -957,6 +989,7 @@ NeonPage.propTypes = {
progress: PropTypes.number,
resetStateAfterRuntimeError: PropTypes.func,
sidebarContent: children,
+ sidebarContentResponsive: PropTypes.bool,
sidebarContainerClassName: PropTypes.string,
sidebarLinks: PropTypes.arrayOf(
PropTypes.shape({
@@ -1006,6 +1039,7 @@ NeonPage.defaultProps = {
progress: null,
resetStateAfterRuntimeError: () => { },
sidebarContent: null,
+ sidebarContentResponsive: false,
sidebarContainerClassName: null,
sidebarLinks: null,
sidebarLinksAdditionalContent: null,
diff --git a/src/lib_components/components/NeonPage/StyleGuide.jsx b/src/lib_components/components/NeonPage/StyleGuide.jsx
index 1a582348..6bf172b0 100644
--- a/src/lib_components/components/NeonPage/StyleGuide.jsx
+++ b/src/lib_components/components/NeonPage/StyleGuide.jsx
@@ -212,6 +212,19 @@ const propRows = [
),
},
+ // sidebarContentResponsive
+ {
+ name: 'sidebarContentResponsive',
+ type: 'boolean',
+ default: 'false',
+ description: (
+
+ When the sidebarContent is set and the responsive behavior of the sidebar
+ is desired, set this property to true. Otherwise, by default, the behavior
+ of the sidebar is up to the implementation of the sidebarContent .
+
+ ),
+ },
// sidebarContainerClassName
{
name: 'sidebarContainerClassName',
@@ -526,6 +539,39 @@ import NeonPage from 'portal-core-components/lib/components/NeonPage';
`}
+
+ Sidebar Content
+
+
+ Use the sidebarContent prop to set custom sidebar content.
+
+
+
+ Sidebar Content)}
+ sidebarContentResponsive
+ >
+ Content
+
+
+
+
+ {`
+Sidebar Content)}
+ sidebarContentResponsive
+>
+ Content
+
+ `}
+
+
Breadcrumbs